diff options
Diffstat (limited to 'main/libpri/libpri-1.4-r1357.patch')
-rw-r--r-- | main/libpri/libpri-1.4-r1357.patch | 39961 |
1 files changed, 39961 insertions, 0 deletions
diff --git a/main/libpri/libpri-1.4-r1357.patch b/main/libpri/libpri-1.4-r1357.patch new file mode 100644 index 0000000000..be827eebb7 --- /dev/null +++ b/main/libpri/libpri-1.4-r1357.patch @@ -0,0 +1,39961 @@ +=================================================================== +--- a/pri_timers.h (.../tags/1.4.10.2) (revision 1357) ++++ b/pri_timers.h (.../branches/1.4) (revision 1357) +@@ -1,97 +0,0 @@ +-/* +- * libpri: An implementation of Primary Rate ISDN +- * +- * Written by Mark Spencer <markster@digium.com> +- * +- * Copyright (C) 2001, Digium, Inc. +- * All Rights Reserved. +- */ +- +-/* +- * See http://www.asterisk.org for more information about +- * the Asterisk project. Please do not directly contact +- * any of the maintainers of this project for assistance; +- * the project provides a web site, mailing lists and IRC +- * channels for your use. +- * +- * This program is free software, distributed under the terms of +- * the GNU General Public License Version 2 as published by the +- * Free Software Foundation. See the LICENSE file included with +- * this program for more details. +- * +- * In addition, when this program is distributed with Asterisk in +- * any form that would qualify as a 'combined work' or as a +- * 'derivative work' (but not mere aggregation), you can redistribute +- * and/or modify the combination under the terms of the license +- * provided with that copy of Asterisk, instead of the license +- * terms granted here. +- */ +- +-#ifndef _PRI_TIMERS_H +-#define _PRI_TIMERS_H +- +-/* -1 means we dont currently support the timer/counter */ +-#define PRI_TIMERS_DEFAULT { \ +- 3, /* N200 */ \ +- -1, /* N201 */ \ +- 3, /* N202 */ \ +- 7, /* K */ \ +- 1000, /* T200 */ \ +- -1, /* T201 */ \ +- 10000, /* T202 */ \ +- 10000, /* T203 */ \ +- -1, /* T300 */ \ +- -1, /* T301 */ \ +- -1, /* T302 */ \ +- -1, /* T303 */ \ +- -1, /* T304 */ \ +- 30000, /* T305 */ \ +- -1, /* T306 */ \ +- -1, /* T307 */ \ +- 4000, /* T308 */ \ +- -1, /* T309 */ \ +- -1, /* T310 */ \ +- 4000, /* T313 */ \ +- -1, /* T314 */ \ +- -1, /* T316 */ \ +- -1, /* T317 */ \ +- -1, /* T318 */ \ +- -1, /* T319 */ \ +- -1, /* T320 */ \ +- -1, /* T321 */ \ +- -1, /* T322 */ \ +- 2500, /* TM20 - Q.921 Appendix IV */ \ +- 3, /* NM20 - Q.921 Appendix IV */ \ +- } +- +-/* XXX Only our default timers are setup now XXX */ +-#define PRI_TIMERS_UNKNOWN PRI_TIMERS_DEFAULT +-#define PRI_TIMERS_NI2 PRI_TIMERS_DEFAULT +-#define PRI_TIMERS_DMS100 PRI_TIMERS_DEFAULT +-#define PRI_TIMERS_LUCENT5E PRI_TIMERS_DEFAULT +-#define PRI_TIMERS_ATT4ESS PRI_TIMERS_DEFAULT +-#define PRI_TIMERS_EUROISDN_E1 PRI_TIMERS_DEFAULT +-#define PRI_TIMERS_EUROISDN_T1 PRI_TIMERS_DEFAULT +-#define PRI_TIMERS_NI1 PRI_TIMERS_DEFAULT +-#define PRI_TIMERS_GR303_EOC PRI_TIMERS_DEFAULT +-#define PRI_TIMERS_GR303_TMC PRI_TIMERS_DEFAULT +-#define PRI_TIMERS_QSIG PRI_TIMERS_DEFAULT +-#define __PRI_TIMERS_GR303_EOC_INT PRI_TIMERS_DEFAULT +-#define __PRI_TIMERS_GR303_TMC_INT PRI_TIMERS_DEFAULT +- +-#define PRI_TIMERS_ALL { PRI_TIMERS_UNKNOWN, \ +- PRI_TIMERS_NI2, \ +- PRI_TIMERS_DMS100, \ +- PRI_TIMERS_LUCENT5E, \ +- PRI_TIMERS_ATT4ESS, \ +- PRI_TIMERS_EUROISDN_E1, \ +- PRI_TIMERS_EUROISDN_T1, \ +- PRI_TIMERS_NI1, \ +- PRI_TIMERS_QSIG, \ +- PRI_TIMERS_GR303_EOC, \ +- PRI_TIMERS_GR303_TMC, \ +- __PRI_TIMERS_GR303_EOC_INT, \ +- __PRI_TIMERS_GR303_TMC_INT, \ +- } +- +-#endif +Index: .version +=================================================================== +--- a/.version (.../tags/1.4.10.2) (revision 1357) ++++ b/.version (.../branches/1.4) (revision 1357) +@@ -1 +1 @@ +-1.4.10.2 ++1.4.10.2-r1357 +Index: prisched.c +=================================================================== +--- a/prisched.c (.../tags/1.4.10.2) (revision 1357) ++++ b/prisched.c (.../branches/1.4) (revision 1357) +@@ -33,25 +33,43 @@ + #include "pri_internal.h" + + ++/*! \brief The maximum number of timers that were active at once. */ + static int maxsched = 0; + + /* Scheduler routines */ +-int pri_schedule_event(struct pri *pri, int ms, void (*function)(void *data), void *data) ++ ++/*! ++ * \brief Start a timer to schedule an event. ++ * ++ * \param ctrl D channel controller. ++ * \param ms Number of milliseconds to scheduled event. ++ * \param function Callback function to call when timeout. ++ * \param data Value to give callback function when timeout. ++ * ++ * \retval 0 if scheduler table is full and could not schedule the event. ++ * \retval id Scheduled event id. ++ */ ++int pri_schedule_event(struct pri *ctrl, int ms, void (*function)(void *data), void *data) + { + int x; + struct timeval tv; ++ + /* Scheduling runs on master channels only */ +- while (pri->master) +- pri = pri->master; +- for (x=1;x<MAX_SCHED;x++) +- if (!pri->pri_sched[x].callback) ++ while (ctrl->master) { ++ ctrl = ctrl->master; ++ } ++ for (x = 0; x < MAX_SCHED; ++x) { ++ if (!ctrl->pri_sched[x].callback) { + break; ++ } ++ } + if (x == MAX_SCHED) { +- pri_error(pri, "No more room in scheduler\n"); +- return -1; ++ pri_error(ctrl, "No more room in scheduler\n"); ++ return 0; + } +- if (x > maxsched) +- maxsched = x; ++ if (x >= maxsched) { ++ maxsched = x + 1; ++ } + gettimeofday(&tv, NULL); + tv.tv_sec += ms / 1000; + tv.tv_usec += (ms % 1000) * 1000; +@@ -59,71 +77,110 @@ + tv.tv_usec -= 1000000; + tv.tv_sec += 1; + } +- pri->pri_sched[x].when = tv; +- pri->pri_sched[x].callback = function; +- pri->pri_sched[x].data = data; +- return x; ++ ctrl->pri_sched[x].when = tv; ++ ctrl->pri_sched[x].callback = function; ++ ctrl->pri_sched[x].data = data; ++ return x + 1; + } + +-struct timeval *pri_schedule_next(struct pri *pri) ++/*! ++ * \brief Determine the time of the next scheduled event to expire. ++ * ++ * \param ctrl D channel controller. ++ * ++ * \return Time of the next scheduled event to expire or NULL if no timers active. ++ */ ++struct timeval *pri_schedule_next(struct pri *ctrl) + { + struct timeval *closest = NULL; + int x; +- /* Check subchannels */ +- if (pri->subchannel) +- closest = pri_schedule_next(pri->subchannel); +- for (x=1;x<MAX_SCHED;x++) { +- if (pri->pri_sched[x].callback && +- (!closest || (closest->tv_sec > pri->pri_sched[x].when.tv_sec) || +- ((closest->tv_sec == pri->pri_sched[x].when.tv_sec) && +- (closest->tv_usec > pri->pri_sched[x].when.tv_usec)))) +- closest = &pri->pri_sched[x].when; ++ ++ /* Scheduling runs on master channels only */ ++ while (ctrl->master) { ++ ctrl = ctrl->master; + } ++ for (x = 0; x < MAX_SCHED; ++x) { ++ if (ctrl->pri_sched[x].callback && (!closest ++ || (closest->tv_sec > ctrl->pri_sched[x].when.tv_sec) ++ || ((closest->tv_sec == ctrl->pri_sched[x].when.tv_sec) ++ && (closest->tv_usec > ctrl->pri_sched[x].when.tv_usec)))) { ++ closest = &ctrl->pri_sched[x].when; ++ } ++ } + return closest; + } + +-static pri_event *__pri_schedule_run(struct pri *pri, struct timeval *tv) ++/*! ++ * \internal ++ * \brief Run all expired timers or return an event generated by an expired timer. ++ * ++ * \param ctrl D channel controller. ++ * \param tv Current time. ++ * ++ * \return Event for upper layer to process or NULL if all expired timers run. ++ */ ++static pri_event *__pri_schedule_run(struct pri *ctrl, struct timeval *tv) + { + int x; + void (*callback)(void *); + void *data; +- pri_event *e; +- if (pri->subchannel) { +- if ((e = __pri_schedule_run(pri->subchannel, tv))) { +- return e; ++ ++ /* Scheduling runs on master channels only */ ++ while (ctrl->master) { ++ ctrl = ctrl->master; ++ } ++ for (x = 0; x < MAX_SCHED; ++x) { ++ if (ctrl->pri_sched[x].callback && ((ctrl->pri_sched[x].when.tv_sec < tv->tv_sec) ++ || ((ctrl->pri_sched[x].when.tv_sec == tv->tv_sec) ++ && (ctrl->pri_sched[x].when.tv_usec <= tv->tv_usec)))) { ++ /* This timer has expired. */ ++ ctrl->schedev = 0; ++ callback = ctrl->pri_sched[x].callback; ++ data = ctrl->pri_sched[x].data; ++ ctrl->pri_sched[x].callback = NULL; ++ callback(data); ++ if (ctrl->schedev) { ++ return &ctrl->ev; ++ } + } + } +- for (x=1;x<MAX_SCHED;x++) { +- if (pri->pri_sched[x].callback && +- ((pri->pri_sched[x].when.tv_sec < tv->tv_sec) || +- ((pri->pri_sched[x].when.tv_sec == tv->tv_sec) && +- (pri->pri_sched[x].when.tv_usec <= tv->tv_usec)))) { +- pri->schedev = 0; +- callback = pri->pri_sched[x].callback; +- data = pri->pri_sched[x].data; +- pri->pri_sched[x].callback = NULL; +- pri->pri_sched[x].data = NULL; +- callback(data); +- if (pri->schedev) +- return &pri->ev; +- } +- } + return NULL; + } + +-pri_event *pri_schedule_run(struct pri *pri) ++/*! ++ * \brief Run all expired timers or return an event generated by an expired timer. ++ * ++ * \param ctrl D channel controller. ++ * ++ * \return Event for upper layer to process or NULL if all expired timers run. ++ */ ++pri_event *pri_schedule_run(struct pri *ctrl) + { + struct timeval tv; ++ + gettimeofday(&tv, NULL); +- return __pri_schedule_run(pri, &tv); ++ return __pri_schedule_run(ctrl, &tv); + } + +- +-void pri_schedule_del(struct pri *pri,int id) ++/*! ++ * \brief Delete a scheduled event. ++ * ++ * \param ctrl D channel controller. ++ * \param id Scheduled event id to delete. ++ * 0 is a disabled/unscheduled event id that is ignored. ++ * 1 - MAX_SCHED is a valid event id. ++ * ++ * \return Nothing ++ */ ++void pri_schedule_del(struct pri *ctrl, int id) + { +- while (pri->master) +- pri = pri->master; +- if ((id >= MAX_SCHED) || (id < 0)) +- pri_error(pri, "Asked to delete sched id %d???\n", id); +- pri->pri_sched[id].callback = NULL; ++ /* Scheduling runs on master channels only */ ++ while (ctrl->master) { ++ ctrl = ctrl->master; ++ } ++ if (0 < id && id <= MAX_SCHED) { ++ ctrl->pri_sched[id - 1].callback = NULL; ++ } else if (id) { ++ pri_error(ctrl, "Asked to delete sched id %d???\n", id); ++ } + } +Index: rose_qsig_mwi.c +=================================================================== +--- a/rose_qsig_mwi.c (.../tags/1.4.10.2) (revision 0) ++++ b/rose_qsig_mwi.c (.../branches/1.4) (revision 1357) +@@ -0,0 +1,790 @@ ++/* ++ * libpri: An implementation of Primary Rate ISDN ++ * ++ * Copyright (C) 2009 Digium, Inc. ++ * ++ * Richard Mudgett <rmudgett@digium.com> ++ * ++ * See http://www.asterisk.org for more information about ++ * the Asterisk project. Please do not directly contact ++ * any of the maintainers of this project for assistance; ++ * the project provides a web site, mailing lists and IRC ++ * channels for your use. ++ * ++ * This program is free software, distributed under the terms of ++ * the GNU General Public License Version 2 as published by the ++ * Free Software Foundation. See the LICENSE file included with ++ * this program for more details. ++ * ++ * In addition, when this program is distributed with Asterisk in ++ * any form that would qualify as a 'combined work' or as a ++ * 'derivative work' (but not mere aggregation), you can redistribute ++ * and/or modify the combination under the terms of the license ++ * provided with that copy of Asterisk, instead of the license ++ * terms granted here. ++ */ ++ ++/*! ++ * \file ++ * \brief Q.SIG ROSE SS-MWI-Operations ++ * ++ * SS-MWI-Operations ECMA-242 Annex E Table E.1 ++ * ++ * \author Richard Mudgett <rmudgett@digium.com> ++ */ ++ ++ ++#include "compat.h" ++#include "libpri.h" ++#include "pri_internal.h" ++#include "rose.h" ++#include "rose_internal.h" ++#include "asn1.h" ++ ++ ++/* ------------------------------------------------------------------- */ ++ ++/*! ++ * \internal ++ * \brief Encode the MsgCentreId type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param msg_centre_id ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *rose_enc_qsig_MsgCentreId(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const struct roseQsigMsgCentreId *msg_centre_id) ++{ ++ unsigned char *seq_len; ++ ++ switch (msg_centre_id->type) { ++ case 0: /* integer */ ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 0, ++ msg_centre_id->u.integer)); ++ break; ++ case 1: /* partyNumber */ ++ /* EXPLICIT tag */ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 1); ++ ASN1_CALL(pos, rose_enc_PartyNumber(ctrl, pos, end, &msg_centre_id->u.number)); ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ break; ++ case 2: /* numericString */ ++ ASN1_CALL(pos, asn1_enc_string_max(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 2, ++ msg_centre_id->u.str, sizeof(msg_centre_id->u.str) - 1)); ++ break; ++ default: ++ ASN1_ENC_ERROR(ctrl, "Unknown MsgCentreId type"); ++ return NULL; ++ } ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the MWIActivate invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_qsig_MWIActivate_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ const struct roseQsigMWIActivateArg *mwi_activate; ++ unsigned char *seq_len; ++ unsigned char *explicit_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ASN1_TAG_SEQUENCE); ++ ++ mwi_activate = &args->qsig.MWIActivate; ++ ASN1_CALL(pos, rose_enc_PartyNumber(ctrl, pos, end, ++ &mwi_activate->served_user_number)); ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, ++ mwi_activate->basic_service)); ++ if (mwi_activate->msg_centre_id_present) { ++ ASN1_CALL(pos, rose_enc_qsig_MsgCentreId(ctrl, pos, end, ++ &mwi_activate->msg_centre_id)); ++ } ++ if (mwi_activate->number_of_messages_present) { ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 3, ++ mwi_activate->number_of_messages)); ++ } ++ if (mwi_activate->originating_number.length) { ++ /* EXPLICIT tag */ ++ ASN1_CONSTRUCTED_BEGIN(explicit_len, pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 4); ++ ASN1_CALL(pos, rose_enc_PartyNumber(ctrl, pos, end, ++ &mwi_activate->originating_number)); ++ ASN1_CONSTRUCTED_END(explicit_len, pos, end); ++ } ++ if (mwi_activate->timestamp_present) { ++ ASN1_CALL(pos, asn1_enc_string_max(pos, end, ASN1_TYPE_GENERALIZED_TIME, ++ mwi_activate->timestamp, sizeof(mwi_activate->timestamp) - 1)); ++ } ++ if (mwi_activate->priority_present) { ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 5, ++ mwi_activate->priority)); ++ } ++ ++ /* No extension to encode */ ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the MWIDeactivate invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_qsig_MWIDeactivate_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ const struct roseQsigMWIDeactivateArg *mwi_deactivate; ++ unsigned char *seq_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ASN1_TAG_SEQUENCE); ++ ++ mwi_deactivate = &args->qsig.MWIDeactivate; ++ ASN1_CALL(pos, rose_enc_PartyNumber(ctrl, pos, end, ++ &mwi_deactivate->served_user_number)); ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, ++ mwi_deactivate->basic_service)); ++ if (mwi_deactivate->msg_centre_id_present) { ++ ASN1_CALL(pos, rose_enc_qsig_MsgCentreId(ctrl, pos, end, ++ &mwi_deactivate->msg_centre_id)); ++ } ++ ++ /* No extension to encode */ ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the MWIInterrogate invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_qsig_MWIInterrogate_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ const struct roseQsigMWIInterrogateArg *mwi_interrogate; ++ unsigned char *seq_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ASN1_TAG_SEQUENCE); ++ ++ mwi_interrogate = &args->qsig.MWIInterrogate; ++ ASN1_CALL(pos, rose_enc_PartyNumber(ctrl, pos, end, ++ &mwi_interrogate->served_user_number)); ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, ++ mwi_interrogate->basic_service)); ++ if (mwi_interrogate->msg_centre_id_present) { ++ ASN1_CALL(pos, rose_enc_qsig_MsgCentreId(ctrl, pos, end, ++ &mwi_interrogate->msg_centre_id)); ++ } ++ ++ /* No extension to encode */ ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Encode the MWIInterrogateResElt type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param tag Component tag to identify the encoded component. ++ * The tag should be ASN1_TAG_SEQUENCE unless the caller implicitly ++ * tags it otherwise. ++ * \param record ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *rose_enc_qsig_MWIInterrogateResElt(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, unsigned tag, ++ const struct roseQsigMWIInterrogateResElt *record) ++{ ++ unsigned char *seq_len; ++ unsigned char *explicit_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, tag); ++ ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, record->basic_service)); ++ if (record->msg_centre_id_present) { ++ ASN1_CALL(pos, rose_enc_qsig_MsgCentreId(ctrl, pos, end, ++ &record->msg_centre_id)); ++ } ++ if (record->number_of_messages_present) { ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 3, ++ record->number_of_messages)); ++ } ++ if (record->originating_number.length) { ++ /* EXPLICIT tag */ ++ ASN1_CONSTRUCTED_BEGIN(explicit_len, pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 4); ++ ASN1_CALL(pos, rose_enc_PartyNumber(ctrl, pos, end, ++ &record->originating_number)); ++ ASN1_CONSTRUCTED_END(explicit_len, pos, end); ++ } ++ if (record->timestamp_present) { ++ ASN1_CALL(pos, asn1_enc_string_max(pos, end, ASN1_TYPE_GENERALIZED_TIME, ++ record->timestamp, sizeof(record->timestamp) - 1)); ++ } ++ if (record->priority_present) { ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 5, ++ record->priority)); ++ } ++ ++ /* No extension to encode */ ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the MWIInterrogate result facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_qsig_MWIInterrogate_RES(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_result_args *args) ++{ ++ unsigned index; ++ unsigned char *seq_len; ++ const struct roseQsigMWIInterrogateRes *mwi_interrogate; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ASN1_TAG_SEQUENCE); ++ ++ mwi_interrogate = &args->qsig.MWIInterrogate; ++ for (index = 0; index < mwi_interrogate->num_records; ++index) { ++ ASN1_CALL(pos, rose_enc_qsig_MWIInterrogateResElt(ctrl, pos, end, ++ ASN1_TAG_SEQUENCE, &mwi_interrogate->list[index])); ++ } ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Decode the MsgCentreId argument parameters. ++ * ++ * \param ctrl D channel controller for any diagnostic messages. ++ * \param name Field name ++ * \param tag Component tag that identified this production. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param msg_centre_id Parameter storage to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++static const unsigned char *rose_dec_qsig_MsgCentreId(struct pri *ctrl, const char *name, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct roseQsigMsgCentreId *msg_centre_id) ++{ ++ int32_t value; ++ size_t str_len; ++ int length; ++ int explicit_offset; ++ const unsigned char *explicit_end; ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s MsgCentreId\n", name); ++ } ++ switch (tag) { ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 0: ++ msg_centre_id->type = 0; /* integer */ ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "integer", tag, pos, end, &value)); ++ msg_centre_id->u.integer = value; ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 1: ++ msg_centre_id->type = 1; /* partyNumber */ ++ ++ /* Remove EXPLICIT tag */ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " Explicit %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(explicit_end, explicit_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, explicit_end, &tag)); ++ ASN1_CALL(pos, rose_dec_PartyNumber(ctrl, "partyNumber", tag, pos, explicit_end, ++ &msg_centre_id->u.number)); ++ ++ ASN1_END_FIXUP(ctrl, pos, explicit_offset, explicit_end, end); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 2: ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 2: ++ msg_centre_id->type = 2; /* numericString */ ++ ASN1_CALL(pos, asn1_dec_string_max(ctrl, "numericString", tag, pos, end, ++ sizeof(msg_centre_id->u.str), msg_centre_id->u.str, &str_len)); ++ break; ++ default: ++ ASN1_DID_NOT_EXPECT_TAG(ctrl, tag); ++ return NULL; ++ } ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the Q.SIG MWIActivate invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_qsig_MWIActivate_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, union rose_msg_invoke_args *args) ++{ ++ int32_t value; ++ size_t str_len; ++ int length; ++ int seq_offset; ++ int explicit_offset; ++ const unsigned char *explicit_end; ++ const unsigned char *seq_end; ++ const unsigned char *save_pos; ++ struct roseQsigMWIActivateArg *mwi_activate; ++ ++ mwi_activate = &args->qsig.MWIActivate; ++ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " MWIActivateArg %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CALL(pos, rose_dec_PartyNumber(ctrl, "servedUserNr", tag, pos, seq_end, ++ &mwi_activate->served_user_number)); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_ENUMERATED); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "basicService", tag, pos, seq_end, &value)); ++ mwi_activate->basic_service = value; ++ ++ /* ++ * A sequence specifies an ordered list of component types. ++ * However, for simplicity we are not checking the order of ++ * the remaining optional components. ++ */ ++ mwi_activate->msg_centre_id_present = 0; ++ mwi_activate->number_of_messages_present = 0; ++ mwi_activate->originating_number.length = 0; ++ mwi_activate->timestamp_present = 0; ++ mwi_activate->priority_present = 0; ++ while (pos < seq_end && *pos != ASN1_INDEF_TERM) { ++ save_pos = pos; ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ switch (tag & ~ASN1_PC_MASK) { ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 0: ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 1: ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 2: ++ ASN1_CALL(pos, rose_dec_qsig_MsgCentreId(ctrl, "msgCentreId", tag, pos, ++ seq_end, &mwi_activate->msg_centre_id)); ++ mwi_activate->msg_centre_id_present = 1; ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 3: ++ /* Must not be constructed but we will not check for it for simplicity. */ ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "nbOfMessages", tag, pos, seq_end, ++ &value)); ++ mwi_activate->number_of_messages = value; ++ mwi_activate->number_of_messages_present = 1; ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 4: ++ /* Must be constructed but we will not check for it for simplicity. */ ++ /* Remove EXPLICIT tag */ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " Explicit %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, seq_end, &length)); ++ ASN1_END_SETUP(explicit_end, explicit_offset, length, pos, seq_end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, explicit_end, &tag)); ++ ASN1_CALL(pos, rose_dec_PartyNumber(ctrl, "originatingNr", tag, pos, ++ explicit_end, &mwi_activate->originating_number)); ++ ++ ASN1_END_FIXUP(ctrl, pos, explicit_offset, explicit_end, seq_end); ++ break; ++ case ASN1_TYPE_GENERALIZED_TIME: ++ ASN1_CALL(pos, asn1_dec_string_max(ctrl, "timestamp", tag, pos, end, ++ sizeof(mwi_activate->timestamp), mwi_activate->timestamp, &str_len)); ++ mwi_activate->timestamp_present = 1; ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 5: ++ /* Must not be constructed but we will not check for it for simplicity. */ ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "priority", tag, pos, seq_end, &value)); ++ mwi_activate->priority = value; ++ mwi_activate->priority_present = 1; ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 6: ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 7: ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " argumentExt %s\n", asn1_tag2str(tag)); ++ } ++ /* Fixup will skip over the manufacturer extension information */ ++ default: ++ pos = save_pos; ++ goto cancel_options; ++ } ++ } ++cancel_options:; ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the Q.SIG MWIDeactivate invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_qsig_MWIDeactivate_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, union rose_msg_invoke_args *args) ++{ ++ int32_t value; ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ const unsigned char *save_pos; ++ struct roseQsigMWIDeactivateArg *mwi_deactivate; ++ ++ mwi_deactivate = &args->qsig.MWIDeactivate; ++ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " MWIDeactivateArg %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CALL(pos, rose_dec_PartyNumber(ctrl, "servedUserNr", tag, pos, seq_end, ++ &mwi_deactivate->served_user_number)); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_ENUMERATED); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "basicService", tag, pos, seq_end, &value)); ++ mwi_deactivate->basic_service = value; ++ ++ /* ++ * A sequence specifies an ordered list of component types. ++ * However, for simplicity we are not checking the order of ++ * the remaining optional components. ++ */ ++ mwi_deactivate->msg_centre_id_present = 0; ++ while (pos < seq_end && *pos != ASN1_INDEF_TERM) { ++ save_pos = pos; ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ switch (tag & ~ASN1_PC_MASK) { ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 0: ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 1: ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 2: ++ ASN1_CALL(pos, rose_dec_qsig_MsgCentreId(ctrl, "msgCentreId", tag, pos, ++ seq_end, &mwi_deactivate->msg_centre_id)); ++ mwi_deactivate->msg_centre_id_present = 1; ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 3: ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 4: ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " argumentExt %s\n", asn1_tag2str(tag)); ++ } ++ /* Fixup will skip over the manufacturer extension information */ ++ default: ++ pos = save_pos; ++ goto cancel_options; ++ } ++ } ++cancel_options:; ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the Q.SIG MWIInterrogate invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_qsig_MWIInterrogate_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, union rose_msg_invoke_args *args) ++{ ++ int32_t value; ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ const unsigned char *save_pos; ++ struct roseQsigMWIInterrogateArg *mwi_interrogate; ++ ++ mwi_interrogate = &args->qsig.MWIInterrogate; ++ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " MWIInterrogateArg %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CALL(pos, rose_dec_PartyNumber(ctrl, "servedUserNr", tag, pos, seq_end, ++ &mwi_interrogate->served_user_number)); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_ENUMERATED); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "basicService", tag, pos, seq_end, &value)); ++ mwi_interrogate->basic_service = value; ++ ++ /* ++ * A sequence specifies an ordered list of component types. ++ * However, for simplicity we are not checking the order of ++ * the remaining optional components. ++ */ ++ mwi_interrogate->msg_centre_id_present = 0; ++ while (pos < seq_end && *pos != ASN1_INDEF_TERM) { ++ save_pos = pos; ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ switch (tag & ~ASN1_PC_MASK) { ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 0: ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 1: ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 2: ++ ASN1_CALL(pos, rose_dec_qsig_MsgCentreId(ctrl, "msgCentreId", tag, pos, ++ seq_end, &mwi_interrogate->msg_centre_id)); ++ mwi_interrogate->msg_centre_id_present = 1; ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 3: ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 4: ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " argumentExt %s\n", asn1_tag2str(tag)); ++ } ++ /* Fixup will skip over the manufacturer extension information */ ++ default: ++ pos = save_pos; ++ goto cancel_options; ++ } ++ } ++cancel_options:; ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Decode the MWIInterrogateResElt argument parameters. ++ * ++ * \param ctrl D channel controller for any diagnostic messages. ++ * \param name Field name ++ * \param tag Component tag that identified this production. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param record Parameter storage to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++static const unsigned char *rose_dec_qsig_MWIInterrogateResElt(struct pri *ctrl, ++ const char *name, unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct roseQsigMWIInterrogateResElt *record) ++{ ++ int32_t value; ++ size_t str_len; ++ int length; ++ int seq_offset; ++ int explicit_offset; ++ const unsigned char *explicit_end; ++ const unsigned char *seq_end; ++ const unsigned char *save_pos; ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " MWIInterrogateResElt %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_ENUMERATED); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "basicService", tag, pos, seq_end, &value)); ++ record->basic_service = value; ++ ++ /* ++ * A sequence specifies an ordered list of component types. ++ * However, for simplicity we are not checking the order of ++ * the remaining optional components. ++ */ ++ record->msg_centre_id_present = 0; ++ record->number_of_messages_present = 0; ++ record->originating_number.length = 0; ++ record->timestamp_present = 0; ++ record->priority_present = 0; ++ while (pos < seq_end && *pos != ASN1_INDEF_TERM) { ++ save_pos = pos; ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ switch (tag & ~ASN1_PC_MASK) { ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 0: ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 1: ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 2: ++ ASN1_CALL(pos, rose_dec_qsig_MsgCentreId(ctrl, "msgCentreId", tag, pos, ++ seq_end, &record->msg_centre_id)); ++ record->msg_centre_id_present = 1; ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 3: ++ /* Must not be constructed but we will not check for it for simplicity. */ ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "nbOfMessages", tag, pos, seq_end, ++ &value)); ++ record->number_of_messages = value; ++ record->number_of_messages_present = 1; ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 4: ++ /* Must be constructed but we will not check for it for simplicity. */ ++ /* Remove EXPLICIT tag */ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " Explicit %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, seq_end, &length)); ++ ASN1_END_SETUP(explicit_end, explicit_offset, length, pos, seq_end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, explicit_end, &tag)); ++ ASN1_CALL(pos, rose_dec_PartyNumber(ctrl, "originatingNr", tag, pos, ++ explicit_end, &record->originating_number)); ++ ++ ASN1_END_FIXUP(ctrl, pos, explicit_offset, explicit_end, seq_end); ++ break; ++ case ASN1_TYPE_GENERALIZED_TIME: ++ ASN1_CALL(pos, asn1_dec_string_max(ctrl, "timestamp", tag, pos, end, ++ sizeof(record->timestamp), record->timestamp, &str_len)); ++ record->timestamp_present = 1; ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 5: ++ /* Must not be constructed but we will not check for it for simplicity. */ ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "priority", tag, pos, seq_end, &value)); ++ record->priority = value; ++ record->priority_present = 1; ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 6: ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 7: ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " argumentExt %s\n", asn1_tag2str(tag)); ++ } ++ /* Fixup will skip over the manufacturer extension information */ ++ default: ++ pos = save_pos; ++ goto cancel_options; ++ } ++ } ++cancel_options:; ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the Q.SIG MWIInterrogate result argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_qsig_MWIInterrogate_RES(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, union rose_msg_result_args *args) ++{ ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ struct roseQsigMWIInterrogateRes *mwi_interrogate; ++ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " MWIInterrogateRes %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ mwi_interrogate = &args->qsig.MWIInterrogate; ++ ++ mwi_interrogate->num_records = 0; ++ while (pos < seq_end && *pos != ASN1_INDEF_TERM) { ++ if (mwi_interrogate->num_records < ARRAY_LEN(mwi_interrogate->list)) { ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ ASN1_CALL(pos, rose_dec_qsig_MWIInterrogateResElt(ctrl, "listEntry", tag, ++ pos, seq_end, &mwi_interrogate->list[mwi_interrogate->num_records])); ++ ++mwi_interrogate->num_records; ++ } else { ++ /* Too many records */ ++ return NULL; ++ } ++ } ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++ ++/* ------------------------------------------------------------------- */ ++/* end rose_qsig_mwi.c */ + +Property changes on: rose_qsig_mwi.c +___________________________________________________________________ +Added: svn:eol-style + + native +Added: svn:mime-type + + text/plain +Added: svn:keywords + + 'Author Date Id Revision' + +Index: rosetest.c +=================================================================== +--- a/rosetest.c (.../tags/1.4.10.2) (revision 0) ++++ b/rosetest.c (.../branches/1.4) (revision 1357) +@@ -0,0 +1,2473 @@ ++/* ++ * libpri: An implementation of Primary Rate ISDN ++ * ++ * Copyright (C) 2009 Digium, Inc. ++ * ++ * Richard Mudgett <rmudgett@digium.com> ++ * ++ * See http://www.asterisk.org for more information about ++ * the Asterisk project. Please do not directly contact ++ * any of the maintainers of this project for assistance; ++ * the project provides a web site, mailing lists and IRC ++ * channels for your use. ++ * ++ * This program is free software, distributed under the terms of ++ * the GNU General Public License Version 2 as published by the ++ * Free Software Foundation. See the LICENSE file included with ++ * this program for more details. ++ * ++ * In addition, when this program is distributed with Asterisk in ++ * any form that would qualify as a 'combined work' or as a ++ * 'derivative work' (but not mere aggregation), you can redistribute ++ * and/or modify the combination under the terms of the license ++ * provided with that copy of Asterisk, instead of the license ++ * terms granted here. ++ */ ++ ++/*! ++ * \file ++ * \brief ROSE encode/decode test program ++ * ++ * \author Richard Mudgett <rmudgett@digium.com> ++ */ ++ ++ ++#include "compat.h" ++#include "libpri.h" ++#include "pri_internal.h" ++#include "rose.h" ++ ++#include <stdio.h> ++#include <stdlib.h> ++ ++ ++/* ------------------------------------------------------------------- */ ++ ++ ++static const struct fac_extension_header fac_headers[] = { ++/* *INDENT-OFF* */ ++ { ++ .nfe_present = 0, ++ }, ++ { ++ .nfe_present = 1, ++ .nfe.source_entity = 1, ++ .nfe.destination_entity = 1, ++ }, ++ { ++ .nfe_present = 1, ++ .nfe.source_entity = 1, ++ .nfe.source_number.plan = 4, ++ .nfe.source_number.length = 4, ++ .nfe.source_number.str = "9834", ++ .nfe.destination_entity = 1, ++ .nfe.destination_number.plan = 4, ++ .nfe.destination_number.length = 4, ++ .nfe.destination_number.str = "9834", ++ }, ++ { ++ .nfe_present = 1, ++ .nfe.source_entity = 1, ++ .nfe.destination_entity = 1, ++ .npp_present = 1, ++ .npp = 19, ++ .interpretation_present = 1, ++ .interpretation = 2, ++ }, ++/* *INDENT-ON* */ ++}; ++ ++ ++static const struct rose_message rose_etsi_msgs[] = { ++/* *INDENT-OFF* */ ++ /* Error messages */ ++ { ++ .type = ROSE_COMP_TYPE_ERROR, ++ .component.error.invoke_id = 82, ++ .component.error.code = ROSE_ERROR_Div_SpecialServiceNr, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_ERROR, ++ .component.error.invoke_id = 8, ++ .component.error.code = ROSE_ERROR_ECT_LinkIdNotAssignedByNetwork, ++ }, ++ ++ /* Reject messages */ ++ { ++ .type = ROSE_COMP_TYPE_REJECT, ++ .component.reject.code = ROSE_REJECT_Gen_BadlyStructuredComponent, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_REJECT, ++ .component.reject.invoke_id_present = 1, ++ .component.reject.invoke_id = 10, ++ .component.reject.code = ROSE_REJECT_Inv_InitiatorReleasing, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_REJECT, ++ .component.reject.invoke_id_present = 1, ++ .component.reject.invoke_id = 11, ++ .component.reject.code = ROSE_REJECT_Res_MistypedResult, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_REJECT, ++ .component.reject.invoke_id_present = 1, ++ .component.reject.invoke_id = 12, ++ .component.reject.code = ROSE_REJECT_Err_ErrorResponseUnexpected, ++ }, ++ ++ /* Anonymous result or result without any arguments. */ ++ { ++ .type = ROSE_COMP_TYPE_RESULT, ++ .component.result.operation = ROSE_None, ++ .component.result.invoke_id = 9, ++ }, ++ ++ /* Advice Of Charge (AOC) */ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_ChargingRequest, ++ .component.invoke.invoke_id = 98, ++ .component.invoke.args.etsi.ChargingRequest.charging_case = 2, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_RESULT, ++ .component.result.operation = ROSE_ETSI_ChargingRequest, ++ .component.result.invoke_id = 99, ++ .component.result.args.etsi.ChargingRequest.type = 0, ++ .component.result.args.etsi.ChargingRequest.u.currency_info.num_records = 1, ++ .component.result.args.etsi.ChargingRequest.u.currency_info.list[0].charged_item = 4, ++ .component.result.args.etsi.ChargingRequest.u.currency_info.list[0].currency_type = 0, ++ .component.result.args.etsi.ChargingRequest.u.currency_info.list[0].u.special_charging_code = 3, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_RESULT, ++ .component.result.operation = ROSE_ETSI_ChargingRequest, ++ .component.result.invoke_id = 100, ++ .component.result.args.etsi.ChargingRequest.type = 0, ++ .component.result.args.etsi.ChargingRequest.u.currency_info.num_records = 1, ++ .component.result.args.etsi.ChargingRequest.u.currency_info.list[0].charged_item = 4, ++ .component.result.args.etsi.ChargingRequest.u.currency_info.list[0].currency_type = 1, ++ .component.result.args.etsi.ChargingRequest.u.currency_info.list[0].u.duration.currency = "Dollars", ++ .component.result.args.etsi.ChargingRequest.u.currency_info.list[0].u.duration.amount.currency = 7, ++ .component.result.args.etsi.ChargingRequest.u.currency_info.list[0].u.duration.amount.multiplier = 1, ++ .component.result.args.etsi.ChargingRequest.u.currency_info.list[0].u.duration.charging_type = 1, ++ .component.result.args.etsi.ChargingRequest.u.currency_info.list[0].u.duration.time.length = 8, ++ .component.result.args.etsi.ChargingRequest.u.currency_info.list[0].u.duration.time.scale = 4, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_RESULT, ++ .component.result.operation = ROSE_ETSI_ChargingRequest, ++ .component.result.invoke_id = 101, ++ .component.result.args.etsi.ChargingRequest.type = 0, ++ .component.result.args.etsi.ChargingRequest.u.currency_info.num_records = 1, ++ .component.result.args.etsi.ChargingRequest.u.currency_info.list[0].charged_item = 4, ++ .component.result.args.etsi.ChargingRequest.u.currency_info.list[0].currency_type = 1, ++ .component.result.args.etsi.ChargingRequest.u.currency_info.list[0].u.duration.currency = "Dollars", ++ .component.result.args.etsi.ChargingRequest.u.currency_info.list[0].u.duration.amount.currency = 7, ++ .component.result.args.etsi.ChargingRequest.u.currency_info.list[0].u.duration.amount.multiplier = 1, ++ .component.result.args.etsi.ChargingRequest.u.currency_info.list[0].u.duration.charging_type = 1, ++ .component.result.args.etsi.ChargingRequest.u.currency_info.list[0].u.duration.time.length = 8, ++ .component.result.args.etsi.ChargingRequest.u.currency_info.list[0].u.duration.time.scale = 4, ++ .component.result.args.etsi.ChargingRequest.u.currency_info.list[0].u.duration.granularity_present = 1, ++ .component.result.args.etsi.ChargingRequest.u.currency_info.list[0].u.duration.granularity.length = 20, ++ .component.result.args.etsi.ChargingRequest.u.currency_info.list[0].u.duration.granularity.scale = 3, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_RESULT, ++ .component.result.operation = ROSE_ETSI_ChargingRequest, ++ .component.result.invoke_id = 102, ++ .component.result.args.etsi.ChargingRequest.type = 0, ++ .component.result.args.etsi.ChargingRequest.u.currency_info.num_records = 1, ++ .component.result.args.etsi.ChargingRequest.u.currency_info.list[0].charged_item = 4, ++ .component.result.args.etsi.ChargingRequest.u.currency_info.list[0].currency_type = 2, ++ .component.result.args.etsi.ChargingRequest.u.currency_info.list[0].u.flat_rate.currency = "Euros", ++ .component.result.args.etsi.ChargingRequest.u.currency_info.list[0].u.flat_rate.amount.currency = 4, ++ .component.result.args.etsi.ChargingRequest.u.currency_info.list[0].u.flat_rate.amount.multiplier = 1, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_RESULT, ++ .component.result.operation = ROSE_ETSI_ChargingRequest, ++ .component.result.invoke_id = 103, ++ .component.result.args.etsi.ChargingRequest.type = 0, ++ .component.result.args.etsi.ChargingRequest.u.currency_info.num_records = 1, ++ .component.result.args.etsi.ChargingRequest.u.currency_info.list[0].charged_item = 4, ++ .component.result.args.etsi.ChargingRequest.u.currency_info.list[0].currency_type = 3, ++ .component.result.args.etsi.ChargingRequest.u.currency_info.list[0].u.volume_rate.currency = "Yen", ++ .component.result.args.etsi.ChargingRequest.u.currency_info.list[0].u.volume_rate.amount.currency = 300, ++ .component.result.args.etsi.ChargingRequest.u.currency_info.list[0].u.volume_rate.amount.multiplier = 5, ++ .component.result.args.etsi.ChargingRequest.u.currency_info.list[0].u.volume_rate.unit = 2, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_RESULT, ++ .component.result.operation = ROSE_ETSI_ChargingRequest, ++ .component.result.invoke_id = 104, ++ .component.result.args.etsi.ChargingRequest.type = 0, ++ .component.result.args.etsi.ChargingRequest.u.currency_info.num_records = 2, ++ .component.result.args.etsi.ChargingRequest.u.currency_info.list[0].charged_item = 4, ++ .component.result.args.etsi.ChargingRequest.u.currency_info.list[0].currency_type = 2, ++ .component.result.args.etsi.ChargingRequest.u.currency_info.list[0].u.flat_rate.currency = "Euros", ++ .component.result.args.etsi.ChargingRequest.u.currency_info.list[0].u.flat_rate.amount.currency = 4, ++ .component.result.args.etsi.ChargingRequest.u.currency_info.list[0].u.flat_rate.amount.multiplier = 1, ++ .component.result.args.etsi.ChargingRequest.u.currency_info.list[1].charged_item = 4, ++ .component.result.args.etsi.ChargingRequest.u.currency_info.list[1].currency_type = 3, ++ .component.result.args.etsi.ChargingRequest.u.currency_info.list[1].u.volume_rate.currency = "Yen", ++ .component.result.args.etsi.ChargingRequest.u.currency_info.list[1].u.volume_rate.amount.currency = 300, ++ .component.result.args.etsi.ChargingRequest.u.currency_info.list[1].u.volume_rate.amount.multiplier = 5, ++ .component.result.args.etsi.ChargingRequest.u.currency_info.list[1].u.volume_rate.unit = 2, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_RESULT, ++ .component.result.operation = ROSE_ETSI_ChargingRequest, ++ .component.result.invoke_id = 105, ++ .component.result.args.etsi.ChargingRequest.type = 0, ++ .component.result.args.etsi.ChargingRequest.u.currency_info.num_records = 1, ++ .component.result.args.etsi.ChargingRequest.u.currency_info.list[0].charged_item = 4, ++ .component.result.args.etsi.ChargingRequest.u.currency_info.list[0].currency_type = 4, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_RESULT, ++ .component.result.operation = ROSE_ETSI_ChargingRequest, ++ .component.result.invoke_id = 106, ++ .component.result.args.etsi.ChargingRequest.type = 0, ++ .component.result.args.etsi.ChargingRequest.u.currency_info.num_records = 1, ++ .component.result.args.etsi.ChargingRequest.u.currency_info.list[0].charged_item = 4, ++ .component.result.args.etsi.ChargingRequest.u.currency_info.list[0].currency_type = 5, ++ }, ++ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_AOCSCurrency, ++ .component.invoke.invoke_id = 107, ++ .component.invoke.args.etsi.AOCSCurrency.type = 0, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_AOCSCurrency, ++ .component.invoke.invoke_id = 108, ++ .component.invoke.args.etsi.AOCSCurrency.type = 1, ++ .component.invoke.args.etsi.AOCSCurrency.currency_info.num_records = 1, ++ .component.invoke.args.etsi.AOCSCurrency.currency_info.list[0].charged_item = 3, ++ .component.invoke.args.etsi.AOCSCurrency.currency_info.list[0].currency_type = 4, ++ }, ++ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_AOCSSpecialArr, ++ .component.invoke.invoke_id = 109, ++ .component.invoke.args.etsi.AOCSSpecialArr.type = 0, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_AOCSSpecialArr, ++ .component.invoke.invoke_id = 110, ++ .component.invoke.args.etsi.AOCSSpecialArr.type = 1, ++ .component.invoke.args.etsi.AOCSSpecialArr.special_arrangement = 9, ++ }, ++ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_AOCDCurrency, ++ .component.invoke.invoke_id = 111, ++ .component.invoke.args.etsi.AOCDCurrency.type = 0, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_AOCDCurrency, ++ .component.invoke.invoke_id = 112, ++ .component.invoke.args.etsi.AOCDCurrency.type = 1, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_AOCDCurrency, ++ .component.invoke.invoke_id = 113, ++ .component.invoke.args.etsi.AOCDCurrency.type = 2, ++ .component.invoke.args.etsi.AOCDCurrency.specific.recorded.currency = "Francs", ++ .component.invoke.args.etsi.AOCDCurrency.specific.recorded.amount.currency = 674, ++ .component.invoke.args.etsi.AOCDCurrency.specific.recorded.amount.multiplier = 3, ++ .component.invoke.args.etsi.AOCDCurrency.specific.type_of_charging_info = 1, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_AOCDCurrency, ++ .component.invoke.invoke_id = 114, ++ .component.invoke.args.etsi.AOCDCurrency.type = 2, ++ .component.invoke.args.etsi.AOCDCurrency.specific.recorded.currency = "Francs", ++ .component.invoke.args.etsi.AOCDCurrency.specific.recorded.amount.currency = 674, ++ .component.invoke.args.etsi.AOCDCurrency.specific.recorded.amount.multiplier = 3, ++ .component.invoke.args.etsi.AOCDCurrency.specific.type_of_charging_info = 1, ++ .component.invoke.args.etsi.AOCDCurrency.specific.billing_id_present = 1, ++ .component.invoke.args.etsi.AOCDCurrency.specific.billing_id = 2, ++ }, ++ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_AOCDChargingUnit, ++ .component.invoke.invoke_id = 115, ++ .component.invoke.args.etsi.AOCDChargingUnit.type = 0, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_AOCDChargingUnit, ++ .component.invoke.invoke_id = 116, ++ .component.invoke.args.etsi.AOCDChargingUnit.type = 1, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_AOCDChargingUnit, ++ .component.invoke.invoke_id = 117, ++ .component.invoke.args.etsi.AOCDChargingUnit.type = 2, ++ .component.invoke.args.etsi.AOCDChargingUnit.specific.recorded.num_records = 1, ++ .component.invoke.args.etsi.AOCDChargingUnit.specific.recorded.list[0].not_available = 1, ++ .component.invoke.args.etsi.AOCDChargingUnit.specific.type_of_charging_info = 1, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_AOCDChargingUnit, ++ .component.invoke.invoke_id = 118, ++ .component.invoke.args.etsi.AOCDChargingUnit.type = 2, ++ .component.invoke.args.etsi.AOCDChargingUnit.specific.recorded.num_records = 1, ++ .component.invoke.args.etsi.AOCDChargingUnit.specific.recorded.list[0].not_available = 0, ++ .component.invoke.args.etsi.AOCDChargingUnit.specific.recorded.list[0].number_of_units = 8523, ++ .component.invoke.args.etsi.AOCDChargingUnit.specific.type_of_charging_info = 1, ++ .component.invoke.args.etsi.AOCDChargingUnit.specific.billing_id_present = 1, ++ .component.invoke.args.etsi.AOCDChargingUnit.specific.billing_id = 2, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_AOCDChargingUnit, ++ .component.invoke.invoke_id = 119, ++ .component.invoke.args.etsi.AOCDChargingUnit.type = 2, ++ .component.invoke.args.etsi.AOCDChargingUnit.specific.recorded.num_records = 1, ++ .component.invoke.args.etsi.AOCDChargingUnit.specific.recorded.list[0].not_available = 1, ++ .component.invoke.args.etsi.AOCDChargingUnit.specific.recorded.list[0].type_of_unit_present = 1, ++ .component.invoke.args.etsi.AOCDChargingUnit.specific.recorded.list[0].type_of_unit = 13, ++ .component.invoke.args.etsi.AOCDChargingUnit.specific.type_of_charging_info = 1, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_AOCDChargingUnit, ++ .component.invoke.invoke_id = 120, ++ .component.invoke.args.etsi.AOCDChargingUnit.type = 2, ++ .component.invoke.args.etsi.AOCDChargingUnit.specific.recorded.num_records = 1, ++ .component.invoke.args.etsi.AOCDChargingUnit.specific.recorded.list[0].not_available = 0, ++ .component.invoke.args.etsi.AOCDChargingUnit.specific.recorded.list[0].number_of_units = 8523, ++ .component.invoke.args.etsi.AOCDChargingUnit.specific.recorded.list[0].type_of_unit_present = 1, ++ .component.invoke.args.etsi.AOCDChargingUnit.specific.recorded.list[0].type_of_unit = 13, ++ .component.invoke.args.etsi.AOCDChargingUnit.specific.type_of_charging_info = 1, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_AOCDChargingUnit, ++ .component.invoke.invoke_id = 121, ++ .component.invoke.args.etsi.AOCDChargingUnit.type = 2, ++ .component.invoke.args.etsi.AOCDChargingUnit.specific.recorded.num_records = 2, ++ .component.invoke.args.etsi.AOCDChargingUnit.specific.recorded.list[0].not_available = 1, ++ .component.invoke.args.etsi.AOCDChargingUnit.specific.recorded.list[1].not_available = 0, ++ .component.invoke.args.etsi.AOCDChargingUnit.specific.recorded.list[1].number_of_units = 8523, ++ .component.invoke.args.etsi.AOCDChargingUnit.specific.recorded.list[1].type_of_unit_present = 1, ++ .component.invoke.args.etsi.AOCDChargingUnit.specific.recorded.list[1].type_of_unit = 13, ++ .component.invoke.args.etsi.AOCDChargingUnit.specific.type_of_charging_info = 1, ++ }, ++ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_AOCECurrency, ++ .component.invoke.invoke_id = 122, ++ .component.invoke.args.etsi.AOCECurrency.type = 0, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_AOCECurrency, ++ .component.invoke.invoke_id = 123, ++ .component.invoke.args.etsi.AOCECurrency.type = 1, ++ .component.invoke.args.etsi.AOCECurrency.currency_info.free_of_charge = 1, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_AOCECurrency, ++ .component.invoke.invoke_id = 124, ++ .component.invoke.args.etsi.AOCECurrency.type = 1, ++ .component.invoke.args.etsi.AOCECurrency.currency_info.free_of_charge = 1, ++ .component.invoke.args.etsi.AOCECurrency.currency_info.charging_association_present = 1, ++ .component.invoke.args.etsi.AOCECurrency.currency_info.charging_association.type = 0, ++ .component.invoke.args.etsi.AOCECurrency.currency_info.charging_association.id = -37, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_AOCECurrency, ++ .component.invoke.invoke_id = 125, ++ .component.invoke.args.etsi.AOCECurrency.type = 1, ++ .component.invoke.args.etsi.AOCECurrency.currency_info.free_of_charge = 1, ++ .component.invoke.args.etsi.AOCECurrency.currency_info.charging_association_present = 1, ++ .component.invoke.args.etsi.AOCECurrency.currency_info.charging_association.type = 1, ++ .component.invoke.args.etsi.AOCECurrency.currency_info.charging_association.number.plan = 0, ++ .component.invoke.args.etsi.AOCECurrency.currency_info.charging_association.number.length = 7, ++ .component.invoke.args.etsi.AOCECurrency.currency_info.charging_association.number.str = "5551212", ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_AOCECurrency, ++ .component.invoke.invoke_id = 126, ++ .component.invoke.args.etsi.AOCECurrency.type = 1, ++ .component.invoke.args.etsi.AOCECurrency.currency_info.free_of_charge = 0, ++ .component.invoke.args.etsi.AOCECurrency.currency_info.specific.recorded.currency = "Francs", ++ .component.invoke.args.etsi.AOCECurrency.currency_info.specific.recorded.amount.currency = 674, ++ .component.invoke.args.etsi.AOCECurrency.currency_info.specific.recorded.amount.multiplier = 3, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_AOCECurrency, ++ .component.invoke.invoke_id = 127, ++ .component.invoke.args.etsi.AOCECurrency.type = 1, ++ .component.invoke.args.etsi.AOCECurrency.currency_info.free_of_charge = 0, ++ .component.invoke.args.etsi.AOCECurrency.currency_info.specific.recorded.currency = "Francs", ++ .component.invoke.args.etsi.AOCECurrency.currency_info.specific.recorded.amount.currency = 674, ++ .component.invoke.args.etsi.AOCECurrency.currency_info.specific.recorded.amount.multiplier = 3, ++ .component.invoke.args.etsi.AOCECurrency.currency_info.charging_association_present = 1, ++ .component.invoke.args.etsi.AOCECurrency.currency_info.charging_association.type = 0, ++ .component.invoke.args.etsi.AOCECurrency.currency_info.charging_association.id = -37, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_AOCECurrency, ++ .component.invoke.invoke_id = 128, ++ .component.invoke.args.etsi.AOCECurrency.type = 1, ++ .component.invoke.args.etsi.AOCECurrency.currency_info.free_of_charge = 0, ++ .component.invoke.args.etsi.AOCECurrency.currency_info.specific.recorded.currency = "Francs", ++ .component.invoke.args.etsi.AOCECurrency.currency_info.specific.recorded.amount.currency = 674, ++ .component.invoke.args.etsi.AOCECurrency.currency_info.specific.recorded.amount.multiplier = 3, ++ .component.invoke.args.etsi.AOCECurrency.currency_info.specific.billing_id_present = 1, ++ .component.invoke.args.etsi.AOCECurrency.currency_info.specific.billing_id = 2, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_AOCECurrency, ++ .component.invoke.invoke_id = 129, ++ .component.invoke.args.etsi.AOCECurrency.type = 1, ++ .component.invoke.args.etsi.AOCECurrency.currency_info.free_of_charge = 0, ++ .component.invoke.args.etsi.AOCECurrency.currency_info.specific.recorded.currency = "Francs", ++ .component.invoke.args.etsi.AOCECurrency.currency_info.specific.recorded.amount.currency = 674, ++ .component.invoke.args.etsi.AOCECurrency.currency_info.specific.recorded.amount.multiplier = 3, ++ .component.invoke.args.etsi.AOCECurrency.currency_info.specific.billing_id_present = 1, ++ .component.invoke.args.etsi.AOCECurrency.currency_info.specific.billing_id = 2, ++ .component.invoke.args.etsi.AOCECurrency.currency_info.charging_association_present = 1, ++ .component.invoke.args.etsi.AOCECurrency.currency_info.charging_association.type = 0, ++ .component.invoke.args.etsi.AOCECurrency.currency_info.charging_association.id = -37, ++ }, ++ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_AOCEChargingUnit, ++ .component.invoke.invoke_id = 130, ++ .component.invoke.args.etsi.AOCEChargingUnit.type = 0, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_AOCEChargingUnit, ++ .component.invoke.invoke_id = 131, ++ .component.invoke.args.etsi.AOCEChargingUnit.type = 1, ++ .component.invoke.args.etsi.AOCEChargingUnit.charging_unit.free_of_charge = 1, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_AOCEChargingUnit, ++ .component.invoke.invoke_id = 132, ++ .component.invoke.args.etsi.AOCEChargingUnit.type = 1, ++ .component.invoke.args.etsi.AOCEChargingUnit.charging_unit.free_of_charge = 1, ++ .component.invoke.args.etsi.AOCEChargingUnit.charging_unit.charging_association_present = 1, ++ .component.invoke.args.etsi.AOCEChargingUnit.charging_unit.charging_association.type = 0, ++ .component.invoke.args.etsi.AOCEChargingUnit.charging_unit.charging_association.id = -37, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_AOCEChargingUnit, ++ .component.invoke.invoke_id = 133, ++ .component.invoke.args.etsi.AOCEChargingUnit.type = 1, ++ .component.invoke.args.etsi.AOCEChargingUnit.charging_unit.free_of_charge = 0, ++ .component.invoke.args.etsi.AOCEChargingUnit.charging_unit.specific.recorded.num_records = 1, ++ .component.invoke.args.etsi.AOCEChargingUnit.charging_unit.specific.recorded.list[0].not_available = 1, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_AOCEChargingUnit, ++ .component.invoke.invoke_id = 134, ++ .component.invoke.args.etsi.AOCEChargingUnit.type = 1, ++ .component.invoke.args.etsi.AOCEChargingUnit.charging_unit.free_of_charge = 0, ++ .component.invoke.args.etsi.AOCEChargingUnit.charging_unit.specific.recorded.num_records = 1, ++ .component.invoke.args.etsi.AOCEChargingUnit.charging_unit.specific.recorded.list[0].not_available = 1, ++ .component.invoke.args.etsi.AOCEChargingUnit.charging_unit.charging_association_present = 1, ++ .component.invoke.args.etsi.AOCEChargingUnit.charging_unit.charging_association.type = 0, ++ .component.invoke.args.etsi.AOCEChargingUnit.charging_unit.charging_association.id = -37, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_AOCEChargingUnit, ++ .component.invoke.invoke_id = 135, ++ .component.invoke.args.etsi.AOCEChargingUnit.type = 1, ++ .component.invoke.args.etsi.AOCEChargingUnit.charging_unit.free_of_charge = 0, ++ .component.invoke.args.etsi.AOCEChargingUnit.charging_unit.specific.recorded.num_records = 1, ++ .component.invoke.args.etsi.AOCEChargingUnit.charging_unit.specific.recorded.list[0].not_available = 1, ++ .component.invoke.args.etsi.AOCEChargingUnit.charging_unit.specific.billing_id_present = 1, ++ .component.invoke.args.etsi.AOCEChargingUnit.charging_unit.specific.billing_id = 2, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_AOCEChargingUnit, ++ .component.invoke.invoke_id = 136, ++ .component.invoke.args.etsi.AOCEChargingUnit.type = 1, ++ .component.invoke.args.etsi.AOCEChargingUnit.charging_unit.free_of_charge = 0, ++ .component.invoke.args.etsi.AOCEChargingUnit.charging_unit.specific.recorded.num_records = 1, ++ .component.invoke.args.etsi.AOCEChargingUnit.charging_unit.specific.recorded.list[0].not_available = 1, ++ .component.invoke.args.etsi.AOCEChargingUnit.charging_unit.specific.billing_id_present = 1, ++ .component.invoke.args.etsi.AOCEChargingUnit.charging_unit.specific.billing_id = 2, ++ .component.invoke.args.etsi.AOCEChargingUnit.charging_unit.charging_association_present = 1, ++ .component.invoke.args.etsi.AOCEChargingUnit.charging_unit.charging_association.type = 0, ++ .component.invoke.args.etsi.AOCEChargingUnit.charging_unit.charging_association.id = -37, ++ }, ++ ++ /* Call diversion */ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_ActivationDiversion, ++ .component.invoke.invoke_id = 67, ++ .component.invoke.linked_id_present = 1, ++ .component.invoke.linked_id = 27, ++ .component.invoke.args.etsi.ActivationDiversion.procedure = 2, ++ .component.invoke.args.etsi.ActivationDiversion.basic_service = 3, ++ .component.invoke.args.etsi.ActivationDiversion.forwarded_to.number.plan = 4, ++ .component.invoke.args.etsi.ActivationDiversion.forwarded_to.number.length = 4, ++ .component.invoke.args.etsi.ActivationDiversion.forwarded_to.number.str = "1803", ++ .component.invoke.args.etsi.ActivationDiversion.served_user_number.plan = 4, ++ .component.invoke.args.etsi.ActivationDiversion.served_user_number.length = 4, ++ .component.invoke.args.etsi.ActivationDiversion.served_user_number.str = "5398", ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_ActivationDiversion, ++ .component.invoke.invoke_id = 68, ++ .component.invoke.args.etsi.ActivationDiversion.procedure = 1, ++ .component.invoke.args.etsi.ActivationDiversion.basic_service = 5, ++ .component.invoke.args.etsi.ActivationDiversion.forwarded_to.number.plan = 4, ++ .component.invoke.args.etsi.ActivationDiversion.forwarded_to.number.length = 4, ++ .component.invoke.args.etsi.ActivationDiversion.forwarded_to.number.str = "1803", ++ }, ++ { ++ .type = ROSE_COMP_TYPE_RESULT, ++ .component.result.operation = ROSE_ETSI_ActivationDiversion, ++ .component.result.invoke_id = 69, ++ }, ++ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_DeactivationDiversion, ++ .component.invoke.invoke_id = 70, ++ .component.invoke.args.etsi.DeactivationDiversion.procedure = 1, ++ .component.invoke.args.etsi.DeactivationDiversion.basic_service = 5, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_RESULT, ++ .component.result.operation = ROSE_ETSI_DeactivationDiversion, ++ .component.result.invoke_id = 71, ++ }, ++ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_ActivationStatusNotificationDiv, ++ .component.invoke.invoke_id = 72, ++ .component.invoke.args.etsi.ActivationStatusNotificationDiv.procedure = 1, ++ .component.invoke.args.etsi.ActivationStatusNotificationDiv.basic_service = 5, ++ .component.invoke.args.etsi.ActivationStatusNotificationDiv.forwarded_to.number.plan = 4, ++ .component.invoke.args.etsi.ActivationStatusNotificationDiv.forwarded_to.number.length = 4, ++ .component.invoke.args.etsi.ActivationStatusNotificationDiv.forwarded_to.number.str = "1803", ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_DeactivationStatusNotificationDiv, ++ .component.invoke.invoke_id = 73, ++ .component.invoke.args.etsi.DeactivationStatusNotificationDiv.procedure = 1, ++ .component.invoke.args.etsi.DeactivationStatusNotificationDiv.basic_service = 5, ++ }, ++ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_InterrogationDiversion, ++ .component.invoke.invoke_id = 74, ++ .component.invoke.args.etsi.InterrogationDiversion.procedure = 1, ++ .component.invoke.args.etsi.InterrogationDiversion.basic_service = 5, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_InterrogationDiversion, ++ .component.invoke.invoke_id = 75, ++ .component.invoke.args.etsi.InterrogationDiversion.procedure = 1, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_RESULT, ++ .component.result.operation = ROSE_ETSI_InterrogationDiversion, ++ .component.result.invoke_id = 76, ++ .component.result.args.etsi.InterrogationDiversion.num_records = 2, ++ .component.result.args.etsi.InterrogationDiversion.list[0].procedure = 2, ++ .component.result.args.etsi.InterrogationDiversion.list[0].basic_service = 5, ++ .component.result.args.etsi.InterrogationDiversion.list[0].forwarded_to.number.plan = 4, ++ .component.result.args.etsi.InterrogationDiversion.list[0].forwarded_to.number.length = 4, ++ .component.result.args.etsi.InterrogationDiversion.list[0].forwarded_to.number.str = "1803", ++ .component.result.args.etsi.InterrogationDiversion.list[1].procedure = 1, ++ .component.result.args.etsi.InterrogationDiversion.list[1].basic_service = 3, ++ .component.result.args.etsi.InterrogationDiversion.list[1].forwarded_to.number.plan = 4, ++ .component.result.args.etsi.InterrogationDiversion.list[1].forwarded_to.number.length = 4, ++ .component.result.args.etsi.InterrogationDiversion.list[1].forwarded_to.number.str = "1903", ++ .component.result.args.etsi.InterrogationDiversion.list[1].served_user_number.plan = 4, ++ .component.result.args.etsi.InterrogationDiversion.list[1].served_user_number.length = 4, ++ .component.result.args.etsi.InterrogationDiversion.list[1].served_user_number.str = "5398", ++ }, ++ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_DiversionInformation, ++ .component.invoke.invoke_id = 77, ++ .component.invoke.args.etsi.DiversionInformation.diversion_reason = 3, ++ .component.invoke.args.etsi.DiversionInformation.basic_service = 5, ++ .component.invoke.args.etsi.DiversionInformation.served_user_subaddress.type = 1, ++ .component.invoke.args.etsi.DiversionInformation.served_user_subaddress.length = 4, ++ .component.invoke.args.etsi.DiversionInformation.served_user_subaddress.u.nsap = "6492", ++ .component.invoke.args.etsi.DiversionInformation.calling_present = 1, ++ .component.invoke.args.etsi.DiversionInformation.calling.presentation = 0, ++ .component.invoke.args.etsi.DiversionInformation.calling.screened.screening_indicator = 3, ++ .component.invoke.args.etsi.DiversionInformation.calling.screened.number.plan = 4, ++ .component.invoke.args.etsi.DiversionInformation.calling.screened.number.length = 4, ++ .component.invoke.args.etsi.DiversionInformation.calling.screened.number.str = "1803", ++ .component.invoke.args.etsi.DiversionInformation.original_called_present = 1, ++ .component.invoke.args.etsi.DiversionInformation.original_called.presentation = 1, ++ .component.invoke.args.etsi.DiversionInformation.last_diverting_present = 1, ++ .component.invoke.args.etsi.DiversionInformation.last_diverting.presentation = 2, ++ .component.invoke.args.etsi.DiversionInformation.last_diverting_reason_present = 1, ++ .component.invoke.args.etsi.DiversionInformation.last_diverting_reason = 3, ++ .component.invoke.args.etsi.DiversionInformation.q931ie.length = 5, ++ .component.invoke.args.etsi.DiversionInformation.q931ie_contents = "79828", ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_DiversionInformation, ++ .component.invoke.invoke_id = 78, ++ .component.invoke.args.etsi.DiversionInformation.diversion_reason = 3, ++ .component.invoke.args.etsi.DiversionInformation.basic_service = 5, ++ .component.invoke.args.etsi.DiversionInformation.calling_present = 1, ++ .component.invoke.args.etsi.DiversionInformation.calling.presentation = 1, ++ .component.invoke.args.etsi.DiversionInformation.original_called_present = 1, ++ .component.invoke.args.etsi.DiversionInformation.original_called.presentation = 2, ++ .component.invoke.args.etsi.DiversionInformation.last_diverting_present = 1, ++ .component.invoke.args.etsi.DiversionInformation.last_diverting.presentation = 1, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_DiversionInformation, ++ .component.invoke.invoke_id = 79, ++ .component.invoke.args.etsi.DiversionInformation.diversion_reason = 2, ++ .component.invoke.args.etsi.DiversionInformation.basic_service = 3, ++ .component.invoke.args.etsi.DiversionInformation.calling_present = 1, ++ .component.invoke.args.etsi.DiversionInformation.calling.presentation = 2, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_DiversionInformation, ++ .component.invoke.invoke_id = 80, ++ .component.invoke.args.etsi.DiversionInformation.diversion_reason = 3, ++ .component.invoke.args.etsi.DiversionInformation.basic_service = 5, ++ .component.invoke.args.etsi.DiversionInformation.calling_present = 1, ++ .component.invoke.args.etsi.DiversionInformation.calling.presentation = 3, ++ .component.invoke.args.etsi.DiversionInformation.calling.screened.screening_indicator = 2, ++ .component.invoke.args.etsi.DiversionInformation.calling.screened.number.plan = 4, ++ .component.invoke.args.etsi.DiversionInformation.calling.screened.number.length = 4, ++ .component.invoke.args.etsi.DiversionInformation.calling.screened.number.str = "1803", ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_DiversionInformation, ++ .component.invoke.invoke_id = 81, ++ .component.invoke.args.etsi.DiversionInformation.diversion_reason = 2, ++ .component.invoke.args.etsi.DiversionInformation.basic_service = 4, ++ .component.invoke.args.etsi.DiversionInformation.q931ie.length = 5, ++ .component.invoke.args.etsi.DiversionInformation.q931ie_contents = "79828", ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_DiversionInformation, ++ .component.invoke.invoke_id = 82, ++ .component.invoke.args.etsi.DiversionInformation.diversion_reason = 2, ++ .component.invoke.args.etsi.DiversionInformation.basic_service = 4, ++ }, ++ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_CallDeflection, ++ .component.invoke.invoke_id = 83, ++ .component.invoke.args.etsi.CallDeflection.deflection.number.plan = 4, ++ .component.invoke.args.etsi.CallDeflection.deflection.number.length = 4, ++ .component.invoke.args.etsi.CallDeflection.deflection.number.str = "1803", ++ .component.invoke.args.etsi.CallDeflection.presentation_allowed_to_diverted_to_user_present = 1, ++ .component.invoke.args.etsi.CallDeflection.presentation_allowed_to_diverted_to_user = 1, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_CallDeflection, ++ .component.invoke.invoke_id = 84, ++ .component.invoke.args.etsi.CallDeflection.deflection.number.plan = 4, ++ .component.invoke.args.etsi.CallDeflection.deflection.number.length = 4, ++ .component.invoke.args.etsi.CallDeflection.deflection.number.str = "1803", ++ .component.invoke.args.etsi.CallDeflection.presentation_allowed_to_diverted_to_user_present = 1, ++ .component.invoke.args.etsi.CallDeflection.presentation_allowed_to_diverted_to_user = 0, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_CallDeflection, ++ .component.invoke.invoke_id = 85, ++ .component.invoke.args.etsi.CallDeflection.deflection.number.plan = 4, ++ .component.invoke.args.etsi.CallDeflection.deflection.number.length = 4, ++ .component.invoke.args.etsi.CallDeflection.deflection.number.str = "1803", ++ }, ++ { ++ .type = ROSE_COMP_TYPE_RESULT, ++ .component.result.operation = ROSE_ETSI_CallDeflection, ++ .component.result.invoke_id = 86, ++ }, ++ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_CallRerouting, ++ .component.invoke.invoke_id = 87, ++ .component.invoke.args.etsi.CallRerouting.rerouting_reason = 3, ++ .component.invoke.args.etsi.CallRerouting.rerouting_counter = 2, ++ .component.invoke.args.etsi.CallRerouting.called_address.number.plan = 4, ++ .component.invoke.args.etsi.CallRerouting.called_address.number.length = 4, ++ .component.invoke.args.etsi.CallRerouting.called_address.number.str = "1803", ++ .component.invoke.args.etsi.CallRerouting.q931ie.length = 129, ++ .component.invoke.args.etsi.CallRerouting.q931ie_contents = ++ "YEHAW." ++ " The quick brown fox jumped over the lazy dog test." ++ " Now is the time for all good men to come to the aid of their country.", ++ .component.invoke.args.etsi.CallRerouting.last_rerouting.presentation = 1, ++ .component.invoke.args.etsi.CallRerouting.subscription_option = 2, ++ .component.invoke.args.etsi.CallRerouting.calling_subaddress.type = 1, ++ .component.invoke.args.etsi.CallRerouting.calling_subaddress.length = 4, ++ .component.invoke.args.etsi.CallRerouting.calling_subaddress.u.nsap = "6492", ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_CallRerouting, ++ .component.invoke.invoke_id = 88, ++ .component.invoke.args.etsi.CallRerouting.rerouting_reason = 3, ++ .component.invoke.args.etsi.CallRerouting.rerouting_counter = 2, ++ .component.invoke.args.etsi.CallRerouting.called_address.number.plan = 4, ++ .component.invoke.args.etsi.CallRerouting.called_address.number.length = 4, ++ .component.invoke.args.etsi.CallRerouting.called_address.number.str = "1803", ++ .component.invoke.args.etsi.CallRerouting.q931ie.length = 2, ++ .component.invoke.args.etsi.CallRerouting.q931ie_contents = "RT", ++ .component.invoke.args.etsi.CallRerouting.last_rerouting.presentation = 1, ++ .component.invoke.args.etsi.CallRerouting.subscription_option = 2, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_CallRerouting, ++ .component.invoke.invoke_id = 89, ++ .component.invoke.args.etsi.CallRerouting.rerouting_reason = 3, ++ .component.invoke.args.etsi.CallRerouting.rerouting_counter = 2, ++ .component.invoke.args.etsi.CallRerouting.called_address.number.plan = 4, ++ .component.invoke.args.etsi.CallRerouting.called_address.number.length = 4, ++ .component.invoke.args.etsi.CallRerouting.called_address.number.str = "1803", ++ .component.invoke.args.etsi.CallRerouting.q931ie.length = 2, ++ .component.invoke.args.etsi.CallRerouting.q931ie_contents = "RT", ++ .component.invoke.args.etsi.CallRerouting.last_rerouting.presentation = 2, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_RESULT, ++ .component.result.operation = ROSE_ETSI_CallRerouting, ++ .component.result.invoke_id = 90, ++ }, ++ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_InterrogateServedUserNumbers, ++ .component.invoke.invoke_id = 91, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_RESULT, ++ .component.result.operation = ROSE_ETSI_InterrogateServedUserNumbers, ++ .component.result.invoke_id = 92, ++ .component.result.args.etsi.InterrogateServedUserNumbers.num_records = 2, ++ .component.result.args.etsi.InterrogateServedUserNumbers.number[0].plan = 4, ++ .component.result.args.etsi.InterrogateServedUserNumbers.number[0].length = 4, ++ .component.result.args.etsi.InterrogateServedUserNumbers.number[0].str = "1803", ++ .component.result.args.etsi.InterrogateServedUserNumbers.number[1].plan = 4, ++ .component.result.args.etsi.InterrogateServedUserNumbers.number[1].length = 4, ++ .component.result.args.etsi.InterrogateServedUserNumbers.number[1].str = "5786", ++ }, ++ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_DivertingLegInformation1, ++ .component.invoke.invoke_id = 93, ++ .component.invoke.args.etsi.DivertingLegInformation1.diversion_reason = 4, ++ .component.invoke.args.etsi.DivertingLegInformation1.subscription_option = 1, ++ .component.invoke.args.etsi.DivertingLegInformation1.diverted_to_present = 1, ++ .component.invoke.args.etsi.DivertingLegInformation1.diverted_to.presentation = 2, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_DivertingLegInformation1, ++ .component.invoke.invoke_id = 94, ++ .component.invoke.args.etsi.DivertingLegInformation1.diversion_reason = 4, ++ .component.invoke.args.etsi.DivertingLegInformation1.subscription_option = 1, ++ }, ++ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_DivertingLegInformation2, ++ .component.invoke.invoke_id = 95, ++ .component.invoke.args.etsi.DivertingLegInformation2.diversion_counter = 3, ++ .component.invoke.args.etsi.DivertingLegInformation2.diversion_reason = 2, ++ .component.invoke.args.etsi.DivertingLegInformation2.diverting_present = 1, ++ .component.invoke.args.etsi.DivertingLegInformation2.diverting.presentation = 2, ++ .component.invoke.args.etsi.DivertingLegInformation2.original_called_present = 1, ++ .component.invoke.args.etsi.DivertingLegInformation2.original_called.presentation = 1, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_DivertingLegInformation2, ++ .component.invoke.invoke_id = 96, ++ .component.invoke.args.etsi.DivertingLegInformation2.diversion_counter = 3, ++ .component.invoke.args.etsi.DivertingLegInformation2.diversion_reason = 2, ++ .component.invoke.args.etsi.DivertingLegInformation2.original_called_present = 1, ++ .component.invoke.args.etsi.DivertingLegInformation2.original_called.presentation = 1, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_DivertingLegInformation2, ++ .component.invoke.invoke_id = 97, ++ .component.invoke.args.etsi.DivertingLegInformation2.diversion_counter = 1, ++ .component.invoke.args.etsi.DivertingLegInformation2.diversion_reason = 2, ++ }, ++ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_DivertingLegInformation3, ++ .component.invoke.invoke_id = 98, ++ .component.invoke.args.etsi.DivertingLegInformation3.presentation_allowed_indicator = 1, ++ }, ++ ++ /* Explicit Call Transfer (ECT) */ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_EctExecute, ++ .component.invoke.invoke_id = 54, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_ExplicitEctExecute, ++ .component.invoke.invoke_id = 55, ++ .component.invoke.args.etsi.ExplicitEctExecute.link_id = 23, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_RequestSubaddress, ++ .component.invoke.invoke_id = 56, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_SubaddressTransfer, ++ .component.invoke.invoke_id = 57, ++ .component.invoke.args.etsi.SubaddressTransfer.subaddress.type = 1, ++ .component.invoke.args.etsi.SubaddressTransfer.subaddress.length = 4, ++ .component.invoke.args.etsi.SubaddressTransfer.subaddress.u.nsap = "6492", ++ }, ++ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_EctLinkIdRequest, ++ .component.invoke.invoke_id = 58, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_RESULT, ++ .component.result.operation = ROSE_ETSI_EctLinkIdRequest, ++ .component.result.invoke_id = 59, ++ .component.result.args.etsi.EctLinkIdRequest.link_id = 76, ++ }, ++ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_EctInform, ++ .component.invoke.invoke_id = 60, ++ .component.invoke.args.etsi.EctInform.status = 1, ++ .component.invoke.args.etsi.EctInform.redirection_present = 1, ++ .component.invoke.args.etsi.EctInform.redirection.presentation = 0, ++ .component.invoke.args.etsi.EctInform.redirection.number.plan = 8, ++ .component.invoke.args.etsi.EctInform.redirection.number.length = 4, ++ .component.invoke.args.etsi.EctInform.redirection.number.str = "6229", ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_EctInform, ++ .component.invoke.invoke_id = 61, ++ .component.invoke.args.etsi.EctInform.status = 1, ++ .component.invoke.args.etsi.EctInform.redirection_present = 1, ++ .component.invoke.args.etsi.EctInform.redirection.presentation = 1, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_EctInform, ++ .component.invoke.invoke_id = 62, ++ .component.invoke.args.etsi.EctInform.status = 1, ++ .component.invoke.args.etsi.EctInform.redirection_present = 1, ++ .component.invoke.args.etsi.EctInform.redirection.presentation = 2, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_EctInform, ++ .component.invoke.invoke_id = 63, ++ .component.invoke.args.etsi.EctInform.status = 1, ++ .component.invoke.args.etsi.EctInform.redirection_present = 1, ++ .component.invoke.args.etsi.EctInform.redirection.presentation = 3, ++ .component.invoke.args.etsi.EctInform.redirection.number.plan = 8, ++ .component.invoke.args.etsi.EctInform.redirection.number.length = 4, ++ .component.invoke.args.etsi.EctInform.redirection.number.str = "3340", ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_EctInform, ++ .component.invoke.invoke_id = 64, ++ .component.invoke.args.etsi.EctInform.status = 1, ++ .component.invoke.args.etsi.EctInform.redirection_present = 0, ++ }, ++ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_ETSI_EctLoopTest, ++ .component.invoke.invoke_id = 65, ++ .component.invoke.args.etsi.EctLoopTest.call_transfer_id = 7, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_RESULT, ++ .component.result.operation = ROSE_ETSI_EctLoopTest, ++ .component.result.invoke_id = 66, ++ .component.result.args.etsi.EctLoopTest.loop_result = 2, ++ }, ++/* *INDENT-ON* */ ++}; ++ ++static unsigned char rose_etsi_indefinite_len[] = { ++/* *INDENT-OFF* */ ++/* ++ * Context Specific/C [1 0x01] <A1> Len:24 <80> ++ * Integer(2 0x02) <02> Len:1 <01> ++ * <44> ++ * Integer(2 0x02) <02> Len:1 <01> ++ * <07> ++ * Sequence/C(48 0x30) <30> Len:16 <80> ++ * Enumerated(10 0x0A) <0A> Len:1 <01> ++ * <01> ++ * Enumerated(10 0x0A) <0A> Len:1 <01> ++ * <05> ++ * Sequence/C(48 0x30) <30> Len:6 <80> ++ * Context Specific [4 0x04] <84> Len:4 <80> ++ * <31 38 30 33> ++ * 0x00, 0x00, ++ * 0x00, 0x00, ++ * NULL(5 0x05) <05> Len:0 <00> ++ * 0x00, 0x00, ++ * 0x00, 0x00 ++ */ ++ 0x91, ++ 0xA1, 0x80, ++ 0x02, 0x01, ++ 0x44, ++ 0x02, 0x01, ++ 0x07, ++ 0x30, 0x80, ++ 0x0A, 0x01, ++ 0x01, ++ 0x0A, 0x01, ++ 0x05, ++ 0x30, 0x80, ++ 0x84, 0x80, ++ 0x31, 0x38, 0x30, 0x33, ++ 0x00, 0x00, ++ 0x00, 0x00, ++ 0x05, 0x00, ++ 0x00, 0x00, ++ 0x00, 0x00, ++ 0x00, 0x00 ++/* *INDENT-ON* */ ++}; ++ ++static unsigned char rose_etsi_unused_indefinite_len[] = { ++/* *INDENT-OFF* */ ++/* ++ * Context Specific/C [1 0x01] <A1> Len:24 <80> ++ * Integer(2 0x02) <02> Len:1 <01> ++ * <44> ++ * Integer(2 0x02) <02> Len:1 <01> ++ * <06> -- EctExecute ++ * Sequence/C(48 0x30) <30> Len:16 <80> ++ * Enumerated(10 0x0A) <0A> Len:1 <01> ++ * <01> ++ * Enumerated(10 0x0A) <0A> Len:1 <01> ++ * <05> ++ * Sequence/C(48 0x30) <30> Len:6 <80> ++ * Context Specific [4 0x04] <84> Len:4 <80> ++ * <31 38 30 33> ++ * 0x00, 0x00, ++ * 0x00, 0x00, ++ * NULL(5 0x05) <05> Len:0 <00> ++ * 0x00, 0x00, ++ * 0x00, 0x00 ++ */ ++ 0x91, ++ 0xA1, 0x80, ++ 0x02, 0x01, ++ 0x44, ++ 0x02, 0x01, ++ 0x06, ++ 0x30, 0x80, ++ 0x0A, 0x01, ++ 0x01, ++ 0x0A, 0x01, ++ 0x05, ++ 0x30, 0x80, ++ 0x84, 0x80, ++ 0x31, 0x38, 0x30, 0x33, ++ 0x00, 0x00, ++ 0x00, 0x00, ++ 0x05, 0x00, ++ 0x00, 0x00, ++ 0x00, 0x00, ++ 0x00, 0x00 ++/* *INDENT-ON* */ ++}; ++ ++static unsigned char rose_etsi_unused[] = { ++/* *INDENT-OFF* */ ++/* ++ * Context Specific/C [1 0x01] <A1> Len:24 <18> ++ * Integer(2 0x02) <02> Len:1 <01> ++ * <44> ++ * Integer(2 0x02) <02> Len:1 <01> ++ * <06> -- EctExecute ++ * Sequence/C(48 0x30) <30> Len:16 <10> ++ * Enumerated(10 0x0A) <0A> Len:1 <01> ++ * <01> ++ * Enumerated(10 0x0A) <0A> Len:1 <01> ++ * <05> ++ * Sequence/C(48 0x30) <30> Len:6 <06> ++ * Context Specific [4 0x04] <84> Len:4 <04> ++ * <31 38 30 33> ++ * NULL(5 0x05) <05> Len:0 <00> ++ */ ++ 0x91, ++ 0xA1, 0x18, ++ 0x02, 0x01, ++ 0x44, ++ 0x02, 0x01, ++ 0x06, ++ 0x30, 0x10, ++ 0x0A, 0x01, ++ 0x01, ++ 0x0A, 0x01, ++ 0x05, ++ 0x30, 0x06, ++ 0x84, 0x04, ++ 0x31, 0x38, 0x30, 0x33, ++ 0x05, 0x00, ++ 0x00, 0x00 ++/* *INDENT-ON* */ ++}; ++ ++static unsigned char rose_etsi_extra[] = { ++/* *INDENT-OFF* */ ++/* ++ * Context Specific/C [1 0x01] <A1> Len:24 <18> ++ * Integer(2 0x02) <02> Len:1 <01> ++ * <44> ++ * Integer(2 0x02) <02> Len:1 <01> ++ * <07> ++ * Sequence/C(48 0x30) <30> Len:16 <10> ++ * Enumerated(10 0x0A) <0A> Len:1 <01> ++ * <01> ++ * Enumerated(10 0x0A) <0A> Len:1 <01> ++ * <05> ++ * Sequence/C(48 0x30) <30> Len:6 <06> ++ * Context Specific [4 0x04] <84> Len:4 <04> ++ * <31 38 30 33> ++ * NULL(5 0x05) <05> Len:0 <00> ++ */ ++ 0x91, ++ 0xA1, 0x18, ++ 0x02, 0x01, ++ 0x44, ++ 0x02, 0x01, ++ 0x07, ++ 0x30, 0x10, ++ 0x0A, 0x01, ++ 0x01, ++ 0x0A, 0x01, ++ 0x05, ++ 0x30, 0x06, ++ 0x84, 0x04, ++ 0x31, 0x38, 0x30, 0x33, ++ 0x05, 0x00, ++ 0x00, 0x00 ++/* *INDENT-ON* */ ++}; ++ ++ ++static const struct rose_message rose_qsig_msgs[] = { ++/* *INDENT-OFF* */ ++ /* Q.SIG Name-Operations */ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_CallingName, ++ .component.invoke.invoke_id = 2, ++ .component.invoke.args.qsig.CallingName.name.presentation = 1, ++ .component.invoke.args.qsig.CallingName.name.char_set = 1, ++ .component.invoke.args.qsig.CallingName.name.length = 7, ++ .component.invoke.args.qsig.CallingName.name.data = "Alphred", ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_CallingName, ++ .component.invoke.invoke_id = 3, ++ .component.invoke.args.qsig.CallingName.name.presentation = 1, ++ .component.invoke.args.qsig.CallingName.name.char_set = 3, ++ .component.invoke.args.qsig.CallingName.name.length = 7, ++ .component.invoke.args.qsig.CallingName.name.data = "Alphred", ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_CallingName, ++ .component.invoke.invoke_id = 4, ++ .component.invoke.args.qsig.CallingName.name.presentation = 2, ++ .component.invoke.args.qsig.CallingName.name.char_set = 1, ++ .component.invoke.args.qsig.CallingName.name.length = 7, ++ .component.invoke.args.qsig.CallingName.name.data = "Alphred", ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_CallingName, ++ .component.invoke.invoke_id = 5, ++ .component.invoke.args.qsig.CallingName.name.presentation = 2, ++ .component.invoke.args.qsig.CallingName.name.char_set = 3, ++ .component.invoke.args.qsig.CallingName.name.length = 7, ++ .component.invoke.args.qsig.CallingName.name.data = "Alphred", ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_CallingName, ++ .component.invoke.invoke_id = 6, ++ .component.invoke.args.qsig.CallingName.name.presentation = 3, ++ .component.invoke.args.qsig.CallingName.name.char_set = 1, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_CallingName, ++ .component.invoke.invoke_id = 7, ++ .component.invoke.args.qsig.CallingName.name.presentation = 4, ++ .component.invoke.args.qsig.CallingName.name.char_set = 1, ++ }, ++ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_CalledName, ++ .component.invoke.invoke_id = 8, ++ .component.invoke.args.qsig.CallingName.name.presentation = 4, ++ .component.invoke.args.qsig.CallingName.name.char_set = 1, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_ConnectedName, ++ .component.invoke.invoke_id = 9, ++ .component.invoke.args.qsig.CallingName.name.presentation = 4, ++ .component.invoke.args.qsig.CallingName.name.char_set = 1, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_BusyName, ++ .component.invoke.invoke_id = 10, ++ .component.invoke.args.qsig.CallingName.name.presentation = 4, ++ .component.invoke.args.qsig.CallingName.name.char_set = 1, ++ }, ++ ++ /* Q.SIG SS-AOC-Operations */ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_ChargeRequest, ++ .component.invoke.invoke_id = 11, ++ .component.invoke.args.qsig.ChargeRequest.num_records = 0, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_ChargeRequest, ++ .component.invoke.invoke_id = 12, ++ .component.invoke.args.qsig.ChargeRequest.num_records = 1, ++ .component.invoke.args.qsig.ChargeRequest.advice_mode_combinations[0] = 3, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_ChargeRequest, ++ .component.invoke.invoke_id = 13, ++ .component.invoke.args.qsig.ChargeRequest.num_records = 2, ++ .component.invoke.args.qsig.ChargeRequest.advice_mode_combinations[0] = 4, ++ .component.invoke.args.qsig.ChargeRequest.advice_mode_combinations[1] = 3, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_RESULT, ++ .component.result.operation = ROSE_QSIG_ChargeRequest, ++ .component.result.invoke_id = 14, ++ .component.result.args.qsig.ChargeRequest.advice_mode_combination = 3, ++ }, ++ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_GetFinalCharge, ++ .component.invoke.invoke_id = 15, ++ }, ++ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_AocFinal, ++ .component.invoke.invoke_id = 16, ++ .component.invoke.args.qsig.AocFinal.type = 0, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_AocFinal, ++ .component.invoke.invoke_id = 17, ++ .component.invoke.args.qsig.AocFinal.type = 1, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_AocFinal, ++ .component.invoke.invoke_id = 18, ++ .component.invoke.args.qsig.AocFinal.type = 2, ++ .component.invoke.args.qsig.AocFinal.specific.recorded.amount.currency = 800, ++ .component.invoke.args.qsig.AocFinal.specific.recorded.amount.multiplier = 2, ++ .component.invoke.args.qsig.AocFinal.specific.recorded.currency = "Rupies", ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_AocFinal, ++ .component.invoke.invoke_id = 19, ++ .component.invoke.args.qsig.AocFinal.type = 2, ++ .component.invoke.args.qsig.AocFinal.specific.recorded.amount.currency = 800, ++ .component.invoke.args.qsig.AocFinal.specific.recorded.amount.multiplier = 2, ++ .component.invoke.args.qsig.AocFinal.specific.recorded.currency = "Rupies", ++ .component.invoke.args.qsig.AocFinal.specific.billing_id_present = 1, ++ .component.invoke.args.qsig.AocFinal.specific.billing_id = 2, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_AocFinal, ++ .component.invoke.invoke_id = 20, ++ .component.invoke.args.qsig.AocFinal.type = 2, ++ .component.invoke.args.qsig.AocFinal.specific.recorded.amount.currency = 800, ++ .component.invoke.args.qsig.AocFinal.specific.recorded.amount.multiplier = 2, ++ .component.invoke.args.qsig.AocFinal.specific.recorded.currency = "Rupies", ++ .component.invoke.args.qsig.AocFinal.charging_association_present = 1, ++ .component.invoke.args.qsig.AocFinal.charging_association.type = 0, ++ .component.invoke.args.qsig.AocFinal.charging_association.id = 200, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_AocFinal, ++ .component.invoke.invoke_id = 21, ++ .component.invoke.args.qsig.AocFinal.type = 2, ++ .component.invoke.args.qsig.AocFinal.specific.recorded.amount.currency = 800, ++ .component.invoke.args.qsig.AocFinal.specific.recorded.amount.multiplier = 2, ++ .component.invoke.args.qsig.AocFinal.specific.recorded.currency = "Rupies", ++ .component.invoke.args.qsig.AocFinal.specific.billing_id_present = 1, ++ .component.invoke.args.qsig.AocFinal.specific.billing_id = 2, ++ .component.invoke.args.qsig.AocFinal.charging_association_present = 1, ++ .component.invoke.args.qsig.AocFinal.charging_association.type = 0, ++ .component.invoke.args.qsig.AocFinal.charging_association.id = 200, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_AocFinal, ++ .component.invoke.invoke_id = 22, ++ .component.invoke.args.qsig.AocFinal.type = 2, ++ .component.invoke.args.qsig.AocFinal.specific.recorded.amount.currency = 800, ++ .component.invoke.args.qsig.AocFinal.specific.recorded.amount.multiplier = 2, ++ .component.invoke.args.qsig.AocFinal.specific.recorded.currency = "Rupies", ++ .component.invoke.args.qsig.AocFinal.charging_association_present = 1, ++ .component.invoke.args.qsig.AocFinal.charging_association.type = 1, ++ .component.invoke.args.qsig.AocFinal.charging_association.number.plan = 4, ++ .component.invoke.args.qsig.AocFinal.charging_association.number.length = 4, ++ .component.invoke.args.qsig.AocFinal.charging_association.number.str = "1802", ++ }, ++ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_AocInterim, ++ .component.invoke.invoke_id = 23, ++ .component.invoke.args.qsig.AocInterim.type = 0, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_AocInterim, ++ .component.invoke.invoke_id = 24, ++ .component.invoke.args.qsig.AocInterim.type = 1, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_AocInterim, ++ .component.invoke.invoke_id = 25, ++ .component.invoke.args.qsig.AocInterim.type = 2, ++ .component.invoke.args.qsig.AocInterim.specific.recorded.amount.currency = 800, ++ .component.invoke.args.qsig.AocInterim.specific.recorded.amount.multiplier = 2, ++ .component.invoke.args.qsig.AocInterim.specific.recorded.currency = "Rupies", ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_AocInterim, ++ .component.invoke.invoke_id = 26, ++ .component.invoke.args.qsig.AocInterim.type = 2, ++ .component.invoke.args.qsig.AocInterim.specific.recorded.amount.currency = 800, ++ .component.invoke.args.qsig.AocInterim.specific.recorded.amount.multiplier = 2, ++ .component.invoke.args.qsig.AocInterim.specific.recorded.currency = "Rupies", ++ .component.invoke.args.qsig.AocInterim.specific.billing_id_present = 1, ++ .component.invoke.args.qsig.AocInterim.specific.billing_id = 2, ++ }, ++ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_AocRate, ++ .component.invoke.invoke_id = 27, ++ .component.invoke.args.qsig.AocRate.type = 0, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_AocRate, ++ .component.invoke.invoke_id = 28, ++ .component.invoke.args.qsig.AocRate.type = 1, ++ .component.invoke.args.qsig.AocRate.currency_info.num_records = 1, ++ .component.invoke.args.qsig.AocRate.currency_info.list[0].charged_item = 4, ++ .component.invoke.args.qsig.AocRate.currency_info.list[0].currency_type = 0, ++ .component.invoke.args.qsig.AocRate.currency_info.list[0].u.special_charging_code = 3, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_AocRate, ++ .component.invoke.invoke_id = 29, ++ .component.invoke.args.qsig.AocRate.type = 1, ++ .component.invoke.args.qsig.AocRate.currency_info.num_records = 1, ++ .component.invoke.args.qsig.AocRate.currency_info.list[0].charged_item = 4, ++ .component.invoke.args.qsig.AocRate.currency_info.list[0].currency_type = 1, ++ .component.invoke.args.qsig.AocRate.currency_info.list[0].u.duration.currency = "Dollars", ++ .component.invoke.args.qsig.AocRate.currency_info.list[0].u.duration.amount.currency = 7, ++ .component.invoke.args.qsig.AocRate.currency_info.list[0].u.duration.amount.multiplier = 1, ++ .component.invoke.args.qsig.AocRate.currency_info.list[0].u.duration.charging_type = 1, ++ .component.invoke.args.qsig.AocRate.currency_info.list[0].u.duration.time.length = 8, ++ .component.invoke.args.qsig.AocRate.currency_info.list[0].u.duration.time.scale = 4, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_AocRate, ++ .component.invoke.invoke_id = 30, ++ .component.invoke.args.qsig.AocRate.type = 1, ++ .component.invoke.args.qsig.AocRate.currency_info.num_records = 1, ++ .component.invoke.args.qsig.AocRate.currency_info.list[0].charged_item = 4, ++ .component.invoke.args.qsig.AocRate.currency_info.list[0].currency_type = 1, ++ .component.invoke.args.qsig.AocRate.currency_info.list[0].u.duration.currency = "Dollars", ++ .component.invoke.args.qsig.AocRate.currency_info.list[0].u.duration.amount.currency = 7, ++ .component.invoke.args.qsig.AocRate.currency_info.list[0].u.duration.amount.multiplier = 1, ++ .component.invoke.args.qsig.AocRate.currency_info.list[0].u.duration.charging_type = 1, ++ .component.invoke.args.qsig.AocRate.currency_info.list[0].u.duration.time.length = 8, ++ .component.invoke.args.qsig.AocRate.currency_info.list[0].u.duration.time.scale = 4, ++ .component.invoke.args.qsig.AocRate.currency_info.list[0].u.duration.granularity_present = 1, ++ .component.invoke.args.qsig.AocRate.currency_info.list[0].u.duration.granularity.length = 20, ++ .component.invoke.args.qsig.AocRate.currency_info.list[0].u.duration.granularity.scale = 3, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_AocRate, ++ .component.invoke.invoke_id = 31, ++ .component.invoke.args.qsig.AocRate.type = 1, ++ .component.invoke.args.qsig.AocRate.currency_info.num_records = 1, ++ .component.invoke.args.qsig.AocRate.currency_info.list[0].charged_item = 4, ++ .component.invoke.args.qsig.AocRate.currency_info.list[0].currency_type = 2, ++ .component.invoke.args.qsig.AocRate.currency_info.list[0].u.flat_rate.currency = "Euros", ++ .component.invoke.args.qsig.AocRate.currency_info.list[0].u.flat_rate.amount.currency = 4, ++ .component.invoke.args.qsig.AocRate.currency_info.list[0].u.flat_rate.amount.multiplier = 1, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_AocRate, ++ .component.invoke.invoke_id = 32, ++ .component.invoke.args.qsig.AocRate.type = 1, ++ .component.invoke.args.qsig.AocRate.currency_info.num_records = 1, ++ .component.invoke.args.qsig.AocRate.currency_info.list[0].charged_item = 4, ++ .component.invoke.args.qsig.AocRate.currency_info.list[0].currency_type = 3, ++ .component.invoke.args.qsig.AocRate.currency_info.list[0].u.volume_rate.currency = "Yen", ++ .component.invoke.args.qsig.AocRate.currency_info.list[0].u.volume_rate.amount.currency = 300, ++ .component.invoke.args.qsig.AocRate.currency_info.list[0].u.volume_rate.amount.multiplier = 5, ++ .component.invoke.args.qsig.AocRate.currency_info.list[0].u.volume_rate.unit = 2, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_AocRate, ++ .component.invoke.invoke_id = 33, ++ .component.invoke.args.qsig.AocRate.type = 1, ++ .component.invoke.args.qsig.AocRate.currency_info.num_records = 2, ++ .component.invoke.args.qsig.AocRate.currency_info.list[0].charged_item = 4, ++ .component.invoke.args.qsig.AocRate.currency_info.list[0].currency_type = 2, ++ .component.invoke.args.qsig.AocRate.currency_info.list[0].u.flat_rate.currency = "Euros", ++ .component.invoke.args.qsig.AocRate.currency_info.list[0].u.flat_rate.amount.currency = 4, ++ .component.invoke.args.qsig.AocRate.currency_info.list[0].u.flat_rate.amount.multiplier = 1, ++ .component.invoke.args.qsig.AocRate.currency_info.list[1].charged_item = 4, ++ .component.invoke.args.qsig.AocRate.currency_info.list[1].currency_type = 3, ++ .component.invoke.args.qsig.AocRate.currency_info.list[1].u.volume_rate.currency = "Yen", ++ .component.invoke.args.qsig.AocRate.currency_info.list[1].u.volume_rate.amount.currency = 300, ++ .component.invoke.args.qsig.AocRate.currency_info.list[1].u.volume_rate.amount.multiplier = 5, ++ .component.invoke.args.qsig.AocRate.currency_info.list[1].u.volume_rate.unit = 2, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_AocRate, ++ .component.invoke.invoke_id = 34, ++ .component.invoke.args.qsig.AocRate.type = 1, ++ .component.invoke.args.qsig.AocRate.currency_info.num_records = 1, ++ .component.invoke.args.qsig.AocRate.currency_info.list[0].charged_item = 4, ++ .component.invoke.args.qsig.AocRate.currency_info.list[0].currency_type = 4, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_AocRate, ++ .component.invoke.invoke_id = 35, ++ .component.invoke.args.qsig.AocRate.type = 1, ++ .component.invoke.args.qsig.AocRate.currency_info.num_records = 1, ++ .component.invoke.args.qsig.AocRate.currency_info.list[0].charged_item = 4, ++ .component.invoke.args.qsig.AocRate.currency_info.list[0].currency_type = 5, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_AocRate, ++ .component.invoke.invoke_id = 36, ++ .component.invoke.args.qsig.AocRate.type = 1, ++ .component.invoke.args.qsig.AocRate.currency_info.num_records = 1, ++ .component.invoke.args.qsig.AocRate.currency_info.list[0].charged_item = 4, ++ .component.invoke.args.qsig.AocRate.currency_info.list[0].currency_type = 6, ++ }, ++ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_AocComplete, ++ .component.invoke.invoke_id = 37, ++ .component.invoke.args.qsig.AocComplete.charged_user_number.plan = 4, ++ .component.invoke.args.qsig.AocComplete.charged_user_number.length = 4, ++ .component.invoke.args.qsig.AocComplete.charged_user_number.str = "8340", ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_AocComplete, ++ .component.invoke.invoke_id = 38, ++ .component.invoke.args.qsig.AocComplete.charged_user_number.plan = 4, ++ .component.invoke.args.qsig.AocComplete.charged_user_number.length = 4, ++ .component.invoke.args.qsig.AocComplete.charged_user_number.str = "8340", ++ .component.invoke.args.qsig.AocComplete.charging_association_present = 1, ++ .component.invoke.args.qsig.AocComplete.charging_association.type = 0, ++ .component.invoke.args.qsig.AocComplete.charging_association.id = 8298, ++ }, ++ ++ { ++ .type = ROSE_COMP_TYPE_RESULT, ++ .component.result.operation = ROSE_QSIG_AocComplete, ++ .component.result.invoke_id = 39, ++ .component.result.args.qsig.AocComplete.charging_option = 2, ++ }, ++ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_AocDivChargeReq, ++ .component.invoke.invoke_id = 40, ++ .component.invoke.args.qsig.AocDivChargeReq.diverting_user_number.plan = 4, ++ .component.invoke.args.qsig.AocDivChargeReq.diverting_user_number.length = 4, ++ .component.invoke.args.qsig.AocDivChargeReq.diverting_user_number.str = "8340", ++ .component.invoke.args.qsig.AocDivChargeReq.diversion_type = 3, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_AocDivChargeReq, ++ .component.invoke.invoke_id = 41, ++ .component.invoke.args.qsig.AocDivChargeReq.diverting_user_number.plan = 4, ++ .component.invoke.args.qsig.AocDivChargeReq.diverting_user_number.length = 4, ++ .component.invoke.args.qsig.AocDivChargeReq.diverting_user_number.str = "8340", ++ .component.invoke.args.qsig.AocDivChargeReq.charging_association_present = 1, ++ .component.invoke.args.qsig.AocDivChargeReq.charging_association.type = 0, ++ .component.invoke.args.qsig.AocDivChargeReq.charging_association.id = 8298, ++ .component.invoke.args.qsig.AocDivChargeReq.diversion_type = 3, ++ }, ++ ++ /* Q.SIG Call-Transfer-Operations (CT) */ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_CallTransferIdentify, ++ .component.invoke.invoke_id = 42, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_RESULT, ++ .component.result.operation = ROSE_QSIG_CallTransferIdentify, ++ .component.result.invoke_id = 43, ++ .component.result.args.qsig.CallTransferIdentify.call_id = "2345", ++ .component.result.args.qsig.CallTransferIdentify.rerouting_number.plan = 4, ++ .component.result.args.qsig.CallTransferIdentify.rerouting_number.length = 4, ++ .component.result.args.qsig.CallTransferIdentify.rerouting_number.str = "8340", ++ }, ++ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_CallTransferAbandon, ++ .component.invoke.invoke_id = 44, ++ }, ++ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_CallTransferInitiate, ++ .component.invoke.invoke_id = 45, ++ .component.invoke.args.qsig.CallTransferInitiate.call_id = "2345", ++ .component.invoke.args.qsig.CallTransferInitiate.rerouting_number.plan = 4, ++ .component.invoke.args.qsig.CallTransferInitiate.rerouting_number.length = 4, ++ .component.invoke.args.qsig.CallTransferInitiate.rerouting_number.str = "8340", ++ }, ++ { ++ .type = ROSE_COMP_TYPE_RESULT, ++ .component.result.operation = ROSE_QSIG_CallTransferInitiate, ++ .component.result.invoke_id = 46, ++ }, ++ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_CallTransferSetup, ++ .component.invoke.invoke_id = 47, ++ .component.invoke.args.qsig.CallTransferSetup.call_id = "23", ++ }, ++ { ++ .type = ROSE_COMP_TYPE_RESULT, ++ .component.result.operation = ROSE_QSIG_CallTransferSetup, ++ .component.result.invoke_id = 48, ++ }, ++ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_CallTransferActive, ++ .component.invoke.invoke_id = 49, ++ .component.invoke.args.qsig.CallTransferActive.connected.presentation = 1, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_CallTransferActive, ++ .component.invoke.invoke_id = 50, ++ .component.invoke.args.qsig.CallTransferActive.connected.presentation = 1, ++ .component.invoke.args.qsig.CallTransferActive.q931ie.length = 2, ++ .component.invoke.args.qsig.CallTransferActive.q931ie_contents = "RT", ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_CallTransferActive, ++ .component.invoke.invoke_id = 51, ++ .component.invoke.args.qsig.CallTransferActive.connected.presentation = 1, ++ .component.invoke.args.qsig.CallTransferActive.connected_name_present = 1, ++ .component.invoke.args.qsig.CallTransferActive.connected_name.presentation = 1, ++ .component.invoke.args.qsig.CallTransferActive.connected_name.char_set = 1, ++ .component.invoke.args.qsig.CallTransferActive.connected_name.length = 7, ++ .component.invoke.args.qsig.CallTransferActive.connected_name.data = "Alphred", ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_CallTransferActive, ++ .component.invoke.invoke_id = 52, ++ .component.invoke.args.qsig.CallTransferActive.connected.presentation = 1, ++ .component.invoke.args.qsig.CallTransferActive.q931ie.length = 2, ++ .component.invoke.args.qsig.CallTransferActive.q931ie_contents = "RT", ++ .component.invoke.args.qsig.CallTransferActive.connected_name_present = 1, ++ .component.invoke.args.qsig.CallTransferActive.connected_name.presentation = 1, ++ .component.invoke.args.qsig.CallTransferActive.connected_name.char_set = 1, ++ .component.invoke.args.qsig.CallTransferActive.connected_name.length = 7, ++ .component.invoke.args.qsig.CallTransferActive.connected_name.data = "Alphred", ++ }, ++ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_CallTransferComplete, ++ .component.invoke.invoke_id = 53, ++ .component.invoke.args.qsig.CallTransferComplete.end_designation = 1, ++ .component.invoke.args.qsig.CallTransferComplete.redirection.presentation = 0, ++ .component.invoke.args.qsig.CallTransferComplete.redirection.screened.screening_indicator = 3, ++ .component.invoke.args.qsig.CallTransferComplete.redirection.screened.number.plan = 4, ++ .component.invoke.args.qsig.CallTransferComplete.redirection.screened.number.length = 4, ++ .component.invoke.args.qsig.CallTransferComplete.redirection.screened.number.str = "8340", ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_CallTransferComplete, ++ .component.invoke.invoke_id = 54, ++ .component.invoke.args.qsig.CallTransferComplete.end_designation = 1, ++ .component.invoke.args.qsig.CallTransferComplete.redirection.presentation = 1, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_CallTransferComplete, ++ .component.invoke.invoke_id = 55, ++ .component.invoke.args.qsig.CallTransferComplete.end_designation = 1, ++ .component.invoke.args.qsig.CallTransferComplete.redirection.presentation = 2, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_CallTransferComplete, ++ .component.invoke.invoke_id = 56, ++ .component.invoke.args.qsig.CallTransferComplete.end_designation = 1, ++ .component.invoke.args.qsig.CallTransferComplete.redirection.presentation = 3, ++ .component.invoke.args.qsig.CallTransferComplete.redirection.screened.screening_indicator = 3, ++ .component.invoke.args.qsig.CallTransferComplete.redirection.screened.number.plan = 4, ++ .component.invoke.args.qsig.CallTransferComplete.redirection.screened.number.length = 4, ++ .component.invoke.args.qsig.CallTransferComplete.redirection.screened.number.str = "8340", ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_CallTransferComplete, ++ .component.invoke.invoke_id = 57, ++ .component.invoke.args.qsig.CallTransferComplete.end_designation = 1, ++ .component.invoke.args.qsig.CallTransferComplete.redirection.presentation = 2, ++ .component.invoke.args.qsig.CallTransferComplete.q931ie.length = 2, ++ .component.invoke.args.qsig.CallTransferComplete.q931ie_contents = "RT", ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_CallTransferComplete, ++ .component.invoke.invoke_id = 58, ++ .component.invoke.args.qsig.CallTransferComplete.end_designation = 1, ++ .component.invoke.args.qsig.CallTransferComplete.redirection.presentation = 2, ++ .component.invoke.args.qsig.CallTransferComplete.redirection_name_present = 1, ++ .component.invoke.args.qsig.CallTransferComplete.redirection_name.presentation = 1, ++ .component.invoke.args.qsig.CallTransferComplete.redirection_name.char_set = 1, ++ .component.invoke.args.qsig.CallTransferComplete.redirection_name.length = 7, ++ .component.invoke.args.qsig.CallTransferComplete.redirection_name.data = "Alphred", ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_CallTransferComplete, ++ .component.invoke.invoke_id = 59, ++ .component.invoke.args.qsig.CallTransferComplete.end_designation = 1, ++ .component.invoke.args.qsig.CallTransferComplete.redirection.presentation = 2, ++ .component.invoke.args.qsig.CallTransferComplete.call_status = 1, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_CallTransferComplete, ++ .component.invoke.invoke_id = 60, ++ .component.invoke.args.qsig.CallTransferComplete.end_designation = 1, ++ .component.invoke.args.qsig.CallTransferComplete.redirection.presentation = 2, ++ .component.invoke.args.qsig.CallTransferComplete.q931ie.length = 2, ++ .component.invoke.args.qsig.CallTransferComplete.q931ie_contents = "RT", ++ .component.invoke.args.qsig.CallTransferComplete.call_status = 1, ++ }, ++ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_CallTransferUpdate, ++ .component.invoke.invoke_id = 61, ++ .component.invoke.args.qsig.CallTransferUpdate.redirection.presentation = 2, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_CallTransferUpdate, ++ .component.invoke.invoke_id = 62, ++ .component.invoke.args.qsig.CallTransferUpdate.redirection.presentation = 2, ++ .component.invoke.args.qsig.CallTransferUpdate.redirection_name_present = 1, ++ .component.invoke.args.qsig.CallTransferUpdate.redirection_name.presentation = 1, ++ .component.invoke.args.qsig.CallTransferUpdate.redirection_name.char_set = 1, ++ .component.invoke.args.qsig.CallTransferUpdate.redirection_name.length = 7, ++ .component.invoke.args.qsig.CallTransferUpdate.redirection_name.data = "Alphred", ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_CallTransferUpdate, ++ .component.invoke.invoke_id = 63, ++ .component.invoke.args.qsig.CallTransferUpdate.redirection.presentation = 2, ++ .component.invoke.args.qsig.CallTransferUpdate.q931ie.length = 2, ++ .component.invoke.args.qsig.CallTransferUpdate.q931ie_contents = "RT", ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_CallTransferUpdate, ++ .component.invoke.invoke_id = 64, ++ .component.invoke.args.qsig.CallTransferUpdate.redirection.presentation = 2, ++ .component.invoke.args.qsig.CallTransferUpdate.redirection_name_present = 1, ++ .component.invoke.args.qsig.CallTransferUpdate.redirection_name.presentation = 1, ++ .component.invoke.args.qsig.CallTransferUpdate.redirection_name.char_set = 1, ++ .component.invoke.args.qsig.CallTransferUpdate.redirection_name.length = 7, ++ .component.invoke.args.qsig.CallTransferUpdate.redirection_name.data = "Alphred", ++ .component.invoke.args.qsig.CallTransferUpdate.q931ie.length = 2, ++ .component.invoke.args.qsig.CallTransferUpdate.q931ie_contents = "RT", ++ }, ++ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_SubaddressTransfer, ++ .component.invoke.invoke_id = 65, ++ .component.invoke.args.qsig.SubaddressTransfer.redirection_subaddress.type = 1, ++ .component.invoke.args.qsig.SubaddressTransfer.redirection_subaddress.length = 4, ++ .component.invoke.args.qsig.SubaddressTransfer.redirection_subaddress.u.nsap = "4356", ++ }, ++ ++ /* Q.SIG Call-Diversion-Operations */ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_ActivateDiversionQ, ++ .component.invoke.invoke_id = 66, ++ .component.invoke.args.qsig.ActivateDiversionQ.procedure = 1, ++ .component.invoke.args.qsig.ActivateDiversionQ.basic_service = 3, ++ .component.invoke.args.qsig.ActivateDiversionQ.diverted_to.number.plan = 4, ++ .component.invoke.args.qsig.ActivateDiversionQ.diverted_to.number.length = 4, ++ .component.invoke.args.qsig.ActivateDiversionQ.diverted_to.number.str = "8340", ++ .component.invoke.args.qsig.ActivateDiversionQ.served_user_number.plan = 4, ++ .component.invoke.args.qsig.ActivateDiversionQ.served_user_number.length = 4, ++ .component.invoke.args.qsig.ActivateDiversionQ.served_user_number.str = "8340", ++ .component.invoke.args.qsig.ActivateDiversionQ.activating_user_number.plan = 4, ++ .component.invoke.args.qsig.ActivateDiversionQ.activating_user_number.length = 4, ++ .component.invoke.args.qsig.ActivateDiversionQ.activating_user_number.str = "8340", ++ }, ++ { ++ .type = ROSE_COMP_TYPE_RESULT, ++ .component.result.operation = ROSE_QSIG_ActivateDiversionQ, ++ .component.result.invoke_id = 67, ++ }, ++ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_DeactivateDiversionQ, ++ .component.invoke.invoke_id = 68, ++ .component.invoke.args.qsig.DeactivateDiversionQ.procedure = 1, ++ .component.invoke.args.qsig.DeactivateDiversionQ.basic_service = 3, ++ .component.invoke.args.qsig.DeactivateDiversionQ.served_user_number.plan = 4, ++ .component.invoke.args.qsig.DeactivateDiversionQ.served_user_number.length = 4, ++ .component.invoke.args.qsig.DeactivateDiversionQ.served_user_number.str = "8340", ++ .component.invoke.args.qsig.DeactivateDiversionQ.deactivating_user_number.plan = 4, ++ .component.invoke.args.qsig.DeactivateDiversionQ.deactivating_user_number.length = 4, ++ .component.invoke.args.qsig.DeactivateDiversionQ.deactivating_user_number.str = "8340", ++ }, ++ { ++ .type = ROSE_COMP_TYPE_RESULT, ++ .component.result.operation = ROSE_QSIG_DeactivateDiversionQ, ++ .component.result.invoke_id = 69, ++ }, ++ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_InterrogateDiversionQ, ++ .component.invoke.invoke_id = 70, ++ .component.invoke.args.qsig.InterrogateDiversionQ.procedure = 1, ++ .component.invoke.args.qsig.InterrogateDiversionQ.basic_service = 3, ++ .component.invoke.args.qsig.InterrogateDiversionQ.served_user_number.plan = 4, ++ .component.invoke.args.qsig.InterrogateDiversionQ.served_user_number.length = 4, ++ .component.invoke.args.qsig.InterrogateDiversionQ.served_user_number.str = "8340", ++ .component.invoke.args.qsig.InterrogateDiversionQ.interrogating_user_number.plan = 4, ++ .component.invoke.args.qsig.InterrogateDiversionQ.interrogating_user_number.length = 4, ++ .component.invoke.args.qsig.InterrogateDiversionQ.interrogating_user_number.str = "8340", ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_InterrogateDiversionQ, ++ .component.invoke.invoke_id = 71, ++ .component.invoke.args.qsig.InterrogateDiversionQ.procedure = 1, ++ .component.invoke.args.qsig.InterrogateDiversionQ.basic_service = 0,/* default */ ++ .component.invoke.args.qsig.InterrogateDiversionQ.served_user_number.plan = 4, ++ .component.invoke.args.qsig.InterrogateDiversionQ.served_user_number.length = 4, ++ .component.invoke.args.qsig.InterrogateDiversionQ.served_user_number.str = "8340", ++ .component.invoke.args.qsig.InterrogateDiversionQ.interrogating_user_number.plan = 4, ++ .component.invoke.args.qsig.InterrogateDiversionQ.interrogating_user_number.length = 4, ++ .component.invoke.args.qsig.InterrogateDiversionQ.interrogating_user_number.str = "8340", ++ }, ++ { ++ .type = ROSE_COMP_TYPE_RESULT, ++ .component.result.operation = ROSE_QSIG_InterrogateDiversionQ, ++ .component.result.invoke_id = 72, ++ .component.result.args.qsig.InterrogateDiversionQ.num_records = 0, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_RESULT, ++ .component.result.operation = ROSE_QSIG_InterrogateDiversionQ, ++ .component.result.invoke_id = 73, ++ .component.result.args.qsig.InterrogateDiversionQ.num_records = 1, ++ .component.result.args.qsig.InterrogateDiversionQ.list[0].served_user_number.plan = 4, ++ .component.result.args.qsig.InterrogateDiversionQ.list[0].served_user_number.length = 4, ++ .component.result.args.qsig.InterrogateDiversionQ.list[0].served_user_number.str = "8340", ++ .component.result.args.qsig.InterrogateDiversionQ.list[0].basic_service = 3, ++ .component.result.args.qsig.InterrogateDiversionQ.list[0].procedure = 2, ++ .component.result.args.qsig.InterrogateDiversionQ.list[0].diverted_to.number.plan = 4, ++ .component.result.args.qsig.InterrogateDiversionQ.list[0].diverted_to.number.length = 4, ++ .component.result.args.qsig.InterrogateDiversionQ.list[0].diverted_to.number.str = "8340", ++ .component.result.args.qsig.InterrogateDiversionQ.list[0].remote_enabled = 0, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_RESULT, ++ .component.result.operation = ROSE_QSIG_InterrogateDiversionQ, ++ .component.result.invoke_id = 74, ++ .component.result.args.qsig.InterrogateDiversionQ.num_records = 1, ++ .component.result.args.qsig.InterrogateDiversionQ.list[0].served_user_number.plan = 4, ++ .component.result.args.qsig.InterrogateDiversionQ.list[0].served_user_number.length = 4, ++ .component.result.args.qsig.InterrogateDiversionQ.list[0].served_user_number.str = "8340", ++ .component.result.args.qsig.InterrogateDiversionQ.list[0].basic_service = 3, ++ .component.result.args.qsig.InterrogateDiversionQ.list[0].procedure = 2, ++ .component.result.args.qsig.InterrogateDiversionQ.list[0].diverted_to.number.plan = 4, ++ .component.result.args.qsig.InterrogateDiversionQ.list[0].diverted_to.number.length = 4, ++ .component.result.args.qsig.InterrogateDiversionQ.list[0].diverted_to.number.str = "8340", ++ .component.result.args.qsig.InterrogateDiversionQ.list[0].remote_enabled = 1, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_RESULT, ++ .component.result.operation = ROSE_QSIG_InterrogateDiversionQ, ++ .component.result.invoke_id = 75, ++ .component.result.args.qsig.InterrogateDiversionQ.num_records = 2, ++ .component.result.args.qsig.InterrogateDiversionQ.list[0].served_user_number.plan = 4, ++ .component.result.args.qsig.InterrogateDiversionQ.list[0].served_user_number.length = 4, ++ .component.result.args.qsig.InterrogateDiversionQ.list[0].served_user_number.str = "8340", ++ .component.result.args.qsig.InterrogateDiversionQ.list[0].basic_service = 3, ++ .component.result.args.qsig.InterrogateDiversionQ.list[0].procedure = 2, ++ .component.result.args.qsig.InterrogateDiversionQ.list[0].diverted_to.number.plan = 4, ++ .component.result.args.qsig.InterrogateDiversionQ.list[0].diverted_to.number.length = 4, ++ .component.result.args.qsig.InterrogateDiversionQ.list[0].diverted_to.number.str = "8340", ++ .component.result.args.qsig.InterrogateDiversionQ.list[1].served_user_number.plan = 4, ++ .component.result.args.qsig.InterrogateDiversionQ.list[1].served_user_number.length = 4, ++ .component.result.args.qsig.InterrogateDiversionQ.list[1].served_user_number.str = "8340", ++ .component.result.args.qsig.InterrogateDiversionQ.list[1].basic_service = 3, ++ .component.result.args.qsig.InterrogateDiversionQ.list[1].procedure = 2, ++ .component.result.args.qsig.InterrogateDiversionQ.list[1].diverted_to.number.plan = 4, ++ .component.result.args.qsig.InterrogateDiversionQ.list[1].diverted_to.number.length = 4, ++ .component.result.args.qsig.InterrogateDiversionQ.list[1].diverted_to.number.str = "8340", ++ .component.result.args.qsig.InterrogateDiversionQ.list[1].remote_enabled = 1, ++ }, ++ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_CheckRestriction, ++ .component.invoke.invoke_id = 76, ++ .component.invoke.args.qsig.CheckRestriction.served_user_number.plan = 4, ++ .component.invoke.args.qsig.CheckRestriction.served_user_number.length = 4, ++ .component.invoke.args.qsig.CheckRestriction.served_user_number.str = "8340", ++ .component.invoke.args.qsig.CheckRestriction.basic_service = 3, ++ .component.invoke.args.qsig.CheckRestriction.diverted_to_number.plan = 4, ++ .component.invoke.args.qsig.CheckRestriction.diverted_to_number.length = 4, ++ .component.invoke.args.qsig.CheckRestriction.diverted_to_number.str = "8340", ++ }, ++ { ++ .type = ROSE_COMP_TYPE_RESULT, ++ .component.result.operation = ROSE_QSIG_CheckRestriction, ++ .component.result.invoke_id = 77, ++ }, ++ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_CallRerouting, ++ .component.invoke.invoke_id = 78, ++ .component.invoke.args.qsig.CallRerouting.rerouting_reason = 3, ++ .component.invoke.args.qsig.CallRerouting.called.number.plan = 4, ++ .component.invoke.args.qsig.CallRerouting.called.number.length = 4, ++ .component.invoke.args.qsig.CallRerouting.called.number.str = "8340", ++ .component.invoke.args.qsig.CallRerouting.diversion_counter = 5, ++ .component.invoke.args.qsig.CallRerouting.q931ie.length = 2, ++ .component.invoke.args.qsig.CallRerouting.q931ie_contents = "RT", ++ .component.invoke.args.qsig.CallRerouting.last_rerouting.presentation = 1, ++ .component.invoke.args.qsig.CallRerouting.subscription_option = 2, ++ .component.invoke.args.qsig.CallRerouting.calling.presentation = 1, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_CallRerouting, ++ .component.invoke.invoke_id = 79, ++ .component.invoke.args.qsig.CallRerouting.rerouting_reason = 3, ++ .component.invoke.args.qsig.CallRerouting.original_rerouting_reason_present = 1, ++ .component.invoke.args.qsig.CallRerouting.original_rerouting_reason = 2, ++ .component.invoke.args.qsig.CallRerouting.called.number.plan = 4, ++ .component.invoke.args.qsig.CallRerouting.called.number.length = 4, ++ .component.invoke.args.qsig.CallRerouting.called.number.str = "8340", ++ .component.invoke.args.qsig.CallRerouting.diversion_counter = 5, ++ .component.invoke.args.qsig.CallRerouting.q931ie.length = 2, ++ .component.invoke.args.qsig.CallRerouting.q931ie_contents = "RT", ++ .component.invoke.args.qsig.CallRerouting.last_rerouting.presentation = 1, ++ .component.invoke.args.qsig.CallRerouting.subscription_option = 2, ++ .component.invoke.args.qsig.CallRerouting.calling_subaddress.type = 1, ++ .component.invoke.args.qsig.CallRerouting.calling_subaddress.length = 4, ++ .component.invoke.args.qsig.CallRerouting.calling_subaddress.u.nsap = "3253", ++ .component.invoke.args.qsig.CallRerouting.calling.presentation = 1, ++ .component.invoke.args.qsig.CallRerouting.calling_name_present = 1, ++ .component.invoke.args.qsig.CallRerouting.calling_name.presentation = 4, ++ .component.invoke.args.qsig.CallRerouting.calling_name.char_set = 1, ++ .component.invoke.args.qsig.CallRerouting.original_called_present = 1, ++ .component.invoke.args.qsig.CallRerouting.original_called.presentation = 2, ++ .component.invoke.args.qsig.CallRerouting.redirecting_name_present = 1, ++ .component.invoke.args.qsig.CallRerouting.redirecting_name.presentation = 4, ++ .component.invoke.args.qsig.CallRerouting.redirecting_name.char_set = 1, ++ .component.invoke.args.qsig.CallRerouting.original_called_name_present = 1, ++ .component.invoke.args.qsig.CallRerouting.original_called_name.presentation = 4, ++ .component.invoke.args.qsig.CallRerouting.original_called_name.char_set = 1, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_RESULT, ++ .component.result.operation = ROSE_QSIG_CallRerouting, ++ .component.result.invoke_id = 80, ++ }, ++ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_DivertingLegInformation1, ++ .component.invoke.invoke_id = 81, ++ .component.invoke.args.qsig.DivertingLegInformation1.diversion_reason = 3, ++ .component.invoke.args.qsig.DivertingLegInformation1.subscription_option = 1, ++ .component.invoke.args.qsig.DivertingLegInformation1.nominated_number.plan = 4, ++ .component.invoke.args.qsig.DivertingLegInformation1.nominated_number.length = 4, ++ .component.invoke.args.qsig.DivertingLegInformation1.nominated_number.str = "8340", ++ }, ++ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_DivertingLegInformation2, ++ .component.invoke.invoke_id = 82, ++ .component.invoke.args.qsig.DivertingLegInformation2.diversion_counter = 6, ++ .component.invoke.args.qsig.DivertingLegInformation2.diversion_reason = 3, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_DivertingLegInformation2, ++ .component.invoke.invoke_id = 83, ++ .component.invoke.args.qsig.DivertingLegInformation2.diversion_counter = 6, ++ .component.invoke.args.qsig.DivertingLegInformation2.diversion_reason = 3, ++ .component.invoke.args.qsig.DivertingLegInformation2.original_diversion_reason_present = 1, ++ .component.invoke.args.qsig.DivertingLegInformation2.original_diversion_reason = 2, ++ .component.invoke.args.qsig.DivertingLegInformation2.diverting_present = 1, ++ .component.invoke.args.qsig.DivertingLegInformation2.diverting.presentation = 2, ++ .component.invoke.args.qsig.DivertingLegInformation2.original_called_present = 1, ++ .component.invoke.args.qsig.DivertingLegInformation2.original_called.presentation = 2, ++ .component.invoke.args.qsig.DivertingLegInformation2.redirecting_name_present = 1, ++ .component.invoke.args.qsig.DivertingLegInformation2.redirecting_name.presentation = 4, ++ .component.invoke.args.qsig.DivertingLegInformation2.redirecting_name.char_set = 1, ++ .component.invoke.args.qsig.DivertingLegInformation2.original_called_name_present = 1, ++ .component.invoke.args.qsig.DivertingLegInformation2.original_called_name.presentation = 4, ++ .component.invoke.args.qsig.DivertingLegInformation2.original_called_name.char_set = 1, ++ }, ++ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_DivertingLegInformation3, ++ .component.invoke.invoke_id = 84, ++ .component.invoke.args.qsig.DivertingLegInformation3.presentation_allowed_indicator = 1, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_DivertingLegInformation3, ++ .component.invoke.invoke_id = 85, ++ .component.invoke.args.qsig.DivertingLegInformation3.presentation_allowed_indicator = 1, ++ .component.invoke.args.qsig.DivertingLegInformation3.redirection_name_present = 1, ++ .component.invoke.args.qsig.DivertingLegInformation3.redirection_name.presentation = 4, ++ .component.invoke.args.qsig.DivertingLegInformation3.redirection_name.char_set = 1, ++ }, ++ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_CfnrDivertedLegFailed, ++ .component.invoke.invoke_id = 86, ++ }, ++ ++ /* Q.SIG SS-MWI-Operations */ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_MWIActivate, ++ .component.invoke.invoke_id = 102, ++ .component.invoke.args.qsig.MWIActivate.served_user_number.plan = 4, ++ .component.invoke.args.qsig.MWIActivate.served_user_number.length = 4, ++ .component.invoke.args.qsig.MWIActivate.served_user_number.str = "9838", ++ .component.invoke.args.qsig.MWIActivate.basic_service = 1, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_MWIActivate, ++ .component.invoke.invoke_id = 103, ++ .component.invoke.args.qsig.MWIActivate.served_user_number.plan = 4, ++ .component.invoke.args.qsig.MWIActivate.served_user_number.length = 4, ++ .component.invoke.args.qsig.MWIActivate.served_user_number.str = "9838", ++ .component.invoke.args.qsig.MWIActivate.basic_service = 1, ++ .component.invoke.args.qsig.MWIActivate.msg_centre_id_present = 1, ++ .component.invoke.args.qsig.MWIActivate.msg_centre_id.type = 0, ++ .component.invoke.args.qsig.MWIActivate.msg_centre_id.u.integer = 532, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_MWIActivate, ++ .component.invoke.invoke_id = 104, ++ .component.invoke.args.qsig.MWIActivate.served_user_number.plan = 4, ++ .component.invoke.args.qsig.MWIActivate.served_user_number.length = 4, ++ .component.invoke.args.qsig.MWIActivate.served_user_number.str = "9838", ++ .component.invoke.args.qsig.MWIActivate.basic_service = 1, ++ .component.invoke.args.qsig.MWIActivate.msg_centre_id_present = 1, ++ .component.invoke.args.qsig.MWIActivate.msg_centre_id.type = 1, ++ .component.invoke.args.qsig.MWIActivate.msg_centre_id.u.number.plan = 4, ++ .component.invoke.args.qsig.MWIActivate.msg_centre_id.u.number.length = 4, ++ .component.invoke.args.qsig.MWIActivate.msg_centre_id.u.number.str = "9838", ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_MWIActivate, ++ .component.invoke.invoke_id = 105, ++ .component.invoke.args.qsig.MWIActivate.served_user_number.plan = 4, ++ .component.invoke.args.qsig.MWIActivate.served_user_number.length = 4, ++ .component.invoke.args.qsig.MWIActivate.served_user_number.str = "9838", ++ .component.invoke.args.qsig.MWIActivate.basic_service = 1, ++ .component.invoke.args.qsig.MWIActivate.msg_centre_id_present = 1, ++ .component.invoke.args.qsig.MWIActivate.msg_centre_id.type = 2, ++ .component.invoke.args.qsig.MWIActivate.msg_centre_id.u.str = "123456", ++ .component.invoke.args.qsig.MWIActivate.number_of_messages_present = 1, ++ .component.invoke.args.qsig.MWIActivate.number_of_messages = 6548, ++ .component.invoke.args.qsig.MWIActivate.originating_number.plan = 4, ++ .component.invoke.args.qsig.MWIActivate.originating_number.length = 4, ++ .component.invoke.args.qsig.MWIActivate.originating_number.str = "9838", ++ .component.invoke.args.qsig.MWIActivate.timestamp_present = 1, ++ .component.invoke.args.qsig.MWIActivate.timestamp = "19970621194530", ++ .component.invoke.args.qsig.MWIActivate.priority_present = 1, ++ .component.invoke.args.qsig.MWIActivate.priority = 7, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_RESULT, ++ .component.result.operation = ROSE_QSIG_MWIActivate, ++ .component.result.invoke_id = 106, ++ }, ++ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_MWIDeactivate, ++ .component.invoke.invoke_id = 107, ++ .component.invoke.args.qsig.MWIDeactivate.served_user_number.plan = 4, ++ .component.invoke.args.qsig.MWIDeactivate.served_user_number.length = 4, ++ .component.invoke.args.qsig.MWIDeactivate.served_user_number.str = "9838", ++ .component.invoke.args.qsig.MWIDeactivate.basic_service = 1, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_MWIDeactivate, ++ .component.invoke.invoke_id = 108, ++ .component.invoke.args.qsig.MWIDeactivate.served_user_number.plan = 4, ++ .component.invoke.args.qsig.MWIDeactivate.served_user_number.length = 4, ++ .component.invoke.args.qsig.MWIDeactivate.served_user_number.str = "9838", ++ .component.invoke.args.qsig.MWIDeactivate.basic_service = 1, ++ .component.invoke.args.qsig.MWIDeactivate.msg_centre_id_present = 1, ++ .component.invoke.args.qsig.MWIDeactivate.msg_centre_id.type = 0, ++ .component.invoke.args.qsig.MWIDeactivate.msg_centre_id.u.integer = 532, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_RESULT, ++ .component.result.operation = ROSE_QSIG_MWIDeactivate, ++ .component.result.invoke_id = 109, ++ }, ++ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_MWIInterrogate, ++ .component.invoke.invoke_id = 110, ++ .component.invoke.args.qsig.MWIInterrogate.served_user_number.plan = 4, ++ .component.invoke.args.qsig.MWIInterrogate.served_user_number.length = 4, ++ .component.invoke.args.qsig.MWIInterrogate.served_user_number.str = "9838", ++ .component.invoke.args.qsig.MWIInterrogate.basic_service = 1, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_QSIG_MWIInterrogate, ++ .component.invoke.invoke_id = 111, ++ .component.invoke.args.qsig.MWIInterrogate.served_user_number.plan = 4, ++ .component.invoke.args.qsig.MWIInterrogate.served_user_number.length = 4, ++ .component.invoke.args.qsig.MWIInterrogate.served_user_number.str = "9838", ++ .component.invoke.args.qsig.MWIInterrogate.basic_service = 1, ++ .component.invoke.args.qsig.MWIInterrogate.msg_centre_id_present = 1, ++ .component.invoke.args.qsig.MWIInterrogate.msg_centre_id.type = 0, ++ .component.invoke.args.qsig.MWIInterrogate.msg_centre_id.u.integer = 532, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_RESULT, ++ .component.result.operation = ROSE_QSIG_MWIInterrogate, ++ .component.result.invoke_id = 112, ++ .component.result.args.qsig.MWIInterrogate.num_records = 1, ++ .component.result.args.qsig.MWIInterrogate.list[0].basic_service = 1, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_RESULT, ++ .component.result.operation = ROSE_QSIG_MWIInterrogate, ++ .component.result.invoke_id = 113, ++ .component.result.args.qsig.MWIInterrogate.num_records = 2, ++ .component.result.args.qsig.MWIInterrogate.list[0].basic_service = 1, ++ .component.result.args.qsig.MWIInterrogate.list[0].msg_centre_id_present = 1, ++ .component.result.args.qsig.MWIInterrogate.list[0].msg_centre_id.type = 0, ++ .component.result.args.qsig.MWIInterrogate.list[0].msg_centre_id.u.integer = 987, ++ .component.result.args.qsig.MWIInterrogate.list[0].number_of_messages_present = 1, ++ .component.result.args.qsig.MWIInterrogate.list[0].number_of_messages = 6548, ++ .component.result.args.qsig.MWIInterrogate.list[0].originating_number.plan = 4, ++ .component.result.args.qsig.MWIInterrogate.list[0].originating_number.length = 4, ++ .component.result.args.qsig.MWIInterrogate.list[0].originating_number.str = "9838", ++ .component.result.args.qsig.MWIInterrogate.list[0].timestamp_present = 1, ++ .component.result.args.qsig.MWIInterrogate.list[0].timestamp = "19970621194530", ++ .component.result.args.qsig.MWIInterrogate.list[0].priority_present = 1, ++ .component.result.args.qsig.MWIInterrogate.list[0].priority = 7, ++ .component.result.args.qsig.MWIInterrogate.list[1].basic_service = 1, ++ }, ++/* *INDENT-ON* */ ++}; ++ ++ ++static const struct rose_message rose_dms100_msgs[] = { ++/* *INDENT-OFF* */ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_DMS100_RLT_OperationInd, ++ .component.invoke.invoke_id = ROSE_DMS100_RLT_OPERATION_IND, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_RESULT, ++ .component.result.operation = ROSE_DMS100_RLT_OperationInd, ++ .component.result.invoke_id = ROSE_DMS100_RLT_OPERATION_IND, ++ .component.result.args.dms100.RLT_OperationInd.call_id = 130363, ++ }, ++ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_DMS100_RLT_ThirdParty, ++ .component.invoke.invoke_id = ROSE_DMS100_RLT_THIRD_PARTY, ++ .component.invoke.args.dms100.RLT_ThirdParty.call_id = 120047, ++ .component.invoke.args.dms100.RLT_ThirdParty.reason = 1, ++ }, ++ { ++ .type = ROSE_COMP_TYPE_RESULT, ++ .component.result.operation = ROSE_DMS100_RLT_ThirdParty, ++ .component.result.invoke_id = ROSE_DMS100_RLT_THIRD_PARTY, ++ }, ++/* *INDENT-ON* */ ++}; ++ ++ ++static const struct rose_message rose_ni2_msgs[] = { ++/* *INDENT-OFF* */ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_NI2_InformationFollowing, ++ .component.invoke.invoke_id = 1, ++ .component.invoke.args.ni2.InformationFollowing.value = 7, ++ }, ++ ++ { ++ .type = ROSE_COMP_TYPE_INVOKE, ++ .component.invoke.operation = ROSE_NI2_InitiateTransfer, ++ .component.invoke.invoke_id = 2, ++ .component.invoke.args.ni2.InitiateTransfer.call_reference = 5, ++ }, ++/* *INDENT-ON* */ ++}; ++ ++/* ------------------------------------------------------------------- */ ++ ++static void rose_pri_message(struct pri *ctrl, char *stuff) ++{ ++ fprintf(stdout, "%s", stuff); ++} ++ ++static void rose_pri_error(struct pri *ctrl, char *stuff) ++{ ++ fprintf(stdout, "%s", stuff); ++ fprintf(stderr, "%s", stuff); ++} ++ ++/*! ++ * \internal ++ * \brief Test ROSE encoding and decoding the given message. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param index Message number to report. ++ * \param header Facility message header data to encode. ++ * \param encode_msg Message data to encode. ++ * ++ * \return Nothing ++ */ ++static void rose_test_msg(struct pri *ctrl, unsigned index, ++ const struct fac_extension_header *header, const struct rose_message *encode_msg) ++{ ++ struct fac_extension_header decoded_header; ++ struct rose_message decoded_msg; ++ unsigned char *enc_pos; ++ unsigned char *enc_end; ++ const unsigned char *dec_pos; ++ const unsigned char *dec_end; ++ ++ static unsigned char buf[1024]; ++ ++ pri_message(ctrl, "\n\n"); ++ enc_end = buf + sizeof(buf); ++ enc_pos = facility_encode_header(ctrl, buf, enc_end, header); ++ if (!enc_pos) { ++ pri_error(ctrl, "Error: Message:%u failed to encode header\n", index); ++ } else { ++ enc_pos = rose_encode(ctrl, enc_pos, enc_end, encode_msg); ++ if (!enc_pos) { ++ pri_error(ctrl, "Error: Message:%u failed to encode ROSE\n", index); ++ } else { ++ pri_message(ctrl, "Message %u encoded length is %u\n", index, ++ (unsigned) (enc_pos - buf)); ++ ++ /* Clear the decoded message contents for comparison. */ ++ memset(&decoded_header, 0, sizeof(decoded_header)); ++ memset(&decoded_msg, 0, sizeof(decoded_msg)); ++ ++ dec_end = enc_pos; ++ dec_pos = facility_decode_header(ctrl, buf, dec_end, &decoded_header); ++ if (!dec_pos) { ++ pri_error(ctrl, "Error: Message:%u failed to decode header\n", index); ++ } else { ++ dec_pos = rose_decode(ctrl, dec_pos, dec_end, &decoded_msg); ++ if (!dec_pos) { ++ pri_error(ctrl, "Error: Message:%u failed to decode ROSE\n", index); ++ } else { ++ if (header ++ && memcmp(header, &decoded_header, sizeof(decoded_header))) { ++ pri_error(ctrl, "Error: Message:%u Header did not match\n", ++ index); ++ } ++ if (memcmp(encode_msg, &decoded_msg, sizeof(decoded_msg))) { ++ pri_error(ctrl, "Error: Message:%u ROSE did not match\n", index); ++ } ++ } ++ } ++ } ++ } ++ pri_message(ctrl, "\n\n" ++ "************************************************************\n"); ++} ++ ++/*! ++ * \internal ++ * \brief Test ROSE decoding messages of unusual encodings. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param name Test name for the encoded message. ++ * \param msg_buf Encoded message to decode. ++ * \param msg_len Length of encoded message buffer. ++ * ++ * \return Nothing ++ */ ++static void rose_test_exception(struct pri *ctrl, const char *name, ++ const unsigned char *msg, size_t msg_len) ++{ ++ const unsigned char *pos; ++ const unsigned char *end; ++ struct fac_extension_header header; ++ struct rose_message decoded_msg; ++ ++ pri_message(ctrl, "\n\n" ++ "%s test: Message encoded length is %u\n", name, (unsigned) msg_len); ++ ++ pos = msg; ++ end = msg + msg_len; ++ pos = facility_decode_header(ctrl, pos, end, &header); ++ if (!pos) { ++ pri_error(ctrl, "Error: %s test: Message failed to decode header\n", name); ++ } else { ++ pos = rose_decode(ctrl, pos, end, &decoded_msg); ++ if (!pos) { ++ pri_error(ctrl, "Error: %s test: Message failed to decode ROSE\n", name); ++ } ++ } ++ ++ pri_message(ctrl, "\n\n" ++ "************************************************************\n"); ++} ++ ++/*! ++ * \brief ROSE encode/decode test program. ++ * ++ * \param argc Program argument count. ++ * \param argv Program argument string array. ++ * ++ * \retval 0 on success. ++ * \retval Nonzero on error. ++ */ ++int main(int argc, char *argv[]) ++{ ++ unsigned index; ++ unsigned offset; ++ static struct pri dummy_ctrl; ++ ++ pri_set_message(rose_pri_message); ++ pri_set_error(rose_pri_error); ++ ++ memset(&dummy_ctrl, 0, sizeof(dummy_ctrl)); ++ dummy_ctrl.debug = PRI_DEBUG_APDU; ++ ++ offset = 0; ++ pri_message(&dummy_ctrl, "Encode/decode message(s)\n"); ++ if (argc <= 1) { ++ dummy_ctrl.switchtype = PRI_SWITCH_EUROISDN_E1; ++ for (index = 0; index < ARRAY_LEN(rose_etsi_msgs); ++index) { ++ rose_test_msg(&dummy_ctrl, index + offset, &fac_headers[0], ++ &rose_etsi_msgs[index]); ++ } ++ offset += ARRAY_LEN(rose_etsi_msgs); ++ ++ dummy_ctrl.switchtype = PRI_SWITCH_QSIG; ++ for (index = 0; index < ARRAY_LEN(rose_qsig_msgs); ++index) { ++ rose_test_msg(&dummy_ctrl, index + offset, ++ &fac_headers[index % ARRAY_LEN(fac_headers)], &rose_qsig_msgs[index]); ++ } ++ offset += ARRAY_LEN(rose_qsig_msgs); ++ ++ dummy_ctrl.switchtype = PRI_SWITCH_DMS100; ++ for (index = 0; index < ARRAY_LEN(rose_dms100_msgs); ++index) { ++ rose_test_msg(&dummy_ctrl, index + offset, &fac_headers[0], ++ &rose_dms100_msgs[index]); ++ } ++ offset += ARRAY_LEN(rose_dms100_msgs); ++ ++ dummy_ctrl.switchtype = PRI_SWITCH_NI2; ++ for (index = 0; index < ARRAY_LEN(rose_ni2_msgs); ++index) { ++ rose_test_msg(&dummy_ctrl, index + offset, &fac_headers[0], ++ &rose_ni2_msgs[index]); ++ } ++ //offset += ARRAY_LEN(rose_ni2_msgs); ++ } else { ++ index = atoi(argv[1]); ++ ++ if (index < ARRAY_LEN(rose_etsi_msgs)) { ++ dummy_ctrl.switchtype = PRI_SWITCH_EUROISDN_E1; ++ rose_test_msg(&dummy_ctrl, index + offset, &fac_headers[0], ++ &rose_etsi_msgs[index]); ++ return 0; ++ } ++ offset += ARRAY_LEN(rose_etsi_msgs); ++ index -= ARRAY_LEN(rose_etsi_msgs); ++ ++ if (index < ARRAY_LEN(rose_qsig_msgs)) { ++ dummy_ctrl.switchtype = PRI_SWITCH_QSIG; ++ rose_test_msg(&dummy_ctrl, index + offset, ++ &fac_headers[index % ARRAY_LEN(fac_headers)], &rose_qsig_msgs[index]); ++ return 0; ++ } ++ offset += ARRAY_LEN(rose_qsig_msgs); ++ index -= ARRAY_LEN(rose_qsig_msgs); ++ ++ if (index < ARRAY_LEN(rose_dms100_msgs)) { ++ dummy_ctrl.switchtype = PRI_SWITCH_DMS100; ++ rose_test_msg(&dummy_ctrl, index + offset, &fac_headers[0], ++ &rose_dms100_msgs[index]); ++ return 0; ++ } ++ offset += ARRAY_LEN(rose_dms100_msgs); ++ index -= ARRAY_LEN(rose_dms100_msgs); ++ ++ if (index < ARRAY_LEN(rose_ni2_msgs)) { ++ dummy_ctrl.switchtype = PRI_SWITCH_NI2; ++ rose_test_msg(&dummy_ctrl, index + offset, &fac_headers[0], ++ &rose_ni2_msgs[index]); ++ return 0; ++ } ++ //offset += ARRAY_LEN(rose_ni2_msgs); ++ //index -= ARRAY_LEN(rose_ni2_msgs); ++ ++ fprintf(stderr, "Invalid option\n"); ++ return 0; ++ } ++ ++/* ------------------------------------------------------------------- */ ++ ++ pri_message(&dummy_ctrl, "\n\n" ++ "Decode unusually encoded messages\n"); ++ ++ dummy_ctrl.switchtype = PRI_SWITCH_EUROISDN_E1; ++ ++ rose_test_exception(&dummy_ctrl, "Extra bytes on end", rose_etsi_extra, ++ sizeof(rose_etsi_extra)); ++ ++ rose_test_exception(&dummy_ctrl, "Indefinite length", rose_etsi_indefinite_len, ++ sizeof(rose_etsi_indefinite_len) - 2); ++ rose_test_exception(&dummy_ctrl, "Indefinite length (extra)", ++ rose_etsi_indefinite_len, sizeof(rose_etsi_indefinite_len)); ++ ++ rose_test_exception(&dummy_ctrl, "Unused components (indefinite length)", ++ rose_etsi_unused_indefinite_len, sizeof(rose_etsi_unused_indefinite_len) - 2); ++ rose_test_exception(&dummy_ctrl, "Unused components (indefinite length, extra)", ++ rose_etsi_unused_indefinite_len, sizeof(rose_etsi_unused_indefinite_len)); ++ ++ rose_test_exception(&dummy_ctrl, "Unused components", rose_etsi_unused, ++ sizeof(rose_etsi_unused) - 2); ++ rose_test_exception(&dummy_ctrl, "Unused components (extra)", rose_etsi_unused, ++ sizeof(rose_etsi_unused)); ++ ++/* ------------------------------------------------------------------- */ ++ ++ pri_message(&dummy_ctrl, "\n\n" ++ "List of operation codes:\n"); ++ for (index = 0; index < ROSE_Num_Operation_Codes; ++index) { ++ pri_message(&dummy_ctrl, "%d: %s\n", index, rose_operation2str(index)); ++ } ++ pri_message(&dummy_ctrl, "\n\n" ++ "************************************************************\n"); ++ ++/* ------------------------------------------------------------------- */ ++ ++ pri_message(&dummy_ctrl, "\n\n" ++ "List of error codes:\n"); ++ for (index = 0; index < ROSE_ERROR_Num_Codes; ++index) { ++ pri_message(&dummy_ctrl, "%d: %s\n", index, rose_error2str(index)); ++ } ++ pri_message(&dummy_ctrl, "\n\n" ++ "************************************************************\n"); ++ ++/* ------------------------------------------------------------------- */ ++ ++ pri_message(&dummy_ctrl, "\n\n"); ++ pri_message(&dummy_ctrl, "sizeof(struct rose_message) = %u\n", ++ (unsigned) sizeof(struct rose_message)); ++ pri_message(&dummy_ctrl, "sizeof(struct rose_msg_invoke) = %u\n", ++ (unsigned) sizeof(struct rose_msg_invoke)); ++ pri_message(&dummy_ctrl, "sizeof(struct rose_msg_result) = %u\n", ++ (unsigned) sizeof(struct rose_msg_result)); ++ pri_message(&dummy_ctrl, "sizeof(struct rose_msg_error) = %u\n", ++ (unsigned) sizeof(struct rose_msg_error)); ++ pri_message(&dummy_ctrl, "sizeof(struct rose_msg_reject) = %u\n", ++ (unsigned) sizeof(struct rose_msg_reject)); ++ pri_message(&dummy_ctrl, "sizeof(union rose_msg_invoke_args) = %u\n", ++ (unsigned) sizeof(union rose_msg_invoke_args)); ++ pri_message(&dummy_ctrl, "sizeof(union rose_msg_result_args) = %u\n", ++ (unsigned) sizeof(union rose_msg_result_args)); ++ ++ pri_message(&dummy_ctrl, "\n"); ++ pri_message(&dummy_ctrl, "sizeof(struct roseQsigForwardingList) = %u\n", ++ (unsigned) sizeof(struct roseQsigForwardingList)); ++ ++ pri_message(&dummy_ctrl, "\n"); ++ pri_message(&dummy_ctrl, "sizeof(struct roseQsigCallRerouting_ARG) = %u\n", ++ (unsigned) sizeof(struct roseQsigCallRerouting_ARG)); ++ pri_message(&dummy_ctrl, "sizeof(struct roseQsigAocRateArg_ARG) = %u\n", ++ (unsigned) sizeof(struct roseQsigAocRateArg_ARG)); ++ pri_message(&dummy_ctrl, "sizeof(struct roseQsigMWIInterrogateRes) = %u\n", ++ (unsigned) sizeof(struct roseQsigMWIInterrogateRes)); ++ ++ pri_message(&dummy_ctrl, "\n"); ++ pri_message(&dummy_ctrl, "sizeof(struct roseEtsiForwardingList) = %u\n", ++ (unsigned) sizeof(struct roseEtsiForwardingList)); ++ pri_message(&dummy_ctrl, "sizeof(struct roseEtsiServedUserNumberList) = %u\n", ++ (unsigned) sizeof(struct roseEtsiServedUserNumberList)); ++ ++ pri_message(&dummy_ctrl, "\n"); ++ pri_message(&dummy_ctrl, "sizeof(struct roseEtsiCallRerouting_ARG) = %u\n", ++ (unsigned) sizeof(struct roseEtsiCallRerouting_ARG)); ++ pri_message(&dummy_ctrl, "sizeof(struct roseEtsiDiversionInformation_ARG) = %u\n", ++ (unsigned) sizeof(struct roseEtsiDiversionInformation_ARG)); ++ pri_message(&dummy_ctrl, "sizeof(struct roseEtsiAOCSCurrencyInfoList) = %u\n", ++ (unsigned) sizeof(struct roseEtsiAOCSCurrencyInfoList)); ++ ++/* ------------------------------------------------------------------- */ ++ ++ return 0; ++} ++ ++/* ------------------------------------------------------------------- */ ++/* end rosetest.c */ + +Property changes on: rosetest.c +___________________________________________________________________ +Added: svn:eol-style + + native +Added: svn:mime-type + + text/plain +Added: svn:keywords + + 'Author Date Id Revision' + +Index: rose_internal.h +=================================================================== +--- a/rose_internal.h (.../tags/1.4.10.2) (revision 0) ++++ b/rose_internal.h (.../branches/1.4) (revision 1357) +@@ -0,0 +1,477 @@ ++/* ++ * libpri: An implementation of Primary Rate ISDN ++ * ++ * Copyright (C) 2009 Digium, Inc. ++ * ++ * Richard Mudgett <rmudgett@digium.com> ++ * ++ * See http://www.asterisk.org for more information about ++ * the Asterisk project. Please do not directly contact ++ * any of the maintainers of this project for assistance; ++ * the project provides a web site, mailing lists and IRC ++ * channels for your use. ++ * ++ * This program is free software, distributed under the terms of ++ * the GNU General Public License Version 2 as published by the ++ * Free Software Foundation. See the LICENSE file included with ++ * this program for more details. ++ * ++ * In addition, when this program is distributed with Asterisk in ++ * any form that would qualify as a 'combined work' or as a ++ * 'derivative work' (but not mere aggregation), you can redistribute ++ * and/or modify the combination under the terms of the license ++ * provided with that copy of Asterisk, instead of the license ++ * terms granted here. ++ */ ++ ++/*! ++ * \file ++ * \brief Internal definitions and prototypes for ROSE. ++ * ++ * \author Richard Mudgett <rmudgett@digium.com> ++ */ ++ ++#ifndef _LIBPRI_ROSE_INTERNAL_H ++#define _LIBPRI_ROSE_INTERNAL_H ++ ++#include "rose.h" ++ ++#ifdef __cplusplus ++extern "C" { ++#endif ++ ++ ++/* ------------------------------------------------------------------- */ ++ ++ ++/* Embedded-Q931-Types */ ++unsigned char *rose_enc_Q931ie(struct pri *ctrl, unsigned char *pos, unsigned char *end, ++ unsigned tag, const struct roseQ931ie *q931ie); ++ ++const unsigned char *rose_dec_Q931ie(struct pri *ctrl, const char *name, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, struct roseQ931ie *q931ie, ++ size_t contents_size); ++ ++/* Addressing-Data-Elements */ ++unsigned char *rose_enc_PartyNumber(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const struct rosePartyNumber *party_number); ++unsigned char *rose_enc_PartySubaddress(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const struct rosePartySubaddress *party_subaddress); ++unsigned char *rose_enc_Address(struct pri *ctrl, unsigned char *pos, unsigned char *end, ++ unsigned tag, const struct roseAddress *address); ++unsigned char *rose_enc_PresentedNumberUnscreened(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const struct rosePresentedNumberUnscreened *party); ++unsigned char *rose_enc_NumberScreened(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, unsigned tag, const struct roseNumberScreened *screened); ++unsigned char *rose_enc_PresentedNumberScreened(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const struct rosePresentedNumberScreened *party); ++unsigned char *rose_enc_AddressScreened(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, unsigned tag, const struct roseAddressScreened *screened); ++unsigned char *rose_enc_PresentedAddressScreened(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const struct rosePresentedAddressScreened *party); ++ ++const unsigned char *rose_dec_PartyNumber(struct pri *ctrl, const char *name, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct rosePartyNumber *party_number); ++const unsigned char *rose_dec_PartySubaddress(struct pri *ctrl, const char *name, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct rosePartySubaddress *party_subaddress); ++const unsigned char *rose_dec_Address(struct pri *ctrl, const char *name, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, struct roseAddress *address); ++const unsigned char *rose_dec_PresentedNumberUnscreened(struct pri *ctrl, ++ const char *name, unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct rosePresentedNumberUnscreened *party); ++const unsigned char *rose_dec_NumberScreened(struct pri *ctrl, const char *name, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct roseNumberScreened *screened); ++const unsigned char *rose_dec_PresentedNumberScreened(struct pri *ctrl, const char *name, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct rosePresentedNumberScreened *party); ++const unsigned char *rose_dec_AddressScreened(struct pri *ctrl, const char *name, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct roseAddressScreened *screened); ++const unsigned char *rose_dec_PresentedAddressScreened(struct pri *ctrl, ++ const char *name, unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct rosePresentedAddressScreened *party); ++ ++/* ETSI Advice-of-Charge (AOC) */ ++unsigned char *rose_enc_etsi_ChargingRequest_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args); ++unsigned char *rose_enc_etsi_ChargingRequest_RES(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_result_args *args); ++unsigned char *rose_enc_etsi_AOCSCurrency_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args); ++unsigned char *rose_enc_etsi_AOCSSpecialArr_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args); ++unsigned char *rose_enc_etsi_AOCDCurrency_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args); ++unsigned char *rose_enc_etsi_AOCDChargingUnit_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args); ++unsigned char *rose_enc_etsi_AOCECurrency_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args); ++unsigned char *rose_enc_etsi_AOCEChargingUnit_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args); ++ ++const unsigned char *rose_dec_etsi_ChargingRequest_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++const unsigned char *rose_dec_etsi_ChargingRequest_RES(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, ++ union rose_msg_result_args *args); ++const unsigned char *rose_dec_etsi_AOCSCurrency_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++const unsigned char *rose_dec_etsi_AOCSSpecialArr_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++const unsigned char *rose_dec_etsi_AOCDCurrency_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++const unsigned char *rose_dec_etsi_AOCDChargingUnit_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++const unsigned char *rose_dec_etsi_AOCECurrency_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++const unsigned char *rose_dec_etsi_AOCEChargingUnit_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++ ++/* ETSI Call Diversion */ ++unsigned char *rose_enc_etsi_ActivationDiversion_ARG(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, const union rose_msg_invoke_args *args); ++unsigned char *rose_enc_etsi_DeactivationDiversion_ARG(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, const union rose_msg_invoke_args *args); ++unsigned char *rose_enc_etsi_ActivationStatusNotificationDiv_ARG(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, const union rose_msg_invoke_args *args); ++unsigned char *rose_enc_etsi_DeactivationStatusNotificationDiv_ARG(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, const union rose_msg_invoke_args *args); ++unsigned char *rose_enc_etsi_InterrogationDiversion_ARG(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, const union rose_msg_invoke_args *args); ++unsigned char *rose_enc_etsi_InterrogationDiversion_RES(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, const union rose_msg_result_args *args); ++unsigned char *rose_enc_etsi_DiversionInformation_ARG(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, const union rose_msg_invoke_args *args); ++unsigned char *rose_enc_etsi_CallDeflection_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args); ++unsigned char *rose_enc_etsi_CallRerouting_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args); ++unsigned char *rose_enc_etsi_InterrogateServedUserNumbers_RES(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, const union rose_msg_result_args *args); ++unsigned char *rose_enc_etsi_DivertingLegInformation1_ARG(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, const union rose_msg_invoke_args *args); ++unsigned char *rose_enc_etsi_DivertingLegInformation2_ARG(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, const union rose_msg_invoke_args *args); ++unsigned char *rose_enc_etsi_DivertingLegInformation3_ARG(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, const union rose_msg_invoke_args *args); ++ ++const unsigned char *rose_dec_etsi_ActivationDiversion_ARG(struct pri *ctrl, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++const unsigned char *rose_dec_etsi_DeactivationDiversion_ARG(struct pri *ctrl, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++const unsigned char *rose_dec_etsi_ActivationStatusNotificationDiv_ARG(struct pri *ctrl, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++const unsigned char *rose_dec_etsi_DeactivationStatusNotificationDiv_ARG(struct pri ++ *ctrl, unsigned tag, const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++const unsigned char *rose_dec_etsi_InterrogationDiversion_ARG(struct pri *ctrl, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++const unsigned char *rose_dec_etsi_InterrogationDiversion_RES(struct pri *ctrl, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ union rose_msg_result_args *args); ++const unsigned char *rose_dec_etsi_DiversionInformation_ARG(struct pri *ctrl, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++const unsigned char *rose_dec_etsi_CallDeflection_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++const unsigned char *rose_dec_etsi_CallRerouting_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++const unsigned char *rose_dec_etsi_InterrogateServedUserNumbers_RES(struct pri *ctrl, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ union rose_msg_result_args *args); ++const unsigned char *rose_dec_etsi_DivertingLegInformation1_ARG(struct pri *ctrl, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++const unsigned char *rose_dec_etsi_DivertingLegInformation2_ARG(struct pri *ctrl, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++const unsigned char *rose_dec_etsi_DivertingLegInformation3_ARG(struct pri *ctrl, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++ ++/* ETSI Explicit Call Transfer (ECT) */ ++unsigned char *rose_enc_etsi_ExplicitEctExecute_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args); ++unsigned char *rose_enc_etsi_SubaddressTransfer_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args); ++unsigned char *rose_enc_etsi_EctLinkIdRequest_RES(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_result_args *args); ++unsigned char *rose_enc_etsi_EctInform_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args); ++unsigned char *rose_enc_etsi_EctLoopTest_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args); ++unsigned char *rose_enc_etsi_EctLoopTest_RES(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_result_args *args); ++ ++const unsigned char *rose_dec_etsi_ExplicitEctExecute_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++const unsigned char *rose_dec_etsi_SubaddressTransfer_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++const unsigned char *rose_dec_etsi_EctLinkIdRequest_RES(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, ++ union rose_msg_result_args *args); ++const unsigned char *rose_dec_etsi_EctInform_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++const unsigned char *rose_dec_etsi_EctLoopTest_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++const unsigned char *rose_dec_etsi_EctLoopTest_RES(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, ++ union rose_msg_result_args *args); ++ ++/* Q.SIG Name-Operations */ ++unsigned char *rose_enc_qsig_Name(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const struct roseQsigName *name); ++ ++const unsigned char *rose_dec_qsig_Name(struct pri *ctrl, const char *fname, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct roseQsigName *name); ++ ++unsigned char *rose_enc_qsig_CallingName_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args); ++unsigned char *rose_enc_qsig_CalledName_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args); ++unsigned char *rose_enc_qsig_ConnectedName_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args); ++unsigned char *rose_enc_qsig_BusyName_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args); ++ ++const unsigned char *rose_dec_qsig_CallingName_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++const unsigned char *rose_dec_qsig_CalledName_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++const unsigned char *rose_dec_qsig_ConnectedName_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++const unsigned char *rose_dec_qsig_BusyName_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++ ++/* ++ * Q.SIG Dummy invoke/result argument used by: ++ * SS-AOC-Operations, ++ * Call-Transfer-Operations, ++ * Call-Diversion-Operations, ++ * and SS-MWI-Operations. ++ */ ++unsigned char *rose_enc_qsig_DummyArg_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args); ++unsigned char *rose_enc_qsig_DummyRes_RES(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_result_args *args); ++ ++const unsigned char *rose_dec_qsig_DummyArg_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++const unsigned char *rose_dec_qsig_DummyRes_RES(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, ++ union rose_msg_result_args *args); ++ ++/* Q.SIG SS-AOC-Operations */ ++unsigned char *rose_enc_qsig_ChargeRequest_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args); ++unsigned char *rose_enc_qsig_ChargeRequest_RES(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_result_args *args); ++unsigned char *rose_enc_qsig_AocFinal_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args); ++unsigned char *rose_enc_qsig_AocInterim_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args); ++unsigned char *rose_enc_qsig_AocRate_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args); ++unsigned char *rose_enc_qsig_AocComplete_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args); ++unsigned char *rose_enc_qsig_AocComplete_RES(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_result_args *args); ++unsigned char *rose_enc_qsig_AocDivChargeReq_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args); ++ ++const unsigned char *rose_dec_qsig_ChargeRequest_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++const unsigned char *rose_dec_qsig_ChargeRequest_RES(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, ++ union rose_msg_result_args *args); ++const unsigned char *rose_dec_qsig_AocFinal_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++const unsigned char *rose_dec_qsig_AocInterim_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++const unsigned char *rose_dec_qsig_AocRate_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++const unsigned char *rose_dec_qsig_AocComplete_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++const unsigned char *rose_dec_qsig_AocComplete_RES(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, ++ union rose_msg_result_args *args); ++const unsigned char *rose_dec_qsig_AocDivChargeReq_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++ ++/* Q.SIG Call-Diversion-Operations */ ++unsigned char *rose_enc_qsig_ActivateDiversionQ_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args); ++unsigned char *rose_enc_qsig_DeactivateDiversionQ_ARG(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, const union rose_msg_invoke_args *args); ++unsigned char *rose_enc_qsig_InterrogateDiversionQ_ARG(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, const union rose_msg_invoke_args *args); ++unsigned char *rose_enc_qsig_InterrogateDiversionQ_RES(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, const union rose_msg_result_args *args); ++unsigned char *rose_enc_qsig_CheckRestriction_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args); ++unsigned char *rose_enc_qsig_CallRerouting_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args); ++unsigned char *rose_enc_qsig_DivertingLegInformation1_ARG(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, const union rose_msg_invoke_args *args); ++unsigned char *rose_enc_qsig_DivertingLegInformation2_ARG(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, const union rose_msg_invoke_args *args); ++unsigned char *rose_enc_qsig_DivertingLegInformation3_ARG(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, const union rose_msg_invoke_args *args); ++ ++const unsigned char *rose_dec_qsig_ActivateDiversionQ_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++const unsigned char *rose_dec_qsig_DeactivateDiversionQ_ARG(struct pri *ctrl, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++const unsigned char *rose_dec_qsig_InterrogateDiversionQ_ARG(struct pri *ctrl, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++const unsigned char *rose_dec_qsig_InterrogateDiversionQ_RES(struct pri *ctrl, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ union rose_msg_result_args *args); ++const unsigned char *rose_dec_qsig_CheckRestriction_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++const unsigned char *rose_dec_qsig_CallRerouting_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++const unsigned char *rose_dec_qsig_DivertingLegInformation1_ARG(struct pri *ctrl, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++const unsigned char *rose_dec_qsig_DivertingLegInformation2_ARG(struct pri *ctrl, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++const unsigned char *rose_dec_qsig_DivertingLegInformation3_ARG(struct pri *ctrl, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++ ++/* Q.SIG Call-Transfer-Operations (CT) */ ++unsigned char *rose_enc_qsig_CallTransferIdentify_RES(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, const union rose_msg_result_args *args); ++unsigned char *rose_enc_qsig_CallTransferInitiate_ARG(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, const union rose_msg_invoke_args *args); ++unsigned char *rose_enc_qsig_CallTransferSetup_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args); ++unsigned char *rose_enc_qsig_CallTransferActive_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args); ++unsigned char *rose_enc_qsig_CallTransferComplete_ARG(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, const union rose_msg_invoke_args *args); ++unsigned char *rose_enc_qsig_CallTransferUpdate_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args); ++unsigned char *rose_enc_qsig_SubaddressTransfer_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args); ++ ++const unsigned char *rose_dec_qsig_CallTransferIdentify_RES(struct pri *ctrl, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ union rose_msg_result_args *args); ++const unsigned char *rose_dec_qsig_CallTransferInitiate_ARG(struct pri *ctrl, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++const unsigned char *rose_dec_qsig_CallTransferSetup_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++const unsigned char *rose_dec_qsig_CallTransferActive_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++const unsigned char *rose_dec_qsig_CallTransferComplete_ARG(struct pri *ctrl, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++const unsigned char *rose_dec_qsig_CallTransferUpdate_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++const unsigned char *rose_dec_qsig_SubaddressTransfer_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++ ++/* Q.SIG SS-MWI-Operations */ ++unsigned char *rose_enc_qsig_MWIActivate_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args); ++unsigned char *rose_enc_qsig_MWIDeactivate_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args); ++unsigned char *rose_enc_qsig_MWIInterrogate_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args); ++unsigned char *rose_enc_qsig_MWIInterrogate_RES(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_result_args *args); ++ ++const unsigned char *rose_dec_qsig_MWIActivate_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++const unsigned char *rose_dec_qsig_MWIDeactivate_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++const unsigned char *rose_dec_qsig_MWIInterrogate_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++const unsigned char *rose_dec_qsig_MWIInterrogate_RES(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, ++ union rose_msg_result_args *args); ++ ++/* Northern Telecom DMS-100 operations */ ++unsigned char *rose_enc_dms100_RLT_OperationInd_RES(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_result_args *args); ++unsigned char *rose_enc_dms100_RLT_ThirdParty_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args); ++ ++const unsigned char *rose_dec_dms100_RLT_OperationInd_RES(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, ++ union rose_msg_result_args *args); ++const unsigned char *rose_dec_dms100_RLT_ThirdParty_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++ ++/* National ISDN 2 (NI2) operations */ ++unsigned char *rose_enc_ni2_InformationFollowing_ARG(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, const union rose_msg_invoke_args *args); ++unsigned char *rose_enc_ni2_InitiateTransfer_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args); ++ ++const unsigned char *rose_dec_ni2_InformationFollowing_ARG(struct pri *ctrl, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++const unsigned char *rose_dec_ni2_InitiateTransfer_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++ ++ ++/* ------------------------------------------------------------------- */ ++ ++#ifdef __cplusplus ++} ++#endif ++ ++#endif /* _LIBPRI_ROSE_INTERNAL_H */ ++/* ------------------------------------------------------------------- */ ++/* end rose_internal.h */ + +Property changes on: rose_internal.h +___________________________________________________________________ +Added: svn:eol-style + + native +Added: svn:mime-type + + text/plain +Added: svn:keywords + + 'Author Date Id Revision' + +Index: pri_q921.h +=================================================================== +--- a/pri_q921.h (.../tags/1.4.10.2) (revision 1357) ++++ b/pri_q921.h (.../branches/1.4) (revision 1357) +@@ -192,4 +192,10 @@ + + extern int q921_transmit_iframe(struct pri *pri, void *buf, int len, int cr); + ++extern int q921_transmit_uiframe(struct pri *pri, void *buf, int len); ++ ++extern pri_event *q921_dchannel_up(struct pri *pri); ++ ++extern pri_event *q921_dchannel_down(struct pri *pri); ++ + #endif +Index: pri_facility.c +=================================================================== +--- a/pri_facility.c (.../tags/1.4.10.2) (revision 1357) ++++ b/pri_facility.c (.../branches/1.4) (revision 1357) +@@ -33,872 +33,1340 @@ + #include "pri_q921.h" + #include "pri_q931.h" + #include "pri_facility.h" ++#include "rose.h" + + #include <stdio.h> + #include <stdlib.h> + #include <string.h> + #include <limits.h> + +-static char *asn1id2text(int id) ++static short get_invokeid(struct pri *ctrl) + { +- static char data[32]; +- static char *strings[] = { +- "none", +- "Boolean", +- "Integer", +- "Bit String", +- "Octet String", +- "NULL", +- "Object Identifier", +- "Object Descriptor", +- "External Reference", +- "Real Number", +- "Enumerated", +- "Embedded PDV", +- "UTF-8 String", +- "Relative Object ID", +- "Reserved (0e)", +- "Reserved (0f)", +- "Sequence", +- "Set", +- "Numeric String", +- "Printable String", +- "Tele-Text String", +- "IA-5 String", +- "UTC Time", +- "Generalized Time", +- }; +- if (id > 0 && id <= 0x18) { +- return strings[id]; +- } else { +- sprintf(data, "Unknown (%02x)", id); +- return data; +- } ++ ctrl = PRI_MASTER(ctrl); ++ return ++ctrl->last_invoke; + } + +-static int asn1_dumprecursive(struct pri *pri, void *comp_ptr, int len, int level) ++static int redirectingreason_from_q931(struct pri *ctrl, int redirectingreason) + { +- unsigned char *vdata = (unsigned char *)comp_ptr; +- struct rose_component *comp; +- int i = 0; +- int j, k, l; +- int clen = 0; ++ int value; + +- while (len > 0) { +- GET_COMPONENT(comp, i, vdata, len); +- pri_message(pri, "%*s%02X %04X", 2 * level, "", comp->type, comp->len); +- if ((comp->type == 0) && (comp->len == 0)) +- return clen + 2; +- if ((comp->type & ASN1_PC_MASK) == ASN1_PRIMITIVE) { +- for (j = 0; j < comp->len; ++j) +- pri_message(pri, " %02X", comp->data[j]); ++ switch (ctrl->switchtype) { ++ case PRI_SWITCH_QSIG: ++ switch (redirectingreason) { ++ case PRI_REDIR_UNKNOWN: ++ value = QSIG_DIVERT_REASON_UNKNOWN; ++ break; ++ case PRI_REDIR_FORWARD_ON_BUSY: ++ value = QSIG_DIVERT_REASON_CFB; ++ break; ++ case PRI_REDIR_FORWARD_ON_NO_REPLY: ++ value = QSIG_DIVERT_REASON_CFNR; ++ break; ++ case PRI_REDIR_UNCONDITIONAL: ++ value = QSIG_DIVERT_REASON_CFU; ++ break; ++ case PRI_REDIR_DEFLECTION: ++ case PRI_REDIR_DTE_OUT_OF_ORDER: ++ case PRI_REDIR_FORWARDED_BY_DTE: ++ pri_message(ctrl, ++ "!! Don't know how to convert Q.931 redirection reason %d to Q.SIG\n", ++ redirectingreason); ++ /* Fall through */ ++ default: ++ value = QSIG_DIVERT_REASON_UNKNOWN; ++ break; + } +- if ((comp->type & ASN1_CLAN_MASK) == ASN1_UNIVERSAL) { +- switch (comp->type & ASN1_TYPE_MASK) { +- case 0: +- pri_message(pri, " (none)"); +- break; +- case ASN1_BOOLEAN: +- pri_message(pri, " (BOOLEAN: %d)", comp->data[0]); +- break; +- case ASN1_INTEGER: +- for (k = l = 0; k < comp->len; ++k) +- l = (l << 8) | comp->data[k]; +- pri_message(pri, " (INTEGER: %d)", l); +- break; +- case ASN1_BITSTRING: +- pri_message(pri, " (BITSTRING:"); +- for (k = 0; k < comp->len; ++k) +- pri_message(pri, " %02x", comp->data[k]); +- pri_message(pri, ")"); +- break; +- case ASN1_OCTETSTRING: +- pri_message(pri, " (OCTETSTRING:"); +- for (k = 0; k < comp->len; ++k) +- pri_message(pri, " %02x", comp->data[k]); +- pri_message(pri, ")"); +- break; +- case ASN1_NULL: +- pri_message(pri, " (NULL)"); +- break; +- case ASN1_OBJECTIDENTIFIER: +- pri_message(pri, " (OBJECTIDENTIFIER:"); +- for (k = 0; k < comp->len; ++k) +- pri_message(pri, " %02x", comp->data[k]); +- pri_message(pri, ")"); +- break; +- case ASN1_ENUMERATED: +- for (k = l = 0; k < comp->len; ++k) +- l = (l << 8) | comp->data[k]; +- pri_message(pri, " (ENUMERATED: %d)", l); +- break; +- case ASN1_SEQUENCE: +- pri_message(pri, " (SEQUENCE)"); +- break; +- default: +- pri_message(pri, " (component %02x - %s)", comp->type, asn1id2text(comp->type & ASN1_TYPE_MASK)); +- break; +- } ++ break; ++ default: ++ switch (redirectingreason) { ++ case PRI_REDIR_UNKNOWN: ++ value = Q952_DIVERT_REASON_UNKNOWN; ++ break; ++ case PRI_REDIR_FORWARD_ON_BUSY: ++ value = Q952_DIVERT_REASON_CFB; ++ break; ++ case PRI_REDIR_FORWARD_ON_NO_REPLY: ++ value = Q952_DIVERT_REASON_CFNR; ++ break; ++ case PRI_REDIR_DEFLECTION: ++ value = Q952_DIVERT_REASON_CD; ++ break; ++ case PRI_REDIR_UNCONDITIONAL: ++ value = Q952_DIVERT_REASON_CFU; ++ break; ++ case PRI_REDIR_DTE_OUT_OF_ORDER: ++ case PRI_REDIR_FORWARDED_BY_DTE: ++ pri_message(ctrl, ++ "!! Don't know how to convert Q.931 redirection reason %d to Q.952\n", ++ redirectingreason); ++ /* Fall through */ ++ default: ++ value = Q952_DIVERT_REASON_UNKNOWN; ++ break; + } +- else if ((comp->type & ASN1_CLAN_MASK) == ASN1_CONTEXT_SPECIFIC) { +- pri_message(pri, " (CONTEXT SPECIFIC [%d])", comp->type & ASN1_TYPE_MASK); ++ break; ++ } ++ ++ return value; ++} ++ ++static int redirectingreason_for_q931(struct pri *ctrl, int redirectingreason) ++{ ++ int value; ++ ++ switch (ctrl->switchtype) { ++ case PRI_SWITCH_QSIG: ++ switch (redirectingreason) { ++ case QSIG_DIVERT_REASON_UNKNOWN: ++ value = PRI_REDIR_UNKNOWN; ++ break; ++ case QSIG_DIVERT_REASON_CFU: ++ value = PRI_REDIR_UNCONDITIONAL; ++ break; ++ case QSIG_DIVERT_REASON_CFB: ++ value = PRI_REDIR_FORWARD_ON_BUSY; ++ break; ++ case QSIG_DIVERT_REASON_CFNR: ++ value = PRI_REDIR_FORWARD_ON_NO_REPLY; ++ break; ++ default: ++ pri_message(ctrl, "!! Unknown Q.SIG diversion reason %d\n", ++ redirectingreason); ++ value = PRI_REDIR_UNKNOWN; ++ break; + } +- else { +- pri_message(pri, " (component %02x)", comp->type); ++ break; ++ default: ++ switch (redirectingreason) { ++ case Q952_DIVERT_REASON_UNKNOWN: ++ value = PRI_REDIR_UNKNOWN; ++ break; ++ case Q952_DIVERT_REASON_CFU: ++ value = PRI_REDIR_UNCONDITIONAL; ++ break; ++ case Q952_DIVERT_REASON_CFB: ++ value = PRI_REDIR_FORWARD_ON_BUSY; ++ break; ++ case Q952_DIVERT_REASON_CFNR: ++ value = PRI_REDIR_FORWARD_ON_NO_REPLY; ++ break; ++ case Q952_DIVERT_REASON_CD: ++ value = PRI_REDIR_DEFLECTION; ++ break; ++ case Q952_DIVERT_REASON_IMMEDIATE: ++ pri_message(ctrl, ++ "!! Dont' know how to convert Q.952 diversion reason IMMEDIATE to PRI analog\n"); ++ value = PRI_REDIR_UNKNOWN; /* ??? */ ++ break; ++ default: ++ pri_message(ctrl, "!! Unknown Q.952 diversion reason %d\n", ++ redirectingreason); ++ value = PRI_REDIR_UNKNOWN; ++ break; + } +- pri_message(pri, "\n"); +- if ((comp->type & ASN1_PC_MASK) == ASN1_CONSTRUCTOR) +- j = asn1_dumprecursive(pri, comp->data, (comp->len ? comp->len : INT_MAX), level+1); +- else +- j = comp->len; +- j += 2; +- len -= j; +- vdata += j; +- clen += j; ++ break; + } +- return clen; ++ ++ return value; + } + +-int asn1_dump(struct pri *pri, void *comp, int len) ++/*! ++ * \brief Convert the Q.931 type-of-number field to facility. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param ton Q.931 ton/plan octet. ++ * ++ * \return PartyNumber enumeration value. ++ */ ++static int typeofnumber_from_q931(struct pri *ctrl, int ton) + { +- return asn1_dumprecursive(pri, comp, len, 0); ++ int value; ++ ++ switch ((ton >> 4) & 0x03) { ++ default: ++ pri_message(ctrl, "!! Unsupported Q.931 TypeOfNumber value (%d)\n", ton); ++ /* fall through */ ++ case PRI_TON_UNKNOWN: ++ value = Q932_TON_UNKNOWN; ++ break; ++ case PRI_TON_INTERNATIONAL: ++ value = Q932_TON_INTERNATIONAL; ++ break; ++ case PRI_TON_NATIONAL: ++ value = Q932_TON_NATIONAL; ++ break; ++ case PRI_TON_NET_SPECIFIC: ++ value = Q932_TON_NET_SPECIFIC; ++ break; ++ case PRI_TON_SUBSCRIBER: ++ value = Q932_TON_SUBSCRIBER; ++ break; ++ case PRI_TON_ABBREVIATED: ++ value = Q932_TON_ABBREVIATED; ++ break; ++ } ++ ++ return value; + } + +-static unsigned char get_invokeid(struct pri *pri) ++static int typeofnumber_for_q931(struct pri *ctrl, int ton) + { +- return ++pri->last_invoke; ++ int value; ++ ++ switch (ton) { ++ default: ++ pri_message(ctrl, "!! Invalid TypeOfNumber %d\n", ton); ++ /* fall through */ ++ case Q932_TON_UNKNOWN: ++ value = PRI_TON_UNKNOWN; ++ break; ++ case Q932_TON_INTERNATIONAL: ++ value = PRI_TON_INTERNATIONAL; ++ break; ++ case Q932_TON_NATIONAL: ++ value = PRI_TON_NATIONAL; ++ break; ++ case Q932_TON_NET_SPECIFIC: ++ value = PRI_TON_NET_SPECIFIC; ++ break; ++ case Q932_TON_SUBSCRIBER: ++ value = PRI_TON_SUBSCRIBER; ++ break; ++ case Q932_TON_ABBREVIATED: ++ value = PRI_TON_ABBREVIATED; ++ break; ++ } ++ ++ return value << 4; + } + +-struct addressingdataelements_presentednumberunscreened { +- char partyaddress[21]; +- char partysubaddress[21]; +- int npi; /* Numbering Plan Indicator */ +- int ton; /* Type Of Number */ +- int pres; /* Presentation */ +-}; ++/*! ++ * \internal ++ * \brief Convert the Q.931 numbering plan field to facility. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param plan Q.931 ton/plan octet. ++ * ++ * \return PartyNumber enumeration value. ++ */ ++static int numbering_plan_from_q931(struct pri *ctrl, int plan) ++{ ++ int value; + +-struct addressingdataelements_presentednumberscreened { +- char partyaddress[21]; +- char partysubaddress[21]; +- int npi; /* Numbering Plan Indicator */ +- int ton; /* Type Of Number */ +- int pres; /* Presentation */ +- int scrind; /* Screening Indicator */ +-}; ++ switch (plan & 0x0F) { ++ default: ++ pri_message(ctrl, "!! Unsupported Q.931 numbering plan value (%d)\n", plan); ++ /* fall through */ ++ case PRI_NPI_UNKNOWN: ++ value = 0; /* unknown */ ++ break; ++ case PRI_NPI_E163_E164: ++ value = 1; /* public */ ++ break; ++ case PRI_NPI_X121: ++ value = 3; /* data */ ++ break; ++ case PRI_NPI_F69: ++ value = 4; /* telex */ ++ break; ++ case PRI_NPI_NATIONAL: ++ value = 8; /* nationalStandard */ ++ break; ++ case PRI_NPI_PRIVATE: ++ value = 5; /* private */ ++ break; ++ } + +-#define PRI_CHECKOVERFLOW(size) \ +- if (msgptr - message + (size) >= sizeof(message)) { \ +- *msgptr = '\0'; \ +- pri_message(pri, "%s", message); \ +- msgptr = message; \ +- } ++ return value; ++} + +-static void dump_apdu(struct pri *pri, unsigned char *c, int len) ++/*! ++ * \internal ++ * \brief Convert the PartyNumber numbering plan to Q.931 plan field value. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param plan PartyNumber enumeration value. ++ * ++ * \return Q.931 plan field value. ++ */ ++static int numbering_plan_for_q931(struct pri *ctrl, int plan) + { +- #define MAX_APDU_LENGTH 255 +- static char hexs[16] = "0123456789ABCDEF"; +- int i; +- char message[(2 + MAX_APDU_LENGTH * 3 + 6 + MAX_APDU_LENGTH + 3)] = ""; /* please adjust here, if you make changes below! */ +- char *msgptr; +- +- msgptr = message; +- *msgptr++ = ' '; +- *msgptr++ = '['; +- for (i=0; i<len; i++) { +- PRI_CHECKOVERFLOW(3); +- *msgptr++ = ' '; +- *msgptr++ = hexs[(c[i] >> 4) & 0x0f]; +- *msgptr++ = hexs[(c[i]) & 0x0f]; ++ int value; ++ ++ switch (plan) { ++ default: ++ pri_message(ctrl, ++ "!! Unsupported PartyNumber to Q.931 numbering plan value (%d)\n", plan); ++ /* fall through */ ++ case 0: /* unknown */ ++ value = PRI_NPI_UNKNOWN; ++ break; ++ case 1: /* public */ ++ value = PRI_NPI_E163_E164; ++ break; ++ case 3: /* data */ ++ value = PRI_NPI_X121; ++ break; ++ case 4: /* telex */ ++ value = PRI_NPI_F69; ++ break; ++ case 5: /* private */ ++ value = PRI_NPI_PRIVATE; ++ break; ++ case 8: /* nationalStandard */ ++ value = PRI_NPI_NATIONAL; ++ break; + } +- PRI_CHECKOVERFLOW(6); +- strcpy(msgptr, " ] - ["); +- msgptr += strlen(msgptr); +- for (i=0; i<len; i++) { +- PRI_CHECKOVERFLOW(1); +- *msgptr++ = ((c[i] < ' ') || (c[i] > '~')) ? '.' : c[i]; ++ ++ return value; ++} ++ ++/*! ++ * \internal ++ * \brief Convert the Q.931 number presentation field to facility. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param presentation Q.931 presentation/screening octet. ++ * \param number_present Non-zero if the number is available. ++ * ++ * \return Presented<Number/Address><Screened/Unscreened> enumeration value. ++ */ ++static int presentation_from_q931(struct pri *ctrl, int presentation, int number_present) ++{ ++ int value; ++ ++ switch (presentation & PRI_PRES_RESTRICTION) { ++ case PRI_PRES_ALLOWED: ++ value = 0; /* presentationAllowed<Number/Address> */ ++ break; ++ default: ++ pri_message(ctrl, "!! Unsupported Q.931 number presentation value (%d)\n", ++ presentation); ++ /* fall through */ ++ case PRI_PRES_RESTRICTED: ++ if (number_present) { ++ value = 3; /* presentationRestricted<Number/Address> */ ++ } else { ++ value = 1; /* presentationRestricted */ ++ } ++ break; ++ case PRI_PRES_UNAVAILABLE: ++ value = 2; /* numberNotAvailableDueToInterworking */ ++ break; + } +- PRI_CHECKOVERFLOW(2); +- *msgptr++ = ']'; +- *msgptr++ = '\n'; +- *msgptr = '\0'; +- pri_message(pri, "%s", message); ++ ++ return value; + } +-#undef PRI_CHECKOVERFLOW + +-int redirectingreason_from_q931(struct pri *pri, int redirectingreason) ++/*! ++ * \internal ++ * \brief Convert the Presented<Number/Address><Screened/Unscreened> presentation ++ * to Q.931 presentation field value. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param presentation Presented<Number/Address><Screened/Unscreened> value. ++ * ++ * \return Q.931 presentation field value. ++ */ ++static int presentation_for_q931(struct pri *ctrl, int presentation) + { +- switch(pri->switchtype) { +- case PRI_SWITCH_QSIG: +- switch(redirectingreason) { +- case PRI_REDIR_UNKNOWN: +- return QSIG_DIVERT_REASON_UNKNOWN; +- case PRI_REDIR_FORWARD_ON_BUSY: +- return QSIG_DIVERT_REASON_CFB; +- case PRI_REDIR_FORWARD_ON_NO_REPLY: +- return QSIG_DIVERT_REASON_CFNR; +- case PRI_REDIR_UNCONDITIONAL: +- return QSIG_DIVERT_REASON_CFU; +- case PRI_REDIR_DEFLECTION: +- case PRI_REDIR_DTE_OUT_OF_ORDER: +- case PRI_REDIR_FORWARDED_BY_DTE: +- pri_message(pri, "!! Don't know how to convert Q.931 redirection reason %d to Q.SIG\n", redirectingreason); +- /* Fall through */ +- default: +- return QSIG_DIVERT_REASON_UNKNOWN; +- } +- default: +- switch(redirectingreason) { +- case PRI_REDIR_UNKNOWN: +- return Q952_DIVERT_REASON_UNKNOWN; +- case PRI_REDIR_FORWARD_ON_BUSY: +- return Q952_DIVERT_REASON_CFB; +- case PRI_REDIR_FORWARD_ON_NO_REPLY: +- return Q952_DIVERT_REASON_CFNR; +- case PRI_REDIR_DEFLECTION: +- return Q952_DIVERT_REASON_CD; +- case PRI_REDIR_UNCONDITIONAL: +- return Q952_DIVERT_REASON_CFU; +- case PRI_REDIR_DTE_OUT_OF_ORDER: +- case PRI_REDIR_FORWARDED_BY_DTE: +- pri_message(pri, "!! Don't know how to convert Q.931 redirection reason %d to Q.952\n", redirectingreason); +- /* Fall through */ +- default: +- return Q952_DIVERT_REASON_UNKNOWN; +- } ++ int value; ++ ++ switch (presentation) { ++ case 0: /* presentationAllowed<Number/Address> */ ++ value = PRI_PRES_ALLOWED; ++ break; ++ default: ++ pri_message(ctrl, ++ "!! Unsupported Presented<Number/Address><Screened/Unscreened> to Q.931 value (%d)\n", ++ presentation); ++ /* fall through */ ++ case 1: /* presentationRestricted */ ++ case 3: /* presentationRestricted<Number/Address> */ ++ value = PRI_PRES_RESTRICTED; ++ break; ++ case 2: /* numberNotAvailableDueToInterworking */ ++ value = PRI_PRES_UNAVAILABLE; ++ break; + } ++ ++ return value; + } + +-static int redirectingreason_for_q931(struct pri *pri, int redirectingreason) ++/*! ++ * \internal ++ * \brief Convert the Q.931 number presentation field to Q.SIG name presentation. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param presentation Q.931 presentation/screening octet. ++ * \param name_present Non-zero if the name is available. ++ * ++ * \return Name presentation enumeration value. ++ */ ++static int qsig_name_presentation_from_q931(struct pri *ctrl, int presentation, int name_present) + { +- switch(pri->switchtype) { +- case PRI_SWITCH_QSIG: +- switch(redirectingreason) { +- case QSIG_DIVERT_REASON_UNKNOWN: +- return PRI_REDIR_UNKNOWN; +- case QSIG_DIVERT_REASON_CFU: +- return PRI_REDIR_UNCONDITIONAL; +- case QSIG_DIVERT_REASON_CFB: +- return PRI_REDIR_FORWARD_ON_BUSY; +- case QSIG_DIVERT_REASON_CFNR: +- return PRI_REDIR_FORWARD_ON_NO_REPLY; +- default: +- pri_message(pri, "!! Unknown Q.SIG diversion reason %d\n", redirectingreason); +- return PRI_REDIR_UNKNOWN; +- } +- default: +- switch(redirectingreason) { +- case Q952_DIVERT_REASON_UNKNOWN: +- return PRI_REDIR_UNKNOWN; +- case Q952_DIVERT_REASON_CFU: +- return PRI_REDIR_UNCONDITIONAL; +- case Q952_DIVERT_REASON_CFB: +- return PRI_REDIR_FORWARD_ON_BUSY; +- case Q952_DIVERT_REASON_CFNR: +- return PRI_REDIR_FORWARD_ON_NO_REPLY; +- case Q952_DIVERT_REASON_CD: +- return PRI_REDIR_DEFLECTION; +- case Q952_DIVERT_REASON_IMMEDIATE: +- pri_message(pri, "!! Dont' know how to convert Q.952 diversion reason IMMEDIATE to PRI analog\n"); +- return PRI_REDIR_UNKNOWN; /* ??? */ +- default: +- pri_message(pri, "!! Unknown Q.952 diversion reason %d\n", redirectingreason); +- return PRI_REDIR_UNKNOWN; +- } ++ int value; ++ ++ switch (presentation & PRI_PRES_RESTRICTION) { ++ case PRI_PRES_ALLOWED: ++ if (name_present) { ++ value = 1; /* presentation_allowed */ ++ } else { ++ value = 4; /* name_not_available */ ++ } ++ break; ++ default: ++ pri_message(ctrl, "!! Unsupported Q.931 number presentation value (%d)\n", ++ presentation); ++ /* fall through */ ++ case PRI_PRES_RESTRICTED: ++ if (name_present) { ++ value = 2; /* presentation_restricted */ ++ } else { ++ value = 3; /* presentation_restricted_null */ ++ } ++ break; ++ case PRI_PRES_UNAVAILABLE: ++ value = 4; /* name_not_available */ ++ break; + } ++ ++ return value; + } + +-int typeofnumber_from_q931(struct pri *pri, int ton) ++/*! ++ * \internal ++ * \brief Convert the Q.SIG name presentation to Q.931 presentation field value. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param presentation Q.SIG name presentation value. ++ * ++ * \return Q.931 presentation field value. ++ */ ++static int qsig_name_presentation_for_q931(struct pri *ctrl, int presentation) + { +- switch(ton) { +- case PRI_TON_INTERNATIONAL: +- return Q932_TON_INTERNATIONAL; +- case PRI_TON_NATIONAL: +- return Q932_TON_NATIONAL; +- case PRI_TON_NET_SPECIFIC: +- return Q932_TON_NET_SPECIFIC; +- case PRI_TON_SUBSCRIBER: +- return Q932_TON_SUBSCRIBER; +- case PRI_TON_ABBREVIATED: +- return Q932_TON_ABBREVIATED; +- case PRI_TON_RESERVED: +- default: +- pri_message(pri, "!! Unsupported Q.931 TypeOfNumber value (%d)\n", ton); +- /* fall through */ +- case PRI_TON_UNKNOWN: +- return Q932_TON_UNKNOWN; ++ int value; ++ ++ switch (presentation) { ++ case 1: /* presentation_allowed */ ++ value = PRI_PRES_ALLOWED; ++ break; ++ default: ++ pri_message(ctrl, ++ "!! Unsupported Q.SIG name presentation to Q.931 value (%d)\n", ++ presentation); ++ /* fall through */ ++ case 2: /* presentation_restricted */ ++ case 3: /* presentation_restricted_null */ ++ value = PRI_PRES_RESTRICTED; ++ break; ++ case 0: /* optional_name_not_present */ ++ case 4: /* name_not_available */ ++ value = PRI_PRES_UNAVAILABLE; ++ break; + } ++ ++ return value; + } + +-static int typeofnumber_for_q931(struct pri *pri, int ton) ++/*! ++ * \internal ++ * \brief Convert number presentation to Q.SIG diversion subscription notification. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param presentation Number presentation value. ++ * ++ * \return Q.SIG diversion subscription notification value. ++ */ ++static int presentation_to_subscription(struct pri *ctrl, int presentation) + { +- switch (ton) { +- case Q932_TON_UNKNOWN: +- return PRI_TON_UNKNOWN; +- case Q932_TON_INTERNATIONAL: +- return PRI_TON_INTERNATIONAL; +- case Q932_TON_NATIONAL: +- return PRI_TON_NATIONAL; +- case Q932_TON_NET_SPECIFIC: +- return PRI_TON_NET_SPECIFIC; +- case Q932_TON_SUBSCRIBER: +- return PRI_TON_SUBSCRIBER; +- case Q932_TON_ABBREVIATED: +- return PRI_TON_ABBREVIATED; +- default: +- pri_message(pri, "!! Invalid Q.932 TypeOfNumber %d\n", ton); +- return PRI_TON_UNKNOWN; ++ /* derive subscription value from presentation value */ ++ ++ switch (presentation & PRI_PRES_RESTRICTION) { ++ case PRI_PRES_ALLOWED: ++ return QSIG_NOTIFICATION_WITH_DIVERTED_TO_NR; ++ case PRI_PRES_RESTRICTED: ++ return QSIG_NOTIFICATION_WITHOUT_DIVERTED_TO_NR; ++ case PRI_PRES_UNAVAILABLE: /* Number not available due to interworking */ ++ return QSIG_NOTIFICATION_WITHOUT_DIVERTED_TO_NR; /* ?? QSIG_NO_NOTIFICATION */ ++ default: ++ pri_message(ctrl, "!! Unknown Q.SIG presentationIndicator 0x%02x\n", ++ presentation); ++ return QSIG_NOTIFICATION_WITHOUT_DIVERTED_TO_NR; + } + } + +-int asn1_name_decode(void * data, int len, char *namebuf, int buflen) ++/*! ++ * \internal ++ * \brief Copy the given rose party number to the q931_party_number ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param q931_number Q.931 party number structure ++ * \param rose_number ROSE party number structure ++ * ++ * \note It is assumed that the q931_number has been initialized before calling. ++ * ++ * \return Nothing ++ */ ++static void rose_copy_number_to_q931(struct pri *ctrl, ++ struct q931_party_number *q931_number, const struct rosePartyNumber *rose_number) + { +- struct rose_component *comp = (struct rose_component*)data; +- int datalen = 0, res = 0; ++ //q931_party_number_init(q931_number); ++ libpri_copy_string(q931_number->str, (char *) rose_number->str, ++ sizeof(q931_number->str)); ++ q931_number->plan = numbering_plan_for_q931(ctrl, rose_number->plan) ++ | typeofnumber_for_q931(ctrl, rose_number->ton); ++ q931_number->valid = 1; ++} + +- if (comp->len == ASN1_LEN_INDEF) { +- datalen = strlen((char *)comp->data); +- res = datalen + 2; +- } else +- datalen = res = comp->len; ++/*! ++ * \internal ++ * \brief Copy the given rose subaddress to the q931_party_subaddress. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param q931_subaddress Q.931 party subaddress structure ++ * \param rose_subaddress ROSE subaddress structure ++ * ++ * \note It is assumed that the q931_subaddress has been initialized before calling. ++ * ++ * \return Nothing ++ */ ++static void rose_copy_subaddress_to_q931(struct pri *ctrl, ++ struct q931_party_subaddress *q931_subaddress, ++ const struct rosePartySubaddress *rose_subaddress) ++{ ++ //q931_party_subaddress_init(q931_subaddress); ++ if (!rose_subaddress->length) { ++ /* Subaddress is not present. */ ++ return; ++ } + +- if (datalen > buflen) { +- /* Truncate */ +- datalen = buflen; ++ switch (rose_subaddress->type) { ++ case 0:/* UserSpecified */ ++ q931_subaddress->type = 2;/* user_specified */ ++ q931_subaddress->valid = 1; ++ q931_subaddress->length = rose_subaddress->length; ++ if (sizeof(q931_subaddress->data) <= q931_subaddress->length) { ++ q931_subaddress->length = sizeof(q931_subaddress->data) - 1; ++ } ++ memcpy(q931_subaddress->data, rose_subaddress->u.user_specified.information, ++ q931_subaddress->length); ++ q931_subaddress->data[q931_subaddress->length] = '\0'; ++ if (rose_subaddress->u.user_specified.odd_count_present) { ++ q931_subaddress->odd_even_indicator = ++ rose_subaddress->u.user_specified.odd_count; ++ } ++ break; ++ case 1:/* NSAP */ ++ q931_subaddress->type = 0;/* nsap */ ++ q931_subaddress->valid = 1; ++ libpri_copy_string((char *) q931_subaddress->data, ++ (char *) rose_subaddress->u.nsap, sizeof(q931_subaddress->data)); ++ q931_subaddress->length = strlen((char *) q931_subaddress->data); ++ break; ++ default: ++ /* Don't know how to encode so assume it is not present. */ ++ break; + } +- memcpy(namebuf, comp->data, datalen); +- return res + 2; + } + +-int asn1_string_encode(unsigned char asn1_type, void *data, int len, int max_len, void *src, int src_len) ++/*! ++ * \internal ++ * \brief Copy the given rose address to the q931_party_id address. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param q931_address Q.931 party id structure to fill address ++ * \param rose_address ROSE address structure ++ * ++ * \note It is assumed that the q931_address has been initialized before calling. ++ * ++ * \return Nothing ++ */ ++static void rose_copy_address_to_q931(struct pri *ctrl, ++ struct q931_party_id *q931_address, const struct roseAddress *rose_address) + { +- struct rose_component *comp = NULL; +- +- if (len < 2 + src_len) +- return -1; ++ rose_copy_number_to_q931(ctrl, &q931_address->number, &rose_address->number); ++ rose_copy_subaddress_to_q931(ctrl, &q931_address->subaddress, ++ &rose_address->subaddress); ++} + +- if (max_len && (src_len > max_len)) +- src_len = max_len; ++/*! ++ * \internal ++ * \brief Copy the given rose presented screened party number to the q931_party_number ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param q931_number Q.931 party number structure ++ * \param rose_presented ROSE presented screened party number structure ++ * ++ * \return Nothing ++ */ ++static void rose_copy_presented_number_screened_to_q931(struct pri *ctrl, ++ struct q931_party_number *q931_number, ++ const struct rosePresentedNumberScreened *rose_presented) ++{ ++ q931_party_number_init(q931_number); ++ q931_number->valid = 1; ++ q931_number->presentation = presentation_for_q931(ctrl, rose_presented->presentation); ++ switch (rose_presented->presentation) { ++ case 0: /* presentationAllowedNumber */ ++ case 3: /* presentationRestrictedNumber */ ++ q931_number->presentation |= ++ (rose_presented->screened.screening_indicator & PRI_PRES_NUMBER_TYPE); ++ rose_copy_number_to_q931(ctrl, q931_number, ++ &rose_presented->screened.number); ++ break; ++ default: ++ q931_number->presentation |= PRI_PRES_USER_NUMBER_UNSCREENED; ++ break; ++ } ++} + +- comp = (struct rose_component *)data; +- comp->type = asn1_type; +- comp->len = src_len; +- memcpy(comp->data, src, src_len); +- +- return 2 + src_len; ++/*! ++ * \internal ++ * \brief Copy the given rose presented unscreened party number to the q931_party_number ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param q931_number Q.931 party number structure ++ * \param rose_presented ROSE presented unscreened party number structure ++ * ++ * \return Nothing ++ */ ++static void rose_copy_presented_number_unscreened_to_q931(struct pri *ctrl, ++ struct q931_party_number *q931_number, ++ const struct rosePresentedNumberUnscreened *rose_presented) ++{ ++ q931_party_number_init(q931_number); ++ q931_number->valid = 1; ++ q931_number->presentation = presentation_for_q931(ctrl, ++ rose_presented->presentation) | PRI_PRES_USER_NUMBER_UNSCREENED; ++ switch (rose_presented->presentation) { ++ case 0: /* presentationAllowedNumber */ ++ case 3: /* presentationRestrictedNumber */ ++ rose_copy_number_to_q931(ctrl, q931_number, &rose_presented->number); ++ break; ++ default: ++ break; ++ } + } + +-int asn1_copy_string(char * buf, int buflen, struct rose_component *comp) ++/*! ++ * \internal ++ * \brief Copy the given rose presented screened party address to the q931_party_number ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param q931_address Q.931 party id structure to fill the address ++ * \param rose_presented ROSE presented screened party address structure ++ * ++ * \return Nothing ++ */ ++static void rose_copy_presented_address_screened_to_q931(struct pri *ctrl, ++ struct q931_party_id *q931_address, ++ const struct rosePresentedAddressScreened *rose_presented) + { +- int res; +- int datalen; ++ q931_party_number_init(&q931_address->number); ++ q931_party_subaddress_init(&q931_address->subaddress); ++ q931_address->number.valid = 1; ++ q931_address->number.presentation = presentation_for_q931(ctrl, ++ rose_presented->presentation); ++ switch (rose_presented->presentation) { ++ case 0: /* presentationAllowedAddress */ ++ case 3: /* presentationRestrictedAddress */ ++ q931_address->number.presentation |= ++ (rose_presented->screened.screening_indicator & PRI_PRES_NUMBER_TYPE); ++ rose_copy_number_to_q931(ctrl, &q931_address->number, ++ &rose_presented->screened.number); ++ rose_copy_subaddress_to_q931(ctrl, &q931_address->subaddress, ++ &rose_presented->screened.subaddress); ++ break; ++ default: ++ q931_address->number.presentation |= PRI_PRES_USER_NUMBER_UNSCREENED; ++ break; ++ } ++} + +- if ((comp->len > buflen) && (comp->len != ASN1_LEN_INDEF)) +- return -1; ++/*! ++ * \internal ++ * \brief Copy the given rose party name to the q931_party_name ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param qsig_name Q.SIG party name structure ++ * \param rose_name Q.SIG ROSE party name structure ++ * ++ * \return Nothing ++ */ ++static void rose_copy_name_to_q931(struct pri *ctrl, ++ struct q931_party_name *qsig_name, const struct roseQsigName *rose_name) ++{ ++ //q931_party_name_init(qsig_name); ++ qsig_name->valid = 1; ++ qsig_name->presentation = qsig_name_presentation_for_q931(ctrl, ++ rose_name->presentation); ++ qsig_name->char_set = rose_name->char_set; ++ libpri_copy_string(qsig_name->str, (char *) rose_name->data, sizeof(qsig_name->str)); ++} + +- if (comp->len == ASN1_LEN_INDEF) { +- datalen = strlen((char*)comp->data); +- res = datalen + 2; +- } else +- res = datalen = comp->len; ++/*! ++ * \internal ++ * \brief Copy the given q931_party_number to the rose party number ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param rose_number ROSE party number structure ++ * \param q931_number Q.931 party number structure ++ * ++ * \return Nothing ++ */ ++static void q931_copy_number_to_rose(struct pri *ctrl, ++ struct rosePartyNumber *rose_number, const struct q931_party_number *q931_number) ++{ ++ rose_number->plan = numbering_plan_from_q931(ctrl, q931_number->plan); ++ rose_number->ton = typeofnumber_from_q931(ctrl, q931_number->plan); ++ /* Truncate the q931_number->str if necessary. */ ++ libpri_copy_string((char *) rose_number->str, q931_number->str, ++ sizeof(rose_number->str)); ++ rose_number->length = strlen((char *) rose_number->str); ++} + +- memcpy(buf, comp->data, datalen); +- buf[datalen] = 0; ++/*! ++ * \internal ++ * \brief Copy the given q931_party_subaddress to the rose subaddress. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param rose_subaddress ROSE subaddress structure ++ * \param q931_subaddress Q.931 party subaddress structure ++ * ++ * \return Nothing ++ */ ++static void q931_copy_subaddress_to_rose(struct pri *ctrl, ++ struct rosePartySubaddress *rose_subaddress, ++ const struct q931_party_subaddress *q931_subaddress) ++{ ++ if (!q931_subaddress->valid) { ++ /* Subaddress is not present. */ ++ rose_subaddress->length = 0; ++ return; ++ } + +- return res; ++ switch (q931_subaddress->type) { ++ case 0: /* NSAP */ ++ rose_subaddress->type = 1;/* NSAP */ ++ libpri_copy_string((char *) rose_subaddress->u.nsap, ++ (char *) q931_subaddress->data, sizeof(rose_subaddress->u.nsap)); ++ rose_subaddress->length = strlen((char *) rose_subaddress->u.nsap); ++ break; ++ case 2: /* user_specified */ ++ rose_subaddress->type = 0;/* UserSpecified */ ++ rose_subaddress->length = q931_subaddress->length; ++ if (sizeof(rose_subaddress->u.user_specified.information) ++ <= rose_subaddress->length) { ++ rose_subaddress->length = ++ sizeof(rose_subaddress->u.user_specified.information) - 1; ++ } else { ++ if (q931_subaddress->odd_even_indicator) { ++ rose_subaddress->u.user_specified.odd_count_present = 1; ++ rose_subaddress->u.user_specified.odd_count = 1; ++ } ++ } ++ memcpy(rose_subaddress->u.user_specified.information, q931_subaddress->data, ++ rose_subaddress->length); ++ rose_subaddress->u.user_specified.information[rose_subaddress->length] = '\0'; ++ break; ++ default: ++ /* Don't know how to encode so assume it is not present. */ ++ rose_subaddress->length = 0; ++ break; ++ } + } + +-static int rose_number_digits_decode(struct pri *pri, q931_call *call, unsigned char *data, int len, struct addressingdataelements_presentednumberunscreened *value) ++/*! ++ * \internal ++ * \brief Copy the given q931_party_id address to the rose address. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param rose_address ROSE address structure ++ * \param q931_address Q.931 party id structure to give address ++ * ++ * \return Nothing ++ */ ++static void q931_copy_address_to_rose(struct pri *ctrl, struct roseAddress *rose_address, ++ const struct q931_party_id *q931_address) + { +- int i = 0; +- struct rose_component *comp = NULL; +- unsigned char *vdata = data; +- int datalen = 0; +- int res = 0; ++ q931_copy_number_to_rose(ctrl, &rose_address->number, &q931_address->number); ++ q931_copy_subaddress_to_rose(ctrl, &rose_address->subaddress, ++ &q931_address->subaddress); ++} + +- do { +- GET_COMPONENT(comp, i, vdata, len); +- CHECK_COMPONENT(comp, ASN1_NUMERICSTRING, "Don't know what to do with PublicPartyNumber ROSE component type 0x%x\n"); +- if(comp->len > 20 && comp->len != ASN1_LEN_INDEF) { +- pri_message(pri, "!! Oversized NumberDigits component (%d)\n", comp->len); +- return -1; +- } +- if (comp->len == ASN1_LEN_INDEF) { +- datalen = strlen((char *)comp->data); +- res = datalen + 2; +- } else +- res = datalen = comp->len; +- +- memcpy(value->partyaddress, comp->data, datalen); +- value->partyaddress[datalen] = '\0'; ++/*! ++ * \internal ++ * \brief Copy the given q931_party_number to the rose presented screened party number ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param rose_presented ROSE presented screened party number structure ++ * \param q931_number Q.931 party number structure ++ * ++ * \return Nothing ++ */ ++static void q931_copy_presented_number_screened_to_rose(struct pri *ctrl, ++ struct rosePresentedNumberScreened *rose_presented, ++ const struct q931_party_number *q931_number) ++{ ++ if (q931_number->valid) { ++ rose_presented->presentation = ++ presentation_from_q931(ctrl, q931_number->presentation, q931_number->str[0]); ++ rose_presented->screened.screening_indicator = ++ q931_number->presentation & PRI_PRES_NUMBER_TYPE; ++ q931_copy_number_to_rose(ctrl, &rose_presented->screened.number, q931_number); ++ } else { ++ rose_presented->presentation = 2;/* numberNotAvailableDueToInterworking */ ++ } ++} + +- return res + 2; ++/*! ++ * \internal ++ * \brief Copy the given q931_party_number to the rose presented unscreened party number ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param rose_presented ROSE presented unscreened party number structure ++ * \param q931_number Q.931 party number structure ++ * ++ * \return Nothing ++ */ ++static void q931_copy_presented_number_unscreened_to_rose(struct pri *ctrl, ++ struct rosePresentedNumberUnscreened *rose_presented, ++ const struct q931_party_number *q931_number) ++{ ++ if (q931_number->valid) { ++ rose_presented->presentation = ++ presentation_from_q931(ctrl, q931_number->presentation, q931_number->str[0]); ++ q931_copy_number_to_rose(ctrl, &rose_presented->number, q931_number); ++ } else { ++ rose_presented->presentation = 2;/* numberNotAvailableDueToInterworking */ + } +- while(0); +- +- return -1; + } + +-static int rose_public_party_number_decode(struct pri *pri, q931_call *call, unsigned char *data, int len, struct addressingdataelements_presentednumberunscreened *value) ++#if 0 /* In case it is needed in the future */ ++/*! ++ * \internal ++ * \brief Copy the given q931_party_number to the rose presented screened party address ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param rose_presented ROSE presented screened party address structure ++ * \param q931_address Q.931 party id structure to get the address ++ * ++ * \return Nothing ++ */ ++static void q931_copy_presented_address_screened_to_rose(struct pri *ctrl, ++ struct rosePresentedAddressScreened *rose_presented, ++ const struct q931_party_id *q931_address) + { +- int i = 0; +- struct rose_component *comp = NULL; +- unsigned char *vdata = data; +- int ton; +- int res = 0; ++ if (q931_address->number.valid) { ++ rose_presented->presentation = ++ presentation_from_q931(ctrl, q931_address->number.presentation, ++ q931_address->number.str[0]); ++ rose_presented->screened.screening_indicator = ++ q931_address->number.presentation & PRI_PRES_NUMBER_TYPE; ++ q931_copy_number_to_rose(ctrl, &rose_presented->screened.number, ++ &q931_address->number); ++ q931_copy_subaddress_to_rose(ctrl, &rose_presented->screened.subaddress, ++ &q931_address->subaddress); ++ } else { ++ rose_presented->presentation = 2;/* numberNotAvailableDueToInterworking */ ++ } ++} ++#endif /* In case it is needed in the future */ + +- if (len < 2) +- return -1; ++/*! ++ * \internal ++ * \brief Copy the given q931_party_name to the rose party name ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param rose_name Q.SIG ROSE party name structure ++ * \param qsig_name Q.SIG party name structure ++ * ++ * \return Nothing ++ */ ++static void q931_copy_name_to_rose(struct pri *ctrl, ++ struct roseQsigName *rose_name, const struct q931_party_name *qsig_name) ++{ ++ if (qsig_name->valid) { ++ rose_name->presentation = qsig_name_presentation_from_q931(ctrl, ++ qsig_name->presentation, qsig_name->str[0]); ++ rose_name->char_set = qsig_name->char_set; ++ /* Truncate the qsig_name->str if necessary. */ ++ libpri_copy_string((char *) rose_name->data, qsig_name->str, sizeof(rose_name->data)); ++ rose_name->length = strlen((char *) rose_name->data); ++ } else { ++ rose_name->presentation = 4;/* name_not_available */ ++ } ++} + +- do { +- GET_COMPONENT(comp, i, vdata, len); +- CHECK_COMPONENT(comp, ASN1_ENUMERATED, "Don't know what to do with PublicPartyNumber ROSE component type 0x%x\n"); +- ASN1_GET_INTEGER(comp, ton); +- NEXT_COMPONENT(comp, i); +- ton = typeofnumber_for_q931(pri, ton); ++/*! ++ * \internal ++ * \brief Encode the Q.SIG DivertingLegInformation1 invoke message. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode the facility ie contents. ++ * \param end End of facility ie contents encoding data buffer. ++ * \param call Call leg from which to encode diversion leg 1. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *enc_qsig_diverting_leg_information1(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, q931_call *call) ++{ ++ struct fac_extension_header header; ++ struct rose_msg_invoke msg; + +- res = rose_number_digits_decode(pri, call, &vdata[i], len-i, value); +- if (res < 0) +- return -1; +- value->ton = ton; ++ memset(&header, 0, sizeof(header)); ++ header.nfe_present = 1; ++ header.nfe.source_entity = 0; /* endPINX */ ++ header.nfe.destination_entity = 0; /* endPINX */ ++ header.interpretation_present = 1; ++ header.interpretation = 0; /* discardAnyUnrecognisedInvokePdu */ ++ pos = facility_encode_header(ctrl, pos, end, &header); ++ if (!pos) { ++ return NULL; ++ } + +- return res + 3; ++ memset(&msg, 0, sizeof(msg)); ++ msg.operation = ROSE_QSIG_DivertingLegInformation1; ++ msg.invoke_id = get_invokeid(ctrl); ++ msg.args.qsig.DivertingLegInformation1.diversion_reason = ++ redirectingreason_from_q931(ctrl, call->redirecting.reason); + +- } while(0); +- return -1; ++ /* subscriptionOption is the redirecting.to.number.presentation */ ++ msg.args.qsig.DivertingLegInformation1.subscription_option = ++ presentation_to_subscription(ctrl, call->redirecting.to.number.presentation); ++ ++ /* nominatedNr is the redirecting.to.number */ ++ q931_copy_number_to_rose(ctrl, ++ &msg.args.qsig.DivertingLegInformation1.nominated_number, ++ &call->redirecting.to.number); ++ ++ pos = rose_encode_invoke(ctrl, pos, end, &msg); ++ ++ return pos; + } + +-static int rose_private_party_number_decode(struct pri *pri, q931_call *call, unsigned char *data, int len, struct addressingdataelements_presentednumberunscreened *value) ++/*! ++ * \internal ++ * \brief Encode the ETSI DivertingLegInformation1 invoke message. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode the facility ie contents. ++ * \param end End of facility ie contents encoding data buffer. ++ * \param call Call leg from which to encode diversion leg 1. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *enc_etsi_diverting_leg_information1(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, q931_call *call) + { +- int i = 0; +- struct rose_component *comp = NULL; +- unsigned char *vdata = data; +- int ton; +- int res = 0; ++ struct rose_msg_invoke msg; + +- if (len < 2) +- return -1; ++ pos = facility_encode_header(ctrl, pos, end, NULL); ++ if (!pos) { ++ return NULL; ++ } + +- do { +- GET_COMPONENT(comp, i, vdata, len); +- CHECK_COMPONENT(comp, ASN1_ENUMERATED, "Don't know what to do with PrivatePartyNumber ROSE component type 0x%x\n"); +- ASN1_GET_INTEGER(comp, ton); +- NEXT_COMPONENT(comp, i); +- ton = typeofnumber_for_q931(pri, ton); ++ memset(&msg, 0, sizeof(msg)); ++ msg.operation = ROSE_ETSI_DivertingLegInformation1; ++ msg.invoke_id = get_invokeid(ctrl); ++ msg.args.etsi.DivertingLegInformation1.diversion_reason = ++ redirectingreason_from_q931(ctrl, call->redirecting.reason); + +- res = rose_number_digits_decode(pri, call, &vdata[i], len-i, value); +- if (res < 0) +- return -1; +- value->ton = ton; ++ if (call->redirecting.to.number.valid) { ++ msg.args.etsi.DivertingLegInformation1.subscription_option = 2; + +- return res + 3; ++ /* divertedToNumber is the redirecting.to.number */ ++ msg.args.etsi.DivertingLegInformation1.diverted_to_present = 1; ++ q931_copy_presented_number_unscreened_to_rose(ctrl, ++ &msg.args.etsi.DivertingLegInformation1.diverted_to, ++ &call->redirecting.to.number); ++ } else { ++ msg.args.etsi.DivertingLegInformation1.subscription_option = 1; ++ } ++ pos = rose_encode_invoke(ctrl, pos, end, &msg); + +- } while(0); +- return -1; ++ return pos; + } + +-static int rose_address_decode(struct pri *pri, q931_call *call, unsigned char *data, int len, struct addressingdataelements_presentednumberunscreened *value) ++/*! ++ * \brief Encode and queue the DivertingLegInformation1 invoke message. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param call Call leg from which to encode diversion leg 1. ++ * ++ * \retval 0 on success. ++ * \retval -1 on error. ++ */ ++int rose_diverting_leg_information1_encode(struct pri *ctrl, q931_call *call) + { +- int i = 0; +- struct rose_component *comp = NULL; +- unsigned char *vdata = data; +- int res = 0; ++ unsigned char buffer[256]; ++ unsigned char *end; + +- do { +- GET_COMPONENT(comp, i, vdata, len); ++ switch (ctrl->switchtype) { ++ case PRI_SWITCH_EUROISDN_E1: ++ case PRI_SWITCH_EUROISDN_T1: ++ end = enc_etsi_diverting_leg_information1(ctrl, buffer, buffer + sizeof(buffer), ++ call); ++ break; ++ case PRI_SWITCH_QSIG: ++ end = enc_qsig_diverting_leg_information1(ctrl, buffer, buffer + sizeof(buffer), ++ call); ++ break; ++ default: ++ return -1; ++ } ++ if (!end) { ++ return -1; ++ } + +- switch(comp->type) { +- case (ASN1_CONTEXT_SPECIFIC | ASN1_CONSTRUCTOR | ASN1_TAG_0): /* [0] unknownPartyNumber */ +- res = rose_number_digits_decode(pri, call, comp->data, comp->len, value); +- if (res < 0) +- return -1; +- value->npi = PRI_NPI_UNKNOWN; +- value->ton = PRI_TON_UNKNOWN; +- break; +- case (ASN1_CONTEXT_SPECIFIC | ASN1_TAG_0): /* [0] unknownPartyNumber */ +- res = asn1_copy_string(value->partyaddress, sizeof(value->partyaddress), comp); +- if (res < 0) +- return -1; +- value->npi = PRI_NPI_UNKNOWN; +- value->ton = PRI_TON_UNKNOWN; +- break; +- case (ASN1_CONTEXT_SPECIFIC | ASN1_CONSTRUCTOR | ASN1_TAG_1): /* [1] publicPartyNumber */ +- res = rose_public_party_number_decode(pri, call, comp->data, comp->len, value); +- if (res < 0) +- return -1; +- value->npi = PRI_NPI_E163_E164; +- break; +- case (ASN1_CONTEXT_SPECIFIC | ASN1_CONSTRUCTOR | ASN1_TAG_2): /* [2] nsapEncodedNumber */ +- pri_message(pri, "!! NsapEncodedNumber isn't handled\n"); +- return -1; +- case (ASN1_CONTEXT_SPECIFIC | ASN1_CONSTRUCTOR | ASN1_TAG_3): /* [3] dataPartyNumber */ +- if(rose_number_digits_decode(pri, call, comp->data, comp->len, value)) +- return -1; +- value->npi = PRI_NPI_X121 /* ??? */; +- value->ton = PRI_TON_UNKNOWN /* ??? */; +- pri_message(pri, "!! dataPartyNumber isn't handled\n"); +- return -1; +- case (ASN1_CONTEXT_SPECIFIC | ASN1_CONSTRUCTOR | ASN1_TAG_4): /* [4] telexPartyNumber */ +- res = rose_number_digits_decode(pri, call, comp->data, comp->len, value); +- if (res < 0) +- return -1; +- value->npi = PRI_NPI_F69 /* ??? */; +- value->ton = PRI_TON_UNKNOWN /* ??? */; +- pri_message(pri, "!! telexPartyNumber isn't handled\n"); +- return -1; +- case (ASN1_CONTEXT_SPECIFIC | ASN1_CONSTRUCTOR | ASN1_TAG_5): /* [5] priavePartyNumber */ +- res = rose_private_party_number_decode(pri, call, comp->data, comp->len, value); +- if (res < 0) +- return -1; +- value->npi = PRI_NPI_PRIVATE; +- break; +- case (ASN1_CONTEXT_SPECIFIC | ASN1_CONSTRUCTOR | ASN1_TAG_8): /* [8] nationalStandardPartyNumber */ +- res = rose_number_digits_decode(pri, call, comp->data, comp->len, value); +- if (res < 0) +- return -1; +- value->npi = PRI_NPI_NATIONAL; +- value->ton = PRI_TON_NATIONAL; +- break; +- default: +- pri_message(pri, "!! Unknown Party number component received 0x%X\n", comp->type); +- return -1; ++ return pri_call_apdu_queue(call, Q931_FACILITY, buffer, end - buffer, NULL); ++} ++ ++/*! ++ * \internal ++ * \brief Encode the Q.SIG DivertingLegInformation2 invoke message. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode the facility ie contents. ++ * \param end End of facility ie contents encoding data buffer. ++ * \param call Call leg from which to encode diversion leg 2. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *enc_qsig_diverting_leg_information2(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, q931_call *call) ++{ ++ struct fac_extension_header header; ++ struct rose_msg_invoke msg; ++ ++ memset(&header, 0, sizeof(header)); ++ header.nfe_present = 1; ++ header.nfe.source_entity = 0; /* endPINX */ ++ header.nfe.destination_entity = 0; /* endPINX */ ++ header.interpretation_present = 1; ++ header.interpretation = 0; /* discardAnyUnrecognisedInvokePdu */ ++ pos = facility_encode_header(ctrl, pos, end, &header); ++ if (!pos) { ++ return NULL; ++ } ++ ++ memset(&msg, 0, sizeof(msg)); ++ msg.operation = ROSE_QSIG_DivertingLegInformation2; ++ msg.invoke_id = get_invokeid(ctrl); ++ ++ /* diversionCounter is the redirecting.count */ ++ msg.args.qsig.DivertingLegInformation2.diversion_counter = call->redirecting.count; ++ ++ msg.args.qsig.DivertingLegInformation2.diversion_reason = ++ redirectingreason_from_q931(ctrl, call->redirecting.reason); ++ ++ /* divertingNr is the redirecting.from.number */ ++ msg.args.qsig.DivertingLegInformation2.diverting_present = 1; ++ q931_copy_presented_number_unscreened_to_rose(ctrl, ++ &msg.args.qsig.DivertingLegInformation2.diverting, ++ &call->redirecting.from.number); ++ ++ /* redirectingName is the redirecting.from.name */ ++ if (call->redirecting.from.name.valid) { ++ msg.args.qsig.DivertingLegInformation2.redirecting_name_present = 1; ++ q931_copy_name_to_rose(ctrl, ++ &msg.args.qsig.DivertingLegInformation2.redirecting_name, ++ &call->redirecting.from.name); ++ } ++ ++ if (1 < call->redirecting.count) { ++ /* originalCalledNr is the redirecting.orig_called.number */ ++ msg.args.qsig.DivertingLegInformation2.original_called_present = 1; ++ q931_copy_presented_number_unscreened_to_rose(ctrl, ++ &msg.args.qsig.DivertingLegInformation2.original_called, ++ &call->redirecting.orig_called.number); ++ ++ msg.args.qsig.DivertingLegInformation2.original_diversion_reason_present = 1; ++ if (call->redirecting.orig_called.number.valid) { ++ msg.args.qsig.DivertingLegInformation2.original_diversion_reason = ++ redirectingreason_from_q931(ctrl, call->redirecting.orig_reason); ++ } else { ++ msg.args.qsig.DivertingLegInformation2.original_diversion_reason = ++ QSIG_DIVERT_REASON_UNKNOWN; + } +- ASN1_FIXUP_LEN(comp, res); +- NEXT_COMPONENT(comp, i); +- if(i < len) +- pri_message(pri, "!! not all information is handled from Address component\n"); +- return res + 2; ++ ++ /* originalCalledName is the redirecting.orig_called.name */ ++ if (call->redirecting.orig_called.name.valid) { ++ msg.args.qsig.DivertingLegInformation2.original_called_name_present = 1; ++ q931_copy_name_to_rose(ctrl, ++ &msg.args.qsig.DivertingLegInformation2.original_called_name, ++ &call->redirecting.orig_called.name); ++ } + } +- while (0); + +- return -1; ++ pos = rose_encode_invoke(ctrl, pos, end, &msg); ++ ++ return pos; + } + +-static int rose_presented_number_unscreened_decode(struct pri *pri, q931_call *call, unsigned char *data, int len, struct addressingdataelements_presentednumberunscreened *value) ++/*! ++ * \internal ++ * \brief Encode the ETSI DivertingLegInformation2 invoke message. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode the facility ie contents. ++ * \param end End of facility ie contents encoding data buffer. ++ * \param call Call leg from which to encode diversion leg 2. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *enc_etsi_diverting_leg_information2(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, q931_call *call) + { +- int i = 0; +- int size = 0; +- struct rose_component *comp = NULL; +- unsigned char *vdata = data; ++ struct rose_msg_invoke msg; + +- /* Fill in default values */ +- value->ton = PRI_TON_UNKNOWN; +- value->npi = PRI_NPI_E163_E164; +- value->pres = -1; /* Data is not available */ ++ pos = facility_encode_header(ctrl, pos, end, NULL); ++ if (!pos) { ++ return NULL; ++ } + +- do { +- GET_COMPONENT(comp, i, vdata, len); ++ memset(&msg, 0, sizeof(msg)); ++ msg.operation = ROSE_ETSI_DivertingLegInformation2; ++ msg.invoke_id = get_invokeid(ctrl); + +- switch(comp->type) { +- case (ASN1_CONTEXT_SPECIFIC | ASN1_CONSTRUCTOR | ASN1_TAG_0): /* [0] presentationAllowedNumber */ +- value->pres = PRES_ALLOWED_USER_NUMBER_NOT_SCREENED; +- size = rose_address_decode(pri, call, comp->data, comp->len, value); +- ASN1_FIXUP_LEN(comp, size); +- return size + 2; +- case (ASN1_CONTEXT_SPECIFIC | ASN1_TAG_1): /* [1] IMPLICIT presentationRestricted */ +- if (comp->len != 0) { /* must be NULL */ +- pri_error(pri, "!! Invalid PresentationRestricted component received (len != 0)\n"); +- return -1; +- } +- value->pres = PRES_PROHIB_USER_NUMBER_NOT_SCREENED; +- return 2; +- case (ASN1_CONTEXT_SPECIFIC | ASN1_TAG_2): /* [2] IMPLICIT numberNotAvailableDueToInterworking */ +- if (comp->len != 0) { /* must be NULL */ +- pri_error(pri, "!! Invalid NumberNotAvailableDueToInterworking component received (len != 0)\n"); +- return -1; +- } +- value->pres = PRES_NUMBER_NOT_AVAILABLE; +- return 2; +- case (ASN1_CONTEXT_SPECIFIC | ASN1_CONSTRUCTOR | ASN1_TAG_3): /* [3] presentationRestrictedNumber */ +- value->pres = PRES_PROHIB_USER_NUMBER_NOT_SCREENED; +- size = rose_address_decode(pri, call, comp->data, comp->len, value) + 2; +- ASN1_FIXUP_LEN(comp, size); +- return size + 2; +- default: +- pri_message(pri, "Invalid PresentedNumberUnscreened component 0x%X\n", comp->type); +- } +- return -1; ++ /* diversionCounter is the redirecting.count */ ++ msg.args.etsi.DivertingLegInformation2.diversion_counter = call->redirecting.count; ++ ++ msg.args.etsi.DivertingLegInformation2.diversion_reason = ++ redirectingreason_from_q931(ctrl, call->redirecting.reason); ++ ++ /* divertingNr is the redirecting.from.number */ ++ msg.args.etsi.DivertingLegInformation2.diverting_present = 1; ++ q931_copy_presented_number_unscreened_to_rose(ctrl, ++ &msg.args.etsi.DivertingLegInformation2.diverting, ++ &call->redirecting.from.number); ++ ++ if (1 < call->redirecting.count) { ++ /* originalCalledNr is the redirecting.orig_called.number */ ++ msg.args.etsi.DivertingLegInformation2.original_called_present = 1; ++ q931_copy_presented_number_unscreened_to_rose(ctrl, ++ &msg.args.etsi.DivertingLegInformation2.original_called, ++ &call->redirecting.orig_called.number); + } +- while (0); + +- return -1; ++ pos = rose_encode_invoke(ctrl, pos, end, &msg); ++ ++ return pos; + } + +-static int rose_diverting_leg_information2_decode(struct pri *pri, q931_call *call, struct rose_component *sequence, int len) ++/*! ++ * \internal ++ * \brief Encode and queue the DivertingLegInformation2 invoke message. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param call Call leg from which to encode diversion leg 2. ++ * ++ * \retval 0 on success. ++ * \retval -1 on error. ++ */ ++static int rose_diverting_leg_information2_encode(struct pri *ctrl, q931_call *call) + { +- int i = 0; +- int diversion_counter; +- int diversion_reason; +- char origcalledname[50] = "", redirectingname[50] = ""; +- struct addressingdataelements_presentednumberunscreened divertingnr; +- struct addressingdataelements_presentednumberunscreened originalcallednr; +- struct rose_component *comp = NULL; +- unsigned char *vdata = sequence->data; +- int res = 0; +- memset(&divertingnr, 0, sizeof(divertingnr)); +- memset(&originalcallednr, 0, sizeof(originalcallednr)); ++ unsigned char buffer[256]; ++ unsigned char *end; + +- /* Data checks */ +- if (sequence->type != (ASN1_CONSTRUCTOR | ASN1_SEQUENCE)) { /* Constructed Sequence */ +- pri_message(pri, "Invalid DivertingLegInformation2Type argument\n"); ++ switch (ctrl->switchtype) { ++ case PRI_SWITCH_EUROISDN_E1: ++ case PRI_SWITCH_EUROISDN_T1: ++ end = enc_etsi_diverting_leg_information2(ctrl, buffer, buffer + sizeof(buffer), ++ call); ++ break; ++ case PRI_SWITCH_QSIG: ++ end = enc_qsig_diverting_leg_information2(ctrl, buffer, buffer + sizeof(buffer), ++ call); ++ break; ++ default: + return -1; + } ++ if (!end) { ++ return -1; ++ } + +- if (sequence->len == ASN1_LEN_INDEF) { +- len -= 4; /* For the 2 extra characters at the end +- * and two characters of header */ +- } else +- len -= 2; ++ return pri_call_apdu_queue(call, Q931_SETUP, buffer, end - buffer, NULL); ++} + +- do { +- /* diversionCounter stuff */ +- GET_COMPONENT(comp, i, vdata, len); +- CHECK_COMPONENT(comp, ASN1_INTEGER, "Don't know what to do it diversionCounter is of type 0x%x\n"); +- ASN1_GET_INTEGER(comp, diversion_counter); +- NEXT_COMPONENT(comp, i); ++/*! ++ * \internal ++ * \brief Encode the Q.SIG DivertingLegInformation3 invoke message. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode the facility ie contents. ++ * \param end End of facility ie contents encoding data buffer. ++ * \param call Call leg from which to encode diversion leg 3. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *enc_qsig_diverting_leg_information3(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, q931_call *call) ++{ ++ struct fac_extension_header header; ++ struct rose_msg_invoke msg; + +- /* diversionReason stuff */ +- GET_COMPONENT(comp, i, vdata, len); +- CHECK_COMPONENT(comp, ASN1_ENUMERATED, "Invalid diversionReason type 0x%X of ROSE divertingLegInformation2 component received\n"); +- ASN1_GET_INTEGER(comp, diversion_reason); +- NEXT_COMPONENT(comp, i); ++ memset(&header, 0, sizeof(header)); ++ header.nfe_present = 1; ++ header.nfe.source_entity = 0; /* endPINX */ ++ header.nfe.destination_entity = 0; /* endPINX */ ++ header.interpretation_present = 1; ++ header.interpretation = 0; /* discardAnyUnrecognisedInvokePdu */ ++ pos = facility_encode_header(ctrl, pos, end, &header); ++ if (!pos) { ++ return NULL; ++ } + +- diversion_reason = redirectingreason_for_q931(pri, diversion_reason); +- +- if(pri->debug & PRI_DEBUG_APDU) +- pri_message(pri, " Redirection reason: %d, total diversions: %d\n", diversion_reason, diversion_counter); +- pri_message(NULL, "Length of message is %d\n", len); ++ memset(&msg, 0, sizeof(msg)); ++ msg.operation = ROSE_QSIG_DivertingLegInformation3; ++ msg.invoke_id = get_invokeid(ctrl); + +- for(; i < len; NEXT_COMPONENT(comp, i)) { +- GET_COMPONENT(comp, i, vdata, len); +- switch(comp->type) { +- case (ASN1_CONTEXT_SPECIFIC | ASN1_TAG_0): +- call->origredirectingreason = redirectingreason_for_q931(pri, comp->data[0]); +- if (pri->debug & PRI_DEBUG_APDU) +- pri_message(pri, " Received reason for original redirection %d\n", call->origredirectingreason); +- break; +- case (ASN1_CONTEXT_SPECIFIC | ASN1_CONSTRUCTOR | ASN1_TAG_1): +- res = rose_presented_number_unscreened_decode(pri, call, comp->data, comp->len, &divertingnr); +- /* TODO: Fix indefinite length form hacks */ +- ASN1_FIXUP_LEN(comp, res); +- comp->len = res; +- if (res < 0) +- return -1; +- if (pri->debug & PRI_DEBUG_APDU) { +- pri_message(pri, " Received divertingNr '%s'\n", divertingnr.partyaddress); +- pri_message(pri, " ton = %d, pres = %d, npi = %d\n", divertingnr.ton, divertingnr.pres, divertingnr.npi); +- } +- break; +- case (ASN1_CONTEXT_SPECIFIC | ASN1_CONSTRUCTOR | ASN1_TAG_2): +- res = rose_presented_number_unscreened_decode(pri, call, comp->data, comp->len, &originalcallednr); +- if (res < 0) +- return -1; +- ASN1_FIXUP_LEN(comp, res); +- comp->len = res; +- if (pri->debug & PRI_DEBUG_APDU) { +- pri_message(pri, " Received originalcallednr '%s'\n", originalcallednr.partyaddress); +- pri_message(pri, " ton = %d, pres = %d, npi = %d\n", originalcallednr.ton, originalcallednr.pres, originalcallednr.npi); +- } +- break; +- case (ASN1_CONTEXT_SPECIFIC | ASN1_CONSTRUCTOR | ASN1_TAG_3): +- res = asn1_name_decode(comp->data, comp->len, redirectingname, sizeof(redirectingname)); +- if (res < 0) +- return -1; +- ASN1_FIXUP_LEN(comp, res); +- comp->len = res; +- if (pri->debug & PRI_DEBUG_APDU) +- pri_message(pri, " Received RedirectingName '%s'\n", redirectingname); +- break; +- case (ASN1_CONTEXT_SPECIFIC | ASN1_CONSTRUCTOR | ASN1_TAG_4): +- res = asn1_name_decode(comp->data, comp->len, origcalledname, sizeof(origcalledname)); +- if (res < 0) +- return -1; +- ASN1_FIXUP_LEN(comp, res); +- comp->len = res; +- if (pri->debug & PRI_DEBUG_APDU) +- pri_message(pri, " Received Originally Called Name '%s'\n", origcalledname); +- break; +- case (ASN1_CONTEXT_SPECIFIC | ASN1_CONSTRUCTOR | ASN1_TAG_5): +- pri_message(pri, "!! Ignoring DivertingLegInformation2 component 0x%X\n", comp->type); +- break; +- default: +- if (comp->type == 0 && comp->len == 0) { +- break; /* Found termination characters */ +- } +- pri_message(pri, "!! Invalid DivertingLegInformation2 component received 0x%X\n", comp->type); +- return -1; +- } +- } ++ /* redirecting.to.number.presentation also indicates if name presentation is allowed */ ++ if ((call->redirecting.to.number.presentation & PRI_PRES_RESTRICTION) == PRI_PRES_ALLOWED) { ++ msg.args.qsig.DivertingLegInformation3.presentation_allowed_indicator = 1; /* TRUE */ + +- if (divertingnr.pres >= 0) { +- call->redirectingplan = divertingnr.npi; +- call->redirectingpres = divertingnr.pres; +- call->redirectingreason = diversion_reason; +- libpri_copy_string(call->redirectingnum, divertingnr.partyaddress, sizeof(call->redirectingnum)); +- pri_message(pri, " Received redirectingnum '%s' (%d)\n", call->redirectingnum, (int)call->redirectingnum[0]); ++ /* redirectionName is the redirecting.to.name */ ++ if (call->redirecting.to.name.valid) { ++ msg.args.qsig.DivertingLegInformation3.redirection_name_present = 1; ++ q931_copy_name_to_rose(ctrl, ++ &msg.args.qsig.DivertingLegInformation3.redirection_name, ++ &call->redirecting.to.name); + } +- if (originalcallednr.pres >= 0) { +- call->origcalledplan = originalcallednr.npi; +- call->origcalledpres = originalcallednr.pres; +- libpri_copy_string(call->origcallednum, originalcallednr.partyaddress, sizeof(call->origcallednum)); +- pri_message(pri, " Received origcallednum '%s' (%d)\n", call->origcallednum, (int)call->origcallednum[0]); +- } +- libpri_copy_string(call->redirectingname, redirectingname, sizeof(call->redirectingname)); +- libpri_copy_string(call->origcalledname, origcalledname, sizeof(call->origcalledname)); +- return 0; + } +- while (0); + +- return -1; ++ pos = rose_encode_invoke(ctrl, pos, end, &msg); ++ ++ return pos; + } +- +-static int rose_diverting_leg_information2_encode(struct pri *pri, q931_call *call) ++ ++/*! ++ * \internal ++ * \brief Encode the ETSI DivertingLegInformation3 invoke message. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode the facility ie contents. ++ * \param end End of facility ie contents encoding data buffer. ++ * \param call Call leg from which to encode diversion leg 3. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *enc_etsi_diverting_leg_information3(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, q931_call *call) + { +- int i = 0, j, compsp = 0; +- struct rose_component *comp, *compstk[10]; +- unsigned char buffer[256]; +- int len = 253; +- +-#if 0 /* This is not required by specifications */ +- if (!strlen(call->callername)) { +- return -1; ++ struct rose_msg_invoke msg; ++ ++ pos = facility_encode_header(ctrl, pos, end, NULL); ++ if (!pos) { ++ return NULL; + } +-#endif + +- buffer[i] = (ASN1_CONTEXT_SPECIFIC | Q932_PROTOCOL_EXTENSIONS); +- i++; +- /* Interpretation component */ +- ASN1_ADD_BYTECOMP(comp, COMP_TYPE_INTERPRETATION, buffer, i, 0x00 /* Discard unrecognized invokes */); +- +- ASN1_ADD_SIMPLE(comp, COMP_TYPE_INVOKE, buffer, i); +- +- ASN1_PUSH(compstk, compsp, comp); +- /* Invoke component contents */ +- /* Invoke ID */ +- ASN1_ADD_BYTECOMP(comp, ASN1_INTEGER, buffer, i, get_invokeid(pri)); +- /* Operation Tag */ +- +- /* ROSE operationId component */ +- ASN1_ADD_BYTECOMP(comp, ASN1_INTEGER, buffer, i, ROSE_DIVERTING_LEG_INFORMATION2); ++ memset(&msg, 0, sizeof(msg)); ++ msg.operation = ROSE_ETSI_DivertingLegInformation3; ++ msg.invoke_id = get_invokeid(ctrl); + +- /* ROSE ARGUMENT component */ +- ASN1_ADD_SIMPLE(comp, (ASN1_CONSTRUCTOR | ASN1_SEQUENCE), buffer, i); +- ASN1_PUSH(compstk, compsp, comp); +- /* ROSE DivertingLegInformation2.diversionCounter component */ +- /* Always is 1 because other isn't available in the current design */ +- ASN1_ADD_BYTECOMP(comp, ASN1_INTEGER, buffer, i, 1); +- +- /* ROSE DivertingLegInformation2.diversionReason component */ +- ASN1_ADD_BYTECOMP(comp, ASN1_ENUMERATED, buffer, i, redirectingreason_from_q931(pri, call->redirectingreason)); +- +- /* ROSE DivertingLegInformation2.divertingNr component */ +- ASN1_ADD_SIMPLE(comp, (ASN1_CONTEXT_SPECIFIC | ASN1_CONSTRUCTOR | ASN1_TAG_1), buffer, i); +- +- ASN1_PUSH(compstk, compsp, comp); +- /* Redirecting information always not screened */ +- +- switch(call->redirectingpres) { +- case PRES_ALLOWED_USER_NUMBER_NOT_SCREENED: +- case PRES_ALLOWED_USER_NUMBER_PASSED_SCREEN: +- if (call->redirectingnum && strlen(call->redirectingnum)) { +- ASN1_ADD_SIMPLE(comp, (ASN1_CONTEXT_SPECIFIC | ASN1_CONSTRUCTOR | ASN1_TAG_0), buffer, i); +- ASN1_PUSH(compstk, compsp, comp); +- /* NPI of redirected number is not supported in the current design */ +- ASN1_ADD_SIMPLE(comp, (ASN1_CONTEXT_SPECIFIC | ASN1_CONSTRUCTOR | ASN1_TAG_1), buffer, i); +- ASN1_PUSH(compstk, compsp, comp); +- ASN1_ADD_BYTECOMP(comp, ASN1_ENUMERATED, buffer, i, typeofnumber_from_q931(pri, call->redirectingplan >> 4)); +- j = asn1_string_encode(ASN1_NUMERICSTRING, &buffer[i], len - i, 20, call->redirectingnum, strlen(call->redirectingnum)); +- if (j < 0) +- return -1; +- +- i += j; +- ASN1_FIXUP(compstk, compsp, buffer, i); +- ASN1_FIXUP(compstk, compsp, buffer, i); +- break; +- } +- /* fall through */ +- case PRES_PROHIB_USER_NUMBER_PASSED_SCREEN: +- case PRES_PROHIB_USER_NUMBER_NOT_SCREENED: +- ASN1_ADD_SIMPLE(comp, (ASN1_CONTEXT_SPECIFIC | ASN1_TAG_1), buffer, i); +- break; +- /* Don't know how to handle this */ +- case PRES_ALLOWED_NETWORK_NUMBER: +- case PRES_PROHIB_NETWORK_NUMBER: +- case PRES_ALLOWED_USER_NUMBER_FAILED_SCREEN: +- case PRES_PROHIB_USER_NUMBER_FAILED_SCREEN: +- ASN1_ADD_SIMPLE(comp, (ASN1_CONTEXT_SPECIFIC | ASN1_TAG_1), buffer, i); +- break; +- default: +- pri_message(pri, "!! Undefined presentation value for redirecting number: %d\n", call->redirectingpres); +- case PRES_NUMBER_NOT_AVAILABLE: +- ASN1_ADD_SIMPLE(comp, (ASN1_CONTEXT_SPECIFIC | ASN1_TAG_2), buffer, i); +- break; ++ if ((call->redirecting.to.number.presentation & PRI_PRES_RESTRICTION) == PRI_PRES_ALLOWED) { ++ msg.args.etsi.DivertingLegInformation3.presentation_allowed_indicator = 1; /* TRUE */ + } +- ASN1_FIXUP(compstk, compsp, buffer, i); + +- /* ROSE DivertingLegInformation2.originalCalledNr component */ +- /* This information isn't supported by current design - duplicate divertingNr */ +- ASN1_ADD_SIMPLE(comp, (ASN1_CONTEXT_SPECIFIC | ASN1_CONSTRUCTOR | ASN1_TAG_2), buffer, i); +- ASN1_PUSH(compstk, compsp, comp); +- /* Redirecting information always not screened */ +- switch(call->redirectingpres) { +- case PRES_ALLOWED_USER_NUMBER_NOT_SCREENED: +- case PRES_ALLOWED_USER_NUMBER_PASSED_SCREEN: +- if (call->redirectingnum && strlen(call->redirectingnum)) { +- ASN1_ADD_SIMPLE(comp, (ASN1_CONTEXT_SPECIFIC | ASN1_CONSTRUCTOR | ASN1_TAG_0), buffer, i); +- ASN1_PUSH(compstk, compsp, comp); +- ASN1_ADD_SIMPLE(comp, (ASN1_CONTEXT_SPECIFIC | ASN1_CONSTRUCTOR | ASN1_TAG_1), buffer, i); +- ASN1_PUSH(compstk, compsp, comp); +- ASN1_ADD_BYTECOMP(comp, ASN1_ENUMERATED, buffer, i, typeofnumber_from_q931(pri, call->redirectingplan >> 4)); +- +- j = asn1_string_encode(ASN1_NUMERICSTRING, &buffer[i], len - i, 20, call->redirectingnum, strlen(call->redirectingnum)); +- if (j < 0) +- return -1; +- +- i += j; +- ASN1_FIXUP(compstk, compsp, buffer, i); +- ASN1_FIXUP(compstk, compsp, buffer, i); +- break; +- } +- /* fall through */ +- case PRES_PROHIB_USER_NUMBER_PASSED_SCREEN: +- case PRES_PROHIB_USER_NUMBER_NOT_SCREENED: +- ASN1_ADD_SIMPLE(comp, (ASN1_CONTEXT_SPECIFIC | ASN1_TAG_1), buffer, i); +- break; +- /* Don't know how to handle this */ +- case PRES_ALLOWED_NETWORK_NUMBER: +- case PRES_PROHIB_NETWORK_NUMBER: +- case PRES_ALLOWED_USER_NUMBER_FAILED_SCREEN: +- case PRES_PROHIB_USER_NUMBER_FAILED_SCREEN: +- ASN1_ADD_SIMPLE(comp, (ASN1_CONTEXT_SPECIFIC | ASN1_TAG_1), buffer, i); +- break; +- default: +- pri_message(pri, "!! Undefined presentation value for redirecting number: %d\n", call->redirectingpres); +- case PRES_NUMBER_NOT_AVAILABLE: +- ASN1_ADD_SIMPLE(comp, (ASN1_CONTEXT_SPECIFIC | ASN1_TAG_2), buffer, i); +- break; ++ pos = rose_encode_invoke(ctrl, pos, end, &msg); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode and queue the DivertingLegInformation3 invoke message. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param call Call leg from which to encode diversion leg 3. ++ * \param messagetype Q.931 message type to add facility ie to. ++ * ++ * \retval 0 on success. ++ * \retval -1 on error. ++ */ ++int rose_diverting_leg_information3_encode(struct pri *ctrl, q931_call *call, ++ int messagetype) ++{ ++ unsigned char buffer[256]; ++ unsigned char *end; ++ ++ switch (ctrl->switchtype) { ++ case PRI_SWITCH_EUROISDN_E1: ++ case PRI_SWITCH_EUROISDN_T1: ++ end = enc_etsi_diverting_leg_information3(ctrl, buffer, buffer + sizeof(buffer), ++ call); ++ break; ++ case PRI_SWITCH_QSIG: ++ end = enc_qsig_diverting_leg_information3(ctrl, buffer, buffer + sizeof(buffer), ++ call); ++ break; ++ default: ++ return -1; + } +- ASN1_FIXUP(compstk, compsp, buffer, i); +- +- /* Fix length of stacked components */ +- while(compsp > 0) { +- ASN1_FIXUP(compstk, compsp, buffer, i); ++ if (!end) { ++ return -1; + } +- +- if (pri_call_apdu_queue(call, Q931_SETUP, buffer, i, NULL, NULL)) +- return -1; +- +- return 0; ++ ++ return pri_call_apdu_queue(call, messagetype, buffer, end - buffer, NULL); + } + +-/* Send the rltThirdParty: Invoke */ +-int rlt_initiate_transfer(struct pri *pri, q931_call *c1, q931_call *c2) ++/*! ++ * \internal ++ * \brief Encode the rltThirdParty invoke message. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode the facility ie contents. ++ * \param end End of facility ie contents encoding data buffer. ++ * \param callwithid Call-ID information to encode. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *enc_dms100_rlt_initiate_transfer(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, const q931_call *callwithid) + { +- int i = 0; ++ struct rose_msg_invoke msg; ++ ++ pos = facility_encode_header(ctrl, pos, end, NULL); ++ if (!pos) { ++ return NULL; ++ } ++ ++ memset(&msg, 0, sizeof(msg)); ++ msg.operation = ROSE_DMS100_RLT_ThirdParty; ++ msg.invoke_id = ROSE_DMS100_RLT_THIRD_PARTY; ++ msg.args.dms100.RLT_ThirdParty.call_id = callwithid->rlt_call_id & 0xFFFFFF; ++ msg.args.dms100.RLT_ThirdParty.reason = 0; /* unused, set to 129 */ ++ pos = rose_encode_invoke(ctrl, pos, end, &msg); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Send the rltThirdParty: Invoke. ++ * ++ * \note For PRI_SWITCH_DMS100 only. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param c1 Q.931 call leg 1 ++ * \param c2 Q.931 call leg 2 ++ * ++ * \retval 0 on success. ++ * \retval -1 on error. ++ */ ++int rlt_initiate_transfer(struct pri *ctrl, q931_call *c1, q931_call *c2) ++{ + unsigned char buffer[256]; +- struct rose_component *comp = NULL, *compstk[10]; +- const unsigned char rlt_3rd_pty = RLT_THIRD_PARTY; +- q931_call *callwithid = NULL, *apdubearer = NULL; +- int compsp = 0; ++ unsigned char *end; ++ q931_call *apdubearer; ++ q931_call *callwithid; + + if (c2->transferable) { + apdubearer = c1; +@@ -906,279 +1374,415 @@ + } else if (c1->transferable) { + apdubearer = c2; + callwithid = c1; +- } else ++ } else { + return -1; ++ } + +- buffer[i++] = (Q932_PROTOCOL_ROSE); +- buffer[i++] = (0x80 | RLT_SERVICE_ID); /* Service Identifier octet */ ++ end = ++ enc_dms100_rlt_initiate_transfer(ctrl, buffer, buffer + sizeof(buffer), ++ callwithid); ++ if (!end) { ++ return -1; ++ } + +- ASN1_ADD_SIMPLE(comp, COMP_TYPE_INVOKE, buffer, i); +- ASN1_PUSH(compstk, compsp, comp); +- +- /* Invoke ID is set to the operation ID */ +- ASN1_ADD_BYTECOMP(comp, ASN1_INTEGER, buffer, i, rlt_3rd_pty); +- +- /* Operation Tag */ +- ASN1_ADD_BYTECOMP(comp, ASN1_INTEGER, buffer, i, rlt_3rd_pty); +- +- /* Additional RLT invoke info - Octet 12 */ +- ASN1_ADD_SIMPLE(comp, (ASN1_CONSTRUCTOR | ASN1_SEQUENCE), buffer, i); +- ASN1_PUSH(compstk, compsp, comp); +- +- ASN1_ADD_WORDCOMP(comp, (ASN1_CONTEXT_SPECIFIC | ASN1_TAG_0), buffer, i, callwithid->rlt_call_id & 0xFFFFFF); /* Length is 3 octets */ +- /* Reason for redirect - unused, set to 129 */ +- ASN1_ADD_BYTECOMP(comp, (ASN1_CONTEXT_SPECIFIC | ASN1_TAG_1), buffer, i, 0); +- ASN1_FIXUP(compstk, compsp, buffer, i); +- ASN1_FIXUP(compstk, compsp, buffer, i); +- +- if (pri_call_apdu_queue(apdubearer, Q931_FACILITY, buffer, i, NULL, NULL)) ++ if (pri_call_apdu_queue(apdubearer, Q931_FACILITY, buffer, end - buffer, NULL)) { + return -1; ++ } + + if (q931_facility(apdubearer->pri, apdubearer)) { +- pri_message(pri, "Could not schedule facility message for call %d\n", apdubearer->cr); ++ pri_message(ctrl, "Could not schedule facility message for call %d\n", ++ apdubearer->cr); + return -1; + } + return 0; + } + +-static int add_dms100_transfer_ability_apdu(struct pri *pri, q931_call *c) ++/*! ++ * \internal ++ * \brief Encode the rltOperationInd invoke message. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode the facility ie contents. ++ * \param end End of facility ie contents encoding data buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *enc_dms100_rlt_transfer_ability(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end) + { +- int i = 0; +- unsigned char buffer[256]; +- struct rose_component *comp = NULL, *compstk[10]; +- const unsigned char rlt_op_ind = RLT_OPERATION_IND; +- int compsp = 0; ++ struct rose_msg_invoke msg; + +- buffer[i++] = (Q932_PROTOCOL_ROSE); /* Note to self: DON'T set the EXT bit */ +- buffer[i++] = (0x80 | RLT_SERVICE_ID); /* Service Identifier octet */ ++ pos = facility_encode_header(ctrl, pos, end, NULL); ++ if (!pos) { ++ return NULL; ++ } + +- ASN1_ADD_SIMPLE(comp, COMP_TYPE_INVOKE, buffer, i); +- ASN1_PUSH(compstk, compsp, comp); ++ memset(&msg, 0, sizeof(msg)); ++ msg.operation = ROSE_DMS100_RLT_OperationInd; ++ msg.invoke_id = ROSE_DMS100_RLT_OPERATION_IND; ++ pos = rose_encode_invoke(ctrl, pos, end, &msg); + +- /* Invoke ID is set to the operation ID */ +- ASN1_ADD_BYTECOMP(comp, ASN1_INTEGER, buffer, i, rlt_op_ind); +- +- /* Operation Tag - basically the same as the invoke ID tag */ +- ASN1_ADD_BYTECOMP(comp, ASN1_INTEGER, buffer, i, rlt_op_ind); +- ASN1_FIXUP(compstk, compsp, buffer, i); +- +- if (pri_call_apdu_queue(c, Q931_SETUP, buffer, i, NULL, NULL)) +- return -1; +- else +- return 0; ++ return pos; + } + +-/* Sending callername information functions */ +-static int add_callername_facility_ies(struct pri *pri, q931_call *c, int cpe) ++/*! ++ * \internal ++ * \brief Send the rltOperationInd: Invoke. ++ * ++ * \note For PRI_SWITCH_DMS100 only. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param call Q.931 call leg ++ * ++ * \retval 0 on success. ++ * \retval -1 on error. ++ */ ++static int add_dms100_transfer_ability_apdu(struct pri *ctrl, q931_call *call) + { +- int res = 0; +- int i = 0; + unsigned char buffer[256]; +- unsigned char namelen = 0; +- struct rose_component *comp = NULL, *compstk[10]; +- int compsp = 0; +- int mymessage = 0; +- static unsigned char op_tag[] = { +- 0x2a, /* informationFollowing 42 */ +- 0x86, +- 0x48, +- 0xce, +- 0x15, +- 0x00, +- 0x04 +- }; +- +- if (!strlen(c->callername)) { ++ unsigned char *end; ++ ++ end = enc_dms100_rlt_transfer_ability(ctrl, buffer, buffer + sizeof(buffer)); ++ if (!end) { + return -1; + } + +- buffer[i++] = (ASN1_CONTEXT_SPECIFIC | Q932_PROTOCOL_EXTENSIONS); +- /* Interpretation component */ ++ return pri_call_apdu_queue(call, Q931_SETUP, buffer, end - buffer, NULL); ++} + +- if (pri->switchtype == PRI_SWITCH_QSIG) { +- ASN1_ADD_SIMPLE(comp, COMP_TYPE_NFE, buffer, i); +- ASN1_PUSH(compstk, compsp, comp); +- ASN1_ADD_BYTECOMP(comp, (ASN1_CONTEXT_SPECIFIC | ASN1_TAG_0), buffer, i, 0); +- ASN1_ADD_BYTECOMP(comp, (ASN1_CONTEXT_SPECIFIC | ASN1_TAG_2), buffer, i, 0); +- ASN1_FIXUP(compstk, compsp, buffer, i); ++/*! ++ * \internal ++ * \brief Encode the NI2 InformationFollowing invoke message. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode the facility ie contents. ++ * \param end End of facility ie contents encoding data buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *enc_ni2_information_following(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end) ++{ ++ struct fac_extension_header header; ++ struct rose_msg_invoke msg; ++ ++ memset(&header, 0, sizeof(header)); ++ header.interpretation_present = 1; ++ header.interpretation = 0; /* discardAnyUnrecognisedInvokePdu */ ++ pos = facility_encode_header(ctrl, pos, end, &header); ++ if (!pos) { ++ return NULL; + } + +- ASN1_ADD_BYTECOMP(comp, COMP_TYPE_INTERPRETATION, buffer, i, 0); ++ memset(&msg, 0, sizeof(msg)); ++ msg.operation = ROSE_NI2_InformationFollowing; ++ msg.invoke_id = get_invokeid(ctrl); ++ msg.args.ni2.InformationFollowing.value = 0; ++ pos = rose_encode_invoke(ctrl, pos, end, &msg); + +- ASN1_ADD_SIMPLE(comp, COMP_TYPE_INVOKE, buffer, i); +- ASN1_PUSH(compstk, compsp, comp); +- /* Invoke ID */ +- ASN1_ADD_BYTECOMP(comp, ASN1_INTEGER, buffer, i, get_invokeid(pri)); ++ return pos; ++} + +- /* Operation Tag */ +- res = asn1_string_encode(ASN1_OBJECTIDENTIFIER, &buffer[i], sizeof(buffer)-i, sizeof(op_tag), op_tag, sizeof(op_tag)); +- if (res < 0) +- return -1; +- i += res; ++/*! ++ * \internal ++ * \brief Encode the Q.SIG CallingName invoke message. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode the facility ie contents. ++ * \param end End of facility ie contents encoding data buffer. ++ * \param name Name data which to encode name. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *enc_qsig_calling_name(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const struct q931_party_name *name) ++{ ++ struct fac_extension_header header; ++ struct rose_msg_invoke msg; + +- ASN1_ADD_BYTECOMP(comp, ASN1_ENUMERATED, buffer, i, 0); +- ASN1_FIXUP(compstk, compsp, buffer, i); +- +- if (!cpe) { +- if (pri_call_apdu_queue(c, Q931_SETUP, buffer, i, NULL, NULL)) +- return -1; ++ memset(&header, 0, sizeof(header)); ++ if (ctrl->switchtype == PRI_SWITCH_QSIG) { ++ header.nfe_present = 1; ++ header.nfe.source_entity = 0; /* endPINX */ ++ header.nfe.destination_entity = 0; /* endPINX */ + } ++ header.interpretation_present = 1; ++ header.interpretation = 0; /* discardAnyUnrecognisedInvokePdu */ ++ pos = facility_encode_header(ctrl, pos, end, &header); ++ if (!pos) { ++ return NULL; ++ } + ++ memset(&msg, 0, sizeof(msg)); ++ msg.operation = ROSE_QSIG_CallingName; ++ msg.invoke_id = get_invokeid(ctrl); + +- /* Now the APDU that contains the information that needs sent. +- * We can reuse the buffer since the queue function doesn't +- * need it. */ ++ /* CallingName */ ++ q931_copy_name_to_rose(ctrl, &msg.args.qsig.CallingName.name, name); + +- i = 0; +- namelen = strlen(c->callername); +- if (namelen > 50) { +- namelen = 50; /* truncate the name */ +- } ++ pos = rose_encode_invoke(ctrl, pos, end, &msg); + +- buffer[i++] = (ASN1_CONTEXT_SPECIFIC | Q932_PROTOCOL_EXTENSIONS); +- /* Interpretation component */ ++ return pos; ++} + +- if (pri->switchtype == PRI_SWITCH_QSIG) { +- ASN1_ADD_SIMPLE(comp, COMP_TYPE_NFE, buffer, i); +- ASN1_PUSH(compstk, compsp, comp); +- ASN1_ADD_BYTECOMP(comp, (ASN1_CONTEXT_SPECIFIC | ASN1_TAG_0), buffer, i, 0); +- ASN1_ADD_BYTECOMP(comp, (ASN1_CONTEXT_SPECIFIC | ASN1_TAG_2), buffer, i, 0); +- ASN1_FIXUP(compstk, compsp, buffer, i); ++/*! ++ * \internal ++ * \brief Send caller name information. ++ * ++ * \note For PRI_SWITCH_NI2 and PRI_SWITCH_QSIG. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param call Call leg from which to encode name. ++ * \param cpe TRUE if we are the CPE side. ++ * ++ * \retval 0 on success. ++ * \retval -1 on error. ++ */ ++static int add_callername_facility_ies(struct pri *ctrl, q931_call *call, int cpe) ++{ ++ unsigned char buffer[256]; ++ unsigned char *end; ++ int mymessage; ++ ++ if (!call->local_id.name.valid) { ++ return 0; + } + +- ASN1_ADD_BYTECOMP(comp, COMP_TYPE_INTERPRETATION, buffer, i, 0); ++ if (ctrl->switchtype == PRI_SWITCH_NI2 && !cpe) { ++ end = enc_ni2_information_following(ctrl, buffer, buffer + sizeof(buffer)); ++ if (!end) { ++ return -1; ++ } + +- ASN1_ADD_SIMPLE(comp, COMP_TYPE_INVOKE, buffer, i); +- ASN1_PUSH(compstk, compsp, comp); ++ if (pri_call_apdu_queue(call, Q931_SETUP, buffer, end - buffer, NULL)) { ++ return -1; ++ } + +- /* Invoke ID */ +- ASN1_ADD_BYTECOMP(comp, ASN1_INTEGER, buffer, i, get_invokeid(pri)); ++ /* ++ * We can reuse the buffer since the queue function doesn't ++ * need it. ++ */ ++ } + +- /* Operation ID: Calling name */ +- ASN1_ADD_BYTECOMP(comp, ASN1_INTEGER, buffer, i, SS_CNID_CALLINGNAME); +- +- res = asn1_string_encode((ASN1_CONTEXT_SPECIFIC | ASN1_TAG_0), &buffer[i], sizeof(buffer)-i, 50, c->callername, namelen); +- if (res < 0) ++ /* CallingName is the local_id.name */ ++ end = enc_qsig_calling_name(ctrl, buffer, buffer + sizeof(buffer), ++ &call->local_id.name); ++ if (!end) { + return -1; +- i += res; +- ASN1_FIXUP(compstk, compsp, buffer, i); ++ } + +- if (cpe) ++ if (cpe) { + mymessage = Q931_SETUP; +- else ++ } else { + mymessage = Q931_FACILITY; ++ } + +- if (pri_call_apdu_queue(c, mymessage, buffer, i, NULL, NULL)) +- return -1; +- +- return 0; ++ return pri_call_apdu_queue(call, mymessage, buffer, end - buffer, NULL); + } + /* End Callername */ + + /* MWI related encode and decode functions */ +-static void mwi_activate_encode_cb(void *data) ++ ++/*! ++ * \internal ++ * \brief Encode the Q.SIG MWIActivate invoke message. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode the facility ie contents. ++ * \param end End of facility ie contents encoding data buffer. ++ * \param req Served user setup request information. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *enc_qsig_mwi_activate_message(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, struct pri_sr *req) + { +- return; ++ struct fac_extension_header header; ++ struct rose_msg_invoke msg; ++ ++ memset(&header, 0, sizeof(header)); ++ header.nfe_present = 1; ++ header.nfe.source_entity = 0; /* endPINX */ ++ header.nfe.destination_entity = 0; /* endPINX */ ++ header.interpretation_present = 1; ++ header.interpretation = 0; /* discardAnyUnrecognisedInvokePdu */ ++ pos = facility_encode_header(ctrl, pos, end, &header); ++ if (!pos) { ++ return NULL; ++ } ++ ++ memset(&msg, 0, sizeof(msg)); ++ msg.operation = ROSE_QSIG_MWIActivate; ++ msg.invoke_id = get_invokeid(ctrl); ++ ++ /* The called.number is the served user */ ++ q931_copy_number_to_rose(ctrl, &msg.args.qsig.MWIActivate.served_user_number, ++ &req->called.number); ++ /* ++ * For now, we will just force the numbering plan to unknown to preserve ++ * the original behaviour. ++ */ ++ msg.args.qsig.MWIActivate.served_user_number.plan = 0; /* unknown */ ++ ++ msg.args.qsig.MWIActivate.basic_service = 1; /* speech */ ++ ++ pos = rose_encode_invoke(ctrl, pos, end, &msg); ++ ++ return pos; + } + +-int mwi_message_send(struct pri* pri, q931_call *call, struct pri_sr *req, int activate) ++/*! ++ * \internal ++ * \brief Encode the Q.SIG MWIDeactivate invoke message. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode the facility ie contents. ++ * \param end End of facility ie contents encoding data buffer. ++ * \param req Served user setup request information. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *enc_qsig_mwi_deactivate_message(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, struct pri_sr *req) + { +- int i = 0; +- unsigned char buffer[255] = ""; +- int destlen = strlen(req->called); +- struct rose_component *comp = NULL, *compstk[10]; +- int compsp = 0; +- int res; ++ struct fac_extension_header header; ++ struct rose_msg_invoke msg; + +- if (destlen <= 0) { +- return -1; +- } else if (destlen > 20) +- destlen = 20; /* Destination number cannot be greater then 20 digits */ ++ memset(&header, 0, sizeof(header)); ++ header.nfe_present = 1; ++ header.nfe.source_entity = 0; /* endPINX */ ++ header.nfe.destination_entity = 0; /* endPINX */ ++ header.interpretation_present = 1; ++ header.interpretation = 0; /* discardAnyUnrecognisedInvokePdu */ ++ pos = facility_encode_header(ctrl, pos, end, &header); ++ if (!pos) { ++ return NULL; ++ } + +- buffer[i++] = (ASN1_CONTEXT_SPECIFIC | Q932_PROTOCOL_EXTENSIONS); +- /* Interpretation component */ ++ memset(&msg, 0, sizeof(msg)); ++ msg.operation = ROSE_QSIG_MWIDeactivate; ++ msg.invoke_id = get_invokeid(ctrl); + +- ASN1_ADD_SIMPLE(comp, COMP_TYPE_NFE, buffer, i); +- ASN1_PUSH(compstk, compsp, comp); +- ASN1_ADD_BYTECOMP(comp, (ASN1_CONTEXT_SPECIFIC | ASN1_TAG_0), buffer, i, 0); +- ASN1_ADD_BYTECOMP(comp, (ASN1_CONTEXT_SPECIFIC | ASN1_TAG_2), buffer, i, 0); +- ASN1_FIXUP(compstk, compsp, buffer, i); ++ /* The called.number is the served user */ ++ q931_copy_number_to_rose(ctrl, &msg.args.qsig.MWIDeactivate.served_user_number, ++ &req->called.number); ++ /* ++ * For now, we will just force the numbering plan to unknown to preserve ++ * the original behaviour. ++ */ ++ msg.args.qsig.MWIDeactivate.served_user_number.plan = 0; /* unknown */ + +- ASN1_ADD_BYTECOMP(comp, COMP_TYPE_INTERPRETATION, buffer, i, 0); ++ msg.args.qsig.MWIDeactivate.basic_service = 1; /* speech */ + +- ASN1_ADD_SIMPLE(comp, COMP_TYPE_INVOKE, buffer, i); +- ASN1_PUSH(compstk, compsp, comp); ++ pos = rose_encode_invoke(ctrl, pos, end, &msg); + +- ASN1_ADD_BYTECOMP(comp, ASN1_INTEGER, buffer, i, get_invokeid(pri)); ++ return pos; ++} + +- ASN1_ADD_BYTECOMP(comp, ASN1_INTEGER, buffer, i, (activate) ? SS_MWI_ACTIVATE : SS_MWI_DEACTIVATE); +- ASN1_ADD_SIMPLE(comp, (ASN1_CONSTRUCTOR | ASN1_SEQUENCE), buffer, i); +- ASN1_PUSH(compstk, compsp, comp); +- /* PartyNumber */ +- res = asn1_string_encode((ASN1_CONTEXT_SPECIFIC | ASN1_TAG_0), &buffer[i], sizeof(buffer)-i, destlen, req->called, destlen); +- +- if (res < 0) ++/*! ++ * \brief Encode and queue the Q.SIG MWIActivate/MWIDeactivate invoke message. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param call Call leg to queue message. ++ * \param req Served user setup request information. ++ * \param activate Nonzero to do the activate message. ++ * ++ * \retval 0 on success. ++ * \retval -1 on error. ++ */ ++int mwi_message_send(struct pri *ctrl, q931_call *call, struct pri_sr *req, int activate) ++{ ++ unsigned char buffer[255]; ++ unsigned char *end; ++ ++ if (!req->called.number.valid || !req->called.number.str[0]) { + return -1; +- i += res; ++ } + +- /* Enumeration: basicService */ +- ASN1_ADD_BYTECOMP(comp, ASN1_ENUMERATED, buffer, i, 1 /* contents: Voice */); +- ASN1_FIXUP(compstk, compsp, buffer, i); +- ASN1_FIXUP(compstk, compsp, buffer, i); ++ if (activate) { ++ end = enc_qsig_mwi_activate_message(ctrl, buffer, buffer + sizeof(buffer), req); ++ } else { ++ end = ++ enc_qsig_mwi_deactivate_message(ctrl, buffer, buffer + sizeof(buffer), req); ++ } ++ if (!end) { ++ return -1; ++ } + +- return pri_call_apdu_queue(call, Q931_SETUP, buffer, i, mwi_activate_encode_cb, NULL); ++ return pri_call_apdu_queue(call, Q931_SETUP, buffer, end - buffer, NULL); + } + /* End MWI */ + + /* EECT functions */ +-int eect_initiate_transfer(struct pri *pri, q931_call *c1, q931_call *c2) ++/*! ++ * \internal ++ * \brief Encode the NI2 InitiateTransfer invoke message. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode the facility ie contents. ++ * \param end End of facility ie contents encoding data buffer. ++ * \param call Call leg from which to encode transfer information. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *enc_ni2_initiate_transfer(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, q931_call *call) + { +- int i = 0; +- int res = 0; +- unsigned char buffer[255] = ""; +- short call_reference = c2->cr ^ 0x8000; /* Let's do the trickery to make sure the flag is correct */ +- struct rose_component *comp = NULL, *compstk[10]; +- int compsp = 0; +- static unsigned char op_tag[] = { +- 0x2A, +- 0x86, +- 0x48, +- 0xCE, +- 0x15, +- 0x00, +- 0x08, +- }; ++ struct rose_msg_invoke msg; + +- buffer[i++] = (ASN1_CONTEXT_SPECIFIC | Q932_PROTOCOL_ROSE); ++ pos = facility_encode_header(ctrl, pos, end, NULL); ++ if (!pos) { ++ return NULL; ++ } + +- ASN1_ADD_SIMPLE(comp, COMP_TYPE_INVOKE, buffer, i); +- ASN1_PUSH(compstk, compsp, comp); ++ memset(&msg, 0, sizeof(msg)); ++ msg.operation = ROSE_NI2_InitiateTransfer; ++ msg.invoke_id = get_invokeid(ctrl); ++ /* Let's do the trickery to make sure the flag is correct */ ++ msg.args.ni2.InitiateTransfer.call_reference = call->cr ^ 0x8000; ++ pos = rose_encode_invoke(ctrl, pos, end, &msg); + +- ASN1_ADD_BYTECOMP(comp, ASN1_INTEGER, buffer, i, get_invokeid(pri)); ++ return pos; ++} + +- res = asn1_string_encode(ASN1_OBJECTIDENTIFIER, &buffer[i], sizeof(buffer)-i, sizeof(op_tag), op_tag, sizeof(op_tag)); +- if (res < 0) ++/*! ++ * \brief Start a 2BCT ++ * ++ * \note Called for PRI_SWITCH_NI2, PRI_SWITCH_LUCENT5E, and PRI_SWITCH_ATT4ESS ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param c1 Q.931 call leg 1 ++ * \param c2 Q.931 call leg 2 ++ * ++ * \retval 0 on success. ++ * \retval -1 on error. ++ */ ++int eect_initiate_transfer(struct pri *ctrl, q931_call *c1, q931_call *c2) ++{ ++ unsigned char buffer[255]; ++ unsigned char *end; ++ ++ end = enc_ni2_initiate_transfer(ctrl, buffer, buffer + sizeof(buffer), c2); ++ if (!end) { + return -1; +- i += res; ++ } + +- ASN1_ADD_SIMPLE(comp, (ASN1_SEQUENCE | ASN1_CONSTRUCTOR), buffer, i); +- ASN1_PUSH(compstk, compsp, comp); +- ASN1_ADD_WORDCOMP(comp, ASN1_INTEGER, buffer, i, call_reference); +- ASN1_FIXUP(compstk, compsp, buffer, i); +- ASN1_FIXUP(compstk, compsp, buffer, i); +- +- res = pri_call_apdu_queue(c1, Q931_FACILITY, buffer, i, NULL, NULL); +- if (res) { +- pri_message(pri, "Could not queue APDU in facility message\n"); ++ if (pri_call_apdu_queue(c1, Q931_FACILITY, buffer, end - buffer, NULL)) { ++ pri_message(ctrl, "Could not queue APDU in facility message\n"); + return -1; + } + + /* Remember that if we queue a facility IE for a facility message we + * have to explicitly send the facility message ourselves */ + +- res = q931_facility(c1->pri, c1); +- if (res) { +- pri_message(pri, "Could not schedule facility message for call %d\n", c1->cr); ++ if (q931_facility(c1->pri, c1)) { ++ pri_message(ctrl, "Could not schedule facility message for call %d\n", c1->cr); + return -1; + } + +@@ -1187,1494 +1791,2316 @@ + /* End EECT */ + + /* QSIG CF CallRerouting */ +-int qsig_cf_callrerouting(struct pri *pri, q931_call *c, const char* dest, const char* original, const char* reason) ++/*! ++ * \internal ++ * \brief Encode the Q.SIG CallRerouting invoke message. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode the facility ie contents. ++ * \param end End of facility ie contents encoding data buffer. ++ * \param call Q.931 call leg. ++ * \param calling Call rerouting/deflecting updated caller data. ++ * \param deflection Call rerouting/deflecting redirection data. ++ * \param subscription_option Diverting user subscription option to specify if caller is notified. ++ * ++ * \note ++ * deflection->to is the new called number and must always be present. ++ * \note ++ * subscription option: ++ * noNotification(0), ++ * notificationWithoutDivertedToNr(1), ++ * notificationWithDivertedToNr(2) ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *enc_qsig_call_rerouting(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, q931_call *call, const struct q931_party_id *calling, ++ const struct q931_party_redirecting *deflection, int subscription_option) + { +-/*CallRerouting ::= OPERATION +- -- Sent from the Served User PINX to the Rerouting PINX +- ARGUMENT SEQUENCE +- { reroutingReason DiversionReason, +- originalReroutingReason [0] IMPLICIT DiversionReason OPTIONAL, +- calledAddress Address, +- diversionCounter INTEGER (1..15), +- pSS1InfoElement PSS1InformationElement, +- -- The basic call information elements Bearer capability, High layer compatibility, Low +- -- layer compatibity, Progress indicator and Party category can be embedded in the +- -- pSS1InfoElement in accordance with 6.5.3.1.5 +- lastReroutingNr [1] PresentedNumberUnscreened, +- subscriptionOption [2] IMPLICIT SubscriptionOption, ++ struct fac_extension_header header; ++ struct rose_msg_invoke msg; ++ unsigned char *q931ie_pos; + +- callingPartySubaddress [3] PartySubaddress OPTIONAL, ++ memset(&header, 0, sizeof(header)); ++ header.nfe_present = 1; ++ header.nfe.source_entity = 0; /* endPINX */ ++ header.nfe.destination_entity = 0; /* endPINX */ ++ header.interpretation_present = 1; ++ header.interpretation = 2; /* rejectAnyUnrecognisedInvokePdu */ ++ pos = facility_encode_header(ctrl, pos, end, &header); ++ if (!pos) { ++ return NULL; ++ } + +- callingNumber [4] PresentedNumberScreened, ++ memset(&msg, 0, sizeof(msg)); ++ msg.operation = ROSE_QSIG_CallRerouting; ++ msg.invoke_id = get_invokeid(ctrl); + +- callingName [5] Name OPTIONAL, +- originalCalledNr [6] PresentedNumberUnscreened OPTIONAL, +- redirectingName [7] Name OPTIONAL, +- originalCalledName [8] Name OPTIONAL, +- extension CHOICE { +- [9] IMPLICIT Extension , +- [10] IMPLICIT SEQUENCE OF Extension } OPTIONAL } +-*/ ++ msg.args.qsig.CallRerouting.rerouting_reason = ++ redirectingreason_from_q931(ctrl, deflection->reason); + +- int i = 0, j; +- int res = 0; +- unsigned char buffer[255] = ""; +- int len = 253; +- struct rose_component *comp = NULL, *compstk[10]; +- int compsp = 0; +- static unsigned char op_tag[] = { +- 0x13, +- }; ++ /* calledAddress is the passed in deflection->to address */ ++ q931_copy_address_to_rose(ctrl, &msg.args.qsig.CallRerouting.called, &deflection->to); + +- buffer[i++] = (ASN1_CONTEXT_SPECIFIC | Q932_PROTOCOL_EXTENSIONS); +- /* Interpretation component */ ++ msg.args.qsig.CallRerouting.diversion_counter = deflection->count; + +- ASN1_ADD_SIMPLE(comp, COMP_TYPE_NFE, buffer, i); +- ASN1_PUSH(compstk, compsp, comp); +- ASN1_ADD_BYTECOMP(comp, (ASN1_CONTEXT_SPECIFIC | ASN1_TAG_0), buffer, i, 0); +- ASN1_ADD_BYTECOMP(comp, (ASN1_CONTEXT_SPECIFIC | ASN1_TAG_2), buffer, i, 0); +- ASN1_FIXUP(compstk, compsp, buffer, i); ++ /* pSS1InfoElement */ ++ q931ie_pos = msg.args.qsig.CallRerouting.q931ie_contents; ++ *q931ie_pos++ = 0x04; /* Bearer Capability IE */ ++ *q931ie_pos++ = 0x03; /* len */ ++ *q931ie_pos++ = 0x80 | call->transcapability; /* Rxed transfer capability. */ ++ *q931ie_pos++ = 0x90; /* circuit mode, 64kbit/s */ ++ *q931ie_pos++ = 0xa3; /* level1 protocol, a-law */ ++ *q931ie_pos++ = 0x95; /* locking shift to codeset 5 (national use) */ ++ *q931ie_pos++ = 0x32; /* Unknown ie */ ++ *q931ie_pos++ = 0x01; /* Unknown ie len */ ++ *q931ie_pos++ = 0x81; /* Unknown ie body */ ++ msg.args.qsig.CallRerouting.q931ie.length = q931ie_pos ++ - msg.args.qsig.CallRerouting.q931ie_contents; + +- ASN1_ADD_BYTECOMP(comp, COMP_TYPE_INTERPRETATION, buffer, i, 2); /* reject - to get feedback from QSIG switch */ ++ /* lastReroutingNr is the passed in deflection->from.number */ ++ q931_copy_presented_number_unscreened_to_rose(ctrl, ++ &msg.args.qsig.CallRerouting.last_rerouting, &deflection->from.number); + +- ASN1_ADD_SIMPLE(comp, COMP_TYPE_INVOKE, buffer, i); +- ASN1_PUSH(compstk, compsp, comp); ++ msg.args.qsig.CallRerouting.subscription_option = subscription_option; + +- ASN1_ADD_BYTECOMP(comp, ASN1_INTEGER, buffer, i, get_invokeid(pri)); ++ /* callingNumber is the passed in calling->number */ ++ q931_copy_presented_number_screened_to_rose(ctrl, ++ &msg.args.qsig.CallRerouting.calling, &calling->number); + +- res = asn1_string_encode(ASN1_INTEGER, &buffer[i], sizeof(buffer)-i, sizeof(op_tag), op_tag, sizeof(op_tag)); +- if (res < 0) +- return -1; +- i += res; ++ /* callingPartySubaddress is the passed in calling->subaddress if valid */ ++ q931_copy_subaddress_to_rose(ctrl, &msg.args.qsig.CallRerouting.calling_subaddress, ++ &calling->subaddress); + +- /* call rerouting argument */ +- ASN1_ADD_SIMPLE(comp, (ASN1_CONSTRUCTOR | ASN1_SEQUENCE), buffer, i); +- ASN1_PUSH(compstk, compsp, comp); ++ /* callingName is the passed in calling->name if valid */ ++ if (calling->name.valid) { ++ msg.args.qsig.CallRerouting.calling_name_present = 1; ++ q931_copy_name_to_rose(ctrl, &msg.args.qsig.CallRerouting.calling_name, ++ &calling->name); ++ } + +- /* reroutingReason DiversionReason */ ++ if (1 < deflection->count) { ++ /* originalCalledNr is the deflection->orig_called.number */ ++ msg.args.qsig.CallRerouting.original_called_present = 1; ++ q931_copy_presented_number_unscreened_to_rose(ctrl, ++ &msg.args.qsig.CallRerouting.original_called, ++ &deflection->orig_called.number); + +- if (reason) { +- if (!strcasecmp(reason, "cfu")) +- ASN1_ADD_BYTECOMP(comp, ASN1_ENUMERATED, buffer, i, 1); /* cfu */ +- else if (!strcasecmp(reason, "cfb")) +- ASN1_ADD_BYTECOMP(comp, ASN1_ENUMERATED, buffer, i, 2); /* cfb */ +- else if (!strcasecmp(reason, "cfnr")) +- ASN1_ADD_BYTECOMP(comp, ASN1_ENUMERATED, buffer, i, 3); /* cfnr */ +- } else { +- ASN1_ADD_BYTECOMP(comp, ASN1_ENUMERATED, buffer, i, 0); /* unknown */ ++ msg.args.qsig.CallRerouting.original_rerouting_reason_present = 1; ++ if (deflection->orig_called.number.valid) { ++ msg.args.qsig.CallRerouting.original_rerouting_reason = ++ redirectingreason_from_q931(ctrl, deflection->orig_reason); ++ } else { ++ msg.args.qsig.CallRerouting.original_rerouting_reason = ++ QSIG_DIVERT_REASON_UNKNOWN; ++ } ++ ++ /* originalCalledName is the deflection->orig_called.name */ ++ if (deflection->orig_called.name.valid) { ++ msg.args.qsig.CallRerouting.original_called_name_present = 1; ++ q931_copy_name_to_rose(ctrl, ++ &msg.args.qsig.CallRerouting.original_called_name, ++ &deflection->orig_called.name); ++ } + } + ++ pos = rose_encode_invoke(ctrl, pos, end, &msg); + +- /* calledAddress Address */ +- /* explicit sequence tag for Address */ +- ASN1_ADD_SIMPLE(comp, (ASN1_CONSTRUCTOR | ASN1_SEQUENCE), buffer, i); +- ASN1_PUSH(compstk, compsp, comp); +- /* implicit choice public party number tag */ +- ASN1_ADD_SIMPLE(comp, (ASN1_CONTEXT_SPECIFIC | ASN1_CONSTRUCTOR | ASN1_TAG_1), buffer, i); +- ASN1_PUSH(compstk, compsp, comp); +- /* type of public party number = unknown */ +- ASN1_ADD_BYTECOMP(comp, ASN1_ENUMERATED, buffer, i, 0); +- /* NumberDigits of public party number */ +- j = asn1_string_encode(ASN1_NUMERICSTRING, &buffer[i], len - i, 20, (char*)dest, strlen(dest)); +- if (j < 0) +- return -1; ++ return pos; ++} + +- i += j; +- ASN1_FIXUP(compstk, compsp, buffer, i); +- ASN1_FIXUP(compstk, compsp, buffer, i); ++/*! ++ * \internal ++ * \brief Encode the ETSI CallRerouting invoke message. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode the facility ie contents. ++ * \param end End of facility ie contents encoding data buffer. ++ * \param call Q.931 call leg. ++ * \param calling Call rerouting/deflecting updated caller data. ++ * \param deflection Call rerouting/deflecting redirection data. ++ * \param subscription_option Diverting user subscription option to specify if caller is notified. ++ * ++ * \note ++ * deflection->to is the new called number and must always be present. ++ * \note ++ * subscription option: ++ * noNotification(0), ++ * notificationWithoutDivertedToNr(1), ++ * notificationWithDivertedToNr(2) ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *enc_etsi_call_rerouting(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, q931_call *call, const struct q931_party_id *calling, ++ const struct q931_party_redirecting *deflection, int subscription_option) ++{ ++ struct rose_msg_invoke msg; ++ unsigned char *q931ie_pos; + +- /* diversionCounter INTEGER (1..15) */ +- ASN1_ADD_BYTECOMP(comp, ASN1_INTEGER, buffer, i, 1); ++ pos = facility_encode_header(ctrl, pos, end, NULL); ++ if (!pos) { ++ return NULL; ++ } + +- /* pSS1InfoElement */ +- ASN1_ADD_SIMPLE(comp, (ASN1_APPLICATION | ASN1_TAG_0 ), buffer, i); +- ASN1_PUSH(compstk, compsp, comp); +- buffer[i++] = (0x04); /* Bearer Capability IE */ +- buffer[i++] = (0x03); /* len */ +- buffer[i++] = (0x80); /* ETSI Standard, Speech */ +- buffer[i++] = (0x90); /* circuit mode, 64kbit/s */ +- buffer[i++] = (0xa3); /* level1 protocol, a-law */ +- buffer[i++] = (0x95); /* locking shift to codeset 5 (national use) */ +- buffer[i++] = (0x32); /* Unknown ie */ +- buffer[i++] = (0x01); /* Unknown ie len */ +- buffer[i++] = (0x81); /* Unknown ie body */ +- ASN1_FIXUP(compstk, compsp, buffer, i); ++ memset(&msg, 0, sizeof(msg)); ++ msg.operation = ROSE_ETSI_CallRerouting; ++ msg.invoke_id = get_invokeid(ctrl); + +- /* lastReroutingNr [1]*/ +- /* implicit optional lastReroutingNr tag */ +- ASN1_ADD_SIMPLE(comp, (ASN1_CONTEXT_SPECIFIC | ASN1_CONSTRUCTOR | ASN1_TAG_1), buffer, i); +- ASN1_PUSH(compstk, compsp, comp); ++ msg.args.etsi.CallRerouting.rerouting_reason = ++ redirectingreason_from_q931(ctrl, deflection->reason); + +- /* implicit choice presented number unscreened tag */ +- ASN1_ADD_SIMPLE(comp, (ASN1_CONTEXT_SPECIFIC | ASN1_CONSTRUCTOR | ASN1_TAG_0), buffer, i); +- ASN1_PUSH(compstk, compsp, comp); ++ /* calledAddress is the passed in deflection->to address */ ++ q931_copy_address_to_rose(ctrl, &msg.args.etsi.CallRerouting.called_address, ++ &deflection->to); + +- /* implicit choice public party number tag */ +- ASN1_ADD_SIMPLE(comp, (ASN1_CONTEXT_SPECIFIC | ASN1_CONSTRUCTOR | ASN1_TAG_1), buffer, i); +- ASN1_PUSH(compstk, compsp, comp); +- /* type of public party number = unknown */ +- ASN1_ADD_BYTECOMP(comp, ASN1_ENUMERATED, buffer, i, 0); +- j = asn1_string_encode(ASN1_NUMERICSTRING, &buffer[i], len - i, 20, original?(char*)original:c->callednum, original?strlen(original):strlen(c->callednum)); +- if (j < 0) +- return -1; ++ msg.args.etsi.CallRerouting.rerouting_counter = deflection->count; + +- i += j; +- ASN1_FIXUP(compstk, compsp, buffer, i); +- ASN1_FIXUP(compstk, compsp, buffer, i); +- ASN1_FIXUP(compstk, compsp, buffer, i); ++ /* q931InfoElement */ ++ q931ie_pos = msg.args.etsi.CallRerouting.q931ie_contents; ++ *q931ie_pos++ = 0x04; /* Bearer Capability IE */ ++ *q931ie_pos++ = 0x03; /* len */ ++ *q931ie_pos++ = 0x80 | call->transcapability; /* Rxed transfer capability. */ ++ *q931ie_pos++ = 0x90; /* circuit mode, 64kbit/s */ ++ *q931ie_pos++ = 0xa3; /* level1 protocol, a-law */ ++ msg.args.etsi.CallRerouting.q931ie.length = q931ie_pos ++ - msg.args.etsi.CallRerouting.q931ie_contents; + +- /* subscriptionOption [2]*/ +- /* implicit optional lastReroutingNr tag */ +- ASN1_ADD_BYTECOMP(comp, (ASN1_CONTEXT_SPECIFIC | ASN1_TAG_2), buffer, i, 0); /* noNotification */ ++ /* lastReroutingNr is the passed in deflection->from.number */ ++ q931_copy_presented_number_unscreened_to_rose(ctrl, ++ &msg.args.etsi.CallRerouting.last_rerouting, &deflection->from.number); + +- /* callingNumber [4]*/ +- /* implicit optional callingNumber tag */ +- ASN1_ADD_SIMPLE(comp, (ASN1_CONTEXT_SPECIFIC | ASN1_CONSTRUCTOR | ASN1_TAG_4), buffer, i); +- ASN1_PUSH(compstk, compsp, comp); ++ msg.args.etsi.CallRerouting.subscription_option = subscription_option; + +- /* implicit choice presented number screened tag */ +- ASN1_ADD_SIMPLE(comp, (ASN1_CONTEXT_SPECIFIC | ASN1_CONSTRUCTOR | ASN1_TAG_0), buffer, i); +- ASN1_PUSH(compstk, compsp, comp); ++ /* callingPartySubaddress is the passed in calling->subaddress if valid */ ++ q931_copy_subaddress_to_rose(ctrl, &msg.args.etsi.CallRerouting.calling_subaddress, ++ &calling->subaddress); + +- /* implicit choice presentationAllowedAddress tag */ +- ASN1_ADD_SIMPLE(comp, (ASN1_CONTEXT_SPECIFIC | ASN1_CONSTRUCTOR | ASN1_TAG_1), buffer, i); +- ASN1_PUSH(compstk, compsp, comp); +- /* type of public party number = subscriber number */ +- ASN1_ADD_BYTECOMP(comp, ASN1_ENUMERATED, buffer, i, 4); +- j = asn1_string_encode(ASN1_NUMERICSTRING, &buffer[i], len - i, 20, c->callernum, strlen(c->callernum)); +- if (j < 0) +- return -1; ++ pos = rose_encode_invoke(ctrl, pos, end, &msg); + +- i += j; +- ASN1_FIXUP(compstk, compsp, buffer, i); ++ return pos; ++} + +- /* Screeening Indicator network provided */ +- ASN1_ADD_BYTECOMP(comp, ASN1_ENUMERATED, buffer, i, 3); ++/*! ++ * \internal ++ * \brief Encode the ETSI CallDeflection invoke message. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode the facility ie contents. ++ * \param end End of facility ie contents encoding data buffer. ++ * \param call Q.931 call leg. ++ * \param deflection Call deflection address. ++ * ++ * \note ++ * deflection is the new called number and must always be present. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *enc_etsi_call_deflection(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, q931_call *call, const struct q931_party_id *deflection) ++{ ++ struct rose_msg_invoke msg; + +- ASN1_FIXUP(compstk, compsp, buffer, i); +- ASN1_FIXUP(compstk, compsp, buffer, i); ++ pos = facility_encode_header(ctrl, pos, end, NULL); ++ if (!pos) { ++ return NULL; ++ } + +- /**/ ++ memset(&msg, 0, sizeof(msg)); ++ msg.operation = ROSE_ETSI_CallDeflection; ++ msg.invoke_id = get_invokeid(ctrl); + +- ASN1_FIXUP(compstk, compsp, buffer, i); +- ASN1_FIXUP(compstk, compsp, buffer, i); ++ /* deflectionAddress is the passed in deflection->to address */ ++ q931_copy_address_to_rose(ctrl, &msg.args.etsi.CallDeflection.deflection, ++ deflection); + +- res = pri_call_apdu_queue(c, Q931_FACILITY, buffer, i, NULL, NULL); +- if (res) { +- pri_message(pri, "Could not queue ADPU in facility message\n"); ++ msg.args.etsi.CallDeflection.presentation_allowed_to_diverted_to_user_present = 1; ++ switch (deflection->number.presentation & PRI_PRES_RESTRICTION) { ++ case PRI_PRES_ALLOWED: ++ msg.args.etsi.CallDeflection.presentation_allowed_to_diverted_to_user = 1; ++ break; ++ default: ++ case PRI_PRES_UNAVAILABLE: ++ case PRI_PRES_RESTRICTED: ++ break; ++ } ++ ++ pos = rose_encode_invoke(ctrl, pos, end, &msg); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Encode and queue the CallRerouting/CallDeflection message. ++ * ++ * \param ctrl D channel controller. ++ * \param call Q.931 call leg. ++ * \param caller Call rerouting/deflecting updated caller data. (NULL if data not updated.) ++ * \param deflection Call rerouting/deflecting redirection data. ++ * \param subscription_option Diverting user subscription option to specify if caller is notified. ++ * ++ * \note ++ * deflection->to is the new called number and must always be present. ++ * \note ++ * subscription option: ++ * noNotification(0), ++ * notificationWithoutDivertedToNr(1), ++ * notificationWithDivertedToNr(2) ++ * ++ * \retval 0 on success. ++ * \retval -1 on error. ++ */ ++static int rose_reroute_request_encode(struct pri *ctrl, q931_call *call, ++ const struct q931_party_id *caller, const struct q931_party_redirecting *deflection, ++ int subscription_option) ++{ ++ unsigned char buffer[256]; ++ unsigned char *end; ++ ++ if (!caller) { ++ /* ++ * We are deflecting an incoming call back to the network. ++ * Therefore, the Caller-ID is the remote party. ++ */ ++ caller = &call->remote_id; ++ } ++ ++ switch (ctrl->switchtype) { ++ case PRI_SWITCH_EUROISDN_E1: ++ case PRI_SWITCH_EUROISDN_T1: ++ if (q931_is_ptmp(ctrl)) { ++ end = ++ enc_etsi_call_deflection(ctrl, buffer, buffer + sizeof(buffer), call, ++ &deflection->to); ++ } else { ++ end = ++ enc_etsi_call_rerouting(ctrl, buffer, buffer + sizeof(buffer), call, ++ caller, deflection, subscription_option); ++ } ++ break; ++ case PRI_SWITCH_QSIG: ++ end = ++ enc_qsig_call_rerouting(ctrl, buffer, buffer + sizeof(buffer), call, caller, ++ deflection, subscription_option); ++ break; ++ default: + return -1; + } ++ if (!end) { ++ return -1; ++ } + +- /* Remember that if we queue a facility IE for a facility message we +- * have to explicitly send the facility message ourselves */ ++ return pri_call_apdu_queue(call, Q931_FACILITY, buffer, end - buffer, NULL); ++} + +- res = q931_facility(c->pri, c); +- if (res) { +- pri_message(pri, "Could not schedule facility message for call %d\n", c->cr); ++/*! ++ * \brief Send the CallRerouting/CallDeflection message. ++ * ++ * \param ctrl D channel controller. ++ * \param call Q.931 call leg. ++ * \param caller Call rerouting/deflecting updated caller data. (NULL if data not updated.) ++ * \param deflection Call rerouting/deflecting redirection data. ++ * \param subscription_option Diverting user subscription option to specify if caller is notified. ++ * ++ * \note ++ * deflection->to is the new called number and must always be present. ++ * \note ++ * subscription option: ++ * noNotification(0), ++ * notificationWithoutDivertedToNr(1), ++ * notificationWithDivertedToNr(2) ++ * ++ * \retval 0 on success. ++ * \retval -1 on error. ++ */ ++int send_reroute_request(struct pri *ctrl, q931_call *call, ++ const struct q931_party_id *caller, const struct q931_party_redirecting *deflection, ++ int subscription_option) ++{ ++ if (!deflection->to.number.str[0]) { ++ /* Must have a deflect to number. That is the point of deflection. */ + return -1; + } ++ if (rose_reroute_request_encode(ctrl, call, caller, deflection, subscription_option) ++ || q931_facility(ctrl, call)) { ++ pri_message(ctrl, ++ "Could not schedule facility message for CallRerouting/CallDeflection message.\n"); ++ return -1; ++ } + + return 0; + } ++ ++/*! ++ * \brief Send the Q.SIG CallRerouting invoke message. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param call Q.931 call leg. ++ * \param dest Destination number. ++ * \param original Original called number. ++ * \param reason Rerouting reason: cfu, cfb, cfnr ++ * ++ * \retval 0 on success. ++ * \retval -1 on error. ++ */ ++int qsig_cf_callrerouting(struct pri *ctrl, q931_call *call, const char *dest, ++ const char *original, const char *reason) ++{ ++ struct q931_party_redirecting reroute; ++ ++ q931_party_redirecting_init(&reroute); ++ ++ /* Rerouting to the dest number. */ ++ reroute.to.number.valid = 1; ++ reroute.to.number.plan = (PRI_TON_UNKNOWN << 4) | PRI_NPI_E163_E164; ++ reroute.to.number.presentation = PRI_PRES_ALLOWED | PRI_PRES_USER_NUMBER_UNSCREENED; ++ libpri_copy_string(reroute.to.number.str, dest, sizeof(reroute.to.number.str)); ++ ++ /* Rerouting from the original number. */ ++ if (original) { ++ reroute.from.number.valid = 1; ++ reroute.from.number.plan = (PRI_TON_UNKNOWN << 4) | PRI_NPI_E163_E164; ++ libpri_copy_string(reroute.from.number.str, original, sizeof(reroute.from.number.str)); ++ } else { ++ q931_party_address_to_id(&reroute.from, &call->called); ++ } ++ reroute.from.number.presentation = PRI_PRES_ALLOWED | PRI_PRES_USER_NUMBER_UNSCREENED; ++ ++ /* Decode the rerouting reason. */ ++ reroute.reason = PRI_REDIR_UNKNOWN; ++ if (!reason) { ++ /* No reason for rerouting given. */ ++ } else if (!strcasecmp(reason, "cfu")) { ++ reroute.reason = PRI_REDIR_UNCONDITIONAL; ++ } else if (!strcasecmp(reason, "cfb")) { ++ reroute.reason = PRI_REDIR_FORWARD_ON_BUSY; ++ } else if (!strcasecmp(reason, "cfnr")) { ++ reroute.reason = PRI_REDIR_FORWARD_ON_NO_REPLY; ++ } ++ ++ reroute.count = (call->redirecting.count < PRI_MAX_REDIRECTS) ++ ? call->redirecting.count + 1 : PRI_MAX_REDIRECTS; ++ ++ if (!call->redirecting.orig_called.number.valid) { ++ /* ++ * Since we do not already have an originally called party, we ++ * must either be the first redirected to party or this call ++ * has not been redirected before. ++ * ++ * Preserve who redirected to us as the originally called party. ++ */ ++ reroute.orig_called = call->redirecting.from; ++ reroute.orig_reason = call->redirecting.reason; ++ } else { ++ reroute.orig_called = call->redirecting.orig_called; ++ reroute.orig_reason = call->redirecting.orig_reason; ++ } ++ ++ return send_reroute_request(ctrl, call, NULL, &reroute, 0 /* noNotification */); ++} + /* End QSIG CC-CallRerouting */ + +-static int anfpr_pathreplacement_respond(struct pri *pri, q931_call *call, q931_ie *ie) ++/* ++ * From Mantis issue 7778 description: (ETS 300 258, ISO 13863) ++ * After both legs of the call are setup and Asterisk has a successful "tromboned" or bridged call ... ++ * Asterisk sees both 'B' channels (from trombone) are on same PRI/technology and initiates "Path Replacement" events ++ * a. Asterisk sends "Transfer Complete" messages to both call legs ++ * b. QSIG Switch sends "PathReplacement" message on one of the legs (random 1-10sec timer expires - 1st leg to send is it!) ++ * c. Asterisk rebroadcasts "PathReplacement" message to other call leg ++ * d. QSIG Switch sends "Disconnect" message on one of the legs (same random timer sequence as above) ++ * e. Asterisk rebroadcasts "Disconnect" message to other call leg ++ * f. QSIG Switch disconnects Asterisk call legs - callers are now within QSIG switch ++ * ++ * Just need to resend the message to the other tromboned leg of the call. ++ */ ++static int anfpr_pathreplacement_respond(struct pri *ctrl, q931_call *call, q931_ie *ie) + { + int res; +- +- res = pri_call_apdu_queue_cleanup(call->bridged_call); +- if (res) { +- pri_message(pri, "Could not Clear queue ADPU\n"); +- return -1; +- } +- ++ ++ pri_call_apdu_queue_cleanup(call->bridged_call); ++ + /* Send message */ +- res = pri_call_apdu_queue(call->bridged_call, Q931_FACILITY, ie->data, ie->len, NULL, NULL); ++ res = pri_call_apdu_queue(call->bridged_call, Q931_FACILITY, ie->data, ie->len, NULL); + if (res) { +- pri_message(pri, "Could not queue ADPU in facility message\n"); +- return -1; ++ pri_message(ctrl, "Could not queue ADPU in facility message\n"); ++ return -1; + } +- ++ + /* Remember that if we queue a facility IE for a facility message we + * have to explicitly send the facility message ourselves */ +- ++ + res = q931_facility(call->bridged_call->pri, call->bridged_call); + if (res) { +- pri_message(pri, "Could not schedule facility message for call %d\n", call->bridged_call->cr); ++ pri_message(ctrl, "Could not schedule facility message for call %d\n", ++ call->bridged_call->cr); + return -1; + } + + return 0; + } ++ + /* AFN-PR */ +-int anfpr_initiate_transfer(struct pri *pri, q931_call *c1, q931_call *c2) ++/*! ++ * \brief Start a Q.SIG path replacement. ++ * ++ * \note Called for PRI_SWITCH_QSIG ++ * ++ * \note Did all the tests to see if we're on the same PRI and ++ * are on a compatible switchtype. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param c1 Q.931 call leg 1 ++ * \param c2 Q.931 call leg 2 ++ * ++ * \retval 0 on success. ++ * \retval -1 on error. ++ */ ++int anfpr_initiate_transfer(struct pri *ctrl, q931_call *c1, q931_call *c2) + { +- /* Did all the tests to see if we're on the same PRI and +- * are on a compatible switchtype */ +- /* TODO */ +- int i = 0; +- int res = 0; +- unsigned char buffer[255] = ""; +- unsigned short call_reference = c2->cr; +- struct rose_component *comp = NULL, *compstk[10]; +- unsigned char buffer2[255] = ""; +- int compsp = 0; +- static unsigned char op_tag[] = { +- 0x0C, +- }; +- +- /* Channel 1 */ +- buffer[i++] = (ASN1_CONTEXT_SPECIFIC | Q932_PROTOCOL_EXTENSIONS); +- +- ASN1_ADD_SIMPLE(comp, COMP_TYPE_NFE, buffer, i); +- ASN1_PUSH(compstk, compsp, comp); +- ASN1_ADD_BYTECOMP(comp, (ASN1_CONTEXT_SPECIFIC | ASN1_TAG_0), buffer, i, 0); +- ASN1_ADD_BYTECOMP(comp, (ASN1_CONTEXT_SPECIFIC | ASN1_TAG_2), buffer, i, 0); +- ASN1_FIXUP(compstk, compsp, buffer, i); +- +- /* Interpretation component */ +- ASN1_ADD_BYTECOMP(comp, COMP_TYPE_INTERPRETATION, buffer, i, 2); /* reject - to get feedback from QSIG switch */ +- +- ASN1_ADD_SIMPLE(comp, COMP_TYPE_INVOKE, buffer, i); +- ASN1_PUSH(compstk, compsp, comp); +- +- ASN1_ADD_BYTECOMP(comp, ASN1_INTEGER, buffer, i, get_invokeid(pri)); +- +- res = asn1_string_encode(ASN1_INTEGER, &buffer[i], sizeof(buffer)-i, sizeof(op_tag), op_tag, sizeof(op_tag)); +- if (res < 0) ++ unsigned char buffer[255]; ++ unsigned char *pos; ++ unsigned char *end; ++ int res; ++ struct fac_extension_header header; ++ struct rose_msg_invoke msg; ++ ++ end = buffer + sizeof(buffer); ++ ++ memset(&header, 0, sizeof(header)); ++ header.nfe_present = 1; ++ header.nfe.source_entity = 0; /* endPINX */ ++ header.nfe.destination_entity = 0; /* endPINX */ ++ header.interpretation_present = 1; ++ header.interpretation = 2; /* rejectAnyUnrecognisedInvokePdu */ ++ pos = facility_encode_header(ctrl, buffer, end, &header); ++ if (!pos) { + return -1; +- i += res; +- +- ASN1_ADD_SIMPLE(comp, (ASN1_SEQUENCE | ASN1_CONSTRUCTOR), buffer, i); +- ASN1_PUSH(compstk, compsp, comp); +- buffer[i++] = (0x0a);/* Enumeration endDesignation */ +- buffer[i++] = (0x01);/* Len */ +- buffer[i++] = (0x00);/* primaryEnd */ +- buffer[i++] = (0x81);/* redirectionNumber = presentationRestricted */ +- buffer[i++] = (0x00);/* Len */ +- buffer[i++] = (0x0a);/* Enumeration callStatus */ +- buffer[i++] = (0x01);/* Len */ +- buffer[i++] = (0x01);/* alerting */ ++ } + +- /* +- * Where does this element come from? It is not in Q.SIG ECMA-178. +- * We send this but we will not accept it. +- * This seems to be a cut and paste error from eect_initiate_transfer(). +- */ +- ASN1_ADD_WORDCOMP(comp, ASN1_INTEGER, buffer, i, call_reference); ++ memset(&msg, 0, sizeof(msg)); ++ msg.operation = ROSE_QSIG_CallTransferComplete; ++ msg.invoke_id = get_invokeid(ctrl); ++ msg.args.qsig.CallTransferComplete.end_designation = 0; /* primaryEnd */ ++ msg.args.qsig.CallTransferComplete.redirection.presentation = 1; /* presentationRestricted */ ++ msg.args.qsig.CallTransferComplete.call_status = 1; /* alerting */ ++ pos = rose_encode_invoke(ctrl, pos, end, &msg); ++ if (!pos) { ++ return -1; ++ } + +- ASN1_FIXUP(compstk, compsp, buffer, i); +- ASN1_FIXUP(compstk, compsp, buffer, i); +- +- res = pri_call_apdu_queue(c1, Q931_FACILITY, buffer, i, NULL, NULL); ++ res = pri_call_apdu_queue(c1, Q931_FACILITY, buffer, pos - buffer, NULL); + if (res) { +- pri_message(pri, "Could not queue ADPU in facility message\n"); ++ pri_message(ctrl, "Could not queue ADPU in facility message\n"); + return -1; + } +- ++ + /* Remember that if we queue a facility IE for a facility message we + * have to explicitly send the facility message ourselves */ +- ++ + res = q931_facility(c1->pri, c1); + if (res) { +- pri_message(pri, "Could not schedule facility message for call %d\n", c1->cr); ++ pri_message(ctrl, "Could not schedule facility message for call %d\n", c1->cr); + return -1; + } +- +- /* Channel 2 */ +- i = 0; +- res = 0; +- compsp = 0; +- +- buffer2[i++] = (ASN1_CONTEXT_SPECIFIC | Q932_PROTOCOL_EXTENSIONS); +- +- ASN1_ADD_SIMPLE(comp, COMP_TYPE_NFE, buffer2, i); +- ASN1_PUSH(compstk, compsp, comp); +- ASN1_ADD_BYTECOMP(comp, (ASN1_CONTEXT_SPECIFIC | ASN1_TAG_0), buffer2, i, 0); +- ASN1_ADD_BYTECOMP(comp, (ASN1_CONTEXT_SPECIFIC | ASN1_TAG_2), buffer2, i, 0); +- ASN1_FIXUP(compstk, compsp, buffer2, i); +- +- /* Interpretation component */ +- ASN1_ADD_BYTECOMP(comp, COMP_TYPE_INTERPRETATION, buffer2, i, 2); /* reject */ +- +- ASN1_ADD_SIMPLE(comp, COMP_TYPE_INVOKE, buffer2, i); +- ASN1_PUSH(compstk, compsp, comp); +- +- ASN1_ADD_BYTECOMP(comp, ASN1_INTEGER, buffer2, i, get_invokeid(pri)); +- +- res = asn1_string_encode(ASN1_INTEGER, &buffer2[i], sizeof(buffer2)-i, sizeof(op_tag), op_tag, sizeof(op_tag)); +- if (res < 0) ++ ++ /* Reuse the previous message header */ ++ pos = facility_encode_header(ctrl, buffer, end, &header); ++ if (!pos) { + return -1; +- i += res; +- +- ASN1_ADD_SIMPLE(comp, (ASN1_SEQUENCE | ASN1_CONSTRUCTOR), buffer2, i); +- ASN1_PUSH(compstk, compsp, comp); +- buffer2[i++] = (0x0a);/* Enumeration endDesignation */ +- buffer2[i++] = (0x01);/* Len */ +- buffer2[i++] = (0x01);/* secondaryEnd */ +- buffer2[i++] = (0x81);/* redirectionNumber = presentationRestricted */ +- buffer2[i++] = (0x00);/* Len */ +- buffer2[i++] = (0x0a);/* Enumeration callStatus */ +- buffer2[i++] = (0x01);/* Len */ +- buffer2[i++] = (0x01);/* alerting */ ++ } + +- /* +- * Where does this element come from? It is not in Q.SIG ECMA-178. +- * We send this but we will not accept it. +- * This seems to be a cut and paste error from eect_initiate_transfer(). +- */ +- ASN1_ADD_WORDCOMP(comp, ASN1_INTEGER, buffer2, i, call_reference); ++ /* Update the previous message */ ++ msg.invoke_id = get_invokeid(ctrl); ++ msg.args.qsig.CallTransferComplete.end_designation = 1; /* secondaryEnd */ ++ pos = rose_encode_invoke(ctrl, pos, end, &msg); ++ if (!pos) { ++ return -1; ++ } + +- ASN1_FIXUP(compstk, compsp, buffer2, i); +- ASN1_FIXUP(compstk, compsp, buffer2, i); +- +- +- res = pri_call_apdu_queue(c2, Q931_FACILITY, buffer2, i, NULL, NULL); ++ res = pri_call_apdu_queue(c2, Q931_FACILITY, buffer, pos - buffer, NULL); + if (res) { +- pri_message(pri, "Could not queue ADPU in facility message\n"); ++ pri_message(ctrl, "Could not queue ADPU in facility message\n"); + return -1; + } +- ++ + /* Remember that if we queue a facility IE for a facility message we + * have to explicitly send the facility message ourselves */ +- ++ + res = q931_facility(c2->pri, c2); + if (res) { +- pri_message(pri, "Could not schedule facility message for call %d\n", c1->cr); ++ pri_message(ctrl, "Could not schedule facility message for call %d\n", c2->cr); + return -1; + } +- ++ + return 0; + } + /* End AFN-PR */ + + /* AOC */ +-static int aoc_aoce_charging_request_decode(struct pri *pri, q931_call *call, unsigned char *data, int len) ++/*! ++ * \internal ++ * \brief Encode the ETSI AOCEChargingUnit invoke message. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode the facility ie contents. ++ * \param end End of facility ie contents encoding data buffer. ++ * \param chargedunits Number of units charged to encode. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *enc_etsi_aoce_charging_unit(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, long chargedunits) + { +- int chargingcase = -1; +- unsigned char *vdata = data; +- struct rose_component *comp = NULL; +- int pos1 = 0; ++ struct rose_msg_invoke msg; + +- if (pri->debug & PRI_DEBUG_AOC) +- dump_apdu (pri, data, len); ++ pos = facility_encode_header(ctrl, pos, end, NULL); ++ if (!pos) { ++ return NULL; ++ } + +- do { +- GET_COMPONENT(comp, pos1, vdata, len); +- CHECK_COMPONENT(comp, ASN1_ENUMERATED, "!! Invalid AOC Charging Request argument. Expected Enumerated (0x0A) but Received 0x%02X\n"); +- ASN1_GET_INTEGER(comp, chargingcase); +- if (chargingcase >= 0 && chargingcase <= 2) { +- if (pri->debug & PRI_DEBUG_APDU) +- pri_message(pri, "Channel %d/%d, Call %d - received AOC charging request - charging case: %i\n", +- call->ds1no, call->channelno, call->cr, chargingcase); +- } else { +- pri_message(pri, "!! unkown AOC ChargingCase: 0x%02X", chargingcase); +- chargingcase = -1; +- } +- NEXT_COMPONENT(comp, pos1); +- } while (pos1 < len); +- if (pos1 < len) { +- pri_message(pri, "!! Only reached position %i in %i bytes long AOC-E structure:", pos1, len ); +- dump_apdu (pri, data, len); +- return -1; /* Aborted before */ ++ memset(&msg, 0, sizeof(msg)); ++ msg.operation = ROSE_ETSI_AOCEChargingUnit; ++ msg.invoke_id = get_invokeid(ctrl); ++ msg.args.etsi.AOCEChargingUnit.type = 1; /* charging_unit */ ++ if (chargedunits <= 0) { ++ msg.args.etsi.AOCEChargingUnit.charging_unit.free_of_charge = 1; ++ } else { ++ msg.args.etsi.AOCEChargingUnit.charging_unit.specific.recorded.num_records = 1; ++ msg.args.etsi.AOCEChargingUnit.charging_unit.specific.recorded.list[0]. ++ number_of_units = chargedunits; + } +- return 0; ++ pos = rose_encode_invoke(ctrl, pos, end, &msg); ++ ++ return pos; + } +- + +-static int aoc_aoce_charging_unit_decode(struct pri *pri, q931_call *call, unsigned char *data, int len) ++/*! ++ * \internal ++ * \brief Send the ETSI AOCEChargingUnit invoke message. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param call Call leg from which to encode AOC. ++ * \param chargedunits Number of units charged to encode. ++ * ++ * \retval 0 on success. ++ * \retval -1 on error. ++ */ ++static int aoc_aoce_charging_unit_encode(struct pri *ctrl, q931_call *call, ++ long chargedunits) + { +- long chargingunits = 0, chargetype = -1, temp, chargeIdentifier = -1; +- unsigned char *vdata = data; +- struct rose_component *comp1 = NULL, *comp2 = NULL, *comp3 = NULL; +- int pos1 = 0, pos2, pos3, sublen2, sublen3; +- struct addressingdataelements_presentednumberunscreened chargednr; ++ unsigned char buffer[255]; ++ unsigned char *end; + +- if (pri->debug & PRI_DEBUG_AOC) +- dump_apdu (pri, data, len); ++ /* sample data: [ 91 a1 12 02 02 3a 78 02 01 24 30 09 30 07 a1 05 30 03 02 01 01 ] */ + +- do { +- GET_COMPONENT(comp1, pos1, vdata, len); /* AOCEChargingUnitInfo */ +- CHECK_COMPONENT(comp1, ASN1_SEQUENCE, "!! Invalid AOC-E Charging Unit argument. Expected Sequence (0x30) but Received 0x%02X\n"); +- SUB_COMPONENT(comp1, pos1); +- GET_COMPONENT(comp1, pos1, vdata, len); +- switch (comp1->type) { +- case (ASN1_SEQUENCE | ASN1_CONSTRUCTOR): /* specificChargingUnits */ +- sublen2 = comp1->len; +- pos2 = pos1; +- comp2 = comp1; +- SUB_COMPONENT(comp2, pos2); +- do { +- GET_COMPONENT(comp2, pos2, vdata, len); +- switch (comp2->type) { +- case (ASN1_CONTEXT_SPECIFIC | ASN1_CONSTRUCTOR | ASN1_TAG_1): /* RecordedUnitsList (0xA1) */ +- SUB_COMPONENT(comp2, pos2); +- GET_COMPONENT(comp2, pos2, vdata, len); +- CHECK_COMPONENT(comp2, ASN1_SEQUENCE, "!! Invalid AOC-E Charging Unit argument. Expected Sequence (0x30) but received 0x02%X\n"); /* RecordedUnits */ +- sublen3 = pos2 + comp2->len; +- pos3 = pos2; +- comp3 = comp2; +- SUB_COMPONENT(comp3, pos3); +- do { +- GET_COMPONENT(comp3, pos3, vdata, len); +- switch (comp3->type) { +- case ASN1_INTEGER: /* numberOfUnits */ +- ASN1_GET_INTEGER(comp3, temp); +- chargingunits += temp; +- case ASN1_NULL: /* notAvailable */ +- break; +- default: +- pri_message(pri, "!! Don't know how to handle 0x%02X in AOC-E RecordedUnits\n", comp3->type); +- } +- NEXT_COMPONENT(comp3, pos3); +- } while (pos3 < sublen3); +- if (pri->debug & PRI_DEBUG_AOC) +- pri_message(pri, "Channel %d/%d, Call %d - received AOC-E charging: %i unit%s\n", +- call->ds1no, call->channelno, call->cr, chargingunits, (chargingunits == 1) ? "" : "s"); +- break; +- case (ASN1_CONTEXT_SPECIFIC | ASN1_CONSTRUCTOR | ASN1_TAG_2): /* AOCEBillingID (0xA2) */ +- SUB_COMPONENT(comp2, pos2); +- GET_COMPONENT(comp2, pos2, vdata, len); +- ASN1_GET_INTEGER(comp2, chargetype); +- pri_message(pri, "!! not handled: Channel %d/%d, Call %d - received AOC-E billing ID: %i\n", +- call->ds1no, call->channelno, call->cr, chargetype); +- break; +- default: +- pri_message(pri, "!! Don't know how to handle 0x%02X in AOC-E RecordedUnitsList\n", comp2->type); +- } +- NEXT_COMPONENT(comp2, pos2); +- } while (pos2 < sublen2); +- break; +- case (ASN1_CONTEXT_SPECIFIC | ASN1_TAG_1): /* freeOfCharge (0x81) */ +- if (pri->debug & PRI_DEBUG_AOC) +- pri_message(pri, "Channel %d/%d, Call %d - received AOC-E free of charge\n", call->ds1no, call->channelno, call->cr); +- chargingunits = 0; +- break; +- default: +- pri_message(pri, "!! Invalid AOC-E specificChargingUnits. Expected Sequence (0x30) or Object Identifier (0x81/0x01) but received 0x%02X\n", comp1->type); +- } +- NEXT_COMPONENT(comp1, pos1); +- GET_COMPONENT(comp1, pos1, vdata, len); /* get optional chargingAssociation. will 'break' when reached end of structure */ +- switch (comp1->type) { +- /* TODO: charged number is untested - please report! */ +- case (ASN1_CONTEXT_SPECIFIC | ASN1_CONSTRUCTOR | ASN1_TAG_0): /* chargedNumber (0xA0) */ +- if(rose_presented_number_unscreened_decode(pri, call, comp1->data, comp1->len, &chargednr) != 0) +- return -1; +- pri_message(pri, "!! not handled: Received ChargedNr '%s' \n", chargednr.partyaddress); +- pri_message(pri, " ton = %d, pres = %d, npi = %d\n", chargednr.ton, chargednr.pres, chargednr.npi); +- break; +- case ASN1_INTEGER: +- ASN1_GET_INTEGER(comp1, chargeIdentifier); +- break; +- default: +- pri_message(pri, "!! Invalid AOC-E chargingAssociation. Expected Object Identifier (0xA0) or Integer (0x02) but received 0x%02X\n", comp1->type); +- } +- NEXT_COMPONENT(comp1, pos1); +- } while (pos1 < len); ++ end = ++ enc_etsi_aoce_charging_unit(ctrl, buffer, buffer + sizeof(buffer), chargedunits); ++ if (!end) { ++ return -1; ++ } + +- if (pos1 < len) { +- pri_message(pri, "!! Only reached position %i in %i bytes long AOC-E structure:", pos1, len ); +- dump_apdu (pri, data, len); +- return -1; /* oops - aborted before */ ++ /* Remember that if we queue a facility IE for a facility message we ++ * have to explicitly send the facility message ourselves */ ++ if (pri_call_apdu_queue(call, Q931_FACILITY, buffer, end - buffer, NULL) ++ || q931_facility(call->pri, call)) { ++ pri_message(ctrl, "Could not schedule facility message for call %d\n", call->cr); ++ return -1; + } +- call->aoc_units = chargingunits; +- ++ + return 0; + } ++/* End AOC */ + +-static int aoc_aoce_charging_unit_encode(struct pri *pri, q931_call *c, long chargedunits) ++/* ===== Call Transfer Supplementary Service (ECMA-178) ===== */ ++ ++/*! ++ * \internal ++ * \brief Encode the Q.SIG CallTransferComplete invoke message. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode the facility ie contents. ++ * \param end End of facility ie contents encoding data buffer. ++ * \param call Call leg from which to encode call transfer. ++ * \param call_status TRUE if call is alerting. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *enc_qsig_call_transfer_complete(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, q931_call *call, int call_status) + { +- /* sample data: [ 91 a1 12 02 02 3a 78 02 01 24 30 09 30 07 a1 05 30 03 02 01 01 ] */ +- int i = 0, res = 0, compsp = 0; +- unsigned char buffer[255] = ""; +- struct rose_component *comp = NULL, *compstk[10]; ++ struct fac_extension_header header; ++ struct rose_msg_invoke msg; + +- /* ROSE protocol (0x91)*/ +- buffer[i++] = (ASN1_CONTEXT_SPECIFIC | Q932_PROTOCOL_ROSE); ++ memset(&header, 0, sizeof(header)); ++ header.nfe_present = 1; ++ header.nfe.source_entity = 0; /* endPINX */ ++ header.nfe.destination_entity = 0; /* endPINX */ ++ header.interpretation_present = 1; ++ header.interpretation = 0; /* discardAnyUnrecognisedInvokePdu */ ++ pos = facility_encode_header(ctrl, pos, end, &header); ++ if (!pos) { ++ return NULL; ++ } + +- /* ROSE Component (0xA1,len)*/ +- ASN1_ADD_SIMPLE(comp, COMP_TYPE_INVOKE, buffer, i); +- ASN1_PUSH(compstk, compsp, comp); ++ memset(&msg, 0, sizeof(msg)); ++ msg.operation = ROSE_QSIG_CallTransferComplete; ++ msg.invoke_id = get_invokeid(ctrl); ++ msg.args.qsig.CallTransferComplete.end_designation = 0; /* primaryEnd */ + +- /* ROSE invokeId component (0x02,len,id)*/ +- ASN1_ADD_WORDCOMP(comp, INVOKE_IDENTIFIER, buffer, i, ++pri->last_invoke); ++ /* redirectionNumber is the local_id.number */ ++ q931_copy_presented_number_screened_to_rose(ctrl, ++ &msg.args.qsig.CallTransferComplete.redirection, &call->local_id.number); + +- /* ROSE operationId component (0x02,0x01,0x24)*/ +- ASN1_ADD_BYTECOMP(comp, ASN1_INTEGER, buffer, i, ROSE_AOC_AOCE_CHARGING_UNIT); ++ /* redirectionName is the local_id.name */ ++ if (call->local_id.name.valid) { ++ msg.args.qsig.CallTransferComplete.redirection_name_present = 1; ++ q931_copy_name_to_rose(ctrl, ++ &msg.args.qsig.CallTransferComplete.redirection_name, ++ &call->local_id.name); ++ } + +- /* AOCEChargingUnitInfo (0x30,len) */ +- ASN1_ADD_SIMPLE(comp, (ASN1_CONSTRUCTOR | ASN1_SEQUENCE), buffer, i); +- ASN1_PUSH(compstk, compsp, comp); ++ if (call_status) { ++ msg.args.qsig.CallTransferComplete.call_status = 1; /* alerting */ ++ } ++ pos = rose_encode_invoke(ctrl, pos, end, &msg); + +- if (chargedunits > 0) { +- /* SpecificChargingUnits (0x30,len) */ +- ASN1_ADD_SIMPLE(comp, (ASN1_CONSTRUCTOR | ASN1_SEQUENCE), buffer, i); +- ASN1_PUSH(compstk, compsp, comp); ++ return pos; ++} + +- /* RecordedUnitsList (0xA1,len) */ +- ASN1_ADD_SIMPLE(comp, (ASN1_CONTEXT_SPECIFIC | ASN1_CONSTRUCTOR | ASN1_TAG_1), buffer, i); +- ASN1_PUSH(compstk, compsp, comp); +- +- /* RecordedUnits (0x30,len) */ +- ASN1_ADD_SIMPLE(comp, (ASN1_CONSTRUCTOR | ASN1_SEQUENCE), buffer, i); +- ASN1_PUSH(compstk, compsp, comp); +- +- /* NumberOfUnits (0x02,len,charge) */ +- ASN1_ADD_BYTECOMP(comp, ASN1_INTEGER, buffer, i, chargedunits); ++/*! ++ * \internal ++ * \brief Encode the ETSI EctInform invoke message. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode the facility ie contents. ++ * \param end End of facility ie contents encoding data buffer. ++ * \param call Call leg from which to encode inform message. ++ * \param call_status TRUE if call is alerting. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *enc_etsi_ect_inform(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, q931_call *call, int call_status) ++{ ++ struct rose_msg_invoke msg; + +- ASN1_FIXUP(compstk, compsp, buffer, i); +- ASN1_FIXUP(compstk, compsp, buffer, i); +- ASN1_FIXUP(compstk, compsp, buffer, i); +- } else { +- /* freeOfCharge (0x81,0) */ +- ASN1_ADD_SIMPLE(comp, (ASN1_CONTEXT_SPECIFIC | ASN1_TAG_1), buffer, i); ++ pos = facility_encode_header(ctrl, pos, end, NULL); ++ if (!pos) { ++ return NULL; + } +- ASN1_FIXUP(compstk, compsp, buffer, i); +- ASN1_FIXUP(compstk, compsp, buffer, i); +- +- if (pri->debug & PRI_DEBUG_AOC) +- dump_apdu (pri, buffer, i); +- +- /* code below is untested */ +- res = pri_call_apdu_queue(c, Q931_FACILITY, buffer, i, NULL, NULL); +- if (res) { +- pri_message(pri, "Could not queue APDU in facility message\n"); +- return -1; ++ ++ memset(&msg, 0, sizeof(msg)); ++ msg.operation = ROSE_ETSI_EctInform; ++ msg.invoke_id = get_invokeid(ctrl); ++ ++ if (!call_status) { ++ msg.args.etsi.EctInform.status = 1;/* active */ ++ ++ /* ++ * EctInform(active) contains the redirectionNumber ++ * redirectionNumber is the local_id.number ++ */ ++ msg.args.etsi.EctInform.redirection_present = 1; ++ q931_copy_presented_number_unscreened_to_rose(ctrl, ++ &msg.args.etsi.EctInform.redirection, &call->local_id.number); + } + +- /* Remember that if we queue a facility IE for a facility message we +- * have to explicitly send the facility message ourselves */ +- res = q931_facility(c->pri, c); +- if (res) { +- pri_message(pri, "Could not schedule facility message for call %d\n", c->cr); ++ pos = rose_encode_invoke(ctrl, pos, end, &msg); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Encode and queue the CallTransferComplete/EctInform invoke message. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param call Call leg from which to encode call transfer. ++ * \param call_status TRUE if call is alerting. ++ * ++ * \retval 0 on success. ++ * \retval -1 on error. ++ */ ++static int rose_call_transfer_complete_encode(struct pri *ctrl, q931_call *call, ++ int call_status) ++{ ++ unsigned char buffer[256]; ++ unsigned char *end; ++ ++ switch (ctrl->switchtype) { ++ case PRI_SWITCH_EUROISDN_E1: ++ case PRI_SWITCH_EUROISDN_T1: ++ end = ++ enc_etsi_ect_inform(ctrl, buffer, buffer + sizeof(buffer), call, call_status); ++ break; ++ case PRI_SWITCH_QSIG: ++ end = ++ enc_qsig_call_transfer_complete(ctrl, buffer, buffer + sizeof(buffer), call, ++ call_status); ++ break; ++ default: + return -1; + } ++ if (!end) { ++ return -1; ++ } + +- return 0; ++ return pri_call_apdu_queue(call, Q931_FACILITY, buffer, end - buffer, NULL); + } +-/* End AOC */ + +-static int rose_calling_name_decode(struct pri *pri, q931_call *call, struct rose_component *choice, int len) ++/* ===== End Call Transfer Supplementary Service (ECMA-178) ===== */ ++ ++/*! ++ * \internal ++ * \brief Encode the Q.SIG CalledName invoke message. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode the facility ie contents. ++ * \param end End of facility ie contents encoding data buffer. ++ * \param name Name data which to encode name. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *enc_qsig_called_name(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const struct q931_party_name *name) + { +- int i = 0; +- struct rose_component *comp = NULL; +- unsigned char *vdata = choice->data; +- int characterSet = 1; +- switch (choice->type) { +- case ROSE_NAME_PRESENTATION_ALLOWED_SIMPLE: +- memcpy(call->callername, choice->data, choice->len); +- call->callername[choice->len] = 0; +- if (pri->debug & PRI_DEBUG_APDU) +- pri_message(pri, " Received simple calling name '%s'\n", call->callername); +- return 0; ++ struct fac_extension_header header; ++ struct rose_msg_invoke msg; + +- case ROSE_NAME_PRESENTATION_ALLOWED_EXTENDED: +- do { +- GET_COMPONENT(comp, i, vdata, len); +- CHECK_COMPONENT(comp, ASN1_OCTETSTRING, "Don't know what to do if nameData is of type 0x%x\n"); +- memcpy(call->callername, comp->data, comp->len); +- call->callername[comp->len] = 0; +- NEXT_COMPONENT(comp, i); ++ memset(&header, 0, sizeof(header)); ++ header.nfe_present = 1; ++ header.nfe.source_entity = 0; /* endPINX */ ++ header.nfe.destination_entity = 0; /* endPINX */ ++ header.interpretation_present = 1; ++ header.interpretation = 0; /* discardAnyUnrecognisedInvokePdu */ ++ pos = facility_encode_header(ctrl, pos, end, &header); ++ if (!pos) { ++ return NULL; ++ } + +- GET_COMPONENT(comp, i, vdata, len); +- CHECK_COMPONENT(comp, ASN1_INTEGER, "Don't know what to do if CharacterSet is of type 0x%x\n"); +- ASN1_GET_INTEGER(comp, characterSet); +- } +- while (0); ++ memset(&msg, 0, sizeof(msg)); ++ msg.operation = ROSE_QSIG_CalledName; ++ msg.invoke_id = get_invokeid(ctrl); + +- if (pri->debug & PRI_DEBUG_APDU) +- pri_message(pri, " Received extended calling name '%s', characterset %d\n", call->callername, characterSet); +- return 0; +- case ROSE_NAME_PRESENTATION_RESTRICTED_SIMPLE: +- case ROSE_NAME_PRESENTATION_RESTRICTED_EXTENDED: +- case ROSE_NAME_PRESENTATION_RESTRICTED_NULL: +- case ROSE_NAME_NOT_AVAIL: +- default: +- if (pri->debug & PRI_DEBUG_APDU) +- pri_message(pri, "Do not handle argument of type 0x%X\n", choice->type); +- return -1; +- } ++ /* CalledName */ ++ q931_copy_name_to_rose(ctrl, &msg.args.qsig.CalledName.name, name); ++ ++ pos = rose_encode_invoke(ctrl, pos, end, &msg); ++ ++ return pos; + } +-/* ===== Call Transfer Supplementary Service (ECMA-178) ===== */ + +-static int rose_party_number_decode(struct pri *pri, q931_call *call, unsigned char *data, int len, struct addressingdataelements_presentednumberunscreened *value) ++/*! ++ * \internal ++ * \brief Encode and queue the Q.SIG CalledName invoke message. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param call Call leg from which to encode name. ++ * ++ * \retval 0 on success. ++ * \retval -1 on error. ++ */ ++int rose_called_name_encode(struct pri *ctrl, q931_call *call, int messagetype) + { +- int i = 0; +- int size = 0; +- struct rose_component *comp = NULL; +- unsigned char *vdata = data; ++ unsigned char buffer[256]; ++ unsigned char *end; + ++ /* CalledName is the local_id.name */ ++ end = enc_qsig_called_name(ctrl, buffer, buffer + sizeof(buffer), ++ &call->local_id.name); ++ if (!end) { ++ return -1; ++ } + +- do { +- GET_COMPONENT(comp, i, vdata, len); ++ return pri_call_apdu_queue(call, messagetype, buffer, end - buffer, NULL); ++} + +- switch(comp->type) { +- case (ASN1_CONTEXT_SPECIFIC | ASN1_CONSTRUCTOR | ASN1_TAG_0): /* [0] IMPLICIT NumberDigits -- default: unknownPartyNumber */ +- if (pri->debug & PRI_DEBUG_APDU) +- pri_message(pri, " PartyNumber: UnknownPartyNumber len=%d\n", len); +- size = rose_number_digits_decode(pri, call, comp->data, comp->len, value); +- if (size < 0) +- return -1; +- value->npi = PRI_NPI_UNKNOWN; +- value->ton = PRI_TON_UNKNOWN; +- break; ++/*! ++ * \internal ++ * \brief Encode the Q.SIG ConnectedName invoke message. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode the facility ie contents. ++ * \param end End of facility ie contents encoding data buffer. ++ * \param name Name data which to encode name. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *enc_qsig_connected_name(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const struct q931_party_name *name) ++{ ++ struct fac_extension_header header; ++ struct rose_msg_invoke msg; + +- case (ASN1_CONTEXT_SPECIFIC | ASN1_CONSTRUCTOR | ASN1_TAG_1): /* [1] IMPLICIT PublicPartyNumber */ +- if (pri->debug & PRI_DEBUG_APDU) +- pri_message(pri, " PartyNumber: PublicPartyNumber len=%d\n", len); +- size = rose_public_party_number_decode(pri, call, comp->data, comp->len, value); +- if (size < 0) +- return -1; +- value->npi = PRI_NPI_E163_E164; +- break; ++ memset(&header, 0, sizeof(header)); ++ header.nfe_present = 1; ++ header.nfe.source_entity = 0; /* endPINX */ ++ header.nfe.destination_entity = 0; /* endPINX */ ++ header.interpretation_present = 1; ++ header.interpretation = 0; /* discardAnyUnrecognisedInvokePdu */ ++ pos = facility_encode_header(ctrl, pos, end, &header); ++ if (!pos) { ++ return NULL; ++ } + +- case (ASN1_CONTEXT_SPECIFIC | ASN1_CONSTRUCTOR | ASN1_TAG_3): /* [3] IMPLICIT NumberDigits -- not used: dataPartyNumber */ +- pri_message(pri, "!! PartyNumber: dataPartyNumber is reserved!\n"); +- size = rose_number_digits_decode(pri, call, comp->data, comp->len, value); +- if (size < 0) +- return -1; +- value->npi = PRI_NPI_X121 /* ??? */; +- value->ton = PRI_TON_UNKNOWN /* ??? */; +- break; ++ memset(&msg, 0, sizeof(msg)); ++ msg.operation = ROSE_QSIG_ConnectedName; ++ msg.invoke_id = get_invokeid(ctrl); + +- case (ASN1_CONTEXT_SPECIFIC | ASN1_CONSTRUCTOR | ASN1_TAG_4): /* [4] IMPLICIT NumberDigits -- not used: telexPartyNumber */ +- pri_message(pri, "!! PartyNumber: telexPartyNumber is reserved!\n"); +- size = rose_number_digits_decode(pri, call, comp->data, comp->len, value); +- if (size < 0) +- return -1; +- value->npi = PRI_NPI_F69 /* ??? */; +- value->ton = PRI_TON_UNKNOWN /* ??? */; +- break; ++ /* ConnectedName */ ++ q931_copy_name_to_rose(ctrl, &msg.args.qsig.ConnectedName.name, name); + +- case (ASN1_CONTEXT_SPECIFIC | ASN1_CONSTRUCTOR | ASN1_TAG_5): /* [5] IMPLICIT PrivatePartyNumber */ +- if (pri->debug & PRI_DEBUG_APDU) +- pri_message(pri, " PartyNumber: PrivatePartyNumber len=%d\n", len); +- size = rose_private_party_number_decode(pri, call, comp->data, comp->len, value); +- if (size < 0) +- return -1; +- value->npi = PRI_NPI_PRIVATE; +- break; ++ pos = rose_encode_invoke(ctrl, pos, end, &msg); + +- case (ASN1_CONTEXT_SPECIFIC | ASN1_CONSTRUCTOR | ASN1_TAG_8): /* [8] IMPLICIT NumberDigits -- not used: nationalStandatdPartyNumber */ +- pri_message(pri, "!! PartyNumber: nationalStandardPartyNumber is reserved!\n"); +- size = rose_number_digits_decode(pri, call, comp->data, comp->len, value); +- if (size < 0) +- return -1; +- value->npi = PRI_NPI_NATIONAL; +- value->ton = PRI_TON_NATIONAL; +- break; ++ return pos; ++} + +- default: +- pri_message(pri, "Invalid PartyNumber component 0x%X\n", comp->type); +- return -1; +- } +- ASN1_FIXUP_LEN(comp, size); +- if (pri->debug & PRI_DEBUG_APDU) +- pri_message(pri, " PartyNumber: '%s' size=%d len=%d\n", value->partyaddress, size, len); +- return size; ++/*! ++ * \internal ++ * \brief Encode and queue the Q.SIG ConnectedName invoke message. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param call Call leg from which to encode name. ++ * ++ * \retval 0 on success. ++ * \retval -1 on error. ++ */ ++int rose_connected_name_encode(struct pri *ctrl, q931_call *call, int messagetype) ++{ ++ unsigned char buffer[256]; ++ unsigned char *end; ++ ++ /* ConnectedName is the local_id.name */ ++ end = enc_qsig_connected_name(ctrl, buffer, buffer + sizeof(buffer), ++ &call->local_id.name); ++ if (!end) { ++ return -1; + } +- while (0); + +- return -1; ++ return pri_call_apdu_queue(call, messagetype, buffer, end - buffer, NULL); + } + +- +-static int rose_number_screened_decode(struct pri *pri, q931_call *call, unsigned char *data, int len, struct addressingdataelements_presentednumberscreened *value) ++/*! ++ * \brief Put the APDU on the call queue. ++ * ++ * \param call Call to enqueue message. ++ * \param messagetype Q.931 message type. ++ * \param apdu Facility ie contents buffer. ++ * \param apdu_len Length of the contents buffer. ++ * \param response Sender supplied information to handle APDU response messages. ++ * NULL if don't care about responses. ++ * ++ * \note ++ * Only APDU messages with an invoke component can supply a response pointer. ++ * If any other APDU messages supply a response pointer then aliasing of the ++ * invoke_id can occur. ++ * ++ * \retval 0 on success. ++ * \retval -1 on error. ++ */ ++int pri_call_apdu_queue(q931_call *call, int messagetype, const unsigned char *apdu, int apdu_len, struct apdu_callback_data *response) + { +- int i = 0; +- int size = 0; +- struct rose_component *comp = NULL; +- unsigned char *vdata = data; ++ struct apdu_event *cur = NULL; ++ struct apdu_event *new_event = NULL; + +- int scrind = -1; +- +- do { +- /* Party Number */ +- GET_COMPONENT(comp, i, vdata, len); +- size = rose_party_number_decode(pri, call, (u_int8_t *)comp, comp->len + 2, (struct addressingdataelements_presentednumberunscreened*) value); +- if (size < 0) ++ if (!call || !messagetype || !apdu ++ || apdu_len < 1 || sizeof(new_event->apdu) < apdu_len) { ++ return -1; ++ } ++ switch (messagetype) { ++ case Q931_FACILITY: ++ break; ++ default: ++ if (q931_is_dummy_call(call)) { ++ pri_error(call->pri, "!! Cannot send %s message on dummy call reference.\n", ++ msg2str(messagetype)); + return -1; +- comp->len = size; +- NEXT_COMPONENT(comp, i); ++ } ++ break; ++ } + +- /* Screening Indicator */ +- GET_COMPONENT(comp, i, vdata, len); +- CHECK_COMPONENT(comp, ASN1_ENUMERATED, "Don't know what to do with NumberScreened ROSE component type 0x%x\n"); +- ASN1_GET_INTEGER(comp, scrind); +- // Todo: scrind = screeningindicator_for_q931(pri, scrind); +- NEXT_COMPONENT(comp, i); ++ new_event = calloc(1, sizeof(*new_event)); ++ if (!new_event) { ++ pri_error(call->pri, "!! Malloc failed!\n"); ++ return -1; ++ } + +- value->scrind = scrind; ++ /* Fill in the APDU event */ ++ new_event->message = messagetype; ++ if (response) { ++ new_event->response = *response; ++ } ++ new_event->call = call; ++ new_event->apdu_len = apdu_len; ++ memcpy(new_event->apdu, apdu, apdu_len); + +- if (pri->debug & PRI_DEBUG_APDU) +- pri_message(pri, " NumberScreened: '%s' ScreeningIndicator=%d i=%d len=%d\n", value->partyaddress, scrind, i, len); +- +- return i-2; // We do not have a sequence header here. ++ /* Append APDU event to the end of the list. */ ++ if (call->apdus) { ++ for (cur = call->apdus; cur->next; cur = cur->next) { ++ } ++ cur->next = new_event; ++ } else { ++ call->apdus = new_event; + } +- while (0); + +- return -1; ++ return 0; + } + +- +-static int rose_presented_number_screened_decode(struct pri *pri, q931_call *call, unsigned char *data, int len, struct addressingdataelements_presentednumberscreened *value) ++/* Used by q931.c to cleanup the apdu queue upon destruction of a call */ ++void pri_call_apdu_queue_cleanup(q931_call *call) + { +- int i = 0; +- int size = 0; +- struct rose_component *comp = NULL; +- unsigned char *vdata = data; ++ struct apdu_event *cur_event; ++ struct apdu_event *free_event; + +- /* Fill in default values */ +- value->ton = PRI_TON_UNKNOWN; +- value->npi = PRI_NPI_UNKNOWN; +- value->pres = -1; /* Data is not available */ ++ if (call) { ++ cur_event = call->apdus; ++ call->apdus = NULL; ++ while (cur_event) { ++ if (cur_event->response.callback) { ++ /* Indicate to callback that the APDU is being cleaned up. */ ++ cur_event->response.callback(APDU_CALLBACK_REASON_CLEANUP, call->pri, ++ call, cur_event, NULL); + +- do { +- GET_COMPONENT(comp, i, vdata, len); ++ /* Stop any response timeout. */ ++ pri_schedule_del(call->pri, cur_event->timer); ++ } ++ free_event = cur_event; ++ cur_event = cur_event->next; ++ free(free_event); ++ } ++ } ++} + +- switch(comp->type) { +- case (ASN1_CONTEXT_SPECIFIC | ASN1_CONSTRUCTOR | ASN1_TAG_0): /* [0] IMPLICIT presentationAllowedNumber */ +- if (pri->debug & PRI_DEBUG_APDU) +- pri_message(pri, " PresentedNumberScreened: presentationAllowedNumber comp->len=%d\n", comp->len); +- value->pres = PRES_ALLOWED_USER_NUMBER_PASSED_SCREEN; +- size = rose_number_screened_decode(pri, call, comp->data, comp->len, value); +- if (size < 0) +- return -1; +- ASN1_FIXUP_LEN(comp, size); +- return size + 2; ++/*! ++ * \internal ++ * \brief Find an outstanding APDU with the given invoke id. ++ * ++ * \param call Call to find APDU. ++ * \param invoke_id Invoke id to match outstanding APDUs in queue. ++ * ++ * \retval apdu_event if found. ++ * \retval NULL if not found. ++ */ ++static struct apdu_event *pri_call_apdu_find(struct q931_call *call, int invoke_id) ++{ ++ struct apdu_event *apdu; + +- case (ASN1_CONTEXT_SPECIFIC | ASN1_TAG_1): /* [1] IMPLICIT presentationRestricted */ +- if (pri->debug & PRI_DEBUG_APDU) +- pri_message(pri, " PresentedNumberScreened: presentationRestricted comp->len=%d\n", comp->len); +- if (comp->len != 0) { /* must be NULL */ +- pri_error(pri, "!! Invalid PresentationRestricted component received (len != 0)\n"); +- return -1; +- } +- value->pres = PRES_PROHIB_USER_NUMBER_PASSED_SCREEN; +- return 2; ++ for (apdu = call->apdus; apdu; apdu = apdu->next) { ++ /* ++ * Note: The APDU cannot be sent and still in the queue without a ++ * callback and timeout timer active. Therefore, an invoke_id of ++ * zero is valid and not just the result of a memset(). ++ */ ++ if (apdu->response.invoke_id == invoke_id && apdu->sent) { ++ break; ++ } ++ } ++ return apdu; ++} + +- case (ASN1_CONTEXT_SPECIFIC | ASN1_TAG_2): /* [2] IMPLICIT numberNotAvailableDueToInterworking */ +- if (pri->debug & PRI_DEBUG_APDU) +- pri_message(pri, " PresentedNumberScreened: NumberNotAvailableDueToInterworking comp->len=%d\n", comp->len); +- if (comp->len != 0) { /* must be NULL */ +- pri_error(pri, "!! Invalid NumberNotAvailableDueToInterworking component received (len != 0)\n"); +- return -1; +- } +- value->pres = PRES_NUMBER_NOT_AVAILABLE; +- if (pri->debug & PRI_DEBUG_APDU) +- pri_message(pri, " PresentedNumberScreened: numberNotAvailableDueToInterworking Type=0x%X i=%d len=%d size=%d\n", comp->type, i, len); +- return 2; ++/*! ++ * \brief Delete the given APDU event from the given call. ++ * ++ * \param call Call to remove the APDU. ++ * \param doomed APDU event to delete. ++ * ++ * \return Nothing ++ */ ++void pri_call_apdu_delete(struct q931_call *call, struct apdu_event *doomed) ++{ ++ struct apdu_event **prev; ++ struct apdu_event *cur; + +- case (ASN1_CONTEXT_SPECIFIC | ASN1_CONSTRUCTOR | ASN1_TAG_3): /* [3] IMPLICIT presentationRestrictedNumber */ +- if (pri->debug & PRI_DEBUG_APDU) +- pri_message(pri, " PresentedNumberScreened: presentationRestrictedNumber comp->len=%d\n", comp->len); +- value->pres = PRES_PROHIB_USER_NUMBER_PASSED_SCREEN; +- size = rose_number_screened_decode(pri, call, comp->data, comp->len, value); +- if (size < 0) +- return -1; +- ASN1_FIXUP_LEN(comp, size); +- return size + 2; ++ /* Find APDU in list. */ ++ for (prev = &call->apdus, cur = call->apdus; ++ cur; ++ prev = &cur->next, cur = cur->next) { ++ if (cur == doomed) { ++ /* Stop any response timeout. */ ++ pri_schedule_del(call->pri, cur->timer); + +- default: +- pri_message(pri, "Invalid PresentedNumberScreened component 0x%X\n", comp->type); ++ /* Remove APDU from list. */ ++ *prev = cur->next; ++ free(cur); ++ break; + } +- return -1; + } +- while (0); +- +- return -1; + } + +- +-static int rose_call_transfer_complete_decode(struct pri *pri, q931_call *call, struct rose_component *sequence, int len) ++/*! \note Only called when sending the SETUP message. */ ++int pri_call_add_standard_apdus(struct pri *ctrl, q931_call *call) + { +- int i = 0; +- struct rose_component *comp = NULL; +- unsigned char *vdata = sequence->data; +- int res = 0; ++ if (!ctrl->sendfacility) { ++ return 0; ++ } + +- int end_designation = 0; +- struct addressingdataelements_presentednumberscreened redirection_number; +- char redirection_name[50] = ""; +- int call_status = 0; +- redirection_number.partyaddress[0] = 0; +- redirection_number.partysubaddress[0] = 0; +- call->callername[0] = 0; +- call->callernum[0] = 0; ++ switch (ctrl->switchtype) { ++ case PRI_SWITCH_EUROISDN_E1: ++ case PRI_SWITCH_EUROISDN_T1: ++ if (q931_is_ptmp(ctrl)) { ++ /* PTMP mode */ ++ break; ++ } ++ /* PTP mode */ ++ if (call->redirecting.count) { ++ rose_diverting_leg_information2_encode(ctrl, call); + ++ /* ++ * Expect a DivertingLegInformation3 to update the COLR of the ++ * redirecting-to party we are attempting to call now. ++ */ ++ call->redirecting.state = Q931_REDIRECTING_STATE_EXPECTING_RX_DIV_LEG_3; ++ } ++ break; ++ case PRI_SWITCH_QSIG: ++ /* For Q.SIG it does network and cpe operations */ ++ if (call->redirecting.count) { ++ rose_diverting_leg_information2_encode(ctrl, call); + +- /* Data checks */ +- if (sequence->type != (ASN1_CONSTRUCTOR | ASN1_SEQUENCE)) { /* Constructed Sequence */ +- pri_message(pri, "Invalid callTransferComplete argument. (Not a sequence)\n"); +- return -1; ++ /* ++ * Expect a DivertingLegInformation3 to update the COLR of the ++ * redirecting-to party we are attempting to call now. ++ */ ++ call->redirecting.state = Q931_REDIRECTING_STATE_EXPECTING_RX_DIV_LEG_3; ++ } ++ add_callername_facility_ies(ctrl, call, 1); ++ break; ++ case PRI_SWITCH_NI2: ++ add_callername_facility_ies(ctrl, call, (ctrl->localtype == PRI_CPE)); ++ break; ++ case PRI_SWITCH_DMS100: ++ if (ctrl->localtype == PRI_CPE) { ++ add_dms100_transfer_ability_apdu(ctrl, call); ++ } ++ break; ++ default: ++ break; + } + +- if (sequence->len == ASN1_LEN_INDEF) { +- len -= 4; /* For the 2 extra characters at the end +- * and two characters of header */ +- } else +- len -= 2; ++ return 0; ++} + +- if (pri->debug & PRI_DEBUG_APDU) +- pri_message(pri, " CT-Complete: len=%d\n", len); ++/*! ++ * \brief Send the CallTransferComplete/EctInform invoke message. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param call Call leg from which to encode call transfer. ++ * \param call_status TRUE if call is alerting. ++ * ++ * \retval 0 on success. ++ * \retval -1 on error. ++ */ ++int send_call_transfer_complete(struct pri *ctrl, q931_call *call, int call_status) ++{ ++ if (rose_call_transfer_complete_encode(ctrl, call, call_status) ++ || q931_facility(ctrl, call)) { ++ pri_message(ctrl, ++ "Could not schedule facility message for call transfer completed.\n"); ++ return -1; ++ } + +- do { +- /* End Designation */ +- GET_COMPONENT(comp, i, vdata, len); +- CHECK_COMPONENT(comp, ASN1_ENUMERATED, "Invalid endDesignation type 0x%X of ROSE callTransferComplete component received\n"); +- ASN1_GET_INTEGER(comp, end_designation); +- NEXT_COMPONENT(comp, i); +- if (pri->debug & PRI_DEBUG_APDU) +- pri_message(pri, " CT-Complete: Received endDesignation=%d\n", end_designation); ++ return 0; ++} + ++/*! ++ * \internal ++ * \brief Encode a plain facility ETSI error code. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode the facility ie contents. ++ * \param end End of facility ie contents encoding data buffer. ++ * \param call Call leg from which to encode error message response. ++ * \param invoke_id Invoke id to put in error message response. ++ * \param code Error code to put in error message response. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *enc_etsi_error(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, q931_call *call, int invoke_id, enum rose_error_code code) ++{ ++ struct rose_msg_error msg; + +- /* Redirection Number */ +- GET_COMPONENT(comp, i, vdata, len); +- res = rose_presented_number_screened_decode(pri, call, (u_int8_t *)comp, comp->len + 2, &redirection_number); +- if (res < 0) +- return -1; +- comp->len = res; +- if (res > 2) { +- if (pri->debug & PRI_DEBUG_APDU) +- pri_message(pri, " CT-Complete: Received redirectionNumber=%s\n", redirection_number.partyaddress); +- strncpy(call->callernum, redirection_number.partyaddress, 20); +- call->callernum[20] = 0; +- } +- NEXT_COMPONENT(comp, i); ++ pos = facility_encode_header(ctrl, pos, end, NULL); ++ if (!pos) { ++ return NULL; ++ } + ++ memset(&msg, 0, sizeof(msg)); ++ msg.invoke_id = invoke_id; ++ msg.code = code; + +-#if 0 /* This one is optional. How do we check if it is there? */ +- /* Basic Call Info Elements */ +- GET_COMPONENT(comp, i, vdata, len); +- NEXT_COMPONENT(comp, i); +-#endif ++ pos = rose_encode_error(ctrl, pos, end, &msg); + ++ return pos; ++} + +- /* Redirection Name */ +- GET_COMPONENT(comp, i, vdata, len); +- res = asn1_name_decode((u_int8_t *)comp, comp->len + 2, redirection_name, sizeof(redirection_name)); +- if (res < 0) +- return -1; +- memcpy(call->callername, comp->data, comp->len); +- call->callername[comp->len] = 0; +- ASN1_FIXUP_LEN(comp, res); +- comp->len = res; +- NEXT_COMPONENT(comp, i); +- if (pri->debug & PRI_DEBUG_APDU) +- pri_message(pri, " CT-Complete: Received redirectionName '%s'\n", redirection_name); ++/*! ++ * \internal ++ * \brief Encode a plain facility Q.SIG error code. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode the facility ie contents. ++ * \param end End of facility ie contents encoding data buffer. ++ * \param call Call leg from which to encode error message response. ++ * \param invoke_id Invoke id to put in error message response. ++ * \param code Error code to put in error message response. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *enc_qsig_error(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, q931_call *call, int invoke_id, enum rose_error_code code) ++{ ++ struct fac_extension_header header; ++ struct rose_msg_error msg; + ++ memset(&header, 0, sizeof(header)); ++ header.nfe_present = 1; ++ header.nfe.source_entity = 0; /* endPINX */ ++ header.nfe.destination_entity = 0; /* endPINX */ ++ header.interpretation_present = 1; ++ header.interpretation = 0; /* discardAnyUnrecognisedInvokePdu */ ++ pos = facility_encode_header(ctrl, pos, end, &header); ++ if (!pos) { ++ return NULL; ++ } + +- /* Call Status */ +- GET_COMPONENT(comp, i, vdata, len); +- CHECK_COMPONENT(comp, ASN1_ENUMERATED, "Invalid callStatus type 0x%X of ROSE callTransferComplete component received\n"); +- ASN1_GET_INTEGER(comp, call_status); +- NEXT_COMPONENT(comp, i); +- if (pri->debug & PRI_DEBUG_APDU) +- pri_message(pri, " CT-Complete: Received callStatus=%d\n", call_status); ++ memset(&msg, 0, sizeof(msg)); ++ msg.invoke_id = invoke_id; ++ msg.code = code; + ++ pos = rose_encode_error(ctrl, pos, end, &msg); + +- /* Argument Extension */ +-#if 0 /* Not supported */ +- GET_COMPONENT(comp, i, vdata, len); +- switch (comp->type) { +- case (ASN1_CONTEXT_SPECIFIC | ASN1_TAG_9): /* [9] IMPLICIT Extension */ +- res = rose_extension_decode(pri, call, comp->data, comp->len, &redirection_number); +- if (res < 0) +- return -1; +- ASN1_FIXUP_LEN(comp, res); +- comp->len = res; ++ return pos; ++} + +- case (ASN1_CONTEXT_SPECIFIC | ASN1_TAG_10): /* [10] IMPLICIT SEQUENCE OF Extension */ +- res = rose_sequence_of_extension_decode(pri, call, comp->data, comp->len, &redirection_number); +- if (res < 0) +- return -1; +- ASN1_FIXUP_LEN(comp, res); +- comp->len = res; ++/*! ++ * \internal ++ * \brief Encode and queue a plain facility error code. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param call Call leg from which to encode error message response. ++ * \param invoke_id Invoke id to put in error message response. ++ * \param code Error code to put in error message response. ++ * ++ * \retval 0 on success. ++ * \retval -1 on error. ++ */ ++static int rose_facility_error_encode(struct pri *ctrl, q931_call *call, int invoke_id, ++ enum rose_error_code code) ++{ ++ unsigned char buffer[256]; ++ unsigned char *end; + +- default: +- pri_message(pri, " CT-Complete: !! Unknown argumentExtension received 0x%X\n", comp->type); +- return -1; +- } +-#else +- GET_COMPONENT(comp, i, vdata, len); +- ASN1_FIXUP_LEN(comp, res); +- NEXT_COMPONENT(comp, i); +-#endif ++ switch (ctrl->switchtype) { ++ case PRI_SWITCH_EUROISDN_E1: ++ case PRI_SWITCH_EUROISDN_T1: ++ end = ++ enc_etsi_error(ctrl, buffer, buffer + sizeof(buffer), call, invoke_id, code); ++ break; ++ case PRI_SWITCH_QSIG: ++ end = ++ enc_qsig_error(ctrl, buffer, buffer + sizeof(buffer), call, invoke_id, code); ++ break; ++ default: ++ return -1; ++ } ++ if (!end) { ++ return -1; ++ } + +- if(i < len) +- pri_message(pri, " CT-Complete: !! not all information is handled !! i=%d / len=%d\n", i, len); ++ return pri_call_apdu_queue(call, Q931_FACILITY, buffer, end - buffer, NULL); ++} + +- return 0; ++/*! ++ * \brief Encode and send a plain facility error code. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param call Call leg from which to encode error message response. ++ * \param invoke_id Invoke id to put in error message response. ++ * \param code Error code to put in error message response. ++ * ++ * \retval 0 on success. ++ * \retval -1 on error. ++ */ ++static int send_facility_error(struct pri *ctrl, q931_call *call, int invoke_id, ++ enum rose_error_code code) ++{ ++ if (rose_facility_error_encode(ctrl, call, invoke_id, code) ++ || q931_facility(ctrl, call)) { ++ pri_message(ctrl, ++ "Could not schedule facility message for error message.\n"); ++ return -1; + } +- while (0); + +- return -1; ++ return 0; + } + +- +-static int rose_call_transfer_update_decode(struct pri *pri, q931_call *call, struct rose_component *sequence, int len) ++/*! ++ * \internal ++ * \brief Encode a plain facility ETSI result ok. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode the facility ie contents. ++ * \param end End of facility ie contents encoding data buffer. ++ * \param call Call leg from which to encode result ok message response. ++ * \param invoke_id Invoke id to put in result ok message response. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *enc_etsi_result_ok(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, q931_call *call, int invoke_id) + { +- int i = 0; +- struct rose_component *comp = NULL; +- unsigned char *vdata = sequence->data; +- int res = 0; ++ struct rose_msg_result msg; + +- struct addressingdataelements_presentednumberscreened redirection_number; +- redirection_number.partyaddress[0] = 0; +- redirection_number.partysubaddress[0] = 0; +- char redirection_name[50] = ""; +- call->callername[0] = 0; +- call->callernum[0] = 0; ++ pos = facility_encode_header(ctrl, pos, end, NULL); ++ if (!pos) { ++ return NULL; ++ } + ++ memset(&msg, 0, sizeof(msg)); ++ msg.invoke_id = invoke_id; ++ msg.operation = ROSE_None; + +- /* Data checks */ +- if (sequence->type != (ASN1_CONSTRUCTOR | ASN1_SEQUENCE)) { /* Constructed Sequence */ +- pri_message(pri, "Invalid callTransferComplete argument. (Not a sequence)\n"); +- return -1; ++ pos = rose_encode_result(ctrl, pos, end, &msg); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Encode a plain facility Q.SIG result ok. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode the facility ie contents. ++ * \param end End of facility ie contents encoding data buffer. ++ * \param call Call leg from which to encode result ok message response. ++ * \param invoke_id Invoke id to put in result ok message response. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *enc_qsig_result_ok(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, q931_call *call, int invoke_id) ++{ ++ struct fac_extension_header header; ++ struct rose_msg_result msg; ++ ++ memset(&header, 0, sizeof(header)); ++ header.nfe_present = 1; ++ header.nfe.source_entity = 0; /* endPINX */ ++ header.nfe.destination_entity = 0; /* endPINX */ ++ header.interpretation_present = 1; ++ header.interpretation = 0; /* discardAnyUnrecognisedInvokePdu */ ++ pos = facility_encode_header(ctrl, pos, end, &header); ++ if (!pos) { ++ return NULL; + } + +- if (sequence->len == ASN1_LEN_INDEF) { +- len -= 4; /* For the 2 extra characters at the end +- * and two characters of header */ +- } else +- len -= 2; ++ memset(&msg, 0, sizeof(msg)); ++ msg.invoke_id = invoke_id; ++ msg.operation = ROSE_None; + +- if (pri->debug & PRI_DEBUG_APDU) +- pri_message(pri, " CT-Complete: len=%d\n", len); ++ pos = rose_encode_result(ctrl, pos, end, &msg); + +- do { +- /* Redirection Number */ +- GET_COMPONENT(comp, i, vdata, len); +- res = rose_presented_number_screened_decode(pri, call, (u_int8_t *)comp, comp->len + 2, &redirection_number); +- if (res < 0) +- return -1; +- comp->len = res; +- if (res > 2) { +- if (pri->debug & PRI_DEBUG_APDU) +- pri_message(pri, " CT-Complete: Received redirectionNumber=%s\n", redirection_number.partyaddress); +- strncpy(call->callernum, redirection_number.partyaddress, 20); +- call->callernum[20] = 0; +- } +- NEXT_COMPONENT(comp, i); ++ return pos; ++} + +- /* Redirection Name */ +- GET_COMPONENT(comp, i, vdata, len); +- res = asn1_name_decode((u_int8_t *)comp, comp->len + 2, redirection_name, sizeof(redirection_name)); +- if (res < 0) +- return -1; +- memcpy(call->callername, comp->data, comp->len); +- call->callername[comp->len] = 0; +- ASN1_FIXUP_LEN(comp, res); +- comp->len = res; +- NEXT_COMPONENT(comp, i); +- if (pri->debug & PRI_DEBUG_APDU) +- pri_message(pri, " CT-Complete: Received redirectionName '%s'\n", redirection_name); ++/*! ++ * \internal ++ * \brief Encode and queue a plain ROSE result ok. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param call Call leg from which to encode result ok message response. ++ * \param msgtype Q.931 message type to put facility ie in. ++ * \param invoke_id Invoke id to put in result ok message response. ++ * ++ * \retval 0 on success. ++ * \retval -1 on error. ++ */ ++static int rose_result_ok_encode(struct pri *ctrl, q931_call *call, int msgtype, int invoke_id) ++{ ++ unsigned char buffer[256]; ++ unsigned char *end; + ++ switch (ctrl->switchtype) { ++ case PRI_SWITCH_EUROISDN_E1: ++ case PRI_SWITCH_EUROISDN_T1: ++ end = ++ enc_etsi_result_ok(ctrl, buffer, buffer + sizeof(buffer), call, invoke_id); ++ break; ++ case PRI_SWITCH_QSIG: ++ end = ++ enc_qsig_result_ok(ctrl, buffer, buffer + sizeof(buffer), call, invoke_id); ++ break; ++ default: ++ return -1; ++ } ++ if (!end) { ++ return -1; ++ } + +-#if 0 /* This one is optional. How do we check if it is there? */ +- /* Basic Call Info Elements */ +- GET_COMPONENT(comp, i, vdata, len); +- NEXT_COMPONENT(comp, i); +-#endif ++ return pri_call_apdu_queue(call, msgtype, buffer, end - buffer, NULL); ++} + ++/*! ++ * \brief Encode and send a FACILITY message with a plain ROSE result ok. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param call Call leg from which to encode result ok message response. ++ * \param invoke_id Invoke id to put in result ok message response. ++ * ++ * \retval 0 on success. ++ * \retval -1 on error. ++ */ ++static int send_facility_result_ok(struct pri *ctrl, q931_call *call, int invoke_id) ++{ ++ if (rose_result_ok_encode(ctrl, call, Q931_FACILITY, invoke_id) ++ || q931_facility(ctrl, call)) { ++ pri_message(ctrl, ++ "Could not schedule facility message for result OK message.\n"); ++ return -1; ++ } + +- /* Argument Extension */ +-#if 0 /* Not supported */ +- GET_COMPONENT(comp, i, vdata, len); +- switch (comp->type) { +- case (ASN1_CONTEXT_SPECIFIC | ASN1_TAG_9): /* [9] IMPLICIT Extension */ +- res = rose_extension_decode(pri, call, comp->data, comp->len, &redirection_number); +- if (res < 0) +- return -1; +- ASN1_FIXUP_LEN(comp, res); +- comp->len = res; ++ return 0; ++} + +- case (ASN1_CONTEXT_SPECIFIC | ASN1_TAG_10): /* [10] IMPLICIT SEQUENCE OF Extension */ +- res = rose_sequence_of_extension_decode(pri, call, comp->data, comp->len, &redirection_number); +- if (res < 0) +- return -1; +- ASN1_FIXUP_LEN(comp, res); +- comp->len = res; ++int pri_rerouting_rsp(struct pri *ctrl, q931_call *call, int invoke_id, enum PRI_REROUTING_RSP_CODE code) ++{ ++ enum rose_error_code rose_err; + +- default: +- pri_message(pri, " CT-Complete: !! Unknown argumentExtension received 0x%X\n", comp->type); +- return -1; +- } +-#else +- GET_COMPONENT(comp, i, vdata, len); +- ASN1_FIXUP_LEN(comp, res); +- NEXT_COMPONENT(comp, i); +-#endif ++ if (!ctrl || !call) { ++ return -1; ++ } + +- if(i < len) +- pri_message(pri, " CT-Complete: !! not all information is handled !! i=%d / len=%d\n", i, len); ++ /* Convert the public rerouting response code to an error code or result ok. */ ++ rose_err = ROSE_ERROR_Gen_ResourceUnavailable; ++ switch (code) { ++ case PRI_REROUTING_RSP_OK_CLEAR: ++ return rose_result_ok_encode(ctrl, call, Q931_DISCONNECT, invoke_id); ++ case PRI_REROUTING_RSP_OK_RETAIN: ++ return send_facility_result_ok(ctrl, call, invoke_id); ++ case PRI_REROUTING_RSP_NOT_SUBSCRIBED: ++ rose_err = ROSE_ERROR_Gen_NotSubscribed; ++ break; ++ case PRI_REROUTING_RSP_NOT_AVAILABLE: ++ rose_err = ROSE_ERROR_Gen_NotAvailable; ++ break; ++ case PRI_REROUTING_RSP_NOT_ALLOWED: ++ rose_err = ROSE_ERROR_Gen_SupplementaryServiceInteractionNotAllowed; ++ break; ++ case PRI_REROUTING_RSP_INVALID_NUMBER: ++ rose_err = ROSE_ERROR_Div_InvalidDivertedToNr; ++ break; ++ case PRI_REROUTING_RSP_SPECIAL_SERVICE_NUMBER: ++ rose_err = ROSE_ERROR_Div_SpecialServiceNr; ++ break; ++ case PRI_REROUTING_RSP_DIVERSION_TO_SELF: ++ rose_err = ROSE_ERROR_Div_DiversionToServedUserNr; ++ break; ++ case PRI_REROUTING_RSP_MAX_DIVERSIONS_EXCEEDED: ++ rose_err = ROSE_ERROR_Div_NumberOfDiversionsExceeded; ++ break; ++ case PRI_REROUTING_RSP_RESOURCE_UNAVAILABLE: ++ rose_err = ROSE_ERROR_Gen_ResourceUnavailable; ++ break; ++ } ++ return send_facility_error(ctrl, call, invoke_id, rose_err); ++} + +- return 0; ++/*! ++ * \brief Handle the ROSE reject message. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param call Call leg from which the message came. ++ * \param msgtype Q.931 message type ie is in. ++ * \param ie Raw ie contents. ++ * \param header Decoded facility header before ROSE. ++ * \param reject Decoded ROSE reject message contents. ++ * ++ * \return Nothing ++ */ ++void rose_handle_reject(struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, ++ const struct fac_extension_header *header, const struct rose_msg_reject *reject) ++{ ++ struct apdu_event *apdu; ++ union apdu_msg_data msg; ++ ++ /* Gripe to the user about getting rejected. */ ++ pri_error(ctrl, "ROSE REJECT:\n"); ++ if (reject->invoke_id_present) { ++ pri_error(ctrl, "\tINVOKE ID: %d\n", reject->invoke_id); + } +- while (0); ++ pri_error(ctrl, "\tPROBLEM: %s\n", rose_reject2str(reject->code)); + +- return -1; ++ switch (ctrl->switchtype) { ++ case PRI_SWITCH_DMS100: ++ /* The DMS-100 switch apparently handles invoke_id as an invoke operation. */ ++ return; ++ default: ++ break; ++ } ++ ++ if (!reject->invoke_id_present) { ++ /* ++ * No invoke id to look up so we cannot match it to any outstanding APDUs. ++ * This REJECT is apparently meant for someone monitoring the link. ++ */ ++ return; ++ } ++ apdu = pri_call_apdu_find(call, reject->invoke_id); ++ if (!apdu) { ++ return; ++ } ++ msg.reject = reject; ++ if (apdu->response.callback(APDU_CALLBACK_REASON_MSG_REJECT, ctrl, call, apdu, &msg)) { ++ pri_call_apdu_delete(call, apdu); ++ } + } + ++/*! ++ * \brief Handle the ROSE error message. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param call Call leg from which the message came. ++ * \param msgtype Q.931 message type ie is in. ++ * \param ie Raw ie contents. ++ * \param header Decoded facility header before ROSE. ++ * \param error Decoded ROSE error message contents. ++ * ++ * \return Nothing ++ */ ++void rose_handle_error(struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, ++ const struct fac_extension_header *header, const struct rose_msg_error *error) ++{ ++ const char *dms100_operation; ++ struct apdu_event *apdu; ++ union apdu_msg_data msg; + +-/* ===== End Call Transfer Supplementary Service (ECMA-178) ===== */ ++ /* Gripe to the user about getting an error. */ ++ pri_error(ctrl, "ROSE RETURN ERROR:\n"); ++ switch (ctrl->switchtype) { ++ case PRI_SWITCH_DMS100: ++ switch (error->invoke_id) { ++ case ROSE_DMS100_RLT_OPERATION_IND: ++ dms100_operation = "RLT_OPERATION_IND"; ++ break; ++ case ROSE_DMS100_RLT_THIRD_PARTY: ++ dms100_operation = "RLT_THIRD_PARTY"; ++ break; ++ default: ++ dms100_operation = NULL; ++ break; ++ } ++ if (dms100_operation) { ++ pri_error(ctrl, "\tOPERATION: %s\n", dms100_operation); ++ break; ++ } ++ /* fall through */ ++ default: ++ pri_error(ctrl, "\tINVOKE ID: %d\n", error->invoke_id); ++ break; ++ } ++ pri_error(ctrl, "\tERROR: %s\n", rose_error2str(error->code)); + ++ switch (ctrl->switchtype) { ++ case PRI_SWITCH_DMS100: ++ /* The DMS-100 switch apparently handles invoke_id as an invoke operation. */ ++ return; ++ default: ++ break; ++ } + ++ apdu = pri_call_apdu_find(call, error->invoke_id); ++ if (!apdu) { ++ return; ++ } ++ msg.error = error; ++ if (apdu->response.callback(APDU_CALLBACK_REASON_MSG_ERROR, ctrl, call, apdu, &msg)) { ++ pri_call_apdu_delete(call, apdu); ++ } ++} + +-int rose_reject_decode(struct pri *pri, q931_call *call, q931_ie *ie, unsigned char *data, int len) ++/*! ++ * \brief Handle the ROSE result message. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param call Call leg from which the message came. ++ * \param msgtype Q.931 message type ie is in. ++ * \param ie Raw ie contents. ++ * \param header Decoded facility header before ROSE. ++ * \param result Decoded ROSE result message contents. ++ * ++ * \return Nothing ++ */ ++void rose_handle_result(struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, ++ const struct fac_extension_header *header, const struct rose_msg_result *result) + { +- int i = 0; +- int problemtag = -1; +- int problem = -1; +- int invokeidvalue = -1; +- unsigned char *vdata = data; +- struct rose_component *comp = NULL; +- char *problemtagstr, *problemstr; +- +- do { +- /* Invoke ID stuff */ +- GET_COMPONENT(comp, i, vdata, len); +- CHECK_COMPONENT(comp, INVOKE_IDENTIFIER, "Don't know what to do if first ROSE component is of type 0x%x\n"); +- ASN1_GET_INTEGER(comp, invokeidvalue); +- NEXT_COMPONENT(comp, i); ++ struct apdu_event *apdu; ++ union apdu_msg_data msg; + +- GET_COMPONENT(comp, i, vdata, len); +- problemtag = comp->type; +- problem = comp->data[0]; +- +- if (pri->switchtype == PRI_SWITCH_DMS100) { +- switch (problemtag) { +- case 0x80: +- problemtagstr = "General problem"; ++ switch (ctrl->switchtype) { ++ case PRI_SWITCH_DMS100: ++ /* The DMS-100 switch apparently handles invoke_id as an invoke operation. */ ++ switch (result->invoke_id) { ++ case ROSE_DMS100_RLT_OPERATION_IND: ++ if (result->operation != ROSE_DMS100_RLT_OperationInd) { ++ pri_message(ctrl, "Invalid Operation value in return result! %s\n", ++ rose_operation2str(result->operation)); + break; +- case 0x81: +- problemtagstr = "Invoke problem"; +- break; +- case 0x82: +- problemtagstr = "Return result problem"; +- break; +- case 0x83: +- problemtagstr = "Return error problem"; +- break; +- default: +- problemtagstr = "Unknown"; + } + +- switch (problem) { +- case 0x00: +- problemstr = "Unrecognized component"; +- break; +- case 0x01: +- problemstr = "Mistyped component"; +- break; +- case 0x02: +- problemstr = "Badly structured component"; +- break; +- default: +- problemstr = "Unknown"; ++ /* We have enough data to transfer the call */ ++ call->rlt_call_id = result->args.dms100.RLT_OperationInd.call_id; ++ call->transferable = 1; ++ break; ++ case ROSE_DMS100_RLT_THIRD_PARTY: ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, "Successfully completed RLT transfer!\n"); + } +- +- pri_error(pri, "ROSE REJECT:\n"); +- pri_error(pri, "\tINVOKE ID: 0x%X\n", invokeidvalue); +- pri_error(pri, "\tPROBLEM TYPE: %s (0x%x)\n", problemtagstr, problemtag); +- pri_error(pri, "\tPROBLEM: %s (0x%x)\n", problemstr, problem); +- +- return 0; +- } else { +- pri_message(pri, "Unable to handle reject on switchtype %d!\n", pri->switchtype); +- return -1; ++ break; ++ default: ++ pri_message(ctrl, "Could not parse invoke of type %d!\n", result->invoke_id); ++ break; + } ++ return; ++ default: ++ break; ++ } + +- } while(0); +- +- return -1; ++ apdu = pri_call_apdu_find(call, result->invoke_id); ++ if (!apdu) { ++ return; ++ } ++ msg.result = result; ++ if (apdu->response.callback(APDU_CALLBACK_REASON_MSG_RESULT, ctrl, call, apdu, &msg)) { ++ pri_call_apdu_delete(call, apdu); ++ } + } +-int rose_return_error_decode(struct pri *pri, q931_call *call, q931_ie *ie, unsigned char *data, int len) ++ ++/*! ++ * \brief Handle the ROSE invoke message. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param call Call leg from which the message came. ++ * \param msgtype Q.931 message type ie is in. ++ * \param ie Raw ie contents. ++ * \param header Decoded facility header before ROSE. ++ * \param invoke Decoded ROSE invoke message contents. ++ * ++ * \return Nothing ++ */ ++void rose_handle_invoke(struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, ++ const struct fac_extension_header *header, const struct rose_msg_invoke *invoke) + { +- int i = 0; +- int errorvalue = -1; +- int invokeidvalue = -1; +- unsigned char *vdata = data; +- struct rose_component *comp = NULL; +- char *invokeidstr, *errorstr; +- +- do { +- /* Invoke ID stuff */ +- GET_COMPONENT(comp, i, vdata, len); +- CHECK_COMPONENT(comp, INVOKE_IDENTIFIER, "Don't know what to do if first ROSE component is of type 0x%x\n"); +- ASN1_GET_INTEGER(comp, invokeidvalue); +- NEXT_COMPONENT(comp, i); ++ struct pri_subcommand *subcmd; ++ struct q931_party_id party_id; ++ struct q931_party_redirecting deflection; + +- GET_COMPONENT(comp, i, vdata, len); +- CHECK_COMPONENT(comp, ASN1_INTEGER, "Don't know what to do if second component in return error is 0x%x\n"); +- ASN1_GET_INTEGER(comp, errorvalue); ++ switch (invoke->operation) { ++#if 0 /* Not handled yet */ ++ case ROSE_ETSI_ActivationDiversion: ++ break; ++ case ROSE_ETSI_DeactivationDiversion: ++ break; ++ case ROSE_ETSI_ActivationStatusNotificationDiv: ++ break; ++ case ROSE_ETSI_DeactivationStatusNotificationDiv: ++ break; ++ case ROSE_ETSI_InterrogationDiversion: ++ break; ++ case ROSE_ETSI_DiversionInformation: ++ break; ++#endif /* Not handled yet */ ++ case ROSE_ETSI_CallDeflection: ++ if (!PRI_MASTER(ctrl)->deflection_support) { ++ send_facility_error(ctrl, call, invoke->invoke_id, ++ ROSE_ERROR_Gen_NotSubscribed); ++ break; ++ } ++ if (!q931_master_pass_event(ctrl, call, msgtype)) { ++ /* Some other user is further along to connecting than this call. */ ++ send_facility_error(ctrl, call, invoke->invoke_id, ++ ROSE_ERROR_Div_IncomingCallAccepted); ++ break; ++ } ++ if (call->master_call->deflection_in_progress) { ++ /* Someone else is already doing a call deflection. */ ++ send_facility_error(ctrl, call, invoke->invoke_id, ++ ROSE_ERROR_Div_RequestAlreadyAccepted); ++ break; ++ } ++ subcmd = q931_alloc_subcommand(ctrl); ++ if (!subcmd) { ++ /* ++ * ROSE_ERROR_Gen_ResourceUnavailable was not in the list of allowed codes, ++ * but we will send it anyway. ++ */ ++ send_facility_error(ctrl, call, invoke->invoke_id, ++ ROSE_ERROR_Gen_ResourceUnavailable); ++ pri_error(ctrl, "ERROR: Too many facility subcommands\n"); ++ break; ++ } + +- if (pri->switchtype == PRI_SWITCH_DMS100) { +- switch (invokeidvalue) { +- case RLT_OPERATION_IND: +- invokeidstr = "RLT_OPERATION_IND"; +- break; +- case RLT_THIRD_PARTY: +- invokeidstr = "RLT_THIRD_PARTY"; +- break; +- default: +- invokeidstr = "Unknown"; +- } ++ call->master_call->deflection_in_progress = 1; + +- switch (errorvalue) { +- case 0x10: +- errorstr = "RLT Bridge Fail"; +- break; +- case 0x11: +- errorstr = "RLT Call ID Not Found"; +- break; +- case 0x12: +- errorstr = "RLT Not Allowed"; +- break; +- case 0x13: +- errorstr = "RLT Switch Equip Congs"; +- break; +- default: +- errorstr = "Unknown"; +- } ++ q931_party_redirecting_init(&deflection); + +- pri_error(pri, "ROSE RETURN ERROR:\n"); +- pri_error(pri, "\tOPERATION: %s\n", invokeidstr); +- pri_error(pri, "\tERROR: %s\n", errorstr); ++ /* Deflecting from the called address. */ ++ q931_party_address_to_id(&deflection.from, &call->called); ++ if (invoke->args.etsi.CallDeflection.presentation_allowed_to_diverted_to_user_present) { ++ deflection.from.number.presentation = ++ invoke->args.etsi.CallDeflection.presentation_allowed_to_diverted_to_user ++ ? PRI_PRES_ALLOWED | PRI_PRES_USER_NUMBER_UNSCREENED ++ : PRI_PRES_RESTRICTED | PRI_PRES_USER_NUMBER_UNSCREENED; ++ } else { ++ deflection.from.number.presentation = ++ PRI_PRES_RESTRICTED | PRI_PRES_USER_NUMBER_UNSCREENED; ++ } + +- return 0; ++ /* Deflecting to the new address. */ ++ rose_copy_address_to_q931(ctrl, &deflection.to, ++ &invoke->args.etsi.CallDeflection.deflection); ++ deflection.to.number.presentation = deflection.from.number.presentation; ++ ++ deflection.count = (call->redirecting.count < PRI_MAX_REDIRECTS) ++ ? call->redirecting.count + 1 : PRI_MAX_REDIRECTS; ++ deflection.reason = PRI_REDIR_DEFLECTION; ++ if (deflection.count == 1) { ++ deflection.orig_called = deflection.from; ++ deflection.orig_reason = deflection.reason; + } else { +- pri_message(pri, "Unable to handle return error on switchtype %d!\n", pri->switchtype); ++ deflection.orig_called = call->redirecting.orig_called; ++ deflection.orig_reason = call->redirecting.orig_reason; + } + +- } while(0); +- +- return -1; +-} ++ subcmd->cmd = PRI_SUBCMD_REROUTING; ++ subcmd->u.rerouting.invoke_id = invoke->invoke_id; ++ subcmd->u.rerouting.subscription_option = 3;/* notApplicable */ ++ q931_party_id_copy_to_pri(&subcmd->u.rerouting.caller, &call->local_id); ++ q931_party_redirecting_copy_to_pri(&subcmd->u.rerouting.deflection, ++ &deflection); ++ break; ++ case ROSE_ETSI_CallRerouting: ++ if (!PRI_MASTER(ctrl)->deflection_support) { ++ send_facility_error(ctrl, call, invoke->invoke_id, ++ ROSE_ERROR_Gen_NotSubscribed); ++ break; ++ } ++ subcmd = q931_alloc_subcommand(ctrl); ++ if (!subcmd) { ++ send_facility_error(ctrl, call, invoke->invoke_id, ++ ROSE_ERROR_Gen_ResourceUnavailable); ++ pri_error(ctrl, "ERROR: Too many facility subcommands\n"); ++ break; ++ } + +-int rose_return_result_decode(struct pri *pri, q931_call *call, q931_ie *ie, unsigned char *data, int len) +-{ +- int i = 0; +- int operationidvalue = -1; +- int invokeidvalue = -1; +- unsigned char *vdata = data; +- struct rose_component *comp = NULL; +- +- do { +- /* Invoke ID stuff */ +- GET_COMPONENT(comp, i, vdata, len); +- CHECK_COMPONENT(comp, INVOKE_IDENTIFIER, "Don't know what to do if first ROSE component is of type 0x%x\n"); +- ASN1_GET_INTEGER(comp, invokeidvalue); +- NEXT_COMPONENT(comp, i); ++ q931_party_redirecting_init(&deflection); + +- if (pri->switchtype == PRI_SWITCH_DMS100) { +- switch (invokeidvalue) { +- case RLT_THIRD_PARTY: +- if (pri->debug & PRI_DEBUG_APDU) pri_message(pri, "Successfully completed RLT transfer!\n"); +- return 0; +- case RLT_OPERATION_IND: +- if (pri->debug & PRI_DEBUG_APDU) pri_message(pri, "Received RLT_OPERATION_IND\n"); +- /* Have to take out the rlt_call_id */ +- GET_COMPONENT(comp, i, vdata, len); +- CHECK_COMPONENT(comp, ASN1_SEQUENCE, "Protocol error detected in parsing RLT_OPERATION_IND return result!\n"); ++ /* Rerouting from the last address. */ ++ rose_copy_presented_number_unscreened_to_q931(ctrl, &deflection.from.number, ++ &invoke->args.etsi.CallRerouting.last_rerouting); + +- /* Traverse the contents of this sequence */ +- /* First is the Operation Value */ +- SUB_COMPONENT(comp, i); +- GET_COMPONENT(comp, i, vdata, len); +- CHECK_COMPONENT(comp, ASN1_INTEGER, "RLT_OPERATION_IND should be of type ASN1_INTEGER!\n"); +- ASN1_GET_INTEGER(comp, operationidvalue); ++ /* Rerouting to the new address. */ ++ rose_copy_address_to_q931(ctrl, &deflection.to, ++ &invoke->args.etsi.CallRerouting.called_address); ++ switch (invoke->args.etsi.CallRerouting.subscription_option) { ++ default: ++ case 0: /* noNotification */ ++ case 1: /* notificationWithoutDivertedToNr */ ++ deflection.to.number.presentation = ++ PRI_PRES_RESTRICTED | PRI_PRES_USER_NUMBER_UNSCREENED; ++ break; ++ case 2: /* notificationWithDivertedToNr */ ++ deflection.to.number.presentation = ++ PRI_PRES_ALLOWED | PRI_PRES_USER_NUMBER_UNSCREENED; ++ break; ++ } + +- if (operationidvalue != RLT_OPERATION_IND) { +- pri_message(pri, "Invalid Operation ID value (0x%x) in return result!\n", operationidvalue); +- return -1; +- } ++ /* Calling party subaddress update. */ ++ party_id = call->local_id; + +- /* Next is the Call ID */ +- NEXT_COMPONENT(comp, i); +- GET_COMPONENT(comp, i, vdata, len); +- CHECK_COMPONENT(comp, ASN1_TAG_0, "Error check failed on Call ID!\n"); +- ASN1_GET_INTEGER(comp, call->rlt_call_id); +- /* We have enough data to transfer the call */ +- call->transferable = 1; ++ deflection.count = invoke->args.etsi.CallRerouting.rerouting_counter; ++ deflection.reason = redirectingreason_for_q931(ctrl, ++ invoke->args.etsi.CallRerouting.rerouting_reason); ++ if (deflection.count == 1) { ++ deflection.orig_called = deflection.from; ++ deflection.orig_reason = deflection.reason; ++ } else { ++ deflection.orig_called = call->redirecting.orig_called; ++ deflection.orig_reason = call->redirecting.orig_reason; ++ } + +- return 0; +- +- default: +- pri_message(pri, "Could not parse invoke of type 0x%x!\n", invokeidvalue); +- return -1; ++ subcmd->cmd = PRI_SUBCMD_REROUTING; ++ subcmd->u.rerouting.invoke_id = invoke->invoke_id; ++ subcmd->u.rerouting.subscription_option = ++ invoke->args.etsi.CallRerouting.subscription_option; ++ q931_party_id_copy_to_pri(&subcmd->u.rerouting.caller, &party_id); ++ q931_party_redirecting_copy_to_pri(&subcmd->u.rerouting.deflection, ++ &deflection); ++ break; ++#if 0 /* Not handled yet */ ++ case ROSE_ETSI_InterrogateServedUserNumbers: ++ break; ++#endif /* Not handled yet */ ++ case ROSE_ETSI_DivertingLegInformation1: ++ if (invoke->args.etsi.DivertingLegInformation1.diverted_to_present) { ++ rose_copy_presented_number_unscreened_to_q931(ctrl, &party_id.number, ++ &invoke->args.etsi.DivertingLegInformation1.diverted_to); ++ /* ++ * We set the presentation value since the sender cannot know the ++ * presentation value preference of the destination party. ++ */ ++ if (party_id.number.str[0]) { ++ party_id.number.presentation = ++ PRI_PRES_ALLOWED | PRI_PRES_USER_NUMBER_UNSCREENED; ++ } else { ++ party_id.number.presentation = ++ PRI_PRES_UNAVAILABLE | PRI_PRES_USER_NUMBER_UNSCREENED; + } +- } else if (pri->switchtype == PRI_SWITCH_QSIG) { +- switch (invokeidvalue) { +- case 0x13: +- if (pri->debug & PRI_DEBUG_APDU) pri_message(pri, "Successfully completed QSIG CF callRerouting!\n"); +- return 0; +- } + } else { +- pri_message(pri, "Unable to handle return result on switchtype %d!\n", pri->switchtype); +- return -1; ++ q931_party_number_init(&party_id.number); ++ party_id.number.valid = 1; + } + +- } while(0); +- +- return -1; +-} ++ /* ++ * Unless otherwise indicated by CONNECT, the divertedToNumber will be ++ * the remote_id.number. ++ */ ++ if (!call->connected_number_in_message) { ++ call->remote_id.number = party_id.number; ++ } + +-int rose_invoke_decode(struct pri *pri, q931_call *call, q931_ie *ie, unsigned char *data, int len) +-{ +- int i = 0; +- int res = 0; +- int operation_tag; +- unsigned char *vdata = data; +- struct rose_component *comp = NULL, *invokeid = NULL, *operationid = NULL; +- +- do { +- /* Invoke ID stuff */ +- GET_COMPONENT(comp, i, vdata, len); +-#if 0 +- CHECK_COMPONENT(comp, INVOKE_IDENTIFIER, "Don't know what to do if first ROSE component is of type 0x%x\n"); +-#endif +- invokeid = comp; +- NEXT_COMPONENT(comp, i); ++ /* divertedToNumber is put in redirecting.to.number */ ++ switch (invoke->args.etsi.DivertingLegInformation1.subscription_option) { ++ default: ++ case 0: /* noNotification */ ++ case 1: /* notificationWithoutDivertedToNr */ ++ q931_party_number_init(&call->redirecting.to.number); ++ call->redirecting.to.number.valid = 1; ++ call->redirecting.to.number.presentation = ++ PRI_PRES_RESTRICTED | PRI_PRES_USER_NUMBER_UNSCREENED; ++ break; ++ case 2: /* notificationWithDivertedToNr */ ++ call->redirecting.to.number = party_id.number; ++ break; ++ } + +- /* Operation Tag */ +- GET_COMPONENT(comp, i, vdata, len); +-#if 0 +- CHECK_COMPONENT(comp, ASN1_INTEGER, "Don't know what to do if second ROSE component is of type 0x%x\n"); +-#endif +- operationid = comp; +- ASN1_GET_INTEGER(comp, operation_tag); +- NEXT_COMPONENT(comp, i); ++ call->redirecting.reason = redirectingreason_for_q931(ctrl, ++ invoke->args.etsi.DivertingLegInformation1.diversion_reason); ++ if (call->redirecting.count < PRI_MAX_REDIRECTS) { ++ ++call->redirecting.count; ++ } ++ call->redirecting.state = Q931_REDIRECTING_STATE_EXPECTING_RX_DIV_LEG_3; ++ break; ++ case ROSE_ETSI_DivertingLegInformation2: ++ call->redirecting.state = Q931_REDIRECTING_STATE_PENDING_TX_DIV_LEG_3; ++ call->redirecting.count = ++ invoke->args.etsi.DivertingLegInformation2.diversion_counter; ++ if (!call->redirecting.count) { ++ /* To be safe, make sure that the count is non-zero. */ ++ call->redirecting.count = 1; ++ } ++ call->redirecting.reason = redirectingreason_for_q931(ctrl, ++ invoke->args.etsi.DivertingLegInformation2.diversion_reason); + +- /* No argument - return with error */ +- if (i >= len) +- return -1; ++ /* divertingNr is put in redirecting.from.number */ ++ if (invoke->args.etsi.DivertingLegInformation2.diverting_present) { ++ rose_copy_presented_number_unscreened_to_q931(ctrl, ++ &call->redirecting.from.number, ++ &invoke->args.etsi.DivertingLegInformation2.diverting); ++ } else if (!call->redirecting_number_in_message) { ++ q931_party_number_init(&call->redirecting.from.number); ++ call->redirecting.from.number.valid = 1; ++ } + +- /* Arguement Tag */ +- GET_COMPONENT(comp, i, vdata, len); +- if (!comp->type) +- return -1; ++ call->redirecting.orig_reason = PRI_REDIR_UNKNOWN; + +- if (pri->debug & PRI_DEBUG_APDU) +- pri_message(pri, " [ Handling operation %d ]\n", operation_tag); +- switch (operation_tag) { +- case SS_CNID_CALLINGNAME: +- if (pri->debug & PRI_DEBUG_APDU) +- pri_message(pri, " Handle Name display operation\n"); +- return rose_calling_name_decode(pri, call, comp, len-i); +- case ROSE_CALL_TRANSFER_IDENTIFY: +- if (pri->debug & PRI_DEBUG_APDU) +- pri_message(pri, "ROSE %i: CallTransferIdentify - not handled!\n", operation_tag); +- dump_apdu (pri, (u_int8_t *)comp, comp->len + 2); +- return -1; +- case ROSE_CALL_TRANSFER_ABANDON: +- if (pri->debug & PRI_DEBUG_APDU) +- pri_message(pri, "ROSE %i: CallTransferAbandon - not handled!\n", operation_tag); +- dump_apdu (pri, (u_int8_t *)comp, comp->len + 2); +- return -1; +- case ROSE_CALL_TRANSFER_INITIATE: +- if (pri->debug & PRI_DEBUG_APDU) +- pri_message(pri, "ROSE %i: CallTransferInitiate - not handled!\n", operation_tag); +- dump_apdu (pri, (u_int8_t *)comp, comp->len + 2); +- return -1; +- case ROSE_CALL_TRANSFER_SETUP: +- if (pri->debug & PRI_DEBUG_APDU) +- pri_message(pri, "ROSE %i: CallTransferSetup - not handled!\n", operation_tag); +- dump_apdu (pri, (u_int8_t *)comp, comp->len + 2); +- return -1; +- case ROSE_CALL_TRANSFER_ACTIVE: +- if (pri->debug & PRI_DEBUG_APDU) +- pri_message(pri, "ROSE %i: CallTransferActive - not handled!\n", operation_tag); +- dump_apdu (pri, (u_int8_t *)comp, comp->len + 2); +- return -1; +- case ROSE_CALL_TRANSFER_COMPLETE: +- if (pri->debug & PRI_DEBUG_APDU) +- { +- pri_message(pri, "ROSE %i: Handle CallTransferComplete\n", operation_tag); +- dump_apdu (pri, (u_int8_t *)comp, comp->len + 2); ++ /* originalCalledNr is put in redirecting.orig_called.number */ ++ if (invoke->args.etsi.DivertingLegInformation2.original_called_present) { ++ rose_copy_presented_number_unscreened_to_q931(ctrl, ++ &call->redirecting.orig_called.number, ++ &invoke->args.etsi.DivertingLegInformation2.original_called); ++ } else { ++ q931_party_number_init(&call->redirecting.orig_called.number); ++ } ++ break; ++ case ROSE_ETSI_DivertingLegInformation3: ++ /* ++ * Unless otherwise indicated by CONNECT, this will be the ++ * remote_id.number.presentation. ++ */ ++ if (!invoke->args.etsi.DivertingLegInformation3.presentation_allowed_indicator) { ++ call->redirecting.to.number.presentation = ++ PRI_PRES_RESTRICTED | PRI_PRES_USER_NUMBER_UNSCREENED; ++ if (!call->connected_number_in_message) { ++ call->remote_id.number.presentation = ++ PRI_PRES_RESTRICTED | PRI_PRES_USER_NUMBER_UNSCREENED; + } +- return rose_call_transfer_complete_decode(pri, call, comp, len-i); +- case ROSE_CALL_TRANSFER_UPDATE: +- if (pri->debug & PRI_DEBUG_APDU) +- { +- pri_message(pri, "ROSE %i: Handle CallTransferUpdate\n", operation_tag); +- dump_apdu (pri, (u_int8_t *)comp, comp->len + 2); ++ } ++ ++ switch (call->redirecting.state) { ++ case Q931_REDIRECTING_STATE_EXPECTING_RX_DIV_LEG_3: ++ call->redirecting.state = Q931_REDIRECTING_STATE_IDLE; ++ subcmd = q931_alloc_subcommand(ctrl); ++ if (!subcmd) { ++ pri_error(ctrl, "ERROR: Too many facility subcommands\n"); ++ break; + } +- return rose_call_transfer_update_decode(pri, call, comp, len-i); +- case ROSE_SUBADDRESS_TRANSFER: +- if (pri->debug & PRI_DEBUG_APDU) +- pri_message(pri, "ROSE %i: SubaddressTransfer - not handled!\n", operation_tag); +- dump_apdu (pri, (u_int8_t *)comp, comp->len + 2); +- return -1; +- case ROSE_DIVERTING_LEG_INFORMATION2: +- if (pri->debug & PRI_DEBUG_APDU) { +- pri_message(pri, "ROSE %i: Handle CallingName\n", operation_tag); +- dump_apdu (pri, (u_int8_t *)comp, comp->len + 2); +- } +- return rose_diverting_leg_information2_decode(pri, call, comp, len-i); +- case ROSE_AOC_NO_CHARGING_INFO_AVAILABLE: +- if (pri->debug & PRI_DEBUG_APDU) { +- pri_message(pri, "ROSE %i: AOC No Charging Info Available - not handled!", operation_tag); +- dump_apdu (pri, comp->data, comp->len); +- } +- return -1; +- case ROSE_AOC_CHARGING_REQUEST: +- return aoc_aoce_charging_request_decode(pri, call, (u_int8_t *)comp, comp->len + 2); +- case ROSE_AOC_AOCS_CURRENCY: +- if (pri->debug & PRI_DEBUG_APDU) { +- pri_message(pri, "ROSE %i: AOC-S Currency - not handled!", operation_tag); +- dump_apdu (pri, (u_int8_t *)comp, comp->len + 2); +- } +- return -1; +- case ROSE_AOC_AOCS_SPECIAL_ARR: +- if (pri->debug & PRI_DEBUG_APDU) { +- pri_message(pri, "ROSE %i: AOC-S Special Array - not handled!", operation_tag); +- dump_apdu (pri, (u_int8_t *)comp, comp->len + 2); +- } +- return -1; +- case ROSE_AOC_AOCD_CURRENCY: +- if (pri->debug & PRI_DEBUG_APDU) { +- pri_message(pri, "ROSE %i: AOC-D Currency - not handled!", operation_tag); +- dump_apdu (pri, (u_int8_t *)comp, comp->len + 2); +- } +- return -1; +- case ROSE_AOC_AOCD_CHARGING_UNIT: +- if (pri->debug & PRI_DEBUG_APDU) { +- pri_message(pri, "ROSE %i: AOC-D Charging Unit - not handled!", operation_tag); +- dump_apdu (pri, (u_int8_t *)comp, comp->len + 2); +- } +- return -1; +- case ROSE_AOC_AOCE_CURRENCY: +- if (pri->debug & PRI_DEBUG_APDU) { +- pri_message(pri, "ROSE %i: AOC-E Currency - not handled!", operation_tag); +- dump_apdu (pri, (u_int8_t *)comp, comp->len + 2); +- } +- return -1; +- case ROSE_AOC_AOCE_CHARGING_UNIT: +- return aoc_aoce_charging_unit_decode(pri, call, (u_int8_t *)comp, comp->len + 2); +- if (0) { /* the following function is currently not used - just to make the compiler happy */ +- aoc_aoce_charging_unit_encode(pri, call, call->aoc_units); /* use this function to forward the aoc-e on a bridged channel */ +- return 0; +- } +- case ROSE_AOC_IDENTIFICATION_OF_CHARGE: +- if (pri->debug & PRI_DEBUG_APDU) { +- pri_message(pri, "ROSE %i: AOC Identification Of Charge - not handled!", operation_tag); +- dump_apdu (pri, (u_int8_t *)comp, comp->len + 2); +- } +- return -1; +- case SS_ANFPR_PATHREPLACEMENT: +- /* Clear Queue */ +- res = pri_call_apdu_queue_cleanup(call->bridged_call); +- if (res) { +- pri_message(pri, "Could not Clear queue ADPU\n"); +- return -1; +- } +- anfpr_pathreplacement_respond(pri, call, ie); +- break; ++ /* Setup redirecting subcommand */ ++ subcmd->cmd = PRI_SUBCMD_REDIRECTING; ++ q931_party_redirecting_copy_to_pri(&subcmd->u.redirecting, ++ &call->redirecting); ++ break; + default: +- if (pri->debug & PRI_DEBUG_APDU) { +- pri_message(pri, "!! Unable to handle ROSE operation %d", operation_tag); +- dump_apdu (pri, (u_int8_t *)comp, comp->len + 2); ++ break; ++ } ++ break; ++ case ROSE_ETSI_ChargingRequest: ++ /* Ignore messsage */ ++ break; ++#if 0 /* Not handled yet */ ++ case ROSE_ETSI_AOCSCurrency: ++ break; ++ case ROSE_ETSI_AOCSSpecialArr: ++ break; ++ case ROSE_ETSI_AOCDCurrency: ++ break; ++ case ROSE_ETSI_AOCDChargingUnit: ++ break; ++ case ROSE_ETSI_AOCECurrency: ++ break; ++#endif /* Not handled yet */ ++ case ROSE_ETSI_AOCEChargingUnit: ++ call->aoc_units = 0; ++ if (invoke->args.etsi.AOCEChargingUnit.type == 1 ++ && !invoke->args.etsi.AOCEChargingUnit.charging_unit.free_of_charge) { ++ unsigned index; ++ ++ for (index = ++ invoke->args.etsi.AOCEChargingUnit.charging_unit.specific.recorded. ++ num_records; index--;) { ++ if (!invoke->args.etsi.AOCEChargingUnit.charging_unit.specific.recorded. ++ list[index].not_available) { ++ call->aoc_units += ++ invoke->args.etsi.AOCEChargingUnit.charging_unit.specific. ++ recorded.list[index].number_of_units; ++ } + } +- return -1; + } +- } while(0); +- +- return -1; +-} ++ /* the following function is currently not used - just to make the compiler happy */ ++ if (0) { ++ /* use this function to forward the aoc-e on a bridged channel */ ++ aoc_aoce_charging_unit_encode(ctrl, call, call->aoc_units); ++ } ++ break; ++#if 0 /* Not handled yet */ ++ case ROSE_ITU_IdentificationOfCharge: ++ break; ++#endif /* Not handled yet */ ++#if 0 /* Not handled yet */ ++ case ROSE_ETSI_EctExecute: ++ break; ++ case ROSE_ETSI_ExplicitEctExecute: ++ break; ++#endif /* Not handled yet */ ++ case ROSE_ETSI_RequestSubaddress: ++ /* Ignore since we are not handling subaddresses yet. */ ++ break; ++#if 0 /* Not handled yet */ ++ case ROSE_ETSI_SubaddressTransfer: ++ break; ++ case ROSE_ETSI_EctLinkIdRequest: ++ break; ++#endif /* Not handled yet */ ++ case ROSE_ETSI_EctInform: ++ /* redirectionNumber is put in remote_id.number */ ++ if (invoke->args.etsi.EctInform.redirection_present) { ++ rose_copy_presented_number_unscreened_to_q931(ctrl, ++ &call->remote_id.number, &invoke->args.etsi.EctInform.redirection); ++ } ++ if (!invoke->args.etsi.EctInform.status) { ++ /* The remote party for the transfer has not answered yet. */ ++ call->incoming_ct_state = INCOMING_CT_STATE_EXPECT_CT_ACTIVE; ++ } else { ++ call->incoming_ct_state = INCOMING_CT_STATE_POST_CONNECTED_LINE; ++ } ++ break; ++#if 0 /* Not handled yet */ ++ case ROSE_ETSI_EctLoopTest: ++ break; ++#endif /* Not handled yet */ ++ case ROSE_QSIG_CallingName: ++ /* CallingName is put in remote_id.name */ ++ rose_copy_name_to_q931(ctrl, &call->remote_id.name, ++ &invoke->args.qsig.CallingName.name); ++ break; ++ case ROSE_QSIG_CalledName: ++ /* CalledName is put in remote_id.name */ ++ rose_copy_name_to_q931(ctrl, &call->remote_id.name, ++ &invoke->args.qsig.CalledName.name); + +-int pri_call_apdu_queue(q931_call *call, int messagetype, void *apdu, int apdu_len, void (*function)(void *data), void *data) +-{ +- struct apdu_event *cur = NULL; +- struct apdu_event *new_event = NULL; ++ /* Setup connected line subcommand */ ++ subcmd = q931_alloc_subcommand(ctrl); ++ if (!subcmd) { ++ pri_error(ctrl, "ERROR: Too many facility subcommands\n"); ++ break; ++ } ++ subcmd->cmd = PRI_SUBCMD_CONNECTED_LINE; ++ q931_party_id_copy_to_pri(&subcmd->u.connected_line.id, &call->remote_id); ++ break; ++ case ROSE_QSIG_ConnectedName: ++ /* ConnectedName is put in remote_id.name */ ++ rose_copy_name_to_q931(ctrl, &call->remote_id.name, ++ &invoke->args.qsig.ConnectedName.name); ++ break; ++#if 0 /* Not handled yet */ ++ case ROSE_QSIG_BusyName: ++ break; ++#endif /* Not handled yet */ ++#if 0 /* Not handled yet */ ++ case ROSE_QSIG_ChargeRequest: ++ break; ++ case ROSE_QSIG_GetFinalCharge: ++ break; ++ case ROSE_QSIG_AocFinal: ++ break; ++ case ROSE_QSIG_AocInterim: ++ break; ++ case ROSE_QSIG_AocRate: ++ break; ++ case ROSE_QSIG_AocComplete: ++ break; ++ case ROSE_QSIG_AocDivChargeReq: ++ break; ++#endif /* Not handled yet */ ++#if 0 /* Not handled yet */ ++ case ROSE_QSIG_CallTransferIdentify: ++ break; ++ case ROSE_QSIG_CallTransferAbandon: ++ break; ++ case ROSE_QSIG_CallTransferInitiate: ++ break; ++ case ROSE_QSIG_CallTransferSetup: ++ break; ++#endif /* Not handled yet */ ++ case ROSE_QSIG_CallTransferActive: ++ call->incoming_ct_state = INCOMING_CT_STATE_POST_CONNECTED_LINE; + +- if (!call || !messagetype || !apdu || (apdu_len < 1) || (apdu_len > 255)) +- return -1; ++ /* connectedAddress is put in remote_id */ ++ rose_copy_presented_address_screened_to_q931(ctrl, &call->remote_id, ++ &invoke->args.qsig.CallTransferActive.connected); + +- if (!(new_event = calloc(1, sizeof(*new_event)))) { +- pri_error(call->pri, "!! Malloc failed!\n"); +- return -1; +- } ++ /* connectedName is put in remote_id.name */ ++ if (invoke->args.qsig.CallTransferActive.connected_name_present) { ++ rose_copy_name_to_q931(ctrl, &call->remote_id.name, ++ &invoke->args.qsig.CallTransferActive.connected_name); ++ } ++ break; ++ case ROSE_QSIG_CallTransferComplete: ++ /* redirectionNumber is put in remote_id.number */ ++ rose_copy_presented_number_screened_to_q931(ctrl, &call->remote_id.number, ++ &invoke->args.qsig.CallTransferComplete.redirection); + +- new_event->message = messagetype; +- new_event->callback = function; +- new_event->data = data; +- memcpy(new_event->apdu, apdu, apdu_len); +- new_event->apdu_len = apdu_len; +- +- if (call->apdus) { +- cur = call->apdus; +- while (cur->next) { +- cur = cur->next; ++ /* redirectionName is put in remote_id.name */ ++ if (invoke->args.qsig.CallTransferComplete.redirection_name_present) { ++ rose_copy_name_to_q931(ctrl, &call->remote_id.name, ++ &invoke->args.qsig.CallTransferComplete.redirection_name); + } +- cur->next = new_event; +- } else +- call->apdus = new_event; + +- return 0; +-} ++ if (invoke->args.qsig.CallTransferComplete.call_status == 1) { ++ /* The remote party for the transfer has not answered yet. */ ++ call->incoming_ct_state = INCOMING_CT_STATE_EXPECT_CT_ACTIVE; ++ } else { ++ call->incoming_ct_state = INCOMING_CT_STATE_POST_CONNECTED_LINE; ++ } ++ break; ++ case ROSE_QSIG_CallTransferUpdate: ++ party_id = call->remote_id; + +-int pri_call_apdu_queue_cleanup(q931_call *call) +-{ +- struct apdu_event *cur_event = NULL, *free_event = NULL; ++ /* redirectionNumber is put in party_id.number */ ++ rose_copy_presented_number_screened_to_q931(ctrl, &party_id.number, ++ &invoke->args.qsig.CallTransferUpdate.redirection); + +- if (call && call->apdus) { +- cur_event = call->apdus; +- while (cur_event) { +- /* TODO: callbacks, some way of giving return res on status of apdu */ +- free_event = cur_event; +- cur_event = cur_event->next; +- free(free_event); ++ /* redirectionName is put in party_id.name */ ++ if (invoke->args.qsig.CallTransferUpdate.redirection_name_present) { ++ rose_copy_name_to_q931(ctrl, &party_id.name, ++ &invoke->args.qsig.CallTransferUpdate.redirection_name); + } +- call->apdus = NULL; +- } + +- return 0; +-} +- +-int pri_call_add_standard_apdus(struct pri *pri, q931_call *call) +-{ +- if (!pri->sendfacility) +- return 0; +- +- if (pri->switchtype == PRI_SWITCH_QSIG) { /* For Q.SIG it does network and cpe operations */ +- if (call->redirectingnum[0]) +- rose_diverting_leg_information2_encode(pri, call); +- add_callername_facility_ies(pri, call, 1); +- return 0; +- } +- +-#if 0 +- if (pri->localtype == PRI_NETWORK) { +- switch (pri->switchtype) { +- case PRI_SWITCH_NI2: +- add_callername_facility_ies(pri, call, 0); ++ if (q931_party_id_cmp(&party_id, &call->remote_id)) { ++ /* The remote_id data has changed. */ ++ call->remote_id = party_id; ++ switch (call->incoming_ct_state) { ++ case INCOMING_CT_STATE_IDLE: ++ call->incoming_ct_state = INCOMING_CT_STATE_POST_CONNECTED_LINE; + break; + default: + break; ++ } + } +- return 0; +- } else if (pri->localtype == PRI_CPE) { +- switch (pri->switchtype) { +- case PRI_SWITCH_NI2: +- add_callername_facility_ies(pri, call, 1); +- break; +- default: +- break; ++ break; ++#if 0 /* Not handled yet */ ++ case ROSE_QSIG_SubaddressTransfer: ++ break; ++#endif /* Not handled yet */ ++ case ROSE_QSIG_PathReplacement: ++ anfpr_pathreplacement_respond(ctrl, call, ie); ++ break; ++#if 0 /* Not handled yet */ ++ case ROSE_QSIG_ActivateDiversionQ: ++ break; ++ case ROSE_QSIG_DeactivateDiversionQ: ++ break; ++ case ROSE_QSIG_InterrogateDiversionQ: ++ break; ++ case ROSE_QSIG_CheckRestriction: ++ break; ++#endif /* Not handled yet */ ++ case ROSE_QSIG_CallRerouting: ++ if (!PRI_MASTER(ctrl)->deflection_support) { ++ send_facility_error(ctrl, call, invoke->invoke_id, ++ ROSE_ERROR_Gen_NotSubscribed); ++ break; + } +- return 0; +- } +-#else +- if (pri->switchtype == PRI_SWITCH_NI2) +- add_callername_facility_ies(pri, call, (pri->localtype == PRI_CPE)); +-#endif ++ subcmd = q931_alloc_subcommand(ctrl); ++ if (!subcmd) { ++ send_facility_error(ctrl, call, invoke->invoke_id, ++ ROSE_ERROR_Gen_ResourceUnavailable); ++ pri_error(ctrl, "ERROR: Too many facility subcommands\n"); ++ break; ++ } + +- if ((pri->switchtype == PRI_SWITCH_DMS100) && (pri->localtype == PRI_CPE)) { +- add_dms100_transfer_ability_apdu(pri, call); +- } ++ q931_party_redirecting_init(&deflection); + ++ /* Rerouting from the last address. */ ++ rose_copy_presented_number_unscreened_to_q931(ctrl, &deflection.from.number, ++ &invoke->args.qsig.CallRerouting.last_rerouting); ++ if (invoke->args.qsig.CallRerouting.redirecting_name_present) { ++ rose_copy_name_to_q931(ctrl, &deflection.from.name, ++ &invoke->args.qsig.CallRerouting.redirecting_name); ++ } + ++ /* Rerouting to the new address. */ ++ rose_copy_address_to_q931(ctrl, &deflection.to, ++ &invoke->args.qsig.CallRerouting.called); ++ switch (invoke->args.qsig.CallRerouting.subscription_option) { ++ default: ++ case 0: /* noNotification */ ++ case 1: /* notificationWithoutDivertedToNr */ ++ deflection.to.number.presentation = ++ PRI_PRES_RESTRICTED | PRI_PRES_USER_NUMBER_UNSCREENED; ++ break; ++ case 2: /* notificationWithDivertedToNr */ ++ deflection.to.number.presentation = ++ PRI_PRES_ALLOWED | PRI_PRES_USER_NUMBER_UNSCREENED; ++ break; ++ } + +- return 0; ++ /* Calling party update. */ ++ party_id = call->local_id; ++ rose_copy_presented_number_screened_to_q931(ctrl, &party_id.number, ++ &invoke->args.qsig.CallRerouting.calling); ++ if (invoke->args.qsig.CallRerouting.calling_name_present) { ++ rose_copy_name_to_q931(ctrl, &party_id.name, ++ &invoke->args.qsig.CallRerouting.calling_name); ++ } ++ ++ deflection.count = invoke->args.qsig.CallRerouting.diversion_counter; ++ deflection.reason = redirectingreason_for_q931(ctrl, ++ invoke->args.qsig.CallRerouting.rerouting_reason); ++ ++ /* Original called party update. */ ++ if (deflection.count == 1) { ++ deflection.orig_called = deflection.from; ++ deflection.orig_reason = deflection.reason; ++ } else { ++ deflection.orig_called = call->redirecting.orig_called; ++ deflection.orig_reason = call->redirecting.orig_reason; ++ } ++ if (invoke->args.qsig.CallRerouting.original_called_present) { ++ rose_copy_presented_number_unscreened_to_q931(ctrl, ++ &deflection.orig_called.number, ++ &invoke->args.qsig.CallRerouting.original_called); ++ } ++ if (invoke->args.qsig.CallRerouting.original_called_name_present) { ++ rose_copy_name_to_q931(ctrl, &deflection.orig_called.name, ++ &invoke->args.qsig.CallRerouting.original_called_name); ++ } ++ if (invoke->args.qsig.CallRerouting.original_rerouting_reason_present) { ++ deflection.orig_reason = redirectingreason_for_q931(ctrl, ++ invoke->args.qsig.CallRerouting.original_rerouting_reason); ++ } ++ ++ subcmd->cmd = PRI_SUBCMD_REROUTING; ++ subcmd->u.rerouting.invoke_id = invoke->invoke_id; ++ subcmd->u.rerouting.subscription_option = ++ invoke->args.qsig.CallRerouting.subscription_option; ++ q931_party_id_copy_to_pri(&subcmd->u.rerouting.caller, &party_id); ++ q931_party_redirecting_copy_to_pri(&subcmd->u.rerouting.deflection, ++ &deflection); ++ break; ++ case ROSE_QSIG_DivertingLegInformation1: ++ q931_party_number_init(&party_id.number); ++ rose_copy_number_to_q931(ctrl, &party_id.number, ++ &invoke->args.qsig.DivertingLegInformation1.nominated_number); ++ if (party_id.number.str[0]) { ++ party_id.number.presentation = ++ PRI_PRES_ALLOWED | PRI_PRES_USER_NUMBER_UNSCREENED; ++ } ++ ++ /* ++ * Unless otherwise indicated by CONNECT, the nominatedNr will be ++ * the remote_id.number. ++ */ ++ if (!call->connected_number_in_message) { ++ call->remote_id.number = party_id.number; ++ } ++ ++ /* nominatedNr is put in redirecting.to.number */ ++ switch (invoke->args.qsig.DivertingLegInformation1.subscription_option) { ++ default: ++ case QSIG_NO_NOTIFICATION: ++ case QSIG_NOTIFICATION_WITHOUT_DIVERTED_TO_NR: ++ q931_party_number_init(&call->redirecting.to.number); ++ call->redirecting.to.number.valid = 1; ++ call->redirecting.to.number.presentation = ++ PRI_PRES_RESTRICTED | PRI_PRES_USER_NUMBER_UNSCREENED; ++ break; ++ case QSIG_NOTIFICATION_WITH_DIVERTED_TO_NR: ++ call->redirecting.to.number = party_id.number; ++ break; ++ } ++ ++ call->redirecting.reason = redirectingreason_for_q931(ctrl, ++ invoke->args.qsig.DivertingLegInformation1.diversion_reason); ++ if (call->redirecting.count < PRI_MAX_REDIRECTS) { ++ ++call->redirecting.count; ++ } ++ call->redirecting.state = Q931_REDIRECTING_STATE_EXPECTING_RX_DIV_LEG_3; ++ break; ++ case ROSE_QSIG_DivertingLegInformation2: ++ call->redirecting.state = Q931_REDIRECTING_STATE_PENDING_TX_DIV_LEG_3; ++ call->redirecting.count = ++ invoke->args.qsig.DivertingLegInformation2.diversion_counter; ++ if (!call->redirecting.count) { ++ /* To be safe, make sure that the count is non-zero. */ ++ call->redirecting.count = 1; ++ } ++ call->redirecting.reason = redirectingreason_for_q931(ctrl, ++ invoke->args.qsig.DivertingLegInformation2.diversion_reason); ++ ++ /* divertingNr is put in redirecting.from.number */ ++ if (invoke->args.qsig.DivertingLegInformation2.diverting_present) { ++ rose_copy_presented_number_unscreened_to_q931(ctrl, ++ &call->redirecting.from.number, ++ &invoke->args.qsig.DivertingLegInformation2.diverting); ++ } else if (!call->redirecting_number_in_message) { ++ q931_party_number_init(&call->redirecting.from.number); ++ call->redirecting.from.number.valid = 1; ++ } ++ ++ /* redirectingName is put in redirecting.from.name */ ++ if (invoke->args.qsig.DivertingLegInformation2.redirecting_name_present) { ++ rose_copy_name_to_q931(ctrl, &call->redirecting.from.name, ++ &invoke->args.qsig.DivertingLegInformation2.redirecting_name); ++ } else { ++ q931_party_name_init(&call->redirecting.from.name); ++ } ++ ++ call->redirecting.orig_reason = PRI_REDIR_UNKNOWN; ++ if (invoke->args.qsig.DivertingLegInformation2.original_diversion_reason_present) { ++ call->redirecting.orig_reason = redirectingreason_for_q931(ctrl, ++ invoke->args.qsig.DivertingLegInformation2.original_diversion_reason); ++ } ++ ++ /* originalCalledNr is put in redirecting.orig_called.number */ ++ if (invoke->args.qsig.DivertingLegInformation2.original_called_present) { ++ rose_copy_presented_number_unscreened_to_q931(ctrl, ++ &call->redirecting.orig_called.number, ++ &invoke->args.qsig.DivertingLegInformation2.original_called); ++ } else { ++ q931_party_number_init(&call->redirecting.orig_called.number); ++ } ++ ++ /* originalCalledName is put in redirecting.orig_called.name */ ++ if (invoke->args.qsig.DivertingLegInformation2.original_called_name_present) { ++ rose_copy_name_to_q931(ctrl, &call->redirecting.orig_called.name, ++ &invoke->args.qsig.DivertingLegInformation2.original_called_name); ++ } else { ++ q931_party_name_init(&call->redirecting.orig_called.name); ++ } ++ break; ++ case ROSE_QSIG_DivertingLegInformation3: ++ /* ++ * Unless otherwise indicated by CONNECT, this will be the ++ * remote_id.number.presentation. ++ */ ++ if (!invoke->args.qsig.DivertingLegInformation3.presentation_allowed_indicator) { ++ call->redirecting.to.number.presentation = ++ PRI_PRES_RESTRICTED | PRI_PRES_USER_NUMBER_UNSCREENED; ++ if (!call->connected_number_in_message) { ++ call->remote_id.number.presentation = ++ PRI_PRES_RESTRICTED | PRI_PRES_USER_NUMBER_UNSCREENED; ++ } ++ } ++ ++ /* redirectionName is put in redirecting.to.name */ ++ if (invoke->args.qsig.DivertingLegInformation3.redirection_name_present) { ++ rose_copy_name_to_q931(ctrl, &call->redirecting.to.name, ++ &invoke->args.qsig.DivertingLegInformation3.redirection_name); ++ if (!invoke->args.qsig.DivertingLegInformation3.presentation_allowed_indicator) { ++ call->redirecting.to.name.presentation = PRI_PRES_RESTRICTED; ++ } ++ } else { ++ q931_party_name_init(&call->redirecting.to.name); ++ } ++ ++ switch (call->redirecting.state) { ++ case Q931_REDIRECTING_STATE_EXPECTING_RX_DIV_LEG_3: ++ call->redirecting.state = Q931_REDIRECTING_STATE_IDLE; ++ subcmd = q931_alloc_subcommand(ctrl); ++ if (!subcmd) { ++ pri_error(ctrl, "ERROR: Too many facility subcommands\n"); ++ break; ++ } ++ /* Setup redirecting subcommand */ ++ subcmd->cmd = PRI_SUBCMD_REDIRECTING; ++ q931_party_redirecting_copy_to_pri(&subcmd->u.redirecting, ++ &call->redirecting); ++ break; ++ default: ++ break; ++ } ++ break; ++#if 0 /* Not handled yet */ ++ case ROSE_QSIG_CfnrDivertedLegFailed: ++ break; ++#endif /* Not handled yet */ ++#if 0 /* Not handled yet */ ++ case ROSE_QSIG_MWIActivate: ++ break; ++ case ROSE_QSIG_MWIDeactivate: ++ break; ++ case ROSE_QSIG_MWIInterrogate: ++ break; ++#endif /* Not handled yet */ ++ default: ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, "!! ROSE invoke operation not handled! %s\n", ++ rose_operation2str(invoke->operation)); ++ } ++ break; ++ } + } +- +Index: pri_facility.h +=================================================================== +--- a/pri_facility.h (.../tags/1.4.10.2) (revision 1357) ++++ b/pri_facility.h (.../branches/1.4) (revision 1357) +@@ -31,123 +31,21 @@ + #define _PRI_FACILITY_H + #include "pri_q931.h" + ++/* Forward declare some structs */ ++struct fac_extension_header; ++struct rose_msg_invoke; ++struct rose_msg_result; ++struct rose_msg_error; ++struct rose_msg_reject; ++ + /* Protocol Profile field */ ++#define Q932_PROTOCOL_MASK 0x1F + #define Q932_PROTOCOL_ROSE 0x11 /* X.219 & X.229 */ + #define Q932_PROTOCOL_CMIP 0x12 /* Q.941 */ + #define Q932_PROTOCOL_ACSE 0x13 /* X.217 & X.227 */ + #define Q932_PROTOCOL_GAT 0x16 + #define Q932_PROTOCOL_EXTENSIONS 0x1F + +-/* Argument values */ +-#define ROSE_NAME_PRESENTATION_ALLOWED_SIMPLE 0x80 +-#define ROSE_NAME_PRESENTATION_RESTRICTED_NULL 0x87 +-#define ROSE_NAME_PRESENTATION_ALLOWED_EXTENDED 0xA1 +-#define ROSE_NAME_PRESENTATION_RESTRICTED_SIMPLE 0xA2 +-#define ROSE_NAME_PRESENTATION_RESTRICTED_EXTENDED 0xA3 +-#define ROSE_NAME_NOT_AVAIL 0x84 +- +-/* Component types */ +-#define COMP_TYPE_INTERPRETATION 0x8B +-#define COMP_TYPE_NETWORK_PROTOCOL_PROFILE 0x92 +-#define COMP_TYPE_INVOKE 0xA1 +-#define COMP_TYPE_RETURN_RESULT 0xA2 +-#define COMP_TYPE_RETURN_ERROR 0xA3 +-#define COMP_TYPE_REJECT 0xA4 +-#define COMP_TYPE_NFE 0xAA +- +-/* Operation ID values */ +-/* Q.952.7 (ECMA-178) ROSE operations (Transfer) */ +-#define ROSE_CALL_TRANSFER_IDENTIFY 7 +-#define ROSE_CALL_TRANSFER_ABANDON 8 +-#define ROSE_CALL_TRANSFER_INITIATE 9 +-#define ROSE_CALL_TRANSFER_SETUP 10 +-#define ROSE_CALL_TRANSFER_ACTIVE 11 +-#define ROSE_CALL_TRANSFER_COMPLETE 12 +-#define ROSE_CALL_TRANSFER_UPDATE 13 +-#define ROSE_SUBADDRESS_TRANSFER 14 +-/* Q.952 ROSE operations (Diverting) */ +-#define ROSE_DIVERTING_LEG_INFORMATION1 18 +-#define ROSE_DIVERTING_LEG_INFORMATION2 0x15 +-#define ROSE_DIVERTING_LEG_INFORMATION3 19 +-/* Q.956 ROSE operations (Advice Of Charge) */ +-#define ROSE_AOC_NO_CHARGING_INFO_AVAILABLE 26 +-#define ROSE_AOC_CHARGING_REQUEST 30 +-#define ROSE_AOC_AOCS_CURRENCY 31 +-#define ROSE_AOC_AOCS_SPECIAL_ARR 32 +-#define ROSE_AOC_AOCD_CURRENCY 33 +-#define ROSE_AOC_AOCD_CHARGING_UNIT 34 +-#define ROSE_AOC_AOCE_CURRENCY 35 +-#define ROSE_AOC_AOCE_CHARGING_UNIT 36 +-#define ROSE_AOC_IDENTIFICATION_OF_CHARGE 37 +-/* Q.SIG operations */ +-#define SS_CNID_CALLINGNAME 0 +-#define SS_ANFPR_PATHREPLACEMENT 4 +-#define SS_DIVERTING_LEG_INFORMATION2 21 +-#define SS_MWI_ACTIVATE 80 +-#define SS_MWI_DEACTIVATE 81 +-#define SS_MWI_INTERROGATE 82 +- +-/* ROSE definitions and data structures */ +-#define INVOKE_IDENTIFIER 0x02 +-#define INVOKE_LINKED_IDENTIFIER 0x80 +-#define INVOKE_NULL_IDENTIFIER __USE_ASN1_NULL +- +-/* ASN.1 Identifier Octet - Data types */ +-#define ASN1_TYPE_MASK 0x1f +-#define ASN1_BOOLEAN 0x01 +-#define ASN1_INTEGER 0x02 +-#define ASN1_BITSTRING 0x03 +-#define ASN1_OCTETSTRING 0x04 +-#define ASN1_NULL 0x05 +-#define ASN1_OBJECTIDENTIFIER 0x06 +-#define ASN1_OBJECTDESCRIPTOR 0x07 +-#define ASN1_EXTERN 0x08 +-#define ASN1_REAL 0x09 +-#define ASN1_ENUMERATED 0x0a +-#define ASN1_EMBEDDEDPDV 0x0b +-#define ASN1_UTF8STRING 0x0c +-#define ASN1_RELATIVEOBJECTID 0x0d +-/* 0x0e & 0x0f are reserved for future ASN.1 editions */ +-#define ASN1_SEQUENCE 0x10 +-#define ASN1_SET 0x11 +-#define ASN1_NUMERICSTRING 0x12 +-#define ASN1_PRINTABLESTRING 0x13 +-#define ASN1_TELETEXSTRING 0x14 +-#define ASN1_IA5STRING 0x16 +-#define ASN1_UTCTIME 0x17 +-#define ASN1_GENERALIZEDTIME 0x18 +- +-/* ASN.1 Identifier Octet - Tags */ +-#define ASN1_TAG_0 0x00 +-#define ASN1_TAG_1 0x01 +-#define ASN1_TAG_2 0x02 +-#define ASN1_TAG_3 0x03 +-#define ASN1_TAG_4 0x04 +-#define ASN1_TAG_5 0x05 +-#define ASN1_TAG_6 0x06 +-#define ASN1_TAG_7 0x07 +-#define ASN1_TAG_8 0x08 +-#define ASN1_TAG_9 0x09 +- +-/* ASN.1 Identifier Octet - Primitive/Constructor Bit */ +-#define ASN1_PC_MASK 0x20 +-#define ASN1_PRIMITIVE 0x00 +-#define ASN1_CONSTRUCTOR 0x20 +- +-/* ASN.1 Identifier Octet - Clan Bits */ +-#define ASN1_CLAN_MASK 0xc0 +-#define ASN1_UNIVERSAL 0x00 +-#define ASN1_APPLICATION 0x40 +-#define ASN1_CONTEXT_SPECIFIC 0x80 +-#define ASN1_PRIVATE 0xc0 +- +-/* ASN.1 Length masks */ +-#define ASN1_LEN_INDEF 0x80 +- +- +-#define INVOKE_OPERATION_INT __USE_ASN1_INTEGER +-#define INVOKE_OBJECT_ID __USE_ASN1_OBJECTIDENTIFIER +- + /* Q.952 Divert cause */ + #define Q952_DIVERT_REASON_UNKNOWN 0x00 + #define Q952_DIVERT_REASON_CFU 0x01 +@@ -169,138 +67,112 @@ + #define Q932_TON_SUBSCRIBER 0x04 + #define Q932_TON_ABBREVIATED 0x06 + +-/* RLT related Operations */ +-#define RLT_SERVICE_ID 0x3e +-#define RLT_OPERATION_IND 0x01 +-#define RLT_THIRD_PARTY 0x02 ++/* Q.SIG Subscription Option. Listed in ECMA-174 */ ++#define QSIG_NO_NOTIFICATION 0x00 ++#define QSIG_NOTIFICATION_WITHOUT_DIVERTED_TO_NR 0x01 ++#define QSIG_NOTIFICATION_WITH_DIVERTED_TO_NR 0x02 + +-struct rose_component { +- u_int8_t type; +- u_int8_t len; +- u_int8_t data[0]; ++/*! Reasons an APDU callback is called. */ ++enum APDU_CALLBACK_REASON { ++ /*! ++ * \brief Send setup error. Abort and cleanup. ++ * \note The message may or may not actually get sent. ++ * \note The callback cannot generate an event subcmd. ++ * \note The callback should not send messages. Out of order messages will result. ++ */ ++ APDU_CALLBACK_REASON_ERROR, ++ /*! ++ * \brief Abort and cleanup. ++ * \note The APDU queue is being destroyed. ++ * \note The callback cannot generate an event subcmd. ++ * \note The callback cannot send messages as the call is likely being destroyed. ++ */ ++ APDU_CALLBACK_REASON_CLEANUP, ++ /*! ++ * \brief Timeout waiting for responses to the message. ++ * \note The callback can generate an event subcmd. ++ * \note The callback can send messages. ++ */ ++ APDU_CALLBACK_REASON_TIMEOUT, ++ /*! ++ * \brief Received a facility response message. ++ * \note The callback can generate an event subcmd. ++ * \note The callback can send messages. ++ */ ++ APDU_CALLBACK_REASON_MSG_RESULT, ++ /*! ++ * \brief Received a facility error message. ++ * \note The callback can generate an event subcmd. ++ * \note The callback can send messages. ++ */ ++ APDU_CALLBACK_REASON_MSG_ERROR, ++ /*! ++ * \brief Received a facility reject message. ++ * \note The callback can generate an event subcmd. ++ * \note The callback can send messages. ++ */ ++ APDU_CALLBACK_REASON_MSG_REJECT, + }; + +-#if 1 +- #define GET_COMPONENT(component, idx, ptr, length) \ +- if ((idx)+2 > (length)) \ +- break; \ +- (component) = (struct rose_component*)&((ptr)[idx]); \ +- if ((idx)+(component)->len+2 > (length)) { \ +- if ((component)->len != ASN1_LEN_INDEF) \ +- pri_message(pri, "Length (%d) of 0x%X component is too long\n", (component)->len, (component)->type); \ +- } +-#else /* Debugging */ +- #define GET_COMPONENT(component, idx, ptr, length) \ +- if ((idx)+2 > (length)) \ +- break; \ +- (component) = (struct rose_component*)&((ptr)[idx]); \ +- if ((idx)+(component)->len+2 > (length)) { \ +- if ((component)->len != 128) \ +- pri_message(pri, "Length (%d) of 0x%X component is too long\n", (component)->len, (component)->type); \ +- } \ +- pri_message(pri, "XX %s:%d Got component %d (0x%02X), length %d\n", __FUNCTION__, __LINE__, (component)->type, (component)->type, (component)->len); \ +- if ((component)->len > 0) { \ +- int zzz; \ +- pri_message(pri, "XX Data:"); \ +- for (zzz = 0; zzz < (component)->len; ++zzz) \ +- pri_message(pri, " %02X", (component)->data[zzz]); \ +- pri_message(pri, "\n"); \ +- } +-#endif ++union apdu_msg_data { ++ const struct rose_msg_result *result; ++ const struct rose_msg_error *error; ++ const struct rose_msg_reject *reject; ++}; + +-#define NEXT_COMPONENT(component, idx) \ +- (idx) += (component)->len + 2 ++union apdu_callback_param { ++ void *ptr; ++ long value; ++ char pad[8]; ++}; + +-#define SUB_COMPONENT(component, idx) \ +- (idx) += 2 ++struct apdu_callback_data { ++ /*! APDU invoke id to match with any response messages. (Result/Error/Reject) */ ++ int invoke_id; ++ /*! ++ * \brief Time to wait for responses to APDU in ms. ++ * \note Set to 0 if send the message only. ++ * \note Set to less than 0 for PRI_TIMER_T_RESPONSE time. ++ */ ++ int timeout_time; ++ /*! ++ * \brief APDU callback function. ++ * ++ * \param reason Reason callback is called. ++ * \param ctrl D channel controller. ++ * \param call Q.931 call leg. ++ * \param apdu APDU queued entry. Do not change! ++ * \param msg APDU response message data. (NULL if was not the reason called.) ++ * ++ * \note ++ * A callback must be supplied if the sender cares about any APDU_CALLBACK_REASON. ++ * ++ * \return TRUE if no more responses are expected. ++ */ ++ int (*callback)(enum APDU_CALLBACK_REASON reason, struct pri *ctrl, struct q931_call *call, struct apdu_event *apdu, const union apdu_msg_data *msg); ++ /*! \brief Sender data for the callback function to identify the particular APDU. */ ++ union apdu_callback_param user; ++}; + +-#define CHECK_COMPONENT(component, comptype, message) \ +- if ((component)->type && ((component)->type & ASN1_TYPE_MASK) != (comptype)) { \ +- pri_message(pri, (message), (component)->type); \ +- asn1_dump(pri, (component), (component)->len+2); \ +- break; \ +- } +- +-#define ASN1_GET_INTEGER(component, variable) \ +- do { \ +- int comp_idx; \ +- (variable) = 0; \ +- for (comp_idx = 0; comp_idx < (component)->len; ++comp_idx) \ +- (variable) = ((variable) << 8) | (component)->data[comp_idx]; \ +- } while (0) ++struct apdu_event { ++ /*! Linked list pointer */ ++ struct apdu_event *next; ++ /*! TRUE if this APDU has been sent. */ ++ int sent; ++ /*! What message to send the ADPU in */ ++ int message; ++ /*! Sender supplied information to handle APDU response messages. */ ++ struct apdu_callback_data response; ++ /*! Q.931 call leg. (Needed for the APDU timeout.) */ ++ struct q931_call *call; ++ /*! Response timeout timer. */ ++ int timer; ++ /*! Length of ADPU */ ++ int apdu_len; ++ /*! ADPU to send */ ++ unsigned char apdu[255]; ++}; + +-#define ASN1_FIXUP_LEN(component, size) \ +- do { \ +- if ((component)->len == ASN1_LEN_INDEF) \ +- size += 2; \ +- } while (0) +- +-#define ASN1_ADD_SIMPLE(component, comptype, ptr, idx) \ +- do { \ +- (component) = (struct rose_component *)&((ptr)[(idx)]); \ +- (component)->type = (comptype); \ +- (component)->len = 0; \ +- (idx) += 2; \ +- } while (0) +- +-#define ASN1_ADD_BYTECOMP(component, comptype, ptr, idx, value) \ +- do { \ +- (component) = (struct rose_component *)&((ptr)[(idx)]); \ +- (component)->type = (comptype); \ +- (component)->len = 1; \ +- (component)->data[0] = (value); \ +- (idx) += 3; \ +- } while (0) +- +-#define ASN1_ADD_WORDCOMP(component, comptype, ptr, idx, value) \ +- do { \ +- int __val = (value); \ +- int __i = 0; \ +- (component) = (struct rose_component *)&((ptr)[(idx)]); \ +- (component)->type = (comptype); \ +- if ((__val >> 24)) \ +- (component)->data[__i++] = (__val >> 24) & 0xff; \ +- if ((__val >> 16)) \ +- (component)->data[__i++] = (__val >> 16) & 0xff; \ +- if ((__val >> 8)) \ +- (component)->data[__i++] = (__val >> 8) & 0xff; \ +- (component)->data[__i++] = __val & 0xff; \ +- (component)->len = __i; \ +- (idx) += 2 + __i; \ +- } while (0) +- +-#define ASN1_PUSH(stack, stackpointer, component) \ +- (stack)[(stackpointer)++] = (component) +- +-#define ASN1_FIXUP(stack, stackpointer, data, idx) \ +- do { \ +- --(stackpointer); \ +- (stack)[(stackpointer)]->len = (unsigned char *)&((data)[(idx)]) - (unsigned char *)(stack)[(stackpointer)] - 2; \ +- } while (0) +- +-/* Decoder for the invoke ROSE component */ +-int rose_invoke_decode(struct pri *pri, struct q931_call *call, q931_ie *ie, unsigned char *data, int len); +- +-/* Decoder for the return result ROSE component */ +-int rose_return_result_decode(struct pri *pri, struct q931_call *call, q931_ie *ie, unsigned char *data, int len); +- +-/* Decoder for the return error ROSE component */ +-int rose_return_error_decode(struct pri *pri, struct q931_call *call, q931_ie *ie, unsigned char *data, int len); +- +-/* Decoder for the reject ROSE component */ +-int rose_reject_decode(struct pri *pri, struct q931_call *call, q931_ie *ie, unsigned char *data, int len); +- +-int asn1_copy_string(char * buf, int buflen, struct rose_component *comp); +- +-int asn1_string_encode(unsigned char asn1_type, void *data, int len, int max_len, void *src, int src_len); +- +-/* Get Name types from ASN.1 */ +-int asn1_name_decode(void * data, int len, char *namebuf, int buflen); +- +-int typeofnumber_from_q931(struct pri *pri, int ton); +- +-int redirectingreason_from_q931(struct pri *pri, int redirectingreason); +- + /* Queues an MWI apdu on a the given call */ + int mwi_message_send(struct pri *pri, q931_call *call, struct pri_sr *req, int activate); + +@@ -310,21 +182,31 @@ + int rlt_initiate_transfer(struct pri *pri, q931_call *c1, q931_call *c2); + + int qsig_cf_callrerouting(struct pri *pri, q931_call *c, const char* dest, const char* original, const char* reason); ++int send_reroute_request(struct pri *ctrl, q931_call *call, const struct q931_party_id *caller, const struct q931_party_redirecting *deflection, int subscription_option); + + /* starts a QSIG Path Replacement */ + int anfpr_initiate_transfer(struct pri *pri, q931_call *c1, q931_call *c2); + +-/* Use this function to queue a facility-IE born APDU onto a call +- * call is the call to use, messagetype is any one of the Q931 messages, +- * apdu is the apdu data, apdu_len is the length of the apdu data */ +-int pri_call_apdu_queue(q931_call *call, int messagetype, void *apdu, int apdu_len, void (*function)(void *data), void *data); ++int send_call_transfer_complete(struct pri *pri, q931_call *call, int call_status); + +-/* Used by q931.c to cleanup the apdu queue upon destruction of a call */ +-int pri_call_apdu_queue_cleanup(q931_call *call); ++int rose_diverting_leg_information1_encode(struct pri *pri, q931_call *call); ++int rose_diverting_leg_information3_encode(struct pri *pri, q931_call *call, int messagetype); + ++int rose_connected_name_encode(struct pri *pri, q931_call *call, int messagetype); ++int rose_called_name_encode(struct pri *pri, q931_call *call, int messagetype); ++ ++int pri_call_apdu_queue(q931_call *call, int messagetype, const unsigned char *apdu, int apdu_len, struct apdu_callback_data *response); ++void pri_call_apdu_queue_cleanup(q931_call *call); ++void pri_call_apdu_delete(struct q931_call *call, struct apdu_event *doomed); ++ + /* Adds the "standard" APDUs to a call */ + int pri_call_add_standard_apdus(struct pri *pri, q931_call *call); + +-int asn1_dump(struct pri *pri, void *comp, int len); ++void asn1_dump(struct pri *ctrl, const unsigned char *start_asn1, const unsigned char *end); + ++void rose_handle_invoke(struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, const struct fac_extension_header *header, const struct rose_msg_invoke *invoke); ++void rose_handle_result(struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, const struct fac_extension_header *header, const struct rose_msg_result *result); ++void rose_handle_error(struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, const struct fac_extension_header *header, const struct rose_msg_error *error); ++void rose_handle_reject(struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, const struct fac_extension_header *header, const struct rose_msg_reject *reject); ++ + #endif /* _PRI_FACILITY_H */ +Index: libpri.h +=================================================================== +--- a/libpri.h (.../tags/1.4.10.2) (revision 1357) ++++ b/libpri.h (.../branches/1.4) (revision 1357) +@@ -26,7 +26,14 @@ + * provided with that copy of Asterisk, instead of the license + * terms granted here. + */ +- ++ ++/* ++ * NOTE: ++ * All new global identifiers that are added to this file MUST be ++ * prefixed with PRI_ or pri_ to indicate that they are part of this ++ * library and to reduce potential naming conflicts. ++ */ ++ + #ifndef _LIBPRI_H + #define _LIBPRI_H + +@@ -73,7 +80,8 @@ + #define PRI_EVENT_ANSWER 8 /* Call has been answered (CONNECT) */ + #define PRI_EVENT_HANGUP_ACK 9 /* Call hangup has been acknowledged */ + #define PRI_EVENT_RESTART_ACK 10 /* Restart complete on a given channel (RESTART_ACKNOWLEDGE) */ +-#define PRI_EVENT_FACNAME 11 /* Caller*ID Name received on Facility */ ++#define PRI_EVENT_FACNAME 11 /* Caller*ID Name received on Facility (DEPRECATED) */ ++#define PRI_EVENT_FACILITY 11 /* Facility received (FACILITY) */ + #define PRI_EVENT_INFO_RECEIVED 12 /* Additional info (digits) received (INFORMATION) */ + #define PRI_EVENT_PROCEEDING 13 /* When we get CALL_PROCEEDING */ + #define PRI_EVENT_SETUP_ACK 14 /* When we get SETUP_ACKNOWLEDGE */ +@@ -81,6 +89,14 @@ + #define PRI_EVENT_NOTIFY 16 /* Notification received (NOTIFY) */ + #define PRI_EVENT_PROGRESS 17 /* When we get PROGRESS */ + #define PRI_EVENT_KEYPAD_DIGIT 18 /* When we receive during ACTIVE state (INFORMATION) */ ++#define PRI_EVENT_SERVICE 19 /* SERVICE maintenance message */ ++#define PRI_EVENT_SERVICE_ACK 20 /* SERVICE maintenance acknowledgement message */ ++#define PRI_EVENT_HOLD 21 /* HOLD request received */ ++#define PRI_EVENT_HOLD_ACK 22 /* HOLD_ACKNOWLEDGE received */ ++#define PRI_EVENT_HOLD_REJ 23 /* HOLD_REJECT received */ ++#define PRI_EVENT_RETRIEVE 24 /* RETRIEVE request received */ ++#define PRI_EVENT_RETRIEVE_ACK 25 /* RETRIEVE_ACKNOWLEDGE received */ ++#define PRI_EVENT_RETRIEVE_REJ 26 /* RETRIEVE_REJECT received */ + + /* Simple states */ + #define PRI_STATE_DOWN 0 +@@ -101,13 +117,13 @@ + #define PRI_PROG_CALLER_RETURNED_TO_ISDN (1 << 9) + + /* Numbering plan identifier */ +-#define PRI_NPI_UNKNOWN 0x0 +-#define PRI_NPI_E163_E164 0x1 +-#define PRI_NPI_X121 0x3 +-#define PRI_NPI_F69 0x4 +-#define PRI_NPI_NATIONAL 0x8 +-#define PRI_NPI_PRIVATE 0x9 +-#define PRI_NPI_RESERVED 0xF ++#define PRI_NPI_UNKNOWN 0x0 /*!< Unknown numbering plan */ ++#define PRI_NPI_E163_E164 0x1 /*!< ISDN/telephony numbering plan (public) */ ++#define PRI_NPI_X121 0x3 /*!< Data numbering plan */ ++#define PRI_NPI_F69 0x4 /*!< Telex numbering plan */ ++#define PRI_NPI_NATIONAL 0x8 /*!< National standard numbering plan */ ++#define PRI_NPI_PRIVATE 0x9 /*!< Private numbering plan */ ++#define PRI_NPI_RESERVED 0xF /*!< Reserved for extension */ + + /* Type of number */ + #define PRI_TON_UNKNOWN 0x0 +@@ -135,16 +151,49 @@ + #define PRI_UNKNOWN 0x0 + + /* Presentation */ +-#define PRES_ALLOWED_USER_NUMBER_NOT_SCREENED 0x00 +-#define PRES_ALLOWED_USER_NUMBER_PASSED_SCREEN 0x01 +-#define PRES_ALLOWED_USER_NUMBER_FAILED_SCREEN 0x02 +-#define PRES_ALLOWED_NETWORK_NUMBER 0x03 +-#define PRES_PROHIB_USER_NUMBER_NOT_SCREENED 0x20 +-#define PRES_PROHIB_USER_NUMBER_PASSED_SCREEN 0x21 +-#define PRES_PROHIB_USER_NUMBER_FAILED_SCREEN 0x22 +-#define PRES_PROHIB_NETWORK_NUMBER 0x23 +-#define PRES_NUMBER_NOT_AVAILABLE 0x43 ++#define PRI_PRES_NUMBER_TYPE 0x03 ++#define PRI_PRES_USER_NUMBER_UNSCREENED 0x00 ++#define PRI_PRES_USER_NUMBER_PASSED_SCREEN 0x01 ++#define PRI_PRES_USER_NUMBER_FAILED_SCREEN 0x02 ++#define PRI_PRES_NETWORK_NUMBER 0x03 + ++#define PRI_PRES_RESTRICTION 0x60 ++#define PRI_PRES_ALLOWED 0x00 ++#define PRI_PRES_RESTRICTED 0x20 ++#define PRI_PRES_UNAVAILABLE 0x40 ++#define PRI_PRES_RESERVED 0x60 ++ ++#define PRES_ALLOWED_USER_NUMBER_NOT_SCREENED \ ++ (PRI_PRES_ALLOWED | PRI_PRES_USER_NUMBER_UNSCREENED) ++ ++#define PRES_ALLOWED_USER_NUMBER_PASSED_SCREEN \ ++ (PRI_PRES_ALLOWED | PRI_PRES_USER_NUMBER_PASSED_SCREEN) ++ ++#define PRES_ALLOWED_USER_NUMBER_FAILED_SCREEN \ ++ (PRI_PRES_ALLOWED | PRI_PRES_USER_NUMBER_FAILED_SCREEN) ++ ++#define PRES_ALLOWED_NETWORK_NUMBER \ ++ (PRI_PRES_ALLOWED | PRI_PRES_NETWORK_NUMBER) ++ ++#define PRES_PROHIB_USER_NUMBER_NOT_SCREENED \ ++ (PRI_PRES_RESTRICTED | PRI_PRES_USER_NUMBER_UNSCREENED) ++ ++#define PRES_PROHIB_USER_NUMBER_PASSED_SCREEN \ ++ (PRI_PRES_RESTRICTED | PRI_PRES_USER_NUMBER_PASSED_SCREEN) ++ ++#define PRES_PROHIB_USER_NUMBER_FAILED_SCREEN \ ++ (PRI_PRES_RESTRICTED | PRI_PRES_USER_NUMBER_FAILED_SCREEN) ++ ++#define PRES_PROHIB_NETWORK_NUMBER \ ++ (PRI_PRES_RESTRICTED | PRI_PRES_NETWORK_NUMBER) ++ ++#define PRES_NUMBER_NOT_AVAILABLE \ ++ (PRI_PRES_UNAVAILABLE | PRI_PRES_NETWORK_NUMBER) ++ ++/* Reverse Charging Indication */ ++#define PRI_REVERSECHARGE_NONE -1 ++#define PRI_REVERSECHARGE_REQUESTED 1 ++ + /* Causes for disconnection */ + #define PRI_CAUSE_UNALLOCATED 1 + #define PRI_CAUSE_NO_ROUTE_TRANSIT_NET 2 /* !Q.SIG */ +@@ -157,6 +206,7 @@ + #define PRI_CAUSE_NO_ANSWER 19 + #define PRI_CAUSE_CALL_REJECTED 21 + #define PRI_CAUSE_NUMBER_CHANGED 22 ++#define PRI_CAUSE_NONSELECTED_USER_CLEARING 26 + #define PRI_CAUSE_DESTINATION_OUT_OF_ORDER 27 + #define PRI_CAUSE_INVALID_NUMBER_FORMAT 28 + #define PRI_CAUSE_FACILITY_REJECTED 29 /* !Q.SIG */ +@@ -169,6 +219,7 @@ + #define PRI_CAUSE_ACCESS_INFO_DISCARDED 43 /* !Q.SIG */ + #define PRI_CAUSE_REQUESTED_CHAN_UNAVAIL 44 + #define PRI_CAUSE_PRE_EMPTED 45 /* !Q.SIG */ ++#define PRI_CAUSE_RESOURCE_UNAVAIL_UNSPECIFIED 47 + #define PRI_CAUSE_FACILITY_NOT_SUBSCRIBED 50 /* !Q.SIG */ + #define PRI_CAUSE_OUTGOING_CALL_BARRED 52 /* !Q.SIG */ + #define PRI_CAUSE_INCOMING_CALL_BARRED 54 /* !Q.SIG */ +@@ -259,8 +310,8 @@ + #define PRI_RATE_ADAPT_ASYNC 0x40 + + /* Notifications */ +-#define PRI_NOTIFY_USER_SUSPENDED 0x00 /* User suspended */ +-#define PRI_NOTIFY_USER_RESUMED 0x01 /* User resumed */ ++#define PRI_NOTIFY_USER_SUSPENDED 0x00 /* User suspended (Q.931) (Call is placed on hold) */ ++#define PRI_NOTIFY_USER_RESUMED 0x01 /* User resumed (Q.931) (Call is taken off hold) */ + #define PRI_NOTIFY_BEARER_CHANGE 0x02 /* Bearer service change (DSS1) */ + #define PRI_NOTIFY_ASN1_COMPONENT 0x03 /* ASN.1 encoded component (DSS1) */ + #define PRI_NOTIFY_COMPLETION_DELAY 0x04 /* Call completion delay */ +@@ -275,12 +326,12 @@ + #define PRI_NOTIFY_CONF_OTHER_DISCONNECTED 0x4a /* Other party disconnected */ + #define PRI_NOTIFY_CONF_FLOATING 0x4b /* Conference floating */ + #define PRI_NOTIFY_WAITING_CALL 0x60 /* Call is waiting call */ +-#define PRI_NOTIFY_DIVERSION_ACTIVATED 0x68 /* Diversion activated (DSS1) */ +-#define PRI_NOTIFY_TRANSFER_ALERTING 0x69 /* Call transfer, alerting */ +-#define PRI_NOTIFY_TRANSFER_ACTIVE 0x6a /* Call transfer, active */ ++#define PRI_NOTIFY_DIVERSION_ACTIVATED 0x68 /* Diversion activated (DSS1) (cfu, cfb, cfnr) (EN 300 207-1 Section 7.2.1) */ ++#define PRI_NOTIFY_TRANSFER_ALERTING 0x69 /* Call transfer, alerting (EN 300 369-1 Section 7.2) */ ++#define PRI_NOTIFY_TRANSFER_ACTIVE 0x6a /* Call transfer, active(answered) (EN 300 369-1 Section 7.2) */ + #define PRI_NOTIFY_REMOTE_HOLD 0x79 /* Remote hold */ + #define PRI_NOTIFY_REMOTE_RETRIEVAL 0x7a /* Remote retrieval */ +-#define PRI_NOTIFY_CALL_DIVERTING 0x7b /* Call is diverting */ ++#define PRI_NOTIFY_CALL_DIVERTING 0x7b /* Call is diverting (EN 300 207-1 Section 7.2.1) */ + + #define PRI_COPY_DIGITS_CALLED_NUMBER + +@@ -305,6 +356,200 @@ + + typedef struct q931_call q931_call; + ++/* Name character set enumeration values */ ++#define PRI_CHAR_SET_UNKNOWN 0 ++#define PRI_CHAR_SET_ISO8859_1 1 ++#define PRI_CHAR_SET_WITHDRAWN 2 ++#define PRI_CHAR_SET_ISO8859_2 3 ++#define PRI_CHAR_SET_ISO8859_3 4 ++#define PRI_CHAR_SET_ISO8859_4 5 ++#define PRI_CHAR_SET_ISO8859_5 6 ++#define PRI_CHAR_SET_ISO8859_7 7 ++#define PRI_CHAR_SET_ISO10646_BMPSTRING 8 ++#define PRI_CHAR_SET_ISO10646_UTF_8STRING 9 ++ ++/*! \brief Q.SIG name information. */ ++struct pri_party_name { ++ /*! \brief TRUE if the name information is valid/present */ ++ int valid; ++ /*! ++ * \brief Q.931 presentation-indicator encoded field ++ * \note Must tollerate the Q.931 screening-indicator field values being present. ++ */ ++ int presentation; ++ /*! ++ * \brief Character set the name is using. ++ * \details ++ * unknown(0), ++ * iso8859-1(1), ++ * enum-value-withdrawn-by-ITU-T(2) ++ * iso8859-2(3), ++ * iso8859-3(4), ++ * iso8859-4(5), ++ * iso8859-5(6), ++ * iso8859-7(7), ++ * iso10646-BmpString(8), ++ * iso10646-utf-8String(9) ++ * \details ++ * Set to iso8859-1(1) if unsure what to use. ++ */ ++ int char_set; ++ /*! \brief Name data with null terminator. */ ++ char str[64]; ++}; ++ ++struct pri_party_number { ++ /*! \brief TRUE if the number information is valid/present */ ++ int valid; ++ /*! \brief Q.931 presentation-indicator and screening-indicator encoded fields */ ++ int presentation; ++ /*! \brief Q.931 Type-Of-Number and numbering-plan encoded fields */ ++ int plan; ++ /*! \brief Number data with null terminator. */ ++ char str[64]; ++}; ++ ++/*! ++ * \note This structure is a place holder for possible future subaddress support ++ * to maintain ABI compatibility. ++ */ ++struct pri_party_subaddress { ++ /*! \brief TRUE if the subaddress information is valid/present */ ++ int valid; ++ /*! ++ * \brief Subaddress type. ++ * \details ++ * nsap(0), ++ * user_specified(2) ++ */ ++ int type; ++ /*! ++ * \brief TRUE if odd number of address signals ++ * \note The odd/even indicator is used when the type of subaddress is ++ * user_specified and the coding is BCD. ++ */ ++ int odd_even_indicator; ++ /*! \brief Length of the subaddress data */ ++ int length; ++ /*! ++ * \brief Subaddress data with null terminator. ++ * \note The null terminator is a convenience only since the data could be ++ * BCD/binary and thus have a null byte as part of the contents. ++ */ ++ unsigned char data[32]; ++}; ++ ++/*! \brief Information needed to identify an endpoint in a call. */ ++struct pri_party_id { ++ /*! \brief Subscriber name */ ++ struct pri_party_name name; ++ /*! \brief Subscriber phone number */ ++ struct pri_party_number number; ++ /*! \brief Subscriber subaddress */ ++ struct pri_party_subaddress subaddress; ++}; ++ ++/*! \brief Connected Line/Party information */ ++struct pri_party_connected_line { ++ /*! Connected party ID */ ++ struct pri_party_id id; ++}; ++ ++/*! ++ * \brief Redirecting Line information. ++ * \details ++ * RDNIS (Redirecting Directory Number Information Service) ++ * Where a call diversion or transfer was invoked. ++ */ ++struct pri_party_redirecting { ++ /*! Who is redirecting the call (Sent to the party the call is redirected toward) */ ++ struct pri_party_id from; ++ /*! Call is redirecting to a new party (Sent to the caller) */ ++ struct pri_party_id to; ++ /*! Originally called party (in cases of multiple redirects) */ ++ struct pri_party_id orig_called; ++ /*! Number of times the call was redirected */ ++ int count; ++ /*! Original reason for redirect (in cases of multiple redirects) */ ++ int orig_reason; ++ /*! Redirection reason */ ++ int reason; ++}; ++ ++/*! ++ * \brief Information for rerouting/deflecting the call. ++ */ ++struct pri_rerouting_data { ++ /*! ++ * \brief Updated caller-id information. ++ * \note The information may have been altered by procedure in the private network. ++ */ ++ struct pri_party_id caller; ++ /*! ++ * \note ++ * deflection.to is the new called number and must always be present. ++ */ ++ struct pri_party_redirecting deflection; ++ /*! ++ * \brief Diverting user subscription option to specify if caller is notified. ++ * \details ++ * noNotification(0), ++ * notificationWithoutDivertedToNr(1), ++ * notificationWithDivertedToNr(2), ++ * notApplicable(3) (Status only.) ++ */ ++ int subscription_option; ++ /*! Invocation ID to use when sending a reply to the call rerouting/deflection request. */ ++ int invoke_id; ++}; ++ ++/* Subcommands derived from supplementary services. */ ++#define PRI_SUBCMD_REDIRECTING 1 ++#define PRI_SUBCMD_CONNECTED_LINE 2 ++#define PRI_SUBCMD_REROUTING 3 ++ ++ ++struct pri_subcommand { ++ /*! PRI_SUBCMD_xxx defined values */ ++ int cmd; ++ union { ++ /*! Reserve room for possible expansion to maintain ABI compatibility. */ ++ char reserve_space[512]; ++ struct pri_party_connected_line connected_line; ++ struct pri_party_redirecting redirecting; ++ struct pri_rerouting_data rerouting; ++ } u; ++}; ++ ++/* Max number of subcommands per event message */ ++#define PRI_MAX_SUBCOMMANDS 8 ++ ++struct pri_subcommands { ++ int counter_subcmd; ++ struct pri_subcommand subcmd[PRI_MAX_SUBCOMMANDS]; ++}; ++ ++ ++/* ++ * Event channel parameter encoding: ++ * 3322 2222 2222 1111 1111 1100 0000 0000 ++ * 1098 7654 3210 9876 5432 1098 7654 3210 ++ * xxxx xxxx xxxx xEDC BBBBBBBBB AAAAAAAAA ++ * ++ * Bit field ++ * A - B channel ++ * B - Span (DS1) (0 - 127) ++ * C - DS1 Explicit bit ++ * D - D channel (cis_call) bit (status only) ++ * E - Call is held bit (status only) ++ * ++ * B channel values: ++ * 0 - No channel (ISDN uses for call waiting feature) ++ * 1-127 - B channel # ++ * 0xFF - Any channel (Also if whole channel value is -1 in event) ++ */ ++ ++ + typedef struct pri_event_generic { + /* Events with no additional information fall in this category */ + int e; +@@ -328,6 +573,7 @@ + int progressmask; + q931_call *call; + char useruserinfo[260]; /* User->User info */ ++ struct pri_subcommands *subcmds; + } pri_event_ringing; + + typedef struct pri_event_answer { +@@ -338,8 +584,10 @@ + int progressmask; + q931_call *call; + char useruserinfo[260]; /* User->User info */ ++ struct pri_subcommands *subcmds; + } pri_event_answer; + ++/*! Deprecated replaced by struct pri_event_facility. */ + typedef struct pri_event_facname { + int e; + char callingname[256]; +@@ -351,6 +599,24 @@ + int callingplan; /* Dialing plan of Calling entity */ + } pri_event_facname; + ++struct pri_event_facility { ++ int e; ++ char callingname[256]; /*!< Deprecated, preserved for struct pri_event_facname compatibility */ ++ char callingnum[256]; /*!< Deprecated, preserved for struct pri_event_facname compatibility */ ++ int channel; ++ int cref; ++ /*! ++ * \brief Master call or normal call. ++ * \note Call pointer known about by upper layer. ++ * \note NULL if dummy call reference. ++ */ ++ q931_call *call; ++ int callingpres; /*!< Presentation of Calling CallerID (Deprecated, preserved for struct pri_event_facname compatibility) */ ++ int callingplan; /*!< Dialing plan of Calling entity (Deprecated, preserved for struct pri_event_facname compatibility) */ ++ struct pri_subcommands *subcmds; ++ q931_call *subcall; /*!< Subcall to send any reply toward. */ ++}; ++ + #define PRI_CALLINGPLANANI + #define PRI_CALLINGPLANRDNIS + typedef struct pri_event_ring { +@@ -376,13 +642,18 @@ + int layer1; /* User layer 1 */ + int complete; /* Have we seen "Complete" i.e. no more number? */ + q931_call *call; /* Opaque call pointer */ +- char callingsubaddr[256]; /* Calling parties subaddress */ ++ char callingsubaddr[256]; /* Calling parties subaddress, backwards compatibility */ + int progress; + int progressmask; + char origcalledname[256]; + char origcallednum[256]; + int callingplanorigcalled; /* Dialing plan of Originally Called Number */ + int origredirectingreason; ++ int reversecharge; ++ struct pri_subcommands *subcmds; ++ struct pri_party_id calling; /* Calling Party's info, initially subaddress' */ ++ struct pri_party_subaddress called_subaddress; /* Called party's subaddress */ ++ char keypad_digits[64]; /* Keypad digits in the SETUP message. */ + } pri_event_ring; + + typedef struct pri_event_hangup { +@@ -390,10 +661,23 @@ + int channel; /* Channel requested */ + int cause; + int cref; +- q931_call *call; /* Opaque call pointer */ ++ q931_call *call; /* Opaque call pointer of call hanging up. */ + long aoc_units; /* Advise of Charge number of charged units */ + char useruserinfo[260]; /* User->User info */ +-} pri_event_hangup; ++ struct pri_subcommands *subcmds; ++ /*! ++ * \brief Opaque held call pointer for possible transfer to active call. ++ * \note The call_held and call_active pointers must not be NULL if ++ * transfer held call on disconnect is available. ++ */ ++ q931_call *call_held; ++ /*! ++ * \brief Opaque active call pointer for possible transfer with held call. ++ * \note The call_held and call_active pointers must not be NULL if ++ * transfer held call on disconnect is available. ++ */ ++ q931_call *call_active; ++} pri_event_hangup; + + typedef struct pri_event_restart_ack { + int e; +@@ -409,18 +693,22 @@ + int progressmask; + int cause; + q931_call *call; ++ struct pri_subcommands *subcmds; + } pri_event_proceeding; +- ++ + typedef struct pri_event_setup_ack { + int e; + int channel; + q931_call *call; ++ struct pri_subcommands *subcmds; + } pri_event_setup_ack; + + typedef struct pri_event_notify { + int e; + int channel; + int info; ++ struct pri_subcommands *subcmds; ++ q931_call *call; + } pri_event_notify; + + typedef struct pri_event_keypad_digit { +@@ -428,14 +716,72 @@ + int channel; + q931_call *call; + char digits[64]; ++ struct pri_subcommands *subcmds; + } pri_event_keypad_digit; + ++typedef struct pri_event_service { ++ int e; ++ int channel; ++ int changestatus; ++} pri_event_service; ++ ++typedef struct pri_event_service_ack { ++ int e; ++ int channel; ++ int changestatus; ++} pri_event_service_ack; ++ ++struct pri_event_hold { ++ int e; ++ int channel; ++ q931_call *call; ++ struct pri_subcommands *subcmds; ++}; ++ ++struct pri_event_hold_ack { ++ int e; ++ int channel; ++ q931_call *call; ++ struct pri_subcommands *subcmds; ++}; ++ ++struct pri_event_hold_rej { ++ int e; ++ int channel; ++ q931_call *call; ++ int cause; ++ struct pri_subcommands *subcmds; ++}; ++ ++struct pri_event_retrieve { ++ int e; ++ int channel; ++ q931_call *call; ++ int flexible; /* Are we flexible with our channel selection? */ ++ struct pri_subcommands *subcmds; ++}; ++ ++struct pri_event_retrieve_ack { ++ int e; ++ int channel; ++ q931_call *call; ++ struct pri_subcommands *subcmds; ++}; ++ ++struct pri_event_retrieve_rej { ++ int e; ++ int channel; ++ q931_call *call; ++ int cause; ++ struct pri_subcommands *subcmds; ++}; ++ + typedef union { + int e; + pri_event_generic gen; /* Generic view */ + pri_event_restart restart; /* Restart view */ + pri_event_error err; /* Error view */ +- pri_event_facname facname; /* Caller*ID Name on Facility */ ++ pri_event_facname facname; /* Caller*ID Name on Facility (Deprecated, use pri_event.facility) */ + pri_event_ring ring; /* Ring */ + pri_event_hangup hangup; /* Hang up */ + pri_event_ringing ringing; /* Ringing */ +@@ -445,6 +791,15 @@ + pri_event_setup_ack setup_ack; /* SETUP_ACKNOWLEDGE structure */ + pri_event_notify notify; /* Notification */ + pri_event_keypad_digit digit; /* Digits that come during a call */ ++ pri_event_service service; /* service message */ ++ pri_event_service_ack service_ack; /* service acknowledgement message */ ++ struct pri_event_facility facility; ++ struct pri_event_hold hold; ++ struct pri_event_hold_ack hold_ack; ++ struct pri_event_hold_rej hold_rej; ++ struct pri_event_retrieve retrieve; ++ struct pri_event_retrieve_ack retrieve_ack; ++ struct pri_event_retrieve_rej retrieve_rej; + } pri_event; + + struct pri; +@@ -456,7 +811,7 @@ + + /* Create a D-channel on a given file descriptor. The file descriptor must be a + channel operating in HDLC mode with FCS computed by the fd's driver. Also it +- must be NON-BLOCKING! Frames received on the fd should include FCS. Nodetype ++ must be NON-BLOCKING! Frames received on the fd should include FCS. Nodetype + must be one of PRI_NETWORK or PRI_CPE. switchtype should be PRI_SWITCH_* */ + struct pri *pri_new(int fd, int nodetype, int switchtype); + struct pri *pri_new_bri(int fd, int ptpmode, int nodetype, int switchtype); +@@ -521,7 +876,7 @@ + + #define PRI_KEYPAD_FACILITY_TX + /* Send a keypad facility string of digits */ +-int pri_keypad_facility(struct pri *pri, q931_call *call, char *digits); ++int pri_keypad_facility(struct pri *pri, q931_call *call, const char *digits); + + /* Answer the incomplete(call without called number) call on the given channel. + Set non-isdn to non-zero if you are not connecting to ISDN equipment */ +@@ -531,6 +886,18 @@ + Set non-isdn to non-zero if you are not connecting to ISDN equipment */ + int pri_answer(struct pri *pri, q931_call *call, int channel, int nonisdn); + ++/*! ++ * \brief Give connected line information to a call ++ * \note Could be used instead of pri_sr_set_caller_party() before calling pri_setup(). ++ */ ++int pri_connected_line_update(struct pri *pri, q931_call *call, const struct pri_party_connected_line *connected); ++ ++/*! ++ * \brief Give redirection information to a call ++ * \note Could be used instead of pri_sr_set_redirecting_parties() before calling pri_setup(). ++ */ ++int pri_redirecting_update(struct pri *pri, q931_call *call, const struct pri_party_redirecting *redirecting); ++ + /* Set CRV reference for GR-303 calls */ + + +@@ -556,9 +923,20 @@ + + int pri_reset(struct pri *pri, int channel); + ++/* handle b-channel maintenance messages */ ++extern int pri_maintenance_service(struct pri *pri, int span, int channel, int changestatus); ++ + /* Create a new call */ + q931_call *pri_new_call(struct pri *pri); + ++/*! ++ * \brief Deterimine if the given call control pointer is a dummy call. ++ * ++ * \retval TRUE if given call is a dummy call. ++ * \retval FALSE otherwise. ++ */ ++int pri_is_dummy_call(q931_call *call); ++ + /* Retrieve CRV reference for GR-303 calls. Returns >0 on success. */ + int pri_get_crv(struct pri *pri, q931_call *call, int *callmode); + +@@ -573,8 +951,8 @@ + extern pri_event *pri_schedule_run_tv(struct pri *pri, const struct timeval *now); + + int pri_call(struct pri *pri, q931_call *c, int transmode, int channel, +- int exclusive, int nonisdn, char *caller, int callerplan, char *callername, int callerpres, +- char *called,int calledplan, int ulayer1); ++ int exclusive, int nonisdn, char *caller, int callerplan, char *callername, int callerpres, ++ char *called, int calledplan, int ulayer1); + + struct pri_sr *pri_sr_new(void); + void pri_sr_free(struct pri_sr *sr); +@@ -582,25 +960,91 @@ + int pri_sr_set_channel(struct pri_sr *sr, int channel, int exclusive, int nonisdn); + int pri_sr_set_bearer(struct pri_sr *sr, int transmode, int userl1); + int pri_sr_set_called(struct pri_sr *sr, char *called, int calledplan, int complete); ++ ++/*! ++ * \brief Set the caller party ID information in the call SETUP record. ++ * ++ * \param sr New call SETUP record. ++ * \param caller Caller party ID information to set. ++ * ++ * \return Nothing ++ */ ++void pri_sr_set_caller_party(struct pri_sr *sr, const struct pri_party_id *caller); ++/*! \note Use pri_sr_set_caller_party() instead to pass more precise caller information. */ + int pri_sr_set_caller(struct pri_sr *sr, char *caller, char *callername, int callerplan, int callerpres); ++ ++/*! ++ * \brief Set the calling subaddress information in the call SETUP record. ++ * ++ * \param sr New call SETUP record. ++ * \param subaddress information to set. ++ * ++ * \return Nothing ++ */ ++void pri_sr_set_caller_subaddress(struct pri_sr *sr, const struct pri_party_subaddress *subaddress); ++ ++/*! ++ * \brief Set the called subaddress information in the call SETUP record. ++ * ++ * \param sr New call SETUP record. ++ * \param subaddress information to set. ++ * ++ * \return Nothing ++ */ ++void pri_sr_set_called_subaddress(struct pri_sr *sr, const struct pri_party_subaddress *subaddress); ++ ++/*! ++ * \brief Set the redirecting information in the call SETUP record. ++ * ++ * \param sr New call SETUP record. ++ * \param caller Redirecting information to set. ++ * ++ * \return Nothing ++ */ ++void pri_sr_set_redirecting_parties(struct pri_sr *sr, const struct pri_party_redirecting *redirecting); ++/*! \note Use pri_sr_set_redirecting_parties() instead to pass more precise redirecting information. */ + int pri_sr_set_redirecting(struct pri_sr *sr, char *num, int plan, int pres, int reason); ++ ++/*! ++ * \brief Set the keypad digits in the call SETUP record. ++ * ++ * \param sr New call SETUP record. ++ * \param keypad_digits Keypad digits to send. ++ * ++ * \return Nothing ++ */ ++void pri_sr_set_keypad_digits(struct pri_sr *sr, const char *keypad_digits); ++ + #define PRI_USER_USER_TX + /* Set the user user field. Warning! don't send binary data accross this field */ + void pri_sr_set_useruser(struct pri_sr *sr, const char *userchars); ++void pri_sr_set_reversecharge(struct pri_sr *sr, int requested); + + void pri_call_set_useruser(q931_call *sr, const char *userchars); + + int pri_setup(struct pri *pri, q931_call *call, struct pri_sr *req); + +-/* Set a call has a call indpendent signalling connection (i.e. no bchan) */ ++/*! ++ * \brief Set a call as a call indpendent signalling connection (i.e. no bchan) ++ * \note Call will automaticlly disconnect after signalling sent. ++ */ + int pri_sr_set_connection_call_independent(struct pri_sr *req); + ++/*! ++ * \brief Set a call as a call indpendent signalling connection (i.e. no bchan) ++ * \note Call will stay connected until explicitly disconnected. ++ */ ++int pri_sr_set_no_channel_call(struct pri_sr *req); ++ + /* Send an MWI indication to a remote location. If activate is non zero, activates, if zero, deactivates */ + int pri_mwi_activate(struct pri *pri, q931_call *c, char *caller, int callerplan, char *callername, int callerpres, char *called, int calledplan); + + /* Send an MWI deactivate request to a remote location */ + int pri_mwi_deactivate(struct pri *pri, q931_call *c, char *caller, int callerplan, char *callername, int callerpres, char *called, int calledplan); + ++/* Set service message support flag */ ++int pri_set_service_message_support(struct pri *pri, int supportflag); ++ + #define PRI_2BCT + /* Attempt to pass the channels back to the NET side if compatable and + * suscribed. Sometimes called 2 bchannel transfer (2BCT) */ +@@ -656,48 +1100,208 @@ + + int pri_callrerouting_facility(struct pri *pri, q931_call *call, const char *dest, const char* original, const char* reason); + ++/*! ++ * \brief Set the call deflection/rerouting feature enable flag. ++ * ++ * \param ctrl D channel controller. ++ * \param enable TRUE to enable call deflection/rerouting feature. ++ * ++ * \return Nothing ++ */ ++void pri_reroute_enable(struct pri *ctrl, int enable); ++ ++/*! ++ * \brief Send the CallRerouting/CallDeflection message. ++ * ++ * \param ctrl D channel controller. ++ * \param call Q.931 call leg. ++ * \param caller Call rerouting/deflecting updated caller data. (NULL if data not updated.) ++ * \param deflection Call rerouting/deflecting redirection data. ++ * \param subscription_option Diverting user subscription option to specify if caller is notified. ++ * ++ * \note ++ * deflection->to is the new called number and must always be present. ++ * \note ++ * subscription option: ++ * noNotification(0), ++ * notificationWithoutDivertedToNr(1), ++ * notificationWithDivertedToNr(2) ++ * ++ * \retval 0 on success. ++ * \retval -1 on error. ++ */ ++int pri_reroute_call(struct pri *ctrl, q931_call *call, const struct pri_party_id *caller, const struct pri_party_redirecting *deflection, int subscription_option); ++ ++enum PRI_REROUTING_RSP_CODE { ++ /*! ++ * Rerouting invocation accepted and the network provider option ++ * "served user call retention on invocation of diversion" ++ * is "clear call on invocation". ++ */ ++ PRI_REROUTING_RSP_OK_CLEAR, ++ /*! ++ * Rerouting invocation accepted and the network provider option ++ * "served user call retention on invocation of diversion" ++ * is "retain call until alerting begins at the deflected-to user". ++ */ ++ PRI_REROUTING_RSP_OK_RETAIN, ++ PRI_REROUTING_RSP_NOT_SUBSCRIBED, ++ PRI_REROUTING_RSP_NOT_AVAILABLE, ++ /*! Supplementary service interaction not allowed. */ ++ PRI_REROUTING_RSP_NOT_ALLOWED, ++ PRI_REROUTING_RSP_INVALID_NUMBER, ++ /*! Deflection to prohibited number (e.g., operator, police, emergency). */ ++ PRI_REROUTING_RSP_SPECIAL_SERVICE_NUMBER, ++ /*! Deflection to served user number. */ ++ PRI_REROUTING_RSP_DIVERSION_TO_SELF, ++ PRI_REROUTING_RSP_MAX_DIVERSIONS_EXCEEDED, ++ PRI_REROUTING_RSP_RESOURCE_UNAVAILABLE, ++}; ++ ++/*! ++ * \brief Send the CallRerouteing/CallDeflection response message. ++ * ++ * \param ctrl D channel controller. ++ * \param call Q.931 call leg. ++ * \param invoke_id Value given by the initiating request. ++ * \param code The result to send. ++ * ++ * \retval 0 on success. ++ * \retval -1 on error. ++ */ ++int pri_rerouting_rsp(struct pri *ctrl, q931_call *call, int invoke_id, enum PRI_REROUTING_RSP_CODE code); ++ ++/*! ++ * \brief Set the call hold feature enable flag. ++ * ++ * \param ctrl D channel controller. ++ * \param enable TRUE to enable call hold feature. ++ * ++ * \return Nothing ++ */ ++void pri_hold_enable(struct pri *ctrl, int enable); ++ ++/*! ++ * \brief Send the HOLD message. ++ * ++ * \param ctrl D channel controller. ++ * \param call Q.931 call leg ++ * ++ * \retval 0 on success. ++ * \retval -1 on error. ++ */ ++int pri_hold(struct pri *ctrl, q931_call *call); ++ ++/*! ++ * \brief Send the HOLD ACKNOWLEDGE message. ++ * ++ * \param ctrl D channel controller. ++ * \param call Q.931 call leg ++ * ++ * \retval 0 on success. ++ * \retval -1 on error. ++ */ ++int pri_hold_ack(struct pri *ctrl, q931_call *call); ++ ++/*! ++ * \brief Send the HOLD REJECT message. ++ * ++ * \param ctrl D channel controller. ++ * \param call Q.931 call leg ++ * \param cause Q.931 cause code for rejecting the hold request. ++ * ++ * \retval 0 on success. ++ * \retval -1 on error. ++ */ ++int pri_hold_rej(struct pri *ctrl, q931_call *call, int cause); ++ ++/*! ++ * \brief Send the RETRIEVE message. ++ * ++ * \param ctrl D channel controller. ++ * \param call Q.931 call leg ++ * \param channel Encoded channel id to use. If zero do not send channel id. ++ * ++ * \retval 0 on success. ++ * \retval -1 on error. ++ */ ++int pri_retrieve(struct pri *ctrl, q931_call *call, int channel); ++ ++/*! ++ * \brief Send the RETRIEVE ACKNOWLEDGE message. ++ * ++ * \param ctrl D channel controller. ++ * \param call Q.931 call leg ++ * \param channel Encoded channel id to use. ++ * ++ * \retval 0 on success. ++ * \retval -1 on error. ++ */ ++int pri_retrieve_ack(struct pri *ctrl, q931_call *call, int channel); ++ ++/*! ++ * \brief Send the RETRIEVE REJECT message. ++ * ++ * \param ctrl D channel controller. ++ * \param call Q.931 call leg ++ * \param cause Q.931 cause code for rejecting the retrieve request. ++ * ++ * \retval 0 on success. ++ * \retval -1 on error. ++ */ ++int pri_retrieve_rej(struct pri *ctrl, q931_call *call, int cause); ++ + /* Get/Set PRI Timers */ + #define PRI_GETSET_TIMERS + int pri_set_timer(struct pri *pri, int timer, int value); + int pri_get_timer(struct pri *pri, int timer); +-int pri_timer2idx(char *timer); ++int pri_timer2idx(const char *timer_name); + +-#define PRI_MAX_TIMERS 32 ++/*! New configurable timers and counters must be added to the end of the list */ ++enum PRI_TIMERS_AND_COUNTERS { ++ PRI_TIMER_N200, /*!< Maximum numer of Q.921 retransmissions */ ++ PRI_TIMER_N201, /*!< Maximum numer of octets in an information field */ ++ PRI_TIMER_N202, /*!< Maximum numer of transmissions of the TEI identity request message */ ++ PRI_TIMER_K, /*!< Maximum number of outstanding I-frames */ + +-#define PRI_TIMER_N200 0 /* Maximum numer of q921 retransmissions */ +-#define PRI_TIMER_N201 1 /* Maximum numer of octets in an information field */ +-#define PRI_TIMER_N202 2 /* Maximum numer of transmissions of the TEI identity request message */ +-#define PRI_TIMER_K 3 /* Maximum number of outstanding I-frames */ ++ PRI_TIMER_T200, /*!< Time between SABME's */ ++ PRI_TIMER_T201, /*!< Minimum time between retransmissions of the TEI Identity check messages */ ++ PRI_TIMER_T202, /*!< Minimum time between transmission of TEI Identity request messages */ ++ PRI_TIMER_T203, /*!< Maximum time without exchanging packets */ + +-#define PRI_TIMER_T200 4 /* time between SABME's */ +-#define PRI_TIMER_T201 5 /* minimum time between retransmissions of the TEI Identity check messages */ +-#define PRI_TIMER_T202 6 /* minimum time between transmission of TEI Identity request messages */ +-#define PRI_TIMER_T203 7 /* maxiumum time without exchanging packets */ ++ PRI_TIMER_T300, ++ PRI_TIMER_T301, /*!< Maximum time to respond to an ALERT */ ++ PRI_TIMER_T302, ++ PRI_TIMER_T303, /*!< Maximum time to wait after sending a SETUP without a response */ ++ PRI_TIMER_T304, ++ PRI_TIMER_T305, /*!< Wait for DISCONNECT acknowledge */ ++ PRI_TIMER_T306, ++ PRI_TIMER_T307, ++ PRI_TIMER_T308, /*!< Wait for RELEASE acknowledge */ ++ PRI_TIMER_T309, /*!< Time active calls can tollerate data link layer being down before clearing. */ ++ PRI_TIMER_T310, /*!< Maximum time between receiving a CALL_PROCEEDING and receiving a ALERT/CONNECT/DISCONNECT/PROGRESS */ ++ PRI_TIMER_T313, /*!< Wait for CONNECT acknowledge, CPE side only */ ++ PRI_TIMER_T314, ++ PRI_TIMER_T316, /*!< Maximum time between transmitting a RESTART and receiving a RESTART ACK */ ++ PRI_TIMER_T317, ++ PRI_TIMER_T318, ++ PRI_TIMER_T319, ++ PRI_TIMER_T320, ++ PRI_TIMER_T321, ++ PRI_TIMER_T322, + +-#define PRI_TIMER_T300 8 +-#define PRI_TIMER_T301 9 /* maximum time to respond to an ALERT */ +-#define PRI_TIMER_T302 10 +-#define PRI_TIMER_T303 11 /* maximum time to wait after sending a SETUP without a response */ +-#define PRI_TIMER_T304 12 +-#define PRI_TIMER_T305 13 +-#define PRI_TIMER_T306 14 +-#define PRI_TIMER_T307 15 +-#define PRI_TIMER_T308 16 +-#define PRI_TIMER_T309 17 +-#define PRI_TIMER_T310 18 /* maximum time between receiving a CALLPROCEEDING and receiving a ALERT/CONNECT/DISCONNECT/PROGRESS */ +-#define PRI_TIMER_T313 19 +-#define PRI_TIMER_T314 20 +-#define PRI_TIMER_T316 21 /* maximum time between transmitting a RESTART and receiving a RESTART ACK */ +-#define PRI_TIMER_T317 22 +-#define PRI_TIMER_T318 23 +-#define PRI_TIMER_T319 24 +-#define PRI_TIMER_T320 25 +-#define PRI_TIMER_T321 26 +-#define PRI_TIMER_T322 27 ++ PRI_TIMER_TM20, /*!< Maximum time awaiting XID response */ ++ PRI_TIMER_NM20, /*!< Number of XID retransmits */ + +-#define PRI_TIMER_TM20 28 /* maximum time avaiting XID response */ +-#define PRI_TIMER_NM20 29 /* number of XID retransmits */ ++ PRI_TIMER_T_HOLD, /*!< Maximum time to wait for HOLD request response. */ ++ PRI_TIMER_T_RETRIEVE, /*!< Maximum time to wait for RETRIEVE request response. */ + ++ PRI_TIMER_T_RESPONSE, /*!< Maximum time to wait for a typical APDU response. */ ++ ++ /* Must be last in the enum list */ ++ PRI_MAX_TIMERS ++}; ++ + /* Get PRI version */ + const char *pri_get_version(void); + +Index: asn1_primitive.c +=================================================================== +--- a/asn1_primitive.c (.../tags/1.4.10.2) (revision 0) ++++ b/asn1_primitive.c (.../branches/1.4) (revision 1357) +@@ -0,0 +1,1306 @@ ++/* ++ * libpri: An implementation of Primary Rate ISDN ++ * ++ * Copyright (C) 2009 Digium, Inc. ++ * ++ * Richard Mudgett <rmudgett@digium.com> ++ * ++ * See http://www.asterisk.org for more information about ++ * the Asterisk project. Please do not directly contact ++ * any of the maintainers of this project for assistance; ++ * the project provides a web site, mailing lists and IRC ++ * channels for your use. ++ * ++ * This program is free software, distributed under the terms of ++ * the GNU General Public License Version 2 as published by the ++ * Free Software Foundation. See the LICENSE file included with ++ * this program for more details. ++ * ++ * In addition, when this program is distributed with Asterisk in ++ * any form that would qualify as a 'combined work' or as a ++ * 'derivative work' (but not mere aggregation), you can redistribute ++ * and/or modify the combination under the terms of the license ++ * provided with that copy of Asterisk, instead of the license ++ * terms granted here. ++ */ ++ ++/*! ++ * \file ++ * \brief ASN.1 BER encode/decode primitives ++ * ++ * \author Richard Mudgett <rmudgett@digium.com> ++ */ ++ ++ ++#include <stdio.h> ++#include <ctype.h> ++ ++#include "compat.h" ++#include "libpri.h" ++#include "pri_internal.h" ++#include "asn1.h" ++ ++/* ------------------------------------------------------------------- */ ++ ++/*! ++ * \internal ++ * \brief Dump the memory contents indicated in printable characters. (Helper function.) ++ * ++ * \param ctrl D channel controller for any diagnostic messages. ++ * \param start Dump memory starting position. ++ * \param end Dump memory ending position. (Not included in dump.) ++ * ++ * \return Nothing ++ */ ++static void asn1_dump_mem_helper(struct pri *ctrl, const unsigned char *start, ++ const unsigned char *end) ++{ ++ pri_message(ctrl, " - \""); ++ for (; start < end; ++start) { ++ pri_message(ctrl, "%c", (isprint(*start)) ? *start : '~'); ++ } ++ pri_message(ctrl, "\"\n"); ++} ++ ++/*! ++ * \internal ++ * \brief Dump the memory contents indicated. ++ * ++ * \param ctrl D channel controller for any diagnostic messages. ++ * \param indent Number of spaces to indent for each new memory dump line. ++ * \param pos Dump memory starting position. ++ * \param length Number of bytes to dump. ++ * ++ * \return Nothing ++ */ ++static void asn1_dump_mem(struct pri *ctrl, unsigned indent, const unsigned char *pos, ++ unsigned length) ++{ ++ const unsigned char *seg_start; ++ const unsigned char *end; ++ unsigned delimiter; ++ unsigned count; ++ ++ seg_start = pos; ++ end = pos + length; ++ if (pos < end) { ++ delimiter = '<'; ++ for (;;) { ++ pri_message(ctrl, "%*s", indent, ""); ++ for (count = 0; count++ < 16 && pos < end;) { ++ pri_message(ctrl, "%c%02X", delimiter, *pos++); ++ delimiter = (count == 8) ? '-' : ' '; ++ } ++ if (end <= pos) { ++ break; ++ } ++ asn1_dump_mem_helper(ctrl, seg_start, pos); ++ seg_start = pos; ++ } ++ } else { ++ pri_message(ctrl, "%*s<", indent, ""); ++ } ++ pri_message(ctrl, ">"); ++ asn1_dump_mem_helper(ctrl, seg_start, end); ++} ++ ++/*! ++ * \brief Convert the given tag value to a descriptive string. ++ * ++ * \param tag Component tag value to convert to a string. ++ * ++ * \return Converted tag string. ++ */ ++const char *asn1_tag2str(unsigned tag) ++{ ++ static const char *primitives[32] = { ++ [ASN1_TYPE_INDEF_TERM] = "Indefinite length terminator", ++ [ASN1_TYPE_BOOLEAN] = "Boolean", ++ [ASN1_TYPE_INTEGER] = "Integer", ++ [ASN1_TYPE_BIT_STRING] = "Bit String", ++ [ASN1_TYPE_OCTET_STRING] = "Octet String", ++ [ASN1_TYPE_NULL] = "NULL", ++ [ASN1_TYPE_OBJECT_IDENTIFIER] = "OID", ++ [ASN1_TYPE_OBJECT_DESCRIPTOR] = "Object Descriptor", ++ [ASN1_TYPE_EXTERN] = "External", ++ [ASN1_TYPE_REAL] = "Real", ++ [ASN1_TYPE_ENUMERATED] = "Enumerated", ++ [ASN1_TYPE_EMBEDDED_PDV] = "Embedded PDV", ++ [ASN1_TYPE_UTF8_STRING] = "UTF8 String", ++ [ASN1_TYPE_RELATIVE_OID] = "Relative OID", ++ [ASN1_TYPE_SEQUENCE] = "Sequence", ++ [ASN1_TYPE_SET] = "Set", ++ [ASN1_TYPE_NUMERIC_STRING] = "Numeric String", ++ [ASN1_TYPE_PRINTABLE_STRING] = "Printable String", ++ [ASN1_TYPE_TELETEX_STRING] = "Teletex String", ++ [ASN1_TYPE_VIDEOTEX_STRING] = "Videotex String", ++ [ASN1_TYPE_IA5_STRING] = "IA5 String", ++ [ASN1_TYPE_UTC_TIME] = "UTC Time", ++ [ASN1_TYPE_GENERALIZED_TIME] = "Generalized Time", ++ [ASN1_TYPE_GRAPHIC_STRING] = "Graphic String", ++ [ASN1_TYPE_VISIBLE_STRING] = "Visible/ISO646 String", ++ [ASN1_TYPE_GENERAL_STRING] = "General String", ++ [ASN1_TYPE_UNIVERSAL_STRING] = "Universal String", ++ [ASN1_TYPE_CHAR_STRING] = "Character String", ++ [ASN1_TYPE_BMP_STRING] = "BMP String", ++ [ASN1_TYPE_EXTENSION] = "Type Extension", ++ }; ++ static char buf[64]; ++ const char *description; ++ unsigned asn1_constructed; /*! TRUE if the tag is constructed. */ ++ unsigned asn1_type; ++ ++ asn1_constructed = ((tag & ASN1_PC_MASK) == ASN1_PC_CONSTRUCTED); ++ asn1_type = tag & ASN1_TYPE_MASK; ++ ++ switch (tag & ASN1_CLASS_MASK) { ++ case ASN1_CLASS_UNIVERSAL: ++ if (tag == (ASN1_CLASS_UNIVERSAL | ASN1_PC_CONSTRUCTED | ASN1_TYPE_INDEF_TERM)) { ++ description = NULL; ++ } else { ++ description = primitives[asn1_type]; ++ } ++ if (!description) { ++ description = "Reserved"; ++ } ++ snprintf(buf, sizeof(buf), "%s%s(%u 0x%02X)", description, ++ asn1_constructed ? "/C" : "", tag, tag); ++ return buf; ++ case ASN1_CLASS_APPLICATION: ++ description = "Application"; ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC: ++ description = "Context Specific"; ++ break; ++ case ASN1_CLASS_PRIVATE: ++ description = "Private"; ++ break; ++ default: ++ snprintf(buf, sizeof(buf), "Unknown tag (%u 0x%02X)", tag, tag); ++ return buf; ++ } ++ snprintf(buf, sizeof(buf), "%s%s [%u 0x%02X]", description, ++ asn1_constructed ? "/C" : "", asn1_type, asn1_type); ++ return buf; ++} ++ ++/*! ++ * \brief Decode the ASN.1 tag value. ++ * ++ * \param tag_pos ASN.1 tag starting position. ++ * \param end End of ASN.1 encoded data buffer. ++ * \param tag Decoded tag value returned on success. ++ * ++ * \retval Next octet after the tag on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *asn1_dec_tag(const unsigned char *tag_pos, const unsigned char *end, ++ unsigned *tag) ++{ ++ unsigned extended_tag; ++ ++ if (end <= tag_pos) { ++ return NULL; ++ } ++ *tag = *tag_pos++; ++ if ((*tag & ASN1_TYPE_MASK) == ASN1_TYPE_EXTENSION) { ++ /* Extract the extended tag value */ ++ extended_tag = 0; ++ do { ++ if (end <= tag_pos) { ++ return NULL; ++ } ++ extended_tag <<= 7; ++ extended_tag |= *tag_pos & ~0x80; ++ } while (*tag_pos++ & 0x80); ++ if (extended_tag && extended_tag < ASN1_TYPE_EXTENSION) { ++ /* ++ * The sender did not need to use the extended format. ++ * This is an encoding error on their part, but we will ++ * accept it anyway. ++ * ++ * Note we cannot return a null tag value from this path. ++ * We would misinterpret the indefinite length ++ * terminator. ++ */ ++ *tag &= ~ASN1_TYPE_MASK; ++ *tag |= extended_tag; ++ } ++ } ++ ++ return tag_pos; ++} ++ ++/*! ++ * \brief Decode the length of an ASN.1 component length. ++ * ++ * \param len_pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param length Decoded length value returned on success. (-1 if indefinite) ++ * ++ * \retval Next octet after the length on success. ++ * \retval NULL on error. ++ * ++ * \note The decoded length is checked to see if there is enough buffer ++ * left for the component body. ++ */ ++const unsigned char *asn1_dec_length(const unsigned char *len_pos, ++ const unsigned char *end, int *length) ++{ ++ unsigned length_size; ++ ++ if (end <= len_pos) { ++ /* Not enough buffer to determine how the length is encoded */ ++ return NULL; ++ } ++ ++ if (*len_pos < 0x80) { ++ /* Short length encoding */ ++ *length = *len_pos++; ++ } else if (*len_pos == 0x80) { ++ /* Indefinite length encoding */ ++ *length = -1; ++ ++len_pos; ++ if (end < len_pos + ASN1_INDEF_TERM_LEN) { ++ /* Not enough buffer for the indefinite length terminator */ ++ return NULL; ++ } ++ return len_pos; ++ } else { ++ /* Long length encoding */ ++ length_size = *len_pos++ & 0x7f; ++ if (length_size == 0x7f) { ++ /* Reserved extension encoding that has not been defined. */ ++ return NULL; ++ } ++ if (end < len_pos + length_size) { ++ /* Not enough buffer for the length value */ ++ return NULL; ++ } ++ *length = 0; ++ while (length_size--) { ++ *length = (*length << 8) | *len_pos++; ++ } ++ } ++ ++ if (end < len_pos + *length) { ++ /* Not enough buffer for the component body. */ ++ return NULL; ++ } ++ return len_pos; ++} ++ ++/*! ++ * \internal ++ * \brief Skip to the end of an indefinite length constructed component helper. ++ * ++ * \param pos ASN.1 tag starting position. ++ * \param end End of ASN.1 decoding data buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++static const unsigned char *asn1_dec_indef_end_fixup_helper(const unsigned char *pos, ++ const unsigned char *end) ++{ ++ unsigned tag; ++ int length; ++ ++ while (pos < end && *pos != ASN1_INDEF_TERM) { ++ ASN1_CALL(pos, asn1_dec_tag(pos, end, &tag)); ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ if (length < 0) { ++ /* Skip over indefinite length sub-component */ ++ if ((tag & ASN1_PC_MASK) == ASN1_PC_CONSTRUCTED ++ || tag == (ASN1_CLASS_UNIVERSAL | ASN1_PC_PRIMITIVE | ASN1_TYPE_SET) ++ || tag == ++ (ASN1_CLASS_UNIVERSAL | ASN1_PC_PRIMITIVE | ASN1_TYPE_SEQUENCE)) { ++ /* This is an ITU encoded indefinite length component. */ ++ ASN1_CALL(pos, asn1_dec_indef_end_fixup_helper(pos, end)); ++ } else { ++ /* This is a non-ITU encoded indefinite length component. */ ++ while (pos < end && *pos != ASN1_INDEF_TERM) { ++ ++pos; ++ } ++ pos += ASN1_INDEF_TERM_LEN; ++ } ++ } else { ++ /* Skip over defininte length sub-component */ ++ pos += length; ++ } ++ } ++ if (end < pos + ASN1_INDEF_TERM_LEN) { ++ return NULL; ++ } ++ ++ return pos + ASN1_INDEF_TERM_LEN; ++} ++ ++/*! ++ * \brief Skip to the end of an indefinite length constructed component. ++ * ++ * \param ctrl D channel controller for any diagnostic messages. ++ * \param pos ASN.1 tag starting position. ++ * \param end End of ASN.1 decoding data buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *asn1_dec_indef_end_fixup(struct pri *ctrl, const unsigned char *pos, ++ const unsigned char *end) ++{ ++ if (pos < end && *pos != ASN1_INDEF_TERM && (ctrl->debug & PRI_DEBUG_APDU)) { ++ pri_message(ctrl, ++ " Skipping unused indefinite length constructed component octets!\n"); ++ } ++ return asn1_dec_indef_end_fixup_helper(pos, end); ++} ++ ++/*! ++ * \brief Decode the boolean primitive. ++ * ++ * \param ctrl D channel controller for any diagnostic messages. ++ * \param name Field name ++ * \param tag Component tag that identified this primitive. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param value Decoded boolean value. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *asn1_dec_boolean(struct pri *ctrl, const char *name, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, int32_t *value) ++{ ++ int length; ++ ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ if (length != 1) { ++ /* ++ * The encoding rules say the length can only be one. ++ * It is rediculus to get anything else anyway. ++ */ ++ return NULL; ++ } ++ ++ *value = *pos++ ? 1 : 0; ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s %s = %d\n", name, asn1_tag2str(tag), *value); ++ } ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the integer type primitive. ++ * ++ * \param ctrl D channel controller for any diagnostic messages. ++ * \param name Field name ++ * \param tag Component tag that identified this primitive. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param value Decoded integer type value. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *asn1_dec_int(struct pri *ctrl, const char *name, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, int32_t *value) ++{ ++ int length; ++ ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ if (length <= 0) { ++ /* ++ * The encoding rules say the length can not be indefinite. ++ * It cannot be empty for that matter either. ++ */ ++ return NULL; ++ } ++ ++#if 1 ++ /* Read value as signed */ ++ if (*pos & 0x80) { ++ /* The value is negative */ ++ *value = -1; ++ } else { ++ *value = 0; ++ } ++#else ++ /* Read value as unsigned */ ++ *value = 0; ++#endif ++ while (length--) { ++ *value = (*value << 8) | *pos; ++ pos++; ++ } ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s %s = %d 0x%04X\n", name, asn1_tag2str(tag), *value, ++ *value); ++ } ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the null primitive. ++ * ++ * \param ctrl D channel controller for any diagnostic messages. ++ * \param name Field name ++ * \param tag Component tag that identified this primitive. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *asn1_dec_null(struct pri *ctrl, const char *name, unsigned tag, ++ const unsigned char *pos, const unsigned char *end) ++{ ++ int length; ++ ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ if (length != 0) { ++ /* ++ * The encoding rules say the length can only be zero. ++ * It is rediculus to get anything else anyway. ++ */ ++ return NULL; ++ } ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s %s\n", name, asn1_tag2str(tag)); ++ } ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the object identifier primitive. ++ * ++ * \param ctrl D channel controller for any diagnostic messages. ++ * \param name Field name ++ * \param tag Component tag that identified this primitive. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param oid Decoded OID type value. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *asn1_dec_oid(struct pri *ctrl, const char *name, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, struct asn1_oid *oid) ++{ ++ int length; ++ unsigned num_values; ++ unsigned value; ++ unsigned delimiter; ++ ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ if (length < 0) { ++ /* ++ * The encoding rules say the length can not be indefinite. ++ */ ++ return NULL; ++ } ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s %s =", name, asn1_tag2str(tag)); ++ } ++ delimiter = ' '; ++ num_values = 0; ++ while (length) { ++ value = 0; ++ for (;;) { ++ --length; ++ value = (value << 7) | (*pos & 0x7F); ++ if (!(*pos++ & 0x80)) { ++ /* Last octet in the OID subidentifier value */ ++ if (num_values < ARRAY_LEN(oid->value)) { ++ oid->value[num_values] = value; ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, "%c%u", delimiter, value); ++ } ++ delimiter = '.'; ++ } else { ++ /* Too many OID subidentifier values */ ++ delimiter = '~'; ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, "%c%u", delimiter, value); ++ } ++ } ++ ++num_values; ++ break; ++ } ++ if (!length) { ++ oid->num_values = 0; ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, "\n" ++ " Last OID subidentifier value not terminated!\n"); ++ } ++ return NULL; ++ } ++ } ++ } ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, "\n"); ++ } ++ ++ if (num_values <= ARRAY_LEN(oid->value)) { ++ oid->num_values = num_values; ++ return pos; ++ } else { ++ /* Need to increase the size of the OID subidentifier list. */ ++ oid->num_values = 0; ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " Too many OID values!\n"); ++ } ++ return NULL; ++ } ++} ++ ++/*! ++ * \brief Decode a binary string primitive. ++ * ++ * \param ctrl D channel controller for any diagnostic messages. ++ * \param name Field name ++ * \param tag Component tag that identified this primitive. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param buf_size Size of the supplied string buffer. (Must be nonzero) ++ * \param str Where to put the decoded string. ++ * \param str_len Length of the decoded string. ++ * ++ * \note The string will be null terminated just in case. ++ * The buffer needs to have enough room for a null terminator. ++ * \note The parse will fail if the parsed string is too large for ++ * the supplied buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *asn1_dec_string_bin(struct pri *ctrl, const char *name, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, size_t buf_size, ++ unsigned char *str, size_t *str_len) ++{ ++ int length; ++ size_t sub_buf_size; ++ size_t sub_str_len; ++ unsigned char *sub_str; ++ ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ if (length < 0) { ++ /* This is an indefinite length string */ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s %s = Indefinite length string\n", name, ++ asn1_tag2str(tag)); ++ } ++ if ((tag & ASN1_PC_MASK) == ASN1_PC_CONSTRUCTED) { ++ /* ++ * This is an ITU encoded indefinite length string ++ * and could contain null. ++ */ ++ ++ /* Ensure that an empty string is null terminated. */ ++ *str = 0; ++ ++ /* Collect all substrings into the original string buffer. */ ++ *str_len = 0; ++ sub_str = str; ++ sub_buf_size = buf_size; ++ for (;;) { ++ ASN1_CALL(pos, asn1_dec_tag(pos, end, &tag)); ++ if (tag == ASN1_INDEF_TERM) { ++ /* End-of-contents octets */ ++ break; ++ } ++ ++ /* Append the substring to the accumulated indefinite string. */ ++ ASN1_CALL(pos, asn1_dec_string_bin(ctrl, name, tag, pos, end, ++ sub_buf_size, sub_str, &sub_str_len)); ++ ++ sub_buf_size -= sub_str_len; ++ sub_str += sub_str_len; ++ *str_len += sub_str_len; ++ } ++ } else { ++ /* ++ * This is a non-ITU encoded indefinite length string ++ * and must not contain null's. ++ */ ++ for (length = 0;; ++length) { ++ if (end <= pos + length) { ++ /* Not enough buffer left */ ++ return NULL; ++ } ++ if (pos[length] == 0) { ++ /* Found End-of-contents octets */ ++ break; ++ } ++ } ++ ++ if (buf_size - 1 < length) { ++ /* The destination buffer is not large enough for the data */ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " String buffer not large enough!\n"); ++ } ++ return NULL; ++ } ++ ++ /* Extract the string and null terminate it. */ ++ memcpy(str, pos, length); ++ str[length] = 0; ++ *str_len = length; ++ ++ pos += length + 1; ++ } ++ if (end <= pos) { ++ /* Not enough buffer left for End-of-contents octets */ ++ return NULL; ++ } ++ if (*pos++ != 0) { ++ /* We actually did not find the End-of-contents octets. */ ++ return NULL; ++ } ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ /* Dump the collected string buffer contents. */ ++ pri_message(ctrl, " Completed string =\n"); ++ asn1_dump_mem(ctrl, 6, str, *str_len); ++ } ++ } else { ++ /* This is a definite length string */ ++ if (buf_size - 1 < length) { ++ /* The destination buffer is not large enough for the data */ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s %s = Buffer not large enough!\n", name, ++ asn1_tag2str(tag)); ++ } ++ return NULL; ++ } ++ ++ /* Extract the string and null terminate it. */ ++ memcpy(str, pos, length); ++ str[length] = 0; ++ *str_len = length; ++ ++ pos += length; ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ /* Dump the collected string buffer contents. */ ++ pri_message(ctrl, " %s %s =\n", name, asn1_tag2str(tag)); ++ asn1_dump_mem(ctrl, 4, str, *str_len); ++ } ++ } ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode a string that can be truncated to a maximum length primitive. ++ * ++ * \param ctrl D channel controller for any diagnostic messages. ++ * \param name Field name ++ * \param tag Component tag that identified this primitive. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param buf_size Size of the supplied string buffer. (Must be nonzero) ++ * \param str Where to put the decoded null terminated string. ++ * \param str_len Length of the decoded string. ++ * (computed for convenience since you could just do a strlen()) ++ * ++ * \note The parsed string will be truncated if the string buffer ++ * cannot contain it. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *asn1_dec_string_max(struct pri *ctrl, const char *name, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, size_t buf_size, ++ unsigned char *str, size_t *str_len) ++{ ++ int length; ++ size_t str_length; ++ size_t sub_buf_size; ++ size_t sub_str_len; ++ unsigned char *sub_str; ++ ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ if (length < 0) { ++ /* This is an indefinite length string */ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s %s = Indefinite length string\n", name, ++ asn1_tag2str(tag)); ++ } ++ if ((tag & ASN1_PC_MASK) == ASN1_PC_CONSTRUCTED) { ++ /* This is an ITU encoded indefinite length string. */ ++ ++ /* Ensure that an empty string is null terminated. */ ++ *str = 0; ++ ++ /* Collect all substrings into the original string buffer. */ ++ *str_len = 0; ++ sub_str = str; ++ sub_buf_size = buf_size; ++ for (;;) { ++ ASN1_CALL(pos, asn1_dec_tag(pos, end, &tag)); ++ if (tag == ASN1_INDEF_TERM) { ++ /* End-of-contents octets */ ++ break; ++ } ++ ++ /* Append the substring to the accumulated indefinite string. */ ++ ASN1_CALL(pos, asn1_dec_string_max(ctrl, name, tag, pos, end, ++ sub_buf_size, sub_str, &sub_str_len)); ++ ++ sub_buf_size -= sub_str_len; ++ sub_str += sub_str_len; ++ *str_len += sub_str_len; ++ } ++ } else { ++ /* This is a non-ITU encoded indefinite length string. */ ++ for (length = 0;; ++length) { ++ if (end <= pos + length) { ++ /* Not enough buffer left */ ++ return NULL; ++ } ++ if (pos[length] == 0) { ++ /* Found End-of-contents octets */ ++ break; ++ } ++ } ++ ++ /* Extract the string, truncate if necessary, and terminate it. */ ++ str_length = (buf_size - 1 < length) ? buf_size - 1 : length; ++ memcpy(str, pos, str_length); ++ str[str_length] = 0; ++ *str_len = str_length; ++ ++ pos += length + 1; ++ } ++ if (end <= pos) { ++ /* Not enough buffer left for End-of-contents octets */ ++ return NULL; ++ } ++ if (*pos++ != 0) { ++ /* We actually did not find the End-of-contents octets. */ ++ return NULL; ++ } ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " Completed string = \"%s\"\n", str); ++ } ++ } else { ++ /* ++ * This is a definite length string ++ * ++ * Extract the string, truncate if necessary, and terminate it. ++ */ ++ str_length = (buf_size - 1 < length) ? buf_size - 1 : length; ++ memcpy(str, pos, str_length); ++ str[str_length] = 0; ++ *str_len = str_length; ++ ++ pos += length; ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s %s = \"%s\"\n", name, asn1_tag2str(tag), str); ++ } ++ } ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Recursive ASN.1 buffer decoding dump helper. ++ * ++ * \param ctrl D channel controller for any diagnostic messages. ++ * \param pos ASN.1 tag starting position. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param level Indentation level to use. ++ * \param indefinite_term TRUE if to stop on an indefinite length terminator. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++static const unsigned char *asn1_dump_helper(struct pri *ctrl, const unsigned char *pos, ++ const unsigned char *end, unsigned level, unsigned indefinite_term) ++{ ++ unsigned delimiter; ++ unsigned tag; ++ int length; ++ const unsigned char *len_pos; ++ ++ while (pos < end && (!indefinite_term || *pos != ASN1_INDEF_TERM)) { ++ /* Decode the tag */ ++ pri_message(ctrl, "%*s", 2 * level, ""); ++ len_pos = asn1_dec_tag(pos, end, &tag); ++ if (!len_pos) { ++ pri_message(ctrl, "Invalid tag encoding!\n"); ++ return NULL; ++ } ++ ++ /* Dump the tag contents. */ ++ pri_message(ctrl, "%s ", asn1_tag2str(tag)); ++ delimiter = '<'; ++ while (pos < len_pos) { ++ pri_message(ctrl, "%c%02X", delimiter, *pos); ++ delimiter = ' '; ++ ++pos; ++ } ++ pri_message(ctrl, "> "); ++ ++ /* Decode the length */ ++ pos = asn1_dec_length(len_pos, end, &length); ++ if (!pos) { ++ pri_message(ctrl, "Invalid length encoding!\n"); ++ return NULL; ++ } ++ ++ /* Dump the length contents. */ ++ if (length < 0) { ++ pri_message(ctrl, "Indefinite length "); ++ } else { ++ pri_message(ctrl, "Len:%d ", length); ++ } ++ delimiter = '<'; ++ while (len_pos < pos) { ++ pri_message(ctrl, "%c%02X", delimiter, *len_pos); ++ delimiter = ' '; ++ ++len_pos; ++ } ++ pri_message(ctrl, ">\n"); ++ ++ /* Dump the body contents */ ++ ++level; ++ if (length < 0) { ++ /* Indefinite length */ ++ if ((tag & ASN1_PC_MASK) == ASN1_PC_CONSTRUCTED) { ++ /* This is an ITU encoded indefinite length component. */ ++ ASN1_CALL(pos, asn1_dump_helper(ctrl, pos, end, level, 1)); ++ } else if (tag == (ASN1_CLASS_UNIVERSAL | ASN1_PC_PRIMITIVE | ASN1_TYPE_SET) ++ || tag == ++ (ASN1_CLASS_UNIVERSAL | ASN1_PC_PRIMITIVE | ASN1_TYPE_SEQUENCE)) { ++ pri_message(ctrl, "%*sThis tag must always be constructed!\n", 2 * level, ++ ""); ++ /* Assume tag was constructed to keep going */ ++ ASN1_CALL(pos, asn1_dump_helper(ctrl, pos, end, level, 1)); ++ } else { ++ /* This is a non-ITU encoded indefinite length component. */ ++ pri_message(ctrl, "%*sNon-ITU indefininte length component.\n", ++ 2 * level, ""); ++ length = 0; ++ while (pos + length < end && pos[length] != ASN1_INDEF_TERM) { ++ ++length; ++ } ++ if (length) { ++ asn1_dump_mem(ctrl, 2 * level, pos, length); ++ pos += length; ++ } ++ } ++ --level; ++ if (end < pos + ASN1_INDEF_TERM_LEN) { ++ pri_message(ctrl, "%*sNot enough room for the End-of-contents octets!\n", ++ 2 * level, ""); ++ pos = end; ++ } else { ++ pri_message(ctrl, "%*sEnd-of-contents <%02X %02X>%s\n", 2 * level, "", ++ pos[0], pos[1], (pos[1] != 0) ? " Invalid!" : ""); ++ pos += ASN1_INDEF_TERM_LEN; ++ } ++ } else { ++ /* Defininte length */ ++ if ((tag & ASN1_PC_MASK) == ASN1_PC_CONSTRUCTED) { ++ /* Dump constructed contents */ ++ ASN1_CALL(pos, asn1_dump_helper(ctrl, pos, pos + length, level, 0)); ++ } else if (tag == (ASN1_CLASS_UNIVERSAL | ASN1_PC_PRIMITIVE | ASN1_TYPE_SET) ++ || tag == ++ (ASN1_CLASS_UNIVERSAL | ASN1_PC_PRIMITIVE | ASN1_TYPE_SEQUENCE)) { ++ /* Assume tag was constructed to keep going */ ++ pri_message(ctrl, "%*sThis tag must always be constructed!\n", 2 * level, ++ ""); ++ ASN1_CALL(pos, asn1_dump_helper(ctrl, pos, pos + length, level, 0)); ++ } else if (0 < length) { ++ /* Dump primitive contents. */ ++ asn1_dump_mem(ctrl, 2 * level, pos, length); ++ pos += length; ++ } ++ --level; ++ } ++ ++#if 0 ++ pri_message(ctrl, "%*sEnd\n", 2 * level, ""); ++#endif ++ } ++ ++ return pos; ++} ++ ++/*! ++ * \brief Dump the given ASN.1 buffer contents. ++ * ++ * \param ctrl D channel controller for any diagnostic messages. ++ * \param start_asn1 First octet in the ASN.1 buffer. (ASN.1 tag starting position) ++ * \param end One beyond the last octet in the ASN.1 buffer. ++ * ++ * \return Nothing ++ */ ++void asn1_dump(struct pri *ctrl, const unsigned char *start_asn1, ++ const unsigned char *end) ++{ ++ pri_message(ctrl, "ASN.1 dump\n"); ++ if (start_asn1) { ++ asn1_dump_helper(ctrl, start_asn1, end, 1, 0); ++ } ++ pri_message(ctrl, "ASN.1 end\n"); ++} ++ ++/*! ++ * \brief Encode the length of an ASN.1 component body of predetermined size. ++ * ++ * \param len_pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param length Predetermined component body length. ++ * ++ * \note The encoding buffer does not need to be checked after calling. ++ * It is already checked to have the requested room. ++ * ++ * \retval Next octet after the length on success. ++ * \retval NULL on error. ++ */ ++unsigned char *asn1_enc_length(unsigned char *len_pos, unsigned char *end, size_t length) ++{ ++ u_int32_t body_length; /* Length of component contents */ ++ u_int32_t value; ++ u_int32_t test_mask; ++ unsigned length_size; /* Length of the length encoding */ ++ ++ body_length = length; ++ ++ /* Determine length encoding length */ ++ if (body_length < 128) { ++ length_size = 1; ++ } else { ++ /* Find most significant octet of 32 bit integer that carries meaning. */ ++ test_mask = 0xFF000000; ++ for (length_size = 4; --length_size;) { ++ if (body_length & test_mask) { ++ /* ++ * Found the first 8 bits of a multiple octet length that ++ * is not all zeroes. ++ */ ++ break; ++ } ++ test_mask >>= 8; ++ } ++ length_size += 1 + 1; ++ } ++ ++ if (end < len_pos + length_size + body_length) { ++ /* No room for the length and component body in the buffer */ ++ return NULL; ++ } ++ ++ /* Encode the component body length */ ++ if (length_size == 1) { ++ *len_pos++ = body_length; ++ } else { ++ *len_pos++ = 0x80 | --length_size; ++ while (length_size--) { ++ value = body_length; ++ value >>= (8 * length_size); ++ *len_pos++ = value & 0xFF; ++ } ++ } ++ ++ return len_pos; ++} ++ ++/*! ++ * \brief Encode the length of an already encoded ASN.1 component. ++ * ++ * \param len_pos Starting position of the ASN.1 component length. ++ * \param component_end Next octet after the component body. ++ * \param end End of ASN.1 encoding data buffer. ++ * ++ * \note The total component size could increase or decrease. ++ * \note The component length field must have been initialized with ++ * ASN1_LEN_INIT() or ASN1_CONSTRUCTED_BEGIN(). ++ * ++ * \retval Next octet after the component body on success. ++ * \retval NULL on error. ++ */ ++unsigned char *asn1_enc_length_fixup(unsigned char *len_pos, ++ unsigned char *component_end, unsigned char *end) ++{ ++ u_int32_t body_length; /* Length of component contents */ ++ u_int32_t value; ++ u_int32_t test_mask; ++ unsigned length_size; /* Length of the length encoding */ ++ ++ if (component_end < len_pos + *len_pos) { ++ /* Sanity check */ ++ return NULL; ++ } ++ ++ body_length = component_end - len_pos - *len_pos; ++ ++ /* Determine length encoding length */ ++ if (body_length < 128) { ++ length_size = 1; ++ } else { ++ /* Find most significant octet of 32 bit integer that carries meaning. */ ++ test_mask = 0xFF000000; ++ for (length_size = 4; --length_size;) { ++ if (body_length & test_mask) { ++ /* ++ * Found the first 8 bits of a multiple octet length that ++ * is not all zeroes. ++ */ ++ break; ++ } ++ test_mask >>= 8; ++ } ++ length_size += 1 + 1; ++ } ++ ++ component_end = len_pos + length_size + body_length; ++ if (end < component_end) { ++ /* No room for the component in the buffer */ ++ return NULL; ++ } ++ if (length_size != *len_pos) { ++ /* Must shift the component body */ ++ memmove(len_pos + length_size, len_pos + *len_pos, body_length); ++ } ++ ++ /* Encode the component body length */ ++ if (length_size == 1) { ++ *len_pos = body_length; ++ } else { ++ *len_pos++ = 0x80 | --length_size; ++ while (length_size--) { ++ value = body_length; ++ value >>= (8 * length_size); ++ *len_pos++ = value & 0xFF; ++ } ++ } ++ ++ return component_end; ++} ++ ++/*! ++ * \brief Encode the boolean primitive. ++ * ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param tag Component tag to identify the encoded component. ++ * \param value Component value to encode. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *asn1_enc_boolean(unsigned char *pos, unsigned char *end, unsigned tag, ++ int32_t value) ++{ ++ if (end < pos + 3) { ++ /* No room for the component in the buffer */ ++ return NULL; ++ } ++ ++ /* Encode component */ ++ *pos++ = tag; ++ *pos++ = 1; ++ *pos++ = value ? 1 : 0; ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the integer type primitive. ++ * ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param tag Component tag to identify the encoded component. ++ * \param value Component value to encode. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *asn1_enc_int(unsigned char *pos, unsigned char *end, unsigned tag, ++ int32_t value) ++{ ++ unsigned count; ++ u_int32_t test_mask; ++ u_int32_t val; ++ ++ /* Find most significant octet of 32 bit integer that carries meaning. */ ++ test_mask = 0xFF800000; ++ val = (u_int32_t) value; ++ for (count = 4; --count;) { ++ if ((val & test_mask) != test_mask && (val & test_mask) != 0) { ++ /* ++ * The first 9 bits of a multiple octet integer is not ++ * all ones or zeroes. ++ */ ++ break; ++ } ++ test_mask >>= 8; ++ } ++ ++ if (end < pos + 3 + count) { ++ /* No room for the component in the buffer */ ++ return NULL; ++ } ++ ++ /* Encode component */ ++ *pos++ = tag; ++ *pos++ = count + 1; ++ do { ++ val = (u_int32_t) value; ++ val >>= (8 * count); ++ *pos++ = val & 0xFF; ++ } while (count--); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the null type primitive. ++ * ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param tag Component tag to identify the encoded component. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *asn1_enc_null(unsigned char *pos, unsigned char *end, unsigned tag) ++{ ++ if (end < pos + 2) { ++ /* No room for the component in the buffer */ ++ return NULL; ++ } ++ ++ /* Encode component */ ++ *pos++ = tag; ++ *pos++ = 0; ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the object identifier (OID) primitive. ++ * ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param tag Component tag to identify the encoded component. ++ * \param oid Component value to encode. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *asn1_enc_oid(unsigned char *pos, unsigned char *end, unsigned tag, ++ const struct asn1_oid *oid) ++{ ++ unsigned char *len_pos; ++ unsigned num_values; ++ unsigned count; ++ u_int32_t value; ++ ++ if (end < pos + 2) { ++ /* No room for the component tag and length in the buffer */ ++ return NULL; ++ } ++ ++ *pos++ = tag; ++ len_pos = pos++; ++ ++ /* For all OID subidentifer values */ ++ for (num_values = 0; num_values < oid->num_values; ++num_values) { ++ /* ++ * Count the number of 7 bit chunks that are needed ++ * to encode the integer. ++ */ ++ value = oid->value[num_values] >> 7; ++ for (count = 0; value; ++count) { ++ /* There are bits still set */ ++ value >>= 7; ++ } ++ ++ if (end < pos + count + 1) { ++ /* No room for the component body in the buffer */ ++ return NULL; ++ } ++ ++ /* Store OID subidentifier value */ ++ do { ++ value = oid->value[num_values]; ++ value >>= (7 * count); ++ *pos++ = (value & 0x7F) | (count ? 0x80 : 0); ++ } while (count--); ++ } ++ ++ /* length */ ++ *len_pos = pos - len_pos - 1; ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the binary string type primitive. ++ * ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param tag Component tag to identify the encoded component. ++ * \param str Binary string to encode. ++ * \param str_len Length of binary string to encode. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *asn1_enc_string_bin(unsigned char *pos, unsigned char *end, unsigned tag, ++ const unsigned char *str, size_t str_len) ++{ ++ if (end < pos + 1) { ++ /* No room for the component tag in the buffer */ ++ return NULL; ++ } ++ ++ /* Encode component */ ++ *pos++ = tag; ++ ASN1_CALL(pos, asn1_enc_length(pos, end, str_len)); ++ memcpy(pos, str, str_len); ++ ++ return pos + str_len; ++} ++ ++/*! ++ * \brief Encode a string that can be truncated to a maximum length primitive. ++ * ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param tag Component tag to identify the encoded component. ++ * \param str Null terminated string to encode. ++ * \param max_len Maximum length of string to encode. ++ * ++ * \note The string will be truncated if it is too long. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *asn1_enc_string_max(unsigned char *pos, unsigned char *end, unsigned tag, ++ const unsigned char *str, size_t max_len) ++{ ++ size_t str_len; ++ ++ str_len = strlen((char *) str); ++ if (max_len < str_len) { ++ str_len = max_len; ++ } ++ return asn1_enc_string_bin(pos, end, tag, str, str_len); ++} ++ ++/* ------------------------------------------------------------------- */ ++/* end asn1_primitive.c */ + +Property changes on: asn1_primitive.c +___________________________________________________________________ +Added: svn:eol-style + + native +Added: svn:mime-type + + text/plain +Added: svn:keywords + + 'Author Date Id Revision' + +Index: pri.c +=================================================================== +--- a/pri.c (.../tags/1.4.10.2) (revision 1357) ++++ b/pri.c (.../branches/1.4) (revision 1357) +@@ -42,8 +42,56 @@ + #include "pri_facility.h" + #include "pri_q921.h" + #include "pri_q931.h" +-#include "pri_timers.h" + ++#define PRI_BIT(a_bit) (1UL << (a_bit)) ++#define PRI_ALL_SWITCHES 0xFFFFFFFF ++ ++struct pri_timer_table { ++ const char *name; ++ enum PRI_TIMERS_AND_COUNTERS number; ++ unsigned long used_by; ++}; ++ ++/*! ++ * \note Sort the timer table entries in the order of the timer name so ++ * pri_dump_info_str() can display them in a consitent order. ++ */ ++static const struct pri_timer_table pri_timer[] = { ++/* *INDENT-OFF* */ ++ /* timer name timer number used by switches */ ++ { "N200", PRI_TIMER_N200, PRI_ALL_SWITCHES }, ++ { "N201", PRI_TIMER_N201, PRI_ALL_SWITCHES }, ++ { "N202", PRI_TIMER_N202, PRI_ALL_SWITCHES }, ++ { "K", PRI_TIMER_K, PRI_ALL_SWITCHES }, ++ { "T200", PRI_TIMER_T200, PRI_ALL_SWITCHES }, ++ { "T202", PRI_TIMER_T202, PRI_ALL_SWITCHES }, ++ { "T203", PRI_TIMER_T203, PRI_ALL_SWITCHES }, ++ { "T300", PRI_TIMER_T300, PRI_ALL_SWITCHES }, ++ { "T301", PRI_TIMER_T301, PRI_ALL_SWITCHES }, ++ { "T302", PRI_TIMER_T302, PRI_ALL_SWITCHES }, ++ { "T303", PRI_TIMER_T303, PRI_ALL_SWITCHES }, ++ { "T304", PRI_TIMER_T304, PRI_ALL_SWITCHES }, ++ { "T305", PRI_TIMER_T305, PRI_ALL_SWITCHES }, ++ { "T306", PRI_TIMER_T306, PRI_ALL_SWITCHES }, ++ { "T307", PRI_TIMER_T307, PRI_ALL_SWITCHES }, ++ { "T308", PRI_TIMER_T308, PRI_ALL_SWITCHES }, ++ { "T309", PRI_TIMER_T309, PRI_ALL_SWITCHES }, ++ { "T310", PRI_TIMER_T310, PRI_ALL_SWITCHES }, ++ { "T313", PRI_TIMER_T313, PRI_ALL_SWITCHES }, ++ { "T314", PRI_TIMER_T314, PRI_ALL_SWITCHES }, ++ { "T316", PRI_TIMER_T316, PRI_ALL_SWITCHES }, ++ { "T317", PRI_TIMER_T317, PRI_ALL_SWITCHES }, ++ { "T318", PRI_TIMER_T318, PRI_ALL_SWITCHES }, ++ { "T319", PRI_TIMER_T319, PRI_ALL_SWITCHES }, ++ { "T320", PRI_TIMER_T320, PRI_ALL_SWITCHES }, ++ { "T321", PRI_TIMER_T321, PRI_ALL_SWITCHES }, ++ { "T322", PRI_TIMER_T322, PRI_ALL_SWITCHES }, ++ { "T-HOLD", PRI_TIMER_T_HOLD, PRI_ALL_SWITCHES }, ++ { "T-RETRIEVE", PRI_TIMER_T_RETRIEVE, PRI_ALL_SWITCHES }, ++ { "T-RESPONSE", PRI_TIMER_T_RESPONSE, PRI_ALL_SWITCHES }, ++/* *INDENT-ON* */ ++}; ++ + char *pri_node2str(int node) + { + switch(node) { +@@ -84,14 +132,39 @@ + } + } + +-static void pri_default_timers(struct pri *pri, int switchtype) ++static void pri_default_timers(struct pri *ctrl, int switchtype) + { +- static const int defaulttimers[20][PRI_MAX_TIMERS] = PRI_TIMERS_ALL; +- int x; ++ unsigned idx; + +- for (x = 0; x<PRI_MAX_TIMERS; x++) { +- pri->timers[x] = defaulttimers[switchtype][x]; ++ /* Initialize all timers/counters to unsupported/disabled. */ ++ for (idx = 0; idx < PRI_MAX_TIMERS; ++idx) { ++ ctrl->timers[idx] = -1; + } ++ ++ /* Set timer values to standard defaults. Time is in ms. */ ++ ctrl->timers[PRI_TIMER_N200] = 3; /* Max numer of Q.921 retransmissions */ ++ ctrl->timers[PRI_TIMER_N202] = 3; /* Max numer of transmissions of the TEI identity request message */ ++ ctrl->timers[PRI_TIMER_K] = 7; /* Max number of outstanding I-frames */ ++ ctrl->timers[PRI_TIMER_T200] = 1000; /* Time between SABME's */ ++ ctrl->timers[PRI_TIMER_T202] = 10 * 1000; /* Min time between transmission of TEI Identity request messages */ ++ ctrl->timers[PRI_TIMER_T203] = 10 * 1000; /* Max time without exchanging packets */ ++ ctrl->timers[PRI_TIMER_T305] = 30 * 1000; /* Wait for DISCONNECT acknowledge */ ++ ctrl->timers[PRI_TIMER_T308] = 4 * 1000; /* Wait for RELEASE acknowledge */ ++ ctrl->timers[PRI_TIMER_T313] = 4 * 1000; /* Wait for CONNECT acknowledge, CPE side only */ ++ ctrl->timers[PRI_TIMER_TM20] = 2500; /* Max time awaiting XID response - Q.921 Appendix IV */ ++ ctrl->timers[PRI_TIMER_NM20] = 3; /* Number of XID retransmits - Q.921 Appendix IV */ ++ ctrl->timers[PRI_TIMER_T303] = 4 * 1000; /* Length between SETUP retransmissions and timeout */ ++ ++ ctrl->timers[PRI_TIMER_T_HOLD] = 4 * 1000; /* Wait for HOLD request response. */ ++ ctrl->timers[PRI_TIMER_T_RETRIEVE] = 4 * 1000;/* Wait for RETRIEVE request response. */ ++ ++ ctrl->timers[PRI_TIMER_T_RESPONSE] = 4 * 1000; /* Maximum time to wait for a typical APDU response. */ ++ ++ /* Set any switch specific override default values */ ++ switch (switchtype) { ++ default: ++ break; ++ } + } + + int pri_set_timer(struct pri *pri, int timer, int value) +@@ -110,66 +183,30 @@ + return pri->timers[timer]; + } + +-int pri_timer2idx(char *timer) ++int pri_set_service_message_support(struct pri *pri, int supportflag) + { +- if (!strcasecmp(timer, "N200")) +- return PRI_TIMER_N200; +- else if (!strcasecmp(timer, "N201")) +- return PRI_TIMER_N201; +- else if (!strcasecmp(timer, "N202")) +- return PRI_TIMER_N202; +- else if (!strcasecmp(timer, "K")) +- return PRI_TIMER_K; +- else if (!strcasecmp(timer, "T200")) +- return PRI_TIMER_T200; +- else if (!strcasecmp(timer, "T202")) +- return PRI_TIMER_T202; +- else if (!strcasecmp(timer, "T203")) +- return PRI_TIMER_T203; +- else if (!strcasecmp(timer, "T300")) +- return PRI_TIMER_T300; +- else if (!strcasecmp(timer, "T301")) +- return PRI_TIMER_T301; +- else if (!strcasecmp(timer, "T302")) +- return PRI_TIMER_T302; +- else if (!strcasecmp(timer, "T303")) +- return PRI_TIMER_T303; +- else if (!strcasecmp(timer, "T304")) +- return PRI_TIMER_T304; +- else if (!strcasecmp(timer, "T305")) +- return PRI_TIMER_T305; +- else if (!strcasecmp(timer, "T306")) +- return PRI_TIMER_T306; +- else if (!strcasecmp(timer, "T307")) +- return PRI_TIMER_T307; +- else if (!strcasecmp(timer, "T308")) +- return PRI_TIMER_T308; +- else if (!strcasecmp(timer, "T309")) +- return PRI_TIMER_T309; +- else if (!strcasecmp(timer, "T310")) +- return PRI_TIMER_T310; +- else if (!strcasecmp(timer, "T313")) +- return PRI_TIMER_T313; +- else if (!strcasecmp(timer, "T314")) +- return PRI_TIMER_T314; +- else if (!strcasecmp(timer, "T316")) +- return PRI_TIMER_T316; +- else if (!strcasecmp(timer, "T317")) +- return PRI_TIMER_T317; +- else if (!strcasecmp(timer, "T318")) +- return PRI_TIMER_T318; +- else if (!strcasecmp(timer, "T319")) +- return PRI_TIMER_T319; +- else if (!strcasecmp(timer, "T320")) +- return PRI_TIMER_T320; +- else if (!strcasecmp(timer, "T321")) +- return PRI_TIMER_T321; +- else if (!strcasecmp(timer, "T322")) +- return PRI_TIMER_T322; +- else ++ if (!pri) { + return -1; ++ } ++ pri->service_message_support = supportflag; ++ return 0; + } + ++int pri_timer2idx(const char *timer_name) ++{ ++ unsigned idx; ++ enum PRI_TIMERS_AND_COUNTERS timer_number; ++ ++ timer_number = -1; ++ for (idx = 0; idx < ARRAY_LEN(pri_timer); ++idx) { ++ if (!strcasecmp(timer_name, pri_timer[idx].name)) { ++ timer_number = pri_timer[idx].number; ++ break; ++ } ++ } ++ return timer_number; ++} ++ + static int __pri_read(struct pri *pri, void *buf, int buflen) + { + int res = read(pri->fd, buf, buflen); +@@ -192,18 +229,53 @@ + return res; + } + +-/* Pass in the master for this function */ + void __pri_free_tei(struct pri * p) + { +- free (p); ++ if (p) { ++ struct q931_call *call; ++ ++ call = p->dummy_call; ++ if (call) { ++ pri_schedule_del(call->pri, call->retranstimer); ++ pri_call_apdu_queue_cleanup(call); ++ } ++ free(p->msg_line); ++ free(p); ++ } + } + + struct pri *__pri_new_tei(int fd, int node, int switchtype, struct pri *master, pri_io_cb rd, pri_io_cb wr, void *userdata, int tei, int bri) + { ++ struct d_ctrl_dummy *dummy_ctrl; + struct pri *p; + +- if (!(p = calloc(1, sizeof(*p)))) +- return NULL; ++ switch (switchtype) { ++ case PRI_SWITCH_GR303_EOC: ++ case PRI_SWITCH_GR303_TMC: ++ case PRI_SWITCH_GR303_TMC_SWITCHING: ++ case PRI_SWITCH_GR303_EOC_PATH: ++ p = calloc(1, sizeof(*p)); ++ if (!p) { ++ return NULL; ++ } ++ dummy_ctrl = NULL; ++ break; ++ default: ++ dummy_ctrl = calloc(1, sizeof(*dummy_ctrl)); ++ if (!dummy_ctrl) { ++ return NULL; ++ } ++ p = &dummy_ctrl->ctrl; ++ break; ++ } ++ if (!master) { ++ /* This is the master record. */ ++ p->msg_line = calloc(1, sizeof(*p->msg_line)); ++ if (!p->msg_line) { ++ free(p); ++ return NULL; ++ } ++ } + + p->bri = bri; + p->fd = fd; +@@ -231,7 +303,14 @@ + p->q931_rxcount = 0; + p->q931_txcount = 0; + #endif +- if (switchtype == PRI_SWITCH_GR303_EOC) { ++ if (dummy_ctrl) { ++ /* Initialize the dummy call reference call record. */ ++ dummy_ctrl->ctrl.dummy_call = &dummy_ctrl->dummy_call; ++ q931_init_call_record(&dummy_ctrl->ctrl, dummy_ctrl->ctrl.dummy_call, ++ Q931_DUMMY_CALL_REFERENCE); ++ } ++ switch (switchtype) { ++ case PRI_SWITCH_GR303_EOC: + p->protodisc = GR303_PROTOCOL_DISCRIMINATOR; + p->sapi = Q921_SAPI_GR303_EOC; + p->tei = Q921_TEI_GR303_EOC_OPS; +@@ -240,7 +319,8 @@ + free(p); + p = NULL; + } +- } else if (switchtype == PRI_SWITCH_GR303_TMC) { ++ break; ++ case PRI_SWITCH_GR303_TMC: + p->protodisc = GR303_PROTOCOL_DISCRIMINATOR; + p->sapi = Q921_SAPI_GR303_TMC_CALLPROC; + p->tei = Q921_TEI_GR303_TMC_CALLPROC; +@@ -249,14 +329,19 @@ + free(p); + p = NULL; + } +- } else if (switchtype == PRI_SWITCH_GR303_TMC_SWITCHING) { ++ break; ++ case PRI_SWITCH_GR303_TMC_SWITCHING: + p->protodisc = GR303_PROTOCOL_DISCRIMINATOR; + p->sapi = Q921_SAPI_GR303_TMC_SWITCHING; + p->tei = Q921_TEI_GR303_TMC_SWITCHING; +- } else if (switchtype == PRI_SWITCH_GR303_EOC_PATH) { ++ break; ++ case PRI_SWITCH_GR303_EOC_PATH: + p->protodisc = GR303_PROTOCOL_DISCRIMINATOR; + p->sapi = Q921_SAPI_GR303_EOC; + p->tei = Q921_TEI_GR303_EOC_PATH; ++ break; ++ default: ++ break; + } + /* Start Q.921 layer, Wait if we're the network */ + if (p) +@@ -327,44 +412,47 @@ + + char *pri_event2str(int id) + { +- switch(id) { +- case PRI_EVENT_DCHAN_UP: +- return "D-Channel Up"; +- case PRI_EVENT_DCHAN_DOWN: +- return "D-channel Down"; +- case PRI_EVENT_RESTART: +- return "Restart channel"; +- case PRI_EVENT_RING: +- return "Ring"; +- case PRI_EVENT_HANGUP: +- return "Hangup"; +- case PRI_EVENT_RINGING: +- return "Ringing"; +- case PRI_EVENT_ANSWER: +- return "Answer"; +- case PRI_EVENT_HANGUP_ACK: +- return "Hangup ACK"; +- case PRI_EVENT_RESTART_ACK: +- return "Restart ACK"; +- case PRI_EVENT_FACNAME: +- return "FacName"; +- case PRI_EVENT_INFO_RECEIVED: +- return "Info Received"; +- case PRI_EVENT_PROCEEDING: +- return "Proceeding"; +- case PRI_EVENT_SETUP_ACK: +- return "Setup ACK"; +- case PRI_EVENT_HANGUP_REQ: +- return "Hangup Req"; +- case PRI_EVENT_NOTIFY: +- return "Notify"; +- case PRI_EVENT_PROGRESS: +- return "Progress"; +- case PRI_EVENT_CONFIG_ERR: +- return "Configuration Error"; +- default: +- return "Unknown Event"; ++ unsigned idx; ++ struct { ++ int id; ++ char *name; ++ } events[] = { ++/* *INDENT-OFF* */ ++ { PRI_EVENT_DCHAN_UP, "D-Channel Up" }, ++ { PRI_EVENT_DCHAN_DOWN, "D-channel Down" }, ++ { PRI_EVENT_RESTART, "Restart channel" }, ++ { PRI_EVENT_CONFIG_ERR, "Configuration Error" }, ++ { PRI_EVENT_RING, "Ring" }, ++ { PRI_EVENT_HANGUP, "Hangup" }, ++ { PRI_EVENT_RINGING, "Ringing" }, ++ { PRI_EVENT_ANSWER, "Answer" }, ++ { PRI_EVENT_HANGUP_ACK, "Hangup ACK" }, ++ { PRI_EVENT_RESTART_ACK, "Restart ACK" }, ++ { PRI_EVENT_FACILITY, "Facility" }, ++ { PRI_EVENT_INFO_RECEIVED, "Info Received" }, ++ { PRI_EVENT_PROCEEDING, "Proceeding" }, ++ { PRI_EVENT_SETUP_ACK, "Setup ACK" }, ++ { PRI_EVENT_HANGUP_REQ, "Hangup Req" }, ++ { PRI_EVENT_NOTIFY, "Notify" }, ++ { PRI_EVENT_PROGRESS, "Progress" }, ++ { PRI_EVENT_KEYPAD_DIGIT, "Keypad Digit" }, ++ { PRI_EVENT_SERVICE, "Service" }, ++ { PRI_EVENT_SERVICE_ACK, "Service ACK" }, ++ { PRI_EVENT_HOLD, "Hold" }, ++ { PRI_EVENT_HOLD_ACK, "Hold Ack" }, ++ { PRI_EVENT_HOLD_REJ, "Hold Rej" }, ++ { PRI_EVENT_RETRIEVE, "Retrieve" }, ++ { PRI_EVENT_RETRIEVE_ACK, "Retrieve ACK" }, ++ { PRI_EVENT_RETRIEVE_REJ, "Retrieve Rej" }, ++/* *INDENT-ON* */ ++ }; ++ ++ for (idx = 0; idx < ARRAY_LEN(events); ++idx) { ++ if (events[idx].id == id) { ++ return events[idx].name; ++ } + } ++ return "Unknown Event"; + } + + pri_event *pri_check_event(struct pri *pri) +@@ -506,7 +594,7 @@ + return q931_information(pri, call, digit); + } + +-int pri_keypad_facility(struct pri *pri, q931_call *call, char *digits) ++int pri_keypad_facility(struct pri *pri, q931_call *call, const char *digits) + { + if (!pri || !call || !digits || !digits[0]) + return -1; +@@ -514,15 +602,6 @@ + return q931_keypad_facility(pri, call, digits); + } + +- +-int pri_callrerouting_facility(struct pri *pri, q931_call *call, const char *dest, const char* original, const char* reason) +-{ +- if (!pri || !call) +- return -1; +- +- return qsig_cf_callrerouting(pri, call, dest, original, reason); +-} +- + int pri_notify(struct pri *pri, q931_call *call, int channel, int info) + { + if (!pri || !call) +@@ -533,7 +612,7 @@ + void pri_destroycall(struct pri *pri, q931_call *call) + { + if (pri && call) +- __q931_destroycall(pri, call); ++ q931_destroycall(pri, call); + return; + } + +@@ -551,6 +630,268 @@ + return q931_connect(pri, call, channel, nonisdn); + } + ++/*! ++ * \internal ++ * \brief Copy the PRI party name to the Q.931 party name structure. ++ * ++ * \param q931_name Q.931 party name structure ++ * \param pri_name PRI party name structure ++ * ++ * \return Nothing ++ */ ++static void pri_copy_party_name_to_q931(struct q931_party_name *q931_name, const struct pri_party_name *pri_name) ++{ ++ q931_party_name_init(q931_name); ++ if (pri_name->valid) { ++ q931_name->valid = 1; ++ q931_name->presentation = pri_name->presentation; ++ q931_name->char_set = pri_name->char_set; ++ libpri_copy_string(q931_name->str, pri_name->str, sizeof(q931_name->str)); ++ } ++} ++ ++/*! ++ * \internal ++ * \brief Copy the PRI party number to the Q.931 party number structure. ++ * ++ * \param q931_number Q.931 party number structure ++ * \param pri_number PRI party number structure ++ * ++ * \return Nothing ++ */ ++static void pri_copy_party_number_to_q931(struct q931_party_number *q931_number, const struct pri_party_number *pri_number) ++{ ++ q931_party_number_init(q931_number); ++ if (pri_number->valid) { ++ q931_number->valid = 1; ++ q931_number->presentation = pri_number->presentation; ++ q931_number->plan = pri_number->plan; ++ libpri_copy_string(q931_number->str, pri_number->str, sizeof(q931_number->str)); ++ } ++} ++ ++/*! ++ * \internal ++ * \brief Copy the PRI party subaddress to the Q.931 party subaddress structure. ++ * ++ * \param q931_subaddress Q.931 party subaddress structure ++ * \param pri_subaddress PRI party subaddress structure ++ * ++ * \return Nothing ++ */ ++static void pri_copy_party_subaddress_to_q931(struct q931_party_subaddress *q931_subaddress, const struct pri_party_subaddress *pri_subaddress) ++{ ++ int length; ++ int maxlen = sizeof(q931_subaddress->data) - 1; ++ ++ q931_party_subaddress_init(q931_subaddress); ++ ++ if (!pri_subaddress->valid) { ++ return; ++ } ++ ++ q931_subaddress->valid = 1; ++ q931_subaddress->type = pri_subaddress->type; ++ ++ length = pri_subaddress->length; ++ if (length > maxlen){ ++ length = maxlen; ++ } else { ++ q931_subaddress->odd_even_indicator = pri_subaddress->odd_even_indicator; ++ } ++ q931_subaddress->length = length; ++ memcpy(q931_subaddress->data, pri_subaddress->data, length); ++ q931_subaddress->data[length] = '\0'; ++} ++ ++/*! ++ * \internal ++ * \brief Copy the PRI party id to the Q.931 party id structure. ++ * ++ * \param q931_id Q.931 party id structure ++ * \param pri_id PRI party id structure ++ * ++ * \return Nothing ++ */ ++static void pri_copy_party_id_to_q931(struct q931_party_id *q931_id, const struct pri_party_id *pri_id) ++{ ++ pri_copy_party_name_to_q931(&q931_id->name, &pri_id->name); ++ pri_copy_party_number_to_q931(&q931_id->number, &pri_id->number); ++ pri_copy_party_subaddress_to_q931(&q931_id->subaddress, &pri_id->subaddress); ++} ++ ++int pri_connected_line_update(struct pri *ctrl, q931_call *call, const struct pri_party_connected_line *connected) ++{ ++ struct q931_party_id party_id; ++ unsigned idx; ++ struct q931_call *subcall; ++ ++ if (!ctrl || !call) { ++ return -1; ++ } ++ ++ pri_copy_party_id_to_q931(&party_id, &connected->id); ++ q931_party_id_fixup(ctrl, &party_id); ++ if (!q931_party_id_cmp(&party_id, &call->local_id)) { ++ /* The local party information did not change so do nothing. */ ++ return 0; ++ } ++ call->local_id = party_id; ++ ++ /* Update all subcalls with new local_id. */ ++ if (call->outboundbroadcast && call->master_call == call) { ++ for (idx = 0; idx < Q931_MAX_TEI; ++idx) { ++ subcall = call->subcalls[idx]; ++ if (subcall) { ++ subcall->local_id = party_id; ++ } ++ } ++ } ++ ++ switch (call->ourcallstate) { ++ case Q931_CALL_STATE_CALL_INITIATED: ++ case Q931_CALL_STATE_OVERLAP_SENDING: ++ case Q931_CALL_STATE_OUTGOING_CALL_PROCEEDING: ++ case Q931_CALL_STATE_CALL_DELIVERED: ++ /* ++ * The local party transferred to someone else before ++ * the remote end answered. ++ */ ++ case Q931_CALL_STATE_ACTIVE: ++ switch (ctrl->switchtype) { ++ case PRI_SWITCH_EUROISDN_E1: ++ case PRI_SWITCH_EUROISDN_T1: ++ if (q931_is_ptmp(ctrl)) { ++ /* PTMP mode */ ++ q931_notify_redirection(ctrl, call, PRI_NOTIFY_TRANSFER_ACTIVE, ++ &call->local_id.number); ++ } else { ++ /* PTP mode */ ++ /* Immediately send EctInform APDU, callStatus=answered(0) */ ++ send_call_transfer_complete(ctrl, call, 0); ++ } ++ break; ++ case PRI_SWITCH_QSIG: ++ /* Immediately send CallTransferComplete APDU, callStatus=answered(0) */ ++ send_call_transfer_complete(ctrl, call, 0); ++ break; ++ default: ++ break; ++ } ++ break; ++ default: ++ /* Just save the data for further developments. */ ++ break; ++ } ++ ++ return 0; ++} ++ ++int pri_redirecting_update(struct pri *ctrl, q931_call *call, const struct pri_party_redirecting *redirecting) ++{ ++ unsigned idx; ++ struct q931_call *subcall; ++ ++ if (!ctrl || !call) { ++ return -1; ++ } ++ ++ /* Save redirecting.to information and reason. */ ++ pri_copy_party_id_to_q931(&call->redirecting.to, &redirecting->to); ++ q931_party_id_fixup(ctrl, &call->redirecting.to); ++ call->redirecting.reason = redirecting->reason; ++ ++ /* ++ * Update all subcalls with new redirecting.to information and reason. ++ * I do not think we will ever have any subcalls when this data is relevant, ++ * but update it just in case. ++ */ ++ if (call->outboundbroadcast && call->master_call == call) { ++ for (idx = 0; idx < Q931_MAX_TEI; ++idx) { ++ subcall = call->subcalls[idx]; ++ if (subcall) { ++ subcall->redirecting.to = call->redirecting.to; ++ subcall->redirecting.reason = redirecting->reason; ++ } ++ } ++ } ++ ++ switch (call->ourcallstate) { ++ case Q931_CALL_STATE_NULL: ++ /* Save the remaining redirecting information before we place a call. */ ++ pri_copy_party_id_to_q931(&call->redirecting.from, &redirecting->from); ++ q931_party_id_fixup(ctrl, &call->redirecting.from); ++ pri_copy_party_id_to_q931(&call->redirecting.orig_called, &redirecting->orig_called); ++ q931_party_id_fixup(ctrl, &call->redirecting.orig_called); ++ call->redirecting.orig_reason = redirecting->orig_reason; ++ if (redirecting->count <= 0) { ++ if (call->redirecting.from.number.valid) { ++ /* ++ * We are redirecting with an unknown count ++ * so assume the count is one. ++ */ ++ call->redirecting.count = 1; ++ } else { ++ call->redirecting.count = 0; ++ } ++ } else if (redirecting->count < PRI_MAX_REDIRECTS) { ++ call->redirecting.count = redirecting->count; ++ } else { ++ call->redirecting.count = PRI_MAX_REDIRECTS; ++ } ++ break; ++ case Q931_CALL_STATE_OVERLAP_RECEIVING: ++ case Q931_CALL_STATE_INCOMING_CALL_PROCEEDING: ++ case Q931_CALL_STATE_CALL_RECEIVED: ++ /* This is an incoming call that has not connected yet. */ ++ if (!call->redirecting.to.number.valid) { ++ /* Not being redirected toward valid number data. Ignore. */ ++ break; ++ } ++ ++ switch (ctrl->switchtype) { ++ case PRI_SWITCH_EUROISDN_E1: ++ case PRI_SWITCH_EUROISDN_T1: ++ if (q931_is_ptmp(ctrl)) { ++ /* PTMP mode */ ++ q931_notify_redirection(ctrl, call, PRI_NOTIFY_CALL_DIVERTING, ++ &call->redirecting.to.number); ++ break; ++ } ++ /* PTP mode - same behaviour as Q.SIG */ ++ /* fall through */ ++ case PRI_SWITCH_QSIG: ++ if (call->redirecting.state != Q931_REDIRECTING_STATE_PENDING_TX_DIV_LEG_3 ++ || strcmp(call->redirecting.to.number.str, call->called.number.str) != 0) { ++ /* immediately send divertingLegInformation1 APDU */ ++ if (rose_diverting_leg_information1_encode(ctrl, call) ++ || q931_facility(ctrl, call)) { ++ pri_message(ctrl, ++ "Could not schedule facility message for divertingLegInfo1\n"); ++ } ++ } ++ call->redirecting.state = Q931_REDIRECTING_STATE_IDLE; ++ ++ /* immediately send divertingLegInformation3 APDU */ ++ if (rose_diverting_leg_information3_encode(ctrl, call, Q931_FACILITY) ++ || q931_facility(ctrl, call)) { ++ pri_message(ctrl, ++ "Could not schedule facility message for divertingLegInfo3\n"); ++ } ++ break; ++ default: ++ break; ++ } ++ break; ++ default: ++ pri_message(ctrl, "Ignored redirecting update because call in state %s(%d).\n", ++ q931_call_state_str(call->ourcallstate), call->ourcallstate); ++ break; ++ } ++ ++ return 0; ++} ++ + #if 0 + /* deprecated routines, use pri_hangup */ + int pri_release(struct pri *pri, q931_call *call, int cause) +@@ -619,7 +960,7 @@ + return -1; + if (cause == -1) + /* normal clear cause */ +- cause = 16; ++ cause = PRI_CAUSE_NORMAL_CLEARING; + return q931_hangup(pri, call, cause); + } + +@@ -630,6 +971,14 @@ + return q931_restart(pri, channel); + } + ++int pri_maintenance_service(struct pri *pri, int span, int channel, int changestatus) ++{ ++ if (!pri) { ++ return -1; ++ } ++ return maintenance_service(pri, span, channel, changestatus); ++} ++ + q931_call *pri_new_call(struct pri *pri) + { + if (!pri) +@@ -637,6 +986,14 @@ + return q931_new_call(pri); + } + ++int pri_is_dummy_call(q931_call *call) ++{ ++ if (!call) { ++ return 0; ++ } ++ return q931_is_dummy_call(call); ++} ++ + void pri_dump_event(struct pri *pri, pri_event *e) + { + if (!pri || !e) +@@ -667,7 +1024,10 @@ + static void pri_sr_init(struct pri_sr *req) + { + memset(req, 0, sizeof(struct pri_sr)); +- ++ q931_party_redirecting_init(&req->redirecting); ++ q931_party_id_init(&req->caller); ++ q931_party_address_init(&req->called); ++ req->reversecharge = PRI_REVERSECHARGE_NONE; + } + + int pri_sr_set_connection_call_independent(struct pri_sr *req) +@@ -675,10 +1035,21 @@ + if (!req) + return -1; + +- req->justsignalling = 1; /* have to set justsignalling for all those pesky IEs we need to setup */ ++ req->cis_call = 1; /* have to set cis_call for all those pesky IEs we need to setup */ ++ req->cis_auto_disconnect = 1; + return 0; + } + ++int pri_sr_set_no_channel_call(struct pri_sr *req) ++{ ++ if (!req) { ++ return -1; ++ } ++ ++ req->cis_call = 1; ++ return 0; ++} ++ + /* Don't call any other pri functions on this */ + int pri_mwi_activate(struct pri *pri, q931_call *c, char *caller, int callerplan, char *callername, int callerpres, char *called, + int calledplan) +@@ -689,14 +1060,9 @@ + + pri_sr_init(&req); + pri_sr_set_connection_call_independent(&req); ++ pri_sr_set_caller(&req, caller, callername, callerplan, callerpres); ++ pri_sr_set_called(&req, called, calledplan, 0); + +- req.caller = caller; +- req.callerplan = callerplan; +- req.callername = callername; +- req.callerpres = callerpres; +- req.called = called; +- req.calledplan = calledplan; +- + if (mwi_message_send(pri, c, &req, 1) < 0) { + pri_message(pri, "Unable to send MWI activate message\n"); + return -1; +@@ -714,14 +1080,9 @@ + + pri_sr_init(&req); + pri_sr_set_connection_call_independent(&req); ++ pri_sr_set_caller(&req, caller, callername, callerplan, callerpres); ++ pri_sr_set_called(&req, called, calledplan, 0); + +- req.caller = caller; +- req.callerplan = callerplan; +- req.callername = callername; +- req.callerpres = callerpres; +- req.called = called; +- req.calledplan = calledplan; +- + if(mwi_message_send(pri, c, &req, 0) < 0) { + pri_message(pri, "Unable to send MWI deactivate message\n"); + return -1; +@@ -740,22 +1101,18 @@ + + int pri_call(struct pri *pri, q931_call *c, int transmode, int channel, int exclusive, + int nonisdn, char *caller, int callerplan, char *callername, int callerpres, char *called, +- int calledplan,int ulayer1) ++ int calledplan, int ulayer1) + { + struct pri_sr req; + if (!pri || !c) + return -1; + pri_sr_init(&req); ++ pri_sr_set_caller(&req, caller, callername, callerplan, callerpres); ++ pri_sr_set_called(&req, called, calledplan, 0); + req.transmode = transmode; + req.channel = channel; + req.exclusive = exclusive; + req.nonisdn = nonisdn; +- req.caller = caller; +- req.callerplan = callerplan; +- req.callername = callername; +- req.callerpres = callerpres; +- req.called = called; +- req.calledplan = calledplan; + req.userl1 = ulayer1; + return q931_setup(pri, c, &req); + } +@@ -773,28 +1130,85 @@ + __pri_error = func; + } + +-void pri_message(struct pri *pri, char *fmt, ...) ++static void pri_old_message(struct pri *ctrl, const char *fmt, va_list *ap) + { + char tmp[1024]; +- va_list ap; +- va_start(ap, fmt); +- vsnprintf(tmp, sizeof(tmp), fmt, ap); +- va_end(ap); ++ ++ vsnprintf(tmp, sizeof(tmp), fmt, *ap); + if (__pri_message) +- __pri_message(pri, tmp); ++ __pri_message(ctrl, tmp); + else + fputs(tmp, stdout); + } + +-void pri_error(struct pri *pri, char *fmt, ...) ++void pri_message(struct pri *ctrl, const char *fmt, ...) + { ++ int added_length; ++ va_list ap; ++ ++ ctrl = PRI_MASTER(ctrl); ++ if (!ctrl || !ctrl->msg_line) { ++ /* Just have to do it the old way. */ ++ va_start(ap, fmt); ++ pri_old_message(ctrl, fmt, &ap); ++ va_end(ap); ++ return; ++ } ++ ++ va_start(ap, fmt); ++ added_length = vsnprintf(ctrl->msg_line->str + ctrl->msg_line->length, ++ sizeof(ctrl->msg_line->str) - ctrl->msg_line->length, fmt, ap); ++ va_end(ap); ++ if (added_length < 0 ++ || sizeof(ctrl->msg_line->str) <= ctrl->msg_line->length + added_length) { ++ static char truncated_output[] = ++ "v-- Error building output or output was truncated. (Next line) --v\n"; ++ ++ /* ++ * This clause should never need to run because the ++ * output line accumulation buffer is quite large. ++ */ ++ ++ /* vsnprintf() error or output string was truncated. */ ++ if (__pri_message) { ++ __pri_message(ctrl, truncated_output); ++ } else { ++ fputs(truncated_output, stdout); ++ } ++ ++ /* Add a terminating '\n' to force a flush of the line. */ ++ ctrl->msg_line->length = strlen(ctrl->msg_line->str); ++ if (ctrl->msg_line->length) { ++ ctrl->msg_line->str[ctrl->msg_line->length - 1] = '\n'; ++ } else { ++ ctrl->msg_line->str[0] = '\n'; ++ ctrl->msg_line->str[1] = '\0'; ++ } ++ } else { ++ ctrl->msg_line->length += added_length; ++ } ++ ++ if (ctrl->msg_line->length ++ && ctrl->msg_line->str[ctrl->msg_line->length - 1] == '\n') { ++ /* The accumulated output line was terminated so send it out. */ ++ ctrl->msg_line->length = 0; ++ if (__pri_message) { ++ __pri_message(ctrl, ctrl->msg_line->str); ++ } else { ++ fputs(ctrl->msg_line->str, stdout); ++ } ++ } ++} ++ ++void pri_error(struct pri *pri, const char *fmt, ...) ++{ + char tmp[1024]; + va_list ap; + va_start(ap, fmt); + vsnprintf(tmp, sizeof(tmp), fmt, ap); + va_end(ap); + if (__pri_error) +- __pri_error(pri, tmp); ++ __pri_error(PRI_MASTER(pri), tmp); + else + fputs(tmp, stderr); + } +@@ -821,49 +1235,102 @@ + return pri->fd; + } + +-char *pri_dump_info_str(struct pri *pri) ++/*! ++ * \internal ++ * \brief Append snprintf output to the given buffer. ++ * ++ * \param buf Buffer currently filling. ++ * \param buf_used Offset into buffer where to put new stuff. ++ * \param buf_size Actual buffer size of buf. ++ * \param format printf format string. ++ * ++ * \return Total buffer space used. ++ */ ++static size_t pri_snprintf(char *buf, size_t buf_used, size_t buf_size, const char *format, ...) __attribute__((format(printf, 4, 5))); ++static size_t pri_snprintf(char *buf, size_t buf_used, size_t buf_size, const char *format, ...) + { +- char buf[4096]; +- int len = 0; ++ va_list args; ++ ++ if (buf_used < buf_size) { ++ va_start(args, format); ++ buf_used += vsnprintf(buf + buf_used, buf_size - buf_used, format, args); ++ va_end(args); ++ } ++ if (buf_size < buf_used) { ++ buf_used = buf_size + 1; ++ } ++ return buf_used; ++} ++ ++char *pri_dump_info_str(struct pri *ctrl) ++{ ++ char *buf; ++ size_t buf_size; ++ size_t used; + #ifdef LIBPRI_COUNTERS + struct q921_frame *f; +- int q921outstanding = 0; ++ unsigned q921outstanding; + #endif +- if (!pri) ++ unsigned idx; ++ unsigned long switch_bit; ++ ++ if (!ctrl) { + return NULL; ++ } + ++ buf_size = 4096; /* This should be bigger than we will ever need. */ ++ buf = malloc(buf_size); ++ if (!buf) { ++ return NULL; ++ } ++ + /* Might be nice to format these a little better */ +- len += sprintf(buf + len, "Switchtype: %s\n", pri_switch2str(pri->switchtype)); +- len += sprintf(buf + len, "Type: %s\n", pri_node2str(pri->localtype)); ++ used = 0; ++ used = pri_snprintf(buf, used, buf_size, "Switchtype: %s\n", ++ pri_switch2str(ctrl->switchtype)); ++ used = pri_snprintf(buf, used, buf_size, "Type: %s\n", pri_node2str(ctrl->localtype)); + #ifdef LIBPRI_COUNTERS + /* Remember that Q921 Counters include Q931 packets (and any retransmissions) */ +- len += sprintf(buf + len, "Q931 RX: %d\n", pri->q931_rxcount); +- len += sprintf(buf + len, "Q931 TX: %d\n", pri->q931_txcount); +- len += sprintf(buf + len, "Q921 RX: %d\n", pri->q921_rxcount); +- len += sprintf(buf + len, "Q921 TX: %d\n", pri->q921_txcount); +- f = pri->txqueue; ++ used = pri_snprintf(buf, used, buf_size, "Q931 RX: %d\n", ctrl->q931_rxcount); ++ used = pri_snprintf(buf, used, buf_size, "Q931 TX: %d\n", ctrl->q931_txcount); ++ used = pri_snprintf(buf, used, buf_size, "Q921 RX: %d\n", ctrl->q921_rxcount); ++ used = pri_snprintf(buf, used, buf_size, "Q921 TX: %d\n", ctrl->q921_txcount); ++ q921outstanding = 0; ++ f = ctrl->txqueue; + while (f) { + q921outstanding++; + f = f->next; + } +- len += sprintf(buf + len, "Q921 Outstanding: %d\n", q921outstanding); ++ used = pri_snprintf(buf, used, buf_size, "Q921 Outstanding: %u\n", q921outstanding); + #endif +- len += sprintf(buf + len, "Window Length: %d/%d\n", pri->windowlen, pri->window); +- len += sprintf(buf + len, "Sentrej: %d\n", pri->sentrej); +- len += sprintf(buf + len, "SolicitFbit: %d\n", pri->solicitfbit); +- len += sprintf(buf + len, "Retrans: %d\n", pri->retrans); +- len += sprintf(buf + len, "Busy: %d\n", pri->busy); +- len += sprintf(buf + len, "Overlap Dial: %d\n", pri->overlapdial); +- len += sprintf(buf + len, "Logical Channel Mapping: %d\n", pri->chan_mapping_logical); +- len += sprintf(buf + len, "T200 Timer: %d\n", pri->timers[PRI_TIMER_T200]); +- len += sprintf(buf + len, "T203 Timer: %d\n", pri->timers[PRI_TIMER_T203]); +- len += sprintf(buf + len, "T305 Timer: %d\n", pri->timers[PRI_TIMER_T305]); +- len += sprintf(buf + len, "T308 Timer: %d\n", pri->timers[PRI_TIMER_T308]); +- len += sprintf(buf + len, "T309 Timer: %d\n", pri->timers[PRI_TIMER_T309]); +- len += sprintf(buf + len, "T313 Timer: %d\n", pri->timers[PRI_TIMER_T313]); +- len += sprintf(buf + len, "N200 Counter: %d\n", pri->timers[PRI_TIMER_N200]); ++ used = pri_snprintf(buf, used, buf_size, "Window Length: %d/%d\n", ctrl->windowlen, ++ ctrl->window); ++ used = pri_snprintf(buf, used, buf_size, "Sentrej: %d\n", ctrl->sentrej); ++ used = pri_snprintf(buf, used, buf_size, "SolicitFbit: %d\n", ctrl->solicitfbit); ++ used = pri_snprintf(buf, used, buf_size, "Retrans: %d\n", ctrl->retrans); ++ used = pri_snprintf(buf, used, buf_size, "Busy: %d\n", ctrl->busy); ++ used = pri_snprintf(buf, used, buf_size, "Overlap Dial: %d\n", ctrl->overlapdial); ++ used = pri_snprintf(buf, used, buf_size, "Logical Channel Mapping: %d\n", ++ ctrl->chan_mapping_logical); ++ used = pri_snprintf(buf, used, buf_size, "Timer and counter settings:\n"); ++ switch_bit = PRI_BIT(ctrl->switchtype); ++ for (idx = 0; idx < ARRAY_LEN(pri_timer); ++idx) { ++ if (pri_timer[idx].used_by & switch_bit) { ++ enum PRI_TIMERS_AND_COUNTERS tmr; + +- return strdup(buf); ++ tmr = pri_timer[idx].number; ++ if (0 <= ctrl->timers[tmr] || tmr == PRI_TIMER_T309) { ++ used = pri_snprintf(buf, used, buf_size, " %s: %d\n", ++ pri_timer[idx].name, ctrl->timers[tmr]); ++ } ++ } ++ } ++ ++ if (buf_size < used) { ++ pri_message(ctrl, ++ "pri_dump_info_str(): Produced output exceeded buffer capacity. (Truncated)\n"); ++ } ++ return buf; + } + + int pri_get_crv(struct pri *pri, q931_call *call, int *callmode) +@@ -913,26 +1380,213 @@ + + int pri_sr_set_called(struct pri_sr *sr, char *called, int calledplan, int numcomplete) + { +- sr->called = called; +- sr->calledplan = calledplan; ++ q931_party_address_init(&sr->called); ++ if (called) { ++ sr->called.number.valid = 1; ++ sr->called.number.plan = calledplan; ++ libpri_copy_string(sr->called.number.str, called, sizeof(sr->called.number.str)); ++ } + sr->numcomplete = numcomplete; + return 0; + } + ++void pri_sr_set_called_subaddress(struct pri_sr *sr, const struct pri_party_subaddress *subaddress) ++{ ++ pri_copy_party_subaddress_to_q931(&sr->called.subaddress, subaddress); ++} ++ + int pri_sr_set_caller(struct pri_sr *sr, char *caller, char *callername, int callerplan, int callerpres) + { +- sr->caller = caller; +- sr->callername = callername; +- sr->callerplan = callerplan; +- sr->callerpres = callerpres; ++ q931_party_id_init(&sr->caller); ++ if (caller) { ++ sr->caller.number.valid = 1; ++ sr->caller.number.presentation = callerpres; ++ sr->caller.number.plan = callerplan; ++ libpri_copy_string(sr->caller.number.str, caller, sizeof(sr->caller.number.str)); ++ ++ if (callername) { ++ sr->caller.name.valid = 1; ++ sr->caller.name.presentation = callerpres; ++ sr->caller.name.char_set = PRI_CHAR_SET_ISO8859_1; ++ libpri_copy_string(sr->caller.name.str, callername, ++ sizeof(sr->caller.name.str)); ++ } ++ } + return 0; + } + ++void pri_sr_set_caller_subaddress(struct pri_sr *sr, const struct pri_party_subaddress *subaddress) ++{ ++ pri_copy_party_subaddress_to_q931(&sr->caller.subaddress, subaddress); ++} ++ ++void pri_sr_set_caller_party(struct pri_sr *sr, const struct pri_party_id *caller) ++{ ++ pri_copy_party_id_to_q931(&sr->caller, caller); ++} ++ + int pri_sr_set_redirecting(struct pri_sr *sr, char *num, int plan, int pres, int reason) + { +- sr->redirectingnum = num; +- sr->redirectingplan = plan; +- sr->redirectingpres = pres; +- sr->redirectingreason = reason; ++ q931_party_redirecting_init(&sr->redirecting); ++ if (num && num[0]) { ++ sr->redirecting.from.number.valid = 1; ++ sr->redirecting.from.number.presentation = pres; ++ sr->redirecting.from.number.plan = plan; ++ libpri_copy_string(sr->redirecting.from.number.str, num, ++ sizeof(sr->redirecting.from.number.str)); ++ ++ sr->redirecting.count = 1; ++ sr->redirecting.reason = reason; ++ } + return 0; + } ++ ++void pri_sr_set_redirecting_parties(struct pri_sr *sr, const struct pri_party_redirecting *redirecting) ++{ ++ pri_copy_party_id_to_q931(&sr->redirecting.from, &redirecting->from); ++ pri_copy_party_id_to_q931(&sr->redirecting.to, &redirecting->to); ++ pri_copy_party_id_to_q931(&sr->redirecting.orig_called, &redirecting->orig_called); ++ sr->redirecting.orig_reason = redirecting->orig_reason; ++ sr->redirecting.reason = redirecting->reason; ++ if (redirecting->count <= 0) { ++ if (sr->redirecting.from.number.valid) { ++ /* ++ * We are redirecting with an unknown count ++ * so assume the count is one. ++ */ ++ sr->redirecting.count = 1; ++ } else { ++ sr->redirecting.count = 0; ++ } ++ } else if (redirecting->count < PRI_MAX_REDIRECTS) { ++ sr->redirecting.count = redirecting->count; ++ } else { ++ sr->redirecting.count = PRI_MAX_REDIRECTS; ++ } ++} ++ ++void pri_sr_set_reversecharge(struct pri_sr *sr, int requested) ++{ ++ sr->reversecharge = requested; ++} ++ ++void pri_sr_set_keypad_digits(struct pri_sr *sr, const char *keypad_digits) ++{ ++ sr->keypad_digits = keypad_digits; ++} ++ ++void pri_hold_enable(struct pri *ctrl, int enable) ++{ ++ ctrl = PRI_MASTER(ctrl); ++ if (ctrl) { ++ ctrl->hold_support = enable ? 1 : 0; ++ } ++} ++ ++int pri_hold(struct pri *ctrl, q931_call *call) ++{ ++ if (!ctrl || !call) { ++ return -1; ++ } ++ return q931_send_hold(ctrl, call); ++} ++ ++int pri_hold_ack(struct pri *ctrl, q931_call *call) ++{ ++ if (!ctrl || !call) { ++ return -1; ++ } ++ return q931_send_hold_ack(ctrl, call); ++} ++ ++int pri_hold_rej(struct pri *ctrl, q931_call *call, int cause) ++{ ++ if (!ctrl || !call) { ++ return -1; ++ } ++ return q931_send_hold_rej(ctrl, call, cause); ++} ++ ++int pri_retrieve(struct pri *ctrl, q931_call *call, int channel) ++{ ++ if (!ctrl || !call) { ++ return -1; ++ } ++ return q931_send_retrieve(ctrl, call, channel); ++} ++ ++int pri_retrieve_ack(struct pri *ctrl, q931_call *call, int channel) ++{ ++ if (!ctrl || !call) { ++ return -1; ++ } ++ return q931_send_retrieve_ack(ctrl, call, channel); ++} ++ ++int pri_retrieve_rej(struct pri *ctrl, q931_call *call, int cause) ++{ ++ if (!ctrl || !call) { ++ return -1; ++ } ++ return q931_send_retrieve_rej(ctrl, call, cause); ++} ++ ++int pri_callrerouting_facility(struct pri *pri, q931_call *call, const char *dest, const char* original, const char* reason) ++{ ++ if (!pri || !call || !dest) ++ return -1; ++ ++ return qsig_cf_callrerouting(pri, call, dest, original, reason); ++} ++ ++void pri_reroute_enable(struct pri *ctrl, int enable) ++{ ++ ctrl = PRI_MASTER(ctrl); ++ if (ctrl) { ++ ctrl->deflection_support = enable ? 1 : 0; ++ } ++} ++ ++int pri_reroute_call(struct pri *ctrl, q931_call *call, const struct pri_party_id *caller, const struct pri_party_redirecting *deflection, int subscription_option) ++{ ++ const struct q931_party_id *caller_id; ++ struct q931_party_id local_caller; ++ struct q931_party_redirecting reroute; ++ ++ if (!ctrl || !call || !deflection) { ++ return -1; ++ } ++ ++ if (caller) { ++ /* Convert the caller update information. */ ++ pri_copy_party_id_to_q931(&local_caller, caller); ++ q931_party_id_fixup(ctrl, &local_caller); ++ caller_id = &local_caller; ++ } else { ++ caller_id = NULL; ++ } ++ ++ /* Convert the deflection information. */ ++ q931_party_redirecting_init(&reroute); ++ pri_copy_party_id_to_q931(&reroute.from, &deflection->from); ++ q931_party_id_fixup(ctrl, &reroute.from); ++ pri_copy_party_id_to_q931(&reroute.to, &deflection->to); ++ q931_party_id_fixup(ctrl, &reroute.to); ++ pri_copy_party_id_to_q931(&reroute.orig_called, &deflection->orig_called); ++ q931_party_id_fixup(ctrl, &reroute.orig_called); ++ reroute.reason = deflection->reason; ++ reroute.orig_reason = deflection->orig_reason; ++ if (deflection->count <= 0) { ++ /* ++ * We are deflecting with an unknown count ++ * so assume the count is one. ++ */ ++ reroute.count = 1; ++ } else if (deflection->count < PRI_MAX_REDIRECTS) { ++ reroute.count = deflection->count; ++ } else { ++ reroute.count = PRI_MAX_REDIRECTS; ++ } ++ ++ return send_reroute_request(ctrl, call, caller_id, &reroute, subscription_option); ++} +Index: Makefile +=================================================================== +--- a/Makefile (.../tags/1.4.10.2) (revision 1357) ++++ b/Makefile (.../branches/1.4) (revision 1357) +@@ -41,15 +41,55 @@ + + STATIC_LIBRARY=libpri.a + DYNAMIC_LIBRARY:=libpri.so.$(SONAME) +-STATIC_OBJS=copy_string.o pri.o q921.o prisched.o q931.o pri_facility.o version.o +-DYNAMIC_OBJS=copy_string.lo pri.lo q921.lo prisched.lo q931.lo pri_facility.lo version.lo +-CFLAGS=-Wall -Werror -Wstrict-prototypes -Wmissing-prototypes -g -fPIC $(ALERTING) $(LIBPRI_COUNTERS) ++STATIC_OBJS= \ ++ copy_string.o \ ++ pri.o \ ++ q921.o \ ++ prisched.o \ ++ q931.o \ ++ pri_facility.o \ ++ asn1_primitive.o \ ++ rose.o \ ++ rose_address.o \ ++ rose_etsi_aoc.o \ ++ rose_etsi_diversion.o \ ++ rose_etsi_ect.o \ ++ rose_other.o \ ++ rose_q931.o \ ++ rose_qsig_aoc.o \ ++ rose_qsig_ct.o \ ++ rose_qsig_diversion.o \ ++ rose_qsig_mwi.o \ ++ rose_qsig_name.o \ ++ version.o ++DYNAMIC_OBJS= \ ++ copy_string.lo \ ++ pri.lo \ ++ q921.lo \ ++ prisched.lo \ ++ q931.lo \ ++ pri_facility.lo \ ++ asn1_primitive.lo \ ++ rose.lo \ ++ rose_address.lo \ ++ rose_etsi_aoc.lo \ ++ rose_etsi_diversion.lo \ ++ rose_etsi_ect.lo \ ++ rose_other.lo \ ++ rose_q931.lo \ ++ rose_qsig_aoc.lo \ ++ rose_qsig_ct.lo \ ++ rose_qsig_diversion.lo \ ++ rose_qsig_mwi.lo \ ++ rose_qsig_name.lo \ ++ version.lo ++CFLAGS+=-Wall -Werror -Wstrict-prototypes -Wmissing-prototypes -g -fPIC $(ALERTING) $(LIBPRI_COUNTERS) $(LIBPRI_OPT) + INSTALL_PREFIX=$(DESTDIR) + INSTALL_BASE=/usr + libdir?=$(INSTALL_BASE)/lib + SOFLAGS:=-Wl,-h$(DYNAMIC_LIBRARY) + LDCONFIG = /sbin/ldconfig +-ifneq (,$(findstring X$(OSARCH)X, XLinuxX XGNU/kFreeBSDX)) ++ifneq (,$(findstring X$(OSARCH)X, XLinuxX XGNU/kFreeBSDX XGNUX)) + LDCONFIG_FLAGS=-n + else + ifeq (${OSARCH},FreeBSD) +@@ -74,7 +114,9 @@ + #A ultrasparc cpu is really v9 but the stock debian stable 3.0 gcc doesnt support it. + ifeq ($(PROC),sparc64) + PROC=ultrasparc +-CFLAGS += -mtune=$(PROC) -O3 -pipe -fomit-frame-pointer -mcpu=v8 ++LIBPRI_OPT = -mtune=$(PROC) -O3 -pipe -fomit-frame-pointer -mcpu=v8 ++else ++LIBPRI_OPT = -O2 + endif + + all: $(STATIC_LIBRARY) $(DYNAMIC_LIBRARY) +@@ -132,6 +174,9 @@ + pridump: pridump.o + $(CC) -o pridump pridump.o -L. -lpri $(CFLAGS) + ++rosetest: rosetest.o ++ $(CC) -o rosetest rosetest.o -L. -lpri $(CFLAGS) ++ + MAKE_DEPS= -MD -MT $@ -MF .$(subst /,_,$@).d -MP + + %.o: %.c +Index: q931.c +=================================================================== +--- a/q931.c (.../tags/1.4.10.2) (revision 1357) ++++ b/q931.c (.../branches/1.4) (revision 1357) +@@ -33,6 +33,7 @@ + #include "pri_q921.h" + #include "pri_q931.h" + #include "pri_facility.h" ++#include "rose.h" + + #include <unistd.h> + #include <stdlib.h> +@@ -67,7 +68,7 @@ + { Q931_RESTART_ACKNOWLEDGE, "RESTART ACKNOWLEDGE", { Q931_RESTART_INDICATOR } }, + + /* Miscellaneous */ +- { Q931_STATUS, "STATUS", { Q931_CAUSE, Q931_CALL_STATE } }, ++ { Q931_STATUS, "STATUS", { Q931_CAUSE, Q931_IE_CALL_STATE } }, + { Q931_STATUS_ENQUIRY, "STATUS ENQUIRY" }, + { Q931_USER_INFORMATION, "USER_INFORMATION" }, + { Q931_SEGMENT, "SEGMENT" }, +@@ -79,22 +80,32 @@ + /* Call Management */ + { Q931_HOLD, "HOLD" }, + { Q931_HOLD_ACKNOWLEDGE, "HOLD ACKNOWLEDGE" }, +- { Q931_HOLD_REJECT, "HOLD REJECT" }, ++ { Q931_HOLD_REJECT, "HOLD REJECT", { Q931_CAUSE } }, + { Q931_RETRIEVE, "RETRIEVE" }, + { Q931_RETRIEVE_ACKNOWLEDGE, "RETRIEVE ACKNOWLEDGE" }, +- { Q931_RETRIEVE_REJECT, "RETRIEVE REJECT" }, ++ { Q931_RETRIEVE_REJECT, "RETRIEVE REJECT", { Q931_CAUSE } }, + { Q931_RESUME, "RESUME" }, + { Q931_RESUME_ACKNOWLEDGE, "RESUME ACKNOWLEDGE", { Q931_CHANNEL_IDENT } }, + { Q931_RESUME_REJECT, "RESUME REJECT", { Q931_CAUSE } }, + { Q931_SUSPEND, "SUSPEND" }, + { Q931_SUSPEND_ACKNOWLEDGE, "SUSPEND ACKNOWLEDGE" }, + { Q931_SUSPEND_REJECT, "SUSPEND REJECT" }, ++}; + +- /* Maintenance */ +- { NATIONAL_SERVICE, "SERVICE" }, +- { NATIONAL_SERVICE_ACKNOWLEDGE, "SERVICE ACKNOWLEDGE" }, ++static int post_handle_q931_message(struct pri *ctrl, struct q931_mh *mh, struct q931_call *c, int missingmand); ++static void nt_ptmp_handle_q931_message(struct pri *ctrl, struct q931_mh *mh, struct q931_call *c, int *allow_event, int *allow_posthandle); ++ ++struct msgtype att_maintenance_msgs[] = { ++ { ATT_SERVICE, "SERVICE", { Q931_CHANNEL_IDENT } }, ++ { ATT_SERVICE_ACKNOWLEDGE, "SERVICE ACKNOWLEDGE", { Q931_CHANNEL_IDENT } }, + }; + ++struct msgtype national_maintenance_msgs[] = { ++ { NATIONAL_SERVICE, "SERVICE", { Q931_CHANNEL_IDENT } }, ++ { NATIONAL_SERVICE_ACKNOWLEDGE, "SERVICE ACKNOWLEDGE", { Q931_CHANNEL_IDENT } }, ++}; ++static int post_handle_maintenance_message(struct pri *ctrl, int protodisc, struct q931_mh *mh, struct q931_call *c); ++ + static struct msgtype causes[] = { + { PRI_CAUSE_UNALLOCATED, "Unallocated (unassigned) number" }, + { PRI_CAUSE_NO_ROUTE_TRANSIT_NET, "No route to specified transmit network" }, +@@ -107,6 +118,7 @@ + { PRI_CAUSE_NO_ANSWER, "User alerting, no answer" }, + { PRI_CAUSE_CALL_REJECTED, "Call Rejected" }, + { PRI_CAUSE_NUMBER_CHANGED, "Number changed" }, ++ { PRI_CAUSE_NONSELECTED_USER_CLEARING, "Non-selected user clearing" }, + { PRI_CAUSE_DESTINATION_OUT_OF_ORDER, "Destination out of order" }, + { PRI_CAUSE_INVALID_NUMBER_FORMAT, "Invalid number format" }, + { PRI_CAUSE_FACILITY_REJECTED, "Facility rejected" }, +@@ -119,13 +131,14 @@ + { PRI_CAUSE_ACCESS_INFO_DISCARDED, "Access information discarded" }, + { PRI_CAUSE_REQUESTED_CHAN_UNAVAIL, "Requested channel not available" }, + { PRI_CAUSE_PRE_EMPTED, "Pre-empted" }, ++ { PRI_CAUSE_RESOURCE_UNAVAIL_UNSPECIFIED, "Resource unavailable, unspecified" }, + { PRI_CAUSE_FACILITY_NOT_SUBSCRIBED, "Facility not subscribed" }, + { PRI_CAUSE_OUTGOING_CALL_BARRED, "Outgoing call barred" }, + { PRI_CAUSE_INCOMING_CALL_BARRED, "Incoming call barred" }, + { PRI_CAUSE_BEARERCAPABILITY_NOTAUTH, "Bearer capability not authorized" }, + { PRI_CAUSE_BEARERCAPABILITY_NOTAVAIL, "Bearer capability not available" }, ++ { PRI_CAUSE_SERVICEOROPTION_NOTAVAIL, "Service or option not available, unspecified" }, + { PRI_CAUSE_BEARERCAPABILITY_NOTIMPL, "Bearer capability not implemented" }, +- { PRI_CAUSE_SERVICEOROPTION_NOTAVAIL, "Service or option not available, unspecified" }, + { PRI_CAUSE_CHAN_NOT_IMPLEMENTED, "Channel not implemented" }, + { PRI_CAUSE_FACILITY_NOT_IMPLEMENTED, "Facility not implemented" }, + { PRI_CAUSE_INVALID_CALL_REFERENCE, "Invalid call reference value" }, +@@ -163,8 +176,9 @@ + { PRI_NSF_CALL_REDIRECTION_SERVICE, "Call Redirection Service" } + }; + +-#define FLAG_PREFERRED 2 +-#define FLAG_EXCLUSIVE 4 ++#define FLAG_WHOLE_INTERFACE 0x01 ++#define FLAG_PREFERRED 0x02 ++#define FLAG_EXCLUSIVE 0x04 + + #define RESET_INDICATOR_CHANNEL 0 + #define RESET_INDICATOR_DS1 6 +@@ -214,26 +228,49 @@ + #define LOC_NETWORK_BEYOND_INTERWORKING 0xa + + static char *ie2str(int ie); +-static char *msg2str(int msg); + + +-#define FUNC_DUMP(name) void ((name))(int full_ie, struct pri *pri, q931_ie *ie, int len, char prefix) +-#define FUNC_RECV(name) int ((name))(int full_ie, struct pri *pri, q931_call *call, int msgtype, q931_ie *ie, int len) +-#define FUNC_SEND(name) int ((name))(int full_ie, struct pri *pri, q931_call *call, int msgtype, q931_ie *ie, int len, int order) ++#define FUNC_DUMP(name) void (name)(int full_ie, struct pri *pri, q931_ie *ie, int len, char prefix) ++#define FUNC_RECV(name) int (name)(int full_ie, struct pri *pri, q931_call *call, int msgtype, q931_ie *ie, int len) ++#define FUNC_SEND(name) int (name)(int full_ie, struct pri *pri, q931_call *call, int msgtype, q931_ie *ie, int len, int order) + + #if 1 + /* Update call state with transition trace. */ +-#define UPDATE_OURCALLSTATE(pri,c,newstate) do {\ +- if (pri->debug & (PRI_DEBUG_Q931_STATE) && c->ourcallstate != newstate) \ +- pri_message(pri, DBGHEAD "call %d on channel %d enters state %d (%s)\n", DBGINFO, \ +- c->cr, c->channelno, newstate, callstate2str(newstate)); \ +- c->ourcallstate = newstate; \ ++#define UPDATE_OURCALLSTATE(ctrl, call, newstate) \ ++ do { \ ++ if (((ctrl)->debug & PRI_DEBUG_Q931_STATE) && (call)->ourcallstate != (newstate)) { \ ++ pri_message((ctrl), \ ++ DBGHEAD "%s %d enters state %d (%s). Hold state: %s\n", \ ++ DBGINFO, ((call) == (call)->master_call) ? "Call" : "Subcall", \ ++ (call)->cr, (newstate), q931_call_state_str(newstate), \ ++ q931_hold_state_str((call)->master_call->hold_state)); \ ++ } \ ++ (call)->ourcallstate = (newstate); \ + } while (0) + #else + /* Update call state with no trace. */ +-#define UPDATE_OURCALLSTATE(pri,c,newstate) c->ourcallstate = newstate ++#define UPDATE_OURCALLSTATE(ctrl, call, newstate) (call)->ourcallstate = (newstate) + #endif + ++#if 1 ++/* Update hold state with transition trace. */ ++#define UPDATE_HOLD_STATE(ctrl, master_call, newstate) \ ++ do { \ ++ if (((ctrl)->debug & PRI_DEBUG_Q931_STATE) \ ++ && (master_call)->hold_state != (newstate)) { \ ++ pri_message((ctrl), \ ++ DBGHEAD "Call %d in state %d (%s) enters Hold state: %s\n", \ ++ DBGINFO, (master_call)->cr, (master_call)->ourcallstate, \ ++ q931_call_state_str((master_call)->ourcallstate), \ ++ q931_hold_state_str(newstate)); \ ++ } \ ++ (master_call)->hold_state = (newstate); \ ++ } while (0) ++#else ++/* Update hold state with no trace. */ ++#define UPDATE_HOLD_STATE(ctrl, master_call, newstate) (master_call)->hold_state = (newstate) ++#endif ++ + struct ie { + /* Maximal count of same IEs at the message (0 - any, 1..n - limited) */ + int max_count; +@@ -249,6 +286,506 @@ + FUNC_SEND(*transmit); + }; + ++/*! ++ * \internal ++ * \brief Encode the channel id information to pass to upper level. ++ * ++ * \param call Q.931 call leg ++ * ++ * \return Encoded channel value. ++ */ ++static int q931_encode_channel(const q931_call *call) ++{ ++ int held_call; ++ int channelno; ++ int ds1no; ++ ++ switch (call->master_call->hold_state) { ++ case Q931_HOLD_STATE_CALL_HELD: ++ case Q931_HOLD_STATE_RETRIEVE_REQ: ++ case Q931_HOLD_STATE_RETRIEVE_IND: ++ held_call = 1 << 18; ++ ++ /* So a -1 does not wipe out the held_call flag. */ ++ channelno = call->channelno & 0xFF; ++ ds1no = call->ds1no & 0xFF; ++ break; ++ default: ++ held_call = 0; ++ channelno = call->channelno; ++ ds1no = call->ds1no; ++ break; ++ } ++ return channelno | (ds1no << 8) | (call->ds1explicit << 16) | (call->cis_call << 17) ++ | held_call; ++} ++ ++/*! ++ * \brief Determine if layer 2 is in PTMP mode. ++ * ++ * \param ctrl D channel controller. ++ * ++ * \retval TRUE if in PTMP mode. ++ * \retval FALSE otherwise. ++ */ ++int q931_is_ptmp(const struct pri *ctrl) ++{ ++ /* Check master control structure */ ++ for (; ctrl->master; ctrl = ctrl->master) { ++ } ++ return ctrl->tei == Q921_TEI_GROUP; ++} ++ ++/*! ++ * \brief Initialize the given struct q931_party_name ++ * ++ * \param name Structure to initialize ++ * ++ * \return Nothing ++ */ ++void q931_party_name_init(struct q931_party_name *name) ++{ ++ name->valid = 0; ++ name->presentation = PRI_PRES_UNAVAILABLE; ++ name->char_set = PRI_CHAR_SET_ISO8859_1; ++ name->str[0] = '\0'; ++} ++ ++/*! ++ * \brief Initialize the given struct q931_party_number ++ * ++ * \param number Structure to initialize ++ * ++ * \return Nothing ++ */ ++void q931_party_number_init(struct q931_party_number *number) ++{ ++ number->valid = 0; ++ number->presentation = PRI_PRES_UNAVAILABLE | PRI_PRES_USER_NUMBER_UNSCREENED; ++ number->plan = (PRI_TON_UNKNOWN << 4) | PRI_NPI_E163_E164; ++ number->str[0] = '\0'; ++} ++ ++/*! ++ * \brief Initialize the given struct q931_party_subaddress ++ * ++ * \param subaddress Structure to initialize ++ * ++ * \return Nothing ++ */ ++void q931_party_subaddress_init(struct q931_party_subaddress *subaddress) ++{ ++ subaddress->valid = 0; ++ subaddress->type = 0; ++ subaddress->odd_even_indicator = 0; ++ subaddress->length = 0; ++ subaddress->data[0] = '\0'; ++} ++ ++/*! ++ * \brief Initialize the given struct q931_party_address ++ * ++ * \param address Structure to initialize ++ * ++ * \return Nothing ++ */ ++void q931_party_address_init(struct q931_party_address *address) ++{ ++ q931_party_number_init(&address->number); ++ q931_party_subaddress_init(&address->subaddress); ++} ++ ++/*! ++ * \brief Initialize the given struct q931_party_id ++ * ++ * \param id Structure to initialize ++ * ++ * \return Nothing ++ */ ++void q931_party_id_init(struct q931_party_id *id) ++{ ++ q931_party_name_init(&id->name); ++ q931_party_number_init(&id->number); ++ q931_party_subaddress_init(&id->subaddress); ++} ++ ++/*! ++ * \brief Initialize the given struct q931_party_redirecting ++ * ++ * \param redirecting Structure to initialize ++ * ++ * \return Nothing ++ */ ++void q931_party_redirecting_init(struct q931_party_redirecting *redirecting) ++{ ++ q931_party_id_init(&redirecting->from); ++ q931_party_id_init(&redirecting->to); ++ q931_party_id_init(&redirecting->orig_called); ++ redirecting->state = Q931_REDIRECTING_STATE_IDLE; ++ redirecting->count = 0; ++ redirecting->orig_reason = PRI_REDIR_UNKNOWN; ++ redirecting->reason = PRI_REDIR_UNKNOWN; ++} ++ ++/*! ++ * \brief Compare the left and right party name. ++ * ++ * \param left Left parameter party name. ++ * \param right Right parameter party name. ++ * ++ * \retval < 0 when left < right. ++ * \retval == 0 when left == right. ++ * \retval > 0 when left > right. ++ */ ++int q931_party_name_cmp(const struct q931_party_name *left, const struct q931_party_name *right) ++{ ++ int cmp; ++ ++ if (!left->valid) { ++ if (!right->valid) { ++ return 0; ++ } ++ return -1; ++ } else if (!right->valid) { ++ return 1; ++ } ++ cmp = left->char_set - right->char_set; ++ if (cmp) { ++ return cmp; ++ } ++ cmp = strcmp(left->str, right->str); ++ if (cmp) { ++ return cmp; ++ } ++ cmp = left->presentation - right->presentation; ++ return cmp; ++} ++ ++/*! ++ * \brief Compare the left and right party number. ++ * ++ * \param left Left parameter party number. ++ * \param right Right parameter party number. ++ * ++ * \retval < 0 when left < right. ++ * \retval == 0 when left == right. ++ * \retval > 0 when left > right. ++ */ ++int q931_party_number_cmp(const struct q931_party_number *left, const struct q931_party_number *right) ++{ ++ int cmp; ++ ++ if (!left->valid) { ++ if (!right->valid) { ++ return 0; ++ } ++ return -1; ++ } else if (!right->valid) { ++ return 1; ++ } ++ cmp = left->plan - right->plan; ++ if (cmp) { ++ return cmp; ++ } ++ cmp = strcmp(left->str, right->str); ++ if (cmp) { ++ return cmp; ++ } ++ cmp = left->presentation - right->presentation; ++ return cmp; ++} ++ ++/*! ++ * \brief Compare the left and right party subaddress. ++ * ++ * \param left Left parameter party subaddress. ++ * \param right Right parameter party subaddress. ++ * ++ * \retval < 0 when left < right. ++ * \retval == 0 when left == right. ++ * \retval > 0 when left > right. ++ */ ++int q931_party_subaddress_cmp(const struct q931_party_subaddress *left, const struct q931_party_subaddress *right) ++{ ++ int cmp; ++ ++ if (!left->valid) { ++ if (!right->valid) { ++ return 0; ++ } ++ return -1; ++ } else if (!right->valid) { ++ return 1; ++ } ++ cmp = left->type - right->type; ++ if (cmp) { ++ return cmp; ++ } ++ cmp = memcmp(left->data, right->data, ++ (left->length < right->length) ? left->length : right->length); ++ if (cmp) { ++ return cmp; ++ } ++ cmp = left->length - right->length; ++ if (cmp) { ++ return cmp; ++ } ++ cmp = left->odd_even_indicator - right->odd_even_indicator; ++ return cmp; ++} ++ ++/*! ++ * \brief Compare the left and right party id. ++ * ++ * \param left Left parameter party id. ++ * \param right Right parameter party id. ++ * ++ * \retval < 0 when left < right. ++ * \retval == 0 when left == right. ++ * \retval > 0 when left > right. ++ */ ++int q931_party_id_cmp(const struct q931_party_id *left, const struct q931_party_id *right) ++{ ++ int cmp; ++ ++ cmp = q931_party_number_cmp(&left->number, &right->number); ++ if (cmp) { ++ return cmp; ++ } ++ cmp = q931_party_subaddress_cmp(&left->subaddress, &right->subaddress); ++ if (cmp) { ++ return cmp; ++ } ++ cmp = q931_party_name_cmp(&left->name, &right->name); ++ return cmp; ++} ++ ++/*! ++ * \brief Copy the Q.931 party name to the PRI party name structure. ++ * ++ * \param pri_name PRI party name structure ++ * \param q931_name Q.931 party name structure ++ * ++ * \return Nothing ++ */ ++void q931_party_name_copy_to_pri(struct pri_party_name *pri_name, const struct q931_party_name *q931_name) ++{ ++ if (q931_name->valid) { ++ pri_name->valid = 1; ++ pri_name->presentation = q931_name->presentation; ++ pri_name->char_set = q931_name->char_set; ++ libpri_copy_string(pri_name->str, q931_name->str, sizeof(pri_name->str)); ++ } else { ++ pri_name->valid = 0; ++ pri_name->presentation = PRI_PRES_UNAVAILABLE; ++ pri_name->char_set = PRI_CHAR_SET_ISO8859_1; ++ pri_name->str[0] = 0; ++ } ++} ++ ++/*! ++ * \brief Copy the Q.931 party number to the PRI party number structure. ++ * ++ * \param pri_number PRI party number structure ++ * \param q931_number Q.931 party number structure ++ * ++ * \return Nothing ++ */ ++void q931_party_number_copy_to_pri(struct pri_party_number *pri_number, const struct q931_party_number *q931_number) ++{ ++ if (q931_number->valid) { ++ pri_number->valid = 1; ++ pri_number->presentation = q931_number->presentation; ++ pri_number->plan = q931_number->plan; ++ libpri_copy_string(pri_number->str, q931_number->str, sizeof(pri_number->str)); ++ } else { ++ pri_number->valid = 0; ++ pri_number->presentation = PRI_PRES_UNAVAILABLE | PRI_PRES_USER_NUMBER_UNSCREENED; ++ pri_number->plan = (PRI_TON_UNKNOWN << 4) | PRI_NPI_E163_E164; ++ pri_number->str[0] = 0; ++ } ++} ++ ++/*! ++ * \brief Copy the Q.931 party subaddress to the PRI party subaddress structure. ++ * ++ * \param pri_subaddress PRI party subaddress structure ++ * \param q931_subaddress Q.931 party subaddress structure ++ * ++ * \return Nothing ++ */ ++void q931_party_subaddress_copy_to_pri(struct pri_party_subaddress *pri_subaddress, const struct q931_party_subaddress *q931_subaddress) ++{ ++ int length; ++ ++ /* ++ * The size of pri_subaddress->data[] is not the same as the size of ++ * q931_subaddress->data[]. ++ */ ++ ++ if (!q931_subaddress->valid) { ++ pri_subaddress->valid = 0; ++ pri_subaddress->type = 0; ++ pri_subaddress->odd_even_indicator = 0; ++ pri_subaddress->length = 0; ++ pri_subaddress->data[0] = '\0'; ++ return; ++ } ++ ++ pri_subaddress->valid = 1; ++ pri_subaddress->type = q931_subaddress->type; ++ pri_subaddress->odd_even_indicator = q931_subaddress->odd_even_indicator; ++ ++ length = q931_subaddress->length; ++ pri_subaddress->length = length; ++ memcpy(pri_subaddress->data, q931_subaddress->data, length); ++ pri_subaddress->data[length] = '\0'; ++} ++ ++/*! ++ * \brief Copy the Q.931 party id to the PRI party id structure. ++ * ++ * \param pri_id PRI party id structure ++ * \param q931_id Q.931 party id structure ++ * ++ * \return Nothing ++ */ ++void q931_party_id_copy_to_pri(struct pri_party_id *pri_id, const struct q931_party_id *q931_id) ++{ ++ q931_party_name_copy_to_pri(&pri_id->name, &q931_id->name); ++ q931_party_number_copy_to_pri(&pri_id->number, &q931_id->number); ++ q931_party_subaddress_copy_to_pri(&pri_id->subaddress, &q931_id->subaddress); ++} ++ ++/*! ++ * \brief Copy the Q.931 redirecting data to the PRI redirecting structure. ++ * ++ * \param pri_redirecting PRI redirecting structure ++ * \param q931_redirecting Q.931 redirecting structure ++ * ++ * \return Nothing ++ */ ++void q931_party_redirecting_copy_to_pri(struct pri_party_redirecting *pri_redirecting, const struct q931_party_redirecting *q931_redirecting) ++{ ++ q931_party_id_copy_to_pri(&pri_redirecting->from, &q931_redirecting->from); ++ q931_party_id_copy_to_pri(&pri_redirecting->to, &q931_redirecting->to); ++ q931_party_id_copy_to_pri(&pri_redirecting->orig_called, ++ &q931_redirecting->orig_called); ++ pri_redirecting->count = q931_redirecting->count; ++ pri_redirecting->orig_reason = q931_redirecting->orig_reason; ++ pri_redirecting->reason = q931_redirecting->reason; ++} ++ ++/*! ++ * \brief Fixup some values in the q931_party_id that may be objectionable by switches. ++ * ++ * \param ctrl D channel controller. ++ * \param id Party ID to tweak. ++ * ++ * \return Nothing ++ */ ++void q931_party_id_fixup(const struct pri *ctrl, struct q931_party_id *id) ++{ ++ switch (ctrl->switchtype) { ++ case PRI_SWITCH_DMS100: ++ case PRI_SWITCH_ATT4ESS: ++ /* Doesn't like certain presentation types */ ++ if (id->number.valid && !(id->number.presentation & 0x7c)) { ++ /* i.e., If presentation is allowed it must be a network number */ ++ id->number.presentation = PRES_ALLOWED_NETWORK_NUMBER; ++ } ++ break; ++ default: ++ break; ++ } ++} ++ ++/*! ++ * \brief Determine the overall presentation value for the given party. ++ * ++ * \param id Party to determine the overall presentation value. ++ * ++ * \return Overall presentation value for the given party. ++ */ ++int q931_party_id_presentation(const struct q931_party_id *id) ++{ ++ int number_priority; ++ int number_value; ++ int number_screening; ++ int name_priority; ++ int name_value; ++ ++ /* Determine name presentation priority. */ ++ if (!id->name.valid) { ++ name_value = PRI_PRES_UNAVAILABLE; ++ name_priority = 3; ++ } else { ++ name_value = id->name.presentation & PRI_PRES_RESTRICTION; ++ switch (name_value) { ++ case PRI_PRES_RESTRICTED: ++ name_priority = 0; ++ break; ++ case PRI_PRES_ALLOWED: ++ name_priority = 1; ++ break; ++ case PRI_PRES_UNAVAILABLE: ++ name_priority = 2; ++ break; ++ default: ++ name_value = PRI_PRES_UNAVAILABLE; ++ name_priority = 3; ++ break; ++ } ++ } ++ ++ /* Determine number presentation priority. */ ++ if (!id->number.valid) { ++ number_screening = PRI_PRES_USER_NUMBER_UNSCREENED; ++ number_value = PRI_PRES_UNAVAILABLE; ++ number_priority = 3; ++ } else { ++ number_screening = id->number.presentation & PRI_PRES_NUMBER_TYPE; ++ number_value = id->number.presentation & PRI_PRES_RESTRICTION; ++ switch (number_value) { ++ case PRI_PRES_RESTRICTED: ++ number_priority = 0; ++ break; ++ case PRI_PRES_ALLOWED: ++ number_priority = 1; ++ break; ++ case PRI_PRES_UNAVAILABLE: ++ number_priority = 2; ++ break; ++ default: ++ number_screening = PRI_PRES_USER_NUMBER_UNSCREENED; ++ number_value = PRI_PRES_UNAVAILABLE; ++ number_priority = 3; ++ break; ++ } ++ } ++ ++ /* Select the wining presentation value. */ ++ if (name_priority < number_priority) { ++ number_value = name_value; ++ } ++ ++ return number_value | number_screening; ++} ++ ++static void q931_clr_subcommands(struct pri *ctrl) ++{ ++ ctrl->subcmds.counter_subcmd = 0; ++} ++ ++struct pri_subcommand *q931_alloc_subcommand(struct pri *ctrl) ++{ ++ if (ctrl->subcmds.counter_subcmd < PRI_MAX_SUBCOMMANDS) { ++ return &ctrl->subcmds.subcmd[ctrl->subcmds.counter_subcmd++]; ++ } ++ ++ return NULL; ++} ++ + static char *code2str(int code, struct msgtype *codes, int max) + { + int x; +@@ -258,15 +795,18 @@ + return "Unknown"; + } + +-static void call_init(struct q931_call *c) ++static char *pritype(int type) + { +- c->forceinvert = -1; +- c->cr = -1; +- c->slotmap = -1; +- c->channelno = -1; +- c->newcall = 1; +- c->ourcallstate = Q931_CALL_STATE_NULL; +- c->peercallstate = Q931_CALL_STATE_NULL; ++ switch (type) { ++ case PRI_CPE: ++ return "CPE"; ++ break; ++ case PRI_NETWORK: ++ return "NET"; ++ break; ++ default: ++ return "UNKNOWN"; ++ } + } + + static char *binary(int b, int len) { +@@ -280,56 +820,85 @@ + return res; + } + +-static FUNC_RECV(receive_channel_id) ++static int receive_channel_id(int full_ie, struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, int len) + { + int x; +- int pos=0; +-#ifdef NO_BRI_SUPPORT +- if (!ie->data[0] & 0x20) { +- pri_error(pri, "!! Not PRI type!?\n"); +- return -1; +- } +-#endif +-#ifndef NOAUTO_CHANNEL_SELECTION_SUPPORT +- if (pri->bri) { +- if (!(ie->data[0] & 3)) +- call->justsignalling = 1; +- else +- call->channelno = ie->data[0] & 3; ++ int pos = 0; ++ int need_extended_channel_octets;/*!< TRUE if octets 3.2 and 3.3 need to be present. */ ++ ++ if (ie->data[0] & 0x08) { ++ call->chanflags = FLAG_EXCLUSIVE; + } else { +- switch (ie->data[0] & 3) { +- case 0: +- call->justsignalling = 1; +- break; +- case 1: +- break; +- default: +- pri_error(pri, "!! Unexpected Channel selection %d\n", ie->data[0] & 3); +- return -1; ++ call->chanflags = FLAG_PREFERRED; ++ } ++ ++ need_extended_channel_octets = 0; ++ if (ie->data[0] & 0x20) { ++ /* PRI encoded interface type */ ++ switch (ie->data[0] & 0x03) { ++ case 0x00: ++ /* No channel */ ++ call->channelno = 0; ++ call->chanflags = FLAG_PREFERRED; ++ break; ++ case 0x01: ++ /* As indicated in following octets */ ++ need_extended_channel_octets = 1; ++ break; ++ case 0x03: ++ /* Any channel */ ++ call->chanflags = FLAG_PREFERRED; ++ break; ++ default: ++ pri_error(ctrl, "!! Unexpected Channel selection %d\n", ie->data[0] & 0x03); ++ return -1; + } ++ } else { ++ /* BRI encoded interface type */ ++ switch (ie->data[0] & 0x03) { ++ case 0x00: ++ /* No channel */ ++ call->channelno = 0; ++ call->chanflags = FLAG_PREFERRED; ++ break; ++ case 0x03: ++ /* Any channel */ ++ call->chanflags = FLAG_PREFERRED; ++ break; ++ default: ++ /* Specified B channel (B1 or B2) */ ++ call->channelno = ie->data[0] & 0x03; ++ break; ++ } + } +-#endif +- if (ie->data[0] & 0x08) +- call->chanflags = FLAG_EXCLUSIVE; +- else +- call->chanflags = FLAG_PREFERRED; ++ + pos++; + if (ie->data[0] & 0x40) { + /* DS1 specified -- stop here */ + call->ds1no = ie->data[1] & 0x7f; + call->ds1explicit = 1; + pos++; +- } else ++ } else { + call->ds1explicit = 0; ++ } + +- if (pos+2 < len) { ++ if (ie->data[0] & 0x04) { ++ /* D channel call. Signaling only. */ ++ call->cis_call = 1; ++ call->chanflags = FLAG_EXCLUSIVE;/* For safety mark this channel as exclusive. */ ++ call->channelno = 0; ++ return 0; ++ } ++ ++ if (need_extended_channel_octets && pos + 2 < len) { + /* More coming */ + if ((ie->data[pos] & 0x0f) != 3) { +- pri_error(pri, "!! Unexpected Channel Type %d\n", ie->data[1] & 0x0f); ++ /* Channel type/mapping is not for B channel units. */ ++ pri_error(ctrl, "!! Unexpected Channel Type %d\n", ie->data[1] & 0x0f); + return -1; + } + if ((ie->data[pos] & 0x60) != 0) { +- pri_error(pri, "!! Invalid CCITT coding %d\n", (ie->data[1] & 0x60) >> 5); ++ pri_error(ctrl, "!! Invalid CCITT coding %d\n", (ie->data[1] & 0x60) >> 5); + return -1; + } + if (ie->data[pos] & 0x10) { +@@ -340,136 +909,167 @@ + call->slotmap <<= 8; + call->slotmap |= ie->data[x + pos]; + } +- return 0; + } else { + pos++; + /* Only expect a particular channel */ + call->channelno = ie->data[pos] & 0x7f; +- if (pri->chan_mapping_logical && call->channelno > 15) ++ if (ctrl->chan_mapping_logical && call->channelno > 15) + call->channelno++; +- return 0; + } +- } else +- return 0; +- return -1; ++ } ++ return 0; + } + +-static FUNC_SEND(transmit_channel_id) ++static int transmit_channel_id(int full_ie, struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, int len, int order) + { +- int pos=0; ++ int pos = 0; + +- + /* We are ready to transmit single IE only */ + if (order > 1) + return 0; +- +- if (call->justsignalling) { +- ie->data[pos++] = 0xac; /* Read the standards docs to figure this out +- ECMA-165 section 7.3 */ ++ ++ if (call->cis_call) { ++ /* ++ * Read the standards docs to figure this out. ++ * Q.SIG ECMA-165 section 7.3 ++ * ITU Q.931 section 4.5.13 ++ */ ++ ie->data[pos++] = ctrl->bri ? 0x8c : 0xac; + return pos + 2; + } +- ++ + /* Start with standard stuff */ +- if (pri->switchtype == PRI_SWITCH_GR303_TMC) ++ if (ctrl->switchtype == PRI_SWITCH_GR303_TMC) + ie->data[pos] = 0x69; +- else if (pri->bri) { ++ else if (ctrl->bri) { + ie->data[pos] = 0x80; +- if (call->channelno > -1) +- ie->data[pos] |= (call->channelno & 0x3); +- } else +- ie->data[pos] = 0xa1; +- /* Add exclusive flag if necessary */ +- if (call->chanflags & FLAG_EXCLUSIVE) ++ ie->data[pos] |= (call->channelno & 0x3); ++ } else { ++ /* PRI */ ++ if (call->slotmap != -1 || (call->chanflags & FLAG_WHOLE_INTERFACE)) { ++ /* Specified channel */ ++ ie->data[pos] = 0xa1; ++ } else if (call->channelno < 0 || call->channelno == 0xff) { ++ /* Any channel */ ++ ie->data[pos] = 0xa3; ++ } else if (!call->channelno) { ++ /* No channel */ ++ ie->data[pos] = 0xa0; ++ } else { ++ /* Specified channel */ ++ ie->data[pos] = 0xa1; ++ } ++ } ++ if (call->chanflags & FLAG_EXCLUSIVE) { ++ /* Channel is exclusive */ + ie->data[pos] |= 0x08; +- else if (!(call->chanflags & FLAG_PREFERRED)) { ++ } else if (!call->chanflags) { + /* Don't need this IE */ + return 0; + } + +- if (((pri->switchtype != PRI_SWITCH_QSIG) && (call->ds1no > 0)) || call->ds1explicit) { +- /* Note that we are specifying the identifier */ ++ if (!ctrl->bri && (((ctrl->switchtype != PRI_SWITCH_QSIG) && (call->ds1no > 0)) || call->ds1explicit)) { ++ /* We are specifying the interface. Octet 3.1 */ + ie->data[pos++] |= 0x40; +- /* We need to use the Channel Identifier Present thingy. Just specify it and we're done */ + ie->data[pos++] = 0x80 | call->ds1no; +- } else +- pos++; ++ } else { ++ ++pos; ++ } + +- if (pri->bri) +- return pos + 2; ++ if (!ctrl->bri && (ie->data[0] & 0x03) == 0x01 /* Specified channel */ ++ && !(call->chanflags & FLAG_WHOLE_INTERFACE)) { ++ /* The 3.2 and 3.3 octets need to be present */ ++ ie->data[pos] = 0x83; ++ if (call->slotmap != -1) { ++ int octet; + +- if ((call->channelno > -1) || (call->slotmap != -1)) { +- /* We'll have the octet 8.2 and 8.3's present */ +- ie->data[pos++] = 0x83; +- if (call->channelno > -1) { ++ /* We have to send a channel map */ ++ ie->data[pos++] |= 0x10; ++ for (octet = 3; octet--;) { ++ ie->data[pos++] = (call->slotmap >> (8 * octet)) & 0xff; ++ } ++ } else { + /* Channel number specified */ +- if (pri->chan_mapping_logical && call->channelno > 16) ++ ++pos; ++ if (ctrl->chan_mapping_logical && call->channelno > 16) { + ie->data[pos++] = 0x80 | (call->channelno - 1); +- else ++ } else { + ie->data[pos++] = 0x80 | call->channelno; +- return pos + 2; ++ } + } +- /* We have to send a channel map */ +- if (call->slotmap != -1) { +- ie->data[pos-1] |= 0x10; +- ie->data[pos++] = (call->slotmap & 0xff0000) >> 16; +- ie->data[pos++] = (call->slotmap & 0xff00) >> 8; +- ie->data[pos++] = (call->slotmap & 0xff); +- return pos + 2; +- } + } +- if (call->ds1no > 0) { +- /* We're done */ +- return pos + 2; +- } +- pri_error(pri, "!! No channel map, no channel, and no ds1? What am I supposed to identify?\n"); +- return -1; ++ ++ return pos + 2; + } + +-static FUNC_DUMP(dump_channel_id) ++static void dump_channel_id(int full_ie, struct pri *ctrl, q931_ie *ie, int len, char prefix) + { +- int pos=0; ++ int pos; + int x; +- int res = 0; +- static const char* msg_chan_sel[] = { +- "No channel selected", "B1 channel", "B2 channel","Any channel selected", +- "No channel selected", "As indicated in following octets", "Reserved","Any channel selected" ++ int res; ++ ++ static const char *msg_chan_sel[] = { ++ "No channel selected", "B1 channel", "B2 channel", "Any channel selected", ++ "No channel selected", "As indicated in following octets", "Reserved", "Any channel selected" + }; + +- pri_message(pri, "%c Channel ID (len=%2d) [ Ext: %d IntID: %s %s Spare: %d %s Dchan: %d\n", +- prefix, len, (ie->data[0] & 0x80) ? 1 : 0, (ie->data[0] & 0x40) ? "Explicit" : "Implicit", +- (ie->data[0] & 0x20) ? "PRI" : "Other", (ie->data[0] & 0x10) ? 1 : 0, +- (ie->data[0] & 0x08) ? "Exclusive" : "Preferred", (ie->data[0] & 0x04) ? 1 : 0); +- pri_message(pri, "%c ChanSel: %s\n", +- prefix, msg_chan_sel[(ie->data[0] & 0x3) + ((ie->data[0]>>3) & 0x4)]); +- pos++; +- len--; +- if (ie->data[0] & 0x40) { ++ pri_message(ctrl, ++ "%c Channel ID (len=%2d) [ Ext: %d IntID: %s %s Spare: %d %s Dchan: %d\n", ++ prefix, len, ++ (ie->data[0] & 0x80) ? 1 : 0, ++ (ie->data[0] & 0x40) ? "Explicit" : "Implicit", ++ (ie->data[0] & 0x20) ? "Other(PRI)" : "BRI", ++ (ie->data[0] & 0x10) ? 1 : 0, ++ (ie->data[0] & 0x08) ? "Exclusive" : "Preferred", ++ (ie->data[0] & 0x04) ? 1 : 0); ++ pri_message(ctrl, "%c ChanSel: %s\n", ++ prefix, msg_chan_sel[(ie->data[0] & 0x03) | ((ie->data[0] >> 3) & 0x04)]); ++ pos = 1; ++ len -= 2; ++ if (ie->data[0] & 0x40) { + /* Explicitly defined DS1 */ +- pri_message(pri, "%c Ext: %d DS1 Identifier: %d \n", prefix, (ie->data[pos] & 0x80) >> 7, ie->data[pos] & 0x7f); +- pos++; ++ do { ++ pri_message(ctrl, "%c Ext: %d DS1 Identifier: %d \n", ++ prefix, (ie->data[pos] & 0x80) >> 7, ie->data[pos] & 0x7f); ++ ++pos; ++ } while (!(ie->data[pos - 1] & 0x80) && pos < len); + } else { + /* Implicitly defined DS1 */ + } +- if (pos+2 < len) { ++ if (pos < len) { + /* Still more information here */ +- pri_message(pri, "%c Ext: %d Coding: %d %s Specified Channel Type: %d\n", +- prefix, (ie->data[pos] & 0x80) >> 7, (ie->data[pos] & 60) >> 5, +- (ie->data[pos] & 0x10) ? "Slot Map" : "Number", ie->data[pos] & 0x0f); +- if (!(ie->data[pos] & 0x10)) { ++ pri_message(ctrl, ++ "%c Ext: %d Coding: %d %s Specified Channel Type: %d\n", ++ prefix, (ie->data[pos] & 0x80) >> 7, (ie->data[pos] & 60) >> 5, ++ (ie->data[pos] & 0x10) ? "Slot Map" : "Number", ie->data[pos] & 0x0f); ++ ++pos; ++ } ++ if (pos < len) { ++ if (!(ie->data[pos - 1] & 0x10)) { + /* Number specified */ +- pos++; +- pri_message(pri, "%c Ext: %d Channel: %d ]\n", prefix, (ie->data[pos] & 0x80) >> 7, +- (ie->data[pos]) & 0x7f); ++ do { ++ pri_message(ctrl, ++ "%c Ext: %d Channel: %d Type: %s%c\n", ++ prefix, (ie->data[pos] & 0x80) >> 7, ++ (ie->data[pos]) & 0x7f, pritype(ctrl->localtype), ++ (pos + 1 < len) ? ' ' : ']'); ++ ++pos; ++ } while (pos < len); + } else { +- pos++; + /* Map specified */ +- for (x=0;x<3;x++) { ++ res = 0; ++ x = 0; ++ do { + res <<= 8; + res |= ie->data[pos++]; +- } +- pri_message(pri, "%c Map: %s ]\n", prefix, binary(res, 24)); ++ ++x; ++ } while (pos < len); ++ pri_message(ctrl, "%c Map len: %d Map: %s ]\n", prefix, ++ x, binary(res, x << 3)); + } +- } else pri_message(pri, " ]\n"); ++ } else { ++ pri_message(ctrl, " ]\n"); ++ } + } + + static char *ri2str(int ri) +@@ -482,20 +1082,20 @@ + return code2str(ri, ris, sizeof(ris) / sizeof(ris[0])); + } + +-static FUNC_DUMP(dump_restart_indicator) ++static void dump_restart_indicator(int full_ie, struct pri *ctrl, q931_ie *ie, int len, char prefix) + { +- pri_message(pri, "%c Restart Indentifier (len=%2d) [ Ext: %d Spare: %d Resetting %s (%d) ]\n", ++ pri_message(ctrl, "%c Restart Indentifier (len=%2d) [ Ext: %d Spare: %d Resetting %s (%d) ]\n", + prefix, len, (ie->data[0] & 0x80) >> 7, (ie->data[0] & 0x78) >> 3, ri2str(ie->data[0] & 0x7), ie->data[0] & 0x7); + } + +-static FUNC_RECV(receive_restart_indicator) ++static int receive_restart_indicator(int full_ie, struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, int len) + { + /* Pretty simple */ + call->ri = ie->data[0] & 0x7; + return 0; + } + +-static FUNC_SEND(transmit_restart_indicator) ++static int transmit_restart_indicator(int full_ie, struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, int len, int order) + { + /* Pretty simple */ + switch(call->ri) { +@@ -509,7 +1109,7 @@ + ie->data[0] = 0xA0 | (call->ri & 0x7); + break; + default: +- pri_error(pri, "!! Invalid restart indicator value %d\n", call->ri); ++ pri_error(ctrl, "!! Invalid restart indicator value %d\n", call->ri); + return-1; + } + return 3; +@@ -607,16 +1207,16 @@ + return code2str(proto, protos, sizeof(protos) / sizeof(protos[0])); + } + +-static FUNC_DUMP(dump_bearer_capability) ++static void dump_bearer_capability(int full_ie, struct pri *ctrl, q931_ie *ie, int len, char prefix) + { + int pos=2; +- pri_message(pri, "%c Bearer Capability (len=%2d) [ Ext: %d Q.931 Std: %d Info transfer capability: %s (%d)\n", ++ pri_message(ctrl, "%c Bearer Capability (len=%2d) [ Ext: %d Q.931 Std: %d Info transfer capability: %s (%d)\n", + prefix, len, (ie->data[0] & 0x80 ) >> 7, (ie->data[0] & 0x60) >> 5, cap2str(ie->data[0] & 0x1f), (ie->data[0] & 0x1f)); +- pri_message(pri, "%c Ext: %d Trans mode/rate: %s (%d)\n", prefix, (ie->data[1] & 0x80) >> 7, mode2str(ie->data[1] & 0x7f), ie->data[1] & 0x7f); ++ pri_message(ctrl, "%c Ext: %d Trans mode/rate: %s (%d)\n", prefix, (ie->data[1] & 0x80) >> 7, mode2str(ie->data[1] & 0x7f), ie->data[1] & 0x7f); + + /* octet 4.1 exists iff mode/rate is multirate */ + if ((ie->data[1] & 0x7f) == 0x18) { +- pri_message(pri, "%c Ext: %d Transfer rate multiplier: %d x 64\n", prefix, (ie->data[2] & 0x80) >> 7, ie->data[2] & 0x7f); ++ pri_message(ctrl, "%c Ext: %d Transfer rate multiplier: %d x 64\n", prefix, (ie->data[2] & 0x80) >> 7, ie->data[2] & 0x7f); + pos++; + } + +@@ -632,7 +1232,7 @@ + too, so we have to do the same for binary compatability */ + u_int8_t layer1 = ie->data[pos] & 0x7f; + +- pri_message(pri, "%c User information layer 1: %s (%d)\n", ++ pri_message(ctrl, "%c User information layer 1: %s (%d)\n", + prefix, l12str(layer1), layer1); + pos++; + +@@ -640,7 +1240,7 @@ + if (pos < len && !(ie->data[pos-1] & 0x80)) { + int ra = ie->data[pos] & 0x7f; + +- pri_message(pri, "%c Async: %d, Negotiation: %d, " ++ pri_message(ctrl, "%c Async: %d, Negotiation: %d, " + "User rate: %s (%#x)\n", + prefix, + ra & PRI_RATE_ADAPT_ASYNC ? 1 : 0, +@@ -654,7 +1254,7 @@ + if (pos < len && !(ie->data[pos-1] & 0x80)) { + u_int8_t data = ie->data[pos]; + if (layer1 == PRI_LAYER_1_ITU_RATE_ADAPT) { +- pri_message(pri, "%c Intermediate rate: %s (%d), " ++ pri_message(ctrl, "%c Intermediate rate: %s (%d), " + "NIC on Tx: %d, NIC on Rx: %d, " + "Flow control on Tx: %d, " + "Flow control on Rx: %d\n", +@@ -665,7 +1265,7 @@ + (data & 0x04)?1:0, + (data & 0x02)?1:0); + } else if (layer1 == PRI_LAYER_1_V120_RATE_ADAPT) { +- pri_message(pri, "%c Hdr: %d, Multiframe: %d, Mode: %d, " ++ pri_message(ctrl, "%c Hdr: %d, Multiframe: %d, Mode: %d, " + "LLI negot: %d, Assignor: %d, " + "In-band neg: %d\n", prefix, + (data & 0x40)?1:0, +@@ -675,7 +1275,8 @@ + (data & 0x04)?1:0, + (data & 0x02)?1:0); + } else { +- pri_message(pri, "%c Unknown octet 5b: 0x%x\n", data ); ++ pri_message(ctrl, "%c Unknown octet 5b: 0x%x\n", ++ prefix, data); + } + pos++; + } +@@ -688,7 +1289,7 @@ + const char *parity[] = {"Odd","?","Even","None", + "zero","one","?","?"}; + +- pri_message(pri, "%c Stop bits: %s, data bits: %s, " ++ pri_message(ctrl, "%c Stop bits: %s, data bits: %s, " + "parity: %s\n", prefix, + stop_bits[(data & 0x60) >> 5], + data_bits[(data & 0x18) >> 3], +@@ -700,7 +1301,7 @@ + /* octet 5d? */ + if (pos < len && !(ie->data[pos-1] & 0x80)) { + u_int8_t data = ie->data[pos]; +- pri_message(pri, "%c Duplex mode: %d, modem type: %d\n", ++ pri_message(ctrl, "%c Duplex mode: %d, modem type: %d\n", + prefix, (data & 0x40) ? 1 : 0,data & 0x3F); + pos++; + } +@@ -710,7 +1311,7 @@ + /* Look for octet 6; this is identified by bits 5,6 == 10 */ + if (pos < len && + (ie->data[pos] & 0x60) == 0x40) { +- pri_message(pri, "%c User information layer 2: %s (%d)\n", ++ pri_message(ctrl, "%c User information layer 2: %s (%d)\n", + prefix, l22str(ie->data[pos] & 0x1f), + ie->data[pos] & 0x1f); + pos++; +@@ -718,7 +1319,7 @@ + + /* Look for octet 7; this is identified by bits 5,6 == 11 */ + if (pos < len && (ie->data[pos] & 0x60) == 0x60) { +- pri_message(pri, "%c User information layer 3: %s (%d)\n", ++ pri_message(ctrl, "%c User information layer 3: %s (%d)\n", + prefix, l32str(ie->data[pos] & 0x1f), + ie->data[pos] & 0x1f); + pos++; +@@ -730,18 +1331,18 @@ + proto = ((ie->data[pos] & 0xF) << 4 ) | + (ie->data[pos+1] & 0xF); + +- pri_message(pri, "%c Network layer: 0x%x\n", prefix, ++ pri_message(ctrl, "%c Network layer: 0x%x\n", prefix, + proto ); + pos += 2; + } + } + } + +-static FUNC_RECV(receive_bearer_capability) ++static int receive_bearer_capability(int full_ie, struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, int len) + { + int pos=2; + if (ie->data[0] & 0x60) { +- pri_error(pri, "!! non-standard Q.931 standard field\n"); ++ pri_error(ctrl, "!! non-standard Q.931 standard field\n"); + return -1; + } + call->transcapability = ie->data[0] & 0x1f; +@@ -788,7 +1389,7 @@ + return 0; + } + +-static FUNC_SEND(transmit_bearer_capability) ++static int transmit_bearer_capability(int full_ie, struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, int len, int order) + { + int tc; + int pos; +@@ -797,20 +1398,20 @@ + if(order > 1) + return 0; + +- tc = call->transcapability; +- if (pri->subchannel && !pri->bri) { ++ if (ctrl->subchannel && !ctrl->bri) { + /* Bearer capability is *hard coded* in GR-303 */ + ie->data[0] = 0x88; + ie->data[1] = 0x90; + return 4; + } +- +- if (call->justsignalling) { ++ ++ if (call->cis_call) { + ie->data[0] = 0xa8; + ie->data[1] = 0x80; + return 4; + } +- ++ ++ tc = call->transcapability; + ie->data[0] = 0x80 | tc; + ie->data[1] = call->transmoderate | 0x80; + +@@ -820,7 +1421,7 @@ + ie->data[pos++] = call->transmultiple | 0x80; + } + +- if ((tc & PRI_TRANS_CAP_DIGITAL) && (pri->switchtype == PRI_SWITCH_EUROISDN_E1) && ++ if ((tc & PRI_TRANS_CAP_DIGITAL) && (ctrl->switchtype == PRI_SWITCH_EUROISDN_E1) && + (call->transmoderate == TRANS_MODE_PACKET)) { + /* Apparently EuroISDN switches don't seem to like user layer 2/3 */ + return 4; +@@ -833,7 +1434,7 @@ + + if (call->transmoderate != TRANS_MODE_PACKET) { + /* If you have an AT&T 4ESS, you don't send any more info */ +- if ((pri->switchtype != PRI_SWITCH_ATT4ESS) && (call->userl1 > -1)) { ++ if ((ctrl->switchtype != PRI_SWITCH_ATT4ESS) && (call->userl1 > -1)) { + ie->data[pos++] = call->userl1 | 0x80; /* XXX Ext bit? XXX */ + if (call->userl1 == PRI_LAYER_1_ITU_RATE_ADAPT) { + ie->data[pos++] = call->rateadaption | 0x80; +@@ -908,6 +1509,30 @@ + return code2str(plan, plans, sizeof(plans) / sizeof(plans[0])); + } + ++/* Calling Party Category (Definitions from Q.763) */ ++static char *cpc2str(int plan) ++{ ++ static struct msgtype plans[] = { ++ { 0, "Unknown Source" }, ++ { 1, "Operator French" }, ++ { 2, "Operator English" }, ++ { 3, "Operator German" }, ++ { 4, "Operator Russian" }, ++ { 5, "Operator Spanish" }, ++ { 6, "Mut Agree Chinese" }, ++ { 7, "Mut Agreement" }, ++ { 8, "Mut Agree Japanese" }, ++ { 9, "National Operator" }, ++ { 10, "Ordinary Toll Caller" }, ++ { 11, "Priority Toll Caller" }, ++ { 12, "Data Call" }, ++ { 13, "Test Call" }, ++ { 14, "Spare" }, ++ { 15, "Pay Phone" }, ++ }; ++ return code2str(plan, plans, ARRAY_LEN(plans)); ++} ++ + char *pri_pres2str(int pres) + { + static struct msgtype press[] = { +@@ -934,51 +1559,134 @@ + num[len] = 0; + } + +-static FUNC_DUMP(dump_called_party_number) ++static void q931_get_subaddr_specific(unsigned char *num, int maxlen, unsigned char *src, int len, char oddflag) + { +- unsigned char cnum[256]; ++ /* User Specified */ ++ int x; ++ char *ptr = (char *) num; + +- q931_get_number(cnum, sizeof(cnum), ie->data + 1, len - 3); +- pri_message(pri, "%c Called Number (len=%2d) [ Ext: %d TON: %s (%d) NPI: %s (%d) '%s' ]\n", +- prefix, len, ie->data[0] >> 7, ton2str((ie->data[0] >> 4) & 0x07), (ie->data[0] >> 4) & 0x07, npi2str(ie->data[0] & 0x0f), ie->data[0] & 0x0f, cnum); ++ if (len <= 0) { ++ num[0] = '\0'; ++ return; ++ } ++ ++ if (((len * 2) + 1) > maxlen) { ++ len = (maxlen / 2) - 1; ++ } ++ ++ for (x = 0; x < (len - 1); ++x) { ++ ptr += sprintf(ptr, "%02x", src[x]); ++ } ++ ++ if (oddflag) { ++ /* ODD */ ++ sprintf(ptr, "%01x", (src[len - 1]) >> 4); ++ } else { ++ /* EVEN */ ++ sprintf(ptr, "%02x", src[len - 1]); ++ } + } + +-static FUNC_DUMP(dump_called_party_subaddr) ++static int transmit_subaddr_helper(int full_ie, struct pri *ctrl, struct q931_party_subaddress *q931_subaddress, int msgtype, q931_ie *ie, int offset, int len, int order) + { ++ size_t datalen; ++ ++ if (!q931_subaddress->valid) { ++ return 0; ++ } ++ ++ datalen = q931_subaddress->length; ++ if (!q931_subaddress->type) { ++ /* 0 = NSAP */ ++ /* 0 = Odd/Even indicator */ ++ ie->data[0] = 0x80; ++ } else { ++ /* 2 = User Specified */ ++ ie->data[0] = q931_subaddress->odd_even_indicator ? 0xA8 : 0xA0; ++ } ++ memcpy(ie->data + offset, q931_subaddress->data, datalen); ++ ++ return datalen + (offset + 2); ++} ++ ++static int receive_subaddr_helper(int full_ie, struct pri *ctrl, struct q931_party_subaddress *q931_subaddress, int msgtype, q931_ie *ie, int offset, int len) ++{ ++ if (len <= 0) { ++ return -1; ++ } ++ ++ q931_subaddress->valid = 1; ++ q931_subaddress->length = len; ++ /* type: 0 = NSAP, 2 = User Specified */ ++ q931_subaddress->type = ((ie->data[0] & 0x70) >> 4); ++ q931_subaddress->odd_even_indicator = (ie->data[0] & 0x08) ? 1 : 0; ++ q931_get_number(q931_subaddress->data, sizeof(q931_subaddress->data), ++ ie->data + offset, len); ++ ++ return 0; ++} ++ ++static void dump_subaddr_helper(int full_ie, struct pri *ctrl, q931_ie *ie, int offset, int len, int datalen, char prefix, const char *named) ++{ + unsigned char cnum[256]; +- q931_get_number(cnum, sizeof(cnum), ie->data + 1, len - 3); +- pri_message(pri, "%c Called Sub-Address (len=%2d) [ Ext: %d Type: %s (%d) O: %d '%s' ]\n", +- prefix, len, ie->data[0] >> 7, ++ ++ if (!(ie->data[0] & 0x70)) { ++ /* NSAP */ ++ q931_get_number(cnum, sizeof(cnum), ie->data + offset, datalen); ++ } else { ++ /* User Specified */ ++ q931_get_subaddr_specific(cnum, sizeof(cnum), ie->data + offset, datalen, ++ ie->data[0] & 0x08); ++ } ++ ++ pri_message(ctrl, ++ "%c %s Sub-Address (len=%2d) [ Ext: %d Type: %s (%d) O: %d '%s' ]\n", ++ prefix, named, len, ie->data[0] >> 7, + subaddrtype2str((ie->data[0] & 0x70) >> 4), (ie->data[0] & 0x70) >> 4, + (ie->data[0] & 0x08) >> 3, cnum); + } + +-static FUNC_DUMP(dump_calling_party_number) ++static void dump_called_party_number(int full_ie, struct pri *ctrl, q931_ie *ie, int len, char prefix) + { + unsigned char cnum[256]; ++ ++ q931_get_number(cnum, sizeof(cnum), ie->data + 1, len - 3); ++ pri_message(ctrl, "%c Called Number (len=%2d) [ Ext: %d TON: %s (%d) NPI: %s (%d) '%s' ]\n", ++ prefix, len, ie->data[0] >> 7, ton2str((ie->data[0] >> 4) & 0x07), (ie->data[0] >> 4) & 0x07, npi2str(ie->data[0] & 0x0f), ie->data[0] & 0x0f, cnum); ++} ++ ++static void dump_called_party_subaddr(int full_ie, struct pri *ctrl, q931_ie *ie, int len, char prefix) ++{ ++ dump_subaddr_helper(full_ie, ctrl, ie, 1 , len, len - 3, prefix, "Called"); ++} ++ ++static void dump_calling_party_number(int full_ie, struct pri *ctrl, q931_ie *ie, int len, char prefix) ++{ ++ unsigned char cnum[256]; + if (ie->data[0] & 0x80) + q931_get_number(cnum, sizeof(cnum), ie->data + 1, len - 3); + else + q931_get_number(cnum, sizeof(cnum), ie->data + 2, len - 4); +- pri_message(pri, "%c Calling Number (len=%2d) [ Ext: %d TON: %s (%d) NPI: %s (%d)\n", prefix, len, ie->data[0] >> 7, ton2str((ie->data[0] >> 4) & 0x07), (ie->data[0] >> 4) & 0x07, npi2str(ie->data[0] & 0x0f), ie->data[0] & 0x0f); ++ pri_message(ctrl, "%c Calling Number (len=%2d) [ Ext: %d TON: %s (%d) NPI: %s (%d)\n", prefix, len, ie->data[0] >> 7, ton2str((ie->data[0] >> 4) & 0x07), (ie->data[0] >> 4) & 0x07, npi2str(ie->data[0] & 0x0f), ie->data[0] & 0x0f); + if (ie->data[0] & 0x80) +- pri_message(pri, "%c Presentation: %s (%d) '%s' ]\n", prefix, pri_pres2str(0), 0, cnum); ++ pri_message(ctrl, "%c Presentation: %s (%d) '%s' ]\n", prefix, pri_pres2str(0), 0, cnum); + else +- pri_message(pri, "%c Presentation: %s (%d) '%s' ]\n", prefix, pri_pres2str(ie->data[1] & 0x7f), ie->data[1] & 0x7f, cnum); ++ pri_message(ctrl, "%c Presentation: %s (%d) '%s' ]\n", prefix, pri_pres2str(ie->data[1] & 0x7f), ie->data[1] & 0x7f, cnum); + } + +-static FUNC_DUMP(dump_calling_party_subaddr) ++static void dump_calling_party_subaddr(int full_ie, struct pri *ctrl, q931_ie *ie, int len, char prefix) + { +- unsigned char cnum[256]; +- q931_get_number(cnum, sizeof(cnum), ie->data + 1, len - 3); +- pri_message(pri, "%c Calling Sub-Address (len=%2d) [ Ext: %d Type: %s (%d) O: %d '%s' ]\n", +- prefix, len, ie->data[0] >> 7, +- subaddrtype2str((ie->data[0] & 0x70) >> 4), (ie->data[0] & 0x70) >> 4, +- (ie->data[0] & 0x08) >> 3, cnum); ++ dump_subaddr_helper(full_ie, ctrl, ie, 1 , len, len - 3, prefix, "Calling"); + } + +-static FUNC_DUMP(dump_redirecting_number) ++static void dump_calling_party_category(int full_ie, struct pri *ctrl, q931_ie *ie, int len, char prefix) + { ++ pri_message(ctrl, "%c Calling Party Category (len=%2d) [ Ext: %d Cat: %s (%d) ]\n", ++ prefix, len, ie->data[0] >> 7, cpc2str(ie->data[0] & 0x0F), ie->data[0] & 0x0F); ++} ++ ++static void dump_redirecting_number(int full_ie, struct pri *ctrl, q931_ie *ie, int len, char prefix) ++{ + unsigned char cnum[256]; + int i = 0; + /* To follow Q.931 (4.5.1), we must search for start of octet 4 by +@@ -986,172 +1694,369 @@ + do { + switch(i) { + case 0: /* Octet 3 */ +- pri_message(pri, "%c Redirecting Number (len=%2d) [ Ext: %d TON: %s (%d) NPI: %s (%d)", ++ pri_message(ctrl, "%c Redirecting Number (len=%2d) [ Ext: %d TON: %s (%d) NPI: %s (%d)", + prefix, len, ie->data[0] >> 7, ton2str((ie->data[0] >> 4) & 0x07), (ie->data[0] >> 4) & 0x07, npi2str(ie->data[0] & 0x0f), ie->data[0] & 0x0f); + break; + case 1: /* Octet 3a */ +- pri_message(pri, "\n%c Ext: %d Presentation: %s (%d)", ++ pri_message(ctrl, "\n%c Ext: %d Presentation: %s (%d)", + prefix, ie->data[1] >> 7, pri_pres2str(ie->data[1] & 0x7f), ie->data[1] & 0x7f); + break; + case 2: /* Octet 3b */ +- pri_message(pri, "\n%c Ext: %d Reason: %s (%d)", ++ pri_message(ctrl, "\n%c Ext: %d Reason: %s (%d)", + prefix, ie->data[2] >> 7, redirection_reason2str(ie->data[2] & 0x7f), ie->data[2] & 0x7f); + break; + } +- } +- while(!(ie->data[i++]& 0x80)); ++ } while(!(ie->data[i++]& 0x80)); + q931_get_number(cnum, sizeof(cnum), ie->data + i, ie->len - i); +- pri_message(pri, " '%s' ]\n", cnum); ++ pri_message(ctrl, " '%s' ]\n", cnum); + } + +-static FUNC_DUMP(dump_connected_number) ++static void dump_redirection_number(int full_ie, struct pri *ctrl, q931_ie *ie, int len, char prefix) + { + unsigned char cnum[256]; + int i = 0; + /* To follow Q.931 (4.5.1), we must search for start of octet 4 by + walking through all bytes until one with ext bit (8) set to 1 */ + do { ++ switch (i) { ++ case 0: /* Octet 3 */ ++ pri_message(ctrl, ++ "%c Redirection Number (len=%2d) [ Ext: %d TON: %s (%d) NPI: %s (%d)", ++ prefix, len, ie->data[0] >> 7, ++ ton2str((ie->data[0] >> 4) & 0x07), (ie->data[0] >> 4) & 0x07, ++ npi2str(ie->data[0] & 0x0f), ie->data[0] & 0x0f); ++ break; ++ case 1: /* Octet 3a */ ++ pri_message(ctrl, "\n%c Ext: %d Presentation: %s (%d)", ++ prefix, ie->data[1] >> 7, pri_pres2str(ie->data[1] & 0x7f), ie->data[1] & 0x7f); ++ break; ++ } ++ } while (!(ie->data[i++] & 0x80)); ++ q931_get_number(cnum, sizeof(cnum), ie->data + i, ie->len - i); ++ pri_message(ctrl, " '%s' ]\n", cnum); ++} ++ ++static int receive_connected_number(int full_ie, struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, int len) ++{ ++ int i = 0; ++ ++ call->connected_number_in_message = 1; ++ call->remote_id.number.valid = 1; ++ call->remote_id.number.presentation = ++ PRI_PRES_ALLOWED | PRI_PRES_USER_NUMBER_UNSCREENED; ++ /* To follow Q.931 (4.5.1), we must search for start of octet 4 by ++ walking through all bytes until one with ext bit (8) set to 1 */ ++ do { ++ switch (i) { ++ case 0: ++ call->remote_id.number.plan = ie->data[i] & 0x7f; ++ break; ++ case 1: ++ /* Keep only the presentation and screening fields */ ++ call->remote_id.number.presentation = ++ ie->data[i] & (PRI_PRES_RESTRICTION | PRI_PRES_NUMBER_TYPE); ++ break; ++ } ++ } while (!(ie->data[i++] & 0x80)); ++ q931_get_number((unsigned char *) call->remote_id.number.str, sizeof(call->remote_id.number.str), ie->data + i, ie->len - i); ++ ++ return 0; ++} ++ ++static int transmit_connected_number(int full_ie, struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, int len, int order) ++{ ++ size_t datalen; ++ ++ if (!call->local_id.number.valid) { ++ return 0; ++ } ++ ++ datalen = strlen(call->local_id.number.str); ++ ie->data[0] = call->local_id.number.plan; ++ ie->data[1] = 0x80 | call->local_id.number.presentation; ++ memcpy(ie->data + 2, call->local_id.number.str, datalen); ++ return datalen + (2 + 2); ++} ++ ++static void dump_connected_number(int full_ie, struct pri *ctrl, q931_ie *ie, int len, char prefix) ++{ ++ unsigned char cnum[256]; ++ int i = 0; ++ /* To follow Q.931 (4.5.1), we must search for start of octet 4 by ++ walking through all bytes until one with ext bit (8) set to 1 */ ++ do { + switch(i) { + case 0: /* Octet 3 */ +- pri_message(pri, "%c Connected Number (len=%2d) [ Ext: %d TON: %s (%d) NPI: %s (%d)", ++ pri_message(ctrl, "%c Connected Number (len=%2d) [ Ext: %d TON: %s (%d) NPI: %s (%d)", + prefix, len, ie->data[0] >> 7, ton2str((ie->data[0] >> 4) & 0x07), (ie->data[0] >> 4) & 0x07, npi2str(ie->data[0] & 0x0f), ie->data[0] & 0x0f); + break; + case 1: /* Octet 3a */ +- pri_message(pri, "\n%c Ext: %d Presentation: %s (%d)", ++ pri_message(ctrl, "\n%c Ext: %d Presentation: %s (%d)", + prefix, ie->data[1] >> 7, pri_pres2str(ie->data[1] & 0x7f), ie->data[1] & 0x7f); + break; + } +- } +- while(!(ie->data[i++]& 0x80)); ++ } while(!(ie->data[i++]& 0x80)); + q931_get_number(cnum, sizeof(cnum), ie->data + i, ie->len - i); +- pri_message(pri, " '%s' ]\n", cnum); ++ pri_message(ctrl, " '%s' ]\n", cnum); + } + ++static int receive_connected_subaddr(int full_ie, struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, int len) ++{ ++ if (len < 3) { ++ return -1; ++ } + +-static FUNC_RECV(receive_redirecting_number) ++ return receive_subaddr_helper(full_ie, ctrl, &call->remote_id.subaddress, msgtype, ie, ++ 1, len - 3); ++} ++ ++static int transmit_connected_subaddr(int full_ie, struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, int len, int order) + { ++ return transmit_subaddr_helper(full_ie, ctrl, &call->local_id.subaddress, msgtype, ie, ++ 1, len, order); ++} ++ ++static void dump_connected_subaddr(int full_ie, struct pri *ctrl, q931_ie *ie, int len, char prefix) ++{ ++ dump_subaddr_helper(full_ie, ctrl, ie, 1 , len, len - 3, prefix, "Connected"); ++} ++ ++static int receive_redirecting_number(int full_ie, struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, int len) ++{ + int i = 0; + ++ call->redirecting_number_in_message = 1; ++ call->redirecting.from.number.valid = 1; ++ call->redirecting.from.number.presentation = ++ PRI_PRES_ALLOWED | PRI_PRES_USER_NUMBER_UNSCREENED; ++ call->redirecting.reason = PRI_REDIR_UNKNOWN; + /* To follow Q.931 (4.5.1), we must search for start of octet 4 by + walking through all bytes until one with ext bit (8) set to 1 */ + do { +- switch(i) { ++ switch (i) { + case 0: +- call->redirectingplan = ie->data[i] & 0x7f; ++ call->redirecting.from.number.plan = ie->data[i] & 0x7f; + break; + case 1: +- call->redirectingpres = ie->data[i] & 0x7f; ++ /* Keep only the presentation and screening fields */ ++ call->redirecting.from.number.presentation = ++ ie->data[i] & (PRI_PRES_RESTRICTION | PRI_PRES_NUMBER_TYPE); + break; + case 2: +- call->redirectingreason = ie->data[i] & 0x0f; ++ call->redirecting.reason = ie->data[i] & 0x0f; + break; + } +- } +- while(!(ie->data[i++] & 0x80)); +- q931_get_number((unsigned char *) call->redirectingnum, sizeof(call->redirectingnum), ie->data + i, ie->len - i); ++ } while (!(ie->data[i++] & 0x80)); ++ q931_get_number((unsigned char *) call->redirecting.from.number.str, sizeof(call->redirecting.from.number.str), ie->data + i, ie->len - i); + return 0; + } + +-static FUNC_SEND(transmit_redirecting_number) ++static int transmit_redirecting_number(int full_ie, struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, int len, int order) + { ++ size_t datalen; ++ + if (order > 1) + return 0; +- if (call->redirectingnum && *call->redirectingnum) { +- ie->data[0] = call->redirectingplan; +- ie->data[1] = call->redirectingpres; +- ie->data[2] = (call->redirectingreason & 0x0f) | 0x80; +- memcpy(ie->data + 3, call->redirectingnum, strlen(call->redirectingnum)); +- return strlen(call->redirectingnum) + 3 + 2; ++ if (!call->redirecting.from.number.valid) { ++ return 0; + } +- return 0; ++ ++ datalen = strlen(call->redirecting.from.number.str); ++ ie->data[0] = call->redirecting.from.number.plan; ++#if 1 ++ /* ETSI and Q.952 do not define the screening field */ ++ ie->data[1] = call->redirecting.from.number.presentation & PRI_PRES_RESTRICTION; ++#else ++ /* Q.931 defines the screening field */ ++ ie->data[1] = call->redirecting.from.number.presentation; ++#endif ++ ie->data[2] = (call->redirecting.reason & 0x0f) | 0x80; ++ memcpy(ie->data + 3, call->redirecting.from.number.str, datalen); ++ return datalen + (3 + 2); + } + +-static FUNC_DUMP(dump_redirecting_subaddr) ++static void dump_redirecting_subaddr(int full_ie, struct pri *ctrl, q931_ie *ie, int len, char prefix) + { +- unsigned char cnum[256]; +- q931_get_number(cnum, sizeof(cnum), ie->data + 2, len - 4); +- pri_message(pri, "%c Redirecting Sub-Address (len=%2d) [ Ext: %d Type: %s (%d) O: %d '%s' ]\n", +- prefix, len, ie->data[0] >> 7, +- subaddrtype2str((ie->data[0] & 0x70) >> 4), (ie->data[0] & 0x70) >> 4, +- (ie->data[0] & 0x08) >> 3, cnum); ++ dump_subaddr_helper(full_ie, ctrl, ie, 2, len, len - 4, prefix, "Redirecting"); + } + +-static FUNC_RECV(receive_calling_party_subaddr) ++static int receive_redirection_number(int full_ie, struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, int len) + { +- /* copy digits to call->callingsubaddr */ +- q931_get_number((unsigned char *) call->callingsubaddr, sizeof(call->callingsubaddr), ie->data + 1, len - 3); ++ int i = 0; ++ ++ call->redirection_number.valid = 1; ++ call->redirection_number.presentation = ++ PRI_PRES_ALLOWED | PRI_PRES_USER_NUMBER_UNSCREENED; ++ /* To follow Q.931 (4.5.1), we must search for start of octet 4 by ++ walking through all bytes until one with ext bit (8) set to 1 */ ++ do { ++ switch (i) { ++ case 0: ++ call->redirection_number.plan = ie->data[i] & 0x7f; ++ break; ++ case 1: ++ /* Keep only the presentation and screening fields */ ++ call->redirection_number.presentation = ++ ie->data[i] & (PRI_PRES_RESTRICTION | PRI_PRES_NUMBER_TYPE); ++ break; ++ } ++ } while (!(ie->data[i++] & 0x80)); ++ q931_get_number((unsigned char *) call->redirection_number.str, sizeof(call->redirection_number.str), ie->data + i, ie->len - i); + return 0; + } + +-static FUNC_RECV(receive_called_party_number) ++static int transmit_redirection_number(int full_ie, struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, int len, int order) + { +- /* copy digits to call->callednum */ +- q931_get_number((unsigned char *) call->callednum, sizeof(call->callednum), ie->data + 1, len - 3); +- call->calledplan = ie->data[0] & 0x7f; +- return 0; ++ size_t datalen; ++ ++ if (order > 1) { ++ return 0; ++ } ++ if (!call->redirection_number.valid) { ++ return 0; ++ } ++ ++ datalen = strlen(call->redirection_number.str); ++ ie->data[0] = call->redirection_number.plan; ++ ie->data[1] = (call->redirection_number.presentation & PRI_PRES_RESTRICTION) | 0x80; ++ memcpy(ie->data + 2, call->redirection_number.str, datalen); ++ return datalen + (2 + 2); + } + +-static FUNC_SEND(transmit_called_party_number) ++static int receive_calling_party_subaddr(int full_ie, struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, int len) + { +- ie->data[0] = 0x80 | call->calledplan; +- if (*call->callednum) +- memcpy(ie->data + 1, call->callednum, strlen(call->callednum)); +- return strlen(call->callednum) + 3; ++ if (len < 3) { ++ return -1; ++ } ++ ++ return receive_subaddr_helper(full_ie, ctrl, &call->remote_id.subaddress, msgtype, ie, ++ 1, len - 3); + } + +-static FUNC_RECV(receive_calling_party_number) ++static int transmit_calling_party_subaddr(int full_ie, struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, int len, int order) + { +- u_int8_t *data; +- size_t length; +- +- if (ie->data[0] & 0x80) { +- data = ie->data + 1; +- length = len - 3; +- call->callerpres = 0; /* PI presentation allowed SI user-provided, not screened */ +- } else { +- data = ie->data + 2; +- length = len - 4; +- call->callerpres = ie->data[1] & 0x7f; ++ return transmit_subaddr_helper(full_ie, ctrl, &call->local_id.subaddress, msgtype, ie, ++ 1, len, order); ++} ++ ++static int receive_called_party_subaddr(int full_ie, struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, int len) ++{ ++ if (len < 3) { ++ return -1; + } ++ return receive_subaddr_helper(full_ie, ctrl, &call->called.subaddress, msgtype, ie, 1, ++ len - 3); ++} + +- if (call->callerpres == PRES_ALLOWED_NETWORK_NUMBER || +- call->callerpres == PRES_PROHIB_NETWORK_NUMBER) { +- q931_get_number((u_int8_t *)call->callerani, sizeof(call->callerani), data, length); +- call->callerplanani = ie->data[0] & 0x7f; ++static int transmit_called_party_subaddr(int full_ie, struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, int len, int order) ++{ ++ return transmit_subaddr_helper(full_ie, ctrl, &call->called.subaddress, msgtype, ie, ++ 1, len, order); ++} + +- if (!*call->callernum) { /*Copy ANI to CallerID if CallerID is not already set */ +- libpri_copy_string(call->callernum, call->callerani, sizeof(call->callernum)); +- call->callerplan = call->callerplanani; ++static int receive_called_party_number(int full_ie, struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, int len) ++{ ++ size_t called_len; ++ size_t max_len; ++ char *called_end; ++ ++ if (len < 3) { ++ return -1; ++ } ++ ++ call->called.number.valid = 1; ++ call->called.number.plan = ie->data[0] & 0x7f; ++ if (msgtype == Q931_SETUP) { ++ q931_get_number((unsigned char *) call->called.number.str, ++ sizeof(call->called.number.str), ie->data + 1, len - 3); ++ } else if (call->ourcallstate == Q931_CALL_STATE_OVERLAP_RECEIVING) { ++ /* ++ * Since we are receiving overlap digits now, we need to append ++ * them to any previously received digits in call->called.number.str. ++ */ ++ called_len = strlen(call->called.number.str); ++ called_end = call->called.number.str + called_len; ++ max_len = (sizeof(call->called.number.str) - 1) - called_len; ++ if (max_len < len - 3) { ++ called_len = max_len; ++ } else { ++ called_len = len - 3; + } +- +- } else { +- q931_get_number((u_int8_t *)call->callernum, sizeof(call->callernum), data, length); +- call->callerplan = ie->data[0] & 0x7f; ++ strncat(called_end, (char *) ie->data + 1, called_len); + } + ++ q931_get_number((unsigned char *) call->overlap_digits, sizeof(call->overlap_digits), ++ ie->data + 1, len - 3); + return 0; + } + +-static FUNC_SEND(transmit_calling_party_number) ++static int transmit_called_party_number(int full_ie, struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, int len, int order) + { +- ie->data[0] = call->callerplan; +- ie->data[1] = 0x80 | call->callerpres; +- if (*call->callernum) +- memcpy(ie->data + 2, call->callernum, strlen(call->callernum)); +- return strlen(call->callernum) + 4; ++ size_t datalen; ++ ++ if (!call->called.number.valid) { ++ return 0; ++ } ++ ++ datalen = strlen(call->overlap_digits); ++ ie->data[0] = 0x80 | call->called.number.plan; ++ memcpy(ie->data + 1, call->overlap_digits, datalen); ++ return datalen + (1 + 2); + } + +-static FUNC_DUMP(dump_user_user) ++static int receive_calling_party_number(int full_ie, struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, int len) + { ++ int i = 0; ++ ++ call->remote_id.number.valid = 1; ++ call->remote_id.number.presentation = ++ PRI_PRES_ALLOWED | PRI_PRES_USER_NUMBER_UNSCREENED; ++ /* To follow Q.931 (4.5.1), we must search for start of octet 4 by ++ walking through all bytes until one with ext bit (8) set to 1 */ ++ do { ++ switch (i) { ++ case 0: ++ call->remote_id.number.plan = ie->data[i] & 0x7f; ++ break; ++ case 1: ++ /* Keep only the presentation and screening fields */ ++ call->remote_id.number.presentation = ++ ie->data[i] & (PRI_PRES_RESTRICTION | PRI_PRES_NUMBER_TYPE); ++ break; ++ } ++ } while (!(ie->data[i++] & 0x80)); ++ q931_get_number((unsigned char *) call->remote_id.number.str, ++ sizeof(call->remote_id.number.str), ie->data + i, ie->len - i); ++ ++ return 0; ++} ++ ++static int transmit_calling_party_number(int full_ie, struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, int len, int order) ++{ ++ size_t datalen; ++ ++ if (!call->local_id.number.valid) { ++ return 0; ++ } ++ ++ datalen = strlen(call->local_id.number.str); ++ ie->data[0] = call->local_id.number.plan; ++ ie->data[1] = 0x80 | call->local_id.number.presentation; ++ memcpy(ie->data + 2, call->local_id.number.str, datalen); ++ return datalen + (2 + 2); ++} ++ ++static void dump_user_user(int full_ie, struct pri *ctrl, q931_ie *ie, int len, char prefix) ++{ + int x; +- pri_message(pri, "%c User-User Information (len=%2d) [", prefix, len); ++ pri_message(ctrl, "%c User-User Information (len=%2d) [", prefix, len); + for (x=0;x<ie->len;x++) +- pri_message(pri, " %02x", ie->data[x] & 0x7f); +- pri_message(pri, " ]\n"); ++ pri_message(ctrl, " %02x", ie->data[x] & 0x7f); ++ pri_message(ctrl, " ]\n"); + } + + +-static FUNC_RECV(receive_user_user) ++static int receive_user_user(int full_ie, struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, int len) + { + call->useruserprotocoldisc = ie->data[0] & 0xff; + if (call->useruserprotocoldisc == 4) /* IA5 */ +@@ -1159,7 +2064,7 @@ + return 0; + } + +-static FUNC_SEND(transmit_user_user) ++static int transmit_user_user(int full_ie, struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, int len, int order) + { + int datalen = strlen(call->useruserinfo); + if (datalen > 0) { +@@ -1180,6 +2085,29 @@ + return 0; + } + ++static void dump_change_status(int full_ie, struct pri *ctrl, q931_ie *ie, int len, char prefix) ++{ ++ int x; ++ ++ pri_message(ctrl, "%c Change Status Information (len=%2d) [", prefix, len); ++ for (x=0; x<ie->len; x++) { ++ pri_message(ctrl, " %02x", ie->data[x] & 0x7f); ++ } ++ pri_message(ctrl, " ]\n"); ++} ++ ++static int receive_change_status(int full_ie, struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, int len) ++{ ++ call->changestatus = ie->data[0] & 0x0f; ++ return 0; ++} ++ ++static int transmit_change_status(int full_ie, struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, int len, int order) ++{ ++ ie->data[0] = 0xc0 | call->changestatus; ++ return 3; ++} ++ + static char *prog2str(int prog) + { + static struct msgtype progs[] = { +@@ -1222,47 +2150,83 @@ + return code2str(loc, locs, sizeof(locs) / sizeof(locs[0])); + } + +-static FUNC_DUMP(dump_progress_indicator) ++static void dump_progress_indicator(int full_ie, struct pri *ctrl, q931_ie *ie, int len, char prefix) + { +- pri_message(pri, "%c Progress Indicator (len=%2d) [ Ext: %d Coding: %s (%d) 0: %d Location: %s (%d)\n", ++ pri_message(ctrl, "%c Progress Indicator (len=%2d) [ Ext: %d Coding: %s (%d) 0: %d Location: %s (%d)\n", + prefix, len, ie->data[0] >> 7, coding2str((ie->data[0] & 0x60) >> 5), (ie->data[0] & 0x60) >> 5, + (ie->data[0] & 0x10) >> 4, loc2str(ie->data[0] & 0xf), ie->data[0] & 0xf); +- pri_message(pri, "%c Ext: %d Progress Description: %s (%d) ]\n", ++ pri_message(ctrl, "%c Ext: %d Progress Description: %s (%d) ]\n", + prefix, ie->data[1] >> 7, prog2str(ie->data[1] & 0x7f), ie->data[1] & 0x7f); + } + +-static FUNC_RECV(receive_display) ++static int receive_display(int full_ie, struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, int len) + { + unsigned char *data; ++ ++ switch (msgtype) { ++ case Q931_SETUP: ++ case Q931_CONNECT: ++ /* ++ * Only keep the display message on SETUP and CONNECT messages ++ * as the remote name. ++ */ ++ break; ++ default: ++ return 0; ++ } ++ ++ call->remote_id.name.valid = 1; ++ + data = ie->data; + if (data[0] & 0x80) { + /* Skip over character set */ + data++; + len--; + } +- q931_get_number((unsigned char *) call->callername, sizeof(call->callername), data, len - 2); ++ call->remote_id.name.char_set = PRI_CHAR_SET_ISO8859_1; ++ ++ q931_get_number((unsigned char *) call->remote_id.name.str, sizeof(call->remote_id.name.str), data, len - 2); ++ if (call->remote_id.name.str[0]) { ++ call->remote_id.name.presentation = PRI_PRES_ALLOWED; ++ } else { ++ call->remote_id.name.presentation = PRI_PRES_RESTRICTED; ++ } + return 0; + } + +-static FUNC_SEND(transmit_display) ++static int transmit_display(int full_ie, struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, int len, int order) + { ++ size_t datalen; + int i; +- +- if ((pri->switchtype == PRI_SWITCH_QSIG) || +- ((pri->switchtype == PRI_SWITCH_EUROISDN_E1) && (pri->localtype == PRI_CPE)) || +- !call->callername[0]) +- return 0; + + i = 0; +- if(pri->switchtype != PRI_SWITCH_EUROISDN_E1) { ++ ++ if (!call->local_id.name.valid || !call->local_id.name.str[0]) { ++ return 0; ++ } ++ switch (ctrl->switchtype) { ++ case PRI_SWITCH_QSIG: ++ /* Q.SIG supports names */ ++ return 0; ++ case PRI_SWITCH_EUROISDN_E1: ++ case PRI_SWITCH_EUROISDN_T1: ++ if (ctrl->localtype == PRI_CPE) { ++ return 0; ++ } ++ break; ++ default: ++ /* Prefix name with character set indicator. */ + ie->data[0] = 0xb1; + ++i; ++ break; + } +- memcpy(ie->data + i, call->callername, strlen(call->callername)); +- return 2 + i + strlen(call->callername); ++ ++ datalen = strlen(call->local_id.name.str); ++ memcpy(ie->data + i, call->local_id.name.str, datalen); ++ return 2 + i + datalen; + } + +-static FUNC_RECV(receive_progress_indicator) ++static int receive_progress_indicator(int full_ie, struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, int len) + { + call->progloc = ie->data[0] & 0xf; + call->progcode = (ie->data[0] & 0x60) >> 5; +@@ -1298,153 +2262,204 @@ + call->progressmask |= PRI_PROG_INTERWORKING_NO_RELEASE_POST_ANSWER; + break; + default: +- pri_error(pri, "XXX Invalid Progress indicator value received: %02x\n",(ie->data[1] & 0x7f)); ++ pri_error(ctrl, "XXX Invalid Progress indicator value received: %02x\n",(ie->data[1] & 0x7f)); + break; + } + return 0; + } + +-static FUNC_SEND(transmit_facility) ++static void q931_apdu_timeout(void *data); ++ ++static int transmit_facility(int full_ie, struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, int len, int order) + { +- struct apdu_event *tmp; +- int i = 0; ++ struct apdu_event **prev; ++ struct apdu_event *cur; ++ int apdu_len; + +- for (tmp = call->apdus; tmp; tmp = tmp->next) { +- if ((tmp->message == msgtype) && !tmp->sent) ++ for (prev = &call->apdus, cur = call->apdus; ++ cur; ++ prev = &cur->next, cur = cur->next) { ++ if (!cur->sent && cur->message == msgtype) { + break; ++ } + } +- +- if (!tmp) /* No APDU found */ ++ if (!cur) { ++ /* No APDU found */ + return 0; ++ } + +- if (tmp->apdu_len > 235) { /* TODO: find out how much space we can use */ +- pri_message(pri, "Requested APDU (%d bytes) is too long\n", tmp->apdu_len); ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, "Adding facility ie contents to send in %s message:\n", ++ msg2str(msgtype)); ++ facility_decode_dump(ctrl, cur->apdu, cur->apdu_len); ++ } ++ ++ if (len < cur->apdu_len) { ++ pri_error(ctrl, ++ "Could not fit facility ie in message. Size needed:%d Available space:%d\n", ++ cur->apdu_len + 2, len); ++ ++ /* Remove APDU from list. */ ++ *prev = cur->next; ++ ++ if (cur->response.callback) { ++ /* Indicate to callback that the APDU had a problem getting sent. */ ++ cur->response.callback(APDU_CALLBACK_REASON_ERROR, ctrl, call, cur, NULL); ++ } ++ ++ free(cur); + return 0; + } +- +- memcpy(&ie->data[i], tmp->apdu, tmp->apdu_len); +- i += tmp->apdu_len; +- tmp->sent = 1; + +- return i + 2; ++ memcpy(ie->data, cur->apdu, cur->apdu_len); ++ apdu_len = cur->apdu_len; ++ cur->sent = 1; ++ ++ if (cur->response.callback && cur->response.timeout_time) { ++ int duration; ++ ++ if (0 < cur->response.timeout_time) { ++ /* Sender specified timeout duration. */ ++ duration = cur->response.timeout_time; ++ } else { ++ /* Sender wants to use the typical timeout duration. */ ++ duration = ctrl->timers[PRI_TIMER_T_RESPONSE]; ++ } ++ cur->timer = pri_schedule_event(ctrl, duration, q931_apdu_timeout, cur); ++ if (!cur->timer) { ++ /* Remove APDU from list. */ ++ *prev = cur->next; ++ ++ /* Indicate to callback that the APDU had a problem getting sent. */ ++ cur->response.callback(APDU_CALLBACK_REASON_ERROR, ctrl, call, cur, NULL); ++ ++ free(cur); ++ } ++ } else if (!cur->timer) { ++ /* Remove APDU from list. */ ++ *prev = cur->next; ++ free(cur); ++ } ++ ++ return apdu_len + 2; + } + +-static FUNC_RECV(receive_facility) ++static int receive_facility(int full_ie, struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, int len) + { +- int i = 0; +- int protocol, next_protocol; +- struct rose_component *comp = NULL; +- enum { +- Q932_STATE_NFE, /* Network facility extension */ +- Q932_STATE_NPP, /* Network protocol profile */ +- Q932_STATE_INTERPRETATION, /* Interpretation component */ +- Q932_STATE_SERVICE /* Service component(s) */ +- } state = Q932_STATE_SERVICE; +-#define Q932_HANDLE_PROC(component, my_state, name, handler) \ +- case component: \ +- if(state > my_state) { \ +- pri_error(pri, "!! %s component received in wrong place\n"); \ +- break; \ +- } \ +- state = my_state; \ +- if (pri->debug) \ +- pri_message(pri, "Handle Q.932 %s component\n", name); \ +- (handler)(pri, call, ie, comp->data, comp->len); \ +- break; +-#define Q932_HANDLE_NULL(component, my_state, name, handle) \ +- case component: \ +- if(state > my_state) { \ +- pri_error(pri, "!! %s component received in wrong place\n"); \ +- break; \ +- } \ +- state = my_state; \ +- if (pri->debug & PRI_DEBUG_APDU) \ +- pri_message(pri, "Q.932 %s component is not handled\n", name); \ +- break; +- +- if (ie->len < 1) ++ /* Delay processing facility ie's till after all other ie's are processed. */ ++ if (MAX_FACILITY_IES <= ctrl->facility.count) { ++ pri_message(ctrl, "!! Too many facility ie's to delay.\n"); + return -1; ++ } ++ /* Make sure we have enough room for the protocol profile ie octet(s) */ ++ if (ie->data + ie->len < ie->data + 2) { ++ return -1; ++ } + +- switch(next_protocol = protocol = (ie->data[i] & 0x1f)) { +- case Q932_PROTOCOL_CMIP: +- case Q932_PROTOCOL_ACSE: +- if (pri->debug & PRI_DEBUG_APDU) +- pri_message(pri, "!! Don't know how to handle Q.932 Protocol Profile of type 0x%X\n", protocol); ++ /* Save the facility ie location for delayed decode. */ ++ ctrl->facility.ie[ctrl->facility.count] = ie; ++ ctrl->facility.codeset[ctrl->facility.count] = Q931_IE_CODESET((unsigned) full_ie); ++ ++ctrl->facility.count; ++ return 0; ++} ++ ++static int process_facility(struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie) ++{ ++ struct fac_extension_header header; ++ struct rose_message rose; ++ const unsigned char *pos; ++ const unsigned char *end; ++ ++ pos = ie->data; ++ end = ie->data + ie->len; ++ ++ /* Make sure we have enough room for the protocol profile ie octet(s) */ ++ if (end < pos + 2) { + return -1; ++ } ++ switch (*pos & Q932_PROTOCOL_MASK) { ++ case Q932_PROTOCOL_ROSE: + case Q932_PROTOCOL_EXTENSIONS: +- state = Q932_STATE_NFE; +- next_protocol = Q932_PROTOCOL_ROSE; + break; +- case Q932_PROTOCOL_ROSE: +- break; + default: +- pri_error(pri, "!! Invalid Q.932 Protocol Profile of type 0x%X received\n", protocol); ++ case Q932_PROTOCOL_CMIP: ++ case Q932_PROTOCOL_ACSE: ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, ++ "!! Don't know how to handle Q.932 Protocol Profile type 0x%X\n", ++ *pos & Q932_PROTOCOL_MASK); ++ } + return -1; + } +- /* Service indicator octet - Just ignore for now */ +- if (!(ie->data[i] & 0x80)) +- i++; +- i++; ++ if (!(*pos & 0x80)) { ++ /* DMS-100 Service indicator octet - Just ignore for now */ ++ ++pos; ++ } ++ ++pos; + +- if (ie->len < 3) ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ asn1_dump(ctrl, pos, end); ++ } ++ ++ pos = fac_dec_extension_header(ctrl, pos, end, &header); ++ if (!pos) { + return -1; +- +- while ((i+1 < ie->len) && (&ie->data[i])) { +- comp = (struct rose_component*)&ie->data[i]; +- if (comp->type) { +- if (protocol == Q932_PROTOCOL_EXTENSIONS) { +- switch (comp->type) { +- Q932_HANDLE_NULL(COMP_TYPE_INTERPRETATION, Q932_STATE_INTERPRETATION, "Interpretation", NULL); +- Q932_HANDLE_NULL(COMP_TYPE_NFE, Q932_STATE_NFE, "Network facility extensions", NULL); +- Q932_HANDLE_NULL(COMP_TYPE_NETWORK_PROTOCOL_PROFILE, Q932_STATE_NPP, "Network protocol profile", NULL); +- default: +- protocol = next_protocol; +- break; +- } +- } +- switch (protocol) { +- case Q932_PROTOCOL_ROSE: +- switch (comp->type) { +- Q932_HANDLE_PROC(COMP_TYPE_INVOKE, Q932_STATE_SERVICE, "ROSE Invoke", rose_invoke_decode); +- Q932_HANDLE_PROC(COMP_TYPE_RETURN_RESULT, Q932_STATE_SERVICE, "ROSE return result", rose_return_result_decode); +- Q932_HANDLE_PROC(COMP_TYPE_RETURN_ERROR, Q932_STATE_SERVICE, "ROSE return error", rose_return_error_decode); +- Q932_HANDLE_PROC(COMP_TYPE_REJECT, Q932_STATE_SERVICE, "ROSE reject", rose_reject_decode); +- default: +- if (pri->debug & PRI_DEBUG_APDU) +- pri_message(pri, "Don't know how to handle ROSE component of type 0x%X\n", comp->type); +- break; +- } +- break; +- case Q932_PROTOCOL_CMIP: +- switch (comp->type) { +- default: +- if (pri->debug & PRI_DEBUG_APDU) +- pri_message(pri, "Don't know how to handle CMIP component of type 0x%X\n", comp->type); +- break; +- } +- break; +- case Q932_PROTOCOL_ACSE: +- switch (comp->type) { +- default: +- if (pri->debug & PRI_DEBUG_APDU) +- pri_message(pri, "Don't know how to handle ACSE component of type 0x%X\n", comp->type); +- break; +- } +- break; +- } ++ } ++ if (header.npp_present) { ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, ++ "!! Don't know how to handle Network Protocol Profile type 0x%X\n", ++ header.npp); + } +- i += (comp->len + 2); ++ return -1; + } +-#undef Q932_HANDLE + ++ pos = rose_decode(ctrl, pos, end, &rose); ++ if (!pos) { ++ return -1; ++ } ++ switch (rose.type) { ++ case ROSE_COMP_TYPE_INVOKE: ++ rose_handle_invoke(ctrl, call, msgtype, ie, &header, &rose.component.invoke); ++ break; ++ case ROSE_COMP_TYPE_RESULT: ++ rose_handle_result(ctrl, call, msgtype, ie, &header, &rose.component.result); ++ break; ++ case ROSE_COMP_TYPE_ERROR: ++ rose_handle_error(ctrl, call, msgtype, ie, &header, &rose.component.error); ++ break; ++ case ROSE_COMP_TYPE_REJECT: ++ rose_handle_reject(ctrl, call, msgtype, ie, &header, &rose.component.reject); ++ break; ++ default: ++ return -1; ++ } + return 0; + } + +-static FUNC_SEND(transmit_progress_indicator) ++static void q931_handle_facilities(struct pri *ctrl, q931_call *call, int msgtype) + { ++ unsigned idx; ++ unsigned codeset; ++ unsigned full_ie; ++ q931_ie *ie; ++ ++ for (idx = 0; idx < ctrl->facility.count; ++idx) { ++ ie = ctrl->facility.ie[idx]; ++ if (ctrl->debug & PRI_DEBUG_Q931_STATE) { ++ codeset = ctrl->facility.codeset[idx]; ++ full_ie = Q931_FULL_IE(codeset, ie->ie); ++ pri_message(ctrl, "-- Delayed processing IE %d (cs%d, %s)\n", ie->ie, codeset, ie2str(full_ie)); ++ } ++ process_facility(ctrl, call, msgtype, ie); ++ } ++} ++ ++static int transmit_progress_indicator(int full_ie, struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, int len, int order) ++{ + int code, mask; + /* Can't send progress indicator on GR-303 -- EVER! */ +- if (pri->subchannel && !pri->bri) ++ if (ctrl->subchannel && !ctrl->bri) + return 0; + if (call->progressmask > 0) { + if (call->progressmask & (mask = PRI_PROG_CALL_NOT_E2E_ISDN)) +@@ -1467,7 +2482,7 @@ + code = Q931_PROG_INTERWORKING_NO_RELEASE_POST_ANSWER; + else { + code = 0; +- pri_error(pri, "XXX Undefined progress bit: %x\n", call->progressmask); ++ pri_error(ctrl, "XXX Undefined progress bit: %x\n", call->progressmask); + } + if (code) { + ie->data[0] = 0x80 | (call->progcode << 5) | (call->progloc); +@@ -1479,82 +2494,139 @@ + /* Leave off */ + return 0; + } +-static FUNC_SEND(transmit_call_state) ++static int transmit_call_state(int full_ie, struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, int len, int order) + { +- if (call->ourcallstate > -1 ) { ++ ie->data[0] = Q931_CALL_STATE_NULL; ++ switch (call->ourcallstate) { ++ case Q931_CALL_STATE_NULL: ++ case Q931_CALL_STATE_CALL_INITIATED: ++ case Q931_CALL_STATE_OVERLAP_SENDING: ++ case Q931_CALL_STATE_OUTGOING_CALL_PROCEEDING: ++ case Q931_CALL_STATE_CALL_DELIVERED: ++ case Q931_CALL_STATE_CALL_PRESENT: ++ case Q931_CALL_STATE_CALL_RECEIVED: ++ case Q931_CALL_STATE_CONNECT_REQUEST: ++ case Q931_CALL_STATE_INCOMING_CALL_PROCEEDING: ++ case Q931_CALL_STATE_ACTIVE: ++ case Q931_CALL_STATE_DISCONNECT_REQUEST: ++ case Q931_CALL_STATE_DISCONNECT_INDICATION: ++ case Q931_CALL_STATE_SUSPEND_REQUEST: ++ case Q931_CALL_STATE_RESUME_REQUEST: ++ case Q931_CALL_STATE_RELEASE_REQUEST: ++ case Q931_CALL_STATE_CALL_ABORT: ++ case Q931_CALL_STATE_OVERLAP_RECEIVING: ++ case Q931_CALL_STATE_CALL_INDEPENDENT_SERVICE: ++ case Q931_CALL_STATE_RESTART_REQUEST: ++ case Q931_CALL_STATE_RESTART: + ie->data[0] = call->ourcallstate; +- return 3; ++ break; ++ case Q931_CALL_STATE_NOT_SET: ++ break; + } +- return 0; ++ return 3; + } + +-static FUNC_RECV(receive_call_state) ++static int receive_call_state(int full_ie, struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, int len) + { + call->sugcallstate = ie->data[0] & 0x3f; + return 0; + } + +-static char *callstate2str(int callstate) ++/*! ++ * \brief Convert the internal Q.931 call state to a string. ++ * ++ * \param callstate Internal Q.931 call state. ++ * ++ * \return String equivalent of the given Q.931 call state. ++ */ ++const char *q931_call_state_str(enum Q931_CALL_STATE callstate) + { + static struct msgtype callstates[] = { +- { 0, "Null" }, +- { 1, "Call Initiated" }, +- { 2, "Overlap sending" }, +- { 3, "Outgoing call Proceeding" }, +- { 4, "Call Delivered" }, +- { 6, "Call Present" }, +- { 7, "Call Received" }, +- { 8, "Connect Request" }, +- { 9, "Incoming Call Proceeding" }, +- { 10, "Active" }, +- { 11, "Disconnect Request" }, +- { 12, "Disconnect Indication" }, +- { 15, "Suspend Request" }, +- { 17, "Resume Request" }, +- { 19, "Release Request" }, +- { 22, "Call Abort" }, +- { 25, "Overlap Receiving" }, +- { 61, "Restart Request" }, +- { 62, "Restart" }, ++/* *INDENT-OFF* */ ++ { Q931_CALL_STATE_NULL, "Null" }, ++ { Q931_CALL_STATE_CALL_INITIATED, "Call Initiated" }, ++ { Q931_CALL_STATE_OVERLAP_SENDING, "Overlap Sending" }, ++ { Q931_CALL_STATE_OUTGOING_CALL_PROCEEDING, "Outgoing Call Proceeding" }, ++ { Q931_CALL_STATE_CALL_DELIVERED, "Call Delivered" }, ++ { Q931_CALL_STATE_CALL_PRESENT, "Call Present" }, ++ { Q931_CALL_STATE_CALL_RECEIVED, "Call Received" }, ++ { Q931_CALL_STATE_CONNECT_REQUEST, "Connect Request" }, ++ { Q931_CALL_STATE_INCOMING_CALL_PROCEEDING, "Incoming Call Proceeding" }, ++ { Q931_CALL_STATE_ACTIVE, "Active" }, ++ { Q931_CALL_STATE_DISCONNECT_REQUEST, "Disconnect Request" }, ++ { Q931_CALL_STATE_DISCONNECT_INDICATION, "Disconnect Indication" }, ++ { Q931_CALL_STATE_SUSPEND_REQUEST, "Suspend Request" }, ++ { Q931_CALL_STATE_RESUME_REQUEST, "Resume Request" }, ++ { Q931_CALL_STATE_RELEASE_REQUEST, "Release Request" }, ++ { Q931_CALL_STATE_CALL_ABORT, "Call Abort" }, ++ { Q931_CALL_STATE_OVERLAP_RECEIVING, "Overlap Receiving" }, ++ { Q931_CALL_STATE_CALL_INDEPENDENT_SERVICE, "Call Independent Service" }, ++ { Q931_CALL_STATE_RESTART_REQUEST, "Restart Request" }, ++ { Q931_CALL_STATE_RESTART, "Restart" }, ++ { Q931_CALL_STATE_NOT_SET, "Not set. Internal use only." }, ++/* *INDENT-ON* */ + }; +- return code2str(callstate, callstates, sizeof(callstates) / sizeof(callstates[0])); ++ return code2str(callstate, callstates, ARRAY_LEN(callstates)); + } + +-static FUNC_DUMP(dump_call_state) ++/*! ++ * \internal ++ * \brief Convert the Q.932 supplementary hold state to a string. ++ * ++ * \param state Q.932 supplementary hold state. ++ * ++ * \return String equivalent of the given hold state. ++ */ ++static const char *q931_hold_state_str(enum Q931_HOLD_STATE state) + { +- pri_message(pri, "%c Call State (len=%2d) [ Ext: %d Coding: %s (%d) Call state: %s (%d)\n", ++ static struct msgtype hold_states[] = { ++/* *INDENT-OFF* */ ++ { Q931_HOLD_STATE_IDLE, "Idle" }, ++ { Q931_HOLD_STATE_HOLD_REQ, "Hold Request" }, ++ { Q931_HOLD_STATE_HOLD_IND, "Hold Indication" }, ++ { Q931_HOLD_STATE_CALL_HELD, "Call Held" }, ++ { Q931_HOLD_STATE_RETRIEVE_REQ, "Retrieve Request" }, ++ { Q931_HOLD_STATE_RETRIEVE_IND, "Retrieve Indication" }, ++/* *INDENT-ON* */ ++ }; ++ return code2str(state, hold_states, ARRAY_LEN(hold_states)); ++} ++ ++static void dump_call_state(int full_ie, struct pri *ctrl, q931_ie *ie, int len, char prefix) ++{ ++ pri_message(ctrl, "%c Call State (len=%2d) [ Ext: %d Coding: %s (%d) Call state: %s (%d)\n", + prefix, len, ie->data[0] >> 7, coding2str((ie->data[0] & 0xC0) >> 6), (ie->data[0] & 0xC0) >> 6, +- callstate2str(ie->data[0] & 0x3f), ie->data[0] & 0x3f); ++ q931_call_state_str(ie->data[0] & 0x3f), ie->data[0] & 0x3f); + } + +-static FUNC_DUMP(dump_call_identity) ++static void dump_call_identity(int full_ie, struct pri *ctrl, q931_ie *ie, int len, char prefix) + { + int x; +- pri_message(pri, "%c Call Identity (len=%2d) [ ", prefix, len); ++ pri_message(ctrl, "%c Call Identity (len=%2d) [ ", prefix, len); + for (x=0;x<ie->len;x++) +- pri_message(pri, "0x%02X ", ie->data[x]); +- pri_message(pri, " ]\n"); ++ pri_message(ctrl, "0x%02X ", ie->data[x]); ++ pri_message(ctrl, " ]\n"); + } + +-static FUNC_DUMP(dump_time_date) ++static void dump_time_date(int full_ie, struct pri *ctrl, q931_ie *ie, int len, char prefix) + { +- pri_message(pri, "%c Time Date (len=%2d) [ ", prefix, len); ++ pri_message(ctrl, "%c Time Date (len=%2d) [ ", prefix, len); + if (ie->len > 0) +- pri_message(pri, "%02d", ie->data[0]); ++ pri_message(ctrl, "%02d", ie->data[0]); + if (ie->len > 1) +- pri_message(pri, "-%02d", ie->data[1]); ++ pri_message(ctrl, "-%02d", ie->data[1]); + if (ie->len > 2) +- pri_message(pri, "-%02d", ie->data[2]); ++ pri_message(ctrl, "-%02d", ie->data[2]); + if (ie->len > 3) +- pri_message(pri, " %02d", ie->data[3]); ++ pri_message(ctrl, " %02d", ie->data[3]); + if (ie->len > 4) +- pri_message(pri, ":%02d", ie->data[4]); ++ pri_message(ctrl, ":%02d", ie->data[4]); + if (ie->len > 5) +- pri_message(pri, ":%02d", ie->data[5]); +- pri_message(pri, " ]\n"); ++ pri_message(ctrl, ":%02d", ie->data[5]); ++ pri_message(ctrl, " ]\n"); + } + +-static FUNC_DUMP(dump_keypad_facility) ++static void dump_keypad_facility(int full_ie, struct pri *ctrl, q931_ie *ie, int len, char prefix) + { + char tmp[64]; + +@@ -1563,10 +2635,10 @@ + + memcpy(tmp, ie->data, ie->len); + tmp[ie->len] = '\0'; +- pri_message(pri, "%c Keypad Facility (len=%2d) [ %s ]\n", prefix, ie->len, tmp ); ++ pri_message(ctrl, "%c Keypad Facility (len=%2d) [ %s ]\n", prefix, ie->len, tmp ); + } + +-static FUNC_RECV(receive_keypad_facility) ++static int receive_keypad_facility(int full_ie, struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, int len) + { + int mylen; + +@@ -1584,27 +2656,19 @@ + return 0; + } + +-static FUNC_SEND(transmit_keypad_facility) ++static int transmit_keypad_facility(int full_ie, struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, int len, int order) + { + int sublen; + + sublen = strlen(call->keypad_digits); +- +- if (sublen > 32) { +- sublen = 32; +- call->keypad_digits[32] = '\0'; +- } +- + if (sublen) { +- libpri_copy_string((char *)ie->data, (char *)call->keypad_digits, sizeof(call->keypad_digits)); +- /* Make sure we clear the field */ +- call->keypad_digits[0] = '\0'; ++ libpri_copy_string((char *) ie->data, call->keypad_digits, sizeof(call->keypad_digits)); + return sublen + 2; + } else + return 0; + } + +-static FUNC_DUMP(dump_display) ++static void dump_display(int full_ie, struct pri *ctrl, q931_ie *ie, int len, char prefix) + { + int x, y; + char *buf = malloc(len + 1); +@@ -1618,7 +2682,7 @@ + for (y=x; x<ie->len; x++) + buf[x] = ie->data[x] & 0x7f; + buf[x] = '\0'; +- pri_message(pri, "%c Display (len=%2d) %s[ %s ]\n", prefix, ie->len, tmp, &buf[y]); ++ pri_message(ctrl, "%c Display (len=%2d) %s[ %s ]\n", prefix, ie->len, tmp, &buf[y]); + free(buf); + } + } +@@ -1626,10 +2690,10 @@ + #define CHECK_OVERFLOW(limit) \ + if (tmpptr - tmp + limit >= sizeof(tmp)) { \ + *tmpptr = '\0'; \ +- pri_message(pri, "%s", tmpptr = tmp); \ ++ pri_message(ctrl, "%s", tmpptr = tmp); \ + } + +-static void dump_ie_data(struct pri *pri, unsigned char *c, int len) ++static void dump_ie_data(struct pri *ctrl, unsigned char *c, int len) + { + static char hexs[16] = "0123456789ABCDEF"; + char tmp[1024], *tmpptr; +@@ -1665,47 +2729,54 @@ + if (lastascii) + *tmpptr++ = '\''; + *tmpptr = '\0'; +- pri_message(pri, "%s", tmp); ++ pri_message(ctrl, "%s", tmp); + } + +-static FUNC_DUMP(dump_facility) ++static void dump_facility(int full_ie, struct pri *ctrl, q931_ie *ie, int len, char prefix) + { +- int dataat = (ie->data[0] & 0x80) ? 1 : 2; +- pri_message(pri, "%c Facility (len=%2d, codeset=%d) [ ", prefix, len, Q931_IE_CODESET(full_ie)); +- dump_ie_data(pri, ie->data, ie->len); +- pri_message(NULL, " ]\n"); ++ pri_message(ctrl, "%c Facility (len=%2d, codeset=%d) [ ", prefix, len, Q931_IE_CODESET(full_ie)); ++ dump_ie_data(ctrl, ie->data, ie->len); ++ pri_message(ctrl, " ]\n"); ++#if 0 /* Lets not dump parse of facility contents here anymore. */ ++ /* ++ * The ASN.1 decode dump has already been done when the facility ie was added to the outgoing ++ * message or the ASN.1 decode dump will be done when the facility ie is processed on incoming ++ * messages. This dump is redundant and very noisy. ++ */ + if (ie->len > 1) { +- pri_message(pri, "PROTOCOL %02X\n", ie->data[0] & ASN1_TYPE_MASK); +- asn1_dump(pri, &ie->data[dataat], ie->len - dataat); ++ int dataat = (ie->data[0] & 0x80) ? 1 : 2; ++ ++ pri_message(ctrl, "PROTOCOL %02X\n", ie->data[0] & Q932_PROTOCOL_MASK); ++ asn1_dump(ctrl, ie->data + dataat, ie->data + ie->len); + } +- ++#endif /* Lets not dump parse of facility contents here anymore. */ + } + +-static FUNC_DUMP(dump_network_spec_fac) ++static void dump_network_spec_fac(int full_ie, struct pri *ctrl, q931_ie *ie, int len, char prefix) + { +- pri_message(pri, "%c Network-Specific Facilities (len=%2d) [ ", prefix, ie->len); ++ pri_message(ctrl, "%c Network-Specific Facilities (len=%2d) [ ", prefix, ie->len); + if (ie->data[0] == 0x00) { +- pri_message(pri, "%s", code2str(ie->data[1], facilities, sizeof(facilities) / sizeof(facilities[0]))); ++ pri_message(ctrl, "%s", code2str(ie->data[1], facilities, ARRAY_LEN(facilities))); + } + else +- dump_ie_data(pri, ie->data, ie->len); +- pri_message(pri, " ]\n"); ++ dump_ie_data(ctrl, ie->data, ie->len); ++ pri_message(ctrl, " ]\n"); + } + +-static FUNC_RECV(receive_network_spec_fac) ++static int receive_network_spec_fac(int full_ie, struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, int len) + { + return 0; + } + +-static FUNC_SEND(transmit_network_spec_fac) ++static int transmit_network_spec_fac(int full_ie, struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, int len, int order) + { + /* We are ready to transmit single IE only */ + if (order > 1) + return 0; + +- if (pri->nsf != PRI_NSF_NONE) { ++ if (ctrl->nsf != PRI_NSF_NONE) { + ie->data[0] = 0x00; +- ie->data[1] = pri->nsf; ++ ie->data[1] = ctrl->nsf; + return 4; + } + /* Leave off */ +@@ -1732,13 +2803,13 @@ + return code2str(cause, causeclasses, sizeof(causeclasses) / sizeof(causeclasses[0])); + } + +-static FUNC_DUMP(dump_cause) ++static void dump_cause(int full_ie, struct pri *ctrl, q931_ie *ie, int len, char prefix) + { + int x; +- pri_message(pri, "%c Cause (len=%2d) [ Ext: %d Coding: %s (%d) Spare: %d Location: %s (%d)\n", ++ pri_message(ctrl, "%c Cause (len=%2d) [ Ext: %d Coding: %s (%d) Spare: %d Location: %s (%d)\n", + prefix, len, ie->data[0] >> 7, coding2str((ie->data[0] & 0x60) >> 5), (ie->data[0] & 0x60) >> 5, + (ie->data[0] & 0x10) >> 4, loc2str(ie->data[0] & 0xf), ie->data[0] & 0xf); +- pri_message(pri, "%c Ext: %d Cause: %s (%d), class = %s (%d) ]\n", ++ pri_message(ctrl, "%c Ext: %d Cause: %s (%d), class = %s (%d) ]\n", + prefix, (ie->data[1] >> 7), pri_cause2str(ie->data[1] & 0x7f), ie->data[1] & 0x7f, + pri_causeclass2str((ie->data[1] & 0x7f) >> 4), (ie->data[1] & 0x7f) >> 4); + if (ie->len < 3) +@@ -1747,29 +2818,29 @@ + switch(ie->data[1] & 0x7f) { + case PRI_CAUSE_IE_NONEXIST: + for (x=2;x<ie->len;x++) +- pri_message(pri, "%c Cause data %d: %02x (%d, %s IE)\n", prefix, x-1, ie->data[x], ie->data[x], ie2str(ie->data[x])); ++ pri_message(ctrl, "%c Cause data %d: %02x (%d, %s IE)\n", prefix, x-1, ie->data[x], ie->data[x], ie2str(ie->data[x])); + break; + case PRI_CAUSE_WRONG_CALL_STATE: + for (x=2;x<ie->len;x++) +- pri_message(pri, "%c Cause data %d: %02x (%d, %s message)\n", prefix, x-1, ie->data[x], ie->data[x], msg2str(ie->data[x])); ++ pri_message(ctrl, "%c Cause data %d: %02x (%d, %s message)\n", prefix, x-1, ie->data[x], ie->data[x], msg2str(ie->data[x])); + break; + case PRI_CAUSE_RECOVERY_ON_TIMER_EXPIRE: +- pri_message(pri, "%c Cause data:", prefix); ++ pri_message(ctrl, "%c Cause data:", prefix); + for (x=2;x<ie->len;x++) +- pri_message(pri, " %02x", ie->data[x]); +- pri_message(pri, " (Timer T"); ++ pri_message(ctrl, " %02x", ie->data[x]); ++ pri_message(ctrl, " (Timer T"); + for (x=2;x<ie->len;x++) +- pri_message(pri, "%c", ((ie->data[x] >= ' ') && (ie->data[x] < 0x7f)) ? ie->data[x] : '.'); +- pri_message(pri, ")\n"); ++ pri_message(ctrl, "%c", ((ie->data[x] >= ' ') && (ie->data[x] < 0x7f)) ? ie->data[x] : '.'); ++ pri_message(ctrl, ")\n"); + break; + default: + for (x=2;x<ie->len;x++) +- pri_message(pri, "%c Cause data %d: %02x (%d)\n", prefix, x-1, ie->data[x], ie->data[x]); ++ pri_message(ctrl, "%c Cause data %d: %02x (%d)\n", prefix, x-1, ie->data[x], ie->data[x]); + break; + } + } + +-static FUNC_RECV(receive_cause) ++static int receive_cause(int full_ie, struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, int len) + { + call->causeloc = ie->data[0] & 0xf; + call->causecode = (ie->data[0] & 0x60) >> 5; +@@ -1777,7 +2848,7 @@ + return 0; + } + +-static FUNC_SEND(transmit_cause) ++static int transmit_cause(int full_ie, struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, int len, int order) + { + /* We are ready to transmit single IE only */ + if (order > 1) +@@ -1793,23 +2864,23 @@ + } + } + +-static FUNC_DUMP(dump_sending_complete) ++static void dump_sending_complete(int full_ie, struct pri *ctrl, q931_ie *ie, int len, char prefix) + { +- pri_message(pri, "%c Sending Complete (len=%2d)\n", prefix, len); ++ pri_message(ctrl, "%c Sending Complete (len=%2d)\n", prefix, len); + } + +-static FUNC_RECV(receive_sending_complete) ++static int receive_sending_complete(int full_ie, struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, int len) + { + /* We've got a "Complete" message: Exect no further digits. */ + call->complete = 1; + return 0; + } + +-static FUNC_SEND(transmit_sending_complete) ++static int transmit_sending_complete(int full_ie, struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, int len, int order) + { +- if ((pri->overlapdial && call->complete) || /* Explicit */ +- (!pri->overlapdial && ((pri->switchtype == PRI_SWITCH_EUROISDN_E1) || +- /* Implicit */ (pri->switchtype == PRI_SWITCH_EUROISDN_T1)))) { ++ if ((ctrl->overlapdial && call->complete) || /* Explicit */ ++ (!ctrl->overlapdial && ((ctrl->switchtype == PRI_SWITCH_EUROISDN_E1) || ++ /* Implicit */ (ctrl->switchtype == PRI_SWITCH_EUROISDN_T1)))) { + /* Include this single-byte IE */ + return 1; + } +@@ -1846,18 +2917,18 @@ + return code2str(info, notifies, sizeof(notifies) / sizeof(notifies[0])); + } + +-static FUNC_DUMP(dump_notify) ++static void dump_notify(int full_ie, struct pri *ctrl, q931_ie *ie, int len, char prefix) + { +- pri_message(pri, "%c Notification indicator (len=%2d): Ext: %d %s (%d)\n", prefix, len, ie->data[0] >> 7, notify2str(ie->data[0] & 0x7f), ie->data[0] & 0x7f); ++ pri_message(ctrl, "%c Notification indicator (len=%2d): Ext: %d %s (%d)\n", prefix, len, ie->data[0] >> 7, notify2str(ie->data[0] & 0x7f), ie->data[0] & 0x7f); + } + +-static FUNC_RECV(receive_notify) ++static int receive_notify(int full_ie, struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, int len) + { + call->notify = ie->data[0] & 0x7F; + return 0; + } + +-static FUNC_SEND(transmit_notify) ++static int transmit_notify(int full_ie, struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, int len, int order) + { + if (call->notify >= 0) { + ie->data[0] = 0x80 | call->notify; +@@ -1866,9 +2937,9 @@ + return 0; + } + +-static FUNC_DUMP(dump_shift) ++static void dump_shift(int full_ie, struct pri *ctrl, q931_ie *ie, int len, char prefix) + { +- pri_message(pri, "%c %sLocking Shift (len=%02d): Requested codeset %d\n", prefix, (full_ie & 8) ? "Non-" : "", len, full_ie & 7); ++ pri_message(ctrl, "%c %sLocking Shift (len=%02d): Requested codeset %d\n", prefix, (full_ie & 8) ? "Non-" : "", len, full_ie & 7); + } + + static char *lineinfo2str(int info) +@@ -1903,21 +2974,21 @@ + return code2str(info, lineinfo, sizeof(lineinfo) / sizeof(lineinfo[0])); + } + +-static FUNC_DUMP(dump_line_information) ++static void dump_line_information(int full_ie, struct pri *ctrl, q931_ie *ie, int len, char prefix) + { +- pri_message(pri, "%c Originating Line Information (len=%02d): %s (%d)\n", prefix, len, lineinfo2str(ie->data[0]), ie->data[0]); ++ pri_message(ctrl, "%c Originating Line Information (len=%02d): %s (%d)\n", prefix, len, lineinfo2str(ie->data[0]), ie->data[0]); + } + +-static FUNC_RECV(receive_line_information) ++static int receive_line_information(int full_ie, struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, int len) + { + call->ani2 = ie->data[0]; + return 0; + } + +-static FUNC_SEND(transmit_line_information) ++static int transmit_line_information(int full_ie, struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, int len, int order) + { + #if 0 /* XXX Is this IE possible for 4ESS only? XXX */ +- if(pri->switchtype == PRI_SWITCH_ATT4ESS) { ++ if(ctrl->switchtype == PRI_SWITCH_ATT4ESS) { + ie->data[0] = 0; + return 3; + } +@@ -1953,53 +3024,54 @@ + return code2str(type, gdtype, sizeof(gdtype) / sizeof(gdtype[0])); + } + +-static FUNC_DUMP(dump_generic_digits) ++static void dump_generic_digits(int full_ie, struct pri *ctrl, q931_ie *ie, int len, char prefix) + { + int encoding; + int type; + int idx; + int value; + if (len < 3) { +- pri_message(pri, "%c Generic Digits (len=%02d): Invalid length\n", prefix, len); ++ pri_message(ctrl, "%c Generic Digits (len=%02d): Invalid length\n", prefix, len); + return; + } + encoding = (ie->data[0] >> 5) & 7; + type = ie->data[0] & 0x1F; +- pri_message(pri, "%c Generic Digits (len=%02d): Encoding %s Type %s\n", prefix, len, gdencoding2str(encoding), gdtype2str(type)); ++ pri_message(ctrl, "%c Generic Digits (len=%02d): Encoding %s Type %s\n", prefix, len, gdencoding2str(encoding), gdtype2str(type)); + if (encoding == 3) { /* Binary */ +- pri_message(pri, "%c Don't know how to handle binary encoding\n"); ++ pri_message(ctrl, "%c Don't know how to handle binary encoding\n", ++ prefix); + return; + } + if (len == 3) /* No number information */ + return; +- pri_message(pri, "%c Digits: "); ++ pri_message(ctrl, "%c Digits: ", prefix); + value = 0; + for(idx = 3; idx < len; ++idx) { + switch(encoding) { + case 0: /* BCD even */ + case 1: /* BCD odd */ +- pri_message(pri, "%d", ie->data[idx-2] & 0x0f); ++ pri_message(ctrl, "%d", ie->data[idx-2] & 0x0f); + value = value * 10 + (ie->data[idx-2] & 0x0f); + if(!encoding || (idx+1 < len)) { /* Special handling for BCD odd */ +- pri_message(pri, "%d", (ie->data[idx-2] >> 4) & 0x0f); ++ pri_message(ctrl, "%d", (ie->data[idx-2] >> 4) & 0x0f); + value = value * 10 + ((ie->data[idx-2] >> 4) & 0x0f); + } + break; + case 2: /* IA5 */ +- pri_message(pri, "%c", ie->data[idx-2]); ++ pri_message(ctrl, "%c", ie->data[idx-2]); + value = value * 10 + ie->data[idx-2] - '0'; + break; + } + } + switch(type) { + case 4: /* Info Digits */ +- pri_message(pri, " - %s", lineinfo2str(value)); ++ pri_message(ctrl, " - %s", lineinfo2str(value)); + break; + } +- pri_message(pri, "\n"); ++ pri_message(ctrl, "\n"); + } + +-static FUNC_RECV(receive_generic_digits) ++static int receive_generic_digits(int full_ie, struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, int len) + { + int encoding; + int type; +@@ -2009,13 +3081,13 @@ + char number[260]; + + if (len < 3) { +- pri_error(pri, "Invalid length of Generic Digits IE\n"); ++ pri_error(ctrl, "Invalid length of Generic Digits IE\n"); + return -1; + } + encoding = (ie->data[0] >> 5) & 7; + type = ie->data[0] & 0x1F; + if (encoding == 3) { /* Binary */ +- pri_message(pri, "!! Unable to handle binary encoded Generic Digits IE\n"); ++ pri_message(ctrl, "!! Unable to handle binary encoded Generic Digits IE\n"); + return 0; + } + if (len == 3) /* No number information */ +@@ -2063,10 +3135,13 @@ + break; + #if 0 + case 5: /* Callid */ +- if (!call->callernum[0]) { +- memcpy(call->callernum, number, sizeof(call->callernum)-1); +- call->callerpres = 0; +- call->callerplan = 0; ++ if (!call->remote_id.number.valid) { ++ call->remote_id.number.valid = 1; ++ call->remote_id.number.presentation = ++ PRI_PRES_ALLOWED | PRI_PRES_USER_NUMBER_UNSCREENED; ++ call->remote_id.number.plan = PRI_UNKNOWN; ++ libpri_copy_string(call->remote_id.number.str, number, ++ sizeof(call->remote_id.number.str)); + } + break; + #endif +@@ -2074,13 +3149,13 @@ + return 0; + } + +-static FUNC_SEND(transmit_generic_digits) ++static int transmit_generic_digits(int full_ie, struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, int len, int order) + { + #if 0 /* XXX Is this IE possible for other switches? XXX */ + if (order > 1) + return 0; + +- if(pri->switchtype == PRI_SWITCH_NI1) { ++ if(ctrl->switchtype == PRI_SWITCH_NI1) { + ie->data[0] = 0x04; /* BCD even, Info Digits */ + ie->data[1] = 0x00; /* POTS */ + return 4; +@@ -2119,35 +3194,54 @@ + } + + +-static FUNC_DUMP(dump_signal) ++static void dump_signal(int full_ie, struct pri *ctrl, q931_ie *ie, int len, char prefix) + { +- pri_message(pri, "%c Signal (len=%02d): ", prefix, len); ++ pri_message(ctrl, "%c Signal (len=%02d): ", prefix, len); + if (len < 3) { +- pri_message(pri, "Invalid length\n"); ++ pri_message(ctrl, "Invalid length\n"); + return; + } +- pri_message(pri, "Signal %s (%d)\n", signal2str(ie->data[0]), ie->data[0]); ++ pri_message(ctrl, "Signal %s (%d)\n", signal2str(ie->data[0]), ie->data[0]); + } + +-static FUNC_DUMP(dump_transit_count) ++static void dump_transit_count(int full_ie, struct pri *ctrl, q931_ie *ie, int len, char prefix) + { + /* Defined in ECMA-225 */ +- pri_message(pri, "%c Transit Count (len=%02d): ", prefix, len); ++ pri_message(ctrl, "%c Transit Count (len=%02d): ", prefix, len); + if (len < 3) { +- pri_message(pri, "Invalid length\n"); ++ pri_message(ctrl, "Invalid length\n"); + return; + } +- pri_message(pri, "Count=%d (0x%02x)\n", ie->data[0] & 0x1f, ie->data[0] & 0x1f); ++ pri_message(ctrl, "Count=%d (0x%02x)\n", ie->data[0] & 0x1f, ie->data[0] & 0x1f); + } + ++static void dump_reverse_charging_indication(int full_ie, struct pri *ctrl, q931_ie *ie, int len, char prefix) ++{ ++ pri_message(ctrl, "%c Reverse Charging Indication (len=%02d): %d\n", prefix, len, ie->data[0] & 0x7); ++} + ++static int receive_reverse_charging_indication(int full_ie, struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, int len) ++{ ++ call->reversecharge = ie->data[0] & 0x7; ++ return 0; ++} ++ ++static int transmit_reverse_charging_indication(int full_ie, struct pri *ctrl, q931_call *call, int msgtype, q931_ie *ie, int len, int order) ++{ ++ if (call->reversecharge != PRI_REVERSECHARGE_NONE) { ++ ie->data[0] = 0x80 | (call->reversecharge & 0x7); ++ return 3; ++ } ++ return 0; ++} ++ + static struct ie ies[] = { + /* Codeset 0 - Common */ +- { 1, NATIONAL_CHANGE_STATUS, "Change Status" }, ++ { 1, NATIONAL_CHANGE_STATUS, "Change Status", dump_change_status, receive_change_status, transmit_change_status }, + { 0, Q931_LOCKING_SHIFT, "Locking Shift", dump_shift }, + { 0, Q931_BEARER_CAPABILITY, "Bearer Capability", dump_bearer_capability, receive_bearer_capability, transmit_bearer_capability }, + { 0, Q931_CAUSE, "Cause", dump_cause, receive_cause, transmit_cause }, +- { 1, Q931_CALL_STATE, "Call State", dump_call_state, receive_call_state, transmit_call_state }, ++ { 1, Q931_IE_CALL_STATE, "Call State", dump_call_state, receive_call_state, transmit_call_state }, + { 0, Q931_CHANNEL_IDENT, "Channel Identification", dump_channel_id, receive_channel_id, transmit_channel_id }, + { 0, Q931_PROGRESS_INDICATOR, "Progress Indicator", dump_progress_indicator, receive_progress_indicator, transmit_progress_indicator }, + { 0, Q931_NETWORK_SPEC_FAC, "Network-Specific Facilities", dump_network_spec_fac, receive_network_spec_fac, transmit_network_spec_fac }, +@@ -2157,11 +3251,11 @@ + { 1, Q931_BINARY_PARAMETERS, "Packet-layer Binary Parameters" }, + { 1, Q931_WINDOW_SIZE, "Packet-layer Window Size" }, + { 1, Q931_CLOSED_USER_GROUP, "Closed User Group" }, +- { 1, Q931_REVERSE_CHARGE_INDIC, "Reverse Charging Indication" }, ++ { 1, Q931_REVERSE_CHARGE_INDIC, "Reverse Charging Indication", dump_reverse_charging_indication, receive_reverse_charging_indication, transmit_reverse_charging_indication }, + { 1, Q931_CALLING_PARTY_NUMBER, "Calling Party Number", dump_calling_party_number, receive_calling_party_number, transmit_calling_party_number }, +- { 1, Q931_CALLING_PARTY_SUBADDR, "Calling Party Subaddress", dump_calling_party_subaddr, receive_calling_party_subaddr }, ++ { 1, Q931_CALLING_PARTY_SUBADDR, "Calling Party Subaddress", dump_calling_party_subaddr, receive_calling_party_subaddr, transmit_calling_party_subaddr }, + { 1, Q931_CALLED_PARTY_NUMBER, "Called Party Number", dump_called_party_number, receive_called_party_number, transmit_called_party_number }, +- { 1, Q931_CALLED_PARTY_SUBADDR, "Called Party Subaddress", dump_called_party_subaddr }, ++ { 1, Q931_CALLED_PARTY_SUBADDR, "Called Party Subaddress", dump_called_party_subaddr, receive_called_party_subaddr, transmit_called_party_subaddr }, + { 0, Q931_REDIRECTING_NUMBER, "Redirecting Number", dump_redirecting_number, receive_redirecting_number, transmit_redirecting_number }, + { 1, Q931_REDIRECTING_SUBADDR, "Redirecting Subaddress", dump_redirecting_subaddr }, + { 0, Q931_TRANSIT_NET_SELECT, "Transit Network Selection" }, +@@ -2170,7 +3264,7 @@ + { 0, Q931_HIGH_LAYER_COMPAT, "High-layer Compatibility" }, + { 1, Q931_PACKET_SIZE, "Packet Size" }, + { 0, Q931_IE_FACILITY, "Facility" , dump_facility, receive_facility, transmit_facility }, +- { 1, Q931_IE_REDIRECTION_NUMBER, "Redirection Number" }, ++ { 1, Q931_IE_REDIRECTION_NUMBER, "Redirection Number", dump_redirection_number, receive_redirection_number, transmit_redirection_number }, + { 1, Q931_IE_REDIRECTION_SUBADDR, "Redirection Subaddress" }, + { 1, Q931_IE_FEATURE_ACTIVATE, "Feature Activation" }, + { 1, Q931_IE_INFO_REQUEST, "Feature Request" }, +@@ -2187,15 +3281,18 @@ + { 1, Q931_IE_USER_USER, "User-User", dump_user_user, receive_user_user, transmit_user_user }, + { 1, Q931_IE_ESCAPE_FOR_EXT, "Escape for Extension" }, + { 1, Q931_IE_CALL_STATUS, "Call Status" }, +- { 1, Q931_IE_CHANGE_STATUS, "Change Status" }, ++ { 1, Q931_IE_CHANGE_STATUS, "Change Status", dump_change_status, receive_change_status, transmit_change_status }, + { 1, Q931_IE_CONNECTED_ADDR, "Connected Number", dump_connected_number }, +- { 1, Q931_IE_CONNECTED_NUM, "Connected Number", dump_connected_number }, ++ { 1, Q931_IE_CONNECTED_NUM, "Connected Number", dump_connected_number, receive_connected_number, transmit_connected_number }, ++ { 1, Q931_IE_CONNECTED_SUBADDR, "Connected Subaddress", dump_connected_subaddr, receive_connected_subaddr, transmit_connected_subaddr }, + { 1, Q931_IE_ORIGINAL_CALLED_NUMBER, "Original Called Number", dump_redirecting_number, receive_redirecting_number, transmit_redirecting_number }, + { 1, Q931_IE_USER_USER_FACILITY, "User-User Facility" }, + { 1, Q931_IE_UPDATE, "Update" }, + { 1, Q931_SENDING_COMPLETE, "Sending Complete", dump_sending_complete, receive_sending_complete, transmit_sending_complete }, + /* Codeset 4 - Q.SIG specific */ + { 1, QSIG_IE_TRANSIT_COUNT | Q931_CODESET(4), "Transit Count", dump_transit_count }, ++ /* Codeset 5 - National specific (ETSI PISN specific) */ ++ { 1, Q931_CALLING_PARTY_CATEGORY, "Calling Party Category", dump_calling_party_category }, + /* Codeset 6 - Network specific */ + { 1, Q931_IE_ORIGINATING_LINE_INFO, "Originating Line Information", dump_line_information, receive_line_information, transmit_line_information }, + { 1, Q931_IE_FACILITY | Q931_CODESET(6), "Facility", dump_facility, receive_facility, transmit_facility }, +@@ -2264,7 +3361,7 @@ + return 2 + ie->len; + } + +-static char *msg2str(int msg) ++const char *msg2str(int msg) + { + unsigned int x; + for (x=0;x<sizeof(msgs) / sizeof(msgs[0]); x++) +@@ -2273,35 +3370,64 @@ + return "Unknown Message Type"; + } + ++static char *maintenance_msg2str(int msg, int pd) ++{ ++ unsigned int x, max; ++ struct msgtype *m = NULL; ++ ++ if (pd == MAINTENANCE_PROTOCOL_DISCRIMINATOR_1) { ++ m = att_maintenance_msgs; ++ max = ARRAY_LEN(att_maintenance_msgs); ++ } else { ++ m = national_maintenance_msgs; ++ max = ARRAY_LEN(national_maintenance_msgs); ++ } ++ ++ for (x = 0; x < max; x++) { ++ if (m[x].msgnum == msg) { ++ return m[x].name; ++ } ++ } ++ return "Unknown Message Type"; ++} ++ ++/* Decode the call reference */ + static inline int q931_cr(q931_h *h) + { +- int cr = 0; ++ int cr; + int x; ++ + if (h->crlen > 3) { + pri_error(NULL, "Call Reference Length Too long: %d\n", h->crlen); +- return -1; ++ return Q931_DUMMY_CALL_REFERENCE; + } + switch (h->crlen) { +- case 2: +- for (x=0;x<h->crlen;x++) { +- cr <<= 8; +- cr |= h->crv[x]; +- } +- break; +- case 1: +- cr = h->crv[0]; +- if (cr & 0x80) { +- cr &= ~0x80; +- cr |= 0x8000; +- } +- break; +- default: +- pri_error(NULL, "Call Reference Length not supported: %d\n", h->crlen); ++ case 2: ++ cr = 0; ++ for (x = 0; x < h->crlen; ++x) { ++ cr <<= 8; ++ cr |= h->crv[x]; ++ } ++ break; ++ case 1: ++ cr = h->crv[0]; ++ if (cr & 0x80) { ++ cr &= ~0x80; ++ cr |= 0x8000; ++ } ++ break; ++ case 0: ++ cr = Q931_DUMMY_CALL_REFERENCE; ++ break; ++ default: ++ pri_error(NULL, "Call Reference Length not supported: %d\n", h->crlen); ++ cr = Q931_DUMMY_CALL_REFERENCE; ++ break; + } + return cr; + } + +-static inline void q931_dumpie(struct pri *pri, int codeset, q931_ie *ie, char prefix) ++static inline void q931_dumpie(struct pri *ctrl, int codeset, q931_ie *ie, char prefix) + { + unsigned int x; + int full_ie = Q931_FULL_IE(codeset, ie->ie); +@@ -2315,7 +3441,7 @@ + for (x = 0; x + 2 < ielen(ie); ++x) + buflen += sprintf(buf + buflen, " %02x", ie->data[x]); + } +- pri_message(pri, "%c [%02x%s]\n", prefix, ie->ie, buf); ++ pri_message(ctrl, "%c [%02x%s]\n", prefix, ie->ie, buf); + free(buf); + + /* Special treatment for shifts */ +@@ -2324,130 +3450,306 @@ + + base_ie = (((full_ie & ~0x7f) == Q931_FULL_IE(0, 0x80)) && ((full_ie & 0x70) != 0x20)) ? full_ie & ~0x0f : full_ie; + +- for (x=0;x<sizeof(ies) / sizeof(ies[0]); x++) ++ for (x = 0; x < ARRAY_LEN(ies); ++x) + if (ies[x].ie == base_ie) { + if (ies[x].dump) +- ies[x].dump(full_ie, pri, ie, ielen(ie), prefix); ++ ies[x].dump(full_ie, ctrl, ie, ielen(ie), prefix); + else +- pri_message(pri, "%c IE: %s (len = %d)\n", prefix, ies[x].name, ielen(ie)); ++ pri_message(ctrl, "%c IE: %s (len = %d)\n", prefix, ies[x].name, ielen(ie)); + return; + } + +- pri_error(pri, "!! %c Unknown IE %d (cs%d, len = %d)\n", prefix, Q931_IE_IE(base_ie), Q931_IE_CODESET(base_ie), ielen(ie)); ++ pri_error(ctrl, "!! %c Unknown IE %d (cs%d, len = %d)\n", prefix, Q931_IE_IE(base_ie), Q931_IE_CODESET(base_ie), ielen(ie)); + } + +-static q931_call *q931_getcall(struct pri *pri, int cr, int outboundnew) ++/*! ++ * \brief Initialize the call record. ++ * ++ * \param ctrl D channel controller. ++ * \param call Q.931 call leg. ++ * ++ * \note The call record is assumed to already be memset() to zero. ++ * ++ * \return Nothing ++ */ ++void q931_init_call_record(struct pri *ctrl, struct q931_call *call, int cr) + { +- q931_call *cur, *prev; ++ call->cr = cr; ++ call->slotmap = -1; ++ call->channelno = -1; ++ if (cr != Q931_DUMMY_CALL_REFERENCE) { ++ call->newcall = 1; ++ } ++ call->ourcallstate = Q931_CALL_STATE_NULL; ++ call->peercallstate = Q931_CALL_STATE_NULL; ++ call->sugcallstate = Q931_CALL_STATE_NOT_SET; ++ call->ri = -1; ++ call->transcapability = -1; ++ call->transmoderate = -1; ++ call->transmultiple = -1; ++ call->userl1 = -1; ++ call->userl2 = -1; ++ call->userl3 = -1; ++ call->rateadaption = -1; ++ call->progress = -1; ++ call->causecode = -1; ++ call->causeloc = -1; ++ call->cause = -1; ++ call->useruserprotocoldisc = -1; ++ call->aoc_units = -1; ++ call->changestatus = -1; ++ call->reversecharge = -1; ++ call->pri_winner = -1; ++ call->master_call = call; ++ q931_party_number_init(&call->redirection_number); ++ q931_party_address_init(&call->called); ++ q931_party_id_init(&call->local_id); ++ q931_party_id_init(&call->remote_id); ++ q931_party_redirecting_init(&call->redirecting); ++ ++ /* PRI is set to whoever called us */ ++ if (BRI_TE_PTMP(ctrl)) { ++ /* ++ * Point to the master to avoid stale pointer problems if ++ * the TEI is removed later. ++ */ ++ call->pri = PRI_MASTER(ctrl); ++ } else { ++ call->pri = ctrl; ++ } ++} ++ ++static q931_call *q931_getcall(struct pri *ctrl, int cr) ++{ ++ q931_call *cur; ++ q931_call *prev; + struct pri *master; + ++ if (cr == Q931_DUMMY_CALL_REFERENCE) { ++ return ctrl->dummy_call; ++ } ++ + /* Find the master - He has the call pool */ +- if (pri->master) +- master = pri->master; +- else +- master = pri; +- ++ master = PRI_MASTER(ctrl); ++ + cur = *master->callpool; + prev = NULL; +- while(cur) { +- if (cur->cr == cr) ++ while (cur) { ++ if (cur->cr == cr) { ++ /* Found existing call. */ ++ switch (ctrl->switchtype) { ++ case PRI_SWITCH_GR303_EOC: ++ case PRI_SWITCH_GR303_EOC_PATH: ++ case PRI_SWITCH_GR303_TMC: ++ case PRI_SWITCH_GR303_TMC_SWITCHING: ++ break; ++ default: ++ if (!ctrl->bri) { ++ /* PRI is set to whoever called us */ ++ cur->pri = ctrl; ++ } ++ break; ++ } + return cur; ++ } + prev = cur; + cur = cur->next; + } ++ + /* No call exists, make a new one */ +- if (pri->debug & PRI_DEBUG_Q931_STATE) +- pri_message(pri, "-- Making new call for cr %d\n", cr); +- +- if (!(cur = calloc(1, sizeof(*cur)))) ++ if (ctrl->debug & PRI_DEBUG_Q931_STATE) { ++ pri_message(ctrl, "-- Making new call for cr %d\n", cr); ++ } ++ ++ cur = calloc(1, sizeof(*cur)); ++ if (!cur) { + return NULL; ++ } + +- call_init(cur); +- /* Call reference */ +- cur->cr = cr; +- /* PRI is set to whoever called us */ +- if (pri->bri && (pri->localtype == PRI_CPE) && pri->subchannel && outboundnew) +- cur->pri = pri->subchannel; +- else +- cur->pri = pri; ++ /* Initialize call structure. */ ++ q931_init_call_record(ctrl, cur, cr); + + /* Append to end of list */ +- if (prev) ++ if (prev) { + prev->next = cur; +- else ++ } else { + *master->callpool = cur; +- ++ } ++ + return cur; + } + +-q931_call *q931_new_call(struct pri *pri) ++q931_call *q931_new_call(struct pri *ctrl) + { + q931_call *cur; + + do { +- cur = *pri->callpool; +- pri->cref++; +- if (!pri->bri) { +- if (pri->cref > 32767) +- pri->cref = 1; ++ cur = *ctrl->callpool; ++ ctrl->cref++; ++ if (!ctrl->bri) { ++ if (ctrl->cref > 32767) ++ ctrl->cref = 1; + } else { +- if (pri->cref > 127) +- pri->cref = 1; ++ if (ctrl->cref > 127) ++ ctrl->cref = 1; + } + while(cur) { +- if (cur->cr == (0x8000 | pri->cref)) ++ if (cur->cr == (0x8000 | ctrl->cref)) + break; + cur = cur->next; + } + } while(cur); + +- return q931_getcall(pri, pri->cref | 0x8000, 1); ++ return q931_getcall(ctrl, ctrl->cref | 0x8000); + } + +-static void q931_destroy(struct pri *pri, int cr, q931_call *c) ++static void stop_t303(struct q931_call *call); ++ ++static void cleanup_and_free_call(struct q931_call *cur) + { +- q931_call *cur, *prev; ++ stop_t303(cur); ++ pri_schedule_del(cur->pri, cur->retranstimer); ++ pri_call_apdu_queue_cleanup(cur); ++ free(cur); ++} + ++static void pri_create_fake_clearing(struct q931_call *c, struct pri *master); ++ ++void q931_destroycall(struct pri *ctrl, q931_call *c) ++{ ++ q931_call *cur; ++ q931_call *prev; ++ q931_call *slave; ++ int i; ++ int slavesleft; ++ int slaveidx; ++ ++ if (q931_is_dummy_call(c)) { ++ /* Cannot destroy the dummy call. */ ++ return; ++ } ++ if (c->master_call != c) { ++ slave = c; ++ c = slave->master_call; ++ } else { ++ slave = NULL; ++ } ++ + /* For destroying, make sure we are using the master span, since it maintains the call pool */ +- for (;pri->master; pri = pri->master); ++ ctrl = PRI_MASTER(ctrl); + + prev = NULL; +- cur = *pri->callpool; +- while(cur) { +- if ((c && (cur == c)) || (!c && (cur->cr == cr))) { ++ cur = *ctrl->callpool; ++ while (cur) { ++ if (cur == c) { ++ slaveidx = -1; ++ if (slave) { ++ for (i = 0; i < Q931_MAX_TEI; i++) { ++ if (cur->subcalls[i] == slave) { ++ if (ctrl->debug & PRI_DEBUG_Q931_STATE) { ++ pri_message(ctrl, "Destroying subcall %p of call %p at %d\n", ++ slave, cur, i); ++ } ++ cleanup_and_free_call(slave); ++ cur->subcalls[i] = NULL; ++ slaveidx = i; ++ break; ++ } ++ } ++ } ++ ++ slavesleft = 0; ++ for (i = 0; i < Q931_MAX_TEI; i++) { ++ if (cur->subcalls[i]) { ++ if (ctrl->debug & PRI_DEBUG_Q931_STATE) { ++ pri_message(ctrl, "Subcall still present at %d\n", i); ++ } ++ slavesleft++; ++ } ++ } ++ ++ /* We have 3 different phases to deal with: ++ * 1.) Sent outbound call, but no response, indicated by t203 present ++ * 2.) Sent outbound call, with responses, indicated by lack of t203 and subcalls present ++ * 3.) Outbound call connected, indicated by pri_winner > -1 ++ * ++ * If chan_dahdi hangs up in phase: ++ * 1.) T303 will be present, and we will fake clear in this case ++ * 2.) pri_winner will be < 0 and subcalls will be present. ++ * 3.) pri_winner will be > -1 and we will free the master when the winner dies. ++ * ++ * If remote ends hang up in phase: ++ * 1.) Impossible, defined by phase. ++ * 2.) When last end hangs up, we should cause a fake clearing. ++ * 3.) Pass events to winner up and be freed when winner is freed ++ * ++ * Exceptional conditions in phase: ++ * 1.) None. ++ * 2.) None. ++ * 3.) We hang up a call so quickly that it hangs up before other competing lines finish hangup sequence ++ * Subcalls present still even though we have hung up the winner. ++ * ++ * So, we could say: ++ * If, when the library user hangs up the master call, and there are more than one subcall up, we fake clear ++ * regardless of whether or not we drop down to one subcall left in the clearing process. ++ * ++ * If there are only one call up, we mirror what it does. ++ * ++ * OR ++ * ++ * Phase 2. them clearing: ++ * For handling of Phase 2 (indicated by not running and pri_winner not present): ++ * We create a fake hangup sequence after all the subcalls have been destroyed and after ++ * ++ * "" us clearing: ++ * For we need to start the fake clearing, but it needs to be half of a fake clearing, not a full one (since we already had a hangup). ++ * ++ * For handling of Phase 3 plus exceptions: ++ * ++ * If pri_winner exists, we mirror him in terms of events (which provides our hangup sequence), and when we have the complete ++ * hangup sequence completed (destroy called on master call), if there still exist non winner subcalls at this time, we declare the master ++ * call as dead and free it when the last subcall clears. ++ */ ++ ++ if ((slave && !slavesleft) && ++ ((cur->pri_winner < 0) || (slave && slaveidx != cur->pri_winner))) { ++ pri_create_fake_clearing(cur, ctrl); ++ return; ++ } ++ ++ if (slavesleft) { ++ return; ++ } ++ ++ /* Master call or normal call destruction. */ ++ if ((cur->pri_winner > -1) && cur->outboundbroadcast) { ++ if (ctrl->debug & PRI_DEBUG_Q931_STATE) { ++ pri_message(ctrl, ++ "Since we already had a winner, we should just be able to kill the call anyways\n"); ++ } ++ } + if (prev) + prev->next = cur->next; + else +- *pri->callpool = cur->next; +- if (pri->debug & PRI_DEBUG_Q931_STATE) +- pri_message(pri, "NEW_HANGUP DEBUG: Destroying the call, ourstate %s, peerstate %s\n",callstate2str(cur->ourcallstate),callstate2str(cur->peercallstate)); +- if (cur->retranstimer) +- pri_schedule_del(pri, cur->retranstimer); +- pri_call_apdu_queue_cleanup(cur); +- free(cur); ++ *ctrl->callpool = cur->next; ++ if (ctrl->debug & PRI_DEBUG_Q931_STATE) ++ pri_message(ctrl, ++ "NEW_HANGUP DEBUG: Destroying the call, ourstate %s, peerstate %s, hold-state %s\n", ++ q931_call_state_str(cur->ourcallstate), ++ q931_call_state_str(cur->peercallstate), ++ q931_hold_state_str(cur->hold_state)); ++ pri_schedule_del(ctrl, cur->hold_timer); ++ cleanup_and_free_call(cur); + return; + } + prev = cur; + cur = cur->next; + } +- pri_error(pri, "Can't destroy call %d!\n", cr); ++ pri_error(ctrl, "Can't destroy call %p cref:%d!\n", c, c->cr); + } + +-static void q931_destroycall(struct pri *pri, int cr) ++static int add_ie(struct pri *ctrl, q931_call *call, int msgtype, int ie, q931_ie *iet, int maxlen, int *codeset) + { +- return q931_destroy(pri, cr, NULL); +-} +- +- +-void __q931_destroycall(struct pri *pri, q931_call *c) +-{ +- if (pri && c) +- q931_destroy(pri,0, c); +- return; +-} +- +-static int add_ie(struct pri *pri, q931_call *call, int msgtype, int ie, q931_ie *iet, int maxlen, int *codeset) +-{ + unsigned int x; + int res, total_res; + int have_shift; +@@ -2473,7 +3775,7 @@ + total_res = 0; + do { + iet->ie = ie; +- res = ies[x].transmit(ie, pri, call, msgtype, iet, maxlen, ++order); ++ res = ies[x].transmit(ie, ctrl, call, msgtype, iet, maxlen, ++order); + /* Error if res < 0 or ignored if res == 0 */ + if (res < 0) + return res; +@@ -2484,8 +3786,7 @@ + maxlen -= res; + iet = (q931_ie *)((char *)iet + res); + } +- } +- while (res > 0 && order < ies_count); ++ } while (res > 0 && order < ies_count); + if (have_shift && total_res) { + if (Q931_IE_CODESET(ies[x].ie)) + *codeset = Q931_IE_CODESET(ies[x].ie); +@@ -2493,12 +3794,12 @@ + } + return total_res; + } else { +- pri_error(pri, "!! Don't know how to add an IE %s (%d)\n", ie2str(ie), ie); ++ pri_error(ctrl, "!! Don't know how to add an IE %s (%d)\n", ie2str(ie), ie); + return -1; + } + } + } +- pri_error(pri, "!! Unknown IE %d (%s)\n", ie, ie2str(ie)); ++ pri_error(ctrl, "!! Unknown IE %d (%s)\n", ie, ie2str(ie)); + return -1; + } + +@@ -2513,25 +3814,37 @@ + return code2str(disc, discs, sizeof(discs) / sizeof(discs[0])); + } + +-void q931_dump(struct pri *pri, q931_h *h, int len, int txrx) ++void q931_dump(struct pri *ctrl, q931_h *h, int len, int txrx) + { + q931_mh *mh; + char c; + int x=0, r; + int cur_codeset; + int codeset; ++ int cref; ++ + c = txrx ? '>' : '<'; +- pri_message(pri, "%c Protocol Discriminator: %s (%d) len=%d\n", c, disc2str(h->pd), h->pd, len); +- pri_message(pri, "%c Call Ref: len=%2d (reference %d/0x%X) (%s)\n", c, h->crlen, q931_cr(h) & 0x7FFF, q931_cr(h) & 0x7FFF, (h->crv[0] & 0x80) ? "Terminator" : "Originator"); ++ pri_message(ctrl, "%c Protocol Discriminator: %s (%d) len=%d\n", c, disc2str(h->pd), h->pd, len); ++ cref = q931_cr(h); ++ pri_message(ctrl, "%c Call Ref: len=%2d (reference %d/0x%X) (%s)\n", ++ c, h->crlen, cref & 0x7FFF, cref & 0x7FFF, ++ (cref == Q931_DUMMY_CALL_REFERENCE) ++ ? "Dummy" ++ : (cref & 0x8000) ? "Terminator" : "Originator"); ++ + /* Message header begins at the end of the call reference number */ + mh = (q931_mh *)(h->contents + h->crlen); +- pri_message(pri, "%c Message type: %s (%d)\n", c, msg2str(mh->msg), mh->msg); ++ if ((h->pd == MAINTENANCE_PROTOCOL_DISCRIMINATOR_1) || (h->pd == MAINTENANCE_PROTOCOL_DISCRIMINATOR_2)) { ++ pri_message(ctrl, "%c Message Type: %s (%d)\n", c, maintenance_msg2str(mh->msg, h->pd), mh->msg); ++ } else { ++ pri_message(ctrl, "%c Message Type: %s (%d)\n", c, msg2str(mh->msg), mh->msg); ++ } + /* Drop length of header, including call reference */ + len -= (h->crlen + 3); + codeset = cur_codeset = 0; + while(x < len) { + r = ielen((q931_ie *)(mh->data + x)); +- q931_dumpie(pri, cur_codeset, (q931_ie *)(mh->data + x), c); ++ q931_dumpie(ctrl, cur_codeset, (q931_ie *)(mh->data + x), c); + switch (mh->data[x] & 0xf8) { + case Q931_LOCKING_SHIFT: + if ((mh->data[x] & 7) > 0) +@@ -2547,86 +3860,106 @@ + x += r; + } + if (x > len) +- pri_error(pri, "XXX Message longer than it should be?? XXX\n"); ++ pri_error(ctrl, "XXX Message longer than it should be?? XXX\n"); + } + +-static int q931_handle_ie(int codeset, struct pri *pri, q931_call *c, int msg, q931_ie *ie) ++static int q931_handle_ie(int codeset, struct pri *ctrl, q931_call *c, int msg, q931_ie *ie) + { + unsigned int x; + int full_ie = Q931_FULL_IE(codeset, ie->ie); +- if (pri->debug & PRI_DEBUG_Q931_STATE) +- pri_message(pri, "-- Processing IE %d (cs%d, %s)\n", ie->ie, codeset, ie2str(full_ie)); ++ if (ctrl->debug & PRI_DEBUG_Q931_STATE) ++ pri_message(ctrl, "-- Processing IE %d (cs%d, %s)\n", ie->ie, codeset, ie2str(full_ie)); + for (x=0;x<sizeof(ies) / sizeof(ies[0]);x++) { + if (full_ie == ies[x].ie) { + if (ies[x].receive) +- return ies[x].receive(full_ie, pri, c, msg, ie, ielen(ie)); ++ return ies[x].receive(full_ie, ctrl, c, msg, ie, ielen(ie)); + else { +- if (pri->debug & PRI_DEBUG_Q931_ANOMALY) +- pri_error(pri, "!! No handler for IE %d (cs%d, %s)\n", ie->ie, codeset, ie2str(full_ie)); ++ if (ctrl->debug & PRI_DEBUG_Q931_ANOMALY) ++ pri_error(ctrl, "!! No handler for IE %d (cs%d, %s)\n", ie->ie, codeset, ie2str(full_ie)); + return -1; + } + } + } +- pri_message(pri, "!! Unknown IE %d (cs%d, %s)\n", ie->ie, codeset, ie2str(full_ie)); ++ pri_message(ctrl, "!! Unknown IE %d (cs%d, %s)\n", ie->ie, codeset, ie2str(full_ie)); + return -1; + } + +-static void init_header(struct pri *pri, q931_call *call, unsigned char *buf, q931_h **hb, q931_mh **mhb, int *len) ++/* Returns header and message header and modifies length in place */ ++static void init_header(struct pri *ctrl, q931_call *call, unsigned char *buf, q931_h **hb, q931_mh **mhb, int *len, int protodisc) + { +- /* Returns header and message header and modifies length in place */ +- q931_h *h = (q931_h *)buf; +- q931_mh * mh; +- h->pd = pri->protodisc; ++ q931_h *h = (q931_h *) buf; ++ q931_mh *mh; ++ unsigned crv; ++ ++ if (protodisc) { ++ h->pd = protodisc; ++ } else { ++ h->pd = ctrl->protodisc; ++ } + h->x0 = 0; /* Reserved 0 */ +- if (!pri->bri) { +- h->crlen = 2; /* Two bytes of Call Reference. Invert the top bit to make it from our sense */ +- if (call->cr || call->forceinvert) { +- h->crv[0] = ((call->cr ^ 0x8000) & 0xff00) >> 8; +- h->crv[1] = (call->cr & 0xff); +- } else { +- /* Unless of course this has no call reference */ +- h->crv[0] = 0; +- h->crv[1] = 0; +- } +- if (pri->subchannel && !pri->bri) { ++ if (q931_is_dummy_call(call)) { ++ h->crlen = 0; ++ } else if (!ctrl->bri) { ++ /* Two bytes of Call Reference. */ ++ h->crlen = 2; ++ /* Invert the top bit to make it from our sense */ ++ crv = (unsigned) call->cr; ++ h->crv[0] = ((crv >> 8) ^ 0x80) & 0xff; ++ h->crv[1] = crv & 0xff; ++ if (ctrl->subchannel && !ctrl->bri) { + /* On GR-303, top bit is always 0 */ + h->crv[0] &= 0x7f; + } + } else { + h->crlen = 1; +- if (call->cr || call->forceinvert) { +- h->crv[0] = (((call->cr ^ 0x8000) & 0x8000) >> 8) | (call->cr & 0x7f); +- } else { +- /* Unless of course this has no call reference */ +- h->crv[0] = 0; +- } ++ /* Invert the top bit to make it from our sense */ ++ crv = (unsigned) call->cr; ++ h->crv[0] = (((crv >> 8) ^ 0x80) & 0x80) | (crv & 0x7f); + } +- mh = (q931_mh *)(h->contents + h->crlen); ++ *hb = h; ++ ++ *len -= 3;/* Protocol discriminator, call reference length, message type id */ ++ *len -= h->crlen; ++ ++ mh = (q931_mh *) (h->contents + h->crlen); + mh->f = 0; +- *hb = h; + *mhb = mh; +- if (h->crlen == 2) +- *len -= 5; +- else +- *len -= 4; +- + } + +-static int q931_xmit(struct pri *pri, q931_h *h, int len, int cr) ++static int q931_xmit(struct pri *ctrl, q931_h *h, int len, int cr, int uiframe) + { +- q921_transmit_iframe(pri, h, len, cr); ++ if (uiframe) { ++ q921_transmit_uiframe(ctrl, h, len); ++ } else { ++ q921_transmit_iframe(ctrl, h, len, cr); ++ } + /* The transmit operation might dump the q921 header, so logging the q931 + message body after the transmit puts the sections of the message in the + right order in the log */ +- if (pri->debug & PRI_DEBUG_Q931_DUMP) +- q931_dump(pri, h, len, 1); ++ if (ctrl->debug & PRI_DEBUG_Q931_DUMP) ++ q931_dump(ctrl, h, len, 1); + #ifdef LIBPRI_COUNTERS +- pri->q931_txcount++; ++ ctrl->q931_txcount++; + #endif + return 0; + } + +-static int send_message(struct pri *pri, q931_call *c, int msgtype, int ies[]) ++/*! ++ * \internal ++ * \brief Build and send the requested message. ++ * ++ * \param ctrl D channel controller. ++ * \param call Q.931 call leg ++ * \param msgtype Q.931 message type to build. ++ * \param ies List of ie's to put in the message. ++ * ++ * \note The ie's in the ie list must be in numerical order. ++ * See Q.931 section 4.5.1 coding rules. ++ * ++ * \retval 0 on success. ++ * \retval -1 on error. ++ */ ++static int send_message(struct pri *ctrl, q931_call *call, int msgtype, int ies[]) + { + unsigned char buf[1024]; + q931_h *h; +@@ -2636,18 +3969,24 @@ + int offset=0; + int x; + int codeset; +- ++ ++ if (call->outboundbroadcast && call->master_call == call && msgtype != Q931_SETUP) { ++ pri_error(ctrl, ++ "Attempting to use master call record to send %s on BRI PTMP NT %p\n", ++ msg2str(msgtype), ctrl); ++ return -1; ++ } ++ + memset(buf, 0, sizeof(buf)); + len = sizeof(buf); +- init_header(pri, c, buf, &h, &mh, &len); +- mh->msg = msgtype; ++ init_header(ctrl, call, buf, &h, &mh, &len, (msgtype >> 8)); ++ mh->msg = msgtype & 0x00ff; + x=0; + codeset = 0; + while(ies[x] > -1) { +- res = add_ie(pri, c, mh->msg, ies[x], (q931_ie *)(mh->data + offset), len, &codeset); +- ++ res = add_ie(ctrl, call, mh->msg, ies[x], (q931_ie *)(mh->data + offset), len, &codeset); + if (res < 0) { +- pri_error(pri, "!! Unable to add IE '%s'\n", ie2str(ies[x])); ++ pri_error(ctrl, "!! Unable to add IE '%s'\n", ie2str(ies[x])); + return -1; + } + +@@ -2657,20 +3996,121 @@ + } + /* Invert the logic */ + len = sizeof(buf) - len; +- q931_xmit(c->pri, h, len, 1); +- c->acked = 1; ++ ++ ctrl = call->pri; ++ if (BRI_TE_PTMP(ctrl)) { ++ /* ++ * Must use the BRI subchannel structure to send with the correct TEI. ++ * Note: If the subchannel is NULL then there is no TEI assigned and ++ * we should not be sending anything out at this time. ++ */ ++ ctrl = ctrl->subchannel; ++ } ++ if (ctrl) { ++ int uiframe; ++ ++ switch (msgtype) { ++ case Q931_SETUP: ++ /* ++ * For NT-PTMP mode, we need to check the following: ++ * MODE = NT-PTMP ++ * MESSAGE = SETUP ++ * ++ * If those are true, we need to send the SETUP in a UI frame ++ * instead of an I-frame. ++ */ ++ if (BRI_NT_PTMP(ctrl)) ++ uiframe = 1; ++ else ++ uiframe = 0; ++ break; ++ case Q931_FACILITY: ++ if (ctrl->tei == Q921_TEI_GROUP) { ++ /* Broadcast TEI. */ ++ if (q931_is_dummy_call(call)) { ++ /* ++ * This is a FACILITY message on the dummy call reference ++ * for the broadcast TEI. ++ */ ++ uiframe = 1; ++ } else { ++ pri_error(ctrl, ++ "Attempting to broadcast %s on cref %d\n", ++ msg2str(msgtype), call->cr); ++ return -1; ++ } ++ } else { ++ uiframe = 0; ++ } ++ break; ++ default: ++ uiframe = 0; ++ break; ++ } ++ if (ctrl->debug & PRI_DEBUG_Q931_STATE) { ++ pri_message(ctrl, ++ "Sending message for call %p on %p TEI/SAPI %d/%d, call->pri is %p, TEI/SAPI %d/%d\n", ++ call, ++ ctrl, ctrl->tei, ctrl->sapi, ++ call->pri, call->pri->tei, call->pri->sapi); ++ } ++ q931_xmit(ctrl, h, len, 1, uiframe); ++ } ++ call->acked = 1; + return 0; + } + +-static int status_ies[] = { Q931_CAUSE, Q931_CALL_STATE, -1 }; ++static int maintenance_service_ies[] = { Q931_IE_CHANGE_STATUS, Q931_CHANNEL_IDENT, -1 }; + +-static int q931_status(struct pri *pri, q931_call *c, int cause) ++int maintenance_service_ack(struct pri *ctrl, q931_call *c) + { ++ int pd = MAINTENANCE_PROTOCOL_DISCRIMINATOR_1; ++ int mt = ATT_SERVICE_ACKNOWLEDGE; ++ ++ if (ctrl->switchtype == PRI_SWITCH_NI2) { ++ pd = MAINTENANCE_PROTOCOL_DISCRIMINATOR_2; ++ mt = NATIONAL_SERVICE_ACKNOWLEDGE; ++ } ++ return send_message(ctrl, c, (pd << 8) | mt, maintenance_service_ies); ++} ++ ++int maintenance_service(struct pri *ctrl, int span, int channel, int changestatus) ++{ ++ struct q931_call *c; ++ int pd = MAINTENANCE_PROTOCOL_DISCRIMINATOR_1; ++ int mt = ATT_SERVICE; ++ ++ c = q931_getcall(ctrl, 0 | 0x8000); ++ if (!c) { ++ return -1; ++ } ++ if (channel > -1) { ++ c->channelno = channel & 0xff; ++ c->chanflags = FLAG_EXCLUSIVE; ++ } else { ++ c->channelno = channel; ++ c->chanflags = FLAG_EXCLUSIVE | FLAG_WHOLE_INTERFACE; ++ } ++ c->ds1no = span; ++ c->ds1explicit = 0; ++ c->changestatus = changestatus; ++ ++ if (ctrl->switchtype == PRI_SWITCH_NI2) { ++ pd = MAINTENANCE_PROTOCOL_DISCRIMINATOR_2; ++ mt = NATIONAL_SERVICE; ++ } ++ return send_message(ctrl, c, (pd << 8) | mt, maintenance_service_ies); ++} ++ ++static int status_ies[] = { Q931_CAUSE, Q931_IE_CALL_STATE, -1 }; ++ ++static int q931_status(struct pri *ctrl, q931_call *c, int cause) ++{ + q931_call *cur = NULL; + if (!cause) + cause = PRI_CAUSE_RESPONSE_TO_STATUS_ENQUIRY; + if (c->cr > -1) +- cur = *pri->callpool; ++ cur = *ctrl->callpool; + while(cur) { + if (cur->cr == c->cr) { + cur->cause=cause; +@@ -2681,79 +4121,156 @@ + cur = cur->next; + } + if (!cur) { +- pri_message(pri, "YYY Here we get reset YYY\n"); ++ pri_message(ctrl, "YYY Here we get reset YYY\n"); + /* something went wrong, respond with "no such call" */ + c->ourcallstate = Q931_CALL_STATE_NULL; + c->peercallstate = Q931_CALL_STATE_NULL; + cur=c; + } +- return send_message(pri, cur, Q931_STATUS, status_ies); ++ return send_message(ctrl, cur, Q931_STATUS, status_ies); + } + +-static int information_ies[] = { Q931_IE_KEYPAD_FACILITY, Q931_CALLED_PARTY_NUMBER, -1 }; ++static int information_ies[] = { Q931_CALLED_PARTY_NUMBER, -1 }; + +-int q931_information(struct pri *pri, q931_call *c, char digit) ++int q931_information(struct pri *ctrl, q931_call *c, char digit) + { +- c->callednum[0] = digit; +- c->callednum[1] = '\0'; +- return send_message(pri, c, Q931_INFORMATION, information_ies); ++ c->overlap_digits[0] = digit; ++ c->overlap_digits[1] = '\0'; ++ ++ /* ++ * Since we are doing overlap dialing now, we need to accumulate ++ * the digits into call->called.number.str. ++ */ ++ c->called.number.valid = 1; ++ if (strlen(c->called.number.str) < sizeof(c->called.number.str) - 1) { ++ /* There is enough room for the new digit. */ ++ strcat(c->called.number.str, c->overlap_digits); ++ } ++ ++ return send_message(ctrl, c, Q931_INFORMATION, information_ies); + } + + static int keypad_facility_ies[] = { Q931_IE_KEYPAD_FACILITY, -1 }; + +-int q931_keypad_facility(struct pri *pri, q931_call *call, char *digits) ++int q931_keypad_facility(struct pri *ctrl, q931_call *call, const char *digits) + { + libpri_copy_string(call->keypad_digits, digits, sizeof(call->keypad_digits)); +- return send_message(pri, call, Q931_INFORMATION, keypad_facility_ies); ++ return send_message(ctrl, call, Q931_INFORMATION, keypad_facility_ies); + } + + static int restart_ack_ies[] = { Q931_CHANNEL_IDENT, Q931_RESTART_INDICATOR, -1 }; + +-static int restart_ack(struct pri *pri, q931_call *c) ++static int restart_ack(struct pri *ctrl, q931_call *c) + { +- UPDATE_OURCALLSTATE(pri, c, Q931_CALL_STATE_NULL); ++ UPDATE_OURCALLSTATE(ctrl, c, Q931_CALL_STATE_NULL); + c->peercallstate = Q931_CALL_STATE_NULL; +- return send_message(pri, c, Q931_RESTART_ACKNOWLEDGE, restart_ack_ies); ++ return send_message(ctrl, c, Q931_RESTART_ACKNOWLEDGE, restart_ack_ies); + } + + static int facility_ies[] = { Q931_IE_FACILITY, -1 }; + +-int q931_facility(struct pri*pri, q931_call *c) ++int q931_facility(struct pri*ctrl, q931_call *c) + { +- return send_message(pri, c, Q931_FACILITY, facility_ies); ++ return send_message(ctrl, c, Q931_FACILITY, facility_ies); + } + +-static int notify_ies[] = { Q931_IE_NOTIFY_IND, -1 }; ++static int notify_ies[] = { Q931_IE_NOTIFY_IND, Q931_IE_REDIRECTION_NUMBER, -1 }; + +-int q931_notify(struct pri *pri, q931_call *c, int channel, int info) ++/*! ++ * \internal ++ * \brief Actually send a NOTIFY message with optional redirection number. ++ * ++ * \param ctrl D channel controller. ++ * \param call Q.931 call leg ++ * \param notify Notification indicator ++ * \param number Redirection number to send if not NULL. ++ * ++ * \retval 0 on success. ++ * \retval -1 on error. ++ */ ++static int q931_notify_redirection_helper(struct pri *ctrl, q931_call *call, int notify, const struct q931_party_number *number) + { +- if ((pri->switchtype == PRI_SWITCH_EUROISDN_T1) || (pri->switchtype != PRI_SWITCH_EUROISDN_E1)) { +- if ((info > 0x2) || (info < 0x00)) ++ if (number) { ++ call->redirection_number = *number; ++ } else { ++ q931_party_number_init(&call->redirection_number); ++ } ++ call->notify = notify; ++ return send_message(ctrl, call, Q931_NOTIFY, notify_ies); ++} ++ ++/*! ++ * \brief Send a NOTIFY message with optional redirection number. ++ * ++ * \param ctrl D channel controller. ++ * \param call Q.931 call leg ++ * \param notify Notification indicator ++ * \param number Redirection number to send if not NULL. ++ * ++ * \retval 0 on success. ++ * \retval -1 on error. ++ */ ++int q931_notify_redirection(struct pri *ctrl, q931_call *call, int notify, const struct q931_party_number *number) ++{ ++ int status; ++ unsigned idx; ++ struct q931_call *subcall; ++ ++ if (call->outboundbroadcast && call->master_call == call) { ++ status = 0; ++ for (idx = 0; idx < Q931_MAX_TEI; ++idx) { ++ subcall = call->subcalls[idx]; ++ if (subcall) { ++ /* Send to all subcalls that have given a positive response. */ ++ switch (subcall->ourcallstate) { ++ case Q931_CALL_STATE_OUTGOING_CALL_PROCEEDING: ++ case Q931_CALL_STATE_CALL_DELIVERED: ++ case Q931_CALL_STATE_ACTIVE: ++ if (q931_notify_redirection_helper(ctrl, subcall, notify, number)) { ++ status = -1; ++ } ++ break; ++ default: ++ break; ++ } ++ } ++ } ++ } else { ++ status = q931_notify_redirection_helper(ctrl, call, notify, number); ++ } ++ return status; ++} ++ ++int q931_notify(struct pri *ctrl, q931_call *c, int channel, int info) ++{ ++ if ((ctrl->switchtype == PRI_SWITCH_EUROISDN_T1) || (ctrl->switchtype != PRI_SWITCH_EUROISDN_E1)) { ++ if ((info > 0x2) || (info < 0x00)) { + return 0; ++ } + } + +- if (info >= 0) +- c->notify = info & 0x7F; +- else +- c->notify = -1; +- return send_message(pri, c, Q931_NOTIFY, notify_ies); ++ if (info >= 0) { ++ info = info & 0x7F; ++ } else { ++ info = -1; ++ } ++ return q931_notify_redirection(ctrl, c, info, NULL); + } + + #ifdef ALERTING_NO_PROGRESS + static int call_progress_ies[] = { -1 }; + #else +-static int call_progress_with_cause_ies[] = { Q931_PROGRESS_INDICATOR, Q931_CAUSE, -1 }; ++static int call_progress_with_cause_ies[] = { Q931_CAUSE, Q931_PROGRESS_INDICATOR, -1 }; + + static int call_progress_ies[] = { Q931_PROGRESS_INDICATOR, -1 }; + #endif + +-int q931_call_progress(struct pri *pri, q931_call *c, int channel, int info) ++int q931_call_progress(struct pri *ctrl, q931_call *c, int channel, int info) + { + if (channel) { + c->ds1no = (channel & 0xff00) >> 8; + c->ds1explicit = (channel & 0x10000) >> 16; +- channel &= 0xff; +- c->channelno = channel; ++ c->channelno = channel & 0xff; + } + + if (info) { +@@ -2762,21 +4279,20 @@ + c->progressmask = PRI_PROG_INBAND_AVAILABLE; + } else { + /* PI is mandatory IE for PROGRESS message - Q.931 3.1.8 */ +- pri_error(pri, "XXX Progress message requested but no information is provided\n"); ++ pri_error(ctrl, "XXX Progress message requested but no information is provided\n"); + c->progressmask = 0; + } + + c->alive = 1; +- return send_message(pri, c, Q931_PROGRESS, call_progress_ies); ++ return send_message(ctrl, c, Q931_PROGRESS, call_progress_ies); + } + +-int q931_call_progress_with_cause(struct pri *pri, q931_call *c, int channel, int info, int cause) ++int q931_call_progress_with_cause(struct pri *ctrl, q931_call *c, int channel, int info, int cause) + { + if (channel) { + c->ds1no = (channel & 0xff00) >> 8; + c->ds1explicit = (channel & 0x10000) >> 16; +- channel &= 0xff; +- c->channelno = channel; ++ c->channelno = channel & 0xff; + } + + if (info) { +@@ -2785,7 +4301,7 @@ + c->progressmask = PRI_PROG_INBAND_AVAILABLE; + } else { + /* PI is mandatory IE for PROGRESS message - Q.931 3.1.8 */ +- pri_error(pri, "XXX Progress message requested but no information is provided\n"); ++ pri_error(ctrl, "XXX Progress message requested but no information is provided\n"); + c->progressmask = 0; + } + +@@ -2794,7 +4310,7 @@ + c->causeloc = LOC_PRIV_NET_LOCAL_USER; + + c->alive = 1; +- return send_message(pri, c, Q931_PROGRESS, call_progress_with_cause_ies); ++ return send_message(ctrl, c, Q931_PROGRESS, call_progress_with_cause_ies); + } + + #ifdef ALERTING_NO_PROGRESS +@@ -2803,17 +4319,16 @@ + static int call_proceeding_ies[] = { Q931_CHANNEL_IDENT, Q931_PROGRESS_INDICATOR, -1 }; + #endif + +-int q931_call_proceeding(struct pri *pri, q931_call *c, int channel, int info) ++int q931_call_proceeding(struct pri *ctrl, q931_call *c, int channel, int info) + { + if (channel) { + c->ds1no = (channel & 0xff00) >> 8; + c->ds1explicit = (channel & 0x10000) >> 16; +- channel &= 0xff; +- c->channelno = channel; ++ c->channelno = channel & 0xff; + } + c->chanflags &= ~FLAG_PREFERRED; + c->chanflags |= FLAG_EXCLUSIVE; +- UPDATE_OURCALLSTATE(pri, c, Q931_CALL_STATE_INCOMING_CALL_PROCEEDING); ++ UPDATE_OURCALLSTATE(ctrl, c, Q931_CALL_STATE_INCOMING_CALL_PROCEEDING); + c->peercallstate = Q931_CALL_STATE_OUTGOING_CALL_PROCEEDING; + if (info) { + c->progloc = LOC_PRIV_NET_LOCAL_USER; +@@ -2823,62 +4338,73 @@ + c->progressmask = 0; + c->proc = 1; + c->alive = 1; +- return send_message(pri, c, Q931_CALL_PROCEEDING, call_proceeding_ies); ++ return send_message(ctrl, c, Q931_CALL_PROCEEDING, call_proceeding_ies); + } + #ifndef ALERTING_NO_PROGRESS +-static int alerting_ies[] = { Q931_PROGRESS_INDICATOR, Q931_IE_USER_USER, -1 }; ++static int alerting_ies[] = { Q931_IE_FACILITY, Q931_PROGRESS_INDICATOR, Q931_IE_USER_USER, -1 }; + #else +-static int alerting_ies[] = { -1 }; ++static int alerting_ies[] = { Q931_IE_FACILITY, -1 }; + #endif + +-int q931_alerting(struct pri *pri, q931_call *c, int channel, int info) ++int q931_alerting(struct pri *ctrl, q931_call *c, int channel, int info) + { + if (!c->proc) +- q931_call_proceeding(pri, c, channel, 0); ++ q931_call_proceeding(ctrl, c, channel, 0); + if (info) { + c->progloc = LOC_PRIV_NET_LOCAL_USER; + c->progcode = CODE_CCITT; + c->progressmask = PRI_PROG_INBAND_AVAILABLE; + } else + c->progressmask = 0; +- UPDATE_OURCALLSTATE(pri, c, Q931_CALL_STATE_CALL_RECEIVED); ++ UPDATE_OURCALLSTATE(ctrl, c, Q931_CALL_STATE_CALL_RECEIVED); + c->peercallstate = Q931_CALL_STATE_CALL_DELIVERED; + c->alive = 1; +- return send_message(pri, c, Q931_ALERTING, alerting_ies); ++ ++ switch (ctrl->switchtype) { ++ case PRI_SWITCH_QSIG: ++ if (c->local_id.name.valid) { ++ /* Send calledName with ALERTING */ ++ rose_called_name_encode(ctrl, c, Q931_ALERTING); ++ } ++ break; ++ default: ++ break; ++ } ++ ++ return send_message(ctrl, c, Q931_ALERTING, alerting_ies); + } + +-static int connect_ies[] = { Q931_CHANNEL_IDENT, Q931_PROGRESS_INDICATOR, -1 }; ++static int setup_ack_ies[] = { Q931_CHANNEL_IDENT, Q931_IE_FACILITY, Q931_PROGRESS_INDICATOR, -1 }; + +-int q931_setup_ack(struct pri *pri, q931_call *c, int channel, int nonisdn) ++int q931_setup_ack(struct pri *ctrl, q931_call *c, int channel, int nonisdn) + { + if (channel) { + c->ds1no = (channel & 0xff00) >> 8; + c->ds1explicit = (channel & 0x10000) >> 16; +- channel &= 0xff; +- c->channelno = channel; ++ c->channelno = channel & 0xff; + } + c->chanflags &= ~FLAG_PREFERRED; + c->chanflags |= FLAG_EXCLUSIVE; +- if (nonisdn && (pri->switchtype != PRI_SWITCH_DMS100)) { ++ if (nonisdn && (ctrl->switchtype != PRI_SWITCH_DMS100)) { + c->progloc = LOC_PRIV_NET_LOCAL_USER; + c->progcode = CODE_CCITT; + c->progressmask = PRI_PROG_CALLED_NOT_ISDN; + } else + c->progressmask = 0; +- UPDATE_OURCALLSTATE(pri, c, Q931_CALL_STATE_OVERLAP_RECEIVING); ++ UPDATE_OURCALLSTATE(ctrl, c, Q931_CALL_STATE_OVERLAP_RECEIVING); + c->peercallstate = Q931_CALL_STATE_OVERLAP_SENDING; + c->alive = 1; +- return send_message(pri, c, Q931_SETUP_ACKNOWLEDGE, connect_ies); ++ return send_message(ctrl, c, Q931_SETUP_ACKNOWLEDGE, setup_ack_ies); + } + + /* T313 expiry, first time */ + static void pri_connect_timeout(void *data) + { + struct q931_call *c = data; +- struct pri *pri = c->pri; +- if (pri->debug & PRI_DEBUG_Q931_STATE) +- pri_message(pri, "Timed out looking for connect acknowledge\n"); +- q931_disconnect(pri, c, PRI_CAUSE_NORMAL_CLEARING); ++ struct pri *ctrl = c->pri; ++ if (ctrl->debug & PRI_DEBUG_Q931_STATE) ++ pri_message(ctrl, "Timed out looking for connect acknowledge\n"); ++ q931_disconnect(ctrl, c, PRI_CAUSE_NORMAL_CLEARING); + + } + +@@ -2886,85 +4412,120 @@ + static void pri_release_timeout(void *data) + { + struct q931_call *c = data; +- struct pri *pri = c->pri; +- if (pri->debug & PRI_DEBUG_Q931_STATE) +- pri_message(pri, "Timed out looking for release complete\n"); ++ struct pri *ctrl = c->pri; ++ if (ctrl->debug & PRI_DEBUG_Q931_STATE) ++ pri_message(ctrl, "Timed out looking for release complete\n"); + c->t308_timedout++; + c->alive = 1; + + /* The call to q931_release will re-schedule T308 */ +- q931_release(pri, c, c->cause); ++ q931_release(ctrl, c, c->cause); + } + + /* T308 expiry, second time */ + static void pri_release_finaltimeout(void *data) + { + struct q931_call *c = data; +- struct pri *pri = c->pri; ++ struct pri *ctrl = c->pri; + c->alive = 1; +- if (pri->debug & PRI_DEBUG_Q931_STATE) +- pri_message(pri, "Final time-out looking for release complete\n"); ++ if (ctrl->debug & PRI_DEBUG_Q931_STATE) ++ pri_message(ctrl, "Final time-out looking for release complete\n"); + c->t308_timedout++; + c->ourcallstate = Q931_CALL_STATE_NULL; + c->peercallstate = Q931_CALL_STATE_NULL; +- pri->schedev = 1; +- pri->ev.e = PRI_EVENT_HANGUP_ACK; +- pri->ev.hangup.channel = c->channelno; +- pri->ev.hangup.cause = c->cause; +- pri->ev.hangup.cref = c->cr; +- pri->ev.hangup.call = c; +- pri->ev.hangup.aoc_units = c->aoc_units; +- libpri_copy_string(pri->ev.hangup.useruserinfo, c->useruserinfo, sizeof(pri->ev.hangup.useruserinfo)); +- q931_hangup(pri, c, c->cause); ++ q931_clr_subcommands(ctrl); ++ ctrl->schedev = 1; ++ ctrl->ev.e = PRI_EVENT_HANGUP_ACK; ++ ctrl->ev.hangup.subcmds = &ctrl->subcmds; ++ ctrl->ev.hangup.channel = q931_encode_channel(c); ++ ctrl->ev.hangup.cause = c->cause; ++ ctrl->ev.hangup.cref = c->cr; ++ ctrl->ev.hangup.call = c->master_call; ++ ctrl->ev.hangup.aoc_units = c->aoc_units; ++ ctrl->ev.hangup.call_held = NULL; ++ ctrl->ev.hangup.call_active = NULL; ++ libpri_copy_string(ctrl->ev.hangup.useruserinfo, c->useruserinfo, sizeof(ctrl->ev.hangup.useruserinfo)); ++ pri_hangup(ctrl, c, c->cause); + } + + /* T305 expiry, first time */ + static void pri_disconnect_timeout(void *data) + { + struct q931_call *c = data; +- struct pri *pri = c->pri; +- if (pri->debug & PRI_DEBUG_Q931_STATE) +- pri_message(pri, "Timed out looking for release\n"); ++ struct pri *ctrl = c->pri; ++ if (ctrl->debug & PRI_DEBUG_Q931_STATE) ++ pri_message(ctrl, "Timed out looking for release\n"); + c->alive = 1; +- q931_release(pri, c, PRI_CAUSE_NORMAL_CLEARING); ++ q931_release(ctrl, c, PRI_CAUSE_NORMAL_CLEARING); + } + +-int q931_connect(struct pri *pri, q931_call *c, int channel, int nonisdn) ++static int connect_ies[] = { ++ Q931_CHANNEL_IDENT, ++ Q931_IE_FACILITY, ++ Q931_PROGRESS_INDICATOR, ++ Q931_DISPLAY, ++ Q931_IE_CONNECTED_NUM, ++ Q931_IE_CONNECTED_SUBADDR, ++ -1 ++}; ++ ++int q931_connect(struct pri *ctrl, q931_call *c, int channel, int nonisdn) + { + if (channel) { + c->ds1no = (channel & 0xff00) >> 8; + c->ds1explicit = (channel & 0x10000) >> 16; +- channel &= 0xff; +- c->channelno = channel; ++ c->channelno = channel & 0xff; + } + c->chanflags &= ~FLAG_PREFERRED; + c->chanflags |= FLAG_EXCLUSIVE; +- if (nonisdn && (pri->switchtype != PRI_SWITCH_DMS100)) { ++ if (nonisdn && (ctrl->switchtype != PRI_SWITCH_DMS100)) { + c->progloc = LOC_PRIV_NET_LOCAL_USER; + c->progcode = CODE_CCITT; + c->progressmask = PRI_PROG_CALLED_NOT_ISDN; + } else + c->progressmask = 0; +- if(pri->localtype == PRI_NETWORK || pri->switchtype == PRI_SWITCH_QSIG) +- UPDATE_OURCALLSTATE(pri, c, Q931_CALL_STATE_ACTIVE); ++ if(ctrl->localtype == PRI_NETWORK || ctrl->switchtype == PRI_SWITCH_QSIG) ++ UPDATE_OURCALLSTATE(ctrl, c, Q931_CALL_STATE_ACTIVE); + else +- UPDATE_OURCALLSTATE(pri, c, Q931_CALL_STATE_CONNECT_REQUEST); ++ UPDATE_OURCALLSTATE(ctrl, c, Q931_CALL_STATE_CONNECT_REQUEST); + c->peercallstate = Q931_CALL_STATE_ACTIVE; + c->alive = 1; + /* Connect request timer */ +- if (c->retranstimer) +- pri_schedule_del(pri, c->retranstimer); ++ pri_schedule_del(ctrl, c->retranstimer); + c->retranstimer = 0; +- if ((c->ourcallstate == Q931_CALL_STATE_CONNECT_REQUEST) && (pri->bri || (!pri->subchannel))) +- c->retranstimer = pri_schedule_event(pri, pri->timers[PRI_TIMER_T313], pri_connect_timeout, c); +- return send_message(pri, c, Q931_CONNECT, connect_ies); ++ if ((c->ourcallstate == Q931_CALL_STATE_CONNECT_REQUEST) && (ctrl->bri || (!ctrl->subchannel))) ++ c->retranstimer = pri_schedule_event(ctrl, ctrl->timers[PRI_TIMER_T313], pri_connect_timeout, c); ++ ++ if (c->redirecting.state == Q931_REDIRECTING_STATE_PENDING_TX_DIV_LEG_3) { ++ c->redirecting.state = Q931_REDIRECTING_STATE_IDLE; ++ /* Send DivertingLegInformation3 with CONNECT. */ ++ c->redirecting.to = c->local_id; ++ if (!c->redirecting.to.number.valid) { ++ q931_party_number_init(&c->redirecting.to.number); ++ c->redirecting.to.number.valid = 1; ++ c->redirecting.to.number.presentation = ++ PRI_PRES_RESTRICTED | PRI_PRES_USER_NUMBER_UNSCREENED; ++ } ++ rose_diverting_leg_information3_encode(ctrl, c, Q931_CONNECT); ++ } ++ switch (ctrl->switchtype) { ++ case PRI_SWITCH_QSIG: ++ if (c->local_id.name.valid) { ++ /* Send connectedName with CONNECT */ ++ rose_connected_name_encode(ctrl, c, Q931_CONNECT); ++ } ++ break; ++ default: ++ break; ++ } ++ return send_message(ctrl, c, Q931_CONNECT, connect_ies); + } + + static int release_ies[] = { Q931_CAUSE, Q931_IE_USER_USER, -1 }; + +-int q931_release(struct pri *pri, q931_call *c, int cause) ++int q931_release(struct pri *ctrl, q931_call *c, int cause) + { +- UPDATE_OURCALLSTATE(pri, c, Q931_CALL_STATE_RELEASE_REQUEST); ++ UPDATE_OURCALLSTATE(ctrl, c, Q931_CALL_STATE_RELEASE_REQUEST); + /* c->peercallstate stays the same */ + if (c->alive) { + c->alive = 0; +@@ -2972,26 +4533,26 @@ + c->causecode = CODE_CCITT; + c->causeloc = LOC_PRIV_NET_LOCAL_USER; + if (c->acked) { +- if (c->retranstimer) +- pri_schedule_del(pri, c->retranstimer); ++ pri_schedule_del(ctrl, c->retranstimer); + if (!c->t308_timedout) { +- c->retranstimer = pri_schedule_event(pri, pri->timers[PRI_TIMER_T308], pri_release_timeout, c); ++ c->retranstimer = pri_schedule_event(ctrl, ctrl->timers[PRI_TIMER_T308], pri_release_timeout, c); + } else { +- c->retranstimer = pri_schedule_event(pri, pri->timers[PRI_TIMER_T308], pri_release_finaltimeout, c); ++ c->retranstimer = pri_schedule_event(ctrl, ctrl->timers[PRI_TIMER_T308], pri_release_finaltimeout, c); + } +- return send_message(pri, c, Q931_RELEASE, release_ies); ++ return send_message(ctrl, c, Q931_RELEASE, release_ies); + } else +- return send_message(pri, c, Q931_RELEASE_COMPLETE, release_ies); /* Yes, release_ies, not release_complete_ies */ ++ return send_message(ctrl, c, Q931_RELEASE_COMPLETE, release_ies); /* Yes, release_ies, not release_complete_ies */ + } else + return 0; + } + + static int restart_ies[] = { Q931_CHANNEL_IDENT, Q931_RESTART_INDICATOR, -1 }; + +-int q931_restart(struct pri *pri, int channel) ++int q931_restart(struct pri *ctrl, int channel) + { + struct q931_call *c; +- c = q931_getcall(pri, 0 | 0x8000, 1); ++ ++ c = q931_getcall(ctrl, 0 | 0x8000); + if (!c) + return -1; + if (!channel) +@@ -2999,20 +4560,19 @@ + c->ri = 0; + c->ds1no = (channel & 0xff00) >> 8; + c->ds1explicit = (channel & 0x10000) >> 16; +- channel &= 0xff; +- c->channelno = channel; ++ c->channelno = channel & 0xff; + c->chanflags &= ~FLAG_PREFERRED; + c->chanflags |= FLAG_EXCLUSIVE; +- UPDATE_OURCALLSTATE(pri, c, Q931_CALL_STATE_RESTART); ++ UPDATE_OURCALLSTATE(ctrl, c, Q931_CALL_STATE_RESTART); + c->peercallstate = Q931_CALL_STATE_RESTART_REQUEST; +- return send_message(pri, c, Q931_RESTART, restart_ies); ++ return send_message(ctrl, c, Q931_RESTART, restart_ies); + } + +-static int disconnect_ies[] = { Q931_CAUSE, Q931_IE_USER_USER, -1 }; ++static int disconnect_ies[] = { Q931_CAUSE, Q931_IE_FACILITY, Q931_IE_USER_USER, -1 }; + +-int q931_disconnect(struct pri *pri, q931_call *c, int cause) ++int q931_disconnect(struct pri *ctrl, q931_call *c, int cause) + { +- UPDATE_OURCALLSTATE(pri, c, Q931_CALL_STATE_DISCONNECT_REQUEST); ++ UPDATE_OURCALLSTATE(ctrl, c, Q931_CALL_STATE_DISCONNECT_REQUEST); + c->peercallstate = Q931_CALL_STATE_DISCONNECT_INDICATION; + if (c->alive) { + c->alive = 0; +@@ -3020,27 +4580,125 @@ + c->causecode = CODE_CCITT; + c->causeloc = LOC_PRIV_NET_LOCAL_USER; + c->sendhangupack = 1; +- if (c->retranstimer) +- pri_schedule_del(pri, c->retranstimer); +- c->retranstimer = pri_schedule_event(pri, pri->timers[PRI_TIMER_T305], pri_disconnect_timeout, c); +- return send_message(pri, c, Q931_DISCONNECT, disconnect_ies); ++ pri_schedule_del(ctrl, c->retranstimer); ++ c->retranstimer = pri_schedule_event(ctrl, ctrl->timers[PRI_TIMER_T305], pri_disconnect_timeout, c); ++ return send_message(ctrl, c, Q931_DISCONNECT, disconnect_ies); + } else + return 0; + } + +-static int setup_ies[] = { Q931_BEARER_CAPABILITY, Q931_CHANNEL_IDENT, Q931_IE_FACILITY, Q931_PROGRESS_INDICATOR, Q931_NETWORK_SPEC_FAC, Q931_DISPLAY, +- Q931_CALLING_PARTY_NUMBER, Q931_CALLED_PARTY_NUMBER, Q931_REDIRECTING_NUMBER, Q931_IE_USER_USER, Q931_SENDING_COMPLETE, +- Q931_IE_ORIGINATING_LINE_INFO, Q931_IE_GENERIC_DIGITS, -1 }; ++static int setup_ies[] = { ++ Q931_BEARER_CAPABILITY, ++ Q931_CHANNEL_IDENT, ++ Q931_IE_FACILITY, ++ Q931_PROGRESS_INDICATOR, ++ Q931_NETWORK_SPEC_FAC, ++ Q931_DISPLAY, ++ Q931_IE_KEYPAD_FACILITY, ++ Q931_REVERSE_CHARGE_INDIC, ++ Q931_CALLING_PARTY_NUMBER, ++ Q931_CALLING_PARTY_SUBADDR, ++ Q931_CALLED_PARTY_NUMBER, ++ Q931_CALLED_PARTY_SUBADDR, ++ Q931_REDIRECTING_NUMBER, ++ Q931_IE_USER_USER, ++ Q931_SENDING_COMPLETE, ++ Q931_IE_ORIGINATING_LINE_INFO, ++ Q931_IE_GENERIC_DIGITS, ++ -1 ++}; + +-static int gr303_setup_ies[] = { Q931_BEARER_CAPABILITY, Q931_CHANNEL_IDENT, -1 }; ++static int gr303_setup_ies[] = { ++ Q931_BEARER_CAPABILITY, ++ Q931_CHANNEL_IDENT, ++ -1 ++}; + +-static int cis_setup_ies[] = { Q931_BEARER_CAPABILITY, Q931_CHANNEL_IDENT, Q931_IE_FACILITY, Q931_CALLED_PARTY_NUMBER, -1 }; ++/*! Call Independent Signalling SETUP ie's */ ++static int cis_setup_ies[] = { ++ Q931_BEARER_CAPABILITY, ++ Q931_CHANNEL_IDENT, ++ Q931_IE_FACILITY, ++ Q931_IE_KEYPAD_FACILITY, ++ Q931_CALLING_PARTY_NUMBER, ++ Q931_CALLING_PARTY_SUBADDR, ++ Q931_CALLED_PARTY_NUMBER, ++ Q931_CALLED_PARTY_SUBADDR, ++ Q931_SENDING_COMPLETE, ++ -1 ++}; + +-int q931_setup(struct pri *pri, q931_call *c, struct pri_sr *req) ++static void stop_t303(struct q931_call *call) + { ++ /* T303 should only be running on the master call */ ++ pri_schedule_del(call->master_call->pri, call->master_call->t303_timer); ++ call->master_call->t303_timer = 0; ++} ++ ++static void t303_expiry(void *data); ++ ++static void start_t303(struct q931_call *call) ++{ ++ if (call->t303_timer) { ++ pri_error(call->pri, "Should not have T303 set when starting again. Stopping first\n"); ++ stop_t303(call); ++ } ++ ++ //pri_error(call->pri, "T303 should be %d\n", call->pri->timers[PRI_TIMER_T303]); ++ call->t303_timer = pri_schedule_event(call->pri, call->pri->timers[PRI_TIMER_T303], t303_expiry, call); ++} ++ ++static void pri_fake_clearing(void *data); ++ ++static void t303_expiry(void *data) ++{ ++ struct q931_call *c = data; ++ struct pri *ctrl = c->pri; + int res; +- +- ++ ++ c->t303_expirycnt++; ++ c->t303_timer = 0; ++ ++ if (c->cause != -1) { ++ /* We got a DISCONNECT, RELEASE, or RELEASE_COMPLETE and no other responses. */ ++ pri_fake_clearing(c); ++ } else if (c->t303_expirycnt < 2) { ++ if (ctrl->subchannel && !ctrl->bri) ++ res = send_message(ctrl, c, Q931_SETUP, gr303_setup_ies); ++ else if (c->cis_call) ++ res = send_message(ctrl, c, Q931_SETUP, cis_setup_ies); ++ else ++ res = send_message(ctrl, c, Q931_SETUP, setup_ies); ++ ++ if (res) { ++ pri_error(c->pri, "Error resending setup message!\n"); ++ } ++ start_t303(c); ++ } else { ++ c->cause = PRI_CAUSE_NO_USER_RESPONSE; ++ pri_fake_clearing(c); ++ } ++} ++ ++int q931_setup(struct pri *ctrl, q931_call *c, struct pri_sr *req) ++{ ++ int res; ++ ++ if (!req->called.number.valid && (!req->keypad_digits || !req->keypad_digits[0])) { ++ /* No called number or keypad digits to send. */ ++ return -1; ++ } ++ ++ c->called = req->called; ++ libpri_copy_string(c->overlap_digits, req->called.number.str, sizeof(c->overlap_digits)); ++ ++ if (req->keypad_digits) { ++ libpri_copy_string(c->keypad_digits, req->keypad_digits, ++ sizeof(c->keypad_digits)); ++ } else { ++ c->keypad_digits[0] = '\0'; ++ } ++ + c->transcapability = req->transmode; + c->transmoderate = TRANS_MODE_64_CIRCUIT; + if (!req->userl1) +@@ -3050,89 +4708,68 @@ + c->userl3 = -1; + c->ds1no = (req->channel & 0xff00) >> 8; + c->ds1explicit = (req->channel & 0x10000) >> 16; +- req->channel &= 0xff; +- if ((pri->localtype == PRI_CPE) && pri->subchannel && !pri->bri) { +- req->channel = 0; +- req->exclusive = 0; ++ if ((ctrl->localtype == PRI_CPE) && ctrl->subchannel && !ctrl->bri) { ++ c->channelno = 0; ++ c->chanflags = 0; ++ } else { ++ c->channelno = req->channel & 0xff; ++ if (req->exclusive) { ++ c->chanflags = FLAG_EXCLUSIVE; ++ } else { ++ c->chanflags = FLAG_PREFERRED; ++ } + } +- +- c->channelno = req->channel; ++ + c->slotmap = -1; + c->nonisdn = req->nonisdn; + c->newcall = 0; +- c->justsignalling = req->justsignalling; ++ c->cis_call = req->cis_call; ++ c->cis_auto_disconnect = req->cis_auto_disconnect; + c->complete = req->numcomplete; +- if (req->exclusive) +- c->chanflags = FLAG_EXCLUSIVE; +- else if (c->channelno) +- c->chanflags = FLAG_PREFERRED; +- if (req->caller) { +- libpri_copy_string(c->callernum, req->caller, sizeof(c->callernum)); +- c->callerplan = req->callerplan; +- if (req->callername) +- libpri_copy_string(c->callername, req->callername, sizeof(c->callername)); +- else +- c->callername[0] = '\0'; +- if ((pri->switchtype == PRI_SWITCH_DMS100) || +- (pri->switchtype == PRI_SWITCH_ATT4ESS)) { +- /* Doesn't like certain presentation types */ +- if (!(req->callerpres & 0x7c)) +- req->callerpres = PRES_ALLOWED_NETWORK_NUMBER; +- } +- c->callerpres = req->callerpres; +- } else { +- c->callernum[0] = '\0'; +- c->callername[0] = '\0'; +- c->callerplan = PRI_UNKNOWN; +- c->callerpres = PRES_NUMBER_NOT_AVAILABLE; ++ ++ if (req->caller.number.valid) { ++ c->local_id = req->caller; ++ q931_party_id_fixup(ctrl, &c->local_id); + } +- if (req->redirectingnum) { +- libpri_copy_string(c->redirectingnum, req->redirectingnum, sizeof(c->redirectingnum)); +- c->redirectingplan = req->redirectingplan; +- if ((pri->switchtype == PRI_SWITCH_DMS100) || +- (pri->switchtype == PRI_SWITCH_ATT4ESS)) { +- /* Doesn't like certain presentation types */ +- if (!(req->redirectingpres & 0x7c)) +- req->redirectingpres = PRES_ALLOWED_NETWORK_NUMBER; +- } +- c->redirectingpres = req->redirectingpres; +- c->redirectingreason = req->redirectingreason; +- } else { +- c->redirectingnum[0] = '\0'; +- c->redirectingplan = PRI_UNKNOWN; +- c->redirectingpres = PRES_NUMBER_NOT_AVAILABLE; +- c->redirectingreason = PRI_REDIR_UNKNOWN; ++ ++ if (req->redirecting.from.number.valid) { ++ c->redirecting = req->redirecting; ++ q931_party_id_fixup(ctrl, &c->redirecting.from); ++ q931_party_id_fixup(ctrl, &c->redirecting.to); ++ q931_party_id_fixup(ctrl, &c->redirecting.orig_called); + } +- if (req->called) { +- libpri_copy_string(c->callednum, req->called, sizeof(c->callednum)); +- c->calledplan = req->calledplan; +- } else +- return -1; + + if (req->useruserinfo) + libpri_copy_string(c->useruserinfo, req->useruserinfo, sizeof(c->useruserinfo)); + else + c->useruserinfo[0] = '\0'; + +- if (req->nonisdn && (pri->switchtype == PRI_SWITCH_NI2)) ++ if (req->nonisdn && (ctrl->switchtype == PRI_SWITCH_NI2)) + c->progressmask = PRI_PROG_CALLER_NOT_ISDN; + else + c->progressmask = 0; + +- pri_call_add_standard_apdus(pri, c); ++ c->reversecharge = req->reversecharge; + +- if (pri->subchannel && !pri->bri) +- res = send_message(pri, c, Q931_SETUP, gr303_setup_ies); +- else if (c->justsignalling) +- res = send_message(pri, c, Q931_SETUP, cis_setup_ies); ++ pri_call_add_standard_apdus(ctrl, c); ++ ++ if (ctrl->subchannel && !ctrl->bri) ++ res = send_message(ctrl, c, Q931_SETUP, gr303_setup_ies); ++ else if (c->cis_call) ++ res = send_message(ctrl, c, Q931_SETUP, cis_setup_ies); + else +- res = send_message(pri, c, Q931_SETUP, setup_ies); ++ res = send_message(ctrl, c, Q931_SETUP, setup_ies); + if (!res) { + c->alive = 1; + /* make sure we call PRI_EVENT_HANGUP_ACK once we send/receive RELEASE_COMPLETE */ + c->sendhangupack = 1; +- UPDATE_OURCALLSTATE(pri, c, Q931_CALL_STATE_CALL_INITIATED); +- c->peercallstate = Q931_CALL_STATE_OVERLAP_SENDING; ++ UPDATE_OURCALLSTATE(ctrl, c, Q931_CALL_STATE_CALL_INITIATED); ++ c->peercallstate = Q931_CALL_STATE_CALL_PRESENT; ++ c->t303_expirycnt = 0; ++ if (BRI_NT_PTMP(ctrl)) { ++ c->outboundbroadcast = 1; ++ } ++ start_t303(c); + } + return res; + +@@ -3140,22 +4777,22 @@ + + static int release_complete_ies[] = { Q931_IE_USER_USER, -1 }; + +-static int q931_release_complete(struct pri *pri, q931_call *c, int cause) ++static int q931_release_complete(struct pri *ctrl, q931_call *c, int cause) + { + int res = 0; +- UPDATE_OURCALLSTATE(pri, c, Q931_CALL_STATE_NULL); ++ UPDATE_OURCALLSTATE(ctrl, c, Q931_CALL_STATE_NULL); + c->peercallstate = Q931_CALL_STATE_NULL; + if (cause > -1) { + c->cause = cause; + c->causecode = CODE_CCITT; + c->causeloc = LOC_PRIV_NET_LOCAL_USER; + /* release_ies has CAUSE in it */ +- res = send_message(pri, c, Q931_RELEASE_COMPLETE, release_ies); ++ res = send_message(ctrl, c, Q931_RELEASE_COMPLETE, release_ies); + } else +- res = send_message(pri, c, Q931_RELEASE_COMPLETE, release_complete_ies); ++ res = send_message(ctrl, c, Q931_RELEASE_COMPLETE, release_complete_ies); + c->alive = 0; + /* release the structure */ +- res += q931_hangup(pri,c,cause); ++ res += pri_hangup(ctrl, c, cause); + return res; + } + +@@ -3163,46 +4800,553 @@ + + static int gr303_connect_acknowledge_ies[] = { Q931_CHANNEL_IDENT, -1 }; + +-static int q931_connect_acknowledge(struct pri *pri, q931_call *c) ++static int q931_connect_acknowledge(struct pri *ctrl, q931_call *c) + { +- if (pri->subchannel && !pri->bri) { +- if (pri->localtype == PRI_CPE) +- return send_message(pri, c, Q931_CONNECT_ACKNOWLEDGE, gr303_connect_acknowledge_ies); ++ if (ctrl->subchannel && !ctrl->bri) { ++ if (ctrl->localtype == PRI_CPE) ++ return send_message(ctrl, c, Q931_CONNECT_ACKNOWLEDGE, gr303_connect_acknowledge_ies); + } else +- return send_message(pri, c, Q931_CONNECT_ACKNOWLEDGE, connect_acknowledge_ies); ++ return send_message(ctrl, c, Q931_CONNECT_ACKNOWLEDGE, connect_acknowledge_ies); + return 0; + } + +-int q931_hangup(struct pri *pri, q931_call *c, int cause) ++/*! ++ * \internal ++ * \brief Find the winning subcall if it exists or current call if not outboundbroadcast. ++ * ++ * \param call Starting Q.931 call record of search. ++ * ++ * \retval winning-call or given call if not outboundbroadcast. ++ * \retval NULL if no winning call yet. ++ */ ++static struct q931_call *q931_find_winning_call(struct q931_call *call) + { ++ struct q931_call *master; ++ ++ master = call->master_call; ++ if (master->outboundbroadcast) { ++ /* We have potential subcalls. Now get the winning call if declared yet. */ ++ if (master->pri_winner < 0) { ++ /* Winner not declared yet.*/ ++ call = NULL; ++ } else { ++ call = master->subcalls[master->pri_winner]; ++ } ++ } ++ return call; ++} ++ ++/*! ++ * \internal ++ * \brief Send HOLD message response wait timeout. ++ * ++ * \param data Q.931 call leg. (Master Q.931 subcall structure) ++ * ++ * \return Nothing ++ */ ++static void q931_hold_timeout(void *data) ++{ ++ struct q931_call *call = data; ++ struct pri *ctrl = call->pri; ++ ++ if (ctrl->debug & PRI_DEBUG_Q931_STATE) { ++ pri_message(ctrl, "Time-out waiting for HOLD response\n"); ++ } ++ ++ /* Ensure that the timer is deleted. */ ++ pri_schedule_del(ctrl, call->hold_timer); ++ call->hold_timer = 0; ++ ++ UPDATE_HOLD_STATE(ctrl, call, Q931_HOLD_STATE_IDLE); ++ ++ q931_clr_subcommands(ctrl); ++ ctrl->schedev = 1; ++ ctrl->ev.e = PRI_EVENT_HOLD_REJ; ++ ctrl->ev.hold_rej.channel = q931_encode_channel(call); ++ ctrl->ev.hold_rej.call = call; ++ ctrl->ev.hold_rej.cause = PRI_CAUSE_MESSAGE_TYPE_NONEXIST; ++ ctrl->ev.hold_rej.subcmds = &ctrl->subcmds; ++} ++ ++/*! ++ * \internal ++ * \brief Determine if a hold request is allowed now. ++ * ++ * \param ctrl D channel controller. ++ * \param call Q.931 call leg. (Master Q.931 subcall structure) ++ * ++ * \retval TRUE if we can send a HOLD request. ++ * \retval FALSE if not allowed. ++ */ ++static int q931_is_hold_allowed(const struct pri *ctrl, const struct q931_call *call) ++{ ++ int allowed; ++ ++ allowed = 0; ++ switch (call->ourcallstate) { ++ case Q931_CALL_STATE_CALL_RECEIVED: ++ case Q931_CALL_STATE_CONNECT_REQUEST: ++ case Q931_CALL_STATE_INCOMING_CALL_PROCEEDING: ++ if (q931_is_ptmp(ctrl)) { ++ /* HOLD request only allowed in these states if point-to-point mode. */ ++ break; ++ } ++ /* Fall through */ ++ case Q931_CALL_STATE_OUTGOING_CALL_PROCEEDING: ++ case Q931_CALL_STATE_CALL_DELIVERED: ++ case Q931_CALL_STATE_ACTIVE: ++ switch (call->hold_state) { ++ case Q931_HOLD_STATE_IDLE: ++ allowed = 1; ++ break; ++ default: ++ break; ++ } ++ break; ++ case Q931_CALL_STATE_DISCONNECT_INDICATION: ++ case Q931_CALL_STATE_RELEASE_REQUEST: ++ /* Ignore HOLD request in these states. */ ++ break; ++ default: ++ break; ++ } ++ ++ return allowed; ++} ++ ++static int hold_ies[] = { ++ -1 ++}; ++ ++/*! ++ * \brief Send the HOLD message. ++ * ++ * \param ctrl D channel controller. ++ * \param call Q.931 call leg. (Master Q.931 subcall structure) ++ * ++ * \retval 0 on success. ++ * \retval -1 on error. ++ */ ++int q931_send_hold(struct pri *ctrl, struct q931_call *call) ++{ ++ struct q931_call *winner; ++ ++ winner = q931_find_winning_call(call); ++ if (!winner || !q931_is_hold_allowed(ctrl, call)) { ++ return -1; ++ } ++ pri_schedule_del(ctrl, call->hold_timer); ++ call->hold_timer = pri_schedule_event(ctrl, ctrl->timers[PRI_TIMER_T_HOLD], ++ q931_hold_timeout, call); ++ if (send_message(ctrl, winner, Q931_HOLD, hold_ies)) { ++ pri_schedule_del(ctrl, call->hold_timer); ++ call->hold_timer = 0; ++ return -1; ++ } ++ UPDATE_HOLD_STATE(ctrl, call, Q931_HOLD_STATE_HOLD_REQ); ++ return 0; ++} ++ ++static int hold_ack_ies[] = { ++ -1 ++}; ++ ++/*! ++ * \brief Send the HOLD ACKNOWLEDGE message. ++ * ++ * \param ctrl D channel controller. ++ * \param call Q.931 call leg. (Master Q.931 subcall structure) ++ * ++ * \retval 0 on success. ++ * \retval -1 on error. ++ */ ++int q931_send_hold_ack(struct pri *ctrl, struct q931_call *call) ++{ ++ struct q931_call *winner; ++ ++ UPDATE_HOLD_STATE(ctrl, call, Q931_HOLD_STATE_CALL_HELD); ++ ++ winner = q931_find_winning_call(call); ++ if (!winner) { ++ return -1; ++ } ++ ++ /* Call is now on hold so forget the channel. */ ++ winner->channelno = 0;/* No channel */ ++ winner->ds1no = 0; ++ winner->ds1explicit = 0; ++ winner->chanflags = 0; ++ ++ return send_message(ctrl, winner, Q931_HOLD_ACKNOWLEDGE, hold_ack_ies); ++} ++ ++static int hold_reject_ies[] = { ++ Q931_CAUSE, ++ -1 ++}; ++ ++/*! ++ * \internal ++ * \brief Send the HOLD REJECT message only. ++ * ++ * \param ctrl D channel controller. ++ * \param call Q.931 call leg. (subcall) ++ * \param cause Q.931 cause code for rejecting the hold request. ++ * ++ * \retval 0 on success. ++ * \retval -1 on error. ++ */ ++static int q931_send_hold_rej_msg(struct pri *ctrl, struct q931_call *call, int cause) ++{ ++ call->cause = cause; ++ call->causecode = CODE_CCITT; ++ call->causeloc = LOC_PRIV_NET_LOCAL_USER; ++ return send_message(ctrl, call, Q931_HOLD_REJECT, hold_reject_ies); ++} ++ ++/*! ++ * \brief Send the HOLD REJECT message. ++ * ++ * \param ctrl D channel controller. ++ * \param call Q.931 call leg. (Master Q.931 subcall structure) ++ * \param cause Q.931 cause code for rejecting the hold request. ++ * ++ * \retval 0 on success. ++ * \retval -1 on error. ++ */ ++int q931_send_hold_rej(struct pri *ctrl, struct q931_call *call, int cause) ++{ ++ struct q931_call *winner; ++ ++ UPDATE_HOLD_STATE(ctrl, call, Q931_HOLD_STATE_IDLE); ++ ++ winner = q931_find_winning_call(call); ++ if (!winner) { ++ return -1; ++ } ++ ++ return q931_send_hold_rej_msg(ctrl, winner, cause); ++} ++ ++/*! ++ * \internal ++ * \brief Send RETRIEVE message response wait timeout. ++ * ++ * \param data Q.931 call leg. (Master Q.931 subcall structure) ++ * ++ * \return Nothing ++ */ ++static void q931_retrieve_timeout(void *data) ++{ ++ struct q931_call *call = data; ++ struct pri *ctrl = call->pri; ++ struct q931_call *winner; ++ ++ if (ctrl->debug & PRI_DEBUG_Q931_STATE) { ++ pri_message(ctrl, "Time-out waiting for RETRIEVE response\n"); ++ } ++ ++ /* Ensure that the timer is deleted. */ ++ pri_schedule_del(ctrl, call->hold_timer); ++ call->hold_timer = 0; ++ ++ UPDATE_HOLD_STATE(ctrl, call, Q931_HOLD_STATE_CALL_HELD); ++ ++ winner = q931_find_winning_call(call); ++ if (winner) { ++ /* Call is still on hold so forget the channel. */ ++ winner->channelno = 0;/* No channel */ ++ winner->ds1no = 0; ++ winner->ds1explicit = 0; ++ winner->chanflags = 0; ++ } ++ ++ q931_clr_subcommands(ctrl); ++ ctrl->schedev = 1; ++ ctrl->ev.e = PRI_EVENT_RETRIEVE_REJ; ++ ctrl->ev.retrieve_rej.channel = q931_encode_channel(call); ++ ctrl->ev.retrieve_rej.call = call; ++ ctrl->ev.retrieve_rej.cause = PRI_CAUSE_MESSAGE_TYPE_NONEXIST; ++ ctrl->ev.retrieve_rej.subcmds = &ctrl->subcmds; ++} ++ ++/*! ++ * \internal ++ * \brief Determine if a retrieve request is allowed now. ++ * ++ * \param ctrl D channel controller. ++ * \param call Q.931 call leg. (Master Q.931 subcall structure) ++ * ++ * \retval TRUE if we can send a RETRIEVE request. ++ * \retval FALSE if not allowed. ++ */ ++static int q931_is_retrieve_allowed(const struct pri *ctrl, const struct q931_call *call) ++{ ++ int allowed; ++ ++ allowed = 0; ++ switch (call->ourcallstate) { ++ case Q931_CALL_STATE_CALL_RECEIVED: ++ case Q931_CALL_STATE_CONNECT_REQUEST: ++ case Q931_CALL_STATE_INCOMING_CALL_PROCEEDING: ++ if (q931_is_ptmp(ctrl)) { ++ /* RETRIEVE request only allowed in these states if point-to-point mode. */ ++ break; ++ } ++ /* Fall through */ ++ case Q931_CALL_STATE_OUTGOING_CALL_PROCEEDING: ++ case Q931_CALL_STATE_CALL_DELIVERED: ++ case Q931_CALL_STATE_ACTIVE: ++ switch (call->hold_state) { ++ case Q931_HOLD_STATE_CALL_HELD: ++ allowed = 1; ++ break; ++ default: ++ break; ++ } ++ break; ++ case Q931_CALL_STATE_DISCONNECT_INDICATION: ++ case Q931_CALL_STATE_RELEASE_REQUEST: ++ /* Ignore RETRIEVE request in these states. */ ++ break; ++ default: ++ break; ++ } ++ ++ return allowed; ++} ++ ++static int retrieve_ies[] = { ++ Q931_CHANNEL_IDENT, ++ -1 ++}; ++ ++/*! ++ * \brief Send the RETRIEVE message. ++ * ++ * \param ctrl D channel controller. ++ * \param call Q.931 call leg. (Master Q.931 subcall structure) ++ * \param channel Encoded channel id to use. If zero do not send channel id. ++ * ++ * \retval 0 on success. ++ * \retval -1 on error. ++ */ ++int q931_send_retrieve(struct pri *ctrl, struct q931_call *call, int channel) ++{ ++ struct q931_call *winner; ++ ++ winner = q931_find_winning_call(call); ++ if (!winner || !q931_is_retrieve_allowed(ctrl, call)) { ++ return -1; ++ } ++ ++ if (channel) { ++ winner->ds1no = (channel & 0xff00) >> 8; ++ winner->ds1explicit = (channel & 0x10000) >> 16; ++ winner->channelno = channel & 0xff; ++ if (ctrl->localtype == PRI_NETWORK) { ++ winner->chanflags = FLAG_EXCLUSIVE; ++ } else { ++ winner->chanflags = FLAG_PREFERRED; ++ } ++ } else { ++ /* Do not send Q931_CHANNEL_IDENT */ ++ winner->chanflags = 0; ++ } ++ ++ pri_schedule_del(ctrl, call->hold_timer); ++ call->hold_timer = pri_schedule_event(ctrl, ctrl->timers[PRI_TIMER_T_RETRIEVE], ++ q931_retrieve_timeout, call); ++ if (send_message(ctrl, winner, Q931_RETRIEVE, retrieve_ies)) { ++ pri_schedule_del(ctrl, call->hold_timer); ++ call->hold_timer = 0; ++ ++ /* Call is still on hold so forget the channel. */ ++ winner->channelno = 0;/* No channel */ ++ winner->ds1no = 0; ++ winner->ds1explicit = 0; ++ winner->chanflags = 0; ++ return -1; ++ } ++ UPDATE_HOLD_STATE(ctrl, call, Q931_HOLD_STATE_RETRIEVE_REQ); ++ return 0; ++} ++ ++static int retrieve_ack_ies[] = { ++ Q931_CHANNEL_IDENT, ++ -1 ++}; ++ ++/*! ++ * \brief Send the RETRIEVE ACKNOWLEDGE message. ++ * ++ * \param ctrl D channel controller. ++ * \param call Q.931 call leg. (Master Q.931 subcall structure) ++ * \param channel Encoded channel id to use. ++ * ++ * \retval 0 on success. ++ * \retval -1 on error. ++ */ ++int q931_send_retrieve_ack(struct pri *ctrl, struct q931_call *call, int channel) ++{ ++ struct q931_call *winner; ++ ++ winner = q931_find_winning_call(call); ++ if (!winner) { ++ return -1; ++ } ++ ++ winner->ds1no = (channel & 0xff00) >> 8; ++ winner->ds1explicit = (channel & 0x10000) >> 16; ++ winner->channelno = channel & 0xff; ++ winner->chanflags = FLAG_EXCLUSIVE; ++ ++ UPDATE_HOLD_STATE(ctrl, call, Q931_HOLD_STATE_IDLE); ++ ++ return send_message(ctrl, winner, Q931_RETRIEVE_ACKNOWLEDGE, retrieve_ack_ies); ++} ++ ++static int retrieve_reject_ies[] = { ++ Q931_CAUSE, ++ -1 ++}; ++ ++/*! ++ * \internal ++ * \brief Send the RETRIEVE REJECT message only. ++ * ++ * \param ctrl D channel controller. ++ * \param call Q.931 call leg. (subcall) ++ * \param cause Q.931 cause code for rejecting the retrieve request. ++ * ++ * \retval 0 on success. ++ * \retval -1 on error. ++ */ ++static int q931_send_retrieve_rej_msg(struct pri *ctrl, struct q931_call *call, int cause) ++{ ++ call->cause = cause; ++ call->causecode = CODE_CCITT; ++ call->causeloc = LOC_PRIV_NET_LOCAL_USER; ++ return send_message(ctrl, call, Q931_RETRIEVE_REJECT, retrieve_reject_ies); ++} ++ ++/*! ++ * \brief Send the RETRIEVE REJECT message. ++ * ++ * \param ctrl D channel controller. ++ * \param call Q.931 call leg. (Master Q.931 subcall structure) ++ * \param cause Q.931 cause code for rejecting the retrieve request. ++ * ++ * \retval 0 on success. ++ * \retval -1 on error. ++ */ ++int q931_send_retrieve_rej(struct pri *ctrl, struct q931_call *call, int cause) ++{ ++ struct q931_call *winner; ++ ++ UPDATE_HOLD_STATE(ctrl, call, Q931_HOLD_STATE_CALL_HELD); ++ ++ winner = q931_find_winning_call(call); ++ if (!winner) { ++ return -1; ++ } ++ ++ /* Call is still on hold so forget the channel. */ ++ winner->channelno = 0;/* No channel */ ++ winner->ds1no = 0; ++ winner->ds1explicit = 0; ++ winner->chanflags = 0; ++ ++ return q931_send_retrieve_rej_msg(ctrl, winner, cause); ++} ++ ++static int pri_internal_clear(void *data); ++ ++/* Fake RELEASE for NT-PTMP initiated SETUPs w/o response */ ++static void pri_fake_clearing(void *data) ++{ ++ struct q931_call *c = data; ++ struct pri *ctrl = c->pri; ++ ++ c->performing_fake_clearing = 1; ++ if (pri_internal_clear(c) == Q931_RES_HAVEEVENT) ++ ctrl->schedev = 1; ++} ++ ++static void pri_create_fake_clearing(struct q931_call *c, struct pri *master) ++{ ++ c->pri = master; ++ ++ pri_schedule_del(master, c->retranstimer); ++ c->retranstimer = pri_schedule_event(master, 0, pri_fake_clearing, c); ++} ++ ++//static int q931_get_subcall_count(struct q931_call *call); ++ ++static int __q931_hangup(struct pri *ctrl, q931_call *c, int cause) ++{ + int disconnect = 1; + int release_compl = 0; +- if (pri->debug & PRI_DEBUG_Q931_STATE) +- pri_message(pri, "NEW_HANGUP DEBUG: Calling q931_hangup, ourstate %s, peerstate %s\n",callstate2str(c->ourcallstate),callstate2str(c->peercallstate)); +- if (!pri || !c) ++ int t303_was_running = c->master_call->t303_timer; ++ ++ if (ctrl->debug & PRI_DEBUG_Q931_STATE) ++ pri_message(ctrl, ++ "NEW_HANGUP DEBUG: Calling q931_hangup, ourstate %s, peerstate %s, hold-state %s\n", ++ q931_call_state_str(c->ourcallstate), ++ q931_call_state_str(c->peercallstate), ++ q931_hold_state_str(c->master_call->hold_state)); ++ if (!ctrl || !c) + return -1; + /* If mandatory IE was missing, insist upon that cause code */ + if (c->cause == PRI_CAUSE_MANDATORY_IE_MISSING) + cause = c->cause; +- if (cause == 34 || cause == 44 || cause == 82 || cause == 1 || cause == 81) { ++ switch (cause) { ++ case PRI_CAUSE_NORMAL_CIRCUIT_CONGESTION: ++ case PRI_CAUSE_REQUESTED_CHAN_UNAVAIL: ++ case PRI_CAUSE_IDENTIFIED_CHANNEL_NOTEXIST: ++ case PRI_CAUSE_UNALLOCATED: ++ case PRI_CAUSE_INVALID_CALL_REFERENCE: + /* We'll send RELEASE_COMPLETE with these causes */ + disconnect = 0; + release_compl = 1; +- } +- if (cause == 6 || cause == 7 || cause == 26) { ++ break; ++ case PRI_CAUSE_CHANNEL_UNACCEPTABLE: ++ case PRI_CAUSE_CALL_AWARDED_DELIVERED: ++ case PRI_CAUSE_NONSELECTED_USER_CLEARING: + /* We'll send RELEASE with these causes */ + disconnect = 0; ++ break; ++ default: ++ break; + } ++ if (c->cis_call) { ++ disconnect = 0; ++ } ++ ++ c->hangupinitiated = 1; ++ stop_t303(c); ++ + /* All other causes we send with DISCONNECT */ + switch(c->ourcallstate) { + case Q931_CALL_STATE_NULL: + if (c->peercallstate == Q931_CALL_STATE_NULL) + /* free the resources if we receive or send REL_COMPL */ +- q931_destroycall(pri, c->cr); ++ pri_destroycall(ctrl, c); + else if (c->peercallstate == Q931_CALL_STATE_RELEASE_REQUEST) +- q931_release_complete(pri,c,cause); ++ q931_release_complete(ctrl,c,cause); + break; + case Q931_CALL_STATE_CALL_INITIATED: ++ if (c->outboundbroadcast && c->master_call == c && t303_was_running) { ++ //c->fakeclearing = 1; ++ //c->alive = 0; ++ /* We need to fake a received clearing sequence in this case... */ ++ if (ctrl->debug & PRI_DEBUG_Q931_STATE) { ++ pri_message(ctrl, "Faking clearing\n"); ++ } ++ pri_create_fake_clearing(c, PRI_MASTER(ctrl)); ++ /* This means that we never got a response from a TEI */ ++ return 0; ++ } + /* we sent SETUP */ + case Q931_CALL_STATE_OVERLAP_SENDING: + /* received SETUP_ACKNOWLEDGE */ +@@ -3221,29 +5365,45 @@ + case Q931_CALL_STATE_OVERLAP_RECEIVING: + /* received SETUP_ACKNOWLEDGE */ + /* send DISCONNECT in general */ +- if (c->peercallstate != Q931_CALL_STATE_NULL && c->peercallstate != Q931_CALL_STATE_DISCONNECT_REQUEST && c->peercallstate != Q931_CALL_STATE_DISCONNECT_INDICATION && c->peercallstate != Q931_CALL_STATE_RELEASE_REQUEST && c->peercallstate != Q931_CALL_STATE_RESTART_REQUEST && c->peercallstate != Q931_CALL_STATE_RESTART) { ++ switch (c->peercallstate) { ++ default: + if (disconnect) +- q931_disconnect(pri,c,cause); ++ q931_disconnect(ctrl,c,cause); + else if (release_compl) +- q931_release_complete(pri,c,cause); ++ q931_release_complete(ctrl,c,cause); + else +- q931_release(pri,c,cause); +- } else +- pri_error(pri, "Wierd, doing nothing but this shouldn't happen, ourstate %s, peerstate %s\n",callstate2str(c->ourcallstate),callstate2str(c->peercallstate)); ++ q931_release(ctrl,c,cause); ++ break; ++ case Q931_CALL_STATE_NULL: ++ case Q931_CALL_STATE_DISCONNECT_REQUEST: ++ case Q931_CALL_STATE_DISCONNECT_INDICATION: ++ case Q931_CALL_STATE_RELEASE_REQUEST: ++ case Q931_CALL_STATE_RESTART_REQUEST: ++ case Q931_CALL_STATE_RESTART: ++ pri_error(ctrl, ++ "Wierd, doing nothing but this shouldn't happen, ourstate %s, peerstate %s\n", ++ q931_call_state_str(c->ourcallstate), ++ q931_call_state_str(c->peercallstate)); ++ break; ++ } + break; + case Q931_CALL_STATE_ACTIVE: + /* received CONNECT */ +- q931_disconnect(pri,c,cause); ++ if (c->cis_call) { ++ q931_release(ctrl, c, cause); ++ break; ++ } ++ q931_disconnect(ctrl,c,cause); + break; + case Q931_CALL_STATE_DISCONNECT_REQUEST: + /* sent DISCONNECT */ +- q931_release(pri,c,cause); ++ q931_release(ctrl,c,cause); + break; + case Q931_CALL_STATE_DISCONNECT_INDICATION: + /* received DISCONNECT */ + if (c->peercallstate == Q931_CALL_STATE_DISCONNECT_REQUEST) { + c->alive = 1; +- q931_release(pri,c,cause); ++ q931_release(ctrl,c,cause); + } + break; + case Q931_CALL_STATE_RELEASE_REQUEST: +@@ -3253,75 +5413,129 @@ + case Q931_CALL_STATE_RESTART: + case Q931_CALL_STATE_RESTART_REQUEST: + /* sent RESTART */ +- pri_error(pri, "q931_hangup shouldn't be called in this state, ourstate %s, peerstate %s\n",callstate2str(c->ourcallstate),callstate2str(c->peercallstate)); ++ pri_error(ctrl, ++ "q931_hangup shouldn't be called in this state, ourstate %s, peerstate %s\n", ++ q931_call_state_str(c->ourcallstate), ++ q931_call_state_str(c->peercallstate)); + break; + default: +- pri_error(pri, "We're not yet handling hanging up when our state is %d, contact support@digium.com, ourstate %s, peerstate %s\n", +- c->ourcallstate, +- callstate2str(c->ourcallstate), +- callstate2str(c->peercallstate)); ++ pri_error(ctrl, ++ "We're not yet handling hanging up when our state is %d, contact support@digium.com, ourstate %s, peerstate %s\n", ++ c->ourcallstate, ++ q931_call_state_str(c->ourcallstate), ++ q931_call_state_str(c->peercallstate)); + return -1; + } + /* we did handle hangup properly at this point */ + return 0; + } + +-int q931_receive(struct pri *pri, q931_h *h, int len) ++static void initiate_hangup_if_needed(struct pri *pri, q931_call *call, int cause); ++ ++int q931_hangup(struct pri *ctrl, q931_call *call, int cause) + { +- q931_mh *mh; +- q931_call *c; +- q931_ie *ie; +- unsigned int x; +- int y; +- int res; +- int r; +- int mandies[MAX_MAND_IES]; +- int missingmand; +- int codeset, cur_codeset; +- int last_ie[8]; +- struct apdu_event *cur = NULL; ++ int i; + +- memset(last_ie, 0, sizeof(last_ie)); +- if (pri->debug & PRI_DEBUG_Q931_DUMP) +- q931_dump(pri, h, len, 0); +-#ifdef LIBPRI_COUNTERS +- pri->q931_rxcount++; +-#endif +- mh = (q931_mh *)(h->contents + h->crlen); +- if ((h->pd == 0x3) || (h->pd == 0x43)) { +- /* This is the weird maintenance stuff. We majorly +- KLUDGE this by changing byte 4 from a 0xf (SERVICE) +- to a 0x7 (SERVICE ACKNOWLEDGE) */ +- h->raw[h->crlen + 2] -= 0x8; +- q931_xmit(pri, h, len, 1); +- return 0; +- } else if (h->pd != pri->protodisc) { +- pri_error(pri, "Warning: unknown/inappropriate protocol discriminator received (%02x/%d)\n", h->pd, h->pd); +- return 0; ++ if (call->master_call->outboundbroadcast) { ++ if (call->master_call == call) { ++ int slaves = 0; ++ ++ /* Master is called with hangup - initiate hangup with slaves */ ++ for (i = 0; i < Q931_MAX_TEI; i++) { ++ if (call->subcalls[i]) { ++ slaves++; ++ if (i == call->master_call->pri_winner) { ++ __q931_hangup(call->subcalls[i]->pri, call->subcalls[i], cause); ++ } else { ++ initiate_hangup_if_needed(call->subcalls[i]->pri, call->subcalls[i], cause); ++ } ++ if (ctrl->debug & PRI_DEBUG_Q931_STATE) { ++ pri_message(ctrl, "%s: Hanging up %d, winner %d\n", __FUNCTION__, ++ i, call->master_call->pri_winner); ++ } ++ } ++ } ++ ++ call->hangupinitiated = 1; ++ ++ if ((!slaves && (call->master_call->pri_winner < 0)) || (call->performing_fake_clearing)) { ++ __q931_hangup(ctrl, call, cause); ++ } ++ if (ctrl->debug & PRI_DEBUG_Q931_STATE) { ++ pri_message(ctrl, "%s: Slaves %d\n", __FUNCTION__, slaves); ++ } ++ return 0; ++ } else { ++ if (ctrl->debug & PRI_DEBUG_Q931_STATE) { ++ pri_message(ctrl, "%s: Slave hangup\n", __FUNCTION__); ++ } ++ return __q931_hangup(ctrl, call, cause); ++ } ++ } else { ++ if (ctrl->debug & PRI_DEBUG_Q931_STATE) { ++ pri_message(ctrl, "%s: other hangup\n", __FUNCTION__); ++ } ++ return __q931_hangup(ctrl, call, cause); + } +- c = q931_getcall(pri, q931_cr(h), 0); +- if (!c) { +- pri_error(pri, "Unable to locate call %d\n", q931_cr(h)); ++ return 0; ++} ++ ++static int prepare_to_handle_maintenance_message(struct pri *ctrl, q931_mh *mh, q931_call *c) ++{ ++ if ((!ctrl) || (!mh) || (!c)) { + return -1; + } +- /* Preliminary handling */ ++ /* SERVICE messages are a superset of messages that can take b-channels ++ * or entire d-channels in and out of service */ + switch(mh->msg) { ++ /* the ATT_SERVICE/ATT_SERVICE_ACKNOWLEDGE and NATIONAL_SERVICE/NATIONAL_SERVICE_ACKNOWLEDGE ++ * are mirrors of each other. We only have to check for one type because they are pre-handled ++ * the same way as each other */ ++ case ATT_SERVICE: ++ case ATT_SERVICE_ACKNOWLEDGE: ++ c->channelno = -1; ++ c->slotmap = -1; ++ c->chanflags = 0; ++ c->ds1explicit = 0; ++ c->ds1no = 0; ++ c->cis_call = 0; ++ c->ri = -1; ++ c->changestatus = -1; ++ break; ++ default: ++ pri_error(ctrl, "!! Don't know how to pre-handle maintenance message type '%d'\n", mh->msg); ++ return -1; ++ } ++ return 0; ++} ++ ++static int prepare_to_handle_q931_message(struct pri *ctrl, q931_mh *mh, q931_call *c) ++{ ++ if ((!ctrl) || (!mh) || (!c)) { ++ return -1; ++ } ++ ++ switch(mh->msg) { + case Q931_RESTART: +- if (pri->debug & PRI_DEBUG_Q931_STATE) +- pri_message(pri, "-- Processing Q.931 Restart\n"); ++ if (ctrl->debug & PRI_DEBUG_Q931_STATE) ++ pri_message(ctrl, "-- Processing Q.931 Restart\n"); + /* Reset information */ + c->channelno = -1; + c->slotmap = -1; + c->chanflags = 0; + c->ds1no = 0; ++ c->ds1explicit = 0; ++ c->cis_call = 0; + c->ri = -1; + break; + case Q931_FACILITY: +- c->callername[0] = '\0'; ++ if (q931_is_dummy_call(c)) { ++ q931_party_address_init(&c->called); ++ } + break; + case Q931_SETUP: +- if (pri->debug & PRI_DEBUG_Q931_STATE) +- pri_message(pri, "-- Processing Q.931 Call Setup\n"); ++ if (ctrl->debug & PRI_DEBUG_Q931_STATE) ++ pri_message(ctrl, "-- Processing Q.931 Call Setup\n"); + c->channelno = -1; + c->slotmap = -1; + c->chanflags = 0; +@@ -3334,29 +5548,25 @@ + c->userl2 = -1; + c->userl3 = -1; + c->rateadaption = -1; +- c->calledplan = -1; +- c->callerplan = -1; +- c->callerpres = -1; +- c->callernum[0] = '\0'; +- c->callednum[0] = '\0'; +- c->callername[0] = '\0'; +- c->callerani[0] = '\0'; +- c->callerplanani = -1; +- c->redirectingplan = -1; +- c->redirectingpres = -1; +- c->redirectingreason = -1; +- c->origcalledplan = -1; +- c->origcalledpres = -1; +- c->origredirectingreason = -1; +- c->redirectingnum[0] = '\0'; +- c->origcallednum[0] = '\0'; +- c->redirectingname[0] = '\0'; +- c->origcalledname[0] = '\0'; ++ ++ q931_party_address_init(&c->called); ++ q931_party_id_init(&c->local_id); ++ q931_party_id_init(&c->remote_id); ++ q931_party_redirecting_init(&c->redirecting); ++ ++ /* ++ * Make sure that keypad and overlap digit buffers are empty in ++ * case they are not in the message. ++ */ ++ c->keypad_digits[0] = '\0'; ++ c->overlap_digits[0] = '\0'; ++ + c->useruserprotocoldisc = -1; + c->useruserinfo[0] = '\0'; + c->complete = 0; + c->nonisdn = 0; + c->aoc_units = -1; ++ c->reversecharge = -1; + /* Fall through */ + case Q931_CONNECT: + case Q931_ALERTING: +@@ -3369,8 +5579,7 @@ + c->progressmask = 0; + break; + case Q931_CONNECT_ACKNOWLEDGE: +- if (c->retranstimer) +- pri_schedule_del(pri, c->retranstimer); ++ pri_schedule_del(ctrl, c->retranstimer); + c->retranstimer = 0; + break; + case Q931_RELEASE: +@@ -3379,14 +5588,12 @@ + c->causecode = -1; + c->causeloc = -1; + c->aoc_units = -1; +- if (c->retranstimer) +- pri_schedule_del(pri, c->retranstimer); ++ pri_schedule_del(ctrl, c->retranstimer); + c->retranstimer = 0; + c->useruserinfo[0] = '\0'; + break; + case Q931_RELEASE_COMPLETE: +- if (c->retranstimer) +- pri_schedule_del(pri, c->retranstimer); ++ pri_schedule_del(ctrl, c->retranstimer); + c->retranstimer = 0; + c->useruserinfo[0] = '\0'; + /* Fall through */ +@@ -3394,45 +5601,252 @@ + c->cause = -1; + c->causecode = -1; + c->causeloc = -1; +- c->sugcallstate = -1; ++ c->sugcallstate = Q931_CALL_STATE_NOT_SET; + c->aoc_units = -1; + break; + case Q931_RESTART_ACKNOWLEDGE: + c->channelno = -1; ++ c->ds1no = 0; ++ c->ds1explicit = 0; ++ c->cis_call = 0; + break; + case Q931_INFORMATION: +- c->callednum[0] = '\0'; ++ /* ++ * Make sure that keypad and overlap digit buffers are empty in ++ * case they are not in the message. ++ */ ++ c->keypad_digits[0] = '\0'; ++ c->overlap_digits[0] = '\0'; + break; + case Q931_STATUS_ENQUIRY: + break; + case Q931_SETUP_ACKNOWLEDGE: + break; + case Q931_NOTIFY: ++ q931_party_number_init(&c->redirection_number); + break; +- case Q931_USER_INFORMATION: +- case Q931_SEGMENT: +- case Q931_CONGESTION_CONTROL: + case Q931_HOLD: ++ break; + case Q931_HOLD_ACKNOWLEDGE: ++ break; + case Q931_HOLD_REJECT: ++ c->cause = -1; ++ break; + case Q931_RETRIEVE: ++ c->channelno = 0xFF; ++ c->ds1no = 0; ++ c->ds1explicit = 0; ++ break; + case Q931_RETRIEVE_ACKNOWLEDGE: ++ break; + case Q931_RETRIEVE_REJECT: ++ c->cause = -1; ++ break; ++ case Q931_USER_INFORMATION: ++ case Q931_SEGMENT: ++ case Q931_CONGESTION_CONTROL: + case Q931_RESUME: + case Q931_RESUME_ACKNOWLEDGE: + case Q931_RESUME_REJECT: + case Q931_SUSPEND: + case Q931_SUSPEND_ACKNOWLEDGE: + case Q931_SUSPEND_REJECT: +- pri_error(pri, "!! Not yet handling pre-handle message type %s (%d)\n", msg2str(mh->msg), mh->msg); ++ pri_error(ctrl, "!! Not yet handling pre-handle message type %s (%d)\n", msg2str(mh->msg), mh->msg); + /* Fall through */ + default: +- pri_error(pri, "!! Don't know how to pre-handle message type %s (%d)\n", msg2str(mh->msg), mh->msg); +- q931_status(pri,c, PRI_CAUSE_MESSAGE_TYPE_NONEXIST); ++ pri_error(ctrl, "!! Don't know how to pre-handle message type %s (%d)\n", msg2str(mh->msg), mh->msg); ++ q931_status(ctrl,c, PRI_CAUSE_MESSAGE_TYPE_NONEXIST); + if (c->newcall) +- q931_destroycall(pri,c->cr); ++ pri_destroycall(ctrl, c); + return -1; + } ++ return 0; ++} ++ ++static struct q931_call *q931_get_subcall_winner(struct q931_call *master) ++{ ++ if (master->pri_winner < 0) { ++ return NULL; ++ } else { ++ return master->subcalls[master->pri_winner]; ++ } ++} ++ ++static void initiate_hangup_if_needed(struct pri *pri, q931_call *call, int cause) ++{ ++ if (!call->hangupinitiated) { ++ q931_hangup(pri, call, cause); ++ call->alive = 0; ++ } ++} ++ ++#if 0 ++static int q931_get_subcall_count(struct q931_call *call) ++{ ++ int count = 0; ++ int i; ++ ++ call = call->master_call; ++ for (i = 0; i < Q931_MAX_TEI; i++) { ++ if (call->subcalls[i]) ++ count++; ++ } ++ ++ return count; ++} ++#endif ++ ++static void q931_set_subcall_winner(struct q931_call *subcall) ++{ ++ struct q931_call *realcall = subcall->master_call; ++ int i; ++ ++ /* Set the winner first */ ++ for (i = 0; i < Q931_MAX_TEI; i++) { ++ if (realcall->subcalls[i] && realcall->subcalls[i] == subcall) { ++ realcall->pri_winner = i; ++ } ++ } ++ if (realcall->pri_winner < 0) { ++ pri_error(subcall->pri, "We should always find the winner in the list!\n"); ++ return; ++ } ++ ++ /* Start tear down of calls that were not chosen */ ++ for (i = 0; i < Q931_MAX_TEI; i++) { ++ if (realcall->subcalls[i] && realcall->subcalls[i] != subcall) { ++ initiate_hangup_if_needed(realcall->subcalls[i]->pri, realcall->subcalls[i], ++ PRI_CAUSE_NONSELECTED_USER_CLEARING); ++ } ++ } ++} ++ ++static struct q931_call *q931_get_subcall(struct pri *ctrl, struct q931_call *master_call) ++{ ++ int i; ++ struct q931_call *cur; ++ int firstfree = -1; ++ ++ /* First try to locate our subcall */ ++ for (i = 0; i < Q931_MAX_TEI; i++) { ++ if (master_call->subcalls[i]) { ++ if (master_call->subcalls[i]->pri == ctrl) { ++ return master_call->subcalls[i]; ++ } ++ } else if (firstfree == -1) { ++ firstfree = i; ++ } ++ } ++ if (firstfree < 0) { ++ pri_error(ctrl, "Tried to add more than %d TEIs to call and failed\n", ++ Q931_MAX_TEI); ++ return NULL; ++ } ++ ++ /* Create new subcall. */ ++ cur = malloc(sizeof(*cur)); ++ if (!cur) { ++ pri_error(ctrl, "Unable to allocate call\n"); ++ return NULL; ++ } ++ *cur = *master_call; ++ cur->pri = ctrl; ++ cur->next = NULL; ++ cur->apdus = NULL; ++ cur->bridged_call = NULL; ++ //cur->master_call = master_call; /* We get this assignment for free. */ ++ for (i = 0; i < Q931_MAX_TEI; ++i) { ++ cur->subcalls[i] = NULL; ++ } ++ cur->t303_timer = 0;/* T303 should only be on on the master call */ ++ cur->hold_timer = 0; ++ cur->retranstimer = 0; ++ ++ /* Assume we sent a SETUP and this is the first response to it from this peer. */ ++ cur->ourcallstate = Q931_CALL_STATE_CALL_INITIATED; ++ cur->peercallstate = Q931_CALL_STATE_CALL_PRESENT; ++ ++ master_call->subcalls[firstfree] = cur; ++ ++ if (ctrl->debug & PRI_DEBUG_Q931_STATE) { ++ pri_message(ctrl, "Adding subcall %p for TEI %d to call %p at position %d\n", ++ cur, ctrl->tei, master_call, firstfree); ++ } ++ /* Should only get here if the TEI is not found */ ++ return cur; ++} ++ ++int q931_receive(struct pri *ctrl, q931_h *h, int len) ++{ ++ q931_mh *mh; ++ q931_call *c; ++ q931_ie *ie; ++ unsigned int x; ++ int y; ++ int res; ++ int r; ++ int mandies[MAX_MAND_IES]; ++ int missingmand; ++ int codeset, cur_codeset; ++ int last_ie[8]; ++ int cref; ++ ++ memset(last_ie, 0, sizeof(last_ie)); ++ if (ctrl->debug & PRI_DEBUG_Q931_DUMP) ++ q931_dump(ctrl, h, len, 0); ++#ifdef LIBPRI_COUNTERS ++ ctrl->q931_rxcount++; ++#endif ++ mh = (q931_mh *)(h->contents + h->crlen); ++ if ((h->pd != ctrl->protodisc) && (h->pd != MAINTENANCE_PROTOCOL_DISCRIMINATOR_1) && (h->pd != MAINTENANCE_PROTOCOL_DISCRIMINATOR_2)) { ++ pri_error(ctrl, "Warning: unknown/inappropriate protocol discriminator received (%02x/%d)\n", h->pd, h->pd); ++ return 0; ++ } ++ if (((h->pd == MAINTENANCE_PROTOCOL_DISCRIMINATOR_1) || (h->pd == MAINTENANCE_PROTOCOL_DISCRIMINATOR_2)) && (!ctrl->service_message_support)) { ++ /* Real service message support has not been enabled (and is OFF in libpri by default), ++ * so we have to revert to the 'traditional' KLUDGE of changing byte 4 from a 0xf (SERVICE) ++ * to a 0x7 (SERVICE ACKNOWLEDGE) */ ++ /* This is the weird maintenance stuff. We majorly ++ KLUDGE this by changing byte 4 from a 0xf (SERVICE) ++ to a 0x7 (SERVICE ACKNOWLEDGE) */ ++ h->raw[h->crlen + 2] -= 0x8; ++ q931_xmit(ctrl, h, len, 1, 0); ++ return 0; ++ } ++ ++ cref = q931_cr(h); ++ c = q931_getcall(ctrl, cref); ++ if (!c) { ++ pri_error(ctrl, "Unable to locate call %d\n", cref); ++ return -1; ++ } ++ if (c->master_call->outboundbroadcast && ctrl != PRI_MASTER(ctrl)) { ++ c = q931_get_subcall(ctrl, c->master_call); ++ if (!c) { ++ pri_error(ctrl, "Unable to locate subcall for %d\n", cref); ++ return -1; ++ } ++ } ++ ++ if (ctrl->debug & PRI_DEBUG_Q931_STATE) { ++ pri_message(ctrl, ++ "Received message for call %p on %p TEI/SAPI %d/%d, call->pri is %p TEI/SAPI %d/%d\n", ++ c, ++ ctrl, ctrl->tei, ctrl->sapi, ++ c->pri, c->pri->tei, c->pri->sapi); ++ } ++ ++ /* Preliminary handling */ ++ ctrl->facility.count = 0; ++ c->connected_number_in_message = 0; ++ c->redirecting_number_in_message = 0; ++ if ((h->pd == MAINTENANCE_PROTOCOL_DISCRIMINATOR_1) || (h->pd == MAINTENANCE_PROTOCOL_DISCRIMINATOR_2)) { ++ prepare_to_handle_maintenance_message(ctrl, mh, c); ++ } else { ++ prepare_to_handle_q931_message(ctrl, mh, c); ++ } ++ q931_clr_subcommands(ctrl); ++ + /* Handle IEs */ + memset(mandies, 0, sizeof(mandies)); + missingmand = 0; +@@ -3453,7 +5867,7 @@ + } + r = ielen(ie); + if (r > len) { +- pri_error(pri, "XXX Message longer than it should be?? XXX\n"); ++ pri_error(ctrl, "XXX Message longer than it should be?? XXX\n"); + return -1; + } + /* Special processing for codeset shifts */ +@@ -3461,16 +5875,16 @@ + case Q931_LOCKING_SHIFT: + y = ie->ie & 7; /* Requested codeset */ + /* Locking shifts couldn't go to lower codeset, and couldn't follows non-locking shifts - verify this */ +- if ((cur_codeset != codeset) && (pri->debug & PRI_DEBUG_Q931_ANOMALY)) +- pri_message(pri, "XXX Locking shift immediately follows non-locking shift (from %d through %d to %d) XXX\n", codeset, cur_codeset, y); ++ if ((cur_codeset != codeset) && (ctrl->debug & PRI_DEBUG_Q931_ANOMALY)) ++ pri_message(ctrl, "XXX Locking shift immediately follows non-locking shift (from %d through %d to %d) XXX\n", codeset, cur_codeset, y); + if (y > 0) { +- if ((y < codeset) && (pri->debug & PRI_DEBUG_Q931_ANOMALY)) +- pri_error(pri, "!! Trying to locked downshift codeset from %d to %d !!\n", codeset, y); ++ if ((y < codeset) && (ctrl->debug & PRI_DEBUG_Q931_ANOMALY)) ++ pri_error(ctrl, "!! Trying to locked downshift codeset from %d to %d !!\n", codeset, y); + codeset = cur_codeset = y; + } + else { + /* Locking shift to codeset 0 is forbidden by all specifications */ +- pri_error(pri, "!! Invalid locking shift to codeset 0 !!\n"); ++ pri_error(ctrl, "!! Invalid locking shift to codeset 0 !!\n"); + } + break; + case Q931_NON_LOCKING_SHIFT: +@@ -3480,24 +5894,24 @@ + /* Sanity check for IE code order */ + if (!(ie->ie & 0x80)) { + if (last_ie[cur_codeset] > ie->ie) { +- if ((pri->debug & PRI_DEBUG_Q931_ANOMALY)) +- pri_message(pri, "XXX Out-of-order IE %d at codeset %d (last was %d)\n", ie->ie, cur_codeset, last_ie[cur_codeset]); ++ if ((ctrl->debug & PRI_DEBUG_Q931_ANOMALY)) ++ pri_message(ctrl, "XXX Out-of-order IE %d at codeset %d (last was %d)\n", ie->ie, cur_codeset, last_ie[cur_codeset]); + } + else + last_ie[cur_codeset] = ie->ie; + } + /* Ignore non-locking shifts for TR41459-based signalling */ +- switch (pri->switchtype) { ++ switch (ctrl->switchtype) { + case PRI_SWITCH_LUCENT5E: + case PRI_SWITCH_ATT4ESS: + if (cur_codeset != codeset) { +- if ((pri->debug & PRI_DEBUG_Q931_DUMP)) +- pri_message(pri, "XXX Ignoring IE %d for temporary codeset %d XXX\n", ie->ie, cur_codeset); ++ if ((ctrl->debug & PRI_DEBUG_Q931_DUMP)) ++ pri_message(ctrl, "XXX Ignoring IE %d for temporary codeset %d XXX\n", ie->ie, cur_codeset); + break; + } + /* Fall through */ + default: +- y = q931_handle_ie(cur_codeset, pri, c, mh->msg, ie); ++ y = q931_handle_ie(cur_codeset, ctrl, c, mh->msg, ie); + /* XXX Applicable to codeset 0 only? XXX */ + if (!cur_codeset && !(ie->ie & 0xf0) && (y < 0)) + mandies[MAX_MAND_IES - 1] = Q931_FULL_IE(cur_codeset, ie->ie); +@@ -3512,33 +5926,562 @@ + for (x=0;x<MAX_MAND_IES;x++) { + if (mandies[x]) { + /* check if there is no channel identification when we're configured as network -> that's not an error */ +- if (((pri->localtype != PRI_NETWORK) || (mh->msg != Q931_SETUP) || (mandies[x] != Q931_CHANNEL_IDENT)) && ++ if (((ctrl->localtype != PRI_NETWORK) || (mh->msg != Q931_SETUP) || (mandies[x] != Q931_CHANNEL_IDENT)) && + ((mh->msg != Q931_PROGRESS) || (mandies[x] != Q931_PROGRESS_INDICATOR))) { +- pri_error(pri, "XXX Missing handling for mandatory IE %d (cs%d, %s) XXX\n", Q931_IE_IE(mandies[x]), Q931_IE_CODESET(mandies[x]), ie2str(mandies[x])); ++ pri_error(ctrl, "XXX Missing handling for mandatory IE %d (cs%d, %s) XXX\n", Q931_IE_IE(mandies[x]), Q931_IE_CODESET(mandies[x]), ie2str(mandies[x])); + missingmand++; + } + } + } +- ++ ++ /* Now handle the facility ie's after all the other ie's were processed. */ ++ q931_handle_facilities(ctrl, c, mh->msg); ++ + /* Post handling */ ++ if ((h->pd == MAINTENANCE_PROTOCOL_DISCRIMINATOR_1) || (h->pd == MAINTENANCE_PROTOCOL_DISCRIMINATOR_2)) { ++ res = post_handle_maintenance_message(ctrl, h->pd, mh, c); ++ } else { ++ int allow_event = 1, allow_posthandle = 1; ++ ++ if (c->master_call->outboundbroadcast) { ++ nt_ptmp_handle_q931_message(ctrl, mh, c, &allow_event, &allow_posthandle); ++ } ++ ++ if (allow_posthandle) { ++ res = post_handle_q931_message(ctrl, mh, c, missingmand); ++ ++ if (res == Q931_RES_HAVEEVENT && !allow_event) { ++ res = 0; ++ } ++ } else { ++ res = 0; ++ } ++ } ++ return res; ++} ++ ++static int post_handle_maintenance_message(struct pri *ctrl, int protodisc, struct q931_mh *mh, struct q931_call *c) ++{ ++ /* Do some maintenance stuff */ ++ if (((protodisc == MAINTENANCE_PROTOCOL_DISCRIMINATOR_1) && (mh->msg == ATT_SERVICE)) ++ || ((protodisc == MAINTENANCE_PROTOCOL_DISCRIMINATOR_2) && (mh->msg == NATIONAL_SERVICE))) { ++ if (c->channelno > 0) { ++ ctrl->ev.e = PRI_EVENT_SERVICE; ++ ctrl->ev.service.channel = q931_encode_channel(c); ++ ctrl->ev.service.changestatus = 0x0f & c->changestatus; ++ } else { ++ switch (0x0f & c->changestatus) { ++ case SERVICE_CHANGE_STATUS_INSERVICE: ++ ctrl->ev.e = PRI_EVENT_DCHAN_UP; ++ q921_dchannel_up(ctrl); ++ break; ++ case SERVICE_CHANGE_STATUS_OUTOFSERVICE: ++ ctrl->ev.e = PRI_EVENT_DCHAN_DOWN; ++ q921_dchannel_down(ctrl); ++ break; ++ default: ++ pri_error(ctrl, "!! Don't know how to handle span service change status '%d'\n", (0x0f & c->changestatus)); ++ return -1; ++ } ++ } ++ maintenance_service_ack(ctrl, c); ++ return Q931_RES_HAVEEVENT; ++ } ++ if (((protodisc == MAINTENANCE_PROTOCOL_DISCRIMINATOR_1) && (mh->msg == ATT_SERVICE_ACKNOWLEDGE)) ++ || ((protodisc == MAINTENANCE_PROTOCOL_DISCRIMINATOR_2) && (mh->msg == NATIONAL_SERVICE_ACKNOWLEDGE))) { ++ if (c->channelno > 0) { ++ ctrl->ev.e = PRI_EVENT_SERVICE_ACK; ++ ctrl->ev.service_ack.channel = q931_encode_channel(c); ++ ctrl->ev.service_ack.changestatus = 0x0f & c->changestatus; ++ } else { ++ switch (0x0f & c->changestatus) { ++ case SERVICE_CHANGE_STATUS_INSERVICE: ++ ctrl->ev.e = PRI_EVENT_DCHAN_UP; ++ q921_dchannel_up(ctrl); ++ break; ++ case SERVICE_CHANGE_STATUS_OUTOFSERVICE: ++ ctrl->ev.e = PRI_EVENT_DCHAN_DOWN; ++ q921_dchannel_down(ctrl); ++ break; ++ default: ++ pri_error(ctrl, "!! Don't know how to handle span service change status '%d'\n", (0x0f & c->changestatus)); ++ return -1; ++ } ++ } ++ return Q931_RES_HAVEEVENT; ++ } ++ ++ pri_error(ctrl, "!! Don't know how to post-handle maintenance message type %d\n", mh->msg); ++ return -1; ++} ++ ++/*! ++ * \internal ++ * \brief Rank the given Q.931 call state for call etablishment. ++ * ++ * \param state Q.931 call state to rank for competing PTMP NT calls. ++ * ++ * \return Call establishment state ranking. ++ */ ++static enum Q931_RANKED_CALL_STATE q931_rank_state(enum Q931_CALL_STATE state) ++{ ++ enum Q931_RANKED_CALL_STATE rank; ++ ++ switch (state) { ++ case Q931_CALL_STATE_CALL_INITIATED: ++ case Q931_CALL_STATE_CALL_PRESENT: ++ rank = Q931_RANKED_CALL_STATE_PRESENT; ++ break; ++ case Q931_CALL_STATE_OVERLAP_SENDING: ++ case Q931_CALL_STATE_OVERLAP_RECEIVING: ++ rank = Q931_RANKED_CALL_STATE_OVERLAP; ++ break; ++ case Q931_CALL_STATE_OUTGOING_CALL_PROCEEDING: ++ case Q931_CALL_STATE_INCOMING_CALL_PROCEEDING: ++ rank = Q931_RANKED_CALL_STATE_PROCEEDING; ++ break; ++ case Q931_CALL_STATE_CALL_DELIVERED: ++ case Q931_CALL_STATE_CALL_RECEIVED: ++ case Q931_CALL_STATE_CONNECT_REQUEST: ++ rank = Q931_RANKED_CALL_STATE_ALERTING; ++ break; ++ case Q931_CALL_STATE_ACTIVE: ++ case Q931_CALL_STATE_CALL_INDEPENDENT_SERVICE: ++ rank = Q931_RANKED_CALL_STATE_CONNECT; ++ break; ++ default: ++ rank = Q931_RANKED_CALL_STATE_OTHER; ++ break; ++ } ++ ++ return rank; ++} ++ ++/*! ++ * \brief Determine if the master will pass an event to the upper layer. ++ * ++ * \param ctrl D channel controller. ++ * \param subcall Q.931 call leg. ++ * \param msg_type Current message type being processed. ++ * ++ * \note This function must parallel nt_ptmp_handle_q931_message(). ++ * ++ * \retval TRUE if the master will pass an event to the upper layer. ++ * \retval FALSE if the event will be blocked. ++ */ ++int q931_master_pass_event(struct pri *ctrl, struct q931_call *subcall, int msg_type) ++{ ++ struct q931_call *winner; ++ struct q931_call *master; ++ enum Q931_RANKED_CALL_STATE master_rank; ++ enum Q931_RANKED_CALL_STATE subcall_rank; ++ int will_pass; /*!< TRUE if the master will pass an event to the upper layer. */ ++ ++ master = subcall->master_call; ++ if (subcall == master) { ++ /* We are the master call so of course the master will pass an event. */ ++ return 1; ++ } ++ ++ winner = q931_get_subcall_winner(master); ++ if (winner && subcall == winner) { ++ /* We are the winner so of course the master will pass an event. */ ++ return 1; ++ } ++ ++ master_rank = q931_rank_state(master->ourcallstate); ++ will_pass = 0; ++ switch (msg_type) { ++ case Q931_SETUP_ACKNOWLEDGE: ++#if 0 /* Overlap dialing in PTMP NT mode not supported at the present time. */ ++ if (master_rank < Q931_RANKED_CALL_STATE_OVERLAP) { ++ will_pass = 1; ++ } ++#endif /* Overlap dialing in PTMP NT mode not supported at the present time. */ ++ break; ++ case Q931_CALL_PROCEEDING: ++ if (master_rank < Q931_RANKED_CALL_STATE_PROCEEDING) { ++ will_pass = 1; ++ } ++ break; ++ case Q931_PROGRESS: ++ /* ++ * We will just ignore this message since there could be multiple devices ++ * competing for this call. Who has access to the B channel at this time ++ * to give in-band signals anyway? ++ */ ++ break; ++ case Q931_ALERTING: ++ if (master_rank < Q931_RANKED_CALL_STATE_ALERTING) { ++ will_pass = 1; ++ } ++ break; ++ case Q931_CONNECT: ++ if (master_rank < Q931_RANKED_CALL_STATE_CONNECT) { ++ /* We are expected to be the winner for the next message. */ ++ will_pass = 1; ++ } ++ break; ++ case Q931_DISCONNECT: ++ case Q931_RELEASE: ++ case Q931_RELEASE_COMPLETE: ++ /* Only deal with the winner. */ ++ break; ++ case Q931_FACILITY: ++ case Q931_NOTIFY: ++ if (!winner) { ++ /* The overlap rank does not count here. */ ++ if (master_rank == Q931_RANKED_CALL_STATE_OVERLAP) { ++ master_rank = Q931_RANKED_CALL_STATE_PRESENT; ++ } ++ subcall_rank = q931_rank_state(subcall->ourcallstate); ++ if (subcall_rank == Q931_RANKED_CALL_STATE_OVERLAP) { ++ subcall_rank = Q931_RANKED_CALL_STATE_PRESENT; ++ } ++ if (master_rank == subcall_rank) { ++ /* ++ * No winner yet but the subcall is as advanced as the master. ++ * Allow the supplementary service event to pass. ++ */ ++ will_pass = 1; ++ } ++ } ++ break; ++ default: ++ /* Only deal with the winner. */ ++ break; ++ } ++ ++ return will_pass; ++} ++ ++/*! ++ * \internal ++ * \brief Handle outboundbroadcast incoming messages for the master_call's state. ++ * ++ * \param ctrl D channel controller. ++ * \param mh Q.931 message type header. ++ * \param subcall Q.931 call leg. ++ * \param allow_event Where to set the allow event to upper layer flag. ++ * \param allow_posthandle Where to set the allow post handle event flag. ++ * ++ * \details ++ * This is where we interact the subcalls state with the master_call's state. ++ * ++ * \note This function must parallel q931_master_pass_event(). ++ * ++ * \return Nothing ++ */ ++static void nt_ptmp_handle_q931_message(struct pri *ctrl, struct q931_mh *mh, struct q931_call *subcall, int *allow_event, int *allow_posthandle) ++{ ++ struct q931_call *master = subcall->master_call; ++ struct q931_call *winner = q931_get_subcall_winner(master); ++ enum Q931_RANKED_CALL_STATE master_rank; ++ enum Q931_RANKED_CALL_STATE subcall_rank; ++ enum Q931_CALL_STATE newstate; ++ ++ /* For broadcast calls, we default to not allowing events to keep events received to a minimum ++ * and to allow post processing, since that is where hangup and subcall state handling and other processing is done */ ++ *allow_event = 0; ++ *allow_posthandle = 1; ++ ++ master_rank = q931_rank_state(master->ourcallstate); ++ ++ switch (mh->msg) { ++ case Q931_SETUP_ACKNOWLEDGE: ++#if 0 /* Overlap dialing in PTMP NT mode not supported at the present time. */ ++ if (master_rank < Q931_RANKED_CALL_STATE_OVERLAP) { ++ *allow_event = 1; ++ UPDATE_OURCALLSTATE(ctrl, master, Q931_CALL_STATE_OVERLAP_SENDING); ++ } ++#endif /* Overlap dialing in PTMP NT mode not supported at the present time. */ ++ break; ++ case Q931_CALL_PROCEEDING: ++ if (master_rank < Q931_RANKED_CALL_STATE_PROCEEDING) { ++ *allow_event = 1; ++ UPDATE_OURCALLSTATE(ctrl, master, Q931_CALL_STATE_OUTGOING_CALL_PROCEEDING); ++ } ++ break; ++ case Q931_PROGRESS: ++ /* ++ * We will just ignore this message since there could be multiple devices ++ * competing for this call. Who has access to the B channel at this time ++ * to give in-band signals anyway? ++ */ ++ break; ++ case Q931_ALERTING: ++ if (master_rank < Q931_RANKED_CALL_STATE_ALERTING) { ++ *allow_event = 1; ++ UPDATE_OURCALLSTATE(ctrl, master, Q931_CALL_STATE_CALL_DELIVERED); ++ } ++ break; ++ case Q931_CONNECT: ++ if (master_rank < Q931_RANKED_CALL_STATE_CONNECT) { ++ UPDATE_OURCALLSTATE(ctrl, master, Q931_CALL_STATE_ACTIVE); ++ q931_set_subcall_winner(subcall); ++ *allow_event = 1; ++ } else { ++ /* Call clearing of non selected calls occurs in ++ * q931_set_subcall_winner() - All we need to do is make sure ++ * that this connect is not acknowledged */ ++ *allow_posthandle = 0; ++ } ++ break; ++ case Q931_DISCONNECT: ++ newstate = Q931_CALL_STATE_DISCONNECT_INDICATION; ++ goto process_hangup; ++ case Q931_RELEASE: ++ case Q931_RELEASE_COMPLETE: ++ newstate = Q931_CALL_STATE_NULL; ++process_hangup: ++ if (!winner) { ++ /* If there's not a winner, we just take the cause and pass it up to the ++ * master_call */ ++ master->cause = subcall->cause; ++ } else { ++ /* There *is* a winner */ ++ if (subcall == winner) { ++ /* .. and we're it: */ ++ *allow_event = 1; ++ UPDATE_OURCALLSTATE(ctrl, master, newstate); ++ } ++ } ++ break; ++ case Q931_FACILITY: ++ case Q931_NOTIFY: ++ if (winner) { ++ if (subcall == winner) { ++ /* Only deal with the winner. */ ++ *allow_event = 1; ++ } ++ } else { ++ /* The overlap rank does not count here. */ ++ if (master_rank == Q931_RANKED_CALL_STATE_OVERLAP) { ++ master_rank = Q931_RANKED_CALL_STATE_PRESENT; ++ } ++ subcall_rank = q931_rank_state(subcall->ourcallstate); ++ if (subcall_rank == Q931_RANKED_CALL_STATE_OVERLAP) { ++ subcall_rank = Q931_RANKED_CALL_STATE_PRESENT; ++ } ++ if (master_rank == subcall_rank) { ++ /* ++ * No winner yet but the subcall is as advanced as the master. ++ * Allow the supplementary service event to pass. ++ */ ++ *allow_event = 1; ++ } ++ } ++ break; ++ default: ++ if (winner && subcall == winner) { ++ /* Only deal with the winner. */ ++ *allow_event = 1; ++ } ++ break; ++ } ++} ++ ++/*! ++ * \internal ++ * \brief Fill in the FACILITY event fields. ++ * ++ * \param ctrl D channel controller. ++ * \param call Q.931 call leg. ++ * ++ * \return Nothing ++ */ ++static void q931_fill_facility_event(struct pri *ctrl, struct q931_call *call) ++{ ++ ctrl->ev.e = PRI_EVENT_FACILITY; ++ ctrl->ev.facility.subcmds = &ctrl->subcmds; ++ ctrl->ev.facility.channel = q931_encode_channel(call); ++ ctrl->ev.facility.cref = call->cr; ++ ctrl->ev.facility.call = call->master_call; ++ ctrl->ev.facility.subcall = call; ++ ++ /* Need to do this for backward compatibility with struct pri_event_facname */ ++ libpri_copy_string(ctrl->ev.facility.callingname, call->remote_id.name.str, ++ sizeof(ctrl->ev.facility.callingname)); ++ libpri_copy_string(ctrl->ev.facility.callingnum, call->remote_id.number.str, ++ sizeof(ctrl->ev.facility.callingnum)); ++ ctrl->ev.facility.callingpres = q931_party_id_presentation(&call->remote_id); ++ ctrl->ev.facility.callingplan = call->remote_id.number.plan; ++} ++ ++/*! ++ * \internal ++ * \brief APDU wait for response message timeout. ++ * ++ * \param data Callback data pointer. ++ * ++ * \return Nothing ++ */ ++static void q931_apdu_timeout(void *data) ++{ ++ struct apdu_event *apdu; ++ struct pri *ctrl; ++ struct q931_call *call; ++ ++ apdu = data; ++ call = apdu->call; ++ ctrl = call->pri; ++ ++ q931_clr_subcommands(ctrl); ++ apdu->response.callback(APDU_CALLBACK_REASON_TIMEOUT, ctrl, call, apdu, NULL); ++ if (ctrl->subcmds.counter_subcmd) { ++ q931_fill_facility_event(ctrl, call); ++ ctrl->schedev = 1; ++ } ++ ++ pri_call_apdu_delete(call, apdu); ++} ++ ++/*! ++ * \internal ++ * \brief Find the active call given the held call. ++ * ++ * \param ctrl D channel controller. ++ * \param held_call Held call to help locate a compatible active call. ++ * ++ * \retval master-active-call on success. ++ * \retval NULL on error. ++ */ ++static struct q931_call *q931_find_held_active_call(struct pri *ctrl, struct q931_call *held_call) ++{ ++ struct pri *master; ++ struct q931_call *cur; ++ struct q931_call *winner; ++ struct q931_call *match; ++ ++ match = NULL; ++ master = PRI_MASTER(ctrl); ++ for (cur = *master->callpool; cur; cur = cur->next) { ++ if (cur->hold_state == Q931_HOLD_STATE_IDLE) { ++ /* Found an active call. */ ++ winner = q931_find_winning_call(cur); ++ if (!winner || (BRI_NT_PTMP(ctrl) && winner->pri != held_call->pri)) { ++ /* There is no winner or the active call does not go to the same TEI. */ ++ continue; ++ } ++ switch (winner->ourcallstate) { ++ case Q931_CALL_STATE_OUTGOING_CALL_PROCEEDING: ++ case Q931_CALL_STATE_CALL_DELIVERED: ++ case Q931_CALL_STATE_CALL_RECEIVED: ++ case Q931_CALL_STATE_CONNECT_REQUEST: ++ case Q931_CALL_STATE_INCOMING_CALL_PROCEEDING: ++ case Q931_CALL_STATE_ACTIVE: ++ break; ++ default: ++ /* Active call not in a good state to transfer. */ ++ continue; ++ } ++ if (q931_party_number_cmp(&winner->remote_id.number, ++ &held_call->remote_id.number)) { ++ /* The remote party number does not match. This is a weak match. */ ++ match = cur; ++ continue; ++ } ++ /* Found an exact match. */ ++ match = cur; ++ break; ++ } ++ } ++ ++ return match; ++} ++ ++/*! ++ * \internal ++ * \brief Find the held call given the active call. ++ * ++ * \param ctrl D channel controller. ++ * \param active_call Active call to help locate a compatible held call. ++ * ++ * \retval master-held-call on success. ++ * \retval NULL on error. ++ */ ++static struct q931_call *q931_find_held_call(struct pri *ctrl, struct q931_call *active_call) ++{ ++ struct pri *master; ++ struct q931_call *cur; ++ struct q931_call *winner; ++ struct q931_call *match; ++ ++ match = NULL; ++ master = PRI_MASTER(ctrl); ++ for (cur = *master->callpool; cur; cur = cur->next) { ++ if (cur->hold_state == Q931_HOLD_STATE_CALL_HELD) { ++ /* Found a held call. */ ++ winner = q931_find_winning_call(cur); ++ if (!winner || (BRI_NT_PTMP(ctrl) && winner->pri != active_call->pri)) { ++ /* There is no winner or the held call does not go to the same TEI. */ ++ continue; ++ } ++ switch (winner->ourcallstate) { ++ case Q931_CALL_STATE_OUTGOING_CALL_PROCEEDING: ++ case Q931_CALL_STATE_CALL_DELIVERED: ++ case Q931_CALL_STATE_CALL_RECEIVED: ++ case Q931_CALL_STATE_CONNECT_REQUEST: ++ case Q931_CALL_STATE_INCOMING_CALL_PROCEEDING: ++ case Q931_CALL_STATE_ACTIVE: ++ break; ++ default: ++ /* Held call not in a good state to transfer. */ ++ continue; ++ } ++ if (q931_party_number_cmp(&winner->remote_id.number, ++ &active_call->remote_id.number)) { ++ /* The remote party number does not match. This is a weak match. */ ++ match = cur; ++ continue; ++ } ++ /* Found an exact match. */ ++ match = cur; ++ break; ++ } ++ } ++ ++ return match; ++} ++ ++/*! ++ * \internal ++ * \brief Process the decoded information in the Q.931 message. ++ * ++ * \param ctrl D channel controller. ++ * \param mh Q.931 message header. ++ * \param c Q.931 call leg. ++ * \param missingmand Number of missing mandatory ie's. ++ * ++ * \retval 0 if no error or event. ++ * \retval Q931_RES_HAVEEVENT if have an event. ++ * \retval -1 on error. ++ */ ++static int post_handle_q931_message(struct pri *ctrl, struct q931_mh *mh, struct q931_call *c, int missingmand) ++{ ++ int res; ++ struct apdu_event *cur = NULL; ++ struct pri_subcommand *subcmd; ++ struct q931_call *master_call; ++ + switch(mh->msg) { + case Q931_RESTART: + if (missingmand) { +- q931_status(pri, c, PRI_CAUSE_MANDATORY_IE_MISSING); +- q931_destroycall(pri, c->cr); ++ q931_status(ctrl, c, PRI_CAUSE_MANDATORY_IE_MISSING); ++ pri_destroycall(ctrl, c); + break; + } +- UPDATE_OURCALLSTATE(pri, c, Q931_CALL_STATE_RESTART); ++ UPDATE_OURCALLSTATE(ctrl, c, Q931_CALL_STATE_RESTART); + c->peercallstate = Q931_CALL_STATE_RESTART_REQUEST; + /* Send back the Restart Acknowledge */ +- restart_ack(pri, c); ++ restart_ack(ctrl, c); + /* Notify user of restart event */ +- pri->ev.e = PRI_EVENT_RESTART; +- pri->ev.restart.channel = c->channelno | (c->ds1no << 8) | (c->ds1explicit << 16); ++ ctrl->ev.e = PRI_EVENT_RESTART; ++ ctrl->ev.restart.channel = q931_encode_channel(c); + return Q931_RES_HAVEEVENT; + case Q931_SETUP: + if (missingmand) { +- q931_release_complete(pri, c, PRI_CAUSE_MANDATORY_IE_MISSING); ++ q931_release_complete(ctrl, c, PRI_CAUSE_MANDATORY_IE_MISSING); + break; + } + /* Must be new call */ +@@ -3548,185 +6491,279 @@ + if (c->progressmask & PRI_PROG_CALLER_NOT_ISDN) + c->nonisdn = 1; + c->newcall = 0; +- UPDATE_OURCALLSTATE(pri, c, Q931_CALL_STATE_CALL_PRESENT); ++ UPDATE_OURCALLSTATE(ctrl, c, Q931_CALL_STATE_CALL_PRESENT); + c->peercallstate = Q931_CALL_STATE_CALL_INITIATED; + /* it's not yet a call since higher level can respond with RELEASE or RELEASE_COMPLETE */ + c->alive = 0; + if (c->transmoderate != TRANS_MODE_64_CIRCUIT) { +- q931_release_complete(pri, c, PRI_CAUSE_BEARERCAPABILITY_NOTIMPL); ++ q931_release_complete(ctrl, c, PRI_CAUSE_BEARERCAPABILITY_NOTIMPL); + break; + } +- pri->ev.e = PRI_EVENT_RING; +- pri->ev.ring.channel = c->channelno | (c->ds1no << 8) | (c->ds1explicit << 16); +- pri->ev.ring.callingpres = c->callerpres; +- pri->ev.ring.callingplan = c->callerplan; +- pri->ev.ring.callingplanani = c->callerplanani; +- pri->ev.ring.callingplanrdnis = c->redirectingplan; +- pri->ev.ring.callingplanorigcalled = c->origcalledplan; +- pri->ev.ring.ani2 = c->ani2; +- libpri_copy_string(pri->ev.ring.callingani, c->callerani, sizeof(pri->ev.ring.callingani)); +- libpri_copy_string(pri->ev.ring.callingnum, c->callernum, sizeof(pri->ev.ring.callingnum)); +- libpri_copy_string(pri->ev.ring.callingname, c->callername, sizeof(pri->ev.ring.callingname)); +- pri->ev.ring.calledplan = c->calledplan; +- libpri_copy_string(pri->ev.ring.callingsubaddr, c->callingsubaddr, sizeof(pri->ev.ring.callingsubaddr)); +- libpri_copy_string(pri->ev.ring.callednum, c->callednum, sizeof(pri->ev.ring.callednum)); +- libpri_copy_string(pri->ev.ring.origcalledname, c->origcalledname, sizeof(pri->ev.ring.origcalledname)); +- libpri_copy_string(pri->ev.ring.origcallednum, c->origcallednum, sizeof(pri->ev.ring.origcallednum)); +- libpri_copy_string(pri->ev.ring.redirectingnum, c->redirectingnum, sizeof(pri->ev.ring.redirectingnum)); +- libpri_copy_string(pri->ev.ring.redirectingname, c->redirectingname, sizeof(pri->ev.ring.redirectingname)); +- libpri_copy_string(pri->ev.ring.useruserinfo, c->useruserinfo, sizeof(pri->ev.ring.useruserinfo)); ++ ++ if (c->redirecting.from.number.valid && !c->redirecting.count) { ++ /* ++ * This is most likely because the redirecting number came in ++ * with the redirecting ie only and not a DivertingLegInformation2. ++ */ ++ c->redirecting.count = 1; ++ } ++ if (c->redirecting.state == Q931_REDIRECTING_STATE_PENDING_TX_DIV_LEG_3) { ++ /* ++ * Valid for Q.SIG and ETSI PRI/BRI-PTP modes: ++ * Setup the redirecting.to informtion so we can identify ++ * if the user wants to manually supply the COLR for this ++ * redirected to number if further redirects could happen. ++ * ++ * All the user needs to do is set the REDIRECTING(to-pres) ++ * to the COLR and REDIRECTING(to-num) = complete-dialed-number ++ * (i.e. CALLERID(dnid)) to be safe after determining that the ++ * incoming call was redirected by checking if the ++ * REDIRECTING(count) is nonzero. ++ */ ++ c->redirecting.to.number = c->called.number; ++ c->redirecting.to.number.presentation = ++ PRI_PRES_RESTRICTED | PRI_PRES_USER_NUMBER_UNSCREENED; ++ } ++ ++ ctrl->ev.e = PRI_EVENT_RING; ++ ctrl->ev.ring.subcmds = &ctrl->subcmds; ++ ctrl->ev.ring.channel = q931_encode_channel(c); ++ ++ /* Calling party information */ ++ ctrl->ev.ring.callingpres = q931_party_id_presentation(&c->remote_id); ++ ctrl->ev.ring.callingplan = c->remote_id.number.plan; ++ if (c->remote_id.number.valid ++ && (c->remote_id.number.presentation == PRES_ALLOWED_NETWORK_NUMBER ++ || c->remote_id.number.presentation == PRES_PROHIB_NETWORK_NUMBER)) { ++ ctrl->ev.ring.callingplanani = c->remote_id.number.plan; ++ libpri_copy_string(ctrl->ev.ring.callingani, c->remote_id.number.str, sizeof(ctrl->ev.ring.callingani)); ++ } else { ++ ctrl->ev.ring.callingplanani = -1; ++ ctrl->ev.ring.callingani[0] = '\0'; ++ } ++ libpri_copy_string(ctrl->ev.ring.callingnum, c->remote_id.number.str, sizeof(ctrl->ev.ring.callingnum)); ++ libpri_copy_string(ctrl->ev.ring.callingname, c->remote_id.name.str, sizeof(ctrl->ev.ring.callingname)); ++ q931_party_id_copy_to_pri(&ctrl->ev.ring.calling, &c->remote_id); ++ /* for backwards compatibility, still need ctrl->ev.ring.callingsubaddr */ ++ if (!c->remote_id.subaddress.type) { /* NSAP: Type = 0 */ ++ libpri_copy_string(ctrl->ev.ring.callingsubaddr, (char *) c->remote_id.subaddress.data, sizeof(ctrl->ev.ring.callingsubaddr)); ++ } else { ++ ctrl->ev.ring.callingsubaddr[0] = '\0'; ++ } ++ ++ ctrl->ev.ring.ani2 = c->ani2; ++ ++ /* Called party information */ ++ ctrl->ev.ring.calledplan = c->called.number.plan; ++ libpri_copy_string(ctrl->ev.ring.callednum, c->called.number.str, sizeof(ctrl->ev.ring.callednum)); ++ q931_party_subaddress_copy_to_pri(&ctrl->ev.ring.called_subaddress, &c->called.subaddress); ++ ++ /* Original called party information (For backward compatibility) */ ++ libpri_copy_string(ctrl->ev.ring.origcalledname, c->redirecting.orig_called.name.str, sizeof(ctrl->ev.ring.origcalledname)); ++ libpri_copy_string(ctrl->ev.ring.origcallednum, c->redirecting.orig_called.number.str, sizeof(ctrl->ev.ring.origcallednum)); ++ ctrl->ev.ring.callingplanorigcalled = c->redirecting.orig_called.number.plan; ++ if (c->redirecting.orig_called.number.valid ++ || c->redirecting.orig_called.name.valid) { ++ ctrl->ev.ring.origredirectingreason = c->redirecting.orig_reason; ++ } else { ++ ctrl->ev.ring.origredirectingreason = -1; ++ } ++ ++ /* Redirecting from party information (For backward compatibility) */ ++ ctrl->ev.ring.callingplanrdnis = c->redirecting.from.number.plan; ++ libpri_copy_string(ctrl->ev.ring.redirectingnum, c->redirecting.from.number.str, sizeof(ctrl->ev.ring.redirectingnum)); ++ libpri_copy_string(ctrl->ev.ring.redirectingname, c->redirecting.from.name.str, sizeof(ctrl->ev.ring.redirectingname)); ++ ++ ctrl->ev.ring.redirectingreason = c->redirecting.reason; ++ ++ libpri_copy_string(ctrl->ev.ring.useruserinfo, c->useruserinfo, sizeof(ctrl->ev.ring.useruserinfo)); + c->useruserinfo[0] = '\0'; +- pri->ev.ring.redirectingreason = c->redirectingreason; +- pri->ev.ring.origredirectingreason = c->origredirectingreason; +- pri->ev.ring.flexible = ! (c->chanflags & FLAG_EXCLUSIVE); +- pri->ev.ring.cref = c->cr; +- pri->ev.ring.call = c; +- pri->ev.ring.layer1 = c->userl1; +- pri->ev.ring.complete = c->complete; +- pri->ev.ring.ctype = c->transcapability; +- pri->ev.ring.redirectingreason = c->redirectingreason; +- pri->ev.ring.progress = c->progress; +- pri->ev.ring.progressmask = c->progressmask; ++ ++ libpri_copy_string(ctrl->ev.ring.keypad_digits, c->keypad_digits, ++ sizeof(ctrl->ev.ring.keypad_digits)); ++ ++ ctrl->ev.ring.flexible = ! (c->chanflags & FLAG_EXCLUSIVE); ++ ctrl->ev.ring.cref = c->cr; ++ ctrl->ev.ring.call = c->master_call; ++ ctrl->ev.ring.layer1 = c->userl1; ++ ctrl->ev.ring.complete = c->complete; ++ ctrl->ev.ring.ctype = c->transcapability; ++ ctrl->ev.ring.progress = c->progress; ++ ctrl->ev.ring.progressmask = c->progressmask; ++ ctrl->ev.ring.reversecharge = c->reversecharge; ++ ++ if (c->redirecting.count) { ++ subcmd = q931_alloc_subcommand(ctrl); ++ if (subcmd) { ++ /* Setup redirecting subcommand */ ++ subcmd->cmd = PRI_SUBCMD_REDIRECTING; ++ q931_party_redirecting_copy_to_pri(&subcmd->u.redirecting, ++ &c->redirecting); ++ } ++ } ++ + return Q931_RES_HAVEEVENT; + case Q931_ALERTING: ++ stop_t303(c); + if (c->newcall) { +- q931_release_complete(pri,c,PRI_CAUSE_INVALID_CALL_REFERENCE); ++ q931_release_complete(ctrl,c,PRI_CAUSE_INVALID_CALL_REFERENCE); + break; + } +- UPDATE_OURCALLSTATE(pri, c, Q931_CALL_STATE_CALL_DELIVERED); ++ UPDATE_OURCALLSTATE(ctrl, c, Q931_CALL_STATE_CALL_DELIVERED); + c->peercallstate = Q931_CALL_STATE_CALL_RECEIVED; +- pri->ev.e = PRI_EVENT_RINGING; +- pri->ev.ringing.channel = c->channelno | (c->ds1no << 8) | (c->ds1explicit << 16); +- pri->ev.ringing.cref = c->cr; +- pri->ev.ringing.call = c; +- pri->ev.ringing.progress = c->progress; +- pri->ev.ringing.progressmask = c->progressmask; +- libpri_copy_string(pri->ev.ringing.useruserinfo, c->useruserinfo, sizeof(pri->ev.ringing.useruserinfo)); ++ ctrl->ev.e = PRI_EVENT_RINGING; ++ ctrl->ev.ringing.subcmds = &ctrl->subcmds; ++ ctrl->ev.ringing.channel = q931_encode_channel(c); ++ ctrl->ev.ringing.cref = c->cr; ++ ctrl->ev.ringing.call = c->master_call; ++ ctrl->ev.ringing.progress = c->progress; ++ ctrl->ev.ringing.progressmask = c->progressmask; ++ ++ libpri_copy_string(ctrl->ev.ringing.useruserinfo, c->useruserinfo, sizeof(ctrl->ev.ringing.useruserinfo)); + c->useruserinfo[0] = '\0'; + +- cur = c->apdus; +- while (cur) { ++ for (cur = c->apdus; cur; cur = cur->next) { + if (!cur->sent && cur->message == Q931_FACILITY) { +- q931_facility(pri, c); ++ q931_facility(ctrl, c); + break; + } +- cur = cur->next; + } + + return Q931_RES_HAVEEVENT; + case Q931_CONNECT: ++ stop_t303(c); + if (c->newcall) { +- q931_release_complete(pri,c,PRI_CAUSE_INVALID_CALL_REFERENCE); ++ q931_release_complete(ctrl,c,PRI_CAUSE_INVALID_CALL_REFERENCE); + break; + } + if (c->ourcallstate == Q931_CALL_STATE_ACTIVE) { +- q931_status(pri, c, PRI_CAUSE_WRONG_MESSAGE); ++ q931_status(ctrl, c, PRI_CAUSE_WRONG_MESSAGE); + break; + } +- UPDATE_OURCALLSTATE(pri, c, Q931_CALL_STATE_ACTIVE); ++ UPDATE_OURCALLSTATE(ctrl, c, Q931_CALL_STATE_ACTIVE); + c->peercallstate = Q931_CALL_STATE_CONNECT_REQUEST; +- pri->ev.e = PRI_EVENT_ANSWER; +- pri->ev.answer.channel = c->channelno | (c->ds1no << 8) | (c->ds1explicit << 16); +- pri->ev.answer.cref = c->cr; +- pri->ev.answer.call = c; +- pri->ev.answer.progress = c->progress; +- pri->ev.answer.progressmask = c->progressmask; +- libpri_copy_string(pri->ev.answer.useruserinfo, c->useruserinfo, sizeof(pri->ev.answer.useruserinfo)); ++ ++ ctrl->ev.e = PRI_EVENT_ANSWER; ++ ctrl->ev.answer.subcmds = &ctrl->subcmds; ++ ctrl->ev.answer.channel = q931_encode_channel(c); ++ ctrl->ev.answer.cref = c->cr; ++ ctrl->ev.answer.call = c->master_call; ++ ctrl->ev.answer.progress = c->progress; ++ ctrl->ev.answer.progressmask = c->progressmask; ++ libpri_copy_string(ctrl->ev.answer.useruserinfo, c->useruserinfo, sizeof(ctrl->ev.answer.useruserinfo)); + c->useruserinfo[0] = '\0'; +- q931_connect_acknowledge(pri, c); +- if (c->justsignalling) { /* Make sure WE release when we initiatie a signalling only connection */ +- q931_release(pri, c, PRI_CAUSE_NORMAL_CLEARING); ++ ++ q931_connect_acknowledge(ctrl, c); ++ ++ if (c->cis_auto_disconnect && c->cis_call) { ++ /* Make sure WE release when we initiate a signalling only connection */ ++ q931_hangup(ctrl, c, PRI_CAUSE_NORMAL_CLEARING); + break; +- } else ++ } else { ++ c->incoming_ct_state = INCOMING_CT_STATE_IDLE; ++ ++ /* Setup connected line subcommand */ ++ subcmd = q931_alloc_subcommand(ctrl); ++ if (subcmd) { ++ subcmd->cmd = PRI_SUBCMD_CONNECTED_LINE; ++ q931_party_id_copy_to_pri(&subcmd->u.connected_line.id, &c->remote_id); ++ } ++ + return Q931_RES_HAVEEVENT; ++ } + case Q931_FACILITY: + if (c->newcall) { +- q931_release_complete(pri,c,PRI_CAUSE_INVALID_CALL_REFERENCE); ++ q931_release_complete(ctrl,c,PRI_CAUSE_INVALID_CALL_REFERENCE); + break; + } +- pri->ev.e = PRI_EVENT_FACNAME; +- libpri_copy_string(pri->ev.facname.callingname, c->callername, sizeof(pri->ev.facname.callingname)); +- libpri_copy_string(pri->ev.facname.callingnum, c->callernum, sizeof(pri->ev.facname.callingnum)); +- pri->ev.facname.channel = c->channelno | (c->ds1no << 8) | (c->ds1explicit << 16); +- pri->ev.facname.callingpres = c->callerpres; +- pri->ev.facname.callingplan = c->callerplan; +- pri->ev.facname.cref = c->cr; +- pri->ev.facname.call = c; +-#if 0 +- pri_message(pri, "Sending facility event (%s/%s)\n", pri->ev.facname.callingname, pri->ev.facname.callingnum); +-#endif +- return Q931_RES_HAVEEVENT; ++ switch (c->incoming_ct_state) { ++ case INCOMING_CT_STATE_POST_CONNECTED_LINE: ++ c->incoming_ct_state = INCOMING_CT_STATE_IDLE; ++ subcmd = q931_alloc_subcommand(ctrl); ++ if (subcmd) { ++ subcmd->cmd = PRI_SUBCMD_CONNECTED_LINE; ++ q931_party_id_copy_to_pri(&subcmd->u.connected_line.id, &c->remote_id); ++ } ++ break; ++ default: ++ break; ++ } ++ if (ctrl->subcmds.counter_subcmd) { ++ q931_fill_facility_event(ctrl, c); ++ return Q931_RES_HAVEEVENT; ++ } ++ break; + case Q931_PROGRESS: + if (missingmand) { +- q931_status(pri, c, PRI_CAUSE_MANDATORY_IE_MISSING); +- q931_destroycall(pri, c->cr); ++ q931_status(ctrl, c, PRI_CAUSE_MANDATORY_IE_MISSING); ++ pri_destroycall(ctrl, c); + break; + } +- pri->ev.e = PRI_EVENT_PROGRESS; +- pri->ev.proceeding.cause = c->cause; ++ ctrl->ev.e = PRI_EVENT_PROGRESS; ++ ctrl->ev.proceeding.cause = c->cause; + /* Fall through */ + case Q931_CALL_PROCEEDING: ++ stop_t303(c); ++ ctrl->ev.proceeding.subcmds = &ctrl->subcmds; + if (c->newcall) { +- q931_release_complete(pri,c,PRI_CAUSE_INVALID_CALL_REFERENCE); ++ q931_release_complete(ctrl,c,PRI_CAUSE_INVALID_CALL_REFERENCE); + break; + } + if ((c->ourcallstate != Q931_CALL_STATE_CALL_INITIATED) && + (c->ourcallstate != Q931_CALL_STATE_OVERLAP_SENDING) && + (c->ourcallstate != Q931_CALL_STATE_CALL_DELIVERED) && + (c->ourcallstate != Q931_CALL_STATE_OUTGOING_CALL_PROCEEDING)) { +- q931_status(pri,c,PRI_CAUSE_WRONG_MESSAGE); ++ q931_status(ctrl,c,PRI_CAUSE_WRONG_MESSAGE); + break; + } +- pri->ev.proceeding.channel = c->channelno | (c->ds1no << 8) | (c->ds1explicit << 16); ++ ctrl->ev.proceeding.channel = q931_encode_channel(c); + if (mh->msg == Q931_CALL_PROCEEDING) { +- pri->ev.e = PRI_EVENT_PROCEEDING; +- UPDATE_OURCALLSTATE(pri, c, Q931_CALL_STATE_OUTGOING_CALL_PROCEEDING); ++ ctrl->ev.e = PRI_EVENT_PROCEEDING; ++ UPDATE_OURCALLSTATE(ctrl, c, Q931_CALL_STATE_OUTGOING_CALL_PROCEEDING); + c->peercallstate = Q931_CALL_STATE_INCOMING_CALL_PROCEEDING; + } +- pri->ev.proceeding.progress = c->progress; +- pri->ev.proceeding.progressmask = c->progressmask; +- pri->ev.proceeding.cref = c->cr; +- pri->ev.proceeding.call = c; ++ ctrl->ev.proceeding.progress = c->progress; ++ ctrl->ev.proceeding.progressmask = c->progressmask; ++ ctrl->ev.proceeding.cref = c->cr; ++ ctrl->ev.proceeding.call = c->master_call; + +- cur = c->apdus; +- while (cur) { ++ for (cur = c->apdus; cur; cur = cur->next) { + if (!cur->sent && cur->message == Q931_FACILITY) { +- q931_facility(pri, c); ++ q931_facility(ctrl, c); + break; + } +- cur = cur->next; + } + return Q931_RES_HAVEEVENT; + case Q931_CONNECT_ACKNOWLEDGE: + if (c->newcall) { +- q931_release_complete(pri,c,PRI_CAUSE_INVALID_CALL_REFERENCE); ++ q931_release_complete(ctrl,c,PRI_CAUSE_INVALID_CALL_REFERENCE); + break; + } + if (!(c->ourcallstate == Q931_CALL_STATE_CONNECT_REQUEST) && + !(c->ourcallstate == Q931_CALL_STATE_ACTIVE && +- (pri->localtype == PRI_NETWORK || pri->switchtype == PRI_SWITCH_QSIG))) { +- q931_status(pri,c,PRI_CAUSE_WRONG_MESSAGE); ++ (ctrl->localtype == PRI_NETWORK || ctrl->switchtype == PRI_SWITCH_QSIG))) { ++ q931_status(ctrl,c,PRI_CAUSE_WRONG_MESSAGE); + break; + } +- UPDATE_OURCALLSTATE(pri, c, Q931_CALL_STATE_ACTIVE); ++ UPDATE_OURCALLSTATE(ctrl, c, Q931_CALL_STATE_ACTIVE); + c->peercallstate = Q931_CALL_STATE_ACTIVE; + break; + case Q931_STATUS: + if (missingmand) { +- q931_status(pri, c, PRI_CAUSE_MANDATORY_IE_MISSING); +- q931_destroycall(pri, c->cr); ++ q931_status(ctrl, c, PRI_CAUSE_MANDATORY_IE_MISSING); ++ pri_destroycall(ctrl, c); + break; + } + if (c->newcall) { + if (c->cr & 0x7fff) +- q931_release_complete(pri,c,PRI_CAUSE_WRONG_CALL_STATE); ++ q931_release_complete(ctrl,c,PRI_CAUSE_WRONG_CALL_STATE); + break; + } + /* Do nothing */ + /* Also when the STATUS asks for the call of an unexisting reference send RELEASE_COMPL */ +- if ((pri->debug & PRI_DEBUG_Q931_ANOMALY) && ++ if ((ctrl->debug & PRI_DEBUG_Q931_ANOMALY) && + (c->cause != PRI_CAUSE_INTERWORKING)) +- pri_error(pri, "Received unsolicited status: %s\n", pri_cause2str(c->cause)); ++ pri_error(ctrl, "Received unsolicited status: %s\n", pri_cause2str(c->cause)); + /* Workaround for S-12 ver 7.3 - it responds for invalid/non-implemented IEs at SETUP with null call state */ + #if 0 + if (!c->sugcallstate && (c->ourcallstate != Q931_CALL_STATE_CALL_INITIATED)) { +@@ -3737,25 +6774,28 @@ + + if (!c->sugcallstate) { + #endif +- pri->ev.hangup.channel = c->channelno | (c->ds1no << 8) | (c->ds1explicit << 16); +- pri->ev.hangup.cause = c->cause; +- pri->ev.hangup.cref = c->cr; +- pri->ev.hangup.call = c; +- pri->ev.hangup.aoc_units = c->aoc_units; +- libpri_copy_string(pri->ev.hangup.useruserinfo, c->useruserinfo, sizeof(pri->ev.hangup.useruserinfo)); ++ ctrl->ev.hangup.subcmds = &ctrl->subcmds; ++ ctrl->ev.hangup.channel = q931_encode_channel(c); ++ ctrl->ev.hangup.cause = c->cause; ++ ctrl->ev.hangup.cref = c->cr; ++ ctrl->ev.hangup.call = c->master_call; ++ ctrl->ev.hangup.aoc_units = c->aoc_units; ++ ctrl->ev.hangup.call_held = NULL; ++ ctrl->ev.hangup.call_active = NULL; ++ libpri_copy_string(ctrl->ev.hangup.useruserinfo, c->useruserinfo, sizeof(ctrl->ev.hangup.useruserinfo)); + /* Free resources */ +- UPDATE_OURCALLSTATE(pri, c, Q931_CALL_STATE_NULL); ++ UPDATE_OURCALLSTATE(ctrl, c, Q931_CALL_STATE_NULL); + c->peercallstate = Q931_CALL_STATE_NULL; + if (c->alive) { +- pri->ev.e = PRI_EVENT_HANGUP; ++ ctrl->ev.e = PRI_EVENT_HANGUP; + res = Q931_RES_HAVEEVENT; + c->alive = 0; + } else if (c->sendhangupack) { + res = Q931_RES_HAVEEVENT; +- pri->ev.e = PRI_EVENT_HANGUP_ACK; +- q931_hangup(pri, c, c->cause); ++ ctrl->ev.e = PRI_EVENT_HANGUP_ACK; ++ pri_hangup(ctrl, c, c->cause); + } else { +- q931_hangup(pri, c, c->cause); ++ pri_hangup(ctrl, c, c->cause); + res = 0; + } + if (res) +@@ -3763,32 +6803,38 @@ + } + break; + case Q931_RELEASE_COMPLETE: +- UPDATE_OURCALLSTATE(pri, c, Q931_CALL_STATE_NULL); ++ c->hangupinitiated = 1; ++ UPDATE_OURCALLSTATE(ctrl, c, Q931_CALL_STATE_NULL); + c->peercallstate = Q931_CALL_STATE_NULL; +- pri->ev.hangup.channel = c->channelno | (c->ds1no << 8) | (c->ds1explicit << 16); +- pri->ev.hangup.cause = c->cause; +- pri->ev.hangup.cref = c->cr; +- pri->ev.hangup.call = c; +- pri->ev.hangup.aoc_units = c->aoc_units; +- libpri_copy_string(pri->ev.hangup.useruserinfo, c->useruserinfo, sizeof(pri->ev.hangup.useruserinfo)); ++ ctrl->ev.hangup.subcmds = &ctrl->subcmds; ++ ctrl->ev.hangup.channel = q931_encode_channel(c); ++ ctrl->ev.hangup.cause = c->cause; ++ ctrl->ev.hangup.cref = c->cr; ++ ctrl->ev.hangup.call = c->master_call; ++ ctrl->ev.hangup.aoc_units = c->aoc_units; ++ ctrl->ev.hangup.call_held = NULL; ++ ctrl->ev.hangup.call_active = NULL; ++ libpri_copy_string(ctrl->ev.hangup.useruserinfo, c->useruserinfo, sizeof(ctrl->ev.hangup.useruserinfo)); + c->useruserinfo[0] = '\0'; + /* Free resources */ + if (c->alive) { +- pri->ev.e = PRI_EVENT_HANGUP; ++ ctrl->ev.e = PRI_EVENT_HANGUP; + res = Q931_RES_HAVEEVENT; + c->alive = 0; + } else if (c->sendhangupack) { + res = Q931_RES_HAVEEVENT; +- pri->ev.e = PRI_EVENT_HANGUP_ACK; +- pri_hangup(pri, c, c->cause); ++ ctrl->ev.e = PRI_EVENT_HANGUP_ACK; ++ pri_hangup(ctrl, c, c->cause); + } else + res = 0; ++ + if (res) + return res; + else +- q931_hangup(pri,c,c->cause); ++ pri_hangup(ctrl,c,c->cause); + break; + case Q931_RELEASE: ++ c->hangupinitiated = 1; + if (missingmand) { + /* Force cause to be mandatory IE missing */ + c->cause = PRI_CAUSE_MANDATORY_IE_MISSING; +@@ -3798,59 +6844,113 @@ + else { + c->peercallstate = Q931_CALL_STATE_RELEASE_REQUEST; + } +- UPDATE_OURCALLSTATE(pri, c, Q931_CALL_STATE_NULL); +- pri->ev.e = PRI_EVENT_HANGUP; +- pri->ev.hangup.channel = c->channelno | (c->ds1no << 8) | (c->ds1explicit << 16); +- pri->ev.hangup.cause = c->cause; +- pri->ev.hangup.cref = c->cr; +- pri->ev.hangup.call = c; +- pri->ev.hangup.aoc_units = c->aoc_units; +- libpri_copy_string(pri->ev.hangup.useruserinfo, c->useruserinfo, sizeof(pri->ev.hangup.useruserinfo)); ++ UPDATE_OURCALLSTATE(ctrl, c, Q931_CALL_STATE_NULL); ++ ctrl->ev.e = PRI_EVENT_HANGUP; ++ ctrl->ev.hangup.subcmds = &ctrl->subcmds; ++ ctrl->ev.hangup.channel = q931_encode_channel(c); ++ ctrl->ev.hangup.cause = c->cause; ++ ctrl->ev.hangup.cref = c->cr; ++ ctrl->ev.hangup.call = c->master_call; ++ ctrl->ev.hangup.aoc_units = c->aoc_units; ++ ctrl->ev.hangup.call_held = NULL; ++ ctrl->ev.hangup.call_active = NULL; ++ libpri_copy_string(ctrl->ev.hangup.useruserinfo, c->useruserinfo, sizeof(ctrl->ev.hangup.useruserinfo)); + c->useruserinfo[0] = '\0'; + /* Don't send release complete if they send us release + while we sent it, assume a NULL state */ + if (c->newcall) +- q931_release_complete(pri,c,PRI_CAUSE_INVALID_CALL_REFERENCE); +- else ++ q931_release_complete(ctrl,c,PRI_CAUSE_INVALID_CALL_REFERENCE); ++ else if (c->outboundbroadcast && (c != q931_get_subcall_winner(c->master_call))) ++ return pri_hangup(ctrl, c, -1); ++ else + return Q931_RES_HAVEEVENT; + break; + case Q931_DISCONNECT: ++ c->hangupinitiated = 1; + if (missingmand) { + /* Still let user call release */ + c->cause = PRI_CAUSE_MANDATORY_IE_MISSING; + } + if (c->newcall) { +- q931_release_complete(pri,c,PRI_CAUSE_INVALID_CALL_REFERENCE); ++ q931_release_complete(ctrl,c,PRI_CAUSE_INVALID_CALL_REFERENCE); + break; + } +- UPDATE_OURCALLSTATE(pri, c, Q931_CALL_STATE_DISCONNECT_INDICATION); ++ ++ /* ++ * Determine if there are any calls that can be proposed for ++ * a transfer of held call on disconnect. ++ */ ++ ctrl->ev.hangup.call_held = NULL; ++ ctrl->ev.hangup.call_active = NULL; ++ switch (c->ourcallstate) { ++ case Q931_CALL_STATE_OUTGOING_CALL_PROCEEDING: ++ case Q931_CALL_STATE_CALL_DELIVERED: ++ case Q931_CALL_STATE_CALL_RECEIVED: ++ case Q931_CALL_STATE_CONNECT_REQUEST: ++ case Q931_CALL_STATE_INCOMING_CALL_PROCEEDING: ++ case Q931_CALL_STATE_ACTIVE: ++ if (c->master_call->hold_state == Q931_HOLD_STATE_CALL_HELD) { ++ /* Held call is being disconnected first. */ ++ ctrl->ev.hangup.call_held = c->master_call; ++ ctrl->ev.hangup.call_active = q931_find_held_active_call(ctrl, c); ++ } else { ++ /* Active call is being disconnected first. */ ++ if (q931_find_winning_call(c) == c) { ++ /* ++ * Only a normal call or the winning call of a broadcast SETUP ++ * can participate in a transfer of held call on disconnet. ++ */ ++ ctrl->ev.hangup.call_active = c->master_call; ++ ctrl->ev.hangup.call_held = q931_find_held_call(ctrl, c); ++ } ++ } ++ break; ++ default: ++ break; ++ } ++ if (ctrl->debug & PRI_DEBUG_Q931_STATE) { ++ if (ctrl->ev.hangup.call_held) { ++ pri_message(ctrl, "-- Found held call: %p cref:%d\n", ++ ctrl->ev.hangup.call_held, ctrl->ev.hangup.call_held->cr); ++ } ++ if (ctrl->ev.hangup.call_active) { ++ pri_message(ctrl, "-- Found active call: %p cref:%d\n", ++ ctrl->ev.hangup.call_active, ctrl->ev.hangup.call_active->cr); ++ } ++ if (ctrl->ev.hangup.call_held && ctrl->ev.hangup.call_active) { ++ pri_message(ctrl, "-- Transfer held call on disconnect possible.\n"); ++ } ++ } ++ ++ UPDATE_OURCALLSTATE(ctrl, c, Q931_CALL_STATE_DISCONNECT_INDICATION); + c->peercallstate = Q931_CALL_STATE_DISCONNECT_REQUEST; + c->sendhangupack = 1; + + /* wait for a RELEASE so that sufficient time has passed + for the inband audio to be heard */ +- if (pri->acceptinbanddisconnect && (c->progressmask & PRI_PROG_INBAND_AVAILABLE)) ++ if (ctrl->acceptinbanddisconnect && (c->progressmask & PRI_PROG_INBAND_AVAILABLE)) + break; + + /* Return such an event */ +- pri->ev.e = PRI_EVENT_HANGUP_REQ; +- pri->ev.hangup.channel = c->channelno | (c->ds1no << 8) | (c->ds1explicit << 16); +- pri->ev.hangup.cause = c->cause; +- pri->ev.hangup.cref = c->cr; +- pri->ev.hangup.call = c; +- pri->ev.hangup.aoc_units = c->aoc_units; +- libpri_copy_string(pri->ev.hangup.useruserinfo, c->useruserinfo, sizeof(pri->ev.hangup.useruserinfo)); ++ ctrl->ev.e = PRI_EVENT_HANGUP_REQ; ++ ctrl->ev.hangup.subcmds = &ctrl->subcmds; ++ ctrl->ev.hangup.channel = q931_encode_channel(c); ++ ctrl->ev.hangup.cause = c->cause; ++ ctrl->ev.hangup.cref = c->cr; ++ ctrl->ev.hangup.call = c->master_call; ++ ctrl->ev.hangup.aoc_units = c->aoc_units; ++ libpri_copy_string(ctrl->ev.hangup.useruserinfo, c->useruserinfo, sizeof(ctrl->ev.hangup.useruserinfo)); + c->useruserinfo[0] = '\0'; + if (c->alive) + return Q931_RES_HAVEEVENT; + else +- q931_hangup(pri,c,c->cause); ++ pri_hangup(ctrl,c,c->cause); + break; + case Q931_RESTART_ACKNOWLEDGE: +- UPDATE_OURCALLSTATE(pri, c, Q931_CALL_STATE_NULL); ++ UPDATE_OURCALLSTATE(ctrl, c, Q931_CALL_STATE_NULL); + c->peercallstate = Q931_CALL_STATE_NULL; +- pri->ev.e = PRI_EVENT_RESTART_ACK; +- pri->ev.restartack.channel = c->channelno | (c->ds1no << 8) | (c->ds1explicit << 16); ++ ctrl->ev.e = PRI_EVENT_RESTART_ACK; ++ ctrl->ev.restartack.channel = q931_encode_channel(c); + return Q931_RES_HAVEEVENT; + case Q931_INFORMATION: + /* XXX We're handling only INFORMATION messages that contain +@@ -3858,80 +6958,376 @@ + + the "Complete" msg which is basically an EOF on further digits + XXX */ + if (c->newcall) { +- q931_release_complete(pri,c,PRI_CAUSE_INVALID_CALL_REFERENCE); ++ q931_release_complete(ctrl,c,PRI_CAUSE_INVALID_CALL_REFERENCE); + break; + } + if (c->ourcallstate != Q931_CALL_STATE_OVERLAP_RECEIVING) { +- pri->ev.e = PRI_EVENT_KEYPAD_DIGIT; +- pri->ev.digit.call = c; +- pri->ev.digit.channel = c->channelno | (c->ds1no << 8); +- libpri_copy_string(pri->ev.digit.digits, c->keypad_digits, sizeof(pri->ev.digit.digits)); +- /* Make sure we clear it out before we return */ +- c->keypad_digits[0] = '\0'; ++ ctrl->ev.e = PRI_EVENT_KEYPAD_DIGIT; ++ ctrl->ev.digit.subcmds = &ctrl->subcmds; ++ ctrl->ev.digit.call = c->master_call; ++ ctrl->ev.digit.channel = q931_encode_channel(c); ++ libpri_copy_string(ctrl->ev.digit.digits, c->keypad_digits, sizeof(ctrl->ev.digit.digits)); + return Q931_RES_HAVEEVENT; + } +- pri->ev.e = PRI_EVENT_INFO_RECEIVED; +- pri->ev.ring.call = c; +- pri->ev.ring.channel = c->channelno | (c->ds1no << 8) | (c->ds1explicit << 16); +- libpri_copy_string(pri->ev.ring.callednum, c->callednum, sizeof(pri->ev.ring.callednum)); +- libpri_copy_string(pri->ev.ring.callingsubaddr, c->callingsubaddr, sizeof(pri->ev.ring.callingsubaddr)); +- pri->ev.ring.complete = c->complete; /* this covers IE 33 (Sending Complete) */ ++ ctrl->ev.e = PRI_EVENT_INFO_RECEIVED; ++ ctrl->ev.ring.subcmds = &ctrl->subcmds; ++ ctrl->ev.ring.call = c->master_call; ++ ctrl->ev.ring.channel = q931_encode_channel(c); ++ libpri_copy_string(ctrl->ev.ring.callednum, c->overlap_digits, sizeof(ctrl->ev.ring.callednum)); ++ ++ q931_party_id_copy_to_pri(&ctrl->ev.ring.calling, &c->remote_id); ++ /* for backwards compatibility, still need ctrl->ev.ring.callingsubaddr */ ++ if (!c->remote_id.subaddress.type) { /* NSAP: Type = 0 */ ++ libpri_copy_string(ctrl->ev.ring.callingsubaddr, (char *) c->remote_id.subaddress.data, sizeof(ctrl->ev.ring.callingsubaddr)); ++ } else { ++ ctrl->ev.ring.callingsubaddr[0] = '\0'; ++ } ++ ++ ctrl->ev.ring.complete = c->complete; /* this covers IE 33 (Sending Complete) */ + return Q931_RES_HAVEEVENT; + case Q931_STATUS_ENQUIRY: + if (c->newcall) { +- q931_release_complete(pri, c, PRI_CAUSE_INVALID_CALL_REFERENCE); ++ q931_release_complete(ctrl, c, PRI_CAUSE_INVALID_CALL_REFERENCE); + } else +- q931_status(pri,c, 0); ++ q931_status(ctrl,c, 0); + break; + case Q931_SETUP_ACKNOWLEDGE: ++ stop_t303(c); + if (c->newcall) { +- q931_release_complete(pri,c,PRI_CAUSE_INVALID_CALL_REFERENCE); ++ q931_release_complete(ctrl,c,PRI_CAUSE_INVALID_CALL_REFERENCE); + break; + } +- UPDATE_OURCALLSTATE(pri, c, Q931_CALL_STATE_OVERLAP_SENDING); ++ UPDATE_OURCALLSTATE(ctrl, c, Q931_CALL_STATE_OVERLAP_SENDING); + c->peercallstate = Q931_CALL_STATE_OVERLAP_RECEIVING; +- pri->ev.e = PRI_EVENT_SETUP_ACK; +- pri->ev.setup_ack.channel = c->channelno | (c->ds1no << 8) | (c->ds1explicit << 16); +- pri->ev.setup_ack.call = c; ++ ctrl->ev.e = PRI_EVENT_SETUP_ACK; ++ ctrl->ev.setup_ack.subcmds = &ctrl->subcmds; ++ ctrl->ev.setup_ack.channel = q931_encode_channel(c); ++ ctrl->ev.setup_ack.call = c->master_call; + +- cur = c->apdus; +- while (cur) { ++ for (cur = c->apdus; cur; cur = cur->next) { + if (!cur->sent && cur->message == Q931_FACILITY) { +- q931_facility(pri, c); ++ q931_facility(ctrl, c); + break; + } +- cur = cur->next; + } + + return Q931_RES_HAVEEVENT; + case Q931_NOTIFY: +- pri->ev.e = PRI_EVENT_NOTIFY; +- pri->ev.notify.channel = c->channelno; +- pri->ev.notify.info = c->notify; +- return Q931_RES_HAVEEVENT; +- case Q931_USER_INFORMATION: +- case Q931_SEGMENT: +- case Q931_CONGESTION_CONTROL: ++ res = 0; ++ switch (c->notify) { ++ case PRI_NOTIFY_CALL_DIVERTING: ++ if (c->redirection_number.valid) { ++ c->redirecting.to.number = c->redirection_number; ++ if (c->redirecting.count < PRI_MAX_REDIRECTS) { ++ ++c->redirecting.count; ++ } ++ switch (c->ourcallstate) { ++ case Q931_CALL_STATE_CALL_DELIVERED: ++ /* Call is deflecting after we have seen an ALERTING message */ ++ c->redirecting.reason = PRI_REDIR_FORWARD_ON_NO_REPLY; ++ break; ++ default: ++ /* Call is deflecting for call forwarding unconditional or busy reason. */ ++ c->redirecting.reason = PRI_REDIR_UNKNOWN; ++ break; ++ } ++ ++ /* Setup redirecting subcommand */ ++ subcmd = q931_alloc_subcommand(ctrl); ++ if (subcmd) { ++ subcmd->cmd = PRI_SUBCMD_REDIRECTING; ++ q931_party_redirecting_copy_to_pri(&subcmd->u.redirecting, ++ &c->redirecting); ++ } ++ } ++ ++ if (ctrl->subcmds.counter_subcmd) { ++ q931_fill_facility_event(ctrl, c); ++ res = Q931_RES_HAVEEVENT; ++ } ++ break; ++ case PRI_NOTIFY_TRANSFER_ALERTING: ++ case PRI_NOTIFY_TRANSFER_ACTIVE: ++ if (c->redirection_number.valid ++ && q931_party_number_cmp(&c->remote_id.number, &c->redirection_number)) { ++ /* The remote party information changed. */ ++ c->remote_id.number = c->redirection_number; ++ ++ /* Setup connected line subcommand */ ++ subcmd = q931_alloc_subcommand(ctrl); ++ if (subcmd) { ++ subcmd->cmd = PRI_SUBCMD_CONNECTED_LINE; ++ q931_party_id_copy_to_pri(&subcmd->u.connected_line.id, ++ &c->remote_id); ++ } ++ } ++ ++ if (ctrl->subcmds.counter_subcmd) { ++ q931_fill_facility_event(ctrl, c); ++ res = Q931_RES_HAVEEVENT; ++ } ++ break; ++ default: ++ ctrl->ev.e = PRI_EVENT_NOTIFY; ++ ctrl->ev.notify.subcmds = &ctrl->subcmds; ++ ctrl->ev.notify.channel = q931_encode_channel(c); ++ ctrl->ev.notify.info = c->notify; ++ ctrl->ev.notify.call = c->master_call; ++ res = Q931_RES_HAVEEVENT; ++ break; ++ } ++ return res; + case Q931_HOLD: ++ res = 0; ++ if (!PRI_MASTER(ctrl)->hold_support) { ++ /* ++ * Blocking any calls from getting on HOLD effectively ++ * disables HOLD/RETRIEVE. ++ */ ++ q931_send_hold_rej_msg(ctrl, c, PRI_CAUSE_FACILITY_NOT_IMPLEMENTED); ++ break; ++ } ++ switch (c->ourcallstate) { ++ case Q931_CALL_STATE_CALL_RECEIVED: ++ case Q931_CALL_STATE_CONNECT_REQUEST: ++ case Q931_CALL_STATE_INCOMING_CALL_PROCEEDING: ++ if (q931_is_ptmp(ctrl)) { ++ /* HOLD request only allowed in these states if point-to-point mode. */ ++ q931_send_hold_rej_msg(ctrl, c, PRI_CAUSE_WRONG_CALL_STATE); ++ break; ++ } ++ /* Fall through */ ++ case Q931_CALL_STATE_OUTGOING_CALL_PROCEEDING: ++ case Q931_CALL_STATE_CALL_DELIVERED: ++ case Q931_CALL_STATE_ACTIVE: ++ if (!q931_find_winning_call(c)) { ++ /* ++ * Only the winning call of a broadcast SETUP can do hold since the ++ * call must be answered first. ++ */ ++ q931_send_hold_rej_msg(ctrl, c, PRI_CAUSE_WRONG_CALL_STATE); ++ break; ++ } ++ master_call = c->master_call; ++ switch (master_call->hold_state) { ++ case Q931_HOLD_STATE_HOLD_REQ: ++ if (ctrl->localtype == PRI_NETWORK) { ++ /* The network ignores HOLD request on a hold collision. */ ++ break; ++ } ++ /* Fall through */ ++ case Q931_HOLD_STATE_IDLE: ++ ctrl->ev.e = PRI_EVENT_HOLD; ++ ctrl->ev.hold.channel = q931_encode_channel(c); ++ ctrl->ev.hold.call = master_call; ++ ctrl->ev.hold.subcmds = &ctrl->subcmds; ++ res = Q931_RES_HAVEEVENT; ++ ++ UPDATE_HOLD_STATE(ctrl, master_call, Q931_HOLD_STATE_HOLD_IND); ++ ++ /* Stop any T-HOLD timer from possible hold collision. */ ++ pri_schedule_del(ctrl, master_call->hold_timer); ++ master_call->hold_timer = 0; ++ break; ++ default: ++ q931_send_hold_rej_msg(ctrl, c, PRI_CAUSE_WRONG_CALL_STATE); ++ break; ++ } ++ break; ++ case Q931_CALL_STATE_DISCONNECT_INDICATION: ++ case Q931_CALL_STATE_RELEASE_REQUEST: ++ /* Ignore HOLD request in these states. */ ++ break; ++ default: ++ q931_send_hold_rej_msg(ctrl, c, PRI_CAUSE_WRONG_CALL_STATE); ++ break; ++ } ++ return res; + case Q931_HOLD_ACKNOWLEDGE: ++ res = 0; ++ master_call = c->master_call; ++ switch (master_call->hold_state) { ++ case Q931_HOLD_STATE_HOLD_REQ: ++ ctrl->ev.e = PRI_EVENT_HOLD_ACK; ++ ctrl->ev.hold_ack.channel = q931_encode_channel(c); ++ ctrl->ev.hold_ack.call = master_call; ++ ctrl->ev.hold_ack.subcmds = &ctrl->subcmds; ++ res = Q931_RES_HAVEEVENT; ++ ++ UPDATE_HOLD_STATE(ctrl, master_call, Q931_HOLD_STATE_CALL_HELD); ++ ++ /* Call is now on hold so forget the channel. */ ++ c->channelno = 0;/* No channel */ ++ c->ds1no = 0; ++ c->ds1explicit = 0; ++ c->chanflags = 0; ++ ++ /* Stop T-HOLD timer */ ++ pri_schedule_del(ctrl, master_call->hold_timer); ++ master_call->hold_timer = 0; ++ break; ++ default: ++ /* Ignore response. Response is late or spurrious. */ ++ break; ++ } ++ return res; + case Q931_HOLD_REJECT: ++ res = 0; ++ master_call = c->master_call; ++ switch (master_call->hold_state) { ++ case Q931_HOLD_STATE_HOLD_REQ: ++ if (missingmand) { ++ /* Still, let hold rejection continue. */ ++ c->cause = PRI_CAUSE_MANDATORY_IE_MISSING; ++ } ++ ctrl->ev.e = PRI_EVENT_HOLD_REJ; ++ ctrl->ev.hold_rej.channel = q931_encode_channel(c); ++ ctrl->ev.hold_rej.call = master_call; ++ ctrl->ev.hold_rej.cause = c->cause; ++ ctrl->ev.hold_rej.subcmds = &ctrl->subcmds; ++ res = Q931_RES_HAVEEVENT; ++ ++ UPDATE_HOLD_STATE(ctrl, master_call, Q931_HOLD_STATE_IDLE); ++ ++ /* Stop T-HOLD timer */ ++ pri_schedule_del(ctrl, master_call->hold_timer); ++ master_call->hold_timer = 0; ++ break; ++ default: ++ /* Ignore response. Response is late or spurrious. */ ++ break; ++ } ++ return res; + case Q931_RETRIEVE: ++ res = 0; ++ switch (c->ourcallstate) { ++ case Q931_CALL_STATE_CALL_RECEIVED: ++ case Q931_CALL_STATE_CONNECT_REQUEST: ++ case Q931_CALL_STATE_INCOMING_CALL_PROCEEDING: ++ if (q931_is_ptmp(ctrl)) { ++ /* RETRIEVE request only allowed in these states if point-to-point mode. */ ++ q931_send_retrieve_rej_msg(ctrl, c, PRI_CAUSE_WRONG_CALL_STATE); ++ break; ++ } ++ /* Fall through */ ++ case Q931_CALL_STATE_OUTGOING_CALL_PROCEEDING: ++ case Q931_CALL_STATE_CALL_DELIVERED: ++ case Q931_CALL_STATE_ACTIVE: ++ if (!q931_find_winning_call(c)) { ++ /* ++ * Only the winning call of a broadcast SETUP can do hold since the ++ * call must be answered first. ++ */ ++ q931_send_retrieve_rej_msg(ctrl, c, PRI_CAUSE_WRONG_CALL_STATE); ++ break; ++ } ++ master_call = c->master_call; ++ switch (master_call->hold_state) { ++ case Q931_HOLD_STATE_RETRIEVE_REQ: ++ if (ctrl->localtype == PRI_NETWORK) { ++ /* The network ignores RETRIEVE request on a retrieve collision. */ ++ break; ++ } ++ /* Fall through */ ++ case Q931_HOLD_STATE_CALL_HELD: ++ ctrl->ev.e = PRI_EVENT_RETRIEVE; ++ ctrl->ev.retrieve.channel = q931_encode_channel(c); ++ ctrl->ev.retrieve.call = master_call; ++ ctrl->ev.retrieve.flexible = !(c->chanflags & FLAG_EXCLUSIVE); ++ ctrl->ev.retrieve.subcmds = &ctrl->subcmds; ++ res = Q931_RES_HAVEEVENT; ++ ++ UPDATE_HOLD_STATE(ctrl, master_call, Q931_HOLD_STATE_RETRIEVE_IND); ++ ++ /* Stop any T-RETRIEVE timer from possible retrieve collision. */ ++ pri_schedule_del(ctrl, master_call->hold_timer); ++ master_call->hold_timer = 0; ++ break; ++ default: ++ q931_send_retrieve_rej_msg(ctrl, c, PRI_CAUSE_WRONG_CALL_STATE); ++ break; ++ } ++ break; ++ case Q931_CALL_STATE_DISCONNECT_INDICATION: ++ case Q931_CALL_STATE_RELEASE_REQUEST: ++ /* Ignore RETRIEVE request in these states. */ ++ break; ++ default: ++ q931_send_retrieve_rej_msg(ctrl, c, PRI_CAUSE_WRONG_CALL_STATE); ++ break; ++ } ++ return res; + case Q931_RETRIEVE_ACKNOWLEDGE: ++ res = 0; ++ master_call = c->master_call; ++ switch (master_call->hold_state) { ++ case Q931_HOLD_STATE_RETRIEVE_REQ: ++ UPDATE_HOLD_STATE(ctrl, master_call, Q931_HOLD_STATE_IDLE); ++ ++ /* Stop T-RETRIEVE timer */ ++ pri_schedule_del(ctrl, master_call->hold_timer); ++ master_call->hold_timer = 0; ++ ++ ctrl->ev.e = PRI_EVENT_RETRIEVE_ACK; ++ ctrl->ev.retrieve_ack.channel = q931_encode_channel(c); ++ ctrl->ev.retrieve_ack.call = master_call; ++ ctrl->ev.retrieve_ack.subcmds = &ctrl->subcmds; ++ res = Q931_RES_HAVEEVENT; ++ break; ++ default: ++ /* Ignore response. Response is late or spurrious. */ ++ break; ++ } ++ return res; + case Q931_RETRIEVE_REJECT: ++ res = 0; ++ master_call = c->master_call; ++ switch (master_call->hold_state) { ++ case Q931_HOLD_STATE_RETRIEVE_REQ: ++ UPDATE_HOLD_STATE(ctrl, master_call, Q931_HOLD_STATE_CALL_HELD); ++ ++ /* Call is still on hold so forget the channel. */ ++ c->channelno = 0;/* No channel */ ++ c->ds1no = 0; ++ c->ds1explicit = 0; ++ c->chanflags = 0; ++ ++ /* Stop T-RETRIEVE timer */ ++ pri_schedule_del(ctrl, master_call->hold_timer); ++ master_call->hold_timer = 0; ++ ++ if (missingmand) { ++ /* Still, let retrive rejection continue. */ ++ c->cause = PRI_CAUSE_MANDATORY_IE_MISSING; ++ } ++ ctrl->ev.e = PRI_EVENT_RETRIEVE_REJ; ++ ctrl->ev.retrieve_rej.channel = q931_encode_channel(c); ++ ctrl->ev.retrieve_rej.call = master_call; ++ ctrl->ev.retrieve_rej.cause = c->cause; ++ ctrl->ev.retrieve_rej.subcmds = &ctrl->subcmds; ++ res = Q931_RES_HAVEEVENT; ++ break; ++ default: ++ /* Ignore response. Response is late or spurrious. */ ++ break; ++ } ++ return res; ++ case Q931_USER_INFORMATION: ++ case Q931_SEGMENT: ++ case Q931_CONGESTION_CONTROL: + case Q931_RESUME: + case Q931_RESUME_ACKNOWLEDGE: + case Q931_RESUME_REJECT: + case Q931_SUSPEND: + case Q931_SUSPEND_ACKNOWLEDGE: + case Q931_SUSPEND_REJECT: +- pri_error(pri, "!! Not yet handling post-handle message type %s (%d)\n", msg2str(mh->msg), mh->msg); ++ pri_error(ctrl, "!! Not yet handling post-handle message type %s (%d)\n", msg2str(mh->msg), mh->msg); + /* Fall through */ + default: +- +- pri_error(pri, "!! Don't know how to post-handle message type %s (%d)\n", msg2str(mh->msg), mh->msg); +- q931_status(pri,c, PRI_CAUSE_MESSAGE_TYPE_NONEXIST); ++ pri_error(ctrl, "!! Don't know how to post-handle message type %s (%d)\n", msg2str(mh->msg), mh->msg); ++ q931_status(ctrl,c, PRI_CAUSE_MESSAGE_TYPE_NONEXIST); + if (c->newcall) +- q931_destroycall(pri,c->cr); ++ pri_destroycall(ctrl, c); + return -1; + } + return 0; +@@ -3941,40 +7337,46 @@ + static int pri_internal_clear(void *data) + { + struct q931_call *c = data; +- struct pri *pri = c->pri; ++ struct pri *ctrl = c->pri; + int res; + +- if (c->retranstimer) +- pri_schedule_del(pri, c->retranstimer); ++ pri_schedule_del(ctrl, c->retranstimer); + c->retranstimer = 0; + c->useruserinfo[0] = '\0'; +- c->cause = -1; ++ //c->cause = -1; + c->causecode = -1; + c->causeloc = -1; +- c->sugcallstate = -1; ++ c->sugcallstate = Q931_CALL_STATE_NOT_SET; + c->aoc_units = -1; + +- UPDATE_OURCALLSTATE(pri, c, Q931_CALL_STATE_NULL); ++ UPDATE_OURCALLSTATE(ctrl, c, Q931_CALL_STATE_NULL); + c->peercallstate = Q931_CALL_STATE_NULL; +- pri->ev.hangup.channel = c->channelno | (c->ds1no << 8) | (c->ds1explicit << 16); +- pri->ev.hangup.cause = c->cause; +- pri->ev.hangup.cref = c->cr; +- pri->ev.hangup.call = c; +- pri->ev.hangup.aoc_units = c->aoc_units; +- libpri_copy_string(pri->ev.hangup.useruserinfo, c->useruserinfo, sizeof(pri->ev.hangup.useruserinfo)); ++ q931_clr_subcommands(ctrl); ++ ctrl->ev.hangup.subcmds = &ctrl->subcmds; ++ ctrl->ev.hangup.channel = q931_encode_channel(c); ++ ctrl->ev.hangup.cause = c->cause; ++ ctrl->ev.hangup.cref = c->cr; ++ ctrl->ev.hangup.call = c->master_call; ++ ctrl->ev.hangup.aoc_units = c->aoc_units; ++ ctrl->ev.hangup.call_held = NULL; ++ ctrl->ev.hangup.call_active = NULL; ++ libpri_copy_string(ctrl->ev.hangup.useruserinfo, c->useruserinfo, sizeof(ctrl->ev.hangup.useruserinfo)); + ++ if (ctrl->debug & PRI_DEBUG_Q931_STATE) { ++ pri_message(ctrl, "clearing, alive %d, hangupack %d\n", c->alive, c->sendhangupack); ++ } + /* Free resources */ + if (c->alive) { +- pri->ev.e = PRI_EVENT_HANGUP; ++ ctrl->ev.e = PRI_EVENT_HANGUP; + res = Q931_RES_HAVEEVENT; + c->alive = 0; + } else if (c->sendhangupack) { + res = Q931_RES_HAVEEVENT; +- pri->ev.e = PRI_EVENT_HANGUP_ACK; +- q931_hangup(pri, c, c->cause); ++ ctrl->ev.e = PRI_EVENT_HANGUP_ACK; ++ pri_hangup(ctrl, c, c->cause); + } else { + res = 0; +- q931_hangup(pri, c, c->cause); ++ pri_hangup(ctrl, c, c->cause); + } + + return res; +@@ -3984,94 +7386,97 @@ + static void pri_dl_down_timeout(void *data) + { + struct q931_call *c = data; +- struct pri *pri = c->pri; +- if (pri->debug & PRI_DEBUG_Q931_STATE) +- pri_message(pri, DBGHEAD "Timed out waiting for data link re-establishment\n", DBGINFO); ++ struct pri *ctrl = c->pri; ++ if (ctrl->debug & PRI_DEBUG_Q931_STATE) ++ pri_message(ctrl, DBGHEAD "Timed out waiting for data link re-establishment\n", DBGINFO); + + c->cause = PRI_CAUSE_DESTINATION_OUT_OF_ORDER; + if (pri_internal_clear(c) == Q931_RES_HAVEEVENT) +- pri->schedev = 1; ++ ctrl->schedev = 1; + } + + /* Handle Layer 2 down event for a non active call. */ + static void pri_dl_down_cancelcall(void *data) + { + struct q931_call *c = data; +- struct pri *pri = c->pri; +- if (pri->debug & PRI_DEBUG_Q931_STATE) +- pri_message(pri, DBGHEAD "Cancel non active call after data link failure\n", DBGINFO); ++ struct pri *ctrl = c->pri; ++ if (ctrl->debug & PRI_DEBUG_Q931_STATE) ++ pri_message(ctrl, DBGHEAD "Cancel non active call after data link failure\n", DBGINFO); + + c->cause = PRI_CAUSE_DESTINATION_OUT_OF_ORDER; + if (pri_internal_clear(c) == Q931_RES_HAVEEVENT) +- pri->schedev = 1; ++ ctrl->schedev = 1; + } + + /* Receive an indication from Layer 2 */ +-void q931_dl_indication(struct pri *pri, int event) ++void q931_dl_indication(struct pri *ctrl, int event) + { + q931_call *cur = NULL; + + /* Just return if T309 is not enabled. */ +- if (!pri || pri->timers[PRI_TIMER_T309] < 0) ++ if (!ctrl || ctrl->timers[PRI_TIMER_T309] < 0) + return; + + switch (event) { + case PRI_EVENT_DCHAN_DOWN: +- pri_message(pri, DBGHEAD "link is DOWN\n", DBGINFO); +- cur = *pri->callpool; ++ pri_message(ctrl, DBGHEAD "link is DOWN\n", DBGINFO); ++ cur = *ctrl->callpool; + while(cur) { + if (cur->ourcallstate == Q931_CALL_STATE_ACTIVE) { + /* For a call in Active state, activate T309 only if there is no timer already running. */ + if (!cur->retranstimer) { +- pri_message(pri, DBGHEAD "activate T309 for call %d on channel %d\n", DBGINFO, cur->cr, cur->channelno); +- cur->retranstimer = pri_schedule_event(pri, pri->timers[PRI_TIMER_T309], pri_dl_down_timeout, cur); ++ pri_message(ctrl, DBGHEAD "activate T309 for call %d on channel %d\n", DBGINFO, cur->cr, cur->channelno); ++ cur->retranstimer = pri_schedule_event(ctrl, ctrl->timers[PRI_TIMER_T309], pri_dl_down_timeout, cur); + } + } else if (cur->ourcallstate != Q931_CALL_STATE_NULL) { + /* For a call that is not in Active state, schedule internal clearing of the call 'ASAP' (delay 0). */ +- pri_message(pri, DBGHEAD "cancel call %d on channel %d in state %d (%s)\n", DBGINFO, +- cur->cr, cur->channelno, cur->ourcallstate, callstate2str(cur->ourcallstate)); +- if (cur->retranstimer) +- pri_schedule_del(pri, cur->retranstimer); +- cur->retranstimer = pri_schedule_event(pri, 0, pri_dl_down_cancelcall, cur); ++ pri_message(ctrl, DBGHEAD "cancel call %d on channel %d in state %d (%s)\n", DBGINFO, ++ cur->cr, cur->channelno, cur->ourcallstate, ++ q931_call_state_str(cur->ourcallstate)); ++ pri_schedule_del(ctrl, cur->retranstimer); ++ cur->retranstimer = pri_schedule_event(ctrl, 0, pri_dl_down_cancelcall, cur); + } + cur = cur->next; + } + break; + case PRI_EVENT_DCHAN_UP: +- pri_message(pri, DBGHEAD "link is UP\n", DBGINFO); +- cur = *pri->callpool; ++ pri_message(ctrl, DBGHEAD "link is UP\n", DBGINFO); ++ cur = *ctrl->callpool; + while(cur) { + if (cur->ourcallstate == Q931_CALL_STATE_ACTIVE && cur->retranstimer) { +- pri_message(pri, DBGHEAD "cancel T309 for call %d on channel %d\n", DBGINFO, cur->cr, cur->channelno); +- pri_schedule_del(pri, cur->retranstimer); ++ pri_message(ctrl, DBGHEAD "cancel T309 for call %d on channel %d\n", DBGINFO, cur->cr, cur->channelno); ++ pri_schedule_del(ctrl, cur->retranstimer); + cur->retranstimer = 0; +- q931_status(pri, cur, PRI_CAUSE_NORMAL_UNSPECIFIED); ++ q931_status(ctrl, cur, PRI_CAUSE_NORMAL_UNSPECIFIED); + } else if (cur->ourcallstate != Q931_CALL_STATE_NULL && + cur->ourcallstate != Q931_CALL_STATE_DISCONNECT_REQUEST && + cur->ourcallstate != Q931_CALL_STATE_DISCONNECT_INDICATION && + cur->ourcallstate != Q931_CALL_STATE_RELEASE_REQUEST) { + + /* The STATUS message sent here is not required by Q.931, but it may help anyway. */ +- q931_status(pri, cur, PRI_CAUSE_NORMAL_UNSPECIFIED); ++ q931_status(ctrl, cur, PRI_CAUSE_NORMAL_UNSPECIFIED); + } + cur = cur->next; + } + break; + default: +- pri_message(pri, DBGHEAD "unexpected event %d.\n", DBGINFO, event); ++ pri_message(ctrl, DBGHEAD "unexpected event %d.\n", DBGINFO, event); + } + } + +-int q931_call_getcrv(struct pri *pri, q931_call *call, int *callmode) ++int q931_call_getcrv(struct pri *ctrl, q931_call *call, int *callmode) + { + if (callmode) + *callmode = call->cr & 0x7; + return ((call->cr & 0x7fff) >> 3); + } + +-int q931_call_setcrv(struct pri *pri, q931_call *call, int crv, int callmode) ++int q931_call_setcrv(struct pri *ctrl, q931_call *call, int crv, int callmode) + { +- call->cr = (crv << 3) & 0x7fff; +- call->cr |= (callmode & 0x7); ++ /* Do not allow changing the dummy call reference */ ++ if (!q931_is_dummy_call(call)) { ++ call->cr = (crv << 3) & 0x7fff; ++ call->cr |= (callmode & 0x7); ++ } + return 0; + } +Index: asn1.h +=================================================================== +--- a/asn1.h (.../tags/1.4.10.2) (revision 0) ++++ b/asn1.h (.../branches/1.4) (revision 1357) +@@ -0,0 +1,257 @@ ++/* ++ * libpri: An implementation of Primary Rate ISDN ++ * ++ * Copyright (C) 2009 Digium, Inc. ++ * ++ * Richard Mudgett <rmudgett@digium.com> ++ * ++ * See http://www.asterisk.org for more information about ++ * the Asterisk project. Please do not directly contact ++ * any of the maintainers of this project for assistance; ++ * the project provides a web site, mailing lists and IRC ++ * channels for your use. ++ * ++ * This program is free software, distributed under the terms of ++ * the GNU General Public License Version 2 as published by the ++ * Free Software Foundation. See the LICENSE file included with ++ * this program for more details. ++ * ++ * In addition, when this program is distributed with Asterisk in ++ * any form that would qualify as a 'combined work' or as a ++ * 'derivative work' (but not mere aggregation), you can redistribute ++ * and/or modify the combination under the terms of the license ++ * provided with that copy of Asterisk, instead of the license ++ * terms granted here. ++ */ ++ ++/*! ++ * \file ++ * \brief ASN.1 definitions and prototypes ++ * ++ * \details ++ * This file contains all ASN.1 primitive data structures and ++ * definitions needed for ROSE component encoding and decoding. ++ * ++ * ROSE - Remote Operations Service Element ++ * ASN.1 - Abstract Syntax Notation 1 ++ * APDU - Application Protocol Data Unit ++ * ++ * \author Richard Mudgett <rmudgett@digium.com> ++ */ ++ ++#ifndef _LIBPRI_ASN1_H ++#define _LIBPRI_ASN1_H ++ ++#include <string.h> ++#include <sys/types.h> ++ ++#ifdef __cplusplus ++extern "C" { ++#endif ++ ++/* ------------------------------------------------------------------- */ ++ ++/*! ASN.1 Identifier Octet - Tag class bits */ ++#define ASN1_CLASS_MASK 0xc0 ++#define ASN1_CLASS_UNIVERSAL 0x00 /*!< Universal primitive data types */ ++#define ASN1_CLASS_APPLICATION 0x40 /*!< Application wide data tag */ ++#define ASN1_CLASS_CONTEXT_SPECIFIC 0x80 /*!< Context specifc data tag */ ++#define ASN1_CLASS_PRIVATE 0xc0 /*!< Private organization data tag */ ++ ++/*! ASN.1 Identifier Octet - Primitive/Constructor bit */ ++#define ASN1_PC_MASK 0x20 ++#define ASN1_PC_PRIMITIVE 0x00 ++#define ASN1_PC_CONSTRUCTED 0x20 ++ ++/*! ASN.1 Identifier Octet - Universal data types */ ++#define ASN1_TYPE_MASK 0x1f ++#define ASN1_TYPE_INDEF_TERM 0x00 /* 0 */ ++#define ASN1_TYPE_BOOLEAN 0x01 /* 1 */ ++#define ASN1_TYPE_INTEGER 0x02 /* 2 */ ++#define ASN1_TYPE_BIT_STRING 0x03 /* 3 */ ++#define ASN1_TYPE_OCTET_STRING 0x04 /* 4 */ ++#define ASN1_TYPE_NULL 0x05 /* 5 */ ++#define ASN1_TYPE_OBJECT_IDENTIFIER 0x06 /* 6 */ ++#define ASN1_TYPE_OBJECT_DESCRIPTOR 0x07 /* 7 */ ++#define ASN1_TYPE_EXTERN 0x08 /* 8 */ ++#define ASN1_TYPE_REAL 0x09 /* 9 */ ++#define ASN1_TYPE_ENUMERATED 0x0a /* 10 */ ++#define ASN1_TYPE_EMBEDDED_PDV 0x0b /* 11 */ ++#define ASN1_TYPE_UTF8_STRING 0x0c /* 12 */ ++#define ASN1_TYPE_RELATIVE_OID 0x0d /* 13 */ ++/* 0x0e & 0x0f are reserved for future ASN.1 editions */ ++#define ASN1_TYPE_SEQUENCE 0x10 /* 16 */ ++#define ASN1_TYPE_SET 0x11 /* 17 */ ++#define ASN1_TYPE_NUMERIC_STRING 0x12 /* 18 */ ++#define ASN1_TYPE_PRINTABLE_STRING 0x13 /* 19 */ ++#define ASN1_TYPE_TELETEX_STRING 0x14 /* 20 */ ++#define ASN1_TYPE_VIDEOTEX_STRING 0x15 /* 21 */ ++#define ASN1_TYPE_IA5_STRING 0x16 /* 22 */ ++#define ASN1_TYPE_UTC_TIME 0x17 /* 23 */ ++#define ASN1_TYPE_GENERALIZED_TIME 0x18 /* 24 */ ++#define ASN1_TYPE_GRAPHIC_STRING 0x19 /* 25 */ ++#define ASN1_TYPE_VISIBLE_STRING 0x1a /* 26 */ ++#define ASN1_TYPE_ISO646_STRING 0x1a /* 26 */ ++#define ASN1_TYPE_GENERAL_STRING 0x1b /* 27 */ ++#define ASN1_TYPE_UNIVERSAL_STRING 0x1c /* 28 */ ++#define ASN1_TYPE_CHAR_STRING 0x1d /* 29 */ ++#define ASN1_TYPE_BMP_STRING 0x1e /* 30 */ ++#define ASN1_TYPE_EXTENSION 0x1f /* 31 */ ++ ++#define ASN1_TAG_SEQUENCE (ASN1_CLASS_UNIVERSAL | ASN1_PC_CONSTRUCTED | ASN1_TYPE_SEQUENCE) ++#define ASN1_TAG_SET (ASN1_CLASS_UNIVERSAL | ASN1_PC_CONSTRUCTED | ASN1_TYPE_SET) ++ ++#define ASN1_INDEF_TERM (ASN1_CLASS_UNIVERSAL | ASN1_PC_PRIMITIVE | ASN1_TYPE_INDEF_TERM) ++#define ASN1_INDEF_TERM_LEN 2 ++ ++struct asn1_oid { ++ /*! \brief Number of subidentifier values in OID list */ ++ u_int16_t num_values; ++ ++ /*! ++ * \brief OID subidentifier value list ++ * \note The first value is really the first two OID subidentifiers. ++ * They are compressed using this formula: ++ * First_Value = (First_Subidentifier * 40) + Second_Subidentifier ++ */ ++ u_int16_t value[10]; ++}; ++ ++#define ASN1_CALL(new_pos, do_it) \ ++ do \ ++ { \ ++ (new_pos) = (do_it); \ ++ if (!(new_pos)) { \ ++ return NULL; \ ++ } \ ++ } while (0) ++ ++/*! \brief Determine the ending position of the set or sequence to verify the length. */ ++#define ASN1_END_SETUP(component_end, offset, length, pos, end) \ ++ do { \ ++ if ((length) < 0) { \ ++ (offset) = ASN1_INDEF_TERM_LEN; \ ++ (component_end) = (end); \ ++ } else { \ ++ (offset) = 0; \ ++ (component_end) = (pos) + (length); \ ++ } \ ++ } while (0) ++ ++/*! \brief Account for the indefinite length terminator of the set or sequence. */ ++#define ASN1_END_FIXUP(ctrl, pos, offset, component_end, end) \ ++ do { \ ++ if (offset) { \ ++ ASN1_CALL((pos), asn1_dec_indef_end_fixup((ctrl), (pos), (end))); \ ++ } else if ((pos) != (component_end)) { \ ++ if ((ctrl)->debug & PRI_DEBUG_APDU) { \ ++ pri_message((ctrl), \ ++ " Skipping unused constructed component octets!\n"); \ ++ } \ ++ (pos) = (component_end); \ ++ } \ ++ } while (0) ++ ++#define ASN1_DID_NOT_EXPECT_TAG(ctrl, tag) \ ++ do { \ ++ if ((ctrl)->debug & PRI_DEBUG_APDU) { \ ++ pri_message((ctrl), " Did not expect: %s\n", asn1_tag2str(tag)); \ ++ } \ ++ } while (0) ++ ++#define ASN1_CHECK_TAG(ctrl, actual_tag, match_tag, expected_tag) \ ++ do { \ ++ if ((match_tag) != (expected_tag)) { \ ++ ASN1_DID_NOT_EXPECT_TAG((ctrl), (actual_tag)); \ ++ return NULL; \ ++ } \ ++ } while (0) ++ ++ ++const unsigned char *asn1_dec_tag(const unsigned char *tag_pos, const unsigned char *end, ++ unsigned *tag); ++const unsigned char *asn1_dec_length(const unsigned char *len_pos, ++ const unsigned char *end, int *length); ++const unsigned char *asn1_dec_indef_end_fixup(struct pri *ctrl, const unsigned char *pos, ++ const unsigned char *end); ++ ++const unsigned char *asn1_dec_boolean(struct pri *ctrl, const char *name, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, int32_t *value); ++const unsigned char *asn1_dec_int(struct pri *ctrl, const char *name, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, int32_t *value); ++const unsigned char *asn1_dec_null(struct pri *ctrl, const char *name, unsigned tag, ++ const unsigned char *pos, const unsigned char *end); ++const unsigned char *asn1_dec_oid(struct pri *ctrl, const char *name, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, struct asn1_oid *oid); ++const unsigned char *asn1_dec_string_bin(struct pri *ctrl, const char *name, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, size_t buf_size, ++ unsigned char *str, size_t *str_len); ++const unsigned char *asn1_dec_string_max(struct pri *ctrl, const char *name, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, size_t buf_size, ++ unsigned char *str, size_t *str_len); ++ ++const char *asn1_tag2str(unsigned tag); ++void asn1_dump(struct pri *ctrl, const unsigned char *start_asn1, ++ const unsigned char *end); ++ ++ ++#define ASN1_LEN_FORM_SHORT 1 /*!< Hint that the final length will be less than 128 octets */ ++#define ASN1_LEN_FORM_LONG_U8 2 /*!< Hint that the final length will be less than 256 octets */ ++#define ASN1_LEN_FORM_LONG_U16 3 /*!< Hint that the final length will be less than 65536 octets */ ++#define ASN1_LEN_INIT(len_pos, end, form_hint) \ ++ do { \ ++ if ((end) < (len_pos) + (form_hint)) { \ ++ return NULL; \ ++ } \ ++ *(len_pos) = (form_hint); \ ++ (len_pos) += (form_hint); \ ++ } while (0) ++ ++#define ASN1_LEN_FIXUP(len_pos, component_end, end) \ ++ ASN1_CALL((component_end), asn1_enc_length_fixup((len_pos), (component_end), (end))) ++ ++/*! \brief Use to begin encoding explicit tags, SET, and SEQUENCE constructed groupings. */ ++#define ASN1_CONSTRUCTED_BEGIN(len_pos_save, pos, end, tag) \ ++ do { \ ++ if ((end) < (pos) + (1 + ASN1_LEN_FORM_SHORT)) { \ ++ return NULL; \ ++ } \ ++ *(pos)++ = (tag) | ASN1_PC_CONSTRUCTED; \ ++ (len_pos_save) = (pos); \ ++ *(pos) = ASN1_LEN_FORM_SHORT; \ ++ (pos) += ASN1_LEN_FORM_SHORT; \ ++ } while (0) ++ ++/*! \brief Use to end encoding explicit tags, SET, and SEQUENCE constructed groupings. */ ++#define ASN1_CONSTRUCTED_END(len_pos, component_end, end) \ ++ ASN1_CALL((component_end), asn1_enc_length_fixup((len_pos), (component_end), (end))) ++ ++#define ASN1_ENC_ERROR(ctrl, msg) \ ++ pri_error((ctrl), "%s error: %s\n", __FUNCTION__, (msg)) ++ ++unsigned char *asn1_enc_length(unsigned char *len_pos, unsigned char *end, ++ size_t str_len); ++unsigned char *asn1_enc_length_fixup(unsigned char *len_pos, ++ unsigned char *component_end, unsigned char *end); ++ ++unsigned char *asn1_enc_boolean(unsigned char *pos, unsigned char *end, unsigned tag, ++ int32_t value); ++unsigned char *asn1_enc_int(unsigned char *pos, unsigned char *end, unsigned tag, ++ int32_t value); ++unsigned char *asn1_enc_null(unsigned char *pos, unsigned char *end, unsigned tag); ++unsigned char *asn1_enc_oid(unsigned char *pos, unsigned char *end, unsigned tag, ++ const struct asn1_oid *oid); ++unsigned char *asn1_enc_string_bin(unsigned char *pos, unsigned char *end, unsigned tag, ++ const unsigned char *str, size_t str_len); ++unsigned char *asn1_enc_string_max(unsigned char *pos, unsigned char *end, unsigned tag, ++ const unsigned char *str, size_t max_len); ++ ++/* ------------------------------------------------------------------- */ ++ ++#ifdef __cplusplus ++} ++#endif ++ ++#endif /* _LIBPRI_ASN1_H */ ++/* ------------------------------------------------------------------- */ ++/* end asn1.h */ + +Property changes on: asn1.h +___________________________________________________________________ +Added: svn:eol-style + + native +Added: svn:mime-type + + text/plain +Added: svn:keywords + + 'Author Date Id Revision' + +Index: rose.c +=================================================================== +--- a/rose.c (.../tags/1.4.10.2) (revision 0) ++++ b/rose.c (.../branches/1.4) (revision 1357) +@@ -0,0 +1,2464 @@ ++/* ++ * libpri: An implementation of Primary Rate ISDN ++ * ++ * Copyright (C) 2009 Digium, Inc. ++ * ++ * Richard Mudgett <rmudgett@digium.com> ++ * ++ * See http://www.asterisk.org for more information about ++ * the Asterisk project. Please do not directly contact ++ * any of the maintainers of this project for assistance; ++ * the project provides a web site, mailing lists and IRC ++ * channels for your use. ++ * ++ * This program is free software, distributed under the terms of ++ * the GNU General Public License Version 2 as published by the ++ * Free Software Foundation. See the LICENSE file included with ++ * this program for more details. ++ * ++ * In addition, when this program is distributed with Asterisk in ++ * any form that would qualify as a 'combined work' or as a ++ * 'derivative work' (but not mere aggregation), you can redistribute ++ * and/or modify the combination under the terms of the license ++ * provided with that copy of Asterisk, instead of the license ++ * terms granted here. ++ */ ++ ++/*! ++ * \file ++ * \brief Remote Operations Service Element (ROSE) main controlling functions ++ * ++ * \author Richard Mudgett <rmudgett@digium.com> ++ */ ++ ++ ++#include <stdio.h> ++ ++#include "compat.h" ++#include "libpri.h" ++#include "pri_internal.h" ++#include "rose.h" ++#include "rose_internal.h" ++#include "asn1.h" ++#include "pri_facility.h" ++ ++ ++#define ROSE_TAG_COMPONENT_INVOKE (ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 1) ++#define ROSE_TAG_COMPONENT_RESULT (ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 2) ++#define ROSE_TAG_COMPONENT_ERROR (ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 3) ++#define ROSE_TAG_COMPONENT_REJECT (ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 4) ++ ++/*! \brief Structure to convert a code value to a string */ ++struct rose_code_strings { ++ /*! \brief Code value to convert to a string */ ++ int code; ++ /*! \brief String equivalent of the associated code value */ ++ const char *name; ++}; ++ ++/*! \brief ROSE invoke/result message conversion table entry. */ ++struct rose_convert_msg { ++ /*! \brief library encoded operation-value */ ++ enum rose_operation operation; ++ /*! ++ * \brief OID prefix values to use when encoding/decoding the operation-value OID ++ * \note NULL if operation-value is a localValue. ++ */ ++ const struct asn1_oid *oid_prefix; ++ /*! \brief Last OID value or localValue for the encoded operation-value */ ++ u_int16_t value; ++ ++ /*! ++ * \brief Encode the ROSE invoke operation-value arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ * ++ * \note The function pointer is NULL if there are no arguments to encode. ++ */ ++ unsigned char *(*encode_invoke_args)(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args); ++ /*! ++ * \brief Encode the ROSE result operation-value arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ * ++ * \note The function pointer is NULL if there are no arguments to encode. ++ */ ++ unsigned char *(*encode_result_args)(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_result_args *args); ++ ++ /*! ++ * \brief Decode the ROSE invoke operation-value arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ * ++ * \note The function pointer is NULL if there are no arguments to decode. ++ */ ++ const unsigned char *(*decode_invoke_args)(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args); ++ /*! ++ * \brief Decode the ROSE result operation-value arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ * ++ * \note The function pointer is NULL if there are no arguments to decode. ++ */ ++ const unsigned char *(*decode_result_args)(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, ++ union rose_msg_result_args *args); ++}; ++ ++/*! \brief ROSE error code conversion table entry. */ ++struct rose_convert_error { ++ /*! \brief library encoded error-value */ ++ enum rose_error_code code; ++ /*! ++ * \brief OID prefix values to use when encoding/decoding the error-value OID ++ * \note NULL if error-value is a localValue. ++ */ ++ const struct asn1_oid *oid_prefix; ++ /*! \brief Last OID value or localValue for the encoded error-value */ ++ u_int16_t value; ++ ++ /*! ++ * \brief Encode the ROSE error parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ * ++ * \note The function pointer is NULL if there are no arguments to encode. ++ */ ++ unsigned char *(*encode_error_args)(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_error_args *args); ++ ++ /*! ++ * \brief Decode the ROSE error parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ * ++ * \note The function pointer is NULL if there are no arguments to decode. ++ */ ++ const unsigned char *(*decode_error_args)(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, ++ union rose_msg_error_args *args); ++}; ++ ++ ++/* ------------------------------------------------------------------- */ ++ ++ ++/* ++ * Note the first value in oid.values[] is really the first two ++ * OID subidentifiers. They are compressed using this formula: ++ * First_Value = (First_Subidentifier * 40) + Second_Subidentifier ++ */ ++ ++/*! \brief ETSI Explicit Call Transfer OID prefix. */ ++static const struct asn1_oid rose_etsi_ect = { ++/* *INDENT-OFF* */ ++ /* {ccitt(0) identified-organization(4) etsi(0) 369 operations-and-errors(1)} */ ++ 4, { 4, 0, 369, 1 } ++/* *INDENT-ON* */ ++}; ++ ++/*! \brief ETSI specific invoke/result encode/decode message table */ ++static const struct rose_convert_msg rose_etsi_msgs[] = { ++/* *INDENT-OFF* */ ++/* ++ * operation, oid_prefix, value, ++ * encode_invoke_args, encode_result_args, ++ * decode_invoke_args, decode_result_args ++ */ ++ /* ++ * localValue's from Diversion-Operations ++ * {ccitt identified-organization etsi(0) 207 operations-and-errors(1)} ++ */ ++ { ++ ROSE_ETSI_ActivationDiversion, NULL, 7, ++ rose_enc_etsi_ActivationDiversion_ARG, NULL, ++ rose_dec_etsi_ActivationDiversion_ARG, NULL ++ }, ++ { ++ ROSE_ETSI_DeactivationDiversion, NULL, 8, ++ rose_enc_etsi_DeactivationDiversion_ARG,NULL, ++ rose_dec_etsi_DeactivationDiversion_ARG,NULL ++ }, ++ { ++ ROSE_ETSI_ActivationStatusNotificationDiv, NULL, 9, ++ rose_enc_etsi_ActivationStatusNotificationDiv_ARG,NULL, ++ rose_dec_etsi_ActivationStatusNotificationDiv_ARG,NULL ++ }, ++ { ++ ROSE_ETSI_DeactivationStatusNotificationDiv,NULL, 10, ++ rose_enc_etsi_DeactivationStatusNotificationDiv_ARG,NULL, ++ rose_dec_etsi_DeactivationStatusNotificationDiv_ARG,NULL ++ }, ++ { ++ ROSE_ETSI_InterrogationDiversion, NULL, 11, ++ rose_enc_etsi_InterrogationDiversion_ARG,rose_enc_etsi_InterrogationDiversion_RES, ++ rose_dec_etsi_InterrogationDiversion_ARG,rose_dec_etsi_InterrogationDiversion_RES ++ }, ++ { ++ ROSE_ETSI_DiversionInformation, NULL, 12, ++ rose_enc_etsi_DiversionInformation_ARG, NULL, ++ rose_dec_etsi_DiversionInformation_ARG, NULL ++ }, ++ { ++ ROSE_ETSI_CallDeflection, NULL, 13, ++ rose_enc_etsi_CallDeflection_ARG, NULL, ++ rose_dec_etsi_CallDeflection_ARG, NULL ++ }, ++ { ++ ROSE_ETSI_CallRerouting, NULL, 14, ++ rose_enc_etsi_CallRerouting_ARG, NULL, ++ rose_dec_etsi_CallRerouting_ARG, NULL ++ }, ++ { ++ ROSE_ETSI_DivertingLegInformation2, NULL, 15, ++ rose_enc_etsi_DivertingLegInformation2_ARG,NULL, ++ rose_dec_etsi_DivertingLegInformation2_ARG,NULL ++ }, ++ { ++ ROSE_ETSI_InterrogateServedUserNumbers, NULL, 17, ++ NULL, rose_enc_etsi_InterrogateServedUserNumbers_RES, ++ NULL, rose_dec_etsi_InterrogateServedUserNumbers_RES ++ }, ++ { ++ ROSE_ETSI_DivertingLegInformation1, NULL, 18, ++ rose_enc_etsi_DivertingLegInformation1_ARG,NULL, ++ rose_dec_etsi_DivertingLegInformation1_ARG,NULL ++ }, ++ { ++ ROSE_ETSI_DivertingLegInformation3, NULL, 19, ++ rose_enc_etsi_DivertingLegInformation3_ARG,NULL, ++ rose_dec_etsi_DivertingLegInformation3_ARG,NULL ++ }, ++ ++ /* ++ * localValue's from Advice-of-Charge-Operations ++ * {ccitt identified-organization etsi (0) 182 operations-and-errors (1)} ++ * ++ * Advice-Of-Charge-at-call-Setup(AOCS) ++ * Advice-Of-Charge-During-the-call(AOCD) ++ * Advice-Of-Charge-at-the-End-of-the-call(AOCE) ++ */ ++ { ++ ROSE_ETSI_ChargingRequest, NULL, 30, ++ rose_enc_etsi_ChargingRequest_ARG, rose_enc_etsi_ChargingRequest_RES, ++ rose_dec_etsi_ChargingRequest_ARG, rose_dec_etsi_ChargingRequest_RES ++ }, ++ { ++ ROSE_ETSI_AOCSCurrency, NULL, 31, ++ rose_enc_etsi_AOCSCurrency_ARG, NULL, ++ rose_dec_etsi_AOCSCurrency_ARG, NULL ++ }, ++ { ++ ROSE_ETSI_AOCSSpecialArr, NULL, 32, ++ rose_enc_etsi_AOCSSpecialArr_ARG, NULL, ++ rose_dec_etsi_AOCSSpecialArr_ARG, NULL ++ }, ++ { ++ ROSE_ETSI_AOCDCurrency, NULL, 33, ++ rose_enc_etsi_AOCDCurrency_ARG, NULL, ++ rose_dec_etsi_AOCDCurrency_ARG, NULL ++ }, ++ { ++ ROSE_ETSI_AOCDChargingUnit, NULL, 34, ++ rose_enc_etsi_AOCDChargingUnit_ARG, NULL, ++ rose_dec_etsi_AOCDChargingUnit_ARG, NULL ++ }, ++ { ++ ROSE_ETSI_AOCECurrency, NULL, 35, ++ rose_enc_etsi_AOCECurrency_ARG, NULL, ++ rose_dec_etsi_AOCECurrency_ARG, NULL ++ }, ++ { ++ ROSE_ETSI_AOCEChargingUnit, NULL, 36, ++ rose_enc_etsi_AOCEChargingUnit_ARG, NULL, ++ rose_dec_etsi_AOCEChargingUnit_ARG, NULL ++ }, ++ ++ /* ++ * localValue's from Explicit-Call-Transfer-Operations-and-Errors ++ * {ccitt identified-organization etsi(0) 369 operations-and-errors(1)} ++ */ ++ { ++ ROSE_ETSI_EctExecute, NULL, 6, ++ NULL, NULL, ++ NULL, NULL ++ }, ++ ++ /* ++ * globalValue's (OIDs) from Explicit-Call-Transfer-Operations-and-Errors ++ * {ccitt identified-organization etsi(0) 369 operations-and-errors(1)} ++ */ ++ { ++ ROSE_ETSI_ExplicitEctExecute, &rose_etsi_ect, 1, ++ rose_enc_etsi_ExplicitEctExecute_ARG, NULL, ++ rose_dec_etsi_ExplicitEctExecute_ARG, NULL ++ }, ++ { ++ ROSE_ETSI_RequestSubaddress, &rose_etsi_ect, 2, ++ NULL, NULL, ++ NULL, NULL ++ }, ++ { ++ ROSE_ETSI_SubaddressTransfer, &rose_etsi_ect, 3, ++ rose_enc_etsi_SubaddressTransfer_ARG, NULL, ++ rose_dec_etsi_SubaddressTransfer_ARG, NULL ++ }, ++ { ++ ROSE_ETSI_EctLinkIdRequest, &rose_etsi_ect, 4, ++ NULL, rose_enc_etsi_EctLinkIdRequest_RES, ++ NULL, rose_dec_etsi_EctLinkIdRequest_RES ++ }, ++ { ++ ROSE_ETSI_EctInform, &rose_etsi_ect, 5, ++ rose_enc_etsi_EctInform_ARG, NULL, ++ rose_dec_etsi_EctInform_ARG, NULL ++ }, ++ { ++ ROSE_ETSI_EctLoopTest, &rose_etsi_ect, 6, ++ rose_enc_etsi_EctLoopTest_ARG, rose_enc_etsi_EctLoopTest_RES, ++ rose_dec_etsi_EctLoopTest_ARG, rose_dec_etsi_EctLoopTest_RES ++ }, ++/* *INDENT-ON* */ ++}; ++ ++ ++/*! \brief ETSI specific error-value converion table */ ++static const struct rose_convert_error rose_etsi_errors[] = { ++/* *INDENT-OFF* */ ++/* ++ * error-code, oid_prefix, value ++ * encode_error_args, decode_error_args ++ */ ++ /* ++ * localValue Errors from General-Errors ++ * {ccitt identified-organization etsi(0) 196 general-errors(2)} ++ */ ++ { ++ ROSE_ERROR_Gen_NotSubscribed, NULL, 0, ++ NULL, NULL ++ }, ++ { ++ ROSE_ERROR_Gen_NotAvailable, NULL, 3, ++ NULL, NULL ++ }, ++ { ++ ROSE_ERROR_Gen_NotImplemented, NULL, 4, ++ NULL, NULL ++ }, ++ { ++ ROSE_ERROR_Gen_InvalidServedUserNr, NULL, 6, ++ NULL, NULL ++ }, ++ { ++ ROSE_ERROR_Gen_InvalidCallState, NULL, 7, ++ NULL, NULL ++ }, ++ { ++ ROSE_ERROR_Gen_BasicServiceNotProvided, NULL, 8, ++ NULL, NULL ++ }, ++ { ++ ROSE_ERROR_Gen_NotIncomingCall, NULL, 9, ++ NULL, NULL ++ }, ++ { ++ ROSE_ERROR_Gen_SupplementaryServiceInteractionNotAllowed,NULL, 10, ++ NULL, NULL ++ }, ++ { ++ ROSE_ERROR_Gen_ResourceUnavailable, NULL, 11, ++ NULL, NULL ++ }, ++ ++ /* ++ * localValue Errors from Diversion-Operations ++ * {ccitt identified-organization etsi(0) 207 operations-and-errors(1)} ++ */ ++ { ++ ROSE_ERROR_Div_InvalidDivertedToNr, NULL, 12, ++ NULL, NULL ++ }, ++ { ++ ROSE_ERROR_Div_SpecialServiceNr, NULL, 14, ++ NULL, NULL ++ }, ++ { ++ ROSE_ERROR_Div_DiversionToServedUserNr, NULL, 15, ++ NULL, NULL ++ }, ++ { ++ ROSE_ERROR_Div_IncomingCallAccepted, NULL, 23, ++ NULL, NULL ++ }, ++ { ++ ROSE_ERROR_Div_NumberOfDiversionsExceeded, NULL, 24, ++ NULL, NULL ++ }, ++ { ++ ROSE_ERROR_Div_NotActivated, NULL, 46, ++ NULL, NULL ++ }, ++ { ++ ROSE_ERROR_Div_RequestAlreadyAccepted, NULL, 48, ++ NULL, NULL ++ }, ++ ++ /* ++ * localValue Errors from Advice-of-Charge-Operations ++ * {ccitt identified-organization etsi (0) 182 operations-and-errors (1)} ++ */ ++ { ++ ROSE_ERROR_AOC_NoChargingInfoAvailable, NULL, 26, ++ NULL, NULL ++ }, ++ ++ /* ++ * globalValue Errors (OIDs) from Explicit-Call-Transfer-Operations-and-Errors ++ * {ccitt identified-organization etsi(0) 369 operations-and-errors(1)} ++ */ ++ { ++ ROSE_ERROR_ECT_LinkIdNotAssignedByNetwork, &rose_etsi_ect, 21, ++ NULL, NULL ++ }, ++/* *INDENT-ON* */ ++}; ++ ++ ++/* ------------------------------------------------------------------- */ ++ ++ ++/*! \brief Q.SIG specific invoke/result encode/decode message table */ ++static const struct rose_convert_msg rose_qsig_msgs[] = { ++/* *INDENT-OFF* */ ++/* ++ * operation, oid_prefix, value, ++ * encode_invoke_args, encode_result_args, ++ * decode_invoke_args, decode_result_args ++ */ ++ /* ++ * localValue's from Q.SIG Name-Operations ++ * { iso(1) standard(0) pss1-name(13868) name-operations(0) } ++ */ ++ { ++ ROSE_QSIG_CallingName, NULL, 0, ++ rose_enc_qsig_CallingName_ARG, NULL, ++ rose_dec_qsig_CallingName_ARG, NULL ++ }, ++ { ++ ROSE_QSIG_CalledName, NULL, 1, ++ rose_enc_qsig_CalledName_ARG, NULL, ++ rose_dec_qsig_CalledName_ARG, NULL ++ }, ++ { ++ ROSE_QSIG_ConnectedName, NULL, 2, ++ rose_enc_qsig_ConnectedName_ARG, NULL, ++ rose_dec_qsig_ConnectedName_ARG, NULL ++ }, ++ { ++ ROSE_QSIG_BusyName, NULL, 3, ++ rose_enc_qsig_BusyName_ARG, NULL, ++ rose_dec_qsig_BusyName_ARG, NULL ++ }, ++ ++ /* ++ * localValue's from Q.SIG SS-AOC-Operations ++ * { iso(1) standard(0) pss1-advice-of-charge(15050) advice-of-charge-operations(0) } ++ */ ++ { ++ ROSE_QSIG_ChargeRequest, NULL, 59, ++ rose_enc_qsig_ChargeRequest_ARG, rose_enc_qsig_ChargeRequest_RES, ++ rose_dec_qsig_ChargeRequest_ARG, rose_dec_qsig_ChargeRequest_RES ++ }, ++ { ++ ROSE_QSIG_GetFinalCharge, NULL, 60, ++ rose_enc_qsig_DummyArg_ARG, NULL, ++ rose_dec_qsig_DummyArg_ARG, NULL ++ }, ++ { ++ ROSE_QSIG_AocFinal, NULL, 61, ++ rose_enc_qsig_AocFinal_ARG, NULL, ++ rose_dec_qsig_AocFinal_ARG, NULL ++ }, ++ { ++ ROSE_QSIG_AocInterim, NULL, 62, ++ rose_enc_qsig_AocInterim_ARG, NULL, ++ rose_dec_qsig_AocInterim_ARG, NULL ++ }, ++ { ++ ROSE_QSIG_AocRate, NULL, 63, ++ rose_enc_qsig_AocRate_ARG, NULL, ++ rose_dec_qsig_AocRate_ARG, NULL ++ }, ++ { ++ ROSE_QSIG_AocComplete, NULL, 64, ++ rose_enc_qsig_AocComplete_ARG, rose_enc_qsig_AocComplete_RES, ++ rose_dec_qsig_AocComplete_ARG, rose_dec_qsig_AocComplete_RES ++ }, ++ { ++ ROSE_QSIG_AocDivChargeReq, NULL, 65, ++ rose_enc_qsig_AocDivChargeReq_ARG, NULL, ++ rose_dec_qsig_AocDivChargeReq_ARG, NULL ++ }, ++ ++ /* ++ * localValue's from Q.SIG Call-Transfer-Operations ++ * { iso(1) standard(0) pss1-call-transfer(13869) call-transfer-operations(0) } ++ */ ++ { ++ ROSE_QSIG_CallTransferIdentify, NULL, 7, ++ rose_enc_qsig_DummyArg_ARG, rose_enc_qsig_CallTransferIdentify_RES, ++ rose_dec_qsig_DummyArg_ARG, rose_dec_qsig_CallTransferIdentify_RES ++ }, ++ { ++ ROSE_QSIG_CallTransferAbandon, NULL, 8, ++ rose_enc_qsig_DummyArg_ARG, NULL, ++ rose_dec_qsig_DummyArg_ARG, NULL ++ }, ++ { ++ ROSE_QSIG_CallTransferInitiate, NULL, 9, ++ rose_enc_qsig_CallTransferInitiate_ARG, rose_enc_qsig_DummyRes_RES, ++ rose_dec_qsig_CallTransferInitiate_ARG, rose_dec_qsig_DummyRes_RES ++ }, ++ { ++ ROSE_QSIG_CallTransferSetup, NULL, 10, ++ rose_enc_qsig_CallTransferSetup_ARG, rose_enc_qsig_DummyRes_RES, ++ rose_dec_qsig_CallTransferSetup_ARG, rose_dec_qsig_DummyRes_RES ++ }, ++ { ++ ROSE_QSIG_CallTransferActive, NULL, 11, ++ rose_enc_qsig_CallTransferActive_ARG, NULL, ++ rose_dec_qsig_CallTransferActive_ARG, NULL ++ }, ++ { ++ ROSE_QSIG_CallTransferComplete, NULL, 12, ++ rose_enc_qsig_CallTransferComplete_ARG, NULL, ++ rose_dec_qsig_CallTransferComplete_ARG, NULL ++ }, ++ { ++ ROSE_QSIG_CallTransferUpdate, NULL, 13, ++ rose_enc_qsig_CallTransferUpdate_ARG, NULL, ++ rose_dec_qsig_CallTransferUpdate_ARG, NULL ++ }, ++ { ++ ROSE_QSIG_SubaddressTransfer, NULL, 14, ++ rose_enc_qsig_SubaddressTransfer_ARG, NULL, ++ rose_dec_qsig_SubaddressTransfer_ARG, NULL ++ }, ++ ++ /* ++ * NOTE: I do not have the specification needed to fully support this ++ * message. Fortunately, all I have to do for this message is to switch ++ * it to the bridged call leg for 2BCT support. ++ */ ++ { ++ ROSE_QSIG_PathReplacement, NULL, 4, ++ NULL, NULL, ++ NULL, NULL ++ }, ++ ++ /* ++ * localValue's from Q.SIG Call-Diversion-Operations ++ * { iso(1) standard(0) pss1-call-diversion(13873) call-diversion-operations(0) } ++ */ ++ { ++ ROSE_QSIG_ActivateDiversionQ, NULL, 15, ++ rose_enc_qsig_ActivateDiversionQ_ARG, rose_enc_qsig_DummyRes_RES, ++ rose_dec_qsig_ActivateDiversionQ_ARG, rose_dec_qsig_DummyRes_RES ++ }, ++ { ++ ROSE_QSIG_DeactivateDiversionQ, NULL, 16, ++ rose_enc_qsig_DeactivateDiversionQ_ARG, rose_enc_qsig_DummyRes_RES, ++ rose_dec_qsig_DeactivateDiversionQ_ARG, rose_dec_qsig_DummyRes_RES ++ }, ++ { ++ ROSE_QSIG_InterrogateDiversionQ, NULL, 17, ++ rose_enc_qsig_InterrogateDiversionQ_ARG,rose_enc_qsig_InterrogateDiversionQ_RES, ++ rose_dec_qsig_InterrogateDiversionQ_ARG,rose_dec_qsig_InterrogateDiversionQ_RES ++ }, ++ { ++ ROSE_QSIG_CheckRestriction, NULL, 18, ++ rose_enc_qsig_CheckRestriction_ARG, rose_enc_qsig_DummyRes_RES, ++ rose_dec_qsig_CheckRestriction_ARG, rose_dec_qsig_DummyRes_RES ++ }, ++ { ++ ROSE_QSIG_CallRerouting, NULL, 19, ++ rose_enc_qsig_CallRerouting_ARG, rose_enc_qsig_DummyRes_RES, ++ rose_dec_qsig_CallRerouting_ARG, rose_dec_qsig_DummyRes_RES ++ }, ++ { ++ ROSE_QSIG_DivertingLegInformation1, NULL, 20, ++ rose_enc_qsig_DivertingLegInformation1_ARG,NULL, ++ rose_dec_qsig_DivertingLegInformation1_ARG,NULL ++ }, ++ { ++ ROSE_QSIG_DivertingLegInformation2, NULL, 21, ++ rose_enc_qsig_DivertingLegInformation2_ARG,NULL, ++ rose_dec_qsig_DivertingLegInformation2_ARG,NULL ++ }, ++ { ++ ROSE_QSIG_DivertingLegInformation3, NULL, 22, ++ rose_enc_qsig_DivertingLegInformation3_ARG,NULL, ++ rose_dec_qsig_DivertingLegInformation3_ARG,NULL ++ }, ++ { ++ ROSE_QSIG_CfnrDivertedLegFailed, NULL, 23, ++ rose_enc_qsig_DummyArg_ARG, NULL, ++ rose_dec_qsig_DummyArg_ARG, NULL ++ }, ++ ++ /* ++ * localValue's from Q.SIG SS-MWI-Operations ++ * { iso(1) standard(0) pss1-message-waiting-indication(15506) message-waiting-operations(0) } ++ */ ++ { ++ ROSE_QSIG_MWIActivate, NULL, 80, ++ rose_enc_qsig_MWIActivate_ARG, rose_enc_qsig_DummyRes_RES, ++ rose_dec_qsig_MWIActivate_ARG, rose_dec_qsig_DummyRes_RES ++ }, ++ { ++ ROSE_QSIG_MWIDeactivate, NULL, 81, ++ rose_enc_qsig_MWIDeactivate_ARG, rose_enc_qsig_DummyRes_RES, ++ rose_dec_qsig_MWIDeactivate_ARG, rose_dec_qsig_DummyRes_RES ++ }, ++ { ++ ROSE_QSIG_MWIInterrogate, NULL, 82, ++ rose_enc_qsig_MWIInterrogate_ARG, rose_enc_qsig_MWIInterrogate_RES, ++ rose_dec_qsig_MWIInterrogate_ARG, rose_dec_qsig_MWIInterrogate_RES ++ }, ++/* *INDENT-ON* */ ++}; ++ ++ ++/*! \brief Q.SIG specific error-value converion table */ ++static const struct rose_convert_error rose_qsig_errors[] = { ++/* *INDENT-OFF* */ ++/* ++ * error-code, oid_prefix, value ++ * encode_error_args, decode_error_args ++ */ ++ /* ++ * localValue Errors from General-Error-List ++ * {ccitt identified-organization q 950 general-error-list(1)} ++ */ ++ { ++ ROSE_ERROR_Gen_NotSubscribed, NULL, 0, ++ NULL, NULL ++ }, ++ { ++ ROSE_ERROR_Gen_RejectedByNetwork, NULL, 1, ++ NULL, NULL ++ }, ++ { ++ ROSE_ERROR_Gen_RejectedByUser, NULL, 2, ++ NULL, NULL ++ }, ++ { ++ ROSE_ERROR_Gen_NotAvailable, NULL, 3, ++ NULL, NULL ++ }, ++ { ++ ROSE_ERROR_Gen_InsufficientInformation, NULL, 5, ++ NULL, NULL ++ }, ++ { ++ ROSE_ERROR_Gen_InvalidServedUserNr, NULL, 6, ++ NULL, NULL ++ }, ++ { ++ ROSE_ERROR_Gen_InvalidCallState, NULL, 7, ++ NULL, NULL ++ }, ++ { ++ ROSE_ERROR_Gen_BasicServiceNotProvided, NULL, 8, ++ NULL, NULL ++ }, ++ { ++ ROSE_ERROR_Gen_NotIncomingCall, NULL, 9, ++ NULL, NULL ++ }, ++ { ++ ROSE_ERROR_Gen_SupplementaryServiceInteractionNotAllowed,NULL, 10, ++ NULL, NULL ++ }, ++ { ++ ROSE_ERROR_Gen_ResourceUnavailable, NULL, 11, ++ NULL, NULL ++ }, ++ { ++ ROSE_ERROR_Gen_CallFailure, NULL, 25, ++ NULL, NULL ++ }, ++ { ++ ROSE_ERROR_Gen_ProceduralError, NULL, 43, ++ NULL, NULL ++ }, ++ ++ /* ++ * From various Q.SIG specifications. ++ * We will ignore the manufacturer specific extension information. ++ */ ++ { ++ ROSE_ERROR_QSIG_Unspecified, NULL, 1008, ++ NULL, NULL ++ }, ++ ++ /* ++ * localValue Errors from Q.SIG SS-AOC-Operations ++ * { iso(1) standard(0) pss1-advice-of-charge(15050) advice-of-charge-operations(0) } ++ */ ++ { ++ ROSE_ERROR_QSIG_AOC_FreeOfCharge, NULL, 1016, ++ NULL, NULL ++ }, ++ ++ /* ++ * localValue's from Q.SIG Call-Transfer-Operations ++ * { iso(1) standard(0) pss1-call-transfer(13869) call-transfer-operations(0) } ++ */ ++ { ++ ROSE_ERROR_QSIG_CT_InvalidReroutingNumber, NULL, 1004, ++ NULL, NULL ++ }, ++ { ++ ROSE_ERROR_QSIG_CT_UnrecognizedCallIdentity,NULL, 1005, ++ NULL, NULL ++ }, ++ { ++ ROSE_ERROR_QSIG_CT_EstablishmentFailure, NULL, 1006, ++ NULL, NULL ++ }, ++ ++ /* ++ * localValue's from Q.SIG Call-Diversion-Operations ++ * { iso(1) standard(0) pss1-call-diversion(13873) call-diversion-operations(0) } ++ */ ++ { ++ ROSE_ERROR_Div_InvalidDivertedToNr, NULL, 12, ++ NULL, NULL ++ }, ++ { ++ ROSE_ERROR_Div_SpecialServiceNr, NULL, 14, ++ NULL, NULL ++ }, ++ { ++ ROSE_ERROR_Div_DiversionToServedUserNr, NULL, 15, ++ NULL, NULL ++ }, ++ { ++ ROSE_ERROR_Div_NumberOfDiversionsExceeded, NULL, 24, ++ NULL, NULL ++ }, ++ { ++ ROSE_ERROR_QSIG_Div_TemporarilyUnavailable, NULL, 1000, ++ NULL, NULL ++ }, ++ { ++ ROSE_ERROR_QSIG_Div_NotAuthorized, NULL, 1007, ++ NULL, NULL ++ }, ++ ++ /* ++ * localValue's from Q.SIG SS-MWI-Operations ++ * { iso(1) standard(0) pss1-message-waiting-indication(15506) message-waiting-operations(0) } ++ */ ++ { ++ ROSE_ERROR_QSIG_InvalidMsgCentreId, NULL, 1018, ++ NULL, NULL ++ }, ++/* *INDENT-ON* */ ++}; ++ ++ ++/* ------------------------------------------------------------------- */ ++ ++ ++/*! \brief DMS-100 specific invoke/result encode/decode message table */ ++static const struct rose_convert_msg rose_dms100_msgs[] = { ++/* *INDENT-OFF* */ ++/* ++ * operation, oid_prefix, value, ++ * encode_invoke_args, encode_result_args, ++ * decode_invoke_args, decode_result_args ++ */ ++ { ++ ROSE_DMS100_RLT_OperationInd, NULL, ROSE_DMS100_RLT_OPERATION_IND, ++ NULL, rose_enc_dms100_RLT_OperationInd_RES, ++ NULL, rose_dec_dms100_RLT_OperationInd_RES ++ }, ++ { ++ ROSE_DMS100_RLT_ThirdParty, NULL, ROSE_DMS100_RLT_THIRD_PARTY, ++ rose_enc_dms100_RLT_ThirdParty_ARG, NULL, ++ rose_dec_dms100_RLT_ThirdParty_ARG, NULL ++ }, ++/* *INDENT-ON* */ ++}; ++ ++ ++/*! \brief DMS-100 specific error-value converion table */ ++static const struct rose_convert_error rose_dms100_errors[] = { ++/* *INDENT-OFF* */ ++/* ++ * error-code, oid_prefix, value ++ * encode_error_args, decode_error_args ++ */ ++ { ++ ROSE_ERROR_DMS100_RLT_BridgeFail, NULL, 0x10, ++ NULL, NULL ++ }, ++ { ++ ROSE_ERROR_DMS100_RLT_CallIDNotFound, NULL, 0x11, ++ NULL, NULL ++ }, ++ { ++ ROSE_ERROR_DMS100_RLT_NotAllowed, NULL, 0x12, ++ NULL, NULL ++ }, ++ { ++ ROSE_ERROR_DMS100_RLT_SwitchEquipCongs, NULL, 0x13, ++ NULL, NULL ++ }, ++/* *INDENT-ON* */ ++}; ++ ++ ++/* ------------------------------------------------------------------- */ ++ ++ ++/* ++ * Note the first value in oid.values[] is really the first two ++ * OID subidentifiers. They are compressed using this formula: ++ * First_Value = (First_Subidentifier * 40) + Second_Subidentifier ++ */ ++ ++static const struct asn1_oid rose_ni2_oid = { ++/* *INDENT-OFF* */ ++ /* { iso(1) member-body(2) usa(840) ansi-t1(10005) operations(0) } */ ++ 4, { 42, 840, 10005, 0 } ++/* *INDENT-ON* */ ++}; ++ ++/*! \brief NI2 specific invoke/result encode/decode message table */ ++static const struct rose_convert_msg rose_ni2_msgs[] = { ++/* *INDENT-OFF* */ ++/* ++ * operation, oid_prefix, value, ++ * encode_invoke_args, encode_result_args, ++ * decode_invoke_args, decode_result_args ++ */ ++ /* NI2 seems to have pirated several Q.SIG messages */ ++ /* ++ * localValue's from Q.SIG Name-Operations ++ * { iso(1) standard(0) pss1-name(13868) name-operations(0) } ++ */ ++ { ++ ROSE_QSIG_CallingName, NULL, 0, ++ rose_enc_qsig_CallingName_ARG, NULL, ++ rose_dec_qsig_CallingName_ARG, NULL ++ }, ++ { ++ ROSE_QSIG_CalledName, NULL, 1, ++ rose_enc_qsig_CalledName_ARG, NULL, ++ rose_dec_qsig_CalledName_ARG, NULL ++ }, ++ { ++ ROSE_QSIG_ConnectedName, NULL, 2, ++ rose_enc_qsig_ConnectedName_ARG, NULL, ++ rose_dec_qsig_ConnectedName_ARG, NULL ++ }, ++ { ++ ROSE_QSIG_BusyName, NULL, 3, ++ rose_enc_qsig_BusyName_ARG, NULL, ++ rose_dec_qsig_BusyName_ARG, NULL ++ }, ++ ++ { ++ ROSE_NI2_InformationFollowing, &rose_ni2_oid, 4, ++ rose_enc_ni2_InformationFollowing_ARG, NULL, ++ rose_dec_ni2_InformationFollowing_ARG, NULL ++ }, ++ ++ /* Also used by PRI_SWITCH_ATT4ESS and PRI_SWITCH_LUCENT5E */ ++ { ++ ROSE_NI2_InitiateTransfer, &rose_ni2_oid, 8, ++ rose_enc_ni2_InitiateTransfer_ARG, NULL, ++ rose_dec_ni2_InitiateTransfer_ARG, NULL ++ }, ++/* *INDENT-ON* */ ++}; ++ ++ ++/*! \brief NI2 specific error-value converion table */ ++static const struct rose_convert_error rose_ni2_errors[] = { ++/* *INDENT-OFF* */ ++/* ++ * error-code, oid_prefix, value ++ * encode_error_args, decode_error_args ++ */ ++ /* ++ * localValue Errors from General-Error-List ++ * {ccitt identified-organization q 950 general-error-list(1)} ++ */ ++ { ++ ROSE_ERROR_Gen_NotSubscribed, NULL, 0, ++ NULL, NULL ++ }, ++ { ++ ROSE_ERROR_Gen_RejectedByNetwork, NULL, 1, ++ NULL, NULL ++ }, ++ { ++ ROSE_ERROR_Gen_RejectedByUser, NULL, 2, ++ NULL, NULL ++ }, ++ { ++ ROSE_ERROR_Gen_NotAvailable, NULL, 3, ++ NULL, NULL ++ }, ++ { ++ ROSE_ERROR_Gen_InsufficientInformation, NULL, 5, ++ NULL, NULL ++ }, ++ { ++ ROSE_ERROR_Gen_InvalidServedUserNr, NULL, 6, ++ NULL, NULL ++ }, ++ { ++ ROSE_ERROR_Gen_InvalidCallState, NULL, 7, ++ NULL, NULL ++ }, ++ { ++ ROSE_ERROR_Gen_BasicServiceNotProvided, NULL, 8, ++ NULL, NULL ++ }, ++ { ++ ROSE_ERROR_Gen_NotIncomingCall, NULL, 9, ++ NULL, NULL ++ }, ++ { ++ ROSE_ERROR_Gen_SupplementaryServiceInteractionNotAllowed,NULL, 10, ++ NULL, NULL ++ }, ++ { ++ ROSE_ERROR_Gen_ResourceUnavailable, NULL, 11, ++ NULL, NULL ++ }, ++ { ++ ROSE_ERROR_Gen_CallFailure, NULL, 25, ++ NULL, NULL ++ }, ++ { ++ ROSE_ERROR_Gen_ProceduralError, NULL, 43, ++ NULL, NULL ++ }, ++/* *INDENT-ON* */ ++}; ++ ++ ++/* ------------------------------------------------------------------- */ ++ ++/*! ++ * \internal ++ * \brief Convert the given code value to a string. ++ * ++ * \param code Code value to convert to a string. ++ * \param arr Array to convert the code to a string. ++ * \param num_elements Number of elements in the conversion array. ++ * ++ * \retval String version of the given code value. ++ */ ++static const char *rose_code2str(int code, const struct rose_code_strings *arr, ++ unsigned num_elements) ++{ ++ static char invalid_code[40]; ++ ++ unsigned index; ++ ++ for (index = 0; index < num_elements; ++index) { ++ if (arr[index].code == code) { ++ return arr[index].name; ++ } ++ } ++ ++ snprintf(invalid_code, sizeof(invalid_code), "Invalid code:%d 0x%X", code, code); ++ return invalid_code; ++} ++ ++/*! ++ * \brief Convert the given operation-value to a string. ++ * ++ * \param operation Operation-value to convert to a string. ++ * ++ * \retval String version of the given operation-value. ++ */ ++const char *rose_operation2str(enum rose_operation operation) ++{ ++ static const struct rose_code_strings arr[] = { ++/* *INDENT-OFF* */ ++ { ROSE_None, "ROSE_None" }, ++ { ROSE_Unknown, "ROSE_Unknown" }, ++ ++ { ROSE_ETSI_ActivationDiversion, "ROSE_ETSI_ActivationDiversion" }, ++ { ROSE_ETSI_DeactivationDiversion, "ROSE_ETSI_DeactivationDiversion" }, ++ { ROSE_ETSI_ActivationStatusNotificationDiv,"ROSE_ETSI_ActivationStatusNotificationDiv" }, ++ { ROSE_ETSI_DeactivationStatusNotificationDiv,"ROSE_ETSI_DeactivationStatusNotificationDiv" }, ++ { ROSE_ETSI_InterrogationDiversion, "ROSE_ETSI_InterrogationDiversion" }, ++ { ROSE_ETSI_DiversionInformation, "ROSE_ETSI_DiversionInformation" }, ++ { ROSE_ETSI_CallDeflection, "ROSE_ETSI_CallDeflection" }, ++ { ROSE_ETSI_CallRerouting, "ROSE_ETSI_CallRerouting" }, ++ { ROSE_ETSI_DivertingLegInformation2, "ROSE_ETSI_DivertingLegInformation2" }, ++ { ROSE_ETSI_InterrogateServedUserNumbers, "ROSE_ETSI_InterrogateServedUserNumbers" }, ++ { ROSE_ETSI_DivertingLegInformation1, "ROSE_ETSI_DivertingLegInformation1" }, ++ { ROSE_ETSI_DivertingLegInformation3, "ROSE_ETSI_DivertingLegInformation3" }, ++ ++ { ROSE_ETSI_EctExecute, "ROSE_ETSI_EctExecute" }, ++ { ROSE_ETSI_ExplicitEctExecute, "ROSE_ETSI_ExplicitEctExecute" }, ++ { ROSE_ETSI_RequestSubaddress, "ROSE_ETSI_RequestSubaddress" }, ++ { ROSE_ETSI_SubaddressTransfer, "ROSE_ETSI_SubaddressTransfer" }, ++ { ROSE_ETSI_EctLinkIdRequest, "ROSE_ETSI_EctLinkIdRequest" }, ++ { ROSE_ETSI_EctInform, "ROSE_ETSI_EctInform" }, ++ { ROSE_ETSI_EctLoopTest, "ROSE_ETSI_EctLoopTest" }, ++ ++ { ROSE_ETSI_ChargingRequest, "ROSE_ETSI_ChargingRequest" }, ++ { ROSE_ETSI_AOCSCurrency, "ROSE_ETSI_AOCSCurrency" }, ++ { ROSE_ETSI_AOCSSpecialArr, "ROSE_ETSI_AOCSSpecialArr" }, ++ { ROSE_ETSI_AOCDCurrency, "ROSE_ETSI_AOCDCurrency" }, ++ { ROSE_ETSI_AOCDChargingUnit, "ROSE_ETSI_AOCDChargingUnit" }, ++ { ROSE_ETSI_AOCECurrency, "ROSE_ETSI_AOCECurrency" }, ++ { ROSE_ETSI_AOCEChargingUnit, "ROSE_ETSI_AOCEChargingUnit" }, ++ ++ { ROSE_QSIG_CallingName, "ROSE_QSIG_CallingName" }, ++ { ROSE_QSIG_CalledName, "ROSE_QSIG_CalledName" }, ++ { ROSE_QSIG_ConnectedName, "ROSE_QSIG_ConnectedName" }, ++ { ROSE_QSIG_BusyName, "ROSE_QSIG_BusyName" }, ++ ++ { ROSE_QSIG_ChargeRequest, "ROSE_QSIG_ChargeRequest" }, ++ { ROSE_QSIG_GetFinalCharge, "ROSE_QSIG_GetFinalCharge" }, ++ { ROSE_QSIG_AocFinal, "ROSE_QSIG_AocFinal" }, ++ { ROSE_QSIG_AocInterim, "ROSE_QSIG_AocInterim" }, ++ { ROSE_QSIG_AocRate, "ROSE_QSIG_AocRate" }, ++ { ROSE_QSIG_AocComplete, "ROSE_QSIG_AocComplete" }, ++ { ROSE_QSIG_AocDivChargeReq, "ROSE_QSIG_AocDivChargeReq" }, ++ ++ { ROSE_QSIG_CallTransferIdentify, "ROSE_QSIG_CallTransferIdentify" }, ++ { ROSE_QSIG_CallTransferAbandon, "ROSE_QSIG_CallTransferAbandon" }, ++ { ROSE_QSIG_CallTransferInitiate, "ROSE_QSIG_CallTransferInitiate" }, ++ { ROSE_QSIG_CallTransferSetup, "ROSE_QSIG_CallTransferSetup" }, ++ { ROSE_QSIG_CallTransferActive, "ROSE_QSIG_CallTransferActive" }, ++ { ROSE_QSIG_CallTransferComplete, "ROSE_QSIG_CallTransferComplete" }, ++ { ROSE_QSIG_CallTransferUpdate, "ROSE_QSIG_CallTransferUpdate" }, ++ { ROSE_QSIG_SubaddressTransfer, "ROSE_QSIG_SubaddressTransfer" }, ++ ++ { ROSE_QSIG_PathReplacement, "ROSE_QSIG_PathReplacement" }, ++ ++ { ROSE_QSIG_ActivateDiversionQ, "ROSE_QSIG_ActivateDiversionQ" }, ++ { ROSE_QSIG_DeactivateDiversionQ, "ROSE_QSIG_DeactivateDiversionQ" }, ++ { ROSE_QSIG_InterrogateDiversionQ, "ROSE_QSIG_InterrogateDiversionQ" }, ++ { ROSE_QSIG_CheckRestriction, "ROSE_QSIG_CheckRestriction" }, ++ { ROSE_QSIG_CallRerouting, "ROSE_QSIG_CallRerouting" }, ++ { ROSE_QSIG_DivertingLegInformation1, "ROSE_QSIG_DivertingLegInformation1" }, ++ { ROSE_QSIG_DivertingLegInformation2, "ROSE_QSIG_DivertingLegInformation2" }, ++ { ROSE_QSIG_DivertingLegInformation3, "ROSE_QSIG_DivertingLegInformation3" }, ++ { ROSE_QSIG_CfnrDivertedLegFailed, "ROSE_QSIG_CfnrDivertedLegFailed" }, ++ ++ { ROSE_QSIG_MWIActivate, "ROSE_QSIG_MWIActivate" }, ++ { ROSE_QSIG_MWIDeactivate, "ROSE_QSIG_MWIDeactivate" }, ++ { ROSE_QSIG_MWIInterrogate, "ROSE_QSIG_MWIInterrogate" }, ++ ++ { ROSE_DMS100_RLT_OperationInd, "ROSE_DMS100_RLT_OperationInd" }, ++ { ROSE_DMS100_RLT_ThirdParty, "ROSE_DMS100_RLT_ThirdParty" }, ++ ++ { ROSE_NI2_InformationFollowing, "ROSE_NI2_InformationFollowing" }, ++ { ROSE_NI2_InitiateTransfer, "ROSE_NI2_InitiateTransfer" }, ++/* *INDENT-ON* */ ++ }; ++ ++ return rose_code2str(operation, arr, ARRAY_LEN(arr)); ++} ++ ++/*! ++ * \brief Convert the given error-value to a string. ++ * ++ * \param code Error-value to convert to a string. ++ * ++ * \retval String version of the given error-value. ++ */ ++const char *rose_error2str(enum rose_error_code code) ++{ ++ static const struct rose_code_strings arr[] = { ++/* *INDENT-OFF* */ ++ { ROSE_ERROR_None, "No error occurred" }, ++ { ROSE_ERROR_Unknown, "Unknown error-value code" }, ++ ++ { ROSE_ERROR_Gen_NotSubscribed, "General: Not Subscribed" }, ++ { ROSE_ERROR_Gen_NotAvailable, "General: Not Available" }, ++ { ROSE_ERROR_Gen_NotImplemented, "General: Not Implemented" }, ++ { ROSE_ERROR_Gen_InvalidServedUserNr, "General: Invalid Served User Number" }, ++ { ROSE_ERROR_Gen_InvalidCallState, "General: Invalid Call State" }, ++ { ROSE_ERROR_Gen_BasicServiceNotProvided, "General: Basic Service Not Provided" }, ++ { ROSE_ERROR_Gen_NotIncomingCall, "General: Not Incoming Call" }, ++ { ROSE_ERROR_Gen_SupplementaryServiceInteractionNotAllowed,"General: Supplementary Service Interaction Not Allowed" }, ++ { ROSE_ERROR_Gen_ResourceUnavailable, "General: Resource Unavailable" }, ++ ++ /* Additional Q.950 General-Errors for Q.SIG */ ++ { ROSE_ERROR_Gen_RejectedByNetwork, "General: Rejected By Network" }, ++ { ROSE_ERROR_Gen_RejectedByUser, "General: Rejected By User" }, ++ { ROSE_ERROR_Gen_InsufficientInformation, "General: Insufficient Information" }, ++ { ROSE_ERROR_Gen_CallFailure, "General: Call Failure" }, ++ { ROSE_ERROR_Gen_ProceduralError, "General: Procedural Error" }, ++ ++ { ROSE_ERROR_Div_InvalidDivertedToNr, "Diversion: Invalid Diverted To Number" }, ++ { ROSE_ERROR_Div_SpecialServiceNr, "Diversion: Special Service Number" }, ++ { ROSE_ERROR_Div_DiversionToServedUserNr, "Diversion: Diversion To Served User Number" }, ++ { ROSE_ERROR_Div_IncomingCallAccepted, "Diversion: Incoming Call Accepted" }, ++ { ROSE_ERROR_Div_NumberOfDiversionsExceeded, "Diversion: Number Of Diversions Exceeded" }, ++ { ROSE_ERROR_Div_NotActivated, "Diversion: Not Activated" }, ++ { ROSE_ERROR_Div_RequestAlreadyAccepted, "Diversion: Request Already Accepted" }, ++ ++ { ROSE_ERROR_AOC_NoChargingInfoAvailable, "AOC: No Charging Info Available" }, ++ ++ { ROSE_ERROR_ECT_LinkIdNotAssignedByNetwork, "ECT: Link ID Not Assigned By Network" }, ++ ++ /* Q.SIG specific errors */ ++ { ROSE_ERROR_QSIG_Unspecified, "Unspecified" }, ++ ++ { ROSE_ERROR_QSIG_AOC_FreeOfCharge, "AOC: FreeOfCharge" }, ++ ++ { ROSE_ERROR_QSIG_CT_InvalidReroutingNumber, "CT: Invalid Rerouting Number" }, ++ { ROSE_ERROR_QSIG_CT_UnrecognizedCallIdentity,"CT: Unrecognized Call Identity" }, ++ { ROSE_ERROR_QSIG_CT_EstablishmentFailure, "CT: Establishment Failure" }, ++ ++ { ROSE_ERROR_QSIG_Div_TemporarilyUnavailable, "Diversion: Temporarily Unavailable" }, ++ { ROSE_ERROR_QSIG_Div_NotAuthorized, "Diversion: Not Authorized" }, ++ ++ { ROSE_ERROR_QSIG_InvalidMsgCentreId, "MWI: Invalid Message Center ID" }, ++ ++ /* DMS-100 specific errors */ ++ { ROSE_ERROR_DMS100_RLT_BridgeFail, "RLT: Bridge Fail" }, ++ { ROSE_ERROR_DMS100_RLT_CallIDNotFound, "RLT: Call ID Not Found" }, ++ { ROSE_ERROR_DMS100_RLT_NotAllowed, "RLT: Not Allowed" }, ++ { ROSE_ERROR_DMS100_RLT_SwitchEquipCongs, "RLT: Switch Equip Congs" }, ++/* *INDENT-ON* */ ++ }; ++ ++ return rose_code2str(code, arr, ARRAY_LEN(arr)); ++} ++ ++/*! ++ * \brief Convert the given reject problem-value to a string. ++ * ++ * \param code Reject problem-value to convert to a string. ++ * ++ * \retval String version of the given reject problem-value. ++ */ ++const char *rose_reject2str(enum rose_reject_code code) ++{ ++ static const struct rose_code_strings arr[] = { ++/* *INDENT-OFF* */ ++ { ROSE_REJECT_None, "No reject occurred" }, ++ { ROSE_REJECT_Unknown, "Unknown reject code" }, ++ ++ { ROSE_REJECT_Gen_UnrecognizedComponent, "General: Unrecognized Component" }, ++ { ROSE_REJECT_Gen_MistypedComponent, "General: Mistyped Component" }, ++ { ROSE_REJECT_Gen_BadlyStructuredComponent, "General: Badly Structured Component" }, ++ ++ { ROSE_REJECT_Inv_DuplicateInvocation, "Invoke: Duplicate Invocation" }, ++ { ROSE_REJECT_Inv_UnrecognizedOperation, "Invoke: Unrecognized Operation" }, ++ { ROSE_REJECT_Inv_MistypedArgument, "Invoke: Mistyped Argument" }, ++ { ROSE_REJECT_Inv_ResourceLimitation, "Invoke: Resource Limitation" }, ++ { ROSE_REJECT_Inv_InitiatorReleasing, "Invoke: Initiator Releasing" }, ++ { ROSE_REJECT_Inv_UnrecognizedLinkedID, "Invoke: Unrecognized Linked ID" }, ++ { ROSE_REJECT_Inv_LinkedResponseUnexpected, "Invoke: Linked Response Unexpected" }, ++ { ROSE_REJECT_Inv_UnexpectedChildOperation, "Invoke: Unexpected Child Operation" }, ++ ++ { ROSE_REJECT_Res_UnrecognizedInvocation, "Result: Unrecognized Invocation" }, ++ { ROSE_REJECT_Res_ResultResponseUnexpected, "Result: Result Response Unexpected" }, ++ { ROSE_REJECT_Res_MistypedResult, "Result: Mistyped Result" }, ++ ++ { ROSE_REJECT_Err_UnrecognizedInvocation, "Error: Unrecognized Invocation" }, ++ { ROSE_REJECT_Err_ErrorResponseUnexpected, "Error: Error Response Unexpected" }, ++ { ROSE_REJECT_Err_UnrecognizedError, "Error: Unrecognized Error" }, ++ { ROSE_REJECT_Err_UnexpectedError, "Error: Unexpected Error" }, ++ { ROSE_REJECT_Err_MistypedParameter, "Error: Mistyped Parameter" }, ++/* *INDENT-ON* */ ++ }; ++ ++ return rose_code2str(code, arr, ARRAY_LEN(arr)); ++} ++ ++/*! ++ * \internal ++ * \brief Find an operation message conversion entry using the operation code. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param operation Library operation-value code. ++ * ++ * \retval Message conversion entry on success. ++ * \retval NULL on error. ++ */ ++static const struct rose_convert_msg *rose_find_msg_by_op_code(struct pri *ctrl, ++ enum rose_operation operation) ++{ ++ const struct rose_convert_msg *found; ++ const struct rose_convert_msg *table; ++ size_t num_entries; ++ size_t index; ++ ++ /* Determine which message conversion table to use */ ++ switch (ctrl->switchtype) { ++ case PRI_SWITCH_EUROISDN_T1: ++ case PRI_SWITCH_EUROISDN_E1: ++ table = rose_etsi_msgs; ++ num_entries = ARRAY_LEN(rose_etsi_msgs); ++ break; ++ case PRI_SWITCH_QSIG: ++ table = rose_qsig_msgs; ++ num_entries = ARRAY_LEN(rose_qsig_msgs); ++ break; ++ case PRI_SWITCH_DMS100: ++ table = rose_dms100_msgs; ++ num_entries = ARRAY_LEN(rose_dms100_msgs); ++ break; ++ case PRI_SWITCH_ATT4ESS: ++ case PRI_SWITCH_LUCENT5E: ++ case PRI_SWITCH_NI2: ++ table = rose_ni2_msgs; ++ num_entries = ARRAY_LEN(rose_ni2_msgs); ++ break; ++ default: ++ return NULL; ++ } ++ ++ /* Search for the table entry */ ++ found = NULL; ++ for (index = 0; index < num_entries; ++index) { ++ if (table[index].operation == operation) { ++ found = &table[index]; ++ break; ++ } ++ } ++ ++ return found; ++} ++ ++/*! ++ * \internal ++ * \brief Find an operation message conversion entry using the ++ * operation-value OID value or localValue. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param oid Search for the full OID if not NULL. ++ * \param local Search for the localValue if OID is NULL. ++ * ++ * \retval Message conversion entry on success. ++ * \retval NULL on error. ++ */ ++static const struct rose_convert_msg *rose_find_msg_by_op_val(struct pri *ctrl, ++ const struct asn1_oid *oid, unsigned local) ++{ ++ const struct rose_convert_msg *found; ++ const struct rose_convert_msg *table; ++ size_t num_entries; ++ size_t index; ++ int sub_index; ++ ++ /* Determine which message conversion table to use */ ++ switch (ctrl->switchtype) { ++ case PRI_SWITCH_EUROISDN_T1: ++ case PRI_SWITCH_EUROISDN_E1: ++ table = rose_etsi_msgs; ++ num_entries = ARRAY_LEN(rose_etsi_msgs); ++ break; ++ case PRI_SWITCH_QSIG: ++ table = rose_qsig_msgs; ++ num_entries = ARRAY_LEN(rose_qsig_msgs); ++ break; ++ case PRI_SWITCH_DMS100: ++ table = rose_dms100_msgs; ++ num_entries = ARRAY_LEN(rose_dms100_msgs); ++ break; ++ case PRI_SWITCH_ATT4ESS: ++ case PRI_SWITCH_LUCENT5E: ++ case PRI_SWITCH_NI2: ++ table = rose_ni2_msgs; ++ num_entries = ARRAY_LEN(rose_ni2_msgs); ++ break; ++ default: ++ return NULL; ++ } ++ ++ /* Search for the table entry */ ++ found = NULL; ++ if (oid) { ++ /* Search for an OID entry */ ++ local = oid->value[oid->num_values - 1]; ++ for (index = 0; index < num_entries; ++index) { ++ if (table[index].value == local && table[index].oid_prefix ++ && table[index].oid_prefix->num_values == oid->num_values - 1) { ++ /* Now lets match the OID prefix subidentifiers */ ++ for (sub_index = oid->num_values - 2; 0 <= sub_index; --sub_index) { ++ if (oid->value[sub_index] ++ != table[index].oid_prefix->value[sub_index]) { ++ break; ++ } ++ } ++ if (sub_index == -1) { ++ /* All of the OID subidentifiers matched */ ++ found = &table[index]; ++ break; ++ } ++ } ++ } ++ } else { ++ /* Search for a localValue entry */ ++ for (index = 0; index < num_entries; ++index) { ++ if (table[index].value == local && !table[index].oid_prefix) { ++ found = &table[index]; ++ break; ++ } ++ } ++ } ++ ++ return found; ++} ++ ++/*! ++ * \internal ++ * \brief Find an error conversion entry using the error code. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param code Library error-value code. ++ * ++ * \retval Error conversion entry on success. ++ * \retval NULL on error. ++ */ ++static const struct rose_convert_error *rose_find_error_by_op_code(struct pri *ctrl, ++ enum rose_error_code code) ++{ ++ const struct rose_convert_error *found; ++ const struct rose_convert_error *table; ++ size_t num_entries; ++ size_t index; ++ ++ /* Determine which error conversion table to use */ ++ switch (ctrl->switchtype) { ++ case PRI_SWITCH_EUROISDN_T1: ++ case PRI_SWITCH_EUROISDN_E1: ++ table = rose_etsi_errors; ++ num_entries = ARRAY_LEN(rose_etsi_errors); ++ break; ++ case PRI_SWITCH_QSIG: ++ table = rose_qsig_errors; ++ num_entries = ARRAY_LEN(rose_qsig_errors); ++ break; ++ case PRI_SWITCH_DMS100: ++ table = rose_dms100_errors; ++ num_entries = ARRAY_LEN(rose_dms100_errors); ++ break; ++ case PRI_SWITCH_ATT4ESS: ++ case PRI_SWITCH_LUCENT5E: ++ case PRI_SWITCH_NI2: ++ table = rose_ni2_errors; ++ num_entries = ARRAY_LEN(rose_ni2_errors); ++ break; ++ default: ++ return NULL; ++ } ++ ++ /* Search for the table entry */ ++ found = NULL; ++ for (index = 0; index < num_entries; ++index) { ++ if (table[index].code == code) { ++ found = &table[index]; ++ break; ++ } ++ } ++ ++ return found; ++} ++ ++/*! ++ * \internal ++ * \brief Find an error conversion entry using the ++ * error-value OID value or localValue. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param oid Search for the full OID if not NULL. ++ * \param local Search for the localValue if OID is NULL. ++ * ++ * \retval Error conversion entry on success. ++ * \retval NULL on error. ++ */ ++static const struct rose_convert_error *rose_find_error_by_op_val(struct pri *ctrl, ++ const struct asn1_oid *oid, unsigned local) ++{ ++ const struct rose_convert_error *found; ++ const struct rose_convert_error *table; ++ size_t num_entries; ++ size_t index; ++ int sub_index; ++ ++ /* Determine which error conversion table to use */ ++ switch (ctrl->switchtype) { ++ case PRI_SWITCH_EUROISDN_T1: ++ case PRI_SWITCH_EUROISDN_E1: ++ table = rose_etsi_errors; ++ num_entries = ARRAY_LEN(rose_etsi_errors); ++ break; ++ case PRI_SWITCH_QSIG: ++ table = rose_qsig_errors; ++ num_entries = ARRAY_LEN(rose_qsig_errors); ++ break; ++ case PRI_SWITCH_DMS100: ++ table = rose_dms100_errors; ++ num_entries = ARRAY_LEN(rose_dms100_errors); ++ break; ++ case PRI_SWITCH_ATT4ESS: ++ case PRI_SWITCH_LUCENT5E: ++ case PRI_SWITCH_NI2: ++ table = rose_ni2_errors; ++ num_entries = ARRAY_LEN(rose_ni2_errors); ++ break; ++ default: ++ return NULL; ++ } ++ ++ /* Search for the table entry */ ++ found = NULL; ++ if (oid) { ++ /* Search for an OID entry */ ++ local = oid->value[oid->num_values - 1]; ++ for (index = 0; index < num_entries; ++index) { ++ if (table[index].value == local && table[index].oid_prefix ++ && table[index].oid_prefix->num_values == oid->num_values - 1) { ++ /* Now lets match the OID prefix subidentifiers */ ++ for (sub_index = oid->num_values - 2; 0 <= sub_index; --sub_index) { ++ if (oid->value[sub_index] ++ != table[index].oid_prefix->value[sub_index]) { ++ break; ++ } ++ } ++ if (sub_index == -1) { ++ /* All of the OID subidentifiers matched */ ++ found = &table[index]; ++ break; ++ } ++ } ++ } ++ } else { ++ /* Search for a localValue entry */ ++ for (index = 0; index < num_entries; ++index) { ++ if (table[index].value == local && !table[index].oid_prefix) { ++ found = &table[index]; ++ break; ++ } ++ } ++ } ++ ++ return found; ++} ++ ++/*! ++ * \internal ++ * \brief Encode the Facility ie component operation-value. ++ * ++ * \param pos Starting position to encode the operation-value. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param oid_prefix Encode as an OID if not NULL. ++ * \param local Encode as a localValue if oid_prefix is NULL ++ * else it is the last OID subidentifier. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *rose_enc_operation_value(unsigned char *pos, unsigned char *end, ++ const struct asn1_oid *oid_prefix, unsigned local) ++{ ++ struct asn1_oid oid; ++ ++ if (oid_prefix) { ++ if (ARRAY_LEN(oid_prefix->value) <= oid_prefix->num_values) { ++ return NULL; ++ } ++ oid = *oid_prefix; ++ oid.value[oid.num_values++] = local; ++ return asn1_enc_oid(pos, end, ASN1_TYPE_OBJECT_IDENTIFIER, &oid); ++ } else { ++ return asn1_enc_int(pos, end, ASN1_TYPE_INTEGER, local); ++ } ++} ++ ++/*! \brief Mapped to rose_enc_operation_value() */ ++#define rose_enc_error_value(pos, end, oid_prefix, local) \ ++ rose_enc_operation_value(pos, end, oid_prefix, local) ++ ++/*! ++ * \brief Encode the invoke component for a ROSE message. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 message. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param msg ROSE invoke message to encode. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_encode_invoke(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const struct rose_msg_invoke *msg) ++{ ++ const struct rose_convert_msg *convert; ++ unsigned char *seq_len; ++ ++ convert = rose_find_msg_by_op_code(ctrl, msg->operation); ++ if (!convert) { ++ return NULL; ++ } ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ROSE_TAG_COMPONENT_INVOKE); ++ ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_INTEGER, msg->invoke_id)); ++ if (msg->linked_id_present) { ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 0, ++ msg->linked_id)); ++ } ++ ASN1_CALL(pos, rose_enc_operation_value(pos, end, convert->oid_prefix, ++ convert->value)); ++ ++ if (convert->encode_invoke_args) { ++ ASN1_CALL(pos, convert->encode_invoke_args(ctrl, pos, end, &msg->args)); ++ } ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the result component for a ROSE message. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 message. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param msg ROSE result message to encode. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_encode_result(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const struct rose_msg_result *msg) ++{ ++ const struct rose_convert_msg *convert; ++ unsigned char *seq_len; ++ unsigned char *op_seq_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ROSE_TAG_COMPONENT_RESULT); ++ ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_INTEGER, msg->invoke_id)); ++ ++ if (msg->operation != ROSE_None) { ++ convert = rose_find_msg_by_op_code(ctrl, msg->operation); ++ if (!convert) { ++ return NULL; ++ } ++ ++ ASN1_CONSTRUCTED_BEGIN(op_seq_len, pos, end, ASN1_TYPE_SEQUENCE); ++ ++ ASN1_CALL(pos, rose_enc_operation_value(pos, end, convert->oid_prefix, ++ convert->value)); ++ ++ if (convert->encode_result_args) { ++ ASN1_CALL(pos, convert->encode_result_args(ctrl, pos, end, &msg->args)); ++ } ++ ++ ASN1_CONSTRUCTED_END(op_seq_len, pos, end); ++ } ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the error component for a ROSE message. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 message. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param msg ROSE error message to encode. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_encode_error(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const struct rose_msg_error *msg) ++{ ++ const struct rose_convert_error *convert; ++ unsigned char *seq_len; ++ ++ convert = rose_find_error_by_op_code(ctrl, msg->code); ++ if (!convert) { ++ return NULL; ++ } ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ROSE_TAG_COMPONENT_ERROR); ++ ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_INTEGER, msg->invoke_id)); ++ ASN1_CALL(pos, rose_enc_error_value(pos, end, convert->oid_prefix, convert->value)); ++ if (convert->encode_error_args) { ++ ASN1_CALL(pos, convert->encode_error_args(ctrl, pos, end, &msg->args)); ++ } ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the reject component for a ROSE message. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 message. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param msg ROSE reject message to encode. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_encode_reject(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const struct rose_msg_reject *msg) ++{ ++ unsigned char *seq_len; ++ unsigned tag; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ROSE_TAG_COMPONENT_REJECT); ++ ++ /* Encode Invoke ID */ ++ if (msg->invoke_id_present) { ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_INTEGER, msg->invoke_id)); ++ } else { ++ ASN1_CALL(pos, asn1_enc_null(pos, end, ASN1_TYPE_NULL)); ++ } ++ ++ /* Encode the reject problem */ ++ switch (msg->code & ~0xFF) { ++ case ROSE_REJECT_BASE(ROSE_REJECT_BASE_General): ++ tag = ASN1_CLASS_CONTEXT_SPECIFIC | 0; ++ break; ++ case ROSE_REJECT_BASE(ROSE_REJECT_BASE_Invoke): ++ tag = ASN1_CLASS_CONTEXT_SPECIFIC | 1; ++ break; ++ case ROSE_REJECT_BASE(ROSE_REJECT_BASE_Result): ++ tag = ASN1_CLASS_CONTEXT_SPECIFIC | 2; ++ break; ++ case ROSE_REJECT_BASE(ROSE_REJECT_BASE_Error): ++ tag = ASN1_CLASS_CONTEXT_SPECIFIC | 3; ++ break; ++ default: ++ return NULL; ++ } ++ ASN1_CALL(pos, asn1_enc_int(pos, end, tag, msg->code & 0xFF)); ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the ROSE message. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 message. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param msg ROSE message to encode. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ * ++ * \note This function only encodes the ROSE contents. It does not include ++ * the protocol profile, NFE, NPP, and interpretation octets defined in ++ * a facility ie that may precede the ROSE contents. These header octets ++ * may already be stored in the encompassing buffer before the starting ++ * position given here. ++ */ ++unsigned char *rose_encode(struct pri *ctrl, unsigned char *pos, unsigned char *end, ++ const struct rose_message *msg) ++{ ++ switch (msg->type) { ++ case ROSE_COMP_TYPE_INVOKE: ++ pos = rose_encode_invoke(ctrl, pos, end, &msg->component.invoke); ++ break; ++ case ROSE_COMP_TYPE_RESULT: ++ pos = rose_encode_result(ctrl, pos, end, &msg->component.result); ++ break; ++ case ROSE_COMP_TYPE_ERROR: ++ pos = rose_encode_error(ctrl, pos, end, &msg->component.error); ++ break; ++ case ROSE_COMP_TYPE_REJECT: ++ pos = rose_encode_reject(ctrl, pos, end, &msg->component.reject); ++ break; ++ default: ++ pos = NULL; ++ break; ++ } ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Encode the NetworkFacilityExtension type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param nfe Network Facility Extension information to encode. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *fac_enc_nfe(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const struct facNetworkFacilityExtension *nfe) ++{ ++ unsigned char *seq_len; ++ unsigned char *explicit_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 10); ++ ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 0, ++ nfe->source_entity)); ++ if (nfe->source_number.length) { ++ /* EXPLICIT tag */ ++ ASN1_CONSTRUCTED_BEGIN(explicit_len, pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 1); ++ ASN1_CALL(pos, rose_enc_PartyNumber(ctrl, pos, end, &nfe->source_number)); ++ ASN1_CONSTRUCTED_END(explicit_len, pos, end); ++ } ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 2, ++ nfe->destination_entity)); ++ if (nfe->destination_number.length) { ++ /* EXPLICIT tag */ ++ ASN1_CONSTRUCTED_BEGIN(explicit_len, pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 3); ++ ASN1_CALL(pos, rose_enc_PartyNumber(ctrl, pos, end, &nfe->destination_number)); ++ ASN1_CONSTRUCTED_END(explicit_len, pos, end); ++ } ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the facility extension header. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param header Facility extension information to encode. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *fac_enc_extension_header(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const struct fac_extension_header *header) ++{ ++ if (header->nfe_present) { ++ ASN1_CALL(pos, fac_enc_nfe(ctrl, pos, end, &header->nfe)); ++ } ++ if (header->npp_present) { ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 18, ++ header->npp)); ++ } ++ if (header->interpretation_present) { ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 11, ++ header->interpretation)); ++ } ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the facility ie contents header. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode the facility ie contents. ++ * \param end End of facility ie contents encoding data buffer. ++ * \param header Facility extension header data to encode (NULL if none). ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *facility_encode_header(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const struct fac_extension_header *header) ++{ ++ /* Make sure we have some room. */ ++ if (end < pos + 2) { ++ return NULL; ++ } ++ ++ switch (ctrl->switchtype) { ++ case PRI_SWITCH_EUROISDN_T1: ++ case PRI_SWITCH_EUROISDN_E1: ++ *pos++ = 0x80 | Q932_PROTOCOL_ROSE; ++ header = NULL; ++ break; ++ case PRI_SWITCH_QSIG: ++ *pos++ = 0x80 | Q932_PROTOCOL_EXTENSIONS; ++ break; ++ case PRI_SWITCH_DMS100: ++ *pos++ = Q932_PROTOCOL_ROSE; /* DON'T set the EXT bit yet. */ ++ *pos++ = 0x80 | ROSE_DMS100_RLT_SERVICE_ID; ++ header = NULL; ++ break; ++ case PRI_SWITCH_ATT4ESS: ++ case PRI_SWITCH_LUCENT5E: ++ case PRI_SWITCH_NI2: ++ if (header) { ++ *pos++ = 0x80 | Q932_PROTOCOL_EXTENSIONS; ++ } else { ++ *pos++ = 0x80 | Q932_PROTOCOL_ROSE; ++ } ++ break; ++ default: ++ return NULL; ++ } ++ ++ if (header) { ++ ASN1_CALL(pos, fac_enc_extension_header(ctrl, pos, end, header)); ++ } ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Decode the ROSE invoke message. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param msg ROSE invoke message data to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++static const unsigned char *rose_decode_invoke(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, struct rose_msg_invoke *msg) ++{ ++ int32_t value; ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ const struct rose_convert_msg *convert; ++ struct asn1_oid oid; ++ unsigned local; ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, "INVOKE Component %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_INTEGER); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "invokeId", tag, pos, seq_end, &value)); ++ msg->invoke_id = value; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ if (tag == (ASN1_CLASS_CONTEXT_SPECIFIC | 0)) { ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "linkedId", tag, pos, seq_end, &value)); ++ msg->linked_id = value; ++ msg->linked_id_present = 1; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ } else { ++ msg->linked_id_present = 0; ++ } ++ ++ /* Decode operation-value */ ++ switch (tag) { ++ case ASN1_TYPE_INTEGER: ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "operationValue", tag, pos, seq_end, &value)); ++ local = value; ++ oid.num_values = 0; ++ break; ++ case ASN1_TYPE_OBJECT_IDENTIFIER: ++ ASN1_CALL(pos, asn1_dec_oid(ctrl, "operationValue", tag, pos, seq_end, &oid)); ++ local = 0; ++ break; ++ default: ++ ASN1_DID_NOT_EXPECT_TAG(ctrl, tag); ++ return NULL; ++ } ++ convert = rose_find_msg_by_op_val(ctrl, (oid.num_values == 0) ? NULL : &oid, local); ++ if (convert) { ++ msg->operation = convert->operation; ++ } else { ++ msg->operation = ROSE_Unknown; ++ } ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " operationValue = %s\n", rose_operation2str(msg->operation)); ++ } ++ ++ /* Decode any expected invoke arguments */ ++ if (convert && convert->decode_invoke_args) { ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CALL(pos, convert->decode_invoke_args(ctrl, tag, pos, seq_end, &msg->args)); ++ } ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Decode the ROSE result message. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param msg ROSE result message data to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++static const unsigned char *rose_decode_result(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, struct rose_msg_result *msg) ++{ ++ int32_t value; ++ int length; ++ int seq_offset; ++ int op_seq_offset; ++ const unsigned char *seq_end; ++ const unsigned char *op_seq_end; ++ const struct rose_convert_msg *convert; ++ struct asn1_oid oid; ++ unsigned local; ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, "RESULT Component %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_INTEGER); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "invokeId", tag, pos, seq_end, &value)); ++ msg->invoke_id = value; ++ ++ /* Decode optional operation sequence */ ++ if (pos < seq_end && *pos != ASN1_INDEF_TERM) { ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " operation %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, seq_end, &length)); ++ ASN1_END_SETUP(op_seq_end, op_seq_offset, length, pos, seq_end); ++ ++ /* Decode operation-value */ ++ ASN1_CALL(pos, asn1_dec_tag(pos, op_seq_end, &tag)); ++ switch (tag) { ++ case ASN1_TYPE_INTEGER: ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "operationValue", tag, pos, op_seq_end, ++ &value)); ++ local = value; ++ oid.num_values = 0; ++ break; ++ case ASN1_TYPE_OBJECT_IDENTIFIER: ++ ASN1_CALL(pos, asn1_dec_oid(ctrl, "operationValue", tag, pos, op_seq_end, ++ &oid)); ++ local = 0; ++ break; ++ default: ++ ASN1_DID_NOT_EXPECT_TAG(ctrl, tag); ++ return NULL; ++ } ++ convert = ++ rose_find_msg_by_op_val(ctrl, (oid.num_values == 0) ? NULL : &oid, local); ++ if (convert) { ++ msg->operation = convert->operation; ++ } else { ++ msg->operation = ROSE_Unknown; ++ } ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " operationValue = %s\n", ++ rose_operation2str(msg->operation)); ++ } ++ ++ /* Decode any expected result arguments */ ++ if (convert && convert->decode_result_args) { ++ ASN1_CALL(pos, asn1_dec_tag(pos, op_seq_end, &tag)); ++ ASN1_CALL(pos, convert->decode_result_args(ctrl, tag, pos, op_seq_end, ++ &msg->args)); ++ } ++ ++ ASN1_END_FIXUP(ctrl, pos, op_seq_offset, op_seq_end, seq_end); ++ } else { ++ msg->operation = ROSE_None; ++ } ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Decode the ROSE error message. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param msg ROSE error message data to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++static const unsigned char *rose_decode_error(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, struct rose_msg_error *msg) ++{ ++ int32_t value; ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ const struct rose_convert_error *convert; ++ struct asn1_oid oid; ++ unsigned local; ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, "ERROR Component %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_INTEGER); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "invokeId", tag, pos, seq_end, &value)); ++ msg->invoke_id = value; ++ ++ /* Decode error-value */ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ switch (tag) { ++ case ASN1_TYPE_INTEGER: ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "errorValue", tag, pos, seq_end, &value)); ++ local = value; ++ oid.num_values = 0; ++ break; ++ case ASN1_TYPE_OBJECT_IDENTIFIER: ++ ASN1_CALL(pos, asn1_dec_oid(ctrl, "errorValue", tag, pos, seq_end, &oid)); ++ local = 0; ++ break; ++ default: ++ ASN1_DID_NOT_EXPECT_TAG(ctrl, tag); ++ return NULL; ++ } ++ convert = ++ rose_find_error_by_op_val(ctrl, (oid.num_values == 0) ? NULL : &oid, local); ++ if (convert) { ++ msg->code = convert->code; ++ } else { ++ msg->code = ROSE_ERROR_Unknown; ++ } ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " errorValue = %s\n", rose_error2str(msg->code)); ++ } ++ ++ /* Decode any expected error parameters */ ++ if (convert && convert->decode_error_args) { ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CALL(pos, convert->decode_error_args(ctrl, tag, pos, seq_end, &msg->args)); ++ } ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Decode the ROSE reject message. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param msg ROSE reject message data to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++static const unsigned char *rose_decode_reject(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, struct rose_msg_reject *msg) ++{ ++ int32_t value; ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, "REJECT Component %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ /* Invoke ID choice */ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ switch (tag) { ++ case ASN1_TYPE_INTEGER: ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "invokeId", tag, pos, seq_end, &value)); ++ msg->invoke_id = value; ++ msg->invoke_id_present = 1; ++ break; ++ case ASN1_TYPE_NULL: ++ ASN1_CALL(pos, asn1_dec_null(ctrl, "invokeId", tag, pos, seq_end)); ++ msg->invoke_id_present = 0; ++ break; ++ default: ++ ASN1_DID_NOT_EXPECT_TAG(ctrl, tag); ++ return NULL; ++ } ++ ++ /* Problem choice */ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ switch (tag) { ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 0: ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "problemGeneral", tag, pos, seq_end, &value)); ++ msg->code = ROSE_REJECT_BASE(ROSE_REJECT_BASE_General) | (value & 0xFF); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 1: ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "problemInvoke", tag, pos, seq_end, &value)); ++ msg->code = ROSE_REJECT_BASE(ROSE_REJECT_BASE_Invoke) | (value & 0xFF); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 2: ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "problemResult", tag, pos, seq_end, &value)); ++ msg->code = ROSE_REJECT_BASE(ROSE_REJECT_BASE_Result) | (value & 0xFF); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 3: ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "problemError", tag, pos, seq_end, &value)); ++ msg->code = ROSE_REJECT_BASE(ROSE_REJECT_BASE_Error) | (value & 0xFF); ++ break; ++ default: ++ ASN1_DID_NOT_EXPECT_TAG(ctrl, tag); ++ return NULL; ++ } ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " problem = %s\n", rose_reject2str(msg->code)); ++ } ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the ROSE message into the given buffer. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position of the ASN.1 component. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param msg Decoded ROSE message contents. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ * ++ * \note This function only decodes the ROSE contents. It does not check ++ * for the protocol profile, NFE, NPP, and interpretation octets defined in ++ * a facility ie that may preceed the ROSE contents. These header octets ++ * may already have been consumed from the encompasing buffer before the ++ * buffer given here. ++ */ ++const unsigned char *rose_decode(struct pri *ctrl, const unsigned char *pos, ++ const unsigned char *end, struct rose_message *msg) ++{ ++ unsigned tag; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, end, &tag)); ++ switch (tag) { ++ case ROSE_TAG_COMPONENT_INVOKE: ++ msg->type = ROSE_COMP_TYPE_INVOKE; ++ ASN1_CALL(pos, rose_decode_invoke(ctrl, tag, pos, end, &msg->component.invoke)); ++ break; ++ case ROSE_TAG_COMPONENT_RESULT: ++ msg->type = ROSE_COMP_TYPE_RESULT; ++ ASN1_CALL(pos, rose_decode_result(ctrl, tag, pos, end, &msg->component.result)); ++ break; ++ case ROSE_TAG_COMPONENT_ERROR: ++ msg->type = ROSE_COMP_TYPE_ERROR; ++ ASN1_CALL(pos, rose_decode_error(ctrl, tag, pos, end, &msg->component.error)); ++ break; ++ case ROSE_TAG_COMPONENT_REJECT: ++ msg->type = ROSE_COMP_TYPE_REJECT; ++ ASN1_CALL(pos, rose_decode_reject(ctrl, tag, pos, end, &msg->component.reject)); ++ break; ++ default: ++ msg->type = ROSE_COMP_TYPE_INVALID; ++ ASN1_DID_NOT_EXPECT_TAG(ctrl, tag); ++ return NULL; ++ } ++ ++ if (pos < end) { ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %u byte(s) of trailing data not consumed.\n", ++ (unsigned) (end - pos)); ++ } ++ } ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Decode the NetworkFacilityExtension argument parameters. ++ * ++ * \param ctrl D channel controller for any diagnostic messages. ++ * \param name Field name ++ * \param tag Component tag that identified this production. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param nfe Parameter storage to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++static const unsigned char *fac_dec_nfe(struct pri *ctrl, const char *name, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, ++ struct facNetworkFacilityExtension *nfe) ++{ ++ int length; ++ int seq_offset; ++ int explicit_offset; ++ const unsigned char *seq_end; ++ const unsigned char *explicit_end; ++ const unsigned char *save_pos; ++ int32_t value; ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s NetworkFacilityExtension %s\n", name, asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_CLASS_CONTEXT_SPECIFIC | 0); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "sourceEntity", tag, pos, seq_end, &value)); ++ nfe->source_entity = value; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ if (tag == (ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 1)) { ++ /* Remove EXPLICIT tag */ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " Explicit %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, seq_end, &length)); ++ ASN1_END_SETUP(explicit_end, explicit_offset, length, pos, seq_end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, explicit_end, &tag)); ++ ASN1_CALL(pos, rose_dec_PartyNumber(ctrl, "sourceEntityAddress", tag, pos, ++ seq_end, &nfe->source_number)); ++ ++ ASN1_END_FIXUP(ctrl, pos, explicit_offset, explicit_end, seq_end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ } else { ++ nfe->source_number.length = 0; ++ } ++ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_CLASS_CONTEXT_SPECIFIC | 2); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "destinationEntity", tag, pos, seq_end, &value)); ++ nfe->destination_entity = value; ++ ++ nfe->destination_number.length = 0; ++ if (pos < seq_end && *pos != ASN1_INDEF_TERM) { ++ save_pos = pos; ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ if (tag == (ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 3)) { ++ /* Remove EXPLICIT tag */ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " Explicit %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, seq_end, &length)); ++ ASN1_END_SETUP(explicit_end, explicit_offset, length, pos, seq_end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, explicit_end, &tag)); ++ ASN1_CALL(pos, rose_dec_PartyNumber(ctrl, "destinationEntityAddress", tag, ++ pos, seq_end, &nfe->destination_number)); ++ ++ ASN1_END_FIXUP(ctrl, pos, explicit_offset, explicit_end, seq_end); ++ } else { ++ pos = save_pos; ++ } ++ } ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the extension header argument parameters. ++ * ++ * \param ctrl D channel controller for any diagnostic messages. ++ * \param pos Starting position of the ASN.1 component. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param header Parameter storage to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *fac_dec_extension_header(struct pri *ctrl, const unsigned char *pos, ++ const unsigned char *end, struct fac_extension_header *header) ++{ ++ int32_t value; ++ unsigned tag; ++ const unsigned char *save_pos; ++ ++ /* ++ * For simplicity we are not checking the order of ++ * the optional header components. ++ */ ++ header->nfe_present = 0; ++ header->npp_present = 0; ++ header->interpretation_present = 0; ++ while (pos < end) { ++ save_pos = pos; ++ ASN1_CALL(pos, asn1_dec_tag(pos, end, &tag)); ++ switch (tag) { ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 10: ++ ASN1_CALL(pos, fac_dec_nfe(ctrl, "nfe", tag, pos, end, &header->nfe)); ++ header->nfe_present = 1; ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 18: ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "networkProtocolProfile", tag, pos, end, ++ &value)); ++ header->npp = value; ++ header->npp_present = 1; ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 11: ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "interpretation", tag, pos, end, &value)); ++ header->interpretation = value; ++ header->interpretation_present = 1; ++ break; ++ default: ++ pos = save_pos; ++ goto cancel_options; ++ } ++ } ++cancel_options:; ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the facility ie contents header. ++ * ++ * \param ctrl D channel controller for any diagnostic messages. ++ * \param pos Starting position of the facility ie contents. ++ * \param end End of facility ie contents. ++ * \param header Parameter storage to fill. ++ * ++ * \retval Start of the next ASN.1 component on success (ROSE message). ++ * \retval NULL on error. ++ */ ++const unsigned char *facility_decode_header(struct pri *ctrl, const unsigned char *pos, ++ const unsigned char *end, struct fac_extension_header *header) ++{ ++ /* Make sure we have enough room for the protocol profile ie octet(s) */ ++ if (end < pos + 2) { ++ return NULL; ++ } ++ switch (*pos & Q932_PROTOCOL_MASK) { ++ case Q932_PROTOCOL_ROSE: ++ case Q932_PROTOCOL_EXTENSIONS: ++ break; ++ default: ++ return NULL; ++ } ++ if (!(*pos & 0x80)) { ++ /* DMS-100 Service indicator octet - Just ignore for now */ ++ ++pos; ++ } ++ ++pos; ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ asn1_dump(ctrl, pos, end); ++ } ++ ++ pos = fac_dec_extension_header(ctrl, pos, end, header); ++ return pos; ++} ++ ++/*! ++ * \brief Decode the facility ie contents for debug purposes. ++ * ++ * \param ctrl D channel controller for any diagnostic messages. ++ * \param buf Buffer containing the facility ie contents. ++ * \param length Length of facility ie contents. ++ * ++ * \return Nothing ++ * ++ * \note Should only be called if PRI_DEBUG_APDU is enabled. Otherwise, ++ * it it does nothing useful. ++ */ ++void facility_decode_dump(struct pri *ctrl, const unsigned char *buf, size_t length) ++{ ++ const unsigned char *pos; ++ const unsigned char *end; ++ union { ++ struct fac_extension_header header; ++ struct rose_message rose; ++ } discard; ++ ++ end = buf + length; ++ pos = facility_decode_header(ctrl, buf, end, &discard.header); ++ while (pos && pos < end) { ++ pos = rose_decode(ctrl, pos, end, &discard.rose); ++ } ++} ++ ++/* ------------------------------------------------------------------- */ ++/* end rose.c */ + +Property changes on: rose.c +___________________________________________________________________ +Added: svn:eol-style + + native +Added: svn:mime-type + + text/plain +Added: svn:keywords + + 'Author Date Id Revision' + +Index: rose_other.c +=================================================================== +--- a/rose_other.c (.../tags/1.4.10.2) (revision 0) ++++ b/rose_other.c (.../branches/1.4) (revision 1357) +@@ -0,0 +1,277 @@ ++/* ++ * libpri: An implementation of Primary Rate ISDN ++ * ++ * Copyright (C) 2009 Digium, Inc. ++ * ++ * Richard Mudgett <rmudgett@digium.com> ++ * ++ * See http://www.asterisk.org for more information about ++ * the Asterisk project. Please do not directly contact ++ * any of the maintainers of this project for assistance; ++ * the project provides a web site, mailing lists and IRC ++ * channels for your use. ++ * ++ * This program is free software, distributed under the terms of ++ * the GNU General Public License Version 2 as published by the ++ * Free Software Foundation. See the LICENSE file included with ++ * this program for more details. ++ * ++ * In addition, when this program is distributed with Asterisk in ++ * any form that would qualify as a 'combined work' or as a ++ * 'derivative work' (but not mere aggregation), you can redistribute ++ * and/or modify the combination under the terms of the license ++ * provided with that copy of Asterisk, instead of the license ++ * terms granted here. ++ */ ++ ++/*! ++ * \file ++ * \brief Switch type operations for: NI2, 4ESS, 5ESS, DMS-100 ++ * ++ * \author Richard Mudgett <rmudgett@digium.com> ++ */ ++ ++ ++#include "compat.h" ++#include "libpri.h" ++#include "pri_internal.h" ++#include "rose.h" ++#include "rose_internal.h" ++#include "asn1.h" ++ ++ ++/* ------------------------------------------------------------------- */ ++ ++/*! ++ * \brief Encode the DMS-100 RLT_OperationInd result facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_dms100_RLT_OperationInd_RES(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_result_args *args) ++{ ++ return asn1_enc_int(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 0, ++ args->dms100.RLT_OperationInd.call_id); ++} ++ ++/*! ++ * \brief Encode the DMS-100 RLT_ThirdParty invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_dms100_RLT_ThirdParty_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ const struct roseDms100RLTThirdParty_ARG *rlt_thirdparty; ++ unsigned char *seq_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ASN1_TAG_SEQUENCE); ++ ++ rlt_thirdparty = &args->dms100.RLT_ThirdParty; ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 0, ++ rlt_thirdparty->call_id)); ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 1, ++ rlt_thirdparty->reason)); ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the DMS-100 RLT_OperationInd result argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_dms100_RLT_OperationInd_RES(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, union rose_msg_result_args *args) ++{ ++ int32_t value; ++ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_CLASS_CONTEXT_SPECIFIC | 0); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "callId", tag, pos, end, &value)); ++ args->dms100.RLT_OperationInd.call_id = value; ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the DMS-100 RLT_ThirdParty invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_dms100_RLT_ThirdParty_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, union rose_msg_invoke_args *args) ++{ ++ int32_t value; ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ struct roseDms100RLTThirdParty_ARG *rlt_third_party; ++ ++ rlt_third_party = &args->dms100.RLT_ThirdParty; ++ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " RLT_ThirdParty %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_CLASS_CONTEXT_SPECIFIC | 0); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "callId", tag, pos, seq_end, &value)); ++ rlt_third_party->call_id = value; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_CLASS_CONTEXT_SPECIFIC | 1); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "reason", tag, pos, seq_end, &value)); ++ rlt_third_party->reason = value; ++ ++ /* Fixup will skip over any OPTIONAL information */ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the NI2 InformationFollowing invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_ni2_InformationFollowing_ARG(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ /* Encode the unknown enumeration value. */ ++ return asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, ++ args->ni2.InformationFollowing.value); ++} ++ ++/*! ++ * \brief Encode the NI2 InitiateTransfer invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_ni2_InitiateTransfer_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ const struct roseNi2InitiateTransfer_ARG *initiate_transfer; ++ unsigned char *seq_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ASN1_TAG_SEQUENCE); ++ ++ initiate_transfer = &args->ni2.InitiateTransfer; ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_INTEGER, ++ initiate_transfer->call_reference)); ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the NI2 InformationFollowing invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_ni2_InformationFollowing_ARG(struct pri *ctrl, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args) ++{ ++ int32_t value; ++ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_ENUMERATED); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "unknown", tag, pos, end, &value)); ++ args->ni2.InformationFollowing.value = value; ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the NI2 InitiateTransfer invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_ni2_InitiateTransfer_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, union rose_msg_invoke_args *args) ++{ ++ int32_t value; ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ struct roseNi2InitiateTransfer_ARG *initiate_transfer; ++ ++ initiate_transfer = &args->ni2.InitiateTransfer; ++ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " InitiateTransfer %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_INTEGER); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "callReference", tag, pos, seq_end, &value)); ++ initiate_transfer->call_reference = value; ++ ++ /* Fixup will skip over any OPTIONAL information */ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/* ------------------------------------------------------------------- */ ++/* end rose_other.c */ + +Property changes on: rose_other.c +___________________________________________________________________ +Added: svn:eol-style + + native +Added: svn:mime-type + + text/plain +Added: svn:keywords + + 'Author Date Id Revision' + +Index: pri_q931.h +=================================================================== +--- a/pri_q931.h (.../tags/1.4.10.2) (revision 1357) ++++ b/pri_q931.h (.../branches/1.4) (revision 1357) +@@ -30,44 +30,6 @@ + #ifndef _PRI_Q931_H + #define _PRI_Q931_H + +-typedef enum q931_state { +- /* User states */ +- U0_NULL_STATE, +- U1_CALL_INITIATED, +- U2_OVERLAP_SENDING, +- U3_OUTGOING_CALL_PROCEEDING, +- U4_CALL_DELIVERED, +- U6_CALL_PRESENT, +- U7_CALL_RECEIVED, +- U8_CONNECT_REQUEST, +- U9_INCOMING_CALL_PROCEEDING, +- U10_ACTIVE, +- U11_DISCONNECT_REQUEST, +- U12_DISCONNECT_INDICATION, +- U15_SUSPEND_REQUEST, +- U17_RESUME_REQUEST, +- U19_RELEASE_REQUEST, +- U25_OVERLAP_RECEIVING, +- /* Network states */ +- N0_NULL_STATE, +- N1_CALL_INITIATED, +- N2_OVERLAP_SENDING, +- N3_OUTGOING_CALL_PROCEEDING, +- N4_CALL_DELIVERED, +- N6_CALL_PRESENT, +- N7_CALL_RECEIVED, +- N8_CONNECT_REQUEST, +- N9_INCOMING_CALL_PROCEEDING, +- N10_ACTIVE, +- N11_DISCONNECT_REQUEST, +- N12_DISCONNECT_INDICATION, +- N15_SUSPEND_REQUEST, +- N17_RESUME_REQUEST, +- N19_RELEASE_REQUEST, +- N22_CALL_ABORT, +- N25_OVERLAP_RECEIVING +-} q931_state; +- + typedef enum q931_mode { + UNKNOWN_MODE, + CIRCUIT_MODE, +@@ -79,13 +41,13 @@ + u_int8_t pd; /* Protocol Discriminator */ + #if __BYTE_ORDER == __BIG_ENDIAN + u_int8_t x0:4; +- u_int8_t crlen:4; ++ u_int8_t crlen:4;/*!< Call reference length */ + #else +- u_int8_t crlen:4; ++ u_int8_t crlen:4;/*!< Call reference length */ + u_int8_t x0:4; + #endif + u_int8_t contents[0]; +- u_int8_t crv[3]; ++ u_int8_t crv[3];/*!< Call reference value */ + } __attribute__ ((packed)) q931_h; + + +@@ -113,6 +75,10 @@ + + #define Q931_PROTOCOL_DISCRIMINATOR 0x08 + #define GR303_PROTOCOL_DISCRIMINATOR 0x4f ++/* AT&T Maintenance Protocol Discriminator */ ++#define MAINTENANCE_PROTOCOL_DISCRIMINATOR_1 0x03 ++/* National Maintenance Protocol Discriminator */ ++#define MAINTENANCE_PROTOCOL_DISCRIMINATOR_2 0x43 + + /* Q.931 / National ISDN Message Types */ + +@@ -157,9 +123,17 @@ + #define Q931_SUSPEND_REJECT 0x21 + + /* Maintenance messages (codeset 0 only) */ +-#define NATIONAL_SERVICE 0x0f +-#define NATIONAL_SERVICE_ACKNOWLEDGE 0x07 ++#define ATT_SERVICE 0x0f ++#define ATT_SERVICE_ACKNOWLEDGE 0x07 ++#define NATIONAL_SERVICE 0x07 ++#define NATIONAL_SERVICE_ACKNOWLEDGE 0x0f + ++#define SERVICE_CHANGE_STATUS_INSERVICE 0 ++#define SERVICE_CHANGE_STATUS_LOOPBACK 1 /* not supported */ ++#define SERVICE_CHANGE_STATUS_OUTOFSERVICE 2 ++#define SERVICE_CHANGE_STATUS_REQCONTINUITYCHECK 3 /* not supported */ ++#define SERVICE_CHANGE_STATUS_SHUTDOWN 4 /* not supported */ ++ + /* Special codeset 0 IE */ + #define NATIONAL_CHANGE_STATUS 0x1 + +@@ -168,10 +142,11 @@ + #define Q931_NON_LOCKING_SHIFT 0x98 + #define Q931_BEARER_CAPABILITY 0x04 + #define Q931_CAUSE 0x08 +-#define Q931_CALL_STATE 0x14 ++#define Q931_IE_CALL_STATE 0x14 + #define Q931_CHANNEL_IDENT 0x18 + #define Q931_PROGRESS_INDICATOR 0x1e + #define Q931_NETWORK_SPEC_FAC 0x20 ++#define Q931_CALLING_PARTY_CATEGORY (0x32 | Q931_CODESET(5)) + #define Q931_INFORMATION_RATE 0x40 + #define Q931_TRANSIT_DELAY 0x42 + #define Q931_TRANS_DELAY_SELECT 0x43 +@@ -200,8 +175,9 @@ + #define Q931_IE_SEGMENTED_MSG 0x00 + #define Q931_IE_CHANGE_STATUS 0x01 + #define Q931_IE_ORIGINATING_LINE_INFO (0x01 | Q931_CODESET(6)) +-#define Q931_IE_CONNECTED_ADDR 0x0C +-#define Q931_IE_CONNECTED_NUM 0x4C ++#define Q931_IE_CONNECTED_ADDR 0x0c ++#define Q931_IE_CONNECTED_NUM 0x4c ++#define Q931_IE_CONNECTED_SUBADDR 0x4d + #define Q931_IE_CALL_IDENTITY 0x10 + #define Q931_IE_FACILITY 0x1c + #define Q931_IE_ENDPOINT_ID 0x26 +@@ -224,31 +200,255 @@ + #define Q931_IE_ESCAPE_FOR_EXT 0x7F + + +-/* Call state stuff */ +-#define Q931_CALL_STATE_NULL 0 +-#define Q931_CALL_STATE_CALL_INITIATED 1 +-#define Q931_CALL_STATE_OVERLAP_SENDING 2 +-#define Q931_CALL_STATE_OUTGOING_CALL_PROCEEDING 3 +-#define Q931_CALL_STATE_CALL_DELIVERED 4 +-#define Q931_CALL_STATE_CALL_PRESENT 6 +-#define Q931_CALL_STATE_CALL_RECEIVED 7 +-#define Q931_CALL_STATE_CONNECT_REQUEST 8 +-#define Q931_CALL_STATE_INCOMING_CALL_PROCEEDING 9 +-#define Q931_CALL_STATE_ACTIVE 10 +-#define Q931_CALL_STATE_DISCONNECT_REQUEST 11 +-#define Q931_CALL_STATE_DISCONNECT_INDICATION 12 +-#define Q931_CALL_STATE_SUSPEND_REQUEST 15 +-#define Q931_CALL_STATE_RESUME_REQUEST 17 +-#define Q931_CALL_STATE_RELEASE_REQUEST 19 +-#define Q931_CALL_STATE_OVERLAP_RECEIVING 25 +-#define Q931_CALL_STATE_RESTART_REQUEST 61 +-#define Q931_CALL_STATE_RESTART 62 ++/*! Q.931 call states */ ++enum Q931_CALL_STATE { ++ /*! ++ * \details ++ * null state (U0): ++ * No call exists. ++ * \details ++ * null state (N0): ++ * No call exists. ++ */ ++ Q931_CALL_STATE_NULL = 0, ++ /*! ++ * \details ++ * call initiated (U1): ++ * This state exists for an outgoing call, when the user requests ++ * call establishment from the network. ++ * \details ++ * call initiated (N1): ++ * This state exists for an outgoing call when the network has received ++ * a call establishment request but has not yet responded. ++ */ ++ Q931_CALL_STATE_CALL_INITIATED = 1, ++ /*! ++ * \details ++ * overlap sending (U2): ++ * This state exists for an outgoing call when the user has ++ * received acknowledgement of the call establishment request which ++ * permits the user to send additional call information to the network ++ * in overlap mode. ++ * \details ++ * overlap sending (N2): ++ * This state exists for an outgoing call when the network has acknowledged ++ * the call establishment request and is prepared to receive additional ++ * call information (if any) in overlap mode. ++ */ ++ Q931_CALL_STATE_OVERLAP_SENDING = 2, ++ /*! ++ * \details ++ * outgoing call proceeding (U3): ++ * This state exists for an outgoing call when the user has ++ * received acknowledgement that the network has received all ++ * call information necessary to effect call establishment. ++ * \details ++ * outgoing call proceeding (N3): ++ * This state exists for an outgoing call when the network has sent ++ * acknowledgement that the network has received all call information ++ * necessary to effect call establishment. ++ */ ++ Q931_CALL_STATE_OUTGOING_CALL_PROCEEDING = 3, ++ /*! ++ * \details ++ * call delivered (U4): ++ * This state exists for an outgoing call when the calling user has ++ * received an indication that remote user alerting has been initiated. ++ * \details ++ * call delivered (N4): ++ * This state exists for an outgoing call when the network has indicated ++ * that remote user alerting has been initiated. ++ */ ++ Q931_CALL_STATE_CALL_DELIVERED = 4, ++ /*! ++ * \details ++ * call present (U6): ++ * This state exists for an incoming call when the user has received a ++ * call establishment request but has not yet responded. ++ * \details ++ * call present (N6): ++ * This state exists for an incoming call when the network has sent a ++ * call establishment request but has not yet received a satisfactory ++ * response. ++ */ ++ Q931_CALL_STATE_CALL_PRESENT = 6, ++ /*! ++ * \details ++ * call received (U7): ++ * This state exists for an incoming call when the user has indicated ++ * alerting but has not yet answered. ++ * \details ++ * call received (N7): ++ * This state exists for an incoming call when the network has received ++ * an indication that the user is alerting but has not yet received an ++ * answer. ++ */ ++ Q931_CALL_STATE_CALL_RECEIVED = 7, ++ /*! ++ * \details ++ * connect request (U8): ++ * This state exists for an incoming call when the user has answered ++ * the call and is waiting to be awarded the call. ++ * \details ++ * connect request (N8): ++ * This state exists for an incoming call when the network has received ++ * an answer but the network has not yet awarded the call. ++ */ ++ Q931_CALL_STATE_CONNECT_REQUEST = 8, ++ /*! ++ * \details ++ * incoming call proceeding (U9): ++ * This state exists for an incoming call when the user has sent ++ * acknowledgement that the user has received all call information ++ * necessary to effect call establishment. ++ * \details ++ * incoming call proceeding (N9): ++ * This state exists for an incoming call when the network has received ++ * acknowledgement that the user has received all call information ++ * necessary to effect call establishment. ++ */ ++ Q931_CALL_STATE_INCOMING_CALL_PROCEEDING = 9, ++ /*! ++ * \details ++ * active (U10): ++ * This state exists for an incoming call when the user has received ++ * an acknowledgement from the network that the user has been awarded ++ * the call. This state exists for an outgoing call when the user has ++ * received an indication that the remote user has answered the call. ++ * \details ++ * active (N10): ++ * This state exists for an incoming call when the network has awarded ++ * the call to the called user. This state exists for an outgoing call ++ * when the network has indicated that the remote user has answered ++ * the call. ++ */ ++ Q931_CALL_STATE_ACTIVE = 10, ++ /*! ++ * \details ++ * disconnect request (U11): ++ * This state exists when the user has requested the network to clear ++ * the end-to-end connection (if any) and is waiting for a response. ++ * \details ++ * disconnect request (N11): ++ * This state exists when the network has received a request from the ++ * user to clear the end-to-end connection (if any). ++ */ ++ Q931_CALL_STATE_DISCONNECT_REQUEST = 11, ++ /*! ++ * \details ++ * disconnect indication (U12): ++ * This state exists when the user has received an invitation to ++ * disconnect because the network has disconnected the end-to-end ++ * connection (if any). ++ * \details ++ * disconnect indication (N12): ++ * This state exists when the network has disconnected the end-to-end ++ * connection (if any) and has sent an invitation to disconnect the ++ * user-network connection. ++ */ ++ Q931_CALL_STATE_DISCONNECT_INDICATION = 12, ++ /*! ++ * \details ++ * suspend request (U15): ++ * This state exists when the user has requested the network to suspend ++ * the call and is waiting for a response. ++ * \details ++ * suspend request (N15): ++ * This state exists when the network has received a request to suspend ++ * the call but has not yet responded. ++ */ ++ Q931_CALL_STATE_SUSPEND_REQUEST = 15, ++ /*! ++ * \details ++ * resume request (U17): ++ * This state exists when the user has requested the network to resume ++ * a previously suspended call and is waiting for a response. ++ * \details ++ * resume request (N17): ++ * This state exists when the network has received a request to resume ++ * a previously suspended call but has not yet responded. ++ */ ++ Q931_CALL_STATE_RESUME_REQUEST = 17, ++ /*! ++ * \details ++ * release request (U19): ++ * This state exists when the user has requested the network to release ++ * and is waiting for a response. ++ * \details ++ * release request (N19): ++ * This state exists when the network has requested the user to release ++ * and is waiting for a response. ++ */ ++ Q931_CALL_STATE_RELEASE_REQUEST = 19, ++ /*! ++ * \details ++ * call abort (N22): ++ * This state exists for an incoming call for the point-to-multipoint ++ * configuration when the call is being cleared before any user has been ++ * awarded the call. ++ */ ++ Q931_CALL_STATE_CALL_ABORT = 22, ++ /*! ++ * \details ++ * overlap receiving (U25): ++ * This state exists for an incoming call when the user has acknowledged ++ * the call establishment request from the network and is prepared to ++ * receive additional call information (if any) in overlap mode. ++ * \details ++ * overlap receiving (N25): ++ * This state exists for an incoming call when the network has received ++ * acknowledgement of the call establishment request which permits the ++ * network to send additional call information (if any) in the overlap ++ * mode. ++ */ ++ Q931_CALL_STATE_OVERLAP_RECEIVING = 25, ++ /*! ++ * \details ++ * call independent service (U31): (From Q.932) ++ * This state exists when a call independent supplementary service ++ * signalling connection is established. ++ * \details ++ * call independent service (N31): (From Q.932) ++ * This state exists when a call independent supplementary service ++ * signalling connection is established. ++ */ ++ Q931_CALL_STATE_CALL_INDEPENDENT_SERVICE = 31, ++ Q931_CALL_STATE_RESTART_REQUEST = 61, ++ Q931_CALL_STATE_RESTART = 62, ++ /*! ++ * \details ++ * Call state has not been set. ++ * Call state does not exist. ++ * Call state not initialized. ++ * Call state internal use only. ++ */ ++ Q931_CALL_STATE_NOT_SET = 0xFF, ++}; + ++/*! Q.931 call establishment state ranking for competing calls in PTMP NT mode. */ ++enum Q931_RANKED_CALL_STATE { ++ /*! Call is present but has no response yet. */ ++ Q931_RANKED_CALL_STATE_PRESENT, ++ /*! Call is collecting digits. */ ++ Q931_RANKED_CALL_STATE_OVERLAP, ++ /*! Call routing is happening. */ ++ Q931_RANKED_CALL_STATE_PROCEEDING, ++ /*! Called party is being alerted of the call. */ ++ Q931_RANKED_CALL_STATE_ALERTING, ++ /*! Call is connected. A winner has been declared. */ ++ Q931_RANKED_CALL_STATE_CONNECT, ++ /*! Call is in some non-call establishment state (likely disconnecting). */ ++ Q931_RANKED_CALL_STATE_OTHER, ++}; + + /* EuroISDN */ + #define Q931_SENDING_COMPLETE 0xa1 + ++extern int maintenance_service(struct pri *pri, int span, int channel, int changestatus); + ++extern int maintenance_service_ack(struct pri *pri, q931_call *call); ++ ++ + /* Q.SIG specific */ + #define QSIG_IE_TRANSIT_COUNT 0x31 + +@@ -268,7 +468,7 @@ + + extern int q931_information(struct pri *pri, q931_call *call, char digit); + +-extern int q931_keypad_facility(struct pri *pri, q931_call *call, char *digits); ++extern int q931_keypad_facility(struct pri *pri, q931_call *call, const char *digits); + + extern int q931_connect(struct pri *pri, q931_call *call, int channel, int nonisdn); + +@@ -291,8 +491,16 @@ + extern int q931_setup(struct pri *pri, q931_call *c, struct pri_sr *req); + extern void q931_dump(struct pri *pri, q931_h *h, int len, int txrx); + +-extern void __q931_destroycall(struct pri *pri, q931_call *c); ++void q931_destroycall(struct pri *pri, q931_call *c); + + extern void q931_dl_indication(struct pri *pri, int event); + ++int q931_send_hold(struct pri *ctrl, struct q931_call *call); ++int q931_send_hold_ack(struct pri *ctrl, struct q931_call *call); ++int q931_send_hold_rej(struct pri *ctrl, struct q931_call *call, int cause); ++ ++int q931_send_retrieve(struct pri *ctrl, struct q931_call *call, int channel); ++int q931_send_retrieve_ack(struct pri *ctrl, struct q931_call *call, int channel); ++int q931_send_retrieve_rej(struct pri *ctrl, struct q931_call *call, int cause); ++ + #endif +Index: rose_qsig_diversion.c +=================================================================== +--- a/rose_qsig_diversion.c (.../tags/1.4.10.2) (revision 0) ++++ b/rose_qsig_diversion.c (.../branches/1.4) (revision 1357) +@@ -0,0 +1,1390 @@ ++/* ++ * libpri: An implementation of Primary Rate ISDN ++ * ++ * Copyright (C) 2009 Digium, Inc. ++ * ++ * Richard Mudgett <rmudgett@digium.com> ++ * ++ * See http://www.asterisk.org for more information about ++ * the Asterisk project. Please do not directly contact ++ * any of the maintainers of this project for assistance; ++ * the project provides a web site, mailing lists and IRC ++ * channels for your use. ++ * ++ * This program is free software, distributed under the terms of ++ * the GNU General Public License Version 2 as published by the ++ * Free Software Foundation. See the LICENSE file included with ++ * this program for more details. ++ * ++ * In addition, when this program is distributed with Asterisk in ++ * any form that would qualify as a 'combined work' or as a ++ * 'derivative work' (but not mere aggregation), you can redistribute ++ * and/or modify the combination under the terms of the license ++ * provided with that copy of Asterisk, instead of the license ++ * terms granted here. ++ */ ++ ++/*! ++ * \file ++ * \brief Q.SIG ROSE Call-Diversion-Operations ++ * ++ * Call-Diversion-Operations ECMA-174 Annex F Table F.1 ++ * ++ * \author Richard Mudgett <rmudgett@digium.com> ++ */ ++ ++ ++#include "compat.h" ++#include "libpri.h" ++#include "pri_internal.h" ++#include "rose.h" ++#include "rose_internal.h" ++#include "asn1.h" ++ ++ ++/* ------------------------------------------------------------------- */ ++ ++/*! ++ * \internal ++ * \brief Encode the IntResult type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param tag Component tag to identify the encoded component. ++ * The tag should be ASN1_TAG_SEQUENCE unless the caller implicitly ++ * tags it otherwise. ++ * \param int_result Forwarding record information to encode. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *rose_enc_qsig_IntResult(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, unsigned tag, const struct roseQsigForwardingRecord *int_result) ++{ ++ unsigned char *seq_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, tag); ++ ++ ASN1_CALL(pos, rose_enc_PartyNumber(ctrl, pos, end, ++ &int_result->served_user_number)); ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, ++ int_result->basic_service)); ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, int_result->procedure)); ++ ASN1_CALL(pos, rose_enc_Address(ctrl, pos, end, ASN1_TAG_SEQUENCE, ++ &int_result->diverted_to)); ++ if (int_result->remote_enabled) { ++ /* Not the DEFAULT value */ ++ ASN1_CALL(pos, asn1_enc_boolean(pos, end, ASN1_TYPE_BOOLEAN, ++ int_result->remote_enabled)); ++ } ++ ++ /* No extension to encode */ ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Encode the IntResultList type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param tag Component tag to identify the encoded component. ++ * The tag should be ASN1_TAG_SET unless the caller implicitly ++ * tags it otherwise. ++ * \param int_result_list Forwarding record list information to encode. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *rose_enc_qsig_IntResultList(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, unsigned tag, ++ const struct roseQsigForwardingList *int_result_list) ++{ ++ unsigned index; ++ unsigned char *set_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(set_len, pos, end, tag); ++ ++ for (index = 0; index < int_result_list->num_records; ++index) { ++ ASN1_CALL(pos, rose_enc_qsig_IntResult(ctrl, pos, end, ASN1_TAG_SEQUENCE, ++ &int_result_list->list[index])); ++ } ++ ++ ASN1_CONSTRUCTED_END(set_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the ActivateDiversionQ invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_qsig_ActivateDiversionQ_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ const struct roseQsigActivateDiversionQ_ARG *activate_diversion_q; ++ unsigned char *seq_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ASN1_TAG_SEQUENCE); ++ ++ activate_diversion_q = &args->qsig.ActivateDiversionQ; ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, ++ activate_diversion_q->procedure)); ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, ++ activate_diversion_q->basic_service)); ++ ASN1_CALL(pos, rose_enc_Address(ctrl, pos, end, ASN1_TAG_SEQUENCE, ++ &activate_diversion_q->diverted_to)); ++ ASN1_CALL(pos, rose_enc_PartyNumber(ctrl, pos, end, ++ &activate_diversion_q->served_user_number)); ++ ASN1_CALL(pos, rose_enc_PartyNumber(ctrl, pos, end, ++ &activate_diversion_q->activating_user_number)); ++ ++ /* No extension to encode */ ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the DeactivateDiversionQ invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_qsig_DeactivateDiversionQ_ARG(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ const struct roseQsigDeactivateDiversionQ_ARG *deactivate_diversion_q; ++ unsigned char *seq_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ASN1_TAG_SEQUENCE); ++ ++ deactivate_diversion_q = &args->qsig.DeactivateDiversionQ; ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, ++ deactivate_diversion_q->procedure)); ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, ++ deactivate_diversion_q->basic_service)); ++ ASN1_CALL(pos, rose_enc_PartyNumber(ctrl, pos, end, ++ &deactivate_diversion_q->served_user_number)); ++ ASN1_CALL(pos, rose_enc_PartyNumber(ctrl, pos, end, ++ &deactivate_diversion_q->deactivating_user_number)); ++ ++ /* No extension to encode */ ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the InterrogateDiversionQ invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_qsig_InterrogateDiversionQ_ARG(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ const struct roseQsigInterrogateDiversionQ_ARG *interrogate_diversion_q; ++ unsigned char *seq_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ASN1_TAG_SEQUENCE); ++ ++ interrogate_diversion_q = &args->qsig.InterrogateDiversionQ; ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, ++ interrogate_diversion_q->procedure)); ++ if (interrogate_diversion_q->basic_service) { ++ /* Not the DEFAULT value */ ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, ++ interrogate_diversion_q->basic_service)); ++ } ++ ASN1_CALL(pos, rose_enc_PartyNumber(ctrl, pos, end, ++ &interrogate_diversion_q->served_user_number)); ++ ASN1_CALL(pos, rose_enc_PartyNumber(ctrl, pos, end, ++ &interrogate_diversion_q->interrogating_user_number)); ++ ++ /* No extension to encode */ ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the InterrogateDiversionQ result facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_qsig_InterrogateDiversionQ_RES(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, const union rose_msg_result_args *args) ++{ ++ return rose_enc_qsig_IntResultList(ctrl, pos, end, ASN1_TAG_SET, ++ &args->qsig.InterrogateDiversionQ); ++} ++ ++/*! ++ * \brief Encode the CheckRestriction invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_qsig_CheckRestriction_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ const struct roseQsigCheckRestriction_ARG *check_restriction; ++ unsigned char *seq_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ASN1_TAG_SEQUENCE); ++ ++ check_restriction = &args->qsig.CheckRestriction; ++ ASN1_CALL(pos, rose_enc_PartyNumber(ctrl, pos, end, ++ &check_restriction->served_user_number)); ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, ++ check_restriction->basic_service)); ++ ASN1_CALL(pos, rose_enc_PartyNumber(ctrl, pos, end, ++ &check_restriction->diverted_to_number)); ++ ++ /* No extension to encode */ ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the CallRerouting invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_qsig_CallRerouting_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ const struct roseQsigCallRerouting_ARG *call_rerouting; ++ unsigned char *seq_len; ++ unsigned char *exp_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ASN1_TAG_SEQUENCE); ++ ++ call_rerouting = &args->qsig.CallRerouting; ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, ++ call_rerouting->rerouting_reason)); ++ if (call_rerouting->original_rerouting_reason_present) { ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 0, ++ call_rerouting->original_rerouting_reason)); ++ } ++ ASN1_CALL(pos, rose_enc_Address(ctrl, pos, end, ASN1_TAG_SEQUENCE, ++ &call_rerouting->called)); ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_INTEGER, ++ call_rerouting->diversion_counter)); ++ ASN1_CALL(pos, rose_enc_Q931ie(ctrl, pos, end, ASN1_CLASS_APPLICATION | 0, ++ &call_rerouting->q931ie)); ++ ++ /* EXPLICIT tag */ ++ ASN1_CONSTRUCTED_BEGIN(exp_len, pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 1); ++ ASN1_CALL(pos, rose_enc_PresentedNumberUnscreened(ctrl, pos, end, ++ &call_rerouting->last_rerouting)); ++ ASN1_CONSTRUCTED_END(exp_len, pos, end); ++ ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 2, ++ call_rerouting->subscription_option)); ++ ++ if (call_rerouting->calling_subaddress.length) { ++ /* EXPLICIT tag */ ++ ASN1_CONSTRUCTED_BEGIN(exp_len, pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 3); ++ ASN1_CALL(pos, rose_enc_PartySubaddress(ctrl, pos, end, ++ &call_rerouting->calling_subaddress)); ++ ASN1_CONSTRUCTED_END(exp_len, pos, end); ++ } ++ ++ /* EXPLICIT tag */ ++ ASN1_CONSTRUCTED_BEGIN(exp_len, pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 4); ++ ASN1_CALL(pos, rose_enc_PresentedNumberScreened(ctrl, pos, end, ++ &call_rerouting->calling)); ++ ASN1_CONSTRUCTED_END(exp_len, pos, end); ++ ++ if (call_rerouting->calling_name_present) { ++ /* EXPLICIT tag */ ++ ASN1_CONSTRUCTED_BEGIN(exp_len, pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 5); ++ ASN1_CALL(pos, rose_enc_qsig_Name(ctrl, pos, end, ++ &call_rerouting->calling_name)); ++ ASN1_CONSTRUCTED_END(exp_len, pos, end); ++ } ++ ++ if (call_rerouting->original_called_present) { ++ /* EXPLICIT tag */ ++ ASN1_CONSTRUCTED_BEGIN(exp_len, pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 6); ++ ASN1_CALL(pos, rose_enc_PresentedNumberUnscreened(ctrl, pos, end, ++ &call_rerouting->original_called)); ++ ASN1_CONSTRUCTED_END(exp_len, pos, end); ++ } ++ ++ if (call_rerouting->redirecting_name_present) { ++ /* EXPLICIT tag */ ++ ASN1_CONSTRUCTED_BEGIN(exp_len, pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 7); ++ ASN1_CALL(pos, rose_enc_qsig_Name(ctrl, pos, end, ++ &call_rerouting->redirecting_name)); ++ ASN1_CONSTRUCTED_END(exp_len, pos, end); ++ } ++ ++ if (call_rerouting->original_called_name_present) { ++ /* EXPLICIT tag */ ++ ASN1_CONSTRUCTED_BEGIN(exp_len, pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 8); ++ ASN1_CALL(pos, rose_enc_qsig_Name(ctrl, pos, end, ++ &call_rerouting->original_called_name)); ++ ASN1_CONSTRUCTED_END(exp_len, pos, end); ++ } ++ ++ /* No extension to encode */ ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the DivertingLegInformation1 invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_qsig_DivertingLegInformation1_ARG(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ const struct roseQsigDivertingLegInformation1_ARG *diverting_leg_information_1; ++ unsigned char *seq_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ASN1_TAG_SEQUENCE); ++ ++ diverting_leg_information_1 = &args->qsig.DivertingLegInformation1; ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, ++ diverting_leg_information_1->diversion_reason)); ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, ++ diverting_leg_information_1->subscription_option)); ++ ASN1_CALL(pos, rose_enc_PartyNumber(ctrl, pos, end, ++ &diverting_leg_information_1->nominated_number)); ++ ++ /* No extension to encode */ ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the DivertingLegInformation2 invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_qsig_DivertingLegInformation2_ARG(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ const struct roseQsigDivertingLegInformation2_ARG *diverting_leg_information_2; ++ unsigned char *seq_len; ++ unsigned char *exp_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ASN1_TAG_SEQUENCE); ++ ++ diverting_leg_information_2 = &args->qsig.DivertingLegInformation2; ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_INTEGER, ++ diverting_leg_information_2->diversion_counter)); ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, ++ diverting_leg_information_2->diversion_reason)); ++ if (diverting_leg_information_2->original_diversion_reason_present) { ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 0, ++ diverting_leg_information_2->original_diversion_reason)); ++ } ++ ++ if (diverting_leg_information_2->diverting_present) { ++ /* EXPLICIT tag */ ++ ASN1_CONSTRUCTED_BEGIN(exp_len, pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 1); ++ ASN1_CALL(pos, rose_enc_PresentedNumberUnscreened(ctrl, pos, end, ++ &diverting_leg_information_2->diverting)); ++ ASN1_CONSTRUCTED_END(exp_len, pos, end); ++ } ++ ++ if (diverting_leg_information_2->original_called_present) { ++ /* EXPLICIT tag */ ++ ASN1_CONSTRUCTED_BEGIN(exp_len, pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 2); ++ ASN1_CALL(pos, rose_enc_PresentedNumberUnscreened(ctrl, pos, end, ++ &diverting_leg_information_2->original_called)); ++ ASN1_CONSTRUCTED_END(exp_len, pos, end); ++ } ++ ++ if (diverting_leg_information_2->redirecting_name_present) { ++ /* EXPLICIT tag */ ++ ASN1_CONSTRUCTED_BEGIN(exp_len, pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 3); ++ ASN1_CALL(pos, rose_enc_qsig_Name(ctrl, pos, end, ++ &diverting_leg_information_2->redirecting_name)); ++ ASN1_CONSTRUCTED_END(exp_len, pos, end); ++ } ++ ++ if (diverting_leg_information_2->original_called_name_present) { ++ /* EXPLICIT tag */ ++ ASN1_CONSTRUCTED_BEGIN(exp_len, pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 4); ++ ASN1_CALL(pos, rose_enc_qsig_Name(ctrl, pos, end, ++ &diverting_leg_information_2->original_called_name)); ++ ASN1_CONSTRUCTED_END(exp_len, pos, end); ++ } ++ ++ /* No extension to encode */ ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the DivertingLegInformation3 invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_qsig_DivertingLegInformation3_ARG(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ const struct roseQsigDivertingLegInformation3_ARG *diverting_leg_information_3; ++ unsigned char *seq_len; ++ unsigned char *exp_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ASN1_TAG_SEQUENCE); ++ ++ diverting_leg_information_3 = &args->qsig.DivertingLegInformation3; ++ ASN1_CALL(pos, asn1_enc_boolean(pos, end, ASN1_TYPE_BOOLEAN, ++ diverting_leg_information_3->presentation_allowed_indicator)); ++ ++ if (diverting_leg_information_3->redirection_name_present) { ++ /* EXPLICIT tag */ ++ ASN1_CONSTRUCTED_BEGIN(exp_len, pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 0); ++ ASN1_CALL(pos, rose_enc_qsig_Name(ctrl, pos, end, ++ &diverting_leg_information_3->redirection_name)); ++ ASN1_CONSTRUCTED_END(exp_len, pos, end); ++ } ++ ++ /* No extension to encode */ ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Decode the IntResult argument parameters. ++ * ++ * \param ctrl D channel controller for any diagnostic messages. ++ * \param name Field name ++ * \param tag Component tag that identified this production. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param int_result Parameter storage to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++static const unsigned char *rose_dec_qsig_IntResult(struct pri *ctrl, const char *name, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct roseQsigForwardingRecord *int_result) ++{ ++ int32_t value; ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ const unsigned char *save_pos; ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s IntResult %s\n", name, asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CALL(pos, rose_dec_PartyNumber(ctrl, "servedUserNr", tag, pos, seq_end, ++ &int_result->served_user_number)); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_ENUMERATED); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "basicService", tag, pos, seq_end, &value)); ++ int_result->basic_service = value; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_ENUMERATED); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "procedure", tag, pos, seq_end, &value)); ++ int_result->procedure = value; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ ASN1_CALL(pos, rose_dec_Address(ctrl, "divertedToAddress", tag, pos, seq_end, ++ &int_result->diverted_to)); ++ ++ /* ++ * A sequence specifies an ordered list of component types. ++ * However, for simplicity we are not checking the order of ++ * the remaining optional components. ++ */ ++ int_result->remote_enabled = 0; /* DEFAULT FALSE */ ++ while (pos < seq_end && *pos != ASN1_INDEF_TERM) { ++ save_pos = pos; ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ switch (tag & ~ASN1_PC_MASK) { ++ case ASN1_TYPE_BOOLEAN: ++ /* Must not be constructed but we will not check for it for simplicity. */ ++ ASN1_CALL(pos, asn1_dec_boolean(ctrl, "remoteEnabled", tag, pos, seq_end, ++ &value)); ++ int_result->remote_enabled = value; ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 1: ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 2: ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " extension %s\n", asn1_tag2str(tag)); ++ } ++ /* Fixup will skip over the manufacturer extension information */ ++ default: ++ pos = save_pos; ++ goto cancel_options; ++ } ++ } ++cancel_options:; ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Decode the IntResultList argument parameters. ++ * ++ * \param ctrl D channel controller for any diagnostic messages. ++ * \param name Field name ++ * \param tag Component tag that identified this production. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param int_result_list Parameter storage to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++static const unsigned char *rose_dec_qsig_IntResultList(struct pri *ctrl, ++ const char *name, unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct roseQsigForwardingList *int_result_list) ++{ ++ int length; ++ int set_offset; ++ const unsigned char *set_end; ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s IntResultList %s\n", name, asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(set_end, set_offset, length, pos, end); ++ ++ int_result_list->num_records = 0; ++ while (pos < set_end && *pos != ASN1_INDEF_TERM) { ++ if (int_result_list->num_records < ARRAY_LEN(int_result_list->list)) { ++ ASN1_CALL(pos, asn1_dec_tag(pos, set_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ ASN1_CALL(pos, rose_dec_qsig_IntResult(ctrl, "listEntry", tag, pos, set_end, ++ &int_result_list->list[int_result_list->num_records])); ++ ++int_result_list->num_records; ++ } else { ++ /* Too many records */ ++ return NULL; ++ } ++ } ++ ++ ASN1_END_FIXUP(ctrl, pos, set_offset, set_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the Q.SIG ActivateDiversionQ invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_qsig_ActivateDiversionQ_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, union rose_msg_invoke_args *args) ++{ ++ int32_t value; ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ struct roseQsigActivateDiversionQ_ARG *activate_diversion_q; ++ ++ activate_diversion_q = &args->qsig.ActivateDiversionQ; ++ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " ActivateDiversionQ %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_ENUMERATED); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "procedure", tag, pos, seq_end, &value)); ++ activate_diversion_q->procedure = value; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_ENUMERATED); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "basicService", tag, pos, seq_end, &value)); ++ activate_diversion_q->basic_service = value; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ ASN1_CALL(pos, rose_dec_Address(ctrl, "divertedToAddress", tag, pos, seq_end, ++ &activate_diversion_q->diverted_to)); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CALL(pos, rose_dec_PartyNumber(ctrl, "servedUserNr", tag, pos, seq_end, ++ &activate_diversion_q->served_user_number)); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CALL(pos, rose_dec_PartyNumber(ctrl, "activatingUserNr", tag, pos, seq_end, ++ &activate_diversion_q->activating_user_number)); ++ ++ /* Fixup will skip over any OPTIONAL manufacturer extension information */ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the Q.SIG DeactivateDiversionQ invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_qsig_DeactivateDiversionQ_ARG(struct pri *ctrl, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args) ++{ ++ int32_t value; ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ struct roseQsigDeactivateDiversionQ_ARG *deactivate_diversion_q; ++ ++ deactivate_diversion_q = &args->qsig.DeactivateDiversionQ; ++ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " DeactivateDiversionQ %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_ENUMERATED); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "procedure", tag, pos, seq_end, &value)); ++ deactivate_diversion_q->procedure = value; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_ENUMERATED); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "basicService", tag, pos, seq_end, &value)); ++ deactivate_diversion_q->basic_service = value; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CALL(pos, rose_dec_PartyNumber(ctrl, "servedUserNr", tag, pos, seq_end, ++ &deactivate_diversion_q->served_user_number)); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CALL(pos, rose_dec_PartyNumber(ctrl, "deactivatingUserNr", tag, pos, seq_end, ++ &deactivate_diversion_q->deactivating_user_number)); ++ ++ /* Fixup will skip over any OPTIONAL manufacturer extension information */ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the Q.SIG InterrogateDiversionQ invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_qsig_InterrogateDiversionQ_ARG(struct pri *ctrl, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args) ++{ ++ int32_t value; ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ struct roseQsigInterrogateDiversionQ_ARG *interrogate_diversion_q; ++ ++ interrogate_diversion_q = &args->qsig.InterrogateDiversionQ; ++ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " InterrogateDiversionQ %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_ENUMERATED); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "procedure", tag, pos, seq_end, &value)); ++ interrogate_diversion_q->procedure = value; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ if (tag == ASN1_TYPE_ENUMERATED) { ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "basicService", tag, pos, seq_end, &value)); ++ interrogate_diversion_q->basic_service = value; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ } else { ++ interrogate_diversion_q->basic_service = 0; /* allServices */ ++ } ++ ++ ASN1_CALL(pos, rose_dec_PartyNumber(ctrl, "servedUserNr", tag, pos, seq_end, ++ &interrogate_diversion_q->served_user_number)); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CALL(pos, rose_dec_PartyNumber(ctrl, "interrogatingUserNr", tag, pos, seq_end, ++ &interrogate_diversion_q->interrogating_user_number)); ++ ++ /* Fixup will skip over any OPTIONAL manufacturer extension information */ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the Q.SIG InterrogateDiversionQ result argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_qsig_InterrogateDiversionQ_RES(struct pri *ctrl, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ union rose_msg_result_args *args) ++{ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SET); ++ return rose_dec_qsig_IntResultList(ctrl, "InterrogateDiversionQ", tag, pos, end, ++ &args->qsig.InterrogateDiversionQ); ++} ++ ++/*! ++ * \brief Decode the Q.SIG CheckRestriction invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_qsig_CheckRestriction_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, union rose_msg_invoke_args *args) ++{ ++ int32_t value; ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ struct roseQsigCheckRestriction_ARG *check_restriction; ++ ++ check_restriction = &args->qsig.CheckRestriction; ++ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " CheckRestriction %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CALL(pos, rose_dec_PartyNumber(ctrl, "servedUserNr", tag, pos, seq_end, ++ &check_restriction->served_user_number)); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_ENUMERATED); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "basicService", tag, pos, seq_end, &value)); ++ check_restriction->basic_service = value; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CALL(pos, rose_dec_PartyNumber(ctrl, "divertedToNr", tag, pos, seq_end, ++ &check_restriction->diverted_to_number)); ++ ++ /* Fixup will skip over any OPTIONAL manufacturer extension information */ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the Q.SIG CallRerouting invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_qsig_CallRerouting_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, union rose_msg_invoke_args *args) ++{ ++ int32_t value; ++ int length; ++ int seq_offset; ++ int explicit_offset; ++ const unsigned char *explicit_end; ++ const unsigned char *seq_end; ++ const unsigned char *save_pos; ++ struct roseQsigCallRerouting_ARG *call_rerouting; ++ ++ call_rerouting = &args->qsig.CallRerouting; ++ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " CallRerouting %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_ENUMERATED); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "reroutingReason", tag, pos, seq_end, &value)); ++ call_rerouting->rerouting_reason = value; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ if (tag == (ASN1_CLASS_CONTEXT_SPECIFIC | 0)) { ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "originalReroutingReason", tag, pos, seq_end, ++ &value)); ++ call_rerouting->original_rerouting_reason = value; ++ call_rerouting->original_rerouting_reason_present = 1; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ } else { ++ call_rerouting->original_rerouting_reason_present = 0; ++ } ++ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ ASN1_CALL(pos, rose_dec_Address(ctrl, "calledAddress", tag, pos, seq_end, ++ &call_rerouting->called)); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_INTEGER); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "diversionCounter", tag, pos, seq_end, &value)); ++ call_rerouting->diversion_counter = value; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag & ~ASN1_PC_MASK, ASN1_CLASS_APPLICATION | 0); ++ ASN1_CALL(pos, rose_dec_Q931ie(ctrl, "pSS1InfoElement", tag, pos, seq_end, ++ &call_rerouting->q931ie, sizeof(call_rerouting->q931ie_contents))); ++ ++ /* Remove EXPLICIT tag */ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ++ ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 1); ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " Explicit %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, seq_end, &length)); ++ ASN1_END_SETUP(explicit_end, explicit_offset, length, pos, seq_end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, explicit_end, &tag)); ++ ASN1_CALL(pos, rose_dec_PresentedNumberUnscreened(ctrl, "lastReroutingNr", tag, pos, ++ explicit_end, &call_rerouting->last_rerouting)); ++ ++ ASN1_END_FIXUP(ctrl, pos, explicit_offset, explicit_end, seq_end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_CLASS_CONTEXT_SPECIFIC | 2); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "subscriptionOption", tag, pos, seq_end, &value)); ++ call_rerouting->subscription_option = value; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ if (tag == (ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 3)) { ++ /* Remove EXPLICIT tag */ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " Explicit %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, seq_end, &length)); ++ ASN1_END_SETUP(explicit_end, explicit_offset, length, pos, seq_end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, explicit_end, &tag)); ++ ASN1_CALL(pos, rose_dec_PartySubaddress(ctrl, "callingPartySubaddress", tag, pos, ++ explicit_end, &call_rerouting->calling_subaddress)); ++ ++ ASN1_END_FIXUP(ctrl, pos, explicit_offset, explicit_end, seq_end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ } else { ++ call_rerouting->calling_subaddress.length = 0; ++ } ++ ++ /* Remove EXPLICIT tag */ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ++ ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 4); ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " Explicit %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, seq_end, &length)); ++ ASN1_END_SETUP(explicit_end, explicit_offset, length, pos, seq_end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, explicit_end, &tag)); ++ ASN1_CALL(pos, rose_dec_PresentedNumberScreened(ctrl, "callingNumber", tag, pos, ++ explicit_end, &call_rerouting->calling)); ++ ++ ASN1_END_FIXUP(ctrl, pos, explicit_offset, explicit_end, seq_end); ++ ++ /* ++ * A sequence specifies an ordered list of component types. ++ * However, for simplicity we are not checking the order of ++ * the remaining optional components. ++ */ ++ call_rerouting->calling_name_present = 0; ++ call_rerouting->redirecting_name_present = 0; ++ call_rerouting->original_called_name_present = 0; ++ call_rerouting->original_called_present = 0; ++ while (pos < seq_end && *pos != ASN1_INDEF_TERM) { ++ save_pos = pos; ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ switch (tag) { ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 5: ++ /* Remove EXPLICIT tag */ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " Explicit %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, seq_end, &length)); ++ ASN1_END_SETUP(explicit_end, explicit_offset, length, pos, seq_end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, explicit_end, &tag)); ++ ASN1_CALL(pos, rose_dec_qsig_Name(ctrl, "callingName", tag, pos, ++ explicit_end, &call_rerouting->calling_name)); ++ call_rerouting->calling_name_present = 1; ++ ++ ASN1_END_FIXUP(ctrl, pos, explicit_offset, explicit_end, seq_end); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 6: ++ /* Remove EXPLICIT tag */ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " Explicit %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, seq_end, &length)); ++ ASN1_END_SETUP(explicit_end, explicit_offset, length, pos, seq_end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, explicit_end, &tag)); ++ ASN1_CALL(pos, rose_dec_PresentedNumberUnscreened(ctrl, "originalCalledNr", ++ tag, pos, explicit_end, &call_rerouting->original_called)); ++ call_rerouting->original_called_present = 1; ++ ++ ASN1_END_FIXUP(ctrl, pos, explicit_offset, explicit_end, seq_end); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 7: ++ /* Remove EXPLICIT tag */ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " Explicit %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, seq_end, &length)); ++ ASN1_END_SETUP(explicit_end, explicit_offset, length, pos, seq_end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, explicit_end, &tag)); ++ ASN1_CALL(pos, rose_dec_qsig_Name(ctrl, "redirectingName", tag, pos, ++ explicit_end, &call_rerouting->redirecting_name)); ++ call_rerouting->redirecting_name_present = 1; ++ ++ ASN1_END_FIXUP(ctrl, pos, explicit_offset, explicit_end, seq_end); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 8: ++ /* Remove EXPLICIT tag */ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " Explicit %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, seq_end, &length)); ++ ASN1_END_SETUP(explicit_end, explicit_offset, length, pos, seq_end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, explicit_end, &tag)); ++ ASN1_CALL(pos, rose_dec_qsig_Name(ctrl, "originalCalledName", tag, pos, ++ explicit_end, &call_rerouting->original_called_name)); ++ call_rerouting->original_called_name_present = 1; ++ ++ ASN1_END_FIXUP(ctrl, pos, explicit_offset, explicit_end, seq_end); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 9: ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 9: ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 10: ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " extension %s\n", asn1_tag2str(tag)); ++ } ++ /* Fixup will skip over the manufacturer extension information */ ++ default: ++ pos = save_pos; ++ goto cancel_options; ++ } ++ } ++cancel_options:; ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the Q.SIG DivertingLegInformation1 invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_qsig_DivertingLegInformation1_ARG(struct pri *ctrl, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args) ++{ ++ int32_t value; ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ struct roseQsigDivertingLegInformation1_ARG *diverting_leg_information_1; ++ ++ diverting_leg_information_1 = &args->qsig.DivertingLegInformation1; ++ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " DivertingLegInformation1 %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_ENUMERATED); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "diversionReason", tag, pos, seq_end, &value)); ++ diverting_leg_information_1->diversion_reason = value; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_ENUMERATED); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "subscriptionOption", tag, pos, seq_end, &value)); ++ diverting_leg_information_1->subscription_option = value; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CALL(pos, rose_dec_PartyNumber(ctrl, "nominatedNr", tag, pos, seq_end, ++ &diverting_leg_information_1->nominated_number)); ++ ++ /* Fixup will skip over any OPTIONAL manufacturer extension information */ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the Q.SIG DivertingLegInformation2 invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_qsig_DivertingLegInformation2_ARG(struct pri *ctrl, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args) ++{ ++ int32_t value; ++ int length; ++ int seq_offset; ++ int explicit_offset; ++ const unsigned char *explicit_end; ++ const unsigned char *seq_end; ++ const unsigned char *save_pos; ++ struct roseQsigDivertingLegInformation2_ARG *diverting_leg_information_2; ++ ++ diverting_leg_information_2 = &args->qsig.DivertingLegInformation2; ++ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " DivertingLegInformation2 %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_INTEGER); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "diversionCounter", tag, pos, seq_end, &value)); ++ diverting_leg_information_2->diversion_counter = value; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_ENUMERATED); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "diversionReason", tag, pos, seq_end, &value)); ++ diverting_leg_information_2->diversion_reason = value; ++ ++ /* ++ * A sequence specifies an ordered list of component types. ++ * However, for simplicity we are not checking the order of ++ * the remaining optional components. ++ */ ++ diverting_leg_information_2->original_diversion_reason_present = 0; ++ diverting_leg_information_2->diverting_present = 0; ++ diverting_leg_information_2->original_called_present = 0; ++ diverting_leg_information_2->redirecting_name_present = 0; ++ diverting_leg_information_2->original_called_name_present = 0; ++ while (pos < seq_end && *pos != ASN1_INDEF_TERM) { ++ save_pos = pos; ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ switch (tag) { ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 0: ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "originalDiversionReason", tag, pos, ++ seq_end, &value)); ++ diverting_leg_information_2->original_diversion_reason = value; ++ diverting_leg_information_2->original_diversion_reason_present = 1; ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 1: ++ /* Remove EXPLICIT tag */ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " Explicit %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, seq_end, &length)); ++ ASN1_END_SETUP(explicit_end, explicit_offset, length, pos, seq_end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, explicit_end, &tag)); ++ ASN1_CALL(pos, rose_dec_PresentedNumberUnscreened(ctrl, "divertingNr", tag, ++ pos, explicit_end, &diverting_leg_information_2->diverting)); ++ diverting_leg_information_2->diverting_present = 1; ++ ++ ASN1_END_FIXUP(ctrl, pos, explicit_offset, explicit_end, seq_end); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 2: ++ /* Remove EXPLICIT tag */ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " Explicit %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, seq_end, &length)); ++ ASN1_END_SETUP(explicit_end, explicit_offset, length, pos, seq_end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, explicit_end, &tag)); ++ ASN1_CALL(pos, rose_dec_PresentedNumberUnscreened(ctrl, "originalCalledNr", ++ tag, pos, explicit_end, &diverting_leg_information_2->original_called)); ++ diverting_leg_information_2->original_called_present = 1; ++ ++ ASN1_END_FIXUP(ctrl, pos, explicit_offset, explicit_end, seq_end); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 3: ++ /* Remove EXPLICIT tag */ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " Explicit %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, seq_end, &length)); ++ ASN1_END_SETUP(explicit_end, explicit_offset, length, pos, seq_end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, explicit_end, &tag)); ++ ASN1_CALL(pos, rose_dec_qsig_Name(ctrl, "redirectingName", tag, pos, ++ explicit_end, &diverting_leg_information_2->redirecting_name)); ++ diverting_leg_information_2->redirecting_name_present = 1; ++ ++ ASN1_END_FIXUP(ctrl, pos, explicit_offset, explicit_end, seq_end); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 4: ++ /* Remove EXPLICIT tag */ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " Explicit %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, seq_end, &length)); ++ ASN1_END_SETUP(explicit_end, explicit_offset, length, pos, seq_end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, explicit_end, &tag)); ++ ASN1_CALL(pos, rose_dec_qsig_Name(ctrl, "originalCalledName", tag, pos, ++ explicit_end, &diverting_leg_information_2->original_called_name)); ++ diverting_leg_information_2->original_called_name_present = 1; ++ ++ ASN1_END_FIXUP(ctrl, pos, explicit_offset, explicit_end, seq_end); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 5: ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 5: ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 6: ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " extension %s\n", asn1_tag2str(tag)); ++ } ++ /* Fixup will skip over the manufacturer extension information */ ++ default: ++ pos = save_pos; ++ goto cancel_options; ++ } ++ } ++cancel_options:; ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the Q.SIG DivertingLegInformation3 invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_qsig_DivertingLegInformation3_ARG(struct pri *ctrl, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args) ++{ ++ int32_t value; ++ int length; ++ int seq_offset; ++ int explicit_offset; ++ const unsigned char *explicit_end; ++ const unsigned char *seq_end; ++ const unsigned char *save_pos; ++ struct roseQsigDivertingLegInformation3_ARG *diverting_leg_information_3; ++ ++ diverting_leg_information_3 = &args->qsig.DivertingLegInformation3; ++ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " DivertingLegInformation3 %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_BOOLEAN); ++ ASN1_CALL(pos, asn1_dec_boolean(ctrl, "presentationAllowedIndicator", tag, pos, ++ seq_end, &value)); ++ diverting_leg_information_3->presentation_allowed_indicator = value; ++ ++ /* ++ * A sequence specifies an ordered list of component types. ++ * However, for simplicity we are not checking the order of ++ * the remaining optional components. ++ */ ++ diverting_leg_information_3->redirection_name_present = 0; ++ while (pos < seq_end && *pos != ASN1_INDEF_TERM) { ++ save_pos = pos; ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ switch (tag) { ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 0: ++ /* Remove EXPLICIT tag */ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " Explicit %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, seq_end, &length)); ++ ASN1_END_SETUP(explicit_end, explicit_offset, length, pos, seq_end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, explicit_end, &tag)); ++ ASN1_CALL(pos, rose_dec_qsig_Name(ctrl, "redirectionName", tag, pos, ++ explicit_end, &diverting_leg_information_3->redirection_name)); ++ diverting_leg_information_3->redirection_name_present = 1; ++ ++ ASN1_END_FIXUP(ctrl, pos, explicit_offset, explicit_end, seq_end); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 1: ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 1: ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 2: ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " extension %s\n", asn1_tag2str(tag)); ++ } ++ /* Fixup will skip over the manufacturer extension information */ ++ default: ++ pos = save_pos; ++ goto cancel_options; ++ } ++ } ++cancel_options:; ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/* ------------------------------------------------------------------- */ ++/* end rose_qsig_diversion.c */ + +Property changes on: rose_qsig_diversion.c +___________________________________________________________________ +Added: svn:eol-style + + native +Added: svn:mime-type + + text/plain +Added: svn:keywords + + 'Author Date Id Revision' + +Index: rose.h +=================================================================== +--- a/rose.h (.../tags/1.4.10.2) (revision 0) ++++ b/rose.h (.../branches/1.4) (revision 1357) +@@ -0,0 +1,3580 @@ ++/* ++ * libpri: An implementation of Primary Rate ISDN ++ * ++ * Copyright (C) 2009 Digium, Inc. ++ * ++ * Richard Mudgett <rmudgett@digium.com> ++ * ++ * See http://www.asterisk.org for more information about ++ * the Asterisk project. Please do not directly contact ++ * any of the maintainers of this project for assistance; ++ * the project provides a web site, mailing lists and IRC ++ * channels for your use. ++ * ++ * This program is free software, distributed under the terms of ++ * the GNU General Public License Version 2 as published by the ++ * Free Software Foundation. See the LICENSE file included with ++ * this program for more details. ++ * ++ * In addition, when this program is distributed with Asterisk in ++ * any form that would qualify as a 'combined work' or as a ++ * 'derivative work' (but not mere aggregation), you can redistribute ++ * and/or modify the combination under the terms of the license ++ * provided with that copy of Asterisk, instead of the license ++ * terms granted here. ++ */ ++ ++/*! ++ * \file ++ * \brief ROSE definitions and prototypes ++ * ++ * \details ++ * This file contains all of the data structures and definitions needed ++ * for ROSE component encoding and decoding. ++ * ++ * ROSE - Remote Operations Service Element ++ * ASN.1 - Abstract Syntax Notation 1 ++ * APDU - Application Protocol Data Unit ++ * ++ * \author Richard Mudgett <rmudgett@digium.com> ++ */ ++ ++#ifndef _LIBPRI_ROSE_H ++#define _LIBPRI_ROSE_H ++ ++#include <string.h> ++#include <sys/types.h> ++ ++#ifdef __cplusplus ++extern "C" { ++#endif ++ ++/* ------------------------------------------------------------------- */ ++ ++ ++/* Northern Telecom DMS-100 RLT related operations */ ++#define ROSE_DMS100_RLT_SERVICE_ID 0x3e ++#define ROSE_DMS100_RLT_OPERATION_IND 0x01 ++#define ROSE_DMS100_RLT_THIRD_PARTY 0x02 ++ ++/*! \brief ROSE operation-value function code */ ++enum rose_operation { ++ /*! \brief No ROSE operation */ ++ ROSE_None, ++ /*! \brief Unknown OID/localValue operation-value code */ ++ ROSE_Unknown, ++ ++/* *INDENT-OFF* */ ++ /* ETSI Diversion-Operations */ ++ ROSE_ETSI_ActivationDiversion, /*!< Invoke/Result */ ++ ROSE_ETSI_DeactivationDiversion, /*!< Invoke/Result */ ++ ROSE_ETSI_ActivationStatusNotificationDiv,/*!< Invoke only */ ++ ROSE_ETSI_DeactivationStatusNotificationDiv,/*!< Invoke only */ ++ ROSE_ETSI_InterrogationDiversion, /*!< Invoke/Result */ ++ ROSE_ETSI_DiversionInformation, /*!< Invoke only */ ++ ROSE_ETSI_CallDeflection, /*!< Invoke/Result */ ++ ROSE_ETSI_CallRerouting, /*!< Invoke/Result */ ++ ROSE_ETSI_InterrogateServedUserNumbers, /*!< Invoke/Result */ ++ ROSE_ETSI_DivertingLegInformation1, /*!< Invoke only */ ++ ROSE_ETSI_DivertingLegInformation2, /*!< Invoke only */ ++ ROSE_ETSI_DivertingLegInformation3, /*!< Invoke only */ ++ ++ /* ++ * ETSI Advice-of-Charge-Operations ++ * ++ * Advice-Of-Charge-at-call-Setup(AOCS) ++ * Advice-Of-Charge-During-the-call(AOCD) ++ * Advice-Of-Charge-at-the-End-of-the-call(AOCE) ++ */ ++ ROSE_ETSI_ChargingRequest, /*!< Invoke/Result */ ++ ROSE_ETSI_AOCSCurrency, /*!< Invoke only */ ++ ROSE_ETSI_AOCSSpecialArr, /*!< Invoke only */ ++ ROSE_ETSI_AOCDCurrency, /*!< Invoke only */ ++ ROSE_ETSI_AOCDChargingUnit, /*!< Invoke only */ ++ ROSE_ETSI_AOCECurrency, /*!< Invoke only */ ++ ROSE_ETSI_AOCEChargingUnit, /*!< Invoke only */ ++ ++ /* ETSI Explicit-Call-Transfer-Operations-and-Errors */ ++ ROSE_ETSI_EctExecute, /*!< Invoke/Result */ ++ ROSE_ETSI_ExplicitEctExecute, /*!< Invoke/Result */ ++ ROSE_ETSI_RequestSubaddress, /*!< Invoke only */ ++ ROSE_ETSI_SubaddressTransfer, /*!< Invoke only */ ++ ROSE_ETSI_EctLinkIdRequest, /*!< Invoke/Result */ ++ ROSE_ETSI_EctInform, /*!< Invoke only */ ++ ROSE_ETSI_EctLoopTest, /*!< Invoke/Result */ ++ ++ /* Q.SIG Name-Operations */ ++ ROSE_QSIG_CallingName, /*!< Invoke only */ ++ ROSE_QSIG_CalledName, /*!< Invoke only */ ++ ROSE_QSIG_ConnectedName, /*!< Invoke only */ ++ ROSE_QSIG_BusyName, /*!< Invoke only */ ++ ++ /* Q.SIG SS-AOC-Operations */ ++ ROSE_QSIG_ChargeRequest, /*!< Invoke/Result */ ++ ROSE_QSIG_GetFinalCharge, /*!< Invoke only */ ++ ROSE_QSIG_AocFinal, /*!< Invoke only */ ++ ROSE_QSIG_AocInterim, /*!< Invoke only */ ++ ROSE_QSIG_AocRate, /*!< Invoke only */ ++ ROSE_QSIG_AocComplete, /*!< Invoke/Result */ ++ ROSE_QSIG_AocDivChargeReq, /*!< Invoke only */ ++ ++ /* Q.SIG Call-Transfer-Operations (CT) */ ++ ROSE_QSIG_CallTransferIdentify, /*!< Invoke/Result */ ++ ROSE_QSIG_CallTransferAbandon, /*!< Invoke only */ ++ ROSE_QSIG_CallTransferInitiate, /*!< Invoke/Result */ ++ ROSE_QSIG_CallTransferSetup, /*!< Invoke/Result */ ++ ROSE_QSIG_CallTransferActive, /*!< Invoke only */ ++ ROSE_QSIG_CallTransferComplete, /*!< Invoke only */ ++ ROSE_QSIG_CallTransferUpdate, /*!< Invoke only */ ++ ROSE_QSIG_SubaddressTransfer, /*!< Invoke only */ ++ ++ ROSE_QSIG_PathReplacement, /*!< Invoke only */ ++ ++ /* Q.SIG Call-Diversion-Operations */ ++ ROSE_QSIG_ActivateDiversionQ, /*!< Invoke/Result */ ++ ROSE_QSIG_DeactivateDiversionQ, /*!< Invoke/Result */ ++ ROSE_QSIG_InterrogateDiversionQ, /*!< Invoke/Result */ ++ ROSE_QSIG_CheckRestriction, /*!< Invoke/Result */ ++ ROSE_QSIG_CallRerouting, /*!< Invoke/Result */ ++ ROSE_QSIG_DivertingLegInformation1, /*!< Invoke only */ ++ ROSE_QSIG_DivertingLegInformation2, /*!< Invoke only */ ++ ROSE_QSIG_DivertingLegInformation3, /*!< Invoke only */ ++ ROSE_QSIG_CfnrDivertedLegFailed, /*!< Invoke only */ ++ ++ /* Q.SIG SS-MWI-Operations */ ++ ROSE_QSIG_MWIActivate, /*!< Invoke/Result */ ++ ROSE_QSIG_MWIDeactivate, /*!< Invoke/Result */ ++ ROSE_QSIG_MWIInterrogate, /*!< Invoke/Result */ ++ ++ /* Northern Telecom DMS-100 RLT related operations */ ++ /*! Invoke/Result: Must set invokeId to ROSE_DMS100_RLT_OPERATION_IND */ ++ ROSE_DMS100_RLT_OperationInd, ++ /*! Invoke/Result: Must set invokeId to ROSE_DMS100_RLT_THIRD_PARTY */ ++ ROSE_DMS100_RLT_ThirdParty, ++ ++ ROSE_NI2_InformationFollowing, /*!< Invoke only? */ ++ ROSE_NI2_InitiateTransfer, /*!< Invoke only? Is this correct operation name? */ ++ ++ ROSE_Num_Operation_Codes /*!< Must be last in the enumeration */ ++/* *INDENT-ON* */ ++}; ++ ++enum rose_error_code { ++ /*! \brief No error occurred */ ++ ROSE_ERROR_None, ++ /*! \brief Unknown OID/localValue error-value code */ ++ ROSE_ERROR_Unknown, ++ ++ /* General-Errors (ETS 300 196) and General-Error-List(Q.950) */ ++ ROSE_ERROR_Gen_NotSubscribed, /*!< also: UserNotSubscribed */ ++ ROSE_ERROR_Gen_NotAvailable, ++ ROSE_ERROR_Gen_NotImplemented, /*!< Not in Q.950 */ ++ ROSE_ERROR_Gen_InvalidServedUserNr, ++ ROSE_ERROR_Gen_InvalidCallState, ++ ROSE_ERROR_Gen_BasicServiceNotProvided, ++ ROSE_ERROR_Gen_NotIncomingCall, ++ ROSE_ERROR_Gen_SupplementaryServiceInteractionNotAllowed, ++ ROSE_ERROR_Gen_ResourceUnavailable, ++ ++ /* Additional General-Error-List(Q.950) */ ++ ROSE_ERROR_Gen_RejectedByNetwork, ++ ROSE_ERROR_Gen_RejectedByUser, ++ ROSE_ERROR_Gen_InsufficientInformation, ++ ROSE_ERROR_Gen_CallFailure, ++ ROSE_ERROR_Gen_ProceduralError, ++ ++ /* ETSI Diversion-Operations */ ++ ROSE_ERROR_Div_InvalidDivertedToNr, ++ ROSE_ERROR_Div_SpecialServiceNr, ++ ROSE_ERROR_Div_DiversionToServedUserNr, ++ ROSE_ERROR_Div_IncomingCallAccepted, ++ ROSE_ERROR_Div_NumberOfDiversionsExceeded, ++ ROSE_ERROR_Div_NotActivated, ++ ROSE_ERROR_Div_RequestAlreadyAccepted, ++ ++ /* ETSI Advice-of-Charge-Operations */ ++ ROSE_ERROR_AOC_NoChargingInfoAvailable, ++ ++ /* ETSI Explicit-Call-Transfer-Operations-and-Errors */ ++ ROSE_ERROR_ECT_LinkIdNotAssignedByNetwork, ++ ++ /* Q.SIG from various specifications */ ++ ROSE_ERROR_QSIG_Unspecified, ++ ++ /* Q.SIG SS-AOC-Operations */ ++ ROSE_ERROR_QSIG_AOC_FreeOfCharge, ++ ++ /* Q.SIG Call-Transfer-Operations (CT) */ ++ ROSE_ERROR_QSIG_CT_InvalidReroutingNumber, ++ ROSE_ERROR_QSIG_CT_UnrecognizedCallIdentity, ++ ROSE_ERROR_QSIG_CT_EstablishmentFailure, ++ ++ /* Q.SIG Call-Diversion-Operations (Additional Q.SIG specific errors) */ ++ ROSE_ERROR_QSIG_Div_TemporarilyUnavailable, ++ ROSE_ERROR_QSIG_Div_NotAuthorized, ++ ++ /* Q.SIG SS-MWI-Operations */ ++ ROSE_ERROR_QSIG_InvalidMsgCentreId, ++ ++ /* Northern Telecom DMS-100 RLT related operations */ ++ ROSE_ERROR_DMS100_RLT_BridgeFail, ++ ROSE_ERROR_DMS100_RLT_CallIDNotFound, ++ ROSE_ERROR_DMS100_RLT_NotAllowed, ++ ROSE_ERROR_DMS100_RLT_SwitchEquipCongs, ++ ++ ROSE_ERROR_Num_Codes /*!< Must be last in the enumeration */ ++}; ++ ++#define ROSE_REJECT_BASE(base) ((base) * 0x100) ++enum rose_reject_base { ++ ROSE_REJECT_BASE_General, ++ ROSE_REJECT_BASE_Invoke, ++ ROSE_REJECT_BASE_Result, ++ ROSE_REJECT_BASE_Error, ++ ++ /*! \brief Must be last in the list */ ++ ROSE_REJECT_BASE_Last ++}; ++ ++/*! ++ * \brief From Facility-Information-Element-Components ++ * {itu-t identified-organization etsi(0) 196 facility-information-element-component(3)} ++ */ ++enum rose_reject_code { ++ /*! \brief Not rejected */ ++ ROSE_REJECT_None = -1, ++ /*! \brief Unknown reject code */ ++ ROSE_REJECT_Unknown = -2, ++ ++/* *INDENT-OFF* */ ++ ROSE_REJECT_Gen_UnrecognizedComponent = ROSE_REJECT_BASE(ROSE_REJECT_BASE_General) + 0, ++ ROSE_REJECT_Gen_MistypedComponent = ROSE_REJECT_BASE(ROSE_REJECT_BASE_General) + 1, ++ ROSE_REJECT_Gen_BadlyStructuredComponent = ROSE_REJECT_BASE(ROSE_REJECT_BASE_General) + 2, ++ ++ ROSE_REJECT_Inv_DuplicateInvocation = ROSE_REJECT_BASE(ROSE_REJECT_BASE_Invoke) + 0, ++ ROSE_REJECT_Inv_UnrecognizedOperation = ROSE_REJECT_BASE(ROSE_REJECT_BASE_Invoke) + 1, ++ ROSE_REJECT_Inv_MistypedArgument = ROSE_REJECT_BASE(ROSE_REJECT_BASE_Invoke) + 2, ++ ROSE_REJECT_Inv_ResourceLimitation = ROSE_REJECT_BASE(ROSE_REJECT_BASE_Invoke) + 3, ++ ROSE_REJECT_Inv_InitiatorReleasing = ROSE_REJECT_BASE(ROSE_REJECT_BASE_Invoke) + 4, ++ ROSE_REJECT_Inv_UnrecognizedLinkedID = ROSE_REJECT_BASE(ROSE_REJECT_BASE_Invoke) + 5, ++ ROSE_REJECT_Inv_LinkedResponseUnexpected = ROSE_REJECT_BASE(ROSE_REJECT_BASE_Invoke) + 6, ++ ROSE_REJECT_Inv_UnexpectedChildOperation = ROSE_REJECT_BASE(ROSE_REJECT_BASE_Invoke) + 7, ++ ++ ROSE_REJECT_Res_UnrecognizedInvocation = ROSE_REJECT_BASE(ROSE_REJECT_BASE_Result) + 0, ++ ROSE_REJECT_Res_ResultResponseUnexpected = ROSE_REJECT_BASE(ROSE_REJECT_BASE_Result) + 1, ++ ROSE_REJECT_Res_MistypedResult = ROSE_REJECT_BASE(ROSE_REJECT_BASE_Result) + 2, ++ ++ ROSE_REJECT_Err_UnrecognizedInvocation = ROSE_REJECT_BASE(ROSE_REJECT_BASE_Error) + 0, ++ ROSE_REJECT_Err_ErrorResponseUnexpected = ROSE_REJECT_BASE(ROSE_REJECT_BASE_Error) + 1, ++ ROSE_REJECT_Err_UnrecognizedError = ROSE_REJECT_BASE(ROSE_REJECT_BASE_Error) + 2, ++ ROSE_REJECT_Err_UnexpectedError = ROSE_REJECT_BASE(ROSE_REJECT_BASE_Error) + 3, ++ ROSE_REJECT_Err_MistypedParameter = ROSE_REJECT_BASE(ROSE_REJECT_BASE_Error) + 4, ++/* *INDENT-ON* */ ++}; ++ ++ ++/* ------------------------------------------------------------------- */ ++ ++ ++/* ++ * Q931InformationElement ::= [APPLICATION 0] IMPLICIT OCTET STRING ++ */ ++struct roseQ931ie { ++ /*! ++ * \brief The Q.931 ie is present if length is nonzero. ++ * (If this field is optional in the message.) ++ */ ++ u_int8_t length; ++ ++ /*! ++ * \brief We mostly just need to store the contents so we will defer ++ * decoding/encoding. ++ * ++ * \note To reduce the size of the structure, the memory for the ++ * ie contents is "allocated" after the structure. ++ * \note Remember the "allocated" memory needs to have room for a ++ * null terminator. ++ */ ++ unsigned char contents[0]; ++}; ++ ++enum { ++ /*! Bearer Capability has a max length of 12. */ ++ ROSE_Q931_MAX_BC = 12, ++ /*! High Layer Compatibility has a max length of 5. */ ++ ROSE_Q931_MAX_HLC = 5, ++ /*! Low Layer Compatibility has a max length of 18. */ ++ ROSE_Q931_MAX_LLC = 18, ++ /*! ++ * User-User Information has a network dependent maximum. ++ * The network dependent maximum is either 35 or 131 octets ++ * in non-USER-INFORMATION messages. ++ */ ++ ROSE_Q931_MAX_USER = 131, ++ /*! ++ * Progress Indicator has a max length of 4. ++ * There can be multiple progress indicator ies. ++ * Q.SIG allows up to 3. ++ * ITU-T allows up to 2. ++ */ ++ ROSE_Q931_MAX_PROGRESS = 3 * 4, ++}; ++ ++ ++/* ------------------------------------------------------------------- */ ++ ++ ++/* ++ * PartyNumber ::= CHOICE { ++ * -- the numbering plan is the default numbering plan of ++ * -- the network. It is recommended that this value is ++ * -- used. ++ * unknownPartyNumber [0] IMPLICIT NumberDigits, ++ * ++ * -- the numbering plan is according to ++ * -- ITU-T Recommendation E.164. ++ * publicPartyNumber [1] IMPLICIT PublicPartyNumber, ++ * ++ * -- ATM endsystem address encoded as an NSAP address. ++ * nsapEncodedNumber [2] IMPLICIT NsapEncodedNumber, ++ * ++ * -- not used, value reserved. ++ * dataPartyNumber [3] IMPLICIT NumberDigits, ++ * ++ * -- not used, value reserved. ++ * telexPartyNumber [4] IMPLICIT NumberDigits, ++ * privatePartyNumber [5] IMPLICIT PrivatePartyNumber, ++ * ++ * -- not used, value reserved. ++ * nationalStandardPartyNumber [8] IMPLICIT NumberDigits ++ * } ++ */ ++struct rosePartyNumber { ++ /*! ++ * \brief Party numbering plan ++ * \details ++ * unknown(0), ++ * public(1) - The numbering plan is according to ITU-T E.164, ++ * nsapEncoded(2), ++ * data(3) - Reserved, ++ * telex(4) - Reserved, ++ * private(5), ++ * nationalStandard(8) - Reserved ++ */ ++ u_int8_t plan; ++ ++ /*! ++ * \brief Type-Of-Number valid for public and private party number plans ++ * \details ++ * public: ++ * unknown(0), ++ * internationalNumber(1), ++ * nationalNumber(2), ++ * networkSpecificNumber(3) - Reserved, ++ * subscriberNumber(4) - Reserved, ++ * abbreviatedNumber(6) ++ * \details ++ * private: ++ * unknown(0), ++ * level2RegionalNumber(1), ++ * level1RegionalNumber(2), ++ * pTNSpecificNumber/pISNSpecificNumber(3), ++ * localNumber(4), ++ * abbreviatedNumber(6) ++ */ ++ u_int8_t ton; ++ ++ /*! \brief Number present if length is nonzero. */ ++ u_int8_t length; ++ ++ /*! \brief Number string data. */ ++ unsigned char str[20 + 1]; ++}; ++ ++/* ++ * NumberScreened ::= SEQUENCE { ++ * partyNumber PartyNumber, ++ * screeningIndicator ScreeningIndicator ++ * } ++ */ ++struct roseNumberScreened { ++ struct rosePartyNumber number; ++ ++ /*! ++ * \details ++ * userProvidedNotScreened(0), ++ * userProvidedVerifiedAndPassed(1), ++ * userProvidedVerifiedAndFailed(2) (Not used, value reserved), ++ * networkProvided(3) ++ */ ++ u_int8_t screening_indicator; ++}; ++ ++/* ++ * PartySubaddress ::= CHOICE { ++ * -- not recommended ++ * UserSpecifiedSubaddress, ++ * ++ * -- according to ITU-T Recommendation X.213 ++ * NSAPSubaddress ++ * } ++ * ++ * UserSpecifiedSubaddress ::= SEQUENCE { ++ * SubaddressInformation, ++ * ++ * -- used when the coding of subaddress is BCD ++ * oddCountIndicator BOOLEAN OPTIONAL ++ * } ++ * ++ * -- specified according to ITU-T Recommendation X.213. Some ++ * -- networks may limit the subaddress value to some other ++ * -- length, e.g. 4 octets ++ * NSAPSubaddress ::= OCTET STRING (SIZE(1..20)) ++ * ++ * -- coded according to user requirements. Some networks may ++ * -- limit the subaddress value to some other length, ++ * -- e.g. 4 octets ++ * SubaddressInformation ::= OCTET STRING (SIZE(1..20)) ++ */ ++struct rosePartySubaddress { ++ /*! \brief Subaddress type UserSpecified(0), NSAP(1) */ ++ u_int8_t type; ++ ++ /*! \brief Subaddress present if length is nonzero */ ++ u_int8_t length; ++ ++ union { ++ /*! \brief Specified according to ITU-T Recommendation X.213 */ ++ unsigned char nsap[20 + 1]; ++ ++ /*! \brief Use of this formatting is not recommended */ ++ struct { ++ /*! \brief TRUE if OddCount present */ ++ u_int8_t odd_count_present; ++ ++ /*! ++ * \brief TRUE if odd number of BCD digits (optional) ++ * \note Used when the coding of subaddress is BCD. ++ */ ++ u_int8_t odd_count; ++ unsigned char information[20 + 1]; ++ } user_specified; ++ } u; ++}; ++ ++/* ++ * Address ::= SEQUENCE { ++ * PartyNumber, ++ * PartySubaddress OPTIONAL ++ * } ++ */ ++struct roseAddress { ++ struct rosePartyNumber number; ++ ++ /*! \brief Subaddress (Optional) */ ++ struct rosePartySubaddress subaddress; ++}; ++ ++/* ++ * AddressScreened ::= SEQUENCE { ++ * PartyNumber, ++ * ScreeningIndicator, ++ * PartySubaddress OPTIONAL ++ * } ++ */ ++struct roseAddressScreened { ++ struct rosePartyNumber number; ++ ++ /*! \brief Subaddress (Optional) */ ++ struct rosePartySubaddress subaddress; ++ ++ /*! ++ * \details ++ * userProvidedNotScreened(0), ++ * userProvidedVerifiedAndPassed(1), ++ * userProvidedVerifiedAndFailed(2) (Not used, value reserved), ++ * networkProvided(3) ++ */ ++ u_int8_t screening_indicator; ++}; ++ ++/* ++ * PresentedNumberUnscreened ::= CHOICE { ++ * presentationAllowedNumber [0] EXPLICIT PartyNumber, ++ * presentationRestricted [1] IMPLICIT NULL, ++ * numberNotAvailableDueToInterworking [2] IMPLICIT NULL, ++ * presentationRestrictedNumber [3] EXPLICIT PartyNumber ++ * } ++ */ ++struct rosePresentedNumberUnscreened { ++ struct rosePartyNumber number; ++ /*! ++ * \brief Number presentation type ++ * \details ++ * presentationAllowedNumber(0), ++ * presentationRestricted(1), ++ * numberNotAvailableDueToInterworking(2), ++ * presentationRestrictedNumber(3) ++ */ ++ u_int8_t presentation; ++}; ++ ++/* ++ * PresentedNumberScreened ::= CHOICE { ++ * presentationAllowedNumber [0] IMPLICIT NumberScreened, ++ * presentationRestricted [1] IMPLICIT NULL, ++ * numberNotAvailableDueToInterworking [2] IMPLICIT NULL, ++ * presentationRestrictedNumber [3] IMPLICIT NumberScreened ++ * } ++ */ ++struct rosePresentedNumberScreened { ++ /*! \brief Screened number */ ++ struct roseNumberScreened screened; ++ /*! ++ * \brief Number presentation type ++ * \details ++ * presentationAllowedNumber(0), ++ * presentationRestricted(1), ++ * numberNotAvailableDueToInterworking(2), ++ * presentationRestrictedNumber(3) ++ */ ++ u_int8_t presentation; ++}; ++ ++/* ++ * PresentedAddressScreened ::= CHOICE { ++ * presentationAllowedAddress [0] IMPLICIT AddressScreened, ++ * presentationRestricted [1] IMPLICIT NULL, ++ * numberNotAvailableDueToInterworking [2] IMPLICIT NULL, ++ * presentationRestrictedAddress [3] IMPLICIT AddressScreened ++ * } ++ */ ++struct rosePresentedAddressScreened { ++ /*! \breif Screened address */ ++ struct roseAddressScreened screened; ++ /*! ++ * \brief Address presentation type ++ * \details ++ * presentationAllowedAddress(0), ++ * presentationRestricted(1), ++ * numberNotAvailableDueToInterworking(2), ++ * presentationRestrictedAddress(3) ++ */ ++ u_int8_t presentation; ++}; ++ ++ ++/* ------------------------------------------------------------------- */ ++ ++ ++/* ++ * Time ::= SEQUENCE { ++ * lengthOfTimeUnit [1] IMPLICIT LengthOfTimeUnit, ++ * scale [2] IMPLICIT Scale ++ * } ++ */ ++struct roseEtsiAOCTime { ++ /*! LengthOfTimeUnit ::= INTEGER (0..16777215) -- 24 bit number */ ++ u_int32_t length; ++ /*! ++ * \details ++ * oneHundredthSecond(0), ++ * oneTenthSecond(1), ++ * oneSecond(2), ++ * tenSeconds(3), ++ * oneMinute(4), ++ * oneHour(5), ++ * twentyFourHours(6) ++ */ ++ u_int8_t scale; ++}; ++ ++/* ++ * Amount ::= SEQUENCE { ++ * currencyAmount [1] IMPLICIT CurrencyAmount, ++ * multiplier [2] IMPLICIT Multiplier ++ * } ++ */ ++struct roseEtsiAOCAmount { ++ /*! CurrencyAmount ::= INTEGER (0..16777215) -- 24 bit number */ ++ u_int32_t currency; ++ /*! ++ * \details ++ * oneThousandth(0), ++ * oneHundredth(1), ++ * oneTenth(2), ++ * one(3), ++ * ten(4), ++ * hundred(5), ++ * thousand(6) ++ */ ++ u_int8_t multiplier; ++}; ++ ++/* ++ * DurationCurrency ::= SEQUENCE { ++ * dCurrency [1] IMPLICIT Currency, ++ * dAmount [2] IMPLICIT Amount, ++ * dChargingType [3] IMPLICIT ChargingType, ++ * dTime [4] IMPLICIT Time, ++ * dGranularity [5] IMPLICIT Time OPTIONAL ++ * } ++ */ ++struct roseEtsiAOCDurationCurrency { ++ struct roseEtsiAOCAmount amount; ++ struct roseEtsiAOCTime time; ++ /*! \breif dGranularity (optional) */ ++ struct roseEtsiAOCTime granularity; ++ /*! Currency ::= IA5String (SIZE (1..10)) -- Name of currency. */ ++ unsigned char currency[10 + 1]; ++ /*! ++ * \details ++ * continuousCharging(0), ++ * stepFunction(1) ++ */ ++ u_int8_t charging_type; ++ /*! TRUE if granularity time is present */ ++ u_int8_t granularity_present; ++}; ++ ++/* ++ * FlatRateCurrency ::= SEQUENCE { ++ * fRCurrency [1] IMPLICIT Currency, ++ * fRAmount [2] IMPLICIT Amount ++ * } ++ */ ++struct roseEtsiAOCFlatRateCurrency { ++ struct roseEtsiAOCAmount amount; ++ /*! Currency ::= IA5String (SIZE (1..10)) -- Name of currency. */ ++ unsigned char currency[10 + 1]; ++}; ++ ++/* ++ * VolumeRateCurrency ::= SEQUENCE { ++ * vRCurrency [1] IMPLICIT Currency, ++ * vRAmount [2] IMPLICIT Amount, ++ * vRVolumeUnit [3] IMPLICIT VolumeUnit ++ * } ++ */ ++struct roseEtsiAOCVolumeRateCurrency { ++ struct roseEtsiAOCAmount amount; ++ /*! Currency ::= IA5String (SIZE (1..10)) -- Name of currency. */ ++ unsigned char currency[10 + 1]; ++ /*! ++ * \brief Volume rate volume unit ++ * \details ++ * octet(0), ++ * segment(1), ++ * message(2) ++ */ ++ u_int8_t unit; ++}; ++ ++/* ++ * AOCSCurrencyInfo ::= SEQUENCE { ++ * chargedItem ChargedItem, ++ * CHOICE { ++ * specialChargingCode SpecialChargingCode, ++ * durationCurrency [1] IMPLICIT DurationCurrency, ++ * flatRateCurrency [2] IMPLICIT FlatRateCurrency, ++ * volumeRateCurrency [3] IMPLICIT VolumeRateCurrency ++ * freeOfCharge [4] IMPLICIT NULL, ++ * currencyInfoNotAvailable [5] IMPLICIT NULL ++ * } ++ * } ++ */ ++struct roseEtsiAOCSCurrencyInfo { ++ union { ++ struct roseEtsiAOCDurationCurrency duration; ++ struct roseEtsiAOCFlatRateCurrency flat_rate; ++ struct roseEtsiAOCVolumeRateCurrency volume_rate; ++ /*! SpecialChargingCode ::= INTEGER (1..10) */ ++ u_int8_t special_charging_code; ++ } u; ++ /*! ++ * \brief Determine what is stored in the union. ++ * \details ++ * specialChargingCode(0), ++ * durationCurrency(1), ++ * flatRateCurrency(2), ++ * volumeRateCurrency(3), ++ * freeOfCharge(4), ++ * currencyInfoNotAvailable(5), ++ */ ++ u_int8_t currency_type; ++ /*! ++ * \brief What service is being billed. ++ * \details ++ * basicCommunication(0), ++ * callAttempt(1), ++ * callSetup(2), ++ * userToUserInfo(3), ++ * operationOfSupplementaryServ(4) ++ */ ++ u_int8_t charged_item; ++}; ++ ++/* ++ * AOCSCurrencyInfoList ::= SEQUENCE SIZE (1..10) OF AOCSCurrencyInfo ++ */ ++struct roseEtsiAOCSCurrencyInfoList { ++ /*! \brief SEQUENCE SIZE (1..10) OF AOCSCurrencyInfo */ ++ struct roseEtsiAOCSCurrencyInfo list[10]; ++ ++ /*! \brief Number of AOCSCurrencyInfo records present */ ++ u_int8_t num_records; ++}; ++ ++/* ++ * RecordedCurrency ::= SEQUENCE { ++ * rCurrency [1] IMPLICIT Currency, ++ * rAmount [2] IMPLICIT Amount ++ * } ++ */ ++struct roseEtsiAOCRecordedCurrency { ++ /*! Amount of currency involved. */ ++ struct roseEtsiAOCAmount amount; ++ /*! Currency ::= IA5String (SIZE (1..10)) -- Name of currency. */ ++ unsigned char currency[10 + 1]; ++}; ++ ++/* ++ * RecordedUnits ::= SEQUENCE { ++ * CHOICE { ++ * recordedNumberOfUnits NumberOfUnits, ++ * notAvailable NULL ++ * }, ++ * recordedTypeOfUnits TypeOfUnit OPTIONAL ++ * } ++ */ ++struct roseEtsiAOCRecordedUnits { ++ /*! \brief recordedNumberOfUnits INTEGER (0..16777215) -- 24 bit number */ ++ u_int32_t number_of_units; ++ /*! \brief TRUE if number_of_units is not available (not present) */ ++ u_int8_t not_available; ++ /*! \brief recordedTypeOfUnits INTEGER (1..16) (optional) */ ++ u_int8_t type_of_unit; ++ /*! \brief TRUE if type_of_unit is present */ ++ u_int8_t type_of_unit_present; ++}; ++ ++/* ++ * RecordedUnitsList ::= SEQUENCE SIZE (1..32) OF RecordedUnits ++ */ ++struct roseEtsiAOCRecordedUnitsList { ++ /*! \brief SEQUENCE SIZE (1..32) OF RecordedUnits */ ++ struct roseEtsiAOCRecordedUnits list[32]; ++ ++ /*! \brief Number of RecordedUnits records present */ ++ u_int8_t num_records; ++}; ++ ++/* ++ * ChargingAssociation ::= CHOICE { ++ * chargedNumber [0] EXPLICIT PartyNumber, ++ * chargeIdentifier ChargeIdentifier ++ * } ++ */ ++struct roseEtsiAOCChargingAssociation { ++ /*! chargeIdentifier: INTEGER (-32768..32767) -- 16 bit number */ ++ int16_t id; ++ /*! chargedNumber */ ++ struct rosePartyNumber number; ++ /*! ++ * \details ++ * charge_identifier(0), ++ * charged_number(1) ++ */ ++ u_int8_t type; ++}; ++ ++/* ++ * AOCECurrencyInfo ::= SEQUENCE { ++ * CHOICE { ++ * freeOfCharge [1] IMPLICIT NULL, ++ * specificCurrency SEQUENCE { ++ * recordedCurrency [1] IMPLICIT RecordedCurrency, ++ * aOCEBillingId [2] IMPLICIT AOCEBillingId OPTIONAL ++ * } ++ * }, ++ * chargingAssociation ChargingAssociation OPTIONAL ++ * } ++ */ ++struct roseEtsiAOCECurrencyInfo { ++ struct { ++ /*! \brief recorded currency */ ++ struct roseEtsiAOCRecordedCurrency recorded; ++ /*! ++ * \brief AOCEBillingId (optional) ++ * \details ++ * normalCharging(0), ++ * reverseCharging(1), ++ * creditCardCharging(2), ++ * callForwardingUnconditional(3), ++ * callForwardingBusy(4), ++ * callForwardingNoReply(5), ++ * callDeflection(6), ++ * callTransfer(7) ++ */ ++ u_int8_t billing_id; ++ /*! \brief TRUE if billing id is present */ ++ u_int8_t billing_id_present; ++ } specific; ++ ++ /*! \brief chargingAssociation (optional) */ ++ struct roseEtsiAOCChargingAssociation charging_association; ++ ++ /*! \brief TRUE if charging_association is present */ ++ u_int8_t charging_association_present; ++ ++ /*! ++ * \brief TRUE if this is free of charge. ++ * \note When TRUE, the contents of specific are not valid. ++ */ ++ u_int8_t free_of_charge; ++}; ++ ++/* ++ * AOCEChargingUnitInfo ::= SEQUENCE { ++ * CHOICE { ++ * freeOfCharge [1] IMPLICIT NULL, ++ * specificChargingUnits SEQUENCE ++ * { ++ * recordedUnitsList [1] IMPLICIT RecordedUnitsList, ++ * aOCEBillingId [2] IMPLICIT AOCEBillingId OPTIONAL ++ * } ++ * }, ++ * chargingAssociation ChargingAssociation OPTIONAL ++ * } ++ */ ++struct roseEtsiAOCEChargingUnitInfo { ++ /*! \brief Not valid if free_of_charge is TRUE */ ++ struct { ++ /*! \brief RecordedUnitsList */ ++ struct roseEtsiAOCRecordedUnitsList recorded; ++ /*! ++ * \brief AOCEBillingId (optional) ++ * \details ++ * normalCharging(0), ++ * reverseCharging(1), ++ * creditCardCharging(2), ++ * callForwardingUnconditional(3), ++ * callForwardingBusy(4), ++ * callForwardingNoReply(5), ++ * callDeflection(6), ++ * callTransfer(7) ++ */ ++ u_int8_t billing_id; ++ /*! \brief TRUE if billing id is present */ ++ u_int8_t billing_id_present; ++ } specific; ++ ++ /*! \brief chargingAssociation (optional) */ ++ struct roseEtsiAOCChargingAssociation charging_association; ++ ++ /*! \brief TRUE if charging_association is present */ ++ u_int8_t charging_association_present; ++ ++ /*! ++ * \brief TRUE if this is free of charge. ++ * \note When TRUE, the contents of specific are not valid. ++ */ ++ u_int8_t free_of_charge; ++}; ++ ++/* ++ * ARGUMENT ChargingCase ++ */ ++struct roseEtsiChargingRequest_ARG { ++ /*! ++ * \details ++ * chargingInformationAtCallSetup(0), ++ * chargingDuringACall(1), ++ * chargingAtTheEndOfACall(2) ++ */ ++ u_int8_t charging_case; ++}; ++ ++/* ++ * RESULT CHOICE { ++ * AOCSCurrencyInfoList, ++ * AOCSSpecialArrInfo, ++ * chargingInfoFollows NULL ++ * } ++ */ ++struct roseEtsiChargingRequest_RES { ++ union { ++ struct roseEtsiAOCSCurrencyInfoList currency_info; ++ ++ /*! AOCSSpecialArrInfo ::= INTEGER (1..10) */ ++ u_int8_t special_arrangement; ++ } u; ++ /*! ++ * \details ++ * currency_info_list(0), ++ * special_arrangement_info(1), ++ * charging_info_follows(2) ++ */ ++ u_int8_t type; ++}; ++ ++/* ++ * ARGUMENT CHOICE { ++ * chargeNotAvailable NULL, ++ * AOCSCurrencyInfoList ++ * } ++ */ ++struct roseEtsiAOCSCurrency_ARG { ++ struct roseEtsiAOCSCurrencyInfoList currency_info; ++ /*! ++ * \details ++ * charge_not_available(0), ++ * currency_info_list(1) ++ */ ++ u_int8_t type; ++}; ++ ++/* ++ * ARGUMENT CHOICE { ++ * chargeNotAvailable NULL, ++ * AOCSSpecialArrInfo ++ * } ++ */ ++struct roseEtsiAOCSSpecialArr_ARG { ++ /*! ++ * \details ++ * charge_not_available(0), ++ * special_arrangement_info(1) ++ */ ++ u_int8_t type; ++ /*! AOCSSpecialArrInfo ::= INTEGER (1..10) */ ++ u_int8_t special_arrangement; ++}; ++ ++/* ++ * ARGUMENT CHOICE { ++ * chargeNotAvailable NULL, ++ * aOCDCurrencyInfo CHOICE { ++ * freeOfCharge [1] IMPLICIT NULL, ++ * specificCurrency SEQUENCE { ++ * recordedCurrency [1] IMPLICIT RecordedCurrency, ++ * typeOfChargingInfo [2] IMPLICIT TypeOfChargingInfo, ++ * aOCDBillingId [3] IMPLICIT AOCDBillingId OPTIONAL ++ * } ++ * } ++ * } ++ */ ++struct roseEtsiAOCDCurrency_ARG { ++ struct { ++ /*! \brief recorded currency */ ++ struct roseEtsiAOCRecordedCurrency recorded; ++ /*! ++ * \brief Type of recorded charging information. ++ * \details ++ * subTotal(0), ++ * total(1) ++ */ ++ u_int8_t type_of_charging_info; ++ /*! ++ * \brief AOCDBillingId (optional) ++ * \details ++ * normalCharging(0), ++ * reverseCharging(1), ++ * creditCardCharging(2) ++ */ ++ u_int8_t billing_id; ++ /*! \brief TRUE if billing id is present */ ++ u_int8_t billing_id_present; ++ } specific; ++ /*! ++ * \details ++ * charge_not_available(0), ++ * free_of_charge(1), ++ * specific_currency(2) ++ */ ++ u_int8_t type; ++}; ++ ++/* ++ * ARGUMENT CHOICE { ++ * chargeNotAvailable NULL, ++ * aOCDChargingUnitInfo CHOICE { ++ * freeOfCharge [1] IMPLICIT NULL, ++ * specificChargingUnits SEQUENCE { ++ * recordedUnitsList [1] IMPLICIT RecordedUnitsList, ++ * typeOfChargingInfo [2] IMPLICIT TypeOfChargingInfo, ++ * aOCDBillingId [3] IMPLICIT AOCDBillingId OPTIONAL ++ * } ++ * } ++ * } ++ */ ++struct roseEtsiAOCDChargingUnit_ARG { ++ struct { ++ /*! \brief RecordedUnitsList */ ++ struct roseEtsiAOCRecordedUnitsList recorded; ++ /*! ++ * \brief Type of recorded charging information. ++ * \details ++ * subTotal(0), ++ * total(1) ++ */ ++ u_int8_t type_of_charging_info; ++ /*! ++ * \brief AOCDBillingId (optional) ++ * \details ++ * normalCharging(0), ++ * reverseCharging(1), ++ * creditCardCharging(2) ++ */ ++ u_int8_t billing_id; ++ /*! \brief TRUE if billing id is present */ ++ u_int8_t billing_id_present; ++ } specific; ++ /*! ++ * \details ++ * charge_not_available(0), ++ * free_of_charge(1), ++ * specific_charging_units(2) ++ */ ++ u_int8_t type; ++}; ++ ++/* ++ * ARGUMENT CHOICE { ++ * chargeNotAvailable NULL, ++ * AOCECurrencyInfo ++ * } ++ */ ++struct roseEtsiAOCECurrency_ARG { ++ struct roseEtsiAOCECurrencyInfo currency_info; ++ /*! ++ * \details ++ * charge_not_available(0), ++ * currency_info(1) ++ */ ++ u_int8_t type; ++}; ++ ++/* ++ * ARGUMENT CHOICE { ++ * chargeNotAvailable NULL, ++ * AOCEChargingUnitInfo ++ * } ++ */ ++struct roseEtsiAOCEChargingUnit_ARG { ++ struct roseEtsiAOCEChargingUnitInfo charging_unit; ++ /*! ++ * \details ++ * charge_not_available(0), ++ * charging_unit(1) ++ */ ++ u_int8_t type; ++}; ++ ++ ++/* ------------------------------------------------------------------- */ ++ ++ ++/* ++ * ARGUMENT SEQUENCE { ++ * procedure Procedure, ++ * basicService BasicService, ++ * forwardedToAddress Address, ++ * servedUserNr ServedUserNr ++ * } ++ */ ++struct roseEtsiActivationDiversion_ARG { ++ /*! \brief Forwarded to address */ ++ struct roseAddress forwarded_to; ++ ++ /*! \brief Forward all numbers if not present (Number length is zero). */ ++ struct rosePartyNumber served_user_number; ++ ++ /*! \details cfu(0), cfb(1), cfnr(2) */ ++ u_int8_t procedure; ++ ++ /*! ++ * \details ++ * allServices(0), ++ * speech(1), ++ * unrestrictedDigitalInformation(2), ++ * audio3k1Hz(3), ++ * unrestrictedDigitalInformationWithTonesAndAnnouncements(4), ++ * multirate(5), ++ * telephony3k1Hz(32), ++ * teletex(33), ++ * telefaxGroup4Class1(34), ++ * videotexSyntaxBased(35), ++ * videotelephony(36), ++ * telefaxGroup2-3(37), ++ * telephony7kHz(38), ++ * euroFileTransfer(39), ++ * fileTransferAndAccessManagement(40), ++ * videoconference(41), ++ * audioGraphicConference(42) ++ */ ++ u_int8_t basic_service; ++}; ++ ++ ++/* ++ * ARGUMENT SEQUENCE { ++ * procedure Procedure, ++ * basicService BasicService, ++ * servedUserNr ServedUserNr ++ * } ++ */ ++struct roseEtsiDeactivationDiversion_ARG { ++ /*! \brief Forward all numbers if not present (Number length is zero). */ ++ struct rosePartyNumber served_user_number; ++ ++ /*! \details cfu(0), cfb(1), cfnr(2) */ ++ u_int8_t procedure; ++ ++ /*! ++ * \details ++ * allServices(0), ++ * speech(1), ++ * unrestrictedDigitalInformation(2), ++ * audio3k1Hz(3), ++ * unrestrictedDigitalInformationWithTonesAndAnnouncements(4), ++ * multirate(5), ++ * telephony3k1Hz(32), ++ * teletex(33), ++ * telefaxGroup4Class1(34), ++ * videotexSyntaxBased(35), ++ * videotelephony(36), ++ * telefaxGroup2-3(37), ++ * telephony7kHz(38), ++ * euroFileTransfer(39), ++ * fileTransferAndAccessManagement(40), ++ * videoconference(41), ++ * audioGraphicConference(42) ++ */ ++ u_int8_t basic_service; ++}; ++ ++ ++/* ++ * ARGUMENT SEQUENCE { ++ * procedure Procedure, ++ * basicService BasicService, ++ * forwardedToAddresss Address, ++ * servedUserNr ServedUserNr ++ * } ++ */ ++struct roseEtsiActivationStatusNotificationDiv_ARG { ++ /*! \brief Forwarded to address */ ++ struct roseAddress forwarded_to; ++ ++ /*! \brief Forward all numbers if not present (Number length is zero). */ ++ struct rosePartyNumber served_user_number; ++ ++ /*! \details cfu(0), cfb(1), cfnr(2) */ ++ u_int8_t procedure; ++ ++ /*! ++ * \details ++ * allServices(0), ++ * speech(1), ++ * unrestrictedDigitalInformation(2), ++ * audio3k1Hz(3), ++ * unrestrictedDigitalInformationWithTonesAndAnnouncements(4), ++ * multirate(5), ++ * telephony3k1Hz(32), ++ * teletex(33), ++ * telefaxGroup4Class1(34), ++ * videotexSyntaxBased(35), ++ * videotelephony(36), ++ * telefaxGroup2-3(37), ++ * telephony7kHz(38), ++ * euroFileTransfer(39), ++ * fileTransferAndAccessManagement(40), ++ * videoconference(41), ++ * audioGraphicConference(42) ++ */ ++ u_int8_t basic_service; ++}; ++ ++/* ++ * ARGUMENT SEQUENCE { ++ * procedure Procedure, ++ * basicService BasicService, ++ * servedUserNr ServedUserNr ++ * } ++ */ ++struct roseEtsiDeactivationStatusNotificationDiv_ARG { ++ /*! \brief Forward all numbers if not present (Number length is zero). */ ++ struct rosePartyNumber served_user_number; ++ ++ /*! \details cfu(0), cfb(1), cfnr(2) */ ++ u_int8_t procedure; ++ ++ /*! ++ * \details ++ * allServices(0), ++ * speech(1), ++ * unrestrictedDigitalInformation(2), ++ * audio3k1Hz(3), ++ * unrestrictedDigitalInformationWithTonesAndAnnouncements(4), ++ * multirate(5), ++ * telephony3k1Hz(32), ++ * teletex(33), ++ * telefaxGroup4Class1(34), ++ * videotexSyntaxBased(35), ++ * videotelephony(36), ++ * telefaxGroup2-3(37), ++ * telephony7kHz(38), ++ * euroFileTransfer(39), ++ * fileTransferAndAccessManagement(40), ++ * videoconference(41), ++ * audioGraphicConference(42) ++ */ ++ u_int8_t basic_service; ++}; ++ ++ ++/* ++ * ARGUMENT SEQUENCE { ++ * procedure Procedure, ++ * basicService BasicService DEFAULT allServices, ++ * servedUserNr ServedUserNr ++ * } ++ */ ++struct roseEtsiInterrogationDiversion_ARG { ++ /*! \brief Forward all numbers if not present (Number length is zero). */ ++ struct rosePartyNumber served_user_number; ++ ++ /*! \details cfu(0), cfb(1), cfnr(2) */ ++ u_int8_t procedure; ++ ++ /*! ++ * \details ++ * DEFAULT allServices ++ * ++ * \details ++ * allServices(0), ++ * speech(1), ++ * unrestrictedDigitalInformation(2), ++ * audio3k1Hz(3), ++ * unrestrictedDigitalInformationWithTonesAndAnnouncements(4), ++ * multirate(5), ++ * telephony3k1Hz(32), ++ * teletex(33), ++ * telefaxGroup4Class1(34), ++ * videotexSyntaxBased(35), ++ * videotelephony(36), ++ * telefaxGroup2-3(37), ++ * telephony7kHz(38), ++ * euroFileTransfer(39), ++ * fileTransferAndAccessManagement(40), ++ * videoconference(41), ++ * audioGraphicConference(42) ++ */ ++ u_int8_t basic_service; ++}; ++ ++/* ++ * IntResult ::= SEQUENCE { ++ * servedUserNr ServedUserNr, ++ * basicService BasicService, ++ * procedure Procedure, ++ * forwardedToAddress Address ++ * } ++ */ ++struct roseEtsiForwardingRecord { ++ /*! \brief Forwarded to address */ ++ struct roseAddress forwarded_to; ++ ++ /*! \brief Forward all numbers if not present (Number length is zero). */ ++ struct rosePartyNumber served_user_number; ++ ++ /*! \details cfu(0), cfb(1), cfnr(2) */ ++ u_int8_t procedure; ++ ++ /*! ++ * \details ++ * allServices(0), ++ * speech(1), ++ * unrestrictedDigitalInformation(2), ++ * audio3k1Hz(3), ++ * unrestrictedDigitalInformationWithTonesAndAnnouncements(4), ++ * multirate(5), ++ * telephony3k1Hz(32), ++ * teletex(33), ++ * telefaxGroup4Class1(34), ++ * videotexSyntaxBased(35), ++ * videotelephony(36), ++ * telefaxGroup2-3(37), ++ * telephony7kHz(38), ++ * euroFileTransfer(39), ++ * fileTransferAndAccessManagement(40), ++ * videoconference(41), ++ * audioGraphicConference(42) ++ */ ++ u_int8_t basic_service; ++}; ++ ++/* ++ * roseInterrogationDiversion_RES ++ * IntResultList ::= SET SIZE (0..29) OF IntResult ++ */ ++struct roseEtsiForwardingList { ++ /*! ++ * \brief SET SIZE (0..29) OF Forwarding Records ++ * \note Reduced the size of the array to conserve ++ * potential stack usage. ++ */ ++ struct roseEtsiForwardingRecord list[10]; ++ ++ /*! \brief Number of Forwarding records present */ ++ u_int8_t num_records; ++}; ++ ++/* ++ * ARGUMENT SEQUENCE { ++ * diversionReason DiversionReason, ++ * basicService BasicService, ++ * servedUserSubaddress PartySubaddress OPTIONAL, ++ * callingAddress [0] EXPLICIT PresentedAddressScreened OPTIONAL, ++ * originalCalledNr [1] EXPLICIT PresentedNumberUnscreened OPTIONAL, ++ * lastDivertingNr [2] EXPLICIT PresentedNumberUnscreened OPTIONAL, ++ * lastDivertingReason [3] EXPLICIT DiversionReason OPTIONAL, ++ * ++ * -- The User-user information element, as specified ++ * -- in ETS 300 102-1 [11], subclause 4.5.29, shall ++ * -- be embedded in the userInfo parameter. ++ * userInfo Q931InformationElement OPTIONAL ++ * } ++ */ ++struct roseEtsiDiversionInformation_ARG { ++ /*! \brief Served user subaddress (Optional) */ ++ struct rosePartySubaddress served_user_subaddress; ++ ++ /*! \brief Calling address (Optional) */ ++ struct rosePresentedAddressScreened calling; ++ ++ /*! \brief Original called number (Optional) */ ++ struct rosePresentedNumberUnscreened original_called; ++ ++ /*! \brief Last diverting number (Optional) */ ++ struct rosePresentedNumberUnscreened last_diverting; ++ ++ /*! \brief User-User information embedded in Q.931 IE (Optional) */ ++ struct roseQ931ie q931ie; ++ /*! \brief q931ie.contents "allocated" after the stucture. */ ++ unsigned char q931ie_contents[ROSE_Q931_MAX_USER + 1]; ++ ++ /*! ++ * \brief Last diverting reason (Optional) ++ * ++ * \details ++ * unknown(0), ++ * cfu(1), ++ * cfb(2), ++ * cfnr(3), ++ * cdAlerting(4), ++ * cdImmediate(5) ++ */ ++ u_int8_t last_diverting_reason; ++ ++ /*! \brief TRUE if CallingAddress is present */ ++ u_int8_t calling_present; ++ ++ /*! \brief TRUE if OriginalCalled is present */ ++ u_int8_t original_called_present; ++ ++ /*! \brief TRUE if LastDiverting is present */ ++ u_int8_t last_diverting_present; ++ ++ /*! \brief TRUE if LastDivertingReason is present */ ++ u_int8_t last_diverting_reason_present; ++ ++ /*! ++ * \details ++ * unknown(0), ++ * cfu(1), ++ * cfb(2), ++ * cfnr(3), ++ * cdAlerting(4), ++ * cdImmediate(5) ++ */ ++ u_int8_t diversion_reason; ++ ++ /*! ++ * \details ++ * allServices(0), ++ * speech(1), ++ * unrestrictedDigitalInformation(2), ++ * audio3k1Hz(3), ++ * unrestrictedDigitalInformationWithTonesAndAnnouncements(4), ++ * multirate(5), ++ * telephony3k1Hz(32), ++ * teletex(33), ++ * telefaxGroup4Class1(34), ++ * videotexSyntaxBased(35), ++ * videotelephony(36), ++ * telefaxGroup2-3(37), ++ * telephony7kHz(38), ++ * euroFileTransfer(39), ++ * fileTransferAndAccessManagement(40), ++ * videoconference(41), ++ * audioGraphicConference(42) ++ */ ++ u_int8_t basic_service; ++}; ++ ++ ++/* ++ * ARGUMENT SEQUENCE { ++ * deflectionAddress Address, ++ * presentationAllowedDivertedToUser PresentationAllowedIndicator OPTIONAL ++ * } ++ */ ++struct roseEtsiCallDeflection_ARG { ++ /*! \brief Deflection address (Deflected-To address) */ ++ struct roseAddress deflection; ++ ++ /*! \brief TRUE if PresentationAllowedToDivertedToUser is present */ ++ u_int8_t presentation_allowed_to_diverted_to_user_present; ++ ++ /*! \brief TRUE if presentation is allowed (Optional) */ ++ u_int8_t presentation_allowed_to_diverted_to_user; ++}; ++ ++ ++/* ++ * ARGUMENT SEQUENCE { ++ * reroutingReason DiversionReason, ++ * calledAddress Address, ++ * reroutingCounter DiversionCounter, ++ * ++ * -- The User-user information element (optional), ++ * -- High layer compatibility information element (optional), ++ * -- Bearer capability information element ++ * -- and Low layer compatibility information element (optional) ++ * -- as specified in ETS 300 102-1 [11] subclause 4.5 shall be ++ * -- embedded in the q931InfoElement. ++ * q931InfoElement Q931InformationElement, ++ * lastReroutingNr [1] EXPLICIT PresentedNumberUnscreened, ++ * subscriptionOption [2] EXPLICIT SubscriptionOption DEFAULT noNotification, ++ * callingPartySubaddress [3] EXPLICIT PartySubaddress OPTIONAL ++ * } ++ */ ++struct roseEtsiCallRerouting_ARG { ++ struct roseAddress called_address; ++ ++ /*! \brief The BC, HLC (optional), LLC (optional), and User-user (optional) information */ ++ struct roseQ931ie q931ie; ++ /*! \brief q931ie.contents "allocated" after the stucture. */ ++ unsigned char q931ie_contents[ROSE_Q931_MAX_BC + ROSE_Q931_MAX_HLC + ++ ROSE_Q931_MAX_LLC + ROSE_Q931_MAX_USER + 1]; ++ ++ /*! \brief Last rerouting number */ ++ struct rosePresentedNumberUnscreened last_rerouting; ++ ++ /*! \brief Calling party subaddress (Optional) */ ++ struct rosePartySubaddress calling_subaddress; ++ ++ /*! ++ * \details ++ * unknown(0), ++ * cfu(1), ++ * cfb(2), ++ * cfnr(3), ++ * cdAlerting(4), ++ * cdImmediate(5) ++ */ ++ u_int8_t rerouting_reason; ++ ++ /*! \brief Range 1-5 */ ++ u_int8_t rerouting_counter; ++ ++ /*! ++ * \details ++ * DEFAULT noNotification ++ * ++ * \details ++ * noNotification(0), ++ * notificationWithoutDivertedToNr(1), ++ * notificationWithDivertedToNr(2) ++ */ ++ u_int8_t subscription_option; ++}; ++ ++ ++/* ++ * roseInterrogateServedUserNumbers_RES ++ * ServedUserNumberList ::= SET SIZE (0..99) OF PartyNumber ++ */ ++struct roseEtsiServedUserNumberList { ++ /*! ++ * \brief SET SIZE (0..99) OF Served user numbers ++ * \note Reduced the size of the array to conserve ++ * potential stack usage. ++ */ ++ struct rosePartyNumber number[20]; ++ ++ /*! \brief Number of Served user numbers present */ ++ u_int8_t num_records; ++}; ++ ++ ++/* ++ * ARGUMENT SEQUENCE { ++ * diversionReason DiversionReason, ++ * subscriptionOption SubscriptionOption, ++ * divertedToNumber PresentedNumberUnscreened OPTIONAL ++ * } ++ */ ++struct roseEtsiDivertingLegInformation1_ARG { ++ /*! \brief Diverted to number (Optional) */ ++ struct rosePresentedNumberUnscreened diverted_to; ++ ++ /*! \brief TRUE if DivertedTo is present */ ++ u_int8_t diverted_to_present; ++ ++ /*! ++ * \details ++ * unknown(0), ++ * cfu(1), ++ * cfb(2), ++ * cfnr(3), ++ * cdAlerting(4), ++ * cdImmediate(5) ++ */ ++ u_int8_t diversion_reason; ++ ++ /*! ++ * \details ++ * noNotification(0), ++ * notificationWithoutDivertedToNr(1), ++ * notificationWithDivertedToNr(2) ++ */ ++ u_int8_t subscription_option; ++}; ++ ++/* ++ * ARGUMENT SEQUENCE { ++ * diversionCounter DiversionCounter, ++ * diversionReason DiversionReason, ++ * divertingNr [1] EXPLICIT PresentedNumberUnscreened OPTIONAL, ++ * originalCalledNr [2] EXPLICIT PresentedNumberUnscreened OPTIONAL ++ * } ++ */ ++struct roseEtsiDivertingLegInformation2_ARG { ++ /*! \brief Diverting number (Optional) */ ++ struct rosePresentedNumberUnscreened diverting; ++ ++ /*! \brief Original called number (Optional) */ ++ struct rosePresentedNumberUnscreened original_called; ++ ++ /*! \brief TRUE if Diverting number is present */ ++ u_int8_t diverting_present; ++ ++ /*! \brief TRUE if OriginalCalled is present */ ++ u_int8_t original_called_present; ++ ++ /*! ++ * \details ++ * unknown(0), ++ * cfu(1), ++ * cfb(2), ++ * cfnr(3), ++ * cdAlerting(4), ++ * cdImmediate(5) ++ */ ++ u_int8_t diversion_reason; ++ ++ /*! \brief Range 1-5 */ ++ u_int8_t diversion_counter; ++}; ++ ++/* ++ * ARGUMENT presentationAllowedIndicator PresentationAllowedIndicator ++ */ ++struct roseEtsiDivertingLegInformation3_ARG { ++ /*! \brief TRUE if presentation is allowed */ ++ u_int8_t presentation_allowed_indicator; ++}; ++ ++ ++/* ------------------------------------------------------------------- */ ++ ++ ++/* ++ * ARGUMENT LinkId ++ */ ++struct roseEtsiExplicitEctExecute_ARG { ++ int16_t link_id; ++}; ++ ++/* ++ * ARGUMENT transferredToSubaddress PartySubaddress ++ */ ++struct roseEtsiSubaddressTransfer_ARG { ++ /*! \brief Transferred to subaddress */ ++ struct rosePartySubaddress subaddress; ++}; ++ ++ ++/* ++ * RESULT LinkId ++ */ ++struct roseEtsiEctLinkIdRequest_RES { ++ int16_t link_id; ++}; ++ ++ ++/* ++ * ARGUMENT SEQUENCE { ++ * ENUMERATED { ++ * alerting (0), ++ * active (1) ++ * }, ++ * redirectionNumber PresentedNumberUnscreened OPTIONAL ++ * } ++ */ ++struct roseEtsiEctInform_ARG { ++ /*! \brief Redirection Number (Optional) */ ++ struct rosePresentedNumberUnscreened redirection; ++ ++ /*! \brief TRUE if the Redirection Number is present */ ++ u_int8_t redirection_present; ++ ++ /*! \details alerting(0), active(1) */ ++ u_int8_t status; ++}; ++ ++ ++/* ++ * ARGUMENT CallTransferIdentity ++ */ ++struct roseEtsiEctLoopTest_ARG { ++ int8_t call_transfer_id; ++}; ++ ++/* ++ * RESULT LoopResult ++ */ ++struct roseEtsiEctLoopTest_RES { ++ /*! ++ * \details ++ * insufficientInformation(0), ++ * noLoopExists(1), ++ * simultaneousTransfer(2) ++ */ ++ u_int8_t loop_result; ++}; ++ ++ ++/* ------------------------------------------------------------------- */ ++ ++ ++/* ++ * Name ::= CHOICE { ++ * -- iso8859-1 is implied in namePresentationAllowedSimple. ++ * namePresentationAllowedSimple [0] IMPLICIT NameData, ++ * namePresentationAllowedExtended [1] IMPLICIT NameSet, ++ * ++ * -- iso8859-1 is implied in namePresentationRestrictedSimple. ++ * namePresentationRestrictedSimple [2] IMPLICIT NameData, ++ * namePresentationRestrictedExtended [3] IMPLICIT NameSet, ++ * ++ * -- namePresentationRestrictedNull shall only be used in the ++ * -- case of interworking where the other network provides an ++ * -- indication that the name is restricted without the name itself. ++ * namePresentationRestrictedNull [7] IMPLICIT NULL, ++ * ++ * nameNotAvailable [4] IMPLICIT NULL ++ * } ++ * ++ * NameSet ::= SEQUENCE { ++ * nameData NameData, ++ * ++ * -- If characterSet is not included, iso8859-1 is implied. ++ * characterSet CharacterSet OPTIONAL -- DEFAULT iso8859-1 ++ * } ++ * ++ * -- The maximum allowed size of the name field is 50 octets. ++ * -- The minimum required size of the name field is 1 octet. ++ * NameData ::= OCTET STRING (SIZE (1..50)) ++ */ ++struct roseQsigName { ++ /*! ++ * \details ++ * optional_name_not_present(0), ++ * presentation_allowed(1), ++ * presentation_restricted(2), ++ * presentation_restricted_null(3), ++ * name_not_available(4) ++ */ ++ u_int8_t presentation; ++ ++ /*! ++ * \details ++ * Set to iso8859-1 if not present in the encoding. ++ * ++ * \details ++ * unknown(0), ++ * iso8859-1(1), ++ * enum-value-withdrawn-by-ITU-T(2) ++ * iso8859-2(3), ++ * iso8859-3(4), ++ * iso8859-4(5), ++ * iso8859-5(6), ++ * iso8859-7(7), ++ * iso10646-BmpString(8), ++ * iso10646-utf-8String(9) ++ */ ++ u_int8_t char_set; ++ ++ /*! \brief Length of name data */ ++ u_int8_t length; ++ ++ /*! \brief Name string data */ ++ unsigned char data[50 + 1]; ++}; ++ ++/* ++ * NOTE: We are not going to record the Extension information ++ * since it is manufacturer specific. ++ * ++ * ARGUMENT CHOICE { ++ * Name, ++ * SEQUENCE { ++ * Name, ++ * CHOICE { ++ * [5] IMPLICIT Extension, ++ * [6] IMPLICIT SEQUENCE OF Extension ++ * } OPTIONAL ++ * } ++ * } ++ */ ++struct roseQsigPartyName_ARG { ++ struct roseQsigName name; ++}; ++ ++ ++/* ------------------------------------------------------------------- */ ++ ++ ++/* ++ * Time ::= SEQUENCE { ++ * lengthOfTimeUnit [1] IMPLICIT LengthOfTimeUnit, ++ * scale [2] IMPLICIT Scale ++ * } ++ */ ++struct roseQsigAOCTime { ++ /*! LengthOfTimeUnit ::= INTEGER (0..16777215) -- 24 bit number */ ++ u_int32_t length; ++ /*! ++ * \details ++ * oneHundredthSecond(0), ++ * oneTenthSecond(1), ++ * oneSecond(2), ++ * tenSeconds(3), ++ * oneMinute(4), ++ * oneHour(5), ++ * twentyFourHours(6) ++ */ ++ u_int8_t scale; ++}; ++ ++/* ++ * Amount ::= SEQUENCE { ++ * currencyAmount [1] IMPLICIT CurrencyAmount, ++ * multiplier [2] IMPLICIT Multiplier ++ * } ++ */ ++struct roseQsigAOCAmount { ++ /*! CurrencyAmount ::= INTEGER (0..16777215) -- 24 bit number */ ++ u_int32_t currency; ++ /*! ++ * \details ++ * oneThousandth(0), ++ * oneHundredth(1), ++ * oneTenth(2), ++ * one(3), ++ * ten(4), ++ * hundred(5), ++ * thousand(6) ++ */ ++ u_int8_t multiplier; ++}; ++ ++/* ++ * DurationCurrency ::= SEQUENCE { ++ * dCurrency [1] IMPLICIT Currency, ++ * dAmount [2] IMPLICIT Amount, ++ * dChargingType [3] IMPLICIT ChargingType, ++ * dTime [4] IMPLICIT Time, ++ * dGranularity [5] IMPLICIT Time OPTIONAL ++ * } ++ */ ++struct roseQsigAOCDurationCurrency { ++ struct roseQsigAOCAmount amount; ++ struct roseQsigAOCTime time; ++ /*! \brief dGranularity (optional) */ ++ struct roseQsigAOCTime granularity; ++ /*! ++ * \brief Name of currency ++ * \details ++ * Currency ::= IA5String (SIZE (0..10)) ++ * \note The empty string is the default currency for the network. ++ */ ++ unsigned char currency[10 + 1]; ++ /*! ++ * \details ++ * continuousCharging(0), ++ * stepFunction(1) ++ */ ++ u_int8_t charging_type; ++ /*! TRUE if granularity time is present */ ++ u_int8_t granularity_present; ++}; ++ ++/* ++ * FlatRateCurrency ::= SEQUENCE { ++ * fRCurrency [1] IMPLICIT Currency, ++ * fRAmount [2] IMPLICIT Amount ++ * } ++ */ ++struct roseQsigAOCFlatRateCurrency { ++ struct roseQsigAOCAmount amount; ++ /*! ++ * \brief Name of currency ++ * \details ++ * Currency ::= IA5String (SIZE (0..10)) ++ * \note The empty string is the default currency for the network. ++ */ ++ unsigned char currency[10 + 1]; ++}; ++ ++/* ++ * VolumeRateCurrency ::= SEQUENCE { ++ * vRCurrency [1] IMPLICIT Currency, ++ * vRAmount [2] IMPLICIT Amount, ++ * vRVolumeUnit [3] IMPLICIT VolumeUnit ++ * } ++ */ ++struct roseQsigAOCVolumeRateCurrency { ++ struct roseQsigAOCAmount amount; ++ /*! ++ * \brief Name of currency ++ * \details ++ * Currency ::= IA5String (SIZE (0..10)) ++ * \note The empty string is the default currency for the network. ++ */ ++ unsigned char currency[10 + 1]; ++ /*! ++ * \brief Volume rate volume unit ++ * \details ++ * octet(0), ++ * segment(1), ++ * message(2) ++ */ ++ u_int8_t unit; ++}; ++ ++/* ++ * AOCSCurrencyInfo ::= SEQUENCE { ++ * chargedItem ChargedItem, ++ * rateType CHOICE { ++ * durationCurrency [1] IMPLICIT DurationCurrency, ++ * flatRateCurrency [2] IMPLICIT FlatRateCurrency, ++ * volumeRateCurrency [3] IMPLICIT VolumeRateCurrency, ++ * specialChargingCode SpecialChargingCode, ++ * freeOfCharge [4] IMPLICIT NULL, ++ * currencyInfoNotAvailable [5] IMPLICIT NULL, ++ * freeOfChargefromBeginning [6] IMPLICIT NULL ++ * } ++ * } ++ */ ++struct roseQsigAOCSCurrencyInfo { ++ union { ++ struct roseQsigAOCDurationCurrency duration; ++ struct roseQsigAOCFlatRateCurrency flat_rate; ++ struct roseQsigAOCVolumeRateCurrency volume_rate; ++ /*! SpecialChargingCode ::= INTEGER (1..10) */ ++ u_int8_t special_charging_code; ++ } u; ++ /*! ++ * \brief Determine what is stored in the union. ++ * \details ++ * specialChargingCode(0), ++ * durationCurrency(1), ++ * flatRateCurrency(2), ++ * volumeRateCurrency(3), ++ * freeOfCharge(4), ++ * currencyInfoNotAvailable(5), ++ * freeOfChargeFromBeginning(6) ++ */ ++ u_int8_t currency_type; ++ /*! ++ * \brief What service is being billed. ++ * \details ++ * basicCommunication(0), ++ * callAttempt(1), ++ * callSetup(2), ++ * userToUserInfo(3), ++ * operationOfSupplementaryServ(4) ++ */ ++ u_int8_t charged_item; ++}; ++ ++/* ++ * AOCSCurrencyInfoList ::= SEQUENCE SIZE (1..10) OF AOCSCurrencyInfo ++ */ ++struct roseQsigAOCSCurrencyInfoList { ++ /*! \brief SEQUENCE SIZE (1..10) OF AOCSCurrencyInfo */ ++ struct roseQsigAOCSCurrencyInfo list[10]; ++ ++ /*! \brief Number of AOCSCurrencyInfo records present */ ++ u_int8_t num_records; ++}; ++ ++/* ++ * RecordedCurrency ::= SEQUENCE { ++ * rCurrency [1] IMPLICIT Currency, ++ * rAmount [2] IMPLICIT Amount ++ * } ++ */ ++struct roseQsigAOCRecordedCurrency { ++ /*! Amount of currency involved. */ ++ struct roseQsigAOCAmount amount; ++ /*! ++ * \brief Name of currency ++ * \details ++ * Currency ::= IA5String (SIZE (0..10)) ++ * \note The empty string is the default currency for the network. ++ */ ++ unsigned char currency[10 + 1]; ++}; ++ ++/* ++ * ChargingAssociation ::= CHOICE { ++ * chargedNumber [0] EXPLICIT PartyNumber, ++ * chargeIdentifier ChargeIdentifier ++ * } ++ */ ++struct roseQsigAOCChargingAssociation { ++ /*! chargeIdentifier: INTEGER (-32768..32767) -- 16 bit number */ ++ int16_t id; ++ /*! chargedNumber */ ++ struct rosePartyNumber number; ++ /*! ++ * \details ++ * charge_identifier(0) ++ * charged_number(1), ++ */ ++ u_int8_t type; ++}; ++ ++/* ++ * AocRateArg ::= SEQUENCE { ++ * aocRate CHOICE ++ * { ++ * chargeNotAvailable NULL, ++ * aocSCurrencyInfoList AOCSCurrencyInfoList ++ * }, ++ * rateArgExtension CHOICE { ++ * extension [1] IMPLICIT Extension, ++ * multipleExtension [2] IMPLICIT SEQUENCE OF Extension ++ * } OPTIONAL ++ * } ++ */ ++struct roseQsigAocRateArg_ARG { ++ struct roseQsigAOCSCurrencyInfoList currency_info; ++ /*! ++ * \details ++ * charge_not_available(0), ++ * currency_info_list(1) ++ */ ++ u_int8_t type; ++}; ++ ++/* ++ * AocInterimArg ::= SEQUENCE { ++ * interimCharge CHOICE { ++ * chargeNotAvailable [0] IMPLICIT NULL, ++ * freeOfCharge [1] IMPLICIT NULL, ++ * specificCurrency SEQUENCE { ++ * recordedCurrency [1] IMPLICIT RecordedCurrency, ++ * interimBillingId [2] IMPLICIT InterimBillingId OPTIONAL ++ * } ++ * }, ++ * interimArgExtension CHOICE { ++ * extension [1] IMPLICIT Extension, ++ * multipleExtension [2] IMPLICIT SEQUENCE OF Extension ++ * } OPTIONAL ++ * } ++ */ ++struct roseQsigAocInterimArg_ARG { ++ struct { ++ /*! \brief recorded currency */ ++ struct roseQsigAOCRecordedCurrency recorded; ++ /*! ++ * \brief InterimBillingId (optional) ++ * \details ++ * normalCharging(0), ++ * creditCardCharging(2) ++ */ ++ u_int8_t billing_id; ++ /*! \brief TRUE if billing id is present */ ++ u_int8_t billing_id_present; ++ } specific; ++ /*! ++ * \details ++ * charge_not_available(0), ++ * free_of_charge(1), ++ * specific_currency(2) ++ */ ++ u_int8_t type; ++}; ++ ++/* ++ * AocFinalArg ::= SEQUENCE { ++ * finalCharge CHOICE { ++ * chargeNotAvailable [0] IMPLICIT NULL, ++ * freeOfCharge [1] IMPLICIT NULL, ++ * specificCurrency SEQUENCE { ++ * recordedCurrency [1] IMPLICIT RecordedCurrency, ++ * finalBillingId [2] IMPLICIT FinalBillingId OPTIONAL ++ * } ++ * }, ++ * chargingAssociation ChargingAssociation OPTIONAL, ++ * finalArgExtension CHOICE { ++ * extension [1] IMPLICIT Extension, ++ * multipleExtension [2] IMPLICIT SEQUENCE OF Extension ++ * } OPTIONAL ++ * } ++ */ ++struct roseQsigAocFinalArg_ARG { ++ struct { ++ /*! \brief recorded currency */ ++ struct roseQsigAOCRecordedCurrency recorded; ++ /*! ++ * \brief FinalBillingId (optional) ++ * \details ++ * normalCharging(0), ++ * creditCardCharging(2), ++ * callForwardingUnconditional(3), ++ * callForwardingBusy(4), ++ * callForwardingNoReply(5), ++ * callDeflection(6), ++ * callTransfer(7) ++ */ ++ u_int8_t billing_id; ++ /*! \brief TRUE if billing id is present */ ++ u_int8_t billing_id_present; ++ } specific; ++ ++ /*! \brief chargingAssociation (optional) */ ++ struct roseQsigAOCChargingAssociation charging_association; ++ ++ /*! \brief TRUE if charging_association is present */ ++ u_int8_t charging_association_present; ++ ++ /*! ++ * \details ++ * charge_not_available(0), ++ * free_of_charge(1), ++ * specific_currency(2) ++ */ ++ u_int8_t type; ++}; ++ ++/* ++ * ChargeRequestArg ::= SEQUENCE { ++ * adviceModeCombinations SEQUENCE SIZE(0..7) OF AdviceModeCombination, ++ * chargeReqArgExtension CHOICE { ++ * extension [1] IMPLICIT Extension, ++ * multipleExtension [2] IMPLICIT SEQUENCE OF Extension ++ * } OPTIONAL ++ * } ++ */ ++struct roseQsigChargeRequestArg_ARG { ++ /*! ++ * \brief SEQUENCE SIZE(0..7) OF AdviceModeCombination ++ * \details ++ * rate(0) <charge rate provision>, ++ * rateInterim(1) <charge rate and interim charge provision>, ++ * rateFinal(2) <charge rate and final charge provision>, ++ * interim(3) <interim charge provision>, ++ * final(4) <final charge provision>, ++ * interimFinal(5) <interim charge and final charge provision>, ++ * rateInterimFinal(6) <charge rate, interim charge and final charge provision> ++ */ ++ u_int8_t advice_mode_combinations[7]; ++ ++ /*! \brief Number of AdviceModeCombination values present */ ++ u_int8_t num_records; ++}; ++ ++/* ++ * ChargeRequestRes ::= SEQUENCE { ++ * adviceModeCombination AdviceModeCombination, ++ * chargeReqResExtension CHOICE { ++ * extension [1] IMPLICIT Extension, ++ * multipleExtension [2] IMPLICIT SEQUENCE OF Extension ++ * } OPTIONAL ++ * } ++ */ ++struct roseQsigChargeRequestRes_RES { ++ /*! ++ * \details ++ * rate(0) <charge rate provision>, ++ * rateInterim(1) <charge rate and interim charge provision>, ++ * rateFinal(2) <charge rate and final charge provision>, ++ * interim(3) <interim charge provision>, ++ * final(4) <final charge provision>, ++ * interimFinal(5) <interim charge and final charge provision>, ++ * rateInterimFinal(6) <charge rate, interim charge and final charge provision> ++ */ ++ u_int8_t advice_mode_combination; ++}; ++ ++/* ++ * AocCompleteArg ::= SEQUENCE { ++ * chargedUser PartyNumber, ++ * chargingAssociation ChargingAssociation OPTIONAL, ++ * completeArgExtension CHOICE { ++ * extension [1] IMPLICIT Extension, ++ * multipleExtension [2] IMPLICIT SEQUENCE OF Extension ++ * } OPTIONAL ++ * } ++ */ ++struct roseQsigAocCompleteArg_ARG { ++ /*! \brief chargingAssociation (optional) */ ++ struct roseQsigAOCChargingAssociation charging_association; ++ ++ struct rosePartyNumber charged_user_number; ++ ++ /*! \brief TRUE if charging_association is present */ ++ u_int8_t charging_association_present; ++}; ++ ++/* ++ * AocCompleteRes ::= SEQUENCE { ++ * chargingOption ChargingOption, ++ * completeResExtension CHOICE { ++ * extension [1] IMPLICIT Extension, ++ * multipleExtension [2] IMPLICIT SEQUENCE OF Extension ++ * } OPTIONAL ++ * } ++ */ ++struct roseQsigAocCompleteRes_RES { ++ /*! ++ * \details ++ * aocFreeOfCharge(0), ++ * aocContinueCharging(1), ++ * aocStopCharging(2) ++ */ ++ u_int8_t charging_option; ++}; ++ ++/* ++ * AocDivChargeReqArg ::= SEQUENCE { ++ * divertingUser PartyNumber, ++ * chargingAssociation ChargingAssociation OPTIONAL, ++ * diversionType DiversionType, ++ * aocDivChargeReqArgExt CHOICE { ++ * extension [1] IMPLICIT Extension, ++ * multipleExtension [2] IMPLICIT SEQUENCE OF Extension ++ * } OPTIONAL ++ * } ++ */ ++struct roseQsigAocDivChargeReqArg_ARG { ++ /*! \brief chargingAssociation (optional) */ ++ struct roseQsigAOCChargingAssociation charging_association; ++ ++ struct rosePartyNumber diverting_user_number; ++ ++ /*! \brief TRUE if charging_association is present */ ++ u_int8_t charging_association_present; ++ ++ /*! ++ * \details ++ * callForwardingUnconditional(0), ++ * callForwardingBusy(1), ++ * callForwardingNoReply(2), ++ * callDeflection(3) ++ */ ++ u_int8_t diversion_type; ++}; ++ ++ ++/* ------------------------------------------------------------------- */ ++ ++ ++/* ++ * CTIdentifyRes ::= SEQUENCE { ++ * callIdentity CallIdentity, ++ * reroutingNumber PartyNumber, ++ * resultExtension CHOICE { ++ * [6] IMPLICIT Extension, ++ * [7] IMPLICIT SEQUENCE OF Extension ++ * } OPTIONAL ++ * } ++ */ ++struct roseQsigCTIdentifyRes_RES { ++ struct rosePartyNumber rerouting_number; ++ ++ /*! \brief CallIdentity ::= NumericString (SIZE (1..4)) */ ++ unsigned char call_id[4 + 1]; ++}; ++ ++/* ++ * CTInitiateArg ::= SEQUENCE { ++ * callIdentity CallIdentity, ++ * reroutingNumber PartyNumber, ++ * argumentExtension CHOICE { ++ * [6] IMPLICIT Extension, ++ * [7] IMPLICIT SEQUENCE OF Extension ++ * } OPTIONAL ++ * } ++ */ ++struct roseQsigCTInitiateArg_ARG { ++ struct rosePartyNumber rerouting_number; ++ ++ /*! \brief CallIdentity ::= NumericString (SIZE (1..4)) */ ++ unsigned char call_id[4 + 1]; ++}; ++ ++/* ++ * CTSetupArg ::= SEQUENCE { ++ * callIdentity CallIdentity, ++ * argumentExtension CHOICE { ++ * [0] IMPLICIT Extension, ++ * [1] IMPLICIT SEQUENCE OF Extension ++ * } OPTIONAL ++ * } ++ */ ++struct roseQsigCTSetupArg_ARG { ++ /*! \brief CallIdentity ::= NumericString (SIZE (1..4)) */ ++ unsigned char call_id[4 + 1]; ++}; ++ ++/* ++ * CTActiveArg ::= SEQUENCE { ++ * connectedAddress PresentedAddressScreened, ++ * ++ * -- ISO/IEC 11572 information elements Party ++ * -- category and Progress indicator are conveyed ++ * basicCallInfoElements PSS1InformationElement OPTIONAL, ++ * connectedName Name OPTIONAL, ++ * argumentExtension CHOICE { ++ * [9] IMPLICIT Extension, ++ * [10] IMPLICIT SEQUENCE OF Extension ++ * } OPTIONAL ++ * } ++ */ ++struct roseQsigCTActiveArg_ARG { ++ /*! \brief connectedAddress */ ++ struct rosePresentedAddressScreened connected; ++ ++ /*! \brief basicCallInfoElements (optional) */ ++ struct roseQ931ie q931ie; ++ /*! \brief q931ie.contents "allocated" after the stucture. */ ++ unsigned char q931ie_contents[ROSE_Q931_MAX_PROGRESS + 1]; ++ ++ /*! \brief connectedName (optional) */ ++ struct roseQsigName connected_name; ++ ++ /*! \brief TRUE if connected_name is present */ ++ u_int8_t connected_name_present; ++}; ++ ++/* ++ * CTCompleteArg ::= SEQUENCE { ++ * endDesignation EndDesignation, ++ * redirectionNumber PresentedNumberScreened, ++ * ++ * -- ISO/IEC 11572 information elements Party ++ * -- category and Progress indicator are conveyed ++ * basicCallInfoElements PSS1InformationElement OPTIONAL, ++ * redirectionName Name OPTIONAL, ++ * callStatus CallStatus DEFAULT answered, ++ * argumentExtension CHOICE { ++ * [9] IMPLICIT Extension, ++ * [10] IMPLICIT SEQUENCE OF Extension ++ * } OPTIONAL ++ * } ++ */ ++struct roseQsigCTCompleteArg_ARG { ++ /*! \brief redirectionNumber */ ++ struct rosePresentedNumberScreened redirection; ++ ++ /*! \brief basicCallInfoElements (optional) */ ++ struct roseQ931ie q931ie; ++ /*! \brief q931ie.contents "allocated" after the stucture. */ ++ unsigned char q931ie_contents[ROSE_Q931_MAX_PROGRESS + 1]; ++ ++ /*! \brief redirectionName (optional) */ ++ struct roseQsigName redirection_name; ++ ++ /*! \brief TRUE if redirection_name is present */ ++ u_int8_t redirection_name_present; ++ ++ /*! ++ * \brief endDesignation ++ * \details ++ * primaryEnd(0), ++ * secondaryEnd(1) ++ */ ++ u_int8_t end_designation; ++ ++ /*! ++ * \brief callStatus ++ * \details ++ * DEFAULT answered ++ * ++ * \details ++ * answered(0), ++ * alerting(1) ++ */ ++ u_int8_t call_status; ++}; ++ ++/* ++ * CTUpdateArg ::= SEQUENCE { ++ * redirectionNumber PresentedNumberScreened, ++ * redirectionName Name OPTIONAL, ++ * ++ * -- ISO/IEC 11572 information elements Party ++ * -- category and Progress indicator are conveyed ++ * basicCallInfoElements PSS1InformationElement OPTIONAL, ++ * argumentExtension CHOICE { ++ * [9] IMPLICIT Extension, ++ * [10] IMPLICIT SEQUENCE OF Extension ++ * } OPTIONAL ++ * } ++ */ ++struct roseQsigCTUpdateArg_ARG { ++ /*! \brief redirectionNumber */ ++ struct rosePresentedNumberScreened redirection; ++ ++ /*! \brief basicCallInfoElements (optional) */ ++ struct roseQ931ie q931ie; ++ /*! \brief q931ie.contents "allocated" after the stucture. */ ++ unsigned char q931ie_contents[ROSE_Q931_MAX_PROGRESS + 1]; ++ ++ /*! \brief redirectionName (optional) */ ++ struct roseQsigName redirection_name; ++ ++ /*! \brief TRUE if redirection_name is present */ ++ u_int8_t redirection_name_present; ++}; ++ ++/* ++ * SubaddressTransferArg ::= SEQUENCE { ++ * redirectionSubaddress PartySubaddress, ++ * argumentExtension CHOICE { ++ * [0] IMPLICIT Extension, ++ * [1] IMPLICIT SEQUENCE OF Extension ++ * } OPTIONAL ++ * } ++ */ ++struct roseQsigSubaddressTransferArg_ARG { ++ struct rosePartySubaddress redirection_subaddress; ++}; ++ ++ ++/* ------------------------------------------------------------------- */ ++ ++ ++/* ++ * IntResult ::= SEQUENCE { ++ * servedUserNr PartyNumber, ++ * basicService BasicService, ++ * procedure Procedure, ++ * divertedToAddress Address, ++ * remoteEnabled BOOLEAN DEFAULT FALSE, ++ * extension CHOICE { ++ * [1] IMPLICIT Extension, ++ * [2] IMPLICIT SEQUENCE OF Extension ++ * } OPTIONAL ++ * } ++ */ ++struct roseQsigForwardingRecord { ++ /*! \brief Diverted to address */ ++ struct roseAddress diverted_to; ++ ++ struct rosePartyNumber served_user_number; ++ ++ /*! ++ * \details ++ * allServices(0), ++ * speech(1), ++ * unrestrictedDigitalInformation(2), ++ * audio3100Hz(3), ++ * telephony(32), ++ * teletex(33), ++ * telefaxGroup4Class1(34), ++ * videotexSyntaxBased(35), ++ * videotelephony(36) ++ */ ++ u_int8_t basic_service; ++ ++ /*! \details cfu(0), cfb(1), cfnr(2) */ ++ u_int8_t procedure; ++ ++ /*! \brief remoteEnabled BOOLEAN DEFAULT FALSE */ ++ u_int8_t remote_enabled; ++}; ++ ++/* ++ * roseQsigInterrogateDiversionQ_REQ ++ * IntResultList ::= SET SIZE (0..29) OF IntResult ++ */ ++struct roseQsigForwardingList { ++ /*! ++ * \brief SET SIZE (0..29) OF Forwarding Records ++ * \note Reduced the size of the array to conserve ++ * potential stack usage. ++ */ ++ struct roseQsigForwardingRecord list[10]; ++ ++ /*! \brief Number of Forwarding records present */ ++ u_int8_t num_records; ++}; ++ ++/* ++ * ARGUMENT SEQUENCE { ++ * procedure Procedure, ++ * basicService BasicService, ++ * divertedToAddress Address, ++ * servedUserNr PartyNumber, ++ * activatingUserNr PartyNumber, ++ * extension CHOICE { ++ * [1] IMPLICIT Extension, ++ * [2] IMPLICIT SEQUENCE OF Extension ++ * } OPTIONAL ++ * } ++ */ ++struct roseQsigActivateDiversionQ_ARG { ++ /*! \brief divertedToAddress */ ++ struct roseAddress diverted_to; ++ ++ struct rosePartyNumber served_user_number; ++ struct rosePartyNumber activating_user_number; ++ ++ /*! \details cfu(0), cfb(1), cfnr(2) */ ++ u_int8_t procedure; ++ ++ /*! ++ * \details ++ * allServices(0), ++ * speech(1), ++ * unrestrictedDigitalInformation(2), ++ * audio3100Hz(3), ++ * telephony(32), ++ * teletex(33), ++ * telefaxGroup4Class1(34), ++ * videotexSyntaxBased(35), ++ * videotelephony(36) ++ */ ++ u_int8_t basic_service; ++}; ++ ++/* ++ * ARGUMENT SEQUENCE { ++ * procedure Procedure, ++ * basicService BasicService, ++ * servedUserNr PartyNumber, ++ * deactivatingUserNr PartyNumber, ++ * extension CHOICE { ++ * [1] IMPLICIT Extension, ++ * [2] IMPLICIT SEQUENCE OF Extension ++ * } OPTIONAL ++ * } ++ */ ++struct roseQsigDeactivateDiversionQ_ARG { ++ struct rosePartyNumber served_user_number; ++ struct rosePartyNumber deactivating_user_number; ++ ++ /*! \details cfu(0), cfb(1), cfnr(2) */ ++ u_int8_t procedure; ++ ++ /*! ++ * \details ++ * allServices(0), ++ * speech(1), ++ * unrestrictedDigitalInformation(2), ++ * audio3100Hz(3), ++ * telephony(32), ++ * teletex(33), ++ * telefaxGroup4Class1(34), ++ * videotexSyntaxBased(35), ++ * videotelephony(36) ++ */ ++ u_int8_t basic_service; ++}; ++ ++/* ++ * ARGUMENT SEQUENCE { ++ * procedure Procedure, ++ * basicService BasicService DEFAULT allServices, ++ * servedUserNr PartyNumber, ++ * interrogatingUserNr PartyNumber, ++ * extension CHOICE { ++ * [1] IMPLICIT Extension, ++ * [2] IMPLICIT SEQUENCE OF Extension ++ * } OPTIONAL ++ * } ++ */ ++struct roseQsigInterrogateDiversionQ_ARG { ++ struct rosePartyNumber served_user_number; ++ struct rosePartyNumber interrogating_user_number; ++ ++ /*! \details cfu(0), cfb(1), cfnr(2) */ ++ u_int8_t procedure; ++ ++ /*! ++ * \details ++ * DEFAULT allServices ++ * ++ * \details ++ * allServices(0), ++ * speech(1), ++ * unrestrictedDigitalInformation(2), ++ * audio3100Hz(3), ++ * telephony(32), ++ * teletex(33), ++ * telefaxGroup4Class1(34), ++ * videotexSyntaxBased(35), ++ * videotelephony(36) ++ */ ++ u_int8_t basic_service; ++}; ++ ++/* ++ * ARGUMENT SEQUENCE { ++ * servedUserNr PartyNumber, ++ * basicService BasicService, ++ * divertedToNr PartyNumber, ++ * extension CHOICE { ++ * [1] IMPLICIT Extension, ++ * [2] IMPLICIT SEQUENCE OF Extension ++ * } OPTIONAL ++ * } ++ */ ++struct roseQsigCheckRestriction_ARG { ++ struct rosePartyNumber served_user_number; ++ struct rosePartyNumber diverted_to_number; ++ ++ /*! ++ * \details ++ * allServices(0), ++ * speech(1), ++ * unrestrictedDigitalInformation(2), ++ * audio3100Hz(3), ++ * telephony(32), ++ * teletex(33), ++ * telefaxGroup4Class1(34), ++ * videotexSyntaxBased(35), ++ * videotelephony(36) ++ */ ++ u_int8_t basic_service; ++}; ++ ++/* ++ * ARGUMENT SEQUENCE { ++ * reroutingReason DiversionReason, ++ * originalReroutingReason [0] IMPLICIT DiversionReason OPTIONAL, ++ * calledAddress Address, ++ * diversionCounter INTEGER (1..15), ++ * ++ * -- The basic call information elements Bearer capability, ++ * -- High layer compatibility, Low layer compatibity ++ * -- and Progress indicator can be embedded in the ++ * -- pSS1InfoElement in accordance with 6.5.3.1.5. ++ * pSS1InfoElement PSS1InformationElement, ++ * lastReroutingNr [1] EXPLICIT PresentedNumberUnscreened, ++ * subscriptionOption [2] IMPLICIT SubscriptionOption, ++ * callingPartySubaddress [3] EXPLICIT PartySubaddress OPTIONAL, ++ * callingNumber [4] EXPLICIT PresentedNumberScreened, ++ * callingName [5] EXPLICIT Name OPTIONAL, ++ * originalCalledNr [6] EXPLICIT PresentedNumberUnscreened OPTIONAL, ++ * redirectingName [7] EXPLICIT Name OPTIONAL, ++ * originalCalledName [8] EXPLICIT Name OPTIONAL, ++ * extension CHOICE { ++ * [9] IMPLICIT Extension, ++ * [10] IMPLICIT SEQUENCE OF Extension ++ * } OPTIONAL ++ * } ++ */ ++struct roseQsigCallRerouting_ARG { ++ /*! \brief calledAddress */ ++ struct roseAddress called; ++ ++ /*! \brief lastReroutingNr */ ++ struct rosePresentedNumberUnscreened last_rerouting; ++ ++ /*! \brief originalCalledNr (optional) */ ++ struct rosePresentedNumberUnscreened original_called; ++ ++ /*! ++ * \brief callingPartySubaddress (optional) ++ * The subaddress is present if the length is nonzero. ++ */ ++ struct rosePartySubaddress calling_subaddress; ++ ++ /*! \brief callingNumber */ ++ struct rosePresentedNumberScreened calling; ++ ++ /*! \brief callingName (optional) */ ++ struct roseQsigName calling_name; ++ ++ /*! \brief redirectingName (optional) */ ++ struct roseQsigName redirecting_name; ++ ++ /*! \brief originalCalledName (optional) */ ++ struct roseQsigName original_called_name; ++ ++ /*! ++ * \brief The BC, HLC (optional), LLC (optional), ++ * and progress indicator(s) information. ++ */ ++ struct roseQ931ie q931ie; ++ /*! \brief q931ie.contents "allocated" after the stucture. */ ++ unsigned char q931ie_contents[ROSE_Q931_MAX_BC + ROSE_Q931_MAX_HLC + ++ ROSE_Q931_MAX_LLC + ROSE_Q931_MAX_PROGRESS + 1]; ++ ++ /*! \brief TRUE if calling_name is present */ ++ u_int8_t calling_name_present; ++ ++ /*! \brief TRUE if redirecting_name is present */ ++ u_int8_t redirecting_name_present; ++ ++ /*! \brief TRUE if original_called_name is present */ ++ u_int8_t original_called_name_present; ++ ++ /*! \brief TRUE if original_called number is present */ ++ u_int8_t original_called_present; ++ ++ /*! ++ * \details ++ * unknown(0), ++ * cfu(1), ++ * cfb(2), ++ * cfnr(3) ++ * ++ * \note The value unknown is only used if received from ++ * another network when interworking. ++ */ ++ u_int8_t rerouting_reason; ++ ++ /*! ++ * \brief originalReroutingReason (optional) ++ * ++ * \details ++ * unknown(0), ++ * cfu(1), ++ * cfb(2), ++ * cfnr(3) ++ * ++ * \note The value unknown is only used if received from ++ * another network when interworking. ++ */ ++ u_int8_t original_rerouting_reason; ++ ++ /*! \brief TRUE if original_rerouting_reason is present */ ++ u_int8_t original_rerouting_reason_present; ++ ++ /*! \brief diversionCounter INTEGER (1..15) */ ++ u_int8_t diversion_counter; ++ ++ /*! ++ * \brief subscriptionOption ++ * ++ * \details ++ * noNotification(0), ++ * notificationWithoutDivertedToNr(1), ++ * notificationWithDivertedToNr(2) ++ */ ++ u_int8_t subscription_option; ++}; ++ ++/* ++ * ARGUMENT SEQUENCE { ++ * diversionReason DiversionReason, ++ * subscriptionOption SubscriptionOption, ++ * nominatedNr PartyNumber, ++ * extension CHOICE { ++ * [9] IMPLICIT Extension, ++ * [10] IMPLICIT SEQUENCE OF Extension ++ * } OPTIONAL ++ * } ++ */ ++struct roseQsigDivertingLegInformation1_ARG { ++ struct rosePartyNumber nominated_number; ++ ++ /*! ++ * \details ++ * unknown(0), ++ * cfu(1), ++ * cfb(2), ++ * cfnr(3) ++ * ++ * \note The value unknown is only used if received from ++ * another network when interworking. ++ */ ++ u_int8_t diversion_reason; ++ ++ /*! ++ * \brief subscriptionOption ++ * ++ * \details ++ * noNotification(0), ++ * notificationWithoutDivertedToNr(1), ++ * notificationWithDivertedToNr(2) ++ */ ++ u_int8_t subscription_option; ++}; ++ ++/* ++ * ARGUMENT SEQUENCE { ++ * diversionCounter INTEGER (1..15), ++ * diversionReason DiversionReason, ++ * originalDiversionReason [0] IMPLICIT DiversionReason OPTIONAL, ++ * ++ * -- The divertingNr element is mandatory except in the case ++ * -- of interworking. ++ * divertingNr [1] EXPLICIT PresentedNumberUnscreened OPTIONAL, ++ * originalCalledNr [2] EXPLICIT PresentedNumberUnscreened OPTIONAL, ++ * redirectingName [3] EXPLICIT Name OPTIONAL, ++ * originalCalledName [4] EXPLICIT Name OPTIONAL, ++ * extension CHOICE { ++ * [5] IMPLICIT Extension, ++ * [6] IMPLICIT SEQUENCE OF Extension ++ * } OPTIONAL ++ * } ++ */ ++struct roseQsigDivertingLegInformation2_ARG { ++ /*! \brief divertingNr (optional) */ ++ struct rosePresentedNumberUnscreened diverting; ++ ++ /*! \brief originalCalledNr (optional) */ ++ struct rosePresentedNumberUnscreened original_called; ++ ++ /*! \brief redirectingName (optional) */ ++ struct roseQsigName redirecting_name; ++ ++ /*! \brief originalCalledName (optional) */ ++ struct roseQsigName original_called_name; ++ ++ /*! \brief diversionCounter INTEGER (1..15) */ ++ u_int8_t diversion_counter; ++ ++ /*! ++ * \details ++ * unknown(0), ++ * cfu(1), ++ * cfb(2), ++ * cfnr(3) ++ * ++ * \note The value unknown is only used if received from ++ * another network when interworking. ++ */ ++ u_int8_t diversion_reason; ++ ++ /*! ++ * \brief originalDiversionReason (optional) ++ * ++ * \details ++ * unknown(0), ++ * cfu(1), ++ * cfb(2), ++ * cfnr(3) ++ * ++ * \note The value unknown is only used if received from ++ * another network when interworking. ++ */ ++ u_int8_t original_diversion_reason; ++ ++ /*! \brief TRUE if original_diversion_reason is present */ ++ u_int8_t original_diversion_reason_present; ++ ++ /*! \brief TRUE if diverting number is present */ ++ u_int8_t diverting_present; ++ ++ /*! \brief TRUE if original_called number is present */ ++ u_int8_t original_called_present; ++ ++ /*! \brief TRUE if redirecting_name is present */ ++ u_int8_t redirecting_name_present; ++ ++ /*! \brief TRUE if original_called_name is present */ ++ u_int8_t original_called_name_present; ++}; ++ ++/* ++ * ARGUMENT SEQUENCE { ++ * presentationAllowedIndicator PresentationAllowedIndicator, -- BOOLEAN ++ * redirectionName [0] EXPLICIT Name OPTIONAL, ++ * extension CHOICE { ++ * [1] IMPLICIT Extension, ++ * [2] IMPLICIT SEQUENCE OF Extension ++ * } OPTIONAL ++ * } ++ */ ++struct roseQsigDivertingLegInformation3_ARG { ++ /*! \brief redirectionName (optional) */ ++ struct roseQsigName redirection_name; ++ ++ /*! \brief TRUE if redirection_name is present */ ++ u_int8_t redirection_name_present; ++ ++ /*! \brief TRUE if presentation is allowed */ ++ u_int8_t presentation_allowed_indicator; ++}; ++ ++ ++/* ------------------------------------------------------------------- */ ++ ++ ++/* ++ * MsgCentreId ::= CHOICE { ++ * integer [0] IMPLICIT INTEGER (0..65535), ++ * ++ * -- The party number must be a complete number as required ++ * -- for routing purposes. ++ * partyNumber [1] EXPLICIT PartyNumber, ++ * numericString [2] IMPLICIT NumericString (SIZE (1..10)) ++ * } ++ */ ++struct roseQsigMsgCentreId { ++ union { ++ /*! \brief INTEGER (0..65535) */ ++ u_int16_t integer; ++ ++ /*! ++ * \note The party number must be a complete number as required ++ * for routing purposes. ++ */ ++ struct rosePartyNumber number; ++ ++ /*! \brief NumericString (SIZE (1..10)) */ ++ unsigned char str[10 + 1]; ++ } u; ++ ++ /*! ++ * \details ++ * integer(0), ++ * partyNumber(1), ++ * numericString(2) ++ */ ++ u_int8_t type; ++}; ++ ++/* ++ * MWIActivateArg ::= SEQUENCE { ++ * servedUserNr PartyNumber, ++ * basicService BasicService, ++ * msgCentreId MsgCentreId OPTIONAL, ++ * nbOfMessages [3] IMPLICIT NbOfMessages OPTIONAL, ++ * originatingNr [4] EXPLICIT PartyNumber OPTIONAL, ++ * timestamp TimeStamp OPTIONAL, ++ * ++ * -- The value 0 means the highest priority and 9 the lowest ++ * priority [5] IMPLICIT INTEGER (0..9) OPTIONAL, ++ * argumentExt CHOICE { ++ * extension [6] IMPLICIT Extension, ++ * multipleExtension [7] IMPLICIT SEQUENCE OF Extension ++ * } OPTIONAL ++ * } ++ */ ++struct roseQsigMWIActivateArg { ++ /*! \brief NbOfMessages ::= INTEGER (0..65535) (optional) */ ++ u_int16_t number_of_messages; ++ ++ /*! \brief msgCentreId (optional) */ ++ struct roseQsigMsgCentreId msg_centre_id; ++ ++ struct rosePartyNumber served_user_number; ++ ++ /*! \brief originatingNr (optional) (Number present if length is nonzero) */ ++ struct rosePartyNumber originating_number; ++ ++ /*! \brief GeneralizedTime (SIZE (12..19)) (optional) */ ++ unsigned char timestamp[19 + 1]; ++ ++ /*! ++ * \details ++ * allServices(0), ++ * speech(1), ++ * unrestrictedDigitalInformation(2), ++ * audio3100Hz(3), ++ * telephony(32), ++ * teletex(33), ++ * telefaxGroup4Class1(34), ++ * videotextSyntaxBased(35), ++ * videotelephony(36), ++ * telefaxGroup2-3(37), ++ * reservedNotUsed1(38), ++ * reservedNotUsed2(39), ++ * reservedNotUsed3(40), ++ * reservedNotUsed4(41), ++ * reservedNotUsed5(42), ++ * email(51), ++ * video(52), ++ * fileTransfer(53), ++ * shortMessageService(54), ++ * speechAndVideo(55), ++ * speechAndFax(56), ++ * speechAndEmail(57), ++ * videoAndFax(58), ++ * videoAndEmail(59), ++ * faxAndEmail(60), ++ * speechVideoAndFax(61), ++ * speechVideoAndEmail(62), ++ * speechFaxAndEmail(63), ++ * videoFaxAndEmail(64), ++ * speechVideoFaxAndEmail(65), ++ * multimediaUnknown(66), ++ * serviceUnknown(67), ++ * futureReserve1(68), ++ * futureReserve2(69), ++ * futureReserve3(70), ++ * futureReserve4(71), ++ * futureReserve5(72), ++ * futureReserve6(73), ++ * futureReserve7(74), ++ * futureReserve8(75) ++ */ ++ u_int8_t basic_service; ++ ++ /*! ++ * \brief INTEGER (0..9) (optional) ++ * \note The value 0 means the highest priority and 9 the lowest. ++ */ ++ u_int8_t priority; ++ ++ /*! \brief TRUE if msg_centre_id is present */ ++ u_int8_t msg_centre_id_present; ++ ++ /*! \brief TRUE if number_of_messages is present */ ++ u_int8_t number_of_messages_present; ++ ++ /*! \brief TRUE if timestamp is present */ ++ u_int8_t timestamp_present; ++ ++ /*! \brief TRUE if priority is present */ ++ u_int8_t priority_present; ++}; ++ ++/* ++ * MWIDeactivateArg ::= SEQUENCE { ++ * servedUserNr PartyNumber, ++ * basicService BasicService, ++ * msgCentreId MsgCentreId OPTIONAL, ++ * argumentExt CHOICE { ++ * extension [3] IMPLICIT Extension, ++ * multipleExtension [4] IMPLICIT SEQUENCE OF Extension ++ * } OPTIONAL ++ * } ++ */ ++struct roseQsigMWIDeactivateArg { ++ /*! \brief msgCentreId (optional) */ ++ struct roseQsigMsgCentreId msg_centre_id; ++ ++ struct rosePartyNumber served_user_number; ++ ++ /*! ++ * \details ++ * allServices(0), ++ * speech(1), ++ * unrestrictedDigitalInformation(2), ++ * audio3100Hz(3), ++ * telephony(32), ++ * teletex(33), ++ * telefaxGroup4Class1(34), ++ * videotextSyntaxBased(35), ++ * videotelephony(36), ++ * telefaxGroup2-3(37), ++ * reservedNotUsed1(38), ++ * reservedNotUsed2(39), ++ * reservedNotUsed3(40), ++ * reservedNotUsed4(41), ++ * reservedNotUsed5(42), ++ * email(51), ++ * video(52), ++ * fileTransfer(53), ++ * shortMessageService(54), ++ * speechAndVideo(55), ++ * speechAndFax(56), ++ * speechAndEmail(57), ++ * videoAndFax(58), ++ * videoAndEmail(59), ++ * faxAndEmail(60), ++ * speechVideoAndFax(61), ++ * speechVideoAndEmail(62), ++ * speechFaxAndEmail(63), ++ * videoFaxAndEmail(64), ++ * speechVideoFaxAndEmail(65), ++ * multimediaUnknown(66), ++ * serviceUnknown(67), ++ * futureReserve1(68), ++ * futureReserve2(69), ++ * futureReserve3(70), ++ * futureReserve4(71), ++ * futureReserve5(72), ++ * futureReserve6(73), ++ * futureReserve7(74), ++ * futureReserve8(75) ++ */ ++ u_int8_t basic_service; ++ ++ /*! \brief TRUE if msg_centre_id is present */ ++ u_int8_t msg_centre_id_present; ++}; ++ ++/* ++ * MWIInterrogateArg ::= SEQUENCE { ++ * servedUserNr PartyNumber, ++ * basicService BasicService, ++ * msgCentreId MsgCentreId OPTIONAL, ++ * argumentExt CHOICE { ++ * extension [3] IMPLICIT Extension, ++ * multipleExtension [4] IMPLICIT SEQUENCE OF Extension ++ * } OPTIONAL ++ * } ++ */ ++struct roseQsigMWIInterrogateArg { ++ /*! \brief msgCentreId (optional) */ ++ struct roseQsigMsgCentreId msg_centre_id; ++ ++ struct rosePartyNumber served_user_number; ++ ++ /*! ++ * \details ++ * allServices(0), ++ * speech(1), ++ * unrestrictedDigitalInformation(2), ++ * audio3100Hz(3), ++ * telephony(32), ++ * teletex(33), ++ * telefaxGroup4Class1(34), ++ * videotextSyntaxBased(35), ++ * videotelephony(36), ++ * telefaxGroup2-3(37), ++ * reservedNotUsed1(38), ++ * reservedNotUsed2(39), ++ * reservedNotUsed3(40), ++ * reservedNotUsed4(41), ++ * reservedNotUsed5(42), ++ * email(51), ++ * video(52), ++ * fileTransfer(53), ++ * shortMessageService(54), ++ * speechAndVideo(55), ++ * speechAndFax(56), ++ * speechAndEmail(57), ++ * videoAndFax(58), ++ * videoAndEmail(59), ++ * faxAndEmail(60), ++ * speechVideoAndFax(61), ++ * speechVideoAndEmail(62), ++ * speechFaxAndEmail(63), ++ * videoFaxAndEmail(64), ++ * speechVideoFaxAndEmail(65), ++ * multimediaUnknown(66), ++ * serviceUnknown(67), ++ * futureReserve1(68), ++ * futureReserve2(69), ++ * futureReserve3(70), ++ * futureReserve4(71), ++ * futureReserve5(72), ++ * futureReserve6(73), ++ * futureReserve7(74), ++ * futureReserve8(75) ++ */ ++ u_int8_t basic_service; ++ ++ /*! \brief TRUE if msg_centre_id is present */ ++ u_int8_t msg_centre_id_present; ++}; ++ ++/* ++ * MWIInterrogateResElt ::= SEQUENCE { ++ * basicService BasicService, ++ * msgCentreId MsgCentreId OPTIONAL, ++ * nbOfMessages [3] IMPLICIT NbOfMessages OPTIONAL, ++ * originatingNr [4] EXPLICIT PartyNumber OPTIONAL, ++ * timestamp TimeStamp OPTIONAL, ++ * ++ * -- The value 0 means the highest priority and 9 the lowest ++ * priority [5] IMPLICIT INTEGER (0..9) OPTIONAL, ++ * argumentExt CHOICE { ++ * extension [6] IMPLICIT Extension, ++ * multipleExtension [7] IMPLICIT SEQUENCE OF Extension ++ * } OPTIONAL ++ * } ++ */ ++struct roseQsigMWIInterrogateResElt { ++ /*! \brief NbOfMessages ::= INTEGER (0..65535) (optional) */ ++ u_int16_t number_of_messages; ++ ++ /*! \brief msgCentreId (optional) */ ++ struct roseQsigMsgCentreId msg_centre_id; ++ ++ /*! \brief originatingNr (optional) (Number present if length is nonzero) */ ++ struct rosePartyNumber originating_number; ++ ++ /*! \brief GeneralizedTime (SIZE (12..19)) (optional) */ ++ unsigned char timestamp[19 + 1]; ++ ++ /*! ++ * \details ++ * allServices(0), ++ * speech(1), ++ * unrestrictedDigitalInformation(2), ++ * audio3100Hz(3), ++ * telephony(32), ++ * teletex(33), ++ * telefaxGroup4Class1(34), ++ * videotextSyntaxBased(35), ++ * videotelephony(36), ++ * telefaxGroup2-3(37), ++ * reservedNotUsed1(38), ++ * reservedNotUsed2(39), ++ * reservedNotUsed3(40), ++ * reservedNotUsed4(41), ++ * reservedNotUsed5(42), ++ * email(51), ++ * video(52), ++ * fileTransfer(53), ++ * shortMessageService(54), ++ * speechAndVideo(55), ++ * speechAndFax(56), ++ * speechAndEmail(57), ++ * videoAndFax(58), ++ * videoAndEmail(59), ++ * faxAndEmail(60), ++ * speechVideoAndFax(61), ++ * speechVideoAndEmail(62), ++ * speechFaxAndEmail(63), ++ * videoFaxAndEmail(64), ++ * speechVideoFaxAndEmail(65), ++ * multimediaUnknown(66), ++ * serviceUnknown(67), ++ * futureReserve1(68), ++ * futureReserve2(69), ++ * futureReserve3(70), ++ * futureReserve4(71), ++ * futureReserve5(72), ++ * futureReserve6(73), ++ * futureReserve7(74), ++ * futureReserve8(75) ++ */ ++ u_int8_t basic_service; ++ ++ /*! ++ * \brief INTEGER (0..9) (optional) ++ * \note The value 0 means the highest priority and 9 the lowest. ++ */ ++ u_int8_t priority; ++ ++ /*! \brief TRUE if msg_centre_id is present */ ++ u_int8_t msg_centre_id_present; ++ ++ /*! \brief TRUE if number_of_messages is present */ ++ u_int8_t number_of_messages_present; ++ ++ /*! \brief TRUE if timestamp is present */ ++ u_int8_t timestamp_present; ++ ++ /*! \brief TRUE if priority is present */ ++ u_int8_t priority_present; ++}; ++ ++/* ++ * MWIInterrogateRes ::= SEQUENCE SIZE(1..10) OF MWIInterrogateResElt ++ */ ++struct roseQsigMWIInterrogateRes { ++ /*! \brief SEQUENCE SIZE(1..10) OF MWIInterrogateResElt */ ++ struct roseQsigMWIInterrogateResElt list[10]; ++ ++ /*! \brief Number of MWIInterrogateResElt records present */ ++ u_int8_t num_records; ++}; ++ ++ ++/* ------------------------------------------------------------------- */ ++ ++ ++/* ++ * Northern Telecom DMS-100 transfer ability result ++ * ++ * callId [0] IMPLICIT INTEGER (0..16777215) -- 24 bit number ++ */ ++struct roseDms100RLTOperationInd_RES { ++ /*! INTEGER (0..16777215) -- 24 bit number */ ++ u_int32_t call_id; ++}; ++ ++/* ++ * Northern Telecom DMS-100 transfer invoke ++ * ++ * ARGUMENT SEQUENCE { ++ * callId [0] IMPLICIT INTEGER (0..16777215), -- 24 bit number ++ * reason [1] IMPLICIT INTEGER ++ * } ++ */ ++struct roseDms100RLTThirdParty_ARG { ++ /*! INTEGER (0..16777215) -- 24 bit number */ ++ u_int32_t call_id; ++ ++ /*! Reason for redirect */ ++ u_int8_t reason; ++}; ++ ++ ++/* ------------------------------------------------------------------- */ ++ ++ ++/* ARGUMENT ENUMERATED */ ++struct roseNi2InformationFollowing_ARG { ++ u_int8_t value; /*!< Unknown enumerated value */ ++}; ++ ++/* ++ * ARGUMENT SEQUENCE { ++ * callReference INTEGER -- 16 bit number ++ * } ++ */ ++struct roseNi2InitiateTransfer_ARG { ++ u_int16_t call_reference; ++}; ++ ++ ++/* ------------------------------------------------------------------- */ ++ ++ ++/*! \brief Facility ie invoke etsi messages with arguments. */ ++union rose_msg_invoke_etsi_args { ++ /* ETSI Advice Of Charge (AOC) */ ++ struct roseEtsiChargingRequest_ARG ChargingRequest; ++ struct roseEtsiAOCSCurrency_ARG AOCSCurrency; ++ struct roseEtsiAOCSSpecialArr_ARG AOCSSpecialArr; ++ struct roseEtsiAOCDCurrency_ARG AOCDCurrency; ++ struct roseEtsiAOCDChargingUnit_ARG AOCDChargingUnit; ++ struct roseEtsiAOCECurrency_ARG AOCECurrency; ++ struct roseEtsiAOCEChargingUnit_ARG AOCEChargingUnit; ++ ++ /* ETSI Call Diversion */ ++ struct roseEtsiActivationDiversion_ARG ActivationDiversion; ++ struct roseEtsiDeactivationDiversion_ARG DeactivationDiversion; ++ struct roseEtsiActivationStatusNotificationDiv_ARG ActivationStatusNotificationDiv; ++ struct roseEtsiDeactivationStatusNotificationDiv_ARG ++ DeactivationStatusNotificationDiv; ++ struct roseEtsiInterrogationDiversion_ARG InterrogationDiversion; ++ struct roseEtsiDiversionInformation_ARG DiversionInformation; ++ struct roseEtsiCallDeflection_ARG CallDeflection; ++ struct roseEtsiCallRerouting_ARG CallRerouting; ++ struct roseEtsiDivertingLegInformation1_ARG DivertingLegInformation1; ++ struct roseEtsiDivertingLegInformation2_ARG DivertingLegInformation2; ++ struct roseEtsiDivertingLegInformation3_ARG DivertingLegInformation3; ++ ++ /* ETSI Explicit Call Transfer (ECT) */ ++ struct roseEtsiExplicitEctExecute_ARG ExplicitEctExecute; ++ struct roseEtsiSubaddressTransfer_ARG SubaddressTransfer; ++ struct roseEtsiEctInform_ARG EctInform; ++ struct roseEtsiEctLoopTest_ARG EctLoopTest; ++}; ++ ++/*! \brief Facility ie result etsi messages with arguments. */ ++union rose_msg_result_etsi_args { ++ /* ETSI Advice Of Charge (AOC) */ ++ struct roseEtsiChargingRequest_RES ChargingRequest; ++ ++ /* ETSI Call Diversion */ ++ struct roseEtsiForwardingList InterrogationDiversion; ++ struct roseEtsiServedUserNumberList InterrogateServedUserNumbers; ++ ++ /* ETSI Explicit Call Transfer (ECT) */ ++ struct roseEtsiEctLinkIdRequest_RES EctLinkIdRequest; ++ struct roseEtsiEctLoopTest_RES EctLoopTest; ++}; ++ ++/*! \brief Facility ie invoke qsig messages with arguments. */ ++union rose_msg_invoke_qsig_args { ++ /* Q.SIG Name-Operations */ ++ struct roseQsigPartyName_ARG CallingName; ++ struct roseQsigPartyName_ARG CalledName; ++ struct roseQsigPartyName_ARG ConnectedName; ++ struct roseQsigPartyName_ARG BusyName; ++ ++ /* Q.SIG SS-AOC-Operations */ ++ struct roseQsigChargeRequestArg_ARG ChargeRequest; ++ struct roseQsigAocFinalArg_ARG AocFinal; ++ struct roseQsigAocInterimArg_ARG AocInterim; ++ struct roseQsigAocRateArg_ARG AocRate; ++ struct roseQsigAocCompleteArg_ARG AocComplete; ++ struct roseQsigAocDivChargeReqArg_ARG AocDivChargeReq; ++ ++ /* Q.SIG Call-Transfer-Operations */ ++ struct roseQsigCTInitiateArg_ARG CallTransferInitiate; ++ struct roseQsigCTSetupArg_ARG CallTransferSetup; ++ struct roseQsigCTActiveArg_ARG CallTransferActive; ++ struct roseQsigCTCompleteArg_ARG CallTransferComplete; ++ struct roseQsigCTUpdateArg_ARG CallTransferUpdate; ++ struct roseQsigSubaddressTransferArg_ARG SubaddressTransfer; ++ ++ /* Q.SIG Call-Diversion-Operations */ ++ struct roseQsigActivateDiversionQ_ARG ActivateDiversionQ; ++ struct roseQsigDeactivateDiversionQ_ARG DeactivateDiversionQ; ++ struct roseQsigInterrogateDiversionQ_ARG InterrogateDiversionQ; ++ struct roseQsigCheckRestriction_ARG CheckRestriction; ++ struct roseQsigCallRerouting_ARG CallRerouting; ++ struct roseQsigDivertingLegInformation1_ARG DivertingLegInformation1; ++ struct roseQsigDivertingLegInformation2_ARG DivertingLegInformation2; ++ struct roseQsigDivertingLegInformation3_ARG DivertingLegInformation3; ++ ++ /* Q.SIG SS-MWI-Operations */ ++ struct roseQsigMWIActivateArg MWIActivate; ++ struct roseQsigMWIDeactivateArg MWIDeactivate; ++ struct roseQsigMWIInterrogateArg MWIInterrogate; ++}; ++ ++/*! \brief Facility ie result qsig messages with arguments. */ ++union rose_msg_result_qsig_args { ++ /* Q.SIG SS-AOC-Operations */ ++ struct roseQsigChargeRequestRes_RES ChargeRequest; ++ struct roseQsigAocCompleteRes_RES AocComplete; ++ ++ /* Q.SIG Call-Transfer-Operations */ ++ struct roseQsigCTIdentifyRes_RES CallTransferIdentify; ++ ++ /* Q.SIG Call-Diversion-Operations */ ++ struct roseQsigForwardingList InterrogateDiversionQ; ++ ++ /* Q.SIG SS-MWI-Operations */ ++ struct roseQsigMWIInterrogateRes MWIInterrogate; ++}; ++ ++/*! \brief Facility ie invoke DMS-100 messages with arguments. */ ++union rose_msg_invoke_dms100_args { ++ struct roseDms100RLTThirdParty_ARG RLT_ThirdParty; ++}; ++ ++/*! \brief Facility ie result DMS-100 messages with arguments. */ ++union rose_msg_result_dms100_args { ++ struct roseDms100RLTOperationInd_RES RLT_OperationInd; ++}; ++ ++/*! \brief Facility ie invoke NI2 messages with arguments. */ ++union rose_msg_invoke_ni2_args { ++ struct roseNi2InformationFollowing_ARG InformationFollowing; ++ struct roseNi2InitiateTransfer_ARG InitiateTransfer; ++}; ++ ++/*! \brief Facility ie result NI2 messages with arguments. */ ++union rose_msg_result_ni2_args { ++ int dummy; /*!< place holder until there are results with parameters */ ++}; ++ ++/*! \brief Facility ie invoke messages with arguments. */ ++union rose_msg_invoke_args { ++ union rose_msg_invoke_etsi_args etsi; ++ union rose_msg_invoke_qsig_args qsig; ++ union rose_msg_invoke_dms100_args dms100; ++ union rose_msg_invoke_ni2_args ni2; ++}; ++ ++/*! \brief Facility ie result messages with arguments. */ ++union rose_msg_result_args { ++ union rose_msg_result_etsi_args etsi; ++ union rose_msg_result_qsig_args qsig; ++ union rose_msg_result_dms100_args dms100; ++ union rose_msg_result_ni2_args ni2; ++}; ++ ++/*! \brief Facility ie error messages with parameters. */ ++union rose_msg_error_args { ++ int dummy; /*!< place holder until there are errors with parameters */ ++}; ++ ++struct rose_msg_invoke { ++ /*! \brief Invoke ID (-32768..32767) */ ++ int16_t invoke_id; ++ /*! \brief Linked ID (-32768..32767) (optional) */ ++ int16_t linked_id; ++ /*! \brief library encoded operation-value */ ++ enum rose_operation operation; ++ /*! \brief TRUE if the Linked ID is present */ ++ u_int8_t linked_id_present; ++ union rose_msg_invoke_args args; ++}; ++ ++struct rose_msg_result { ++ /*! \brief Invoke ID (-32768..32767) */ ++ int16_t invoke_id; ++ /*! ++ * \brief library encoded operation-value ++ * \note Set to ROSE_None if the operation sequence is not present. ++ * \note ETSI and Q.SIG imply that if a return result does not have ++ * any arguments then you must rely upon the invokeId value to ++ * distinguish between return results because the operation-value is ++ * not present. ++ */ ++ enum rose_operation operation; ++ union rose_msg_result_args args; ++}; ++ ++struct rose_msg_error { ++ /*! \brief Invoke ID (-32768..32767) */ ++ int16_t invoke_id; ++ /*! \brief library encoded error-value */ ++ enum rose_error_code code; ++ union rose_msg_error_args args; ++}; ++ ++struct rose_msg_reject { ++ /*! \brief Invoke ID (-32768..32767) (optional) */ ++ int16_t invoke_id; ++ /*! \brief TRUE if the Invoke ID is present */ ++ u_int8_t invoke_id_present; ++ /*! \brief library encoded problem-value */ ++ enum rose_reject_code code; ++}; ++ ++enum rose_component_type { ++ ROSE_COMP_TYPE_INVALID, ++ ROSE_COMP_TYPE_INVOKE, ++ ROSE_COMP_TYPE_RESULT, ++ ROSE_COMP_TYPE_ERROR, ++ ROSE_COMP_TYPE_REJECT ++}; ++ ++struct rose_message { ++ /*! \brief invoke, result, error, reject */ ++ enum rose_component_type type; ++ union { ++ struct rose_msg_invoke invoke; ++ struct rose_msg_result result; ++ struct rose_msg_error error; ++ struct rose_msg_reject reject; ++ } component; ++}; ++ ++/* ++ * NetworkFacilityExtension ::= [10] IMPLICIT SEQUENCE { ++ * sourceEntity [0] IMPLICIT EntityType, ++ * sourceEntityAddress [1] EXPLICIT AddressInformation OPTIONAL, ++ * destinationEntity [2] IMPLICIT EntityType, ++ * destinationEntityAddress [3] EXPLICIT AddressInformation OPTIONAL ++ * } ++ * ++ * AddressInformation ::= PartyNumber ++ */ ++struct facNetworkFacilityExtension { ++ /*! \brief sourceEntityAddress (optional) (Number present if length is nonzero) */ ++ struct rosePartyNumber source_number; ++ /*! \brief destinationEntityAddress (optional) (Number present if length is nonzero) */ ++ struct rosePartyNumber destination_number; ++ ++ /*! ++ * \details ++ * endPINX(0), ++ * anyTypeOfPINX(1) ++ */ ++ u_int8_t source_entity; ++ ++ /*! ++ * \details ++ * endPINX(0), ++ * anyTypeOfPINX(1) ++ */ ++ u_int8_t destination_entity; ++}; ++ ++/* ++ * The network extensions header is a sequence of the following components: ++ * ++ * nfe NetworkFacilityExtension OPTIONAL, ++ * npp NetworkProtocolProfile OPTIONAL, ++ * interpretation InterpretationApdu OPTIONAL ++ * ++ * NetworkProtocolProfile ::= [18] IMPLICIT INTEGER (0..254) ++ * ++ * InterpretationApdu ::= [11] IMPLICIT ENUMERATED { ++ * discardAnyUnrecognisedInvokePdu(0), ++ * ++ * -- this value also applies to Call independent signalling connections ++ * -- see clause 8.1.2 (ECMA-165) ++ * clearCallIfAnyInvokePduNotRecognised(1), ++ * ++ * -- this coding is implied by the absence of an ++ * -- interpretation APDU. ++ * rejectAnyUnrecognisedInvokePdu(2) ++ * } ++ */ ++struct fac_extension_header { ++ /*! \brief Network Facility Extension component */ ++ struct facNetworkFacilityExtension nfe; ++ ++ /*! \brief Network Protocol Profile component */ ++ u_int8_t npp; ++ ++ /*! ++ * \brief interpretation component ++ * ++ * \details ++ * discardAnyUnrecognisedInvokePdu(0), ++ * clearCallIfAnyInvokePduNotRecognised(1), ++ * rejectAnyUnrecognisedInvokePdu(2) ++ */ ++ u_int8_t interpretation; ++ ++ /*! \brief TRUE if nfe is present */ ++ u_int8_t nfe_present; ++ ++ /*! \brief TRUE if npp is present */ ++ u_int8_t npp_present; ++ ++ /*! \brief TRUE if interpretation is present */ ++ u_int8_t interpretation_present; ++}; ++ ++const char *rose_operation2str(enum rose_operation operation); ++const char *rose_error2str(enum rose_error_code code); ++const char *rose_reject2str(enum rose_reject_code code); ++ ++unsigned char *rose_encode_invoke(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const struct rose_msg_invoke *msg); ++unsigned char *rose_encode_result(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const struct rose_msg_result *msg); ++unsigned char *rose_encode_error(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const struct rose_msg_error *msg); ++unsigned char *rose_encode_reject(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const struct rose_msg_reject *msg); ++ ++unsigned char *rose_encode(struct pri *ctrl, unsigned char *pos, unsigned char *end, ++ const struct rose_message *msg); ++const unsigned char *rose_decode(struct pri *ctrl, const unsigned char *pos, ++ const unsigned char *end, struct rose_message *msg); ++ ++unsigned char *fac_enc_extension_header(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const struct fac_extension_header *header); ++unsigned char *facility_encode_header(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const struct fac_extension_header *header); ++ ++const unsigned char *fac_dec_extension_header(struct pri *ctrl, const unsigned char *pos, ++ const unsigned char *end, struct fac_extension_header *header); ++const unsigned char *facility_decode_header(struct pri *ctrl, const unsigned char *pos, ++ const unsigned char *end, struct fac_extension_header *header); ++ ++void facility_decode_dump(struct pri *ctrl, const unsigned char *buf, size_t length); ++ ++/* ------------------------------------------------------------------- */ ++ ++#ifdef __cplusplus ++} ++#endif ++ ++#endif /* _LIBPRI_ROSE_H */ ++/* ------------------------------------------------------------------- */ ++/* end rose.h */ + +Property changes on: rose.h +___________________________________________________________________ +Added: svn:eol-style + + native +Added: svn:mime-type + + text/plain +Added: svn:keywords + + 'Author Date Id Revision' + +Index: rose_address.c +=================================================================== +--- a/rose_address.c (.../tags/1.4.10.2) (revision 0) ++++ b/rose_address.c (.../branches/1.4) (revision 1357) +@@ -0,0 +1,983 @@ ++/* ++ * libpri: An implementation of Primary Rate ISDN ++ * ++ * Copyright (C) 2009 Digium, Inc. ++ * ++ * Richard Mudgett <rmudgett@digium.com> ++ * ++ * See http://www.asterisk.org for more information about ++ * the Asterisk project. Please do not directly contact ++ * any of the maintainers of this project for assistance; ++ * the project provides a web site, mailing lists and IRC ++ * channels for your use. ++ * ++ * This program is free software, distributed under the terms of ++ * the GNU General Public License Version 2 as published by the ++ * Free Software Foundation. See the LICENSE file included with ++ * this program for more details. ++ * ++ * In addition, when this program is distributed with Asterisk in ++ * any form that would qualify as a 'combined work' or as a ++ * 'derivative work' (but not mere aggregation), you can redistribute ++ * and/or modify the combination under the terms of the license ++ * provided with that copy of Asterisk, instead of the license ++ * terms granted here. ++ */ ++ ++/*! ++ * \file ++ * \brief ROSE Addressing-Data-Elements ++ * ++ * Addressing-Data-Elements ETS 300 196-1 D.3 ++ * ++ * \author Richard Mudgett <rmudgett@digium.com> ++ */ ++ ++ ++#include "compat.h" ++#include "libpri.h" ++#include "pri_internal.h" ++#include "rose.h" ++#include "rose_internal.h" ++#include "asn1.h" ++ ++ ++/* ------------------------------------------------------------------- */ ++ ++/*! ++ * \internal ++ * \brief Encode the public or private network PartyNumber type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param tag Component tag to identify the encoded component. ++ * The tag should be ASN1_TAG_SEQUENCE unless the caller implicitly ++ * tags it otherwise. ++ * \param number ++ * \param length_of_number ++ * \param type_of_number ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *rose_enc_NetworkPartyNumber(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, unsigned tag, const unsigned char *number, ++ size_t length_of_number, u_int8_t type_of_number) ++{ ++ unsigned char *seq_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, tag); ++ ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, type_of_number)); ++ ASN1_CALL(pos, asn1_enc_string_bin(pos, end, ASN1_TYPE_NUMERIC_STRING, number, ++ length_of_number)); ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the PartyNumber type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param party_number ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_PartyNumber(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const struct rosePartyNumber *party_number) ++{ ++ switch (party_number->plan) { ++ case 0: /* Unknown PartyNumber */ ++ ASN1_CALL(pos, asn1_enc_string_bin(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 0, ++ party_number->str, party_number->length)); ++ break; ++ case 1: /* Public PartyNumber */ ++ ASN1_CALL(pos, rose_enc_NetworkPartyNumber(ctrl, pos, end, ++ ASN1_CLASS_CONTEXT_SPECIFIC | 1, party_number->str, party_number->length, ++ party_number->ton)); ++ break; ++ case 2: /* NSAP encoded PartyNumber */ ++ ASN1_CALL(pos, asn1_enc_string_bin(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 2, ++ party_number->str, party_number->length)); ++ break; ++ case 3: /* Data PartyNumber (Not used) */ ++ ASN1_CALL(pos, asn1_enc_string_bin(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 3, ++ party_number->str, party_number->length)); ++ break; ++ case 4: /* Telex PartyNumber (Not used) */ ++ ASN1_CALL(pos, asn1_enc_string_bin(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 4, ++ party_number->str, party_number->length)); ++ break; ++ case 5: /* Private PartyNumber */ ++ ASN1_CALL(pos, rose_enc_NetworkPartyNumber(ctrl, pos, end, ++ ASN1_CLASS_CONTEXT_SPECIFIC | 5, party_number->str, party_number->length, ++ party_number->ton)); ++ break; ++ case 8: /* National Standard PartyNumber (Not used) */ ++ ASN1_CALL(pos, asn1_enc_string_bin(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 8, ++ party_number->str, party_number->length)); ++ break; ++ default: ++ ASN1_ENC_ERROR(ctrl, "Unknown numbering plan"); ++ return NULL; ++ } ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the PartySubaddress type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param party_subaddress ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_PartySubaddress(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const struct rosePartySubaddress *party_subaddress) ++{ ++ unsigned char *seq_len; ++ ++ switch (party_subaddress->type) { ++ case 0: /* UserSpecified */ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ASN1_TAG_SEQUENCE); ++ ++ ASN1_CALL(pos, asn1_enc_string_bin(pos, end, ASN1_TYPE_OCTET_STRING, ++ party_subaddress->u.user_specified.information, party_subaddress->length)); ++ if (party_subaddress->u.user_specified.odd_count_present) { ++ ASN1_CALL(pos, asn1_enc_boolean(pos, end, ASN1_TYPE_BOOLEAN, ++ party_subaddress->u.user_specified.odd_count)); ++ } ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ break; ++ case 1: /* NSAP */ ++ ASN1_CALL(pos, asn1_enc_string_bin(pos, end, ASN1_TYPE_OCTET_STRING, ++ party_subaddress->u.nsap, party_subaddress->length)); ++ break; ++ default: ++ ASN1_ENC_ERROR(ctrl, "Unknown subaddress type"); ++ return NULL; ++ } ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the Address type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param tag Component tag to identify the encoded component. ++ * The tag should be ASN1_TAG_SEQUENCE unless the caller implicitly ++ * tags it otherwise. ++ * \param address ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_Address(struct pri *ctrl, unsigned char *pos, unsigned char *end, ++ unsigned tag, const struct roseAddress *address) ++{ ++ unsigned char *seq_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, tag); ++ ++ ASN1_CALL(pos, rose_enc_PartyNumber(ctrl, pos, end, &address->number)); ++ if (address->subaddress.length) { ++ ASN1_CALL(pos, rose_enc_PartySubaddress(ctrl, pos, end, &address->subaddress)); ++ } ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the PresentedNumberUnscreened type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param party ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_PresentedNumberUnscreened(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const struct rosePresentedNumberUnscreened *party) ++{ ++ unsigned char *seq_len; ++ ++ switch (party->presentation) { ++ case 0: /* presentationAllowedNumber */ ++ /* EXPLICIT tag */ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 0); ++ ASN1_CALL(pos, rose_enc_PartyNumber(ctrl, pos, end, &party->number)); ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ break; ++ case 1: /* presentationRestricted */ ++ ASN1_CALL(pos, asn1_enc_null(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 1)); ++ break; ++ case 2: /* numberNotAvailableDueToInterworking */ ++ ASN1_CALL(pos, asn1_enc_null(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 2)); ++ break; ++ case 3: /* presentationRestrictedNumber */ ++ /* EXPLICIT tag */ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 3); ++ ASN1_CALL(pos, rose_enc_PartyNumber(ctrl, pos, end, &party->number)); ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ break; ++ default: ++ ASN1_ENC_ERROR(ctrl, "Unknown presentation type"); ++ return NULL; ++ } ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the NumberScreened type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param tag Component tag to identify the encoded component. ++ * The tag should be ASN1_TAG_SEQUENCE unless the caller implicitly ++ * tags it otherwise. ++ * \param screened ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_NumberScreened(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, unsigned tag, const struct roseNumberScreened *screened) ++{ ++ unsigned char *seq_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, tag); ++ ++ ASN1_CALL(pos, rose_enc_PartyNumber(ctrl, pos, end, &screened->number)); ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, ++ screened->screening_indicator)); ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the PresentedNumberScreened type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param party ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_PresentedNumberScreened(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const struct rosePresentedNumberScreened *party) ++{ ++ switch (party->presentation) { ++ case 0: /* presentationAllowedNumber */ ++ ASN1_CALL(pos, rose_enc_NumberScreened(ctrl, pos, end, ++ ASN1_CLASS_CONTEXT_SPECIFIC | 0, &party->screened)); ++ break; ++ case 1: /* presentationRestricted */ ++ ASN1_CALL(pos, asn1_enc_null(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 1)); ++ break; ++ case 2: /* numberNotAvailableDueToInterworking */ ++ ASN1_CALL(pos, asn1_enc_null(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 2)); ++ break; ++ case 3: /* presentationRestrictedNumber */ ++ ASN1_CALL(pos, rose_enc_NumberScreened(ctrl, pos, end, ++ ASN1_CLASS_CONTEXT_SPECIFIC | 3, &party->screened)); ++ break; ++ default: ++ ASN1_ENC_ERROR(ctrl, "Unknown presentation type"); ++ return NULL; ++ } ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the AddressScreened type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param tag Component tag to identify the encoded component. ++ * The tag should be ASN1_TAG_SEQUENCE unless the caller implicitly ++ * tags it otherwise. ++ * \param screened ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_AddressScreened(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, unsigned tag, const struct roseAddressScreened *screened) ++{ ++ unsigned char *seq_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, tag); ++ ++ ASN1_CALL(pos, rose_enc_PartyNumber(ctrl, pos, end, &screened->number)); ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, ++ screened->screening_indicator)); ++ if (screened->subaddress.length) { ++ ASN1_CALL(pos, rose_enc_PartySubaddress(ctrl, pos, end, &screened->subaddress)); ++ } ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the PresentedAddressScreened type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param party ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_PresentedAddressScreened(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const struct rosePresentedAddressScreened *party) ++{ ++ switch (party->presentation) { ++ case 0: /* presentationAllowedAddress */ ++ ASN1_CALL(pos, rose_enc_AddressScreened(ctrl, pos, end, ++ ASN1_CLASS_CONTEXT_SPECIFIC | 0, &party->screened)); ++ break; ++ case 1: /* presentationRestricted */ ++ ASN1_CALL(pos, asn1_enc_null(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 1)); ++ break; ++ case 2: /* numberNotAvailableDueToInterworking */ ++ ASN1_CALL(pos, asn1_enc_null(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 2)); ++ break; ++ case 3: /* presentationRestrictedAddress */ ++ ASN1_CALL(pos, rose_enc_AddressScreened(ctrl, pos, end, ++ ASN1_CLASS_CONTEXT_SPECIFIC | 3, &party->screened)); ++ break; ++ default: ++ ASN1_ENC_ERROR(ctrl, "Unknown presentation type"); ++ return NULL; ++ } ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Decode the NumberDigits PartyNumber argument parameters. ++ * ++ * \param ctrl D channel controller for any diagnostic messages. ++ * \param name Field name ++ * \param tag Component tag that identified this production. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param party_number Parameter storage to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++static const unsigned char *rose_dec_NumberDigits(struct pri *ctrl, const char *name, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct rosePartyNumber *party_number) ++{ ++ size_t str_len; ++ ++ ASN1_CALL(pos, asn1_dec_string_max(ctrl, name, tag, pos, end, ++ sizeof(party_number->str), party_number->str, &str_len)); ++ party_number->length = str_len; ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Decode the NSAP PartyNumber argument parameters. ++ * ++ * \param ctrl D channel controller for any diagnostic messages. ++ * \param name Field name ++ * \param tag Component tag that identified this production. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param party_number Parameter storage to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++static const unsigned char *rose_dec_NSAPPartyNumber(struct pri *ctrl, const char *name, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct rosePartyNumber *party_number) ++{ ++ size_t str_len; ++ ++ ASN1_CALL(pos, asn1_dec_string_bin(ctrl, name, tag, pos, end, ++ sizeof(party_number->str), party_number->str, &str_len)); ++ party_number->length = str_len; ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Decode the public or private network PartyNumber argument parameters. ++ * ++ * \param ctrl D channel controller for any diagnostic messages. ++ * \param name Field name ++ * \param tag Component tag that identified this production. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param party_number Parameter storage to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++static const unsigned char *rose_dec_NetworkPartyNumber(struct pri *ctrl, ++ const char *name, unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct rosePartyNumber *party_number) ++{ ++ int32_t value; ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s %s\n", name, asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_ENUMERATED); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "typeOfNumber", tag, pos, seq_end, &value)); ++ party_number->ton = value; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag & ~ASN1_PC_MASK, ASN1_TYPE_NUMERIC_STRING); ++ ASN1_CALL(pos, rose_dec_NumberDigits(ctrl, "numberDigits", tag, pos, seq_end, ++ party_number)); ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the PartyNumber argument parameters. ++ * ++ * \param ctrl D channel controller for any diagnostic messages. ++ * \param name Field name ++ * \param tag Component tag that identified this production. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param party_number Parameter storage to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_PartyNumber(struct pri *ctrl, const char *name, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct rosePartyNumber *party_number) ++{ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s PartyNumber\n", name); ++ } ++ party_number->ton = 0; /* unknown */ ++ switch (tag & ~ASN1_PC_MASK) { ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 0: ++ party_number->plan = 0; /* Unknown PartyNumber */ ++ ASN1_CALL(pos, rose_dec_NumberDigits(ctrl, "unknownPartyNumber", tag, pos, end, ++ party_number)); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 1: ++ /* Must be constructed but we will not check for it for simplicity. */ ++ party_number->plan = 1; /* Public PartyNumber */ ++ ASN1_CALL(pos, rose_dec_NetworkPartyNumber(ctrl, "publicPartyNumber", tag, pos, ++ end, party_number)); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 2: ++ party_number->plan = 2; /* NSAP encoded PartyNumber */ ++ ASN1_CALL(pos, rose_dec_NSAPPartyNumber(ctrl, "nsapEncodedPartyNumber", tag, pos, ++ end, party_number)); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 3: ++ party_number->plan = 3; /* Data PartyNumber (Not used) */ ++ ASN1_CALL(pos, rose_dec_NumberDigits(ctrl, "dataPartyNumber", tag, pos, end, ++ party_number)); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 4: ++ party_number->plan = 4; /* Telex PartyNumber (Not used) */ ++ ASN1_CALL(pos, rose_dec_NumberDigits(ctrl, "telexPartyNumber", tag, pos, end, ++ party_number)); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 5: ++ /* Must be constructed but we will not check for it for simplicity. */ ++ party_number->plan = 5; /* Private PartyNumber */ ++ ASN1_CALL(pos, rose_dec_NetworkPartyNumber(ctrl, "privatePartyNumber", tag, pos, ++ end, party_number)); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 8: ++ party_number->plan = 8; /* National Standard PartyNumber (Not used) */ ++ ASN1_CALL(pos, rose_dec_NumberDigits(ctrl, "nationalStandardPartyNumber", tag, ++ pos, end, party_number)); ++ break; ++ default: ++ ASN1_DID_NOT_EXPECT_TAG(ctrl, tag); ++ return NULL; ++ } ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Decode the User PartySubaddress argument parameters. ++ * ++ * \param ctrl D channel controller for any diagnostic messages. ++ * \param name Field name ++ * \param tag Component tag that identified this production. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param party_subaddress Parameter storage to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++static const unsigned char *rose_dec_UserSubaddress(struct pri *ctrl, const char *name, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct rosePartySubaddress *party_subaddress) ++{ ++ size_t str_len; ++ int32_t odd_count; ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ ++ party_subaddress->type = 0; /* UserSpecified */ ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s UserSpecified %s\n", name, asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ /* SubaddressInformation */ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag & ~ASN1_PC_MASK, ASN1_TYPE_OCTET_STRING); ++ ASN1_CALL(pos, asn1_dec_string_bin(ctrl, "subaddressInformation", tag, pos, seq_end, ++ sizeof(party_subaddress->u.user_specified.information), ++ party_subaddress->u.user_specified.information, &str_len)); ++ party_subaddress->length = str_len; ++ ++ if (pos < seq_end && *pos != ASN1_INDEF_TERM) { ++ /* ++ * The optional odd count indicator must be present since there ++ * is something left. ++ */ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_BOOLEAN); ++ ASN1_CALL(pos, asn1_dec_boolean(ctrl, "oddCount", tag, pos, seq_end, ++ &odd_count)); ++ party_subaddress->u.user_specified.odd_count = odd_count; ++ party_subaddress->u.user_specified.odd_count_present = 1; ++ } else { ++ party_subaddress->u.user_specified.odd_count = 0; ++ party_subaddress->u.user_specified.odd_count_present = 0; ++ } ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Decode the NSAP PartySubaddress argument parameters. ++ * ++ * \param ctrl D channel controller for any diagnostic messages. ++ * \param name Field name ++ * \param tag Component tag that identified this production. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param party_subaddress Parameter storage to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++static const unsigned char *rose_dec_NSAPSubaddress(struct pri *ctrl, const char *name, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct rosePartySubaddress *party_subaddress) ++{ ++ size_t str_len; ++ ++ party_subaddress->type = 1; /* NSAP */ ++ ++ ASN1_CALL(pos, asn1_dec_string_bin(ctrl, name, tag, pos, end, ++ sizeof(party_subaddress->u.nsap), party_subaddress->u.nsap, &str_len)); ++ party_subaddress->length = str_len; ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the PartySubaddress argument parameters. ++ * ++ * \param ctrl D channel controller for any diagnostic messages. ++ * \param name Field name ++ * \param tag Component tag that identified this production. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param party_subaddress Parameter storage to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_PartySubaddress(struct pri *ctrl, const char *name, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct rosePartySubaddress *party_subaddress) ++{ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s PartySubaddress\n", name); ++ } ++ switch (tag) { ++ case ASN1_TAG_SEQUENCE: ++ ASN1_CALL(pos, rose_dec_UserSubaddress(ctrl, "user", tag, pos, end, ++ party_subaddress)); ++ break; ++ case ASN1_TYPE_OCTET_STRING: ++ case ASN1_TYPE_OCTET_STRING | ASN1_PC_CONSTRUCTED: ++ ASN1_CALL(pos, rose_dec_NSAPSubaddress(ctrl, "nsap", tag, pos, end, ++ party_subaddress)); ++ break; ++ default: ++ ASN1_DID_NOT_EXPECT_TAG(ctrl, tag); ++ return NULL; ++ } ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the Address argument parameters. ++ * ++ * \param ctrl D channel controller for any diagnostic messages. ++ * \param name Field name ++ * \param tag Component tag that identified this production. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param address Parameter storage to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_Address(struct pri *ctrl, const char *name, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, struct roseAddress *address) ++{ ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s Address %s\n", name, asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CALL(pos, rose_dec_PartyNumber(ctrl, "partyNumber", tag, pos, seq_end, ++ &address->number)); ++ ++ if (pos < seq_end && *pos != ASN1_INDEF_TERM) { ++ /* The optional subaddress must be present since there is something left. */ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CALL(pos, rose_dec_PartySubaddress(ctrl, "partySubaddress", tag, pos, ++ seq_end, &address->subaddress)); ++ } else { ++ address->subaddress.length = 0; /* Subaddress not present */ ++ } ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the PresentedNumberUnscreened argument parameters. ++ * ++ * \param ctrl D channel controller for any diagnostic messages. ++ * \param name Field name ++ * \param tag Component tag that identified this production. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param party Parameter storage to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_PresentedNumberUnscreened(struct pri *ctrl, ++ const char *name, unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct rosePresentedNumberUnscreened *party) ++{ ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s PresentedNumberUnscreened\n", name); ++ } ++ switch (tag) { ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 0: ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " Explicit %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ party->presentation = 0; /* presentationAllowedNumber */ ++ ASN1_CALL(pos, rose_dec_PartyNumber(ctrl, "presentationAllowedNumber", tag, pos, ++ seq_end, &party->number)); ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 1: ++ party->presentation = 1; /* presentationRestricted */ ++ ASN1_CALL(pos, asn1_dec_null(ctrl, "presentationRestricted", tag, pos, end)); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 2: ++ party->presentation = 2; /* numberNotAvailableDueToInterworking */ ++ ASN1_CALL(pos, asn1_dec_null(ctrl, "numberNotAvailableDueToInterworking", tag, ++ pos, end)); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 3: ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " Explicit %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ party->presentation = 3; /* presentationRestrictedNumber */ ++ ASN1_CALL(pos, rose_dec_PartyNumber(ctrl, "presentationRestrictedNumber", tag, ++ pos, seq_end, &party->number)); ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ break; ++ default: ++ ASN1_DID_NOT_EXPECT_TAG(ctrl, tag); ++ return NULL; ++ } ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the NumberScreened argument parameters. ++ * ++ * \param ctrl D channel controller for any diagnostic messages. ++ * \param name Field name ++ * \param tag Component tag that identified this production. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param screened Parameter storage to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_NumberScreened(struct pri *ctrl, const char *name, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct roseNumberScreened *screened) ++{ ++ int32_t value; ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s NumberScreened %s\n", name, asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CALL(pos, rose_dec_PartyNumber(ctrl, "partyNumber", tag, pos, seq_end, ++ &screened->number)); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_ENUMERATED); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "screeningIndicator", tag, pos, seq_end, &value)); ++ screened->screening_indicator = value; ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the PresentedNumberScreened argument parameters. ++ * ++ * \param ctrl D channel controller for any diagnostic messages. ++ * \param name Field name ++ * \param tag Component tag that identified this production. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param party Parameter storage to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_PresentedNumberScreened(struct pri *ctrl, const char *name, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct rosePresentedNumberScreened *party) ++{ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s PresentedNumberScreened\n", name); ++ } ++ switch (tag) { ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 0: ++ party->presentation = 0; /* presentationAllowedNumber */ ++ ASN1_CALL(pos, rose_dec_NumberScreened(ctrl, "presentationAllowedNumber", tag, ++ pos, end, &party->screened)); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 1: ++ party->presentation = 1; /* presentationRestricted */ ++ ASN1_CALL(pos, asn1_dec_null(ctrl, "presentationRestricted", tag, pos, end)); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 2: ++ party->presentation = 2; /* numberNotAvailableDueToInterworking */ ++ ASN1_CALL(pos, asn1_dec_null(ctrl, "numberNotAvailableDueToInterworking", tag, ++ pos, end)); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 3: ++ party->presentation = 3; /* presentationRestrictedNumber */ ++ ASN1_CALL(pos, rose_dec_NumberScreened(ctrl, "presentationRestrictedNumber", tag, ++ pos, end, &party->screened)); ++ break; ++ default: ++ ASN1_DID_NOT_EXPECT_TAG(ctrl, tag); ++ return NULL; ++ } ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the AddressScreened argument parameters. ++ * ++ * \param ctrl D channel controller for any diagnostic messages. ++ * \param name Field name ++ * \param tag Component tag that identified this production. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param screened Parameter storage to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_AddressScreened(struct pri *ctrl, const char *name, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct roseAddressScreened *screened) ++{ ++ int32_t value; ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s AddressScreened %s\n", name, asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CALL(pos, rose_dec_PartyNumber(ctrl, "partyNumber", tag, pos, seq_end, ++ &screened->number)); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_ENUMERATED); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "screeningIndicator", tag, pos, seq_end, &value)); ++ screened->screening_indicator = value; ++ ++ if (pos < seq_end && *pos != ASN1_INDEF_TERM) { ++ /* The optional subaddress must be present since there is something left. */ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CALL(pos, rose_dec_PartySubaddress(ctrl, "partySubaddress", tag, pos, ++ seq_end, &screened->subaddress)); ++ } else { ++ screened->subaddress.length = 0; /* Subaddress not present */ ++ } ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the PresentedAddressScreened argument parameters. ++ * ++ * \param ctrl D channel controller for any diagnostic messages. ++ * \param name Field name ++ * \param tag Component tag that identified this production. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param party Parameter storage to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_PresentedAddressScreened(struct pri *ctrl, ++ const char *name, unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct rosePresentedAddressScreened *party) ++{ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s PresentedAddressScreened\n", name); ++ } ++ switch (tag) { ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 0: ++ party->presentation = 0; /* presentationAllowedAddress */ ++ ASN1_CALL(pos, rose_dec_AddressScreened(ctrl, "presentationAllowedAddress", tag, ++ pos, end, &party->screened)); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 1: ++ party->presentation = 1; /* presentationRestricted */ ++ ASN1_CALL(pos, asn1_dec_null(ctrl, "presentationRestricted", tag, pos, end)); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 2: ++ party->presentation = 2; /* numberNotAvailableDueToInterworking */ ++ ASN1_CALL(pos, asn1_dec_null(ctrl, "numberNotAvailableDueToInterworking", tag, ++ pos, end)); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 3: ++ party->presentation = 3; /* presentationRestrictedAddress */ ++ ASN1_CALL(pos, rose_dec_AddressScreened(ctrl, "presentationRestrictedAddress", ++ tag, pos, end, &party->screened)); ++ break; ++ default: ++ ASN1_DID_NOT_EXPECT_TAG(ctrl, tag); ++ return NULL; ++ } ++ ++ return pos; ++} ++ ++/* ------------------------------------------------------------------- */ ++/* end rose_address.c */ + +Property changes on: rose_address.c +___________________________________________________________________ +Added: svn:eol-style + + native +Added: svn:mime-type + + text/plain +Added: svn:keywords + + 'Author Date Id Revision' + +Index: rose_qsig_aoc.c +=================================================================== +--- a/rose_qsig_aoc.c (.../tags/1.4.10.2) (revision 0) ++++ b/rose_qsig_aoc.c (.../branches/1.4) (revision 1357) +@@ -0,0 +1,1714 @@ ++/* ++ * libpri: An implementation of Primary Rate ISDN ++ * ++ * Copyright (C) 2009 Digium, Inc. ++ * ++ * Richard Mudgett <rmudgett@digium.com> ++ * ++ * See http://www.asterisk.org for more information about ++ * the Asterisk project. Please do not directly contact ++ * any of the maintainers of this project for assistance; ++ * the project provides a web site, mailing lists and IRC ++ * channels for your use. ++ * ++ * This program is free software, distributed under the terms of ++ * the GNU General Public License Version 2 as published by the ++ * Free Software Foundation. See the LICENSE file included with ++ * this program for more details. ++ * ++ * In addition, when this program is distributed with Asterisk in ++ * any form that would qualify as a 'combined work' or as a ++ * 'derivative work' (but not mere aggregation), you can redistribute ++ * and/or modify the combination under the terms of the license ++ * provided with that copy of Asterisk, instead of the license ++ * terms granted here. ++ */ ++ ++/*! ++ * \file ++ * \brief Q.SIG ROSE Advice-Of-Charge (AOC) operations ++ * ++ * SS-AOC-Operations ECMA-212 Annex E Table E.1 ++ * ++ * \author Richard Mudgett <rmudgett@digium.com> ++ */ ++ ++ ++#include "compat.h" ++#include "libpri.h" ++#include "pri_internal.h" ++#include "rose.h" ++#include "rose_internal.h" ++#include "asn1.h" ++ ++ ++/* ------------------------------------------------------------------- */ ++ ++/*! ++ * \internal ++ * \brief Encode the Time type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param tag Component tag to identify the encoded component. ++ * The tag should be ASN1_TAG_SEQUENCE unless the caller implicitly ++ * tags it otherwise. ++ * \param time Time information to encode. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *rose_enc_qsig_AOC_Time(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, unsigned tag, const struct roseQsigAOCTime *time) ++{ ++ unsigned char *seq_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, tag); ++ ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 1, ++ time->length)); ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 2, time->scale)); ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Encode the Amount type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param tag Component tag to identify the encoded component. ++ * The tag should be ASN1_TAG_SEQUENCE unless the caller implicitly ++ * tags it otherwise. ++ * \param amount Amount information to encode. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *rose_enc_qsig_AOC_Amount(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, unsigned tag, const struct roseQsigAOCAmount *amount) ++{ ++ unsigned char *seq_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, tag); ++ ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 1, ++ amount->currency)); ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 2, ++ amount->multiplier)); ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Encode the RecordedCurrency type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param tag Component tag to identify the encoded component. ++ * The tag should be ASN1_TAG_SEQUENCE unless the caller implicitly ++ * tags it otherwise. ++ * \param recorded Recorded currency information to encode. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *rose_enc_qsig_AOC_RecordedCurrency(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, unsigned tag, ++ const struct roseQsigAOCRecordedCurrency *recorded) ++{ ++ unsigned char *seq_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, tag); ++ ++ ASN1_CALL(pos, asn1_enc_string_max(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 1, ++ recorded->currency, sizeof(recorded->currency) - 1)); ++ ASN1_CALL(pos, rose_enc_qsig_AOC_Amount(ctrl, pos, end, ++ ASN1_CLASS_CONTEXT_SPECIFIC | 2, &recorded->amount)); ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Encode the DurationCurrency type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param tag Component tag to identify the encoded component. ++ * The tag should be ASN1_TAG_SEQUENCE unless the caller implicitly ++ * tags it otherwise. ++ * \param duration Duration currency information to encode. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *rose_enc_qsig_AOC_DurationCurrency(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, unsigned tag, ++ const struct roseQsigAOCDurationCurrency *duration) ++{ ++ unsigned char *seq_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, tag); ++ ++ ASN1_CALL(pos, asn1_enc_string_max(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 1, ++ duration->currency, sizeof(duration->currency) - 1)); ++ ASN1_CALL(pos, rose_enc_qsig_AOC_Amount(ctrl, pos, end, ++ ASN1_CLASS_CONTEXT_SPECIFIC | 2, &duration->amount)); ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 3, ++ duration->charging_type)); ++ ASN1_CALL(pos, rose_enc_qsig_AOC_Time(ctrl, pos, end, ++ ASN1_CLASS_CONTEXT_SPECIFIC | 4, &duration->time)); ++ if (duration->granularity_present) { ++ ASN1_CALL(pos, rose_enc_qsig_AOC_Time(ctrl, pos, end, ++ ASN1_CLASS_CONTEXT_SPECIFIC | 5, &duration->granularity)); ++ } ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Encode the FlatRateCurrency type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param tag Component tag to identify the encoded component. ++ * The tag should be ASN1_TAG_SEQUENCE unless the caller implicitly ++ * tags it otherwise. ++ * \param flat_rate Flat rate currency information to encode. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *rose_enc_qsig_AOC_FlatRateCurrency(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, unsigned tag, ++ const struct roseQsigAOCFlatRateCurrency *flat_rate) ++{ ++ unsigned char *seq_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, tag); ++ ++ ASN1_CALL(pos, asn1_enc_string_max(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 1, ++ flat_rate->currency, sizeof(flat_rate->currency) - 1)); ++ ASN1_CALL(pos, rose_enc_qsig_AOC_Amount(ctrl, pos, end, ++ ASN1_CLASS_CONTEXT_SPECIFIC | 2, &flat_rate->amount)); ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Encode the VolumeRateCurrency type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param tag Component tag to identify the encoded component. ++ * The tag should be ASN1_TAG_SEQUENCE unless the caller implicitly ++ * tags it otherwise. ++ * \param volume_rate Volume rate currency information to encode. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *rose_enc_qsig_AOC_VolumeRateCurrency(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, unsigned tag, ++ const struct roseQsigAOCVolumeRateCurrency *volume_rate) ++{ ++ unsigned char *seq_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, tag); ++ ++ ASN1_CALL(pos, asn1_enc_string_max(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 1, ++ volume_rate->currency, sizeof(volume_rate->currency) - 1)); ++ ASN1_CALL(pos, rose_enc_qsig_AOC_Amount(ctrl, pos, end, ++ ASN1_CLASS_CONTEXT_SPECIFIC | 2, &volume_rate->amount)); ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 3, ++ volume_rate->unit)); ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Encode the AOCSCurrencyInfo type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param tag Component tag to identify the encoded component. ++ * The tag should be ASN1_TAG_SEQUENCE unless the caller implicitly ++ * tags it otherwise. ++ * \param currency_info Currency information record to encode. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *rose_enc_qsig_AOCSCurrencyInfo(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, unsigned tag, ++ const struct roseQsigAOCSCurrencyInfo *currency_info) ++{ ++ unsigned char *seq_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, tag); ++ ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, ++ currency_info->charged_item)); ++ ++ switch (currency_info->currency_type) { ++ case 0: /* specialChargingCode */ ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_INTEGER, ++ currency_info->u.special_charging_code)); ++ break; ++ case 1: /* durationCurrency */ ++ ASN1_CALL(pos, rose_enc_qsig_AOC_DurationCurrency(ctrl, pos, end, ++ ASN1_CLASS_CONTEXT_SPECIFIC | 1, ¤cy_info->u.duration)); ++ break; ++ case 2: /* flatRateCurrency */ ++ ASN1_CALL(pos, rose_enc_qsig_AOC_FlatRateCurrency(ctrl, pos, end, ++ ASN1_CLASS_CONTEXT_SPECIFIC | 2, ¤cy_info->u.flat_rate)); ++ break; ++ case 3: /* volumeRateCurrency */ ++ ASN1_CALL(pos, rose_enc_qsig_AOC_VolumeRateCurrency(ctrl, pos, end, ++ ASN1_CLASS_CONTEXT_SPECIFIC | 3, ¤cy_info->u.volume_rate)); ++ break; ++ case 4: /* freeOfCharge */ ++ ASN1_CALL(pos, asn1_enc_null(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 4)); ++ break; ++ case 5: /* currencyInfoNotAvailable */ ++ ASN1_CALL(pos, asn1_enc_null(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 5)); ++ break; ++ case 6: /* freeOfChargeFromBeginning */ ++ ASN1_CALL(pos, asn1_enc_null(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 6)); ++ break; ++ default: ++ ASN1_ENC_ERROR(ctrl, "Unknown currency type"); ++ return NULL; ++ } ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Encode the AOCSCurrencyInfoList type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param tag Component tag to identify the encoded component. ++ * The tag should be ASN1_TAG_SEQUENCE unless the caller implicitly ++ * tags it otherwise. ++ * \param currency_info Currency information list to encode. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *rose_enc_qsig_AOCSCurrencyInfoList(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, unsigned tag, ++ const struct roseQsigAOCSCurrencyInfoList *currency_info) ++{ ++ unsigned index; ++ unsigned char *seq_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, tag); ++ ++ for (index = 0; index < currency_info->num_records; ++index) { ++ ASN1_CALL(pos, rose_enc_qsig_AOCSCurrencyInfo(ctrl, pos, end, ASN1_TAG_SEQUENCE, ++ ¤cy_info->list[index])); ++ } ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Encode the ChargingAssociation type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param charging Charging association information to encode. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *rose_enc_qsig_AOC_ChargingAssociation(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, ++ const struct roseQsigAOCChargingAssociation *charging) ++{ ++ unsigned char *explicit_len; ++ ++ switch (charging->type) { ++ case 0: /* charge_identifier */ ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_INTEGER, charging->id)); ++ break; ++ case 1: /* charged_number */ ++ /* EXPLICIT tag */ ++ ASN1_CONSTRUCTED_BEGIN(explicit_len, pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 0); ++ ASN1_CALL(pos, rose_enc_PartyNumber(ctrl, pos, end, &charging->number)); ++ ASN1_CONSTRUCTED_END(explicit_len, pos, end); ++ break; ++ default: ++ ASN1_ENC_ERROR(ctrl, "Unknown ChargingAssociation type"); ++ return NULL; ++ } ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the Q.SIG ChargeRequest invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_qsig_ChargeRequest_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ unsigned index; ++ unsigned char *seq_len; ++ unsigned char *advice_len; ++ const struct roseQsigChargeRequestArg_ARG *charge_request; ++ ++ charge_request = &args->qsig.ChargeRequest; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ASN1_TAG_SEQUENCE); ++ ++ /* SEQUENCE SIZE(0..7) OF AdviceModeCombination */ ++ ASN1_CONSTRUCTED_BEGIN(advice_len, pos, end, ASN1_TAG_SEQUENCE); ++ for (index = 0; index < charge_request->num_records; ++index) { ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, ++ charge_request->advice_mode_combinations[index])); ++ } ++ ASN1_CONSTRUCTED_END(advice_len, pos, end); ++ ++ /* No extension to encode */ ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the Q.SIG ChargeRequest result facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_qsig_ChargeRequest_RES(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_result_args *args) ++{ ++ unsigned char *seq_len; ++ const struct roseQsigChargeRequestRes_RES *charge_request; ++ ++ charge_request = &args->qsig.ChargeRequest; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ASN1_TAG_SEQUENCE); ++ ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, ++ charge_request->advice_mode_combination)); ++ ++ /* No extension to encode */ ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the Q.SIG AocFinal invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_qsig_AocFinal_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ unsigned char *seq_len; ++ unsigned char *specific_len; ++ const struct roseQsigAocFinalArg_ARG *aoc_final; ++ ++ aoc_final = &args->qsig.AocFinal; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ASN1_TAG_SEQUENCE); ++ ++ switch (aoc_final->type) { ++ case 0: /* charge_not_available */ ++ ASN1_CALL(pos, asn1_enc_null(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 0)); ++ break; ++ case 1: /* free_of_charge */ ++ ASN1_CALL(pos, asn1_enc_null(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 1)); ++ break; ++ case 2: /* specific_currency */ ++ ASN1_CONSTRUCTED_BEGIN(specific_len, pos, end, ASN1_TAG_SEQUENCE); ++ ++ ASN1_CALL(pos, rose_enc_qsig_AOC_RecordedCurrency(ctrl, pos, end, ++ ASN1_CLASS_CONTEXT_SPECIFIC | 1, &aoc_final->specific.recorded)); ++ ++ if (aoc_final->specific.billing_id_present) { ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 2, ++ aoc_final->specific.billing_id)); ++ } ++ ++ ASN1_CONSTRUCTED_END(specific_len, pos, end); ++ break; ++ default: ++ ASN1_ENC_ERROR(ctrl, "Unknown AocFinal type"); ++ return NULL; ++ } ++ ++ if (aoc_final->charging_association_present) { ++ ASN1_CALL(pos, rose_enc_qsig_AOC_ChargingAssociation(ctrl, pos, end, ++ &aoc_final->charging_association)); ++ } ++ ++ /* No extension to encode */ ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the Q.SIG AocInterim invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_qsig_AocInterim_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ unsigned char *seq_len; ++ unsigned char *specific_len; ++ const struct roseQsigAocInterimArg_ARG *aoc_interim; ++ ++ aoc_interim = &args->qsig.AocInterim; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ASN1_TAG_SEQUENCE); ++ ++ switch (aoc_interim->type) { ++ case 0: /* charge_not_available */ ++ ASN1_CALL(pos, asn1_enc_null(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 0)); ++ break; ++ case 1: /* free_of_charge */ ++ ASN1_CALL(pos, asn1_enc_null(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 1)); ++ break; ++ case 2: /* specific_currency */ ++ ASN1_CONSTRUCTED_BEGIN(specific_len, pos, end, ASN1_TAG_SEQUENCE); ++ ++ ASN1_CALL(pos, rose_enc_qsig_AOC_RecordedCurrency(ctrl, pos, end, ++ ASN1_CLASS_CONTEXT_SPECIFIC | 1, &aoc_interim->specific.recorded)); ++ ++ if (aoc_interim->specific.billing_id_present) { ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 2, ++ aoc_interim->specific.billing_id)); ++ } ++ ++ ASN1_CONSTRUCTED_END(specific_len, pos, end); ++ break; ++ default: ++ ASN1_ENC_ERROR(ctrl, "Unknown AocInterim type"); ++ return NULL; ++ } ++ ++ /* No extension to encode */ ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the Q.SIG AocRate invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_qsig_AocRate_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ unsigned char *seq_len; ++ const struct roseQsigAocRateArg_ARG *aoc_rate; ++ ++ aoc_rate = &args->qsig.AocRate; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ASN1_TAG_SEQUENCE); ++ ++ switch (aoc_rate->type) { ++ case 0: /* charge_not_available */ ++ ASN1_CALL(pos, asn1_enc_null(pos, end, ASN1_TYPE_NULL)); ++ break; ++ case 1: /* currency_info_list */ ++ ASN1_CALL(pos, rose_enc_qsig_AOCSCurrencyInfoList(ctrl, pos, end, ++ ASN1_TAG_SEQUENCE, &aoc_rate->currency_info)); ++ break; ++ default: ++ ASN1_ENC_ERROR(ctrl, "Unknown AocRate type"); ++ return NULL; ++ } ++ ++ /* No extension to encode */ ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the Q.SIG AocComplete invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_qsig_AocComplete_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ unsigned char *seq_len; ++ const struct roseQsigAocCompleteArg_ARG *aoc_complete; ++ ++ aoc_complete = &args->qsig.AocComplete; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ASN1_TAG_SEQUENCE); ++ ++ ASN1_CALL(pos, rose_enc_PartyNumber(ctrl, pos, end, ++ &aoc_complete->charged_user_number)); ++ ++ if (aoc_complete->charging_association_present) { ++ ASN1_CALL(pos, rose_enc_qsig_AOC_ChargingAssociation(ctrl, pos, end, ++ &aoc_complete->charging_association)); ++ } ++ ++ /* No extension to encode */ ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the Q.SIG AocComplete result facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_qsig_AocComplete_RES(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_result_args *args) ++{ ++ unsigned char *seq_len; ++ const struct roseQsigAocCompleteRes_RES *aoc_complete; ++ ++ aoc_complete = &args->qsig.AocComplete; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ASN1_TAG_SEQUENCE); ++ ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, ++ aoc_complete->charging_option)); ++ ++ /* No extension to encode */ ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the Q.SIG AocDivChargeReq invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_qsig_AocDivChargeReq_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ unsigned char *seq_len; ++ const struct roseQsigAocDivChargeReqArg_ARG *aoc_div_charge_req; ++ ++ aoc_div_charge_req = &args->qsig.AocDivChargeReq; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ASN1_TAG_SEQUENCE); ++ ++ ASN1_CALL(pos, rose_enc_PartyNumber(ctrl, pos, end, ++ &aoc_div_charge_req->diverting_user_number)); ++ ++ if (aoc_div_charge_req->charging_association_present) { ++ ASN1_CALL(pos, rose_enc_qsig_AOC_ChargingAssociation(ctrl, pos, end, ++ &aoc_div_charge_req->charging_association)); ++ } ++ ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, ++ aoc_div_charge_req->diversion_type)); ++ ++ /* No extension to encode */ ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Decode the Time type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param name Field name ++ * \param tag Component tag that identified this production. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param time Parameter storage to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++static const unsigned char *rose_dec_qsig_AOC_Time(struct pri *ctrl, const char *name, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct roseQsigAOCTime *time) ++{ ++ int32_t value; ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s Time %s\n", name, asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_CLASS_CONTEXT_SPECIFIC | 1); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "lengthOfTimeUnit", tag, pos, seq_end, &value)); ++ time->length = value; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_CLASS_CONTEXT_SPECIFIC | 2); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "scale", tag, pos, seq_end, &value)); ++ time->scale = value; ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Decode the Amount type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param name Field name ++ * \param tag Component tag that identified this production. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param amount Parameter storage to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++static const unsigned char *rose_dec_qsig_AOC_Amount(struct pri *ctrl, const char *name, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct roseQsigAOCAmount *amount) ++{ ++ int32_t value; ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s Amount %s\n", name, asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_CLASS_CONTEXT_SPECIFIC | 1); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "currencyAmount", tag, pos, seq_end, &value)); ++ amount->currency = value; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_CLASS_CONTEXT_SPECIFIC | 2); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "multiplier", tag, pos, seq_end, &value)); ++ amount->multiplier = value; ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Decode the RecordedCurrency type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param name Field name ++ * \param tag Component tag that identified this production. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param recorded Parameter storage to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++static const unsigned char *rose_dec_qsig_AOC_RecordedCurrency(struct pri *ctrl, ++ const char *name, unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct roseQsigAOCRecordedCurrency *recorded) ++{ ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ size_t str_len; ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s RecordedCurrency %s\n", name, asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag & ~ASN1_PC_MASK, tag, ASN1_CLASS_CONTEXT_SPECIFIC | 1); ++ ASN1_CALL(pos, asn1_dec_string_max(ctrl, "rCurrency", tag, pos, seq_end, ++ sizeof(recorded->currency), recorded->currency, &str_len)); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ++ ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 2); ++ ASN1_CALL(pos, rose_dec_qsig_AOC_Amount(ctrl, "rAmount", tag, pos, seq_end, ++ &recorded->amount)); ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Decode the DurationCurrency type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param name Field name ++ * \param tag Component tag that identified this production. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param duration Parameter storage to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++static const unsigned char *rose_dec_qsig_AOC_DurationCurrency(struct pri *ctrl, ++ const char *name, unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct roseQsigAOCDurationCurrency *duration) ++{ ++ int32_t value; ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ size_t str_len; ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s DurationCurrency %s\n", name, asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag & ~ASN1_PC_MASK, tag, ASN1_CLASS_CONTEXT_SPECIFIC | 1); ++ ASN1_CALL(pos, asn1_dec_string_max(ctrl, "dCurrency", tag, pos, seq_end, ++ sizeof(duration->currency), duration->currency, &str_len)); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ++ ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 2); ++ ASN1_CALL(pos, rose_dec_qsig_AOC_Amount(ctrl, "dAmount", tag, pos, seq_end, ++ &duration->amount)); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_CLASS_CONTEXT_SPECIFIC | 3); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "dChargingType", tag, pos, seq_end, &value)); ++ duration->charging_type = value; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ++ ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 4); ++ ASN1_CALL(pos, rose_dec_qsig_AOC_Time(ctrl, "dTime", tag, pos, seq_end, ++ &duration->time)); ++ ++ if (pos < seq_end && *pos != ASN1_INDEF_TERM) { ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ++ ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 5); ++ ASN1_CALL(pos, rose_dec_qsig_AOC_Time(ctrl, "dGranularity", tag, pos, seq_end, ++ &duration->granularity)); ++ duration->granularity_present = 1; ++ } else { ++ duration->granularity_present = 0; ++ } ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Decode the FlatRateCurrency type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param name Field name ++ * \param tag Component tag that identified this production. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param flat_rate Parameter storage to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++static const unsigned char *rose_dec_qsig_AOC_FlatRateCurrency(struct pri *ctrl, ++ const char *name, unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct roseQsigAOCFlatRateCurrency *flat_rate) ++{ ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ size_t str_len; ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s FlatRateCurrency %s\n", name, asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag & ~ASN1_PC_MASK, tag, ASN1_CLASS_CONTEXT_SPECIFIC | 1); ++ ASN1_CALL(pos, asn1_dec_string_max(ctrl, "fRCurrency", tag, pos, seq_end, ++ sizeof(flat_rate->currency), flat_rate->currency, &str_len)); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ++ ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 2); ++ ASN1_CALL(pos, rose_dec_qsig_AOC_Amount(ctrl, "fRAmount", tag, pos, seq_end, ++ &flat_rate->amount)); ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Decode the VolumeRateCurrency type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param name Field name ++ * \param tag Component tag that identified this production. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param volume_rate Parameter storage to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++static const unsigned char *rose_dec_qsig_AOC_VolumeRateCurrency(struct pri *ctrl, ++ const char *name, unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct roseQsigAOCVolumeRateCurrency *volume_rate) ++{ ++ int32_t value; ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ size_t str_len; ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s VolumeRateCurrency %s\n", name, asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag & ~ASN1_PC_MASK, tag, ASN1_CLASS_CONTEXT_SPECIFIC | 1); ++ ASN1_CALL(pos, asn1_dec_string_max(ctrl, "vRCurrency", tag, pos, seq_end, ++ sizeof(volume_rate->currency), volume_rate->currency, &str_len)); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ++ ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 2); ++ ASN1_CALL(pos, rose_dec_qsig_AOC_Amount(ctrl, "vRAmount", tag, pos, seq_end, ++ &volume_rate->amount)); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_CLASS_CONTEXT_SPECIFIC | 3); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "vRVolumeUnit", tag, pos, seq_end, &value)); ++ volume_rate->unit = value; ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Decode the AOCSCurrencyInfo type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param name Field name ++ * \param tag Component tag that identified this production. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param currency_info Parameter storage to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++static const unsigned char *rose_dec_qsig_AOCSCurrencyInfo(struct pri *ctrl, ++ const char *name, unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct roseQsigAOCSCurrencyInfo *currency_info) ++{ ++ int32_t value; ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s AOCSCurrencyInfo %s\n", name, asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_ENUMERATED); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "chargedItem", tag, pos, seq_end, &value)); ++ currency_info->charged_item = value; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ switch (tag) { ++ case ASN1_TYPE_INTEGER: ++ currency_info->currency_type = 0; /* specialChargingCode */ ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "specialChargingCode", tag, pos, seq_end, ++ &value)); ++ currency_info->u.special_charging_code = value; ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 1: ++ currency_info->currency_type = 1; /* durationCurrency */ ++ ASN1_CALL(pos, rose_dec_qsig_AOC_DurationCurrency(ctrl, "durationCurrency", tag, ++ pos, seq_end, ¤cy_info->u.duration)); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 2: ++ currency_info->currency_type = 2; /* flatRateCurrency */ ++ ASN1_CALL(pos, rose_dec_qsig_AOC_FlatRateCurrency(ctrl, "flatRateCurrency", tag, ++ pos, seq_end, ¤cy_info->u.flat_rate)); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 3: ++ currency_info->currency_type = 3; /* volumeRateCurrency */ ++ ASN1_CALL(pos, rose_dec_qsig_AOC_VolumeRateCurrency(ctrl, "volumeRateCurrency", ++ tag, pos, seq_end, ¤cy_info->u.volume_rate)); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 4: ++ currency_info->currency_type = 4; /* freeOfCharge */ ++ ASN1_CALL(pos, asn1_dec_null(ctrl, "freeOfCharge", tag, pos, seq_end)); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 5: ++ currency_info->currency_type = 5; /* currencyInfoNotAvailable */ ++ ASN1_CALL(pos, asn1_dec_null(ctrl, "currencyInfoNotAvailable", tag, pos, ++ seq_end)); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 6: ++ currency_info->currency_type = 6; /* freeOfChargeFromBeginning */ ++ ASN1_CALL(pos, asn1_dec_null(ctrl, "freeOfChargeFromBeginning", tag, pos, ++ seq_end)); ++ break; ++ default: ++ ASN1_DID_NOT_EXPECT_TAG(ctrl, tag); ++ return NULL; ++ } ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Decode the AOCSCurrencyInfoList type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param name Field name ++ * \param tag Component tag that identified this production. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param currency_info Parameter storage to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++static const unsigned char *rose_dec_qsig_AOCSCurrencyInfoList(struct pri *ctrl, ++ const char *name, unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct roseQsigAOCSCurrencyInfoList *currency_info) ++{ ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s AOCSCurrencyInfoList %s\n", name, asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ currency_info->num_records = 0; ++ while (pos < seq_end && *pos != ASN1_INDEF_TERM) { ++ if (currency_info->num_records < ARRAY_LEN(currency_info->list)) { ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ ASN1_CALL(pos, rose_dec_qsig_AOCSCurrencyInfo(ctrl, "listEntry", tag, pos, ++ seq_end, ¤cy_info->list[currency_info->num_records])); ++ ++currency_info->num_records; ++ } else { ++ /* Too many records */ ++ return NULL; ++ } ++ } ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Decode the ChargingAssociation type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param name Field name ++ * \param tag Component tag that identified this production. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param charging Parameter storage to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++static const unsigned char *rose_dec_qsig_AOC_ChargingAssociation(struct pri *ctrl, ++ const char *name, unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct roseQsigAOCChargingAssociation *charging) ++{ ++ int32_t value; ++ int length; ++ int explicit_offset; ++ const unsigned char *explicit_end; ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s ChargingAssociation\n", name); ++ } ++ switch (tag) { ++ case ASN1_TYPE_INTEGER: ++ charging->type = 0; /* charge_identifier */ ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "chargeIdentifier", tag, pos, end, &value)); ++ charging->id = value; ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 0: ++ charging->type = 1; /* charged_number */ ++ ++ /* Remove EXPLICIT tag */ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " Explicit %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(explicit_end, explicit_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, explicit_end, &tag)); ++ ASN1_CALL(pos, rose_dec_PartyNumber(ctrl, "chargedNumber", tag, pos, ++ explicit_end, &charging->number)); ++ ++ ASN1_END_FIXUP(ctrl, pos, explicit_offset, explicit_end, end); ++ break; ++ default: ++ ASN1_DID_NOT_EXPECT_TAG(ctrl, tag); ++ return NULL; ++ } ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the Q.SIG ChargeRequest invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_qsig_ChargeRequest_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, union rose_msg_invoke_args *args) ++{ ++ int32_t value; ++ int length; ++ int seq_offset; ++ int advice_offset; ++ const unsigned char *seq_end; ++ const unsigned char *advice_end; ++ struct roseQsigChargeRequestArg_ARG *charge_request; ++ ++ charge_request = &args->qsig.ChargeRequest; ++ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " ChargeRequest %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ /* SEQUENCE SIZE(0..7) OF AdviceModeCombination */ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " adviceModeCombinations %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, seq_end, &length)); ++ ASN1_END_SETUP(advice_end, advice_offset, length, pos, seq_end); ++ ++ /* Decode SIZE(0..7) OF AdviceModeCombination */ ++ charge_request->num_records = 0; ++ while (pos < advice_end && *pos != ASN1_INDEF_TERM) { ++ if (charge_request->num_records < ++ ARRAY_LEN(charge_request->advice_mode_combinations)) { ++ ASN1_CALL(pos, asn1_dec_tag(pos, advice_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_ENUMERATED); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "adviceModeCombination", tag, pos, ++ advice_end, &value)); ++ charge_request->advice_mode_combinations[charge_request->num_records] = ++ value; ++ ++charge_request->num_records; ++ } else { ++ /* Too many records */ ++ return NULL; ++ } ++ } ++ ++ ASN1_END_FIXUP(ctrl, pos, advice_offset, advice_end, seq_end); ++ ++ /* Fixup will skip over any OPTIONAL manufacturer extension information */ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the Q.SIG ChargeRequest result argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_qsig_ChargeRequest_RES(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, union rose_msg_result_args *args) ++{ ++ int32_t value; ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ struct roseQsigChargeRequestRes_RES *charge_request; ++ ++ charge_request = &args->qsig.ChargeRequest; ++ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " ChargeRequest %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_ENUMERATED); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "adviceModeCombination", tag, pos, seq_end, ++ &value)); ++ charge_request->advice_mode_combination = value; ++ ++ /* Fixup will skip over any OPTIONAL manufacturer extension information */ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the Q.SIG AocFinal invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_qsig_AocFinal_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, union rose_msg_invoke_args *args) ++{ ++ int32_t value; ++ int length; ++ int seq_offset; ++ int specific_offset; ++ const unsigned char *seq_end; ++ const unsigned char *specific_end; ++ const unsigned char *save_pos; ++ struct roseQsigAocFinalArg_ARG *aoc_final; ++ ++ aoc_final = &args->qsig.AocFinal; ++ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " AocFinal %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ switch (tag) { ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 0: ++ aoc_final->type = 0; /* charge_not_available */ ++ ASN1_CALL(pos, asn1_dec_null(ctrl, "chargeNotAvailable", tag, pos, seq_end)); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 1: ++ aoc_final->type = 1; /* free_of_charge */ ++ ASN1_CALL(pos, asn1_dec_null(ctrl, "freeOfCharge", tag, pos, seq_end)); ++ break; ++ case ASN1_TAG_SEQUENCE: ++ aoc_final->type = 2; /* specific_currency */ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " specificCurrency %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(specific_end, specific_offset, length, pos, seq_end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, specific_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ++ ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 1); ++ ASN1_CALL(pos, rose_dec_qsig_AOC_RecordedCurrency(ctrl, "recordedCurrency", tag, ++ pos, specific_end, &aoc_final->specific.recorded)); ++ ++ if (pos < specific_end && *pos != ASN1_INDEF_TERM) { ++ ASN1_CALL(pos, asn1_dec_tag(pos, specific_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_CLASS_CONTEXT_SPECIFIC | 2); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "finalBillingId", tag, pos, specific_end, ++ &value)); ++ aoc_final->specific.billing_id = value; ++ aoc_final->specific.billing_id_present = 1; ++ } else { ++ aoc_final->specific.billing_id_present = 0; ++ } ++ ++ ASN1_END_FIXUP(ctrl, pos, specific_offset, specific_end, seq_end); ++ break; ++ default: ++ ASN1_DID_NOT_EXPECT_TAG(ctrl, tag); ++ return NULL; ++ } ++ ++ /* ++ * A sequence specifies an ordered list of component types. ++ * However, for simplicity we are not checking the order of ++ * the remaining optional components. ++ */ ++ aoc_final->charging_association_present = 0; ++ while (pos < seq_end && *pos != ASN1_INDEF_TERM) { ++ save_pos = pos; ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ switch (tag) { ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 0: ++ case ASN1_TYPE_INTEGER: ++ ASN1_CALL(pos, rose_dec_qsig_AOC_ChargingAssociation(ctrl, ++ "chargingAssociation", tag, pos, seq_end, ++ &aoc_final->charging_association)); ++ aoc_final->charging_association_present = 1; ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 1: ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 2: ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " finalArgExtension %s\n", asn1_tag2str(tag)); ++ } ++ /* Fixup will skip over the manufacturer extension information */ ++ default: ++ pos = save_pos; ++ goto cancel_options; ++ } ++ } ++cancel_options:; ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the Q.SIG AocInterim invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_qsig_AocInterim_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, union rose_msg_invoke_args *args) ++{ ++ int32_t value; ++ int length; ++ int seq_offset; ++ int specific_offset; ++ const unsigned char *seq_end; ++ const unsigned char *specific_end; ++ struct roseQsigAocInterimArg_ARG *aoc_interim; ++ ++ aoc_interim = &args->qsig.AocInterim; ++ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " AocInterim %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ switch (tag) { ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 0: ++ aoc_interim->type = 0; /* charge_not_available */ ++ ASN1_CALL(pos, asn1_dec_null(ctrl, "chargeNotAvailable", tag, pos, seq_end)); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 1: ++ aoc_interim->type = 1; /* free_of_charge */ ++ ASN1_CALL(pos, asn1_dec_null(ctrl, "freeOfCharge", tag, pos, seq_end)); ++ break; ++ case ASN1_TAG_SEQUENCE: ++ aoc_interim->type = 2; /* specific_currency */ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " specificCurrency %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(specific_end, specific_offset, length, pos, seq_end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, specific_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ++ ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 1); ++ ASN1_CALL(pos, rose_dec_qsig_AOC_RecordedCurrency(ctrl, "recordedCurrency", tag, ++ pos, specific_end, &aoc_interim->specific.recorded)); ++ ++ if (pos < specific_end && *pos != ASN1_INDEF_TERM) { ++ ASN1_CALL(pos, asn1_dec_tag(pos, specific_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_CLASS_CONTEXT_SPECIFIC | 2); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "interimBillingId", tag, pos, specific_end, ++ &value)); ++ aoc_interim->specific.billing_id = value; ++ aoc_interim->specific.billing_id_present = 1; ++ } else { ++ aoc_interim->specific.billing_id_present = 0; ++ } ++ ++ ASN1_END_FIXUP(ctrl, pos, specific_offset, specific_end, seq_end); ++ break; ++ default: ++ ASN1_DID_NOT_EXPECT_TAG(ctrl, tag); ++ return NULL; ++ } ++ ++ /* Fixup will skip over any OPTIONAL manufacturer extension information */ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the Q.SIG AocRate invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_qsig_AocRate_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, union rose_msg_invoke_args *args) ++{ ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ struct roseQsigAocRateArg_ARG *aoc_rate; ++ ++ aoc_rate = &args->qsig.AocRate; ++ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " AocRate %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ switch (tag) { ++ case ASN1_TYPE_NULL: ++ aoc_rate->type = 0; /* charge_not_available */ ++ ASN1_CALL(pos, asn1_dec_null(ctrl, "chargeNotAvailable", tag, pos, seq_end)); ++ break; ++ case ASN1_TAG_SEQUENCE: ++ aoc_rate->type = 1; /* currency_info_list */ ++ ASN1_CALL(pos, rose_dec_qsig_AOCSCurrencyInfoList(ctrl, "aocSCurrencyInfoList", ++ tag, pos, seq_end, &aoc_rate->currency_info)); ++ break; ++ default: ++ ASN1_DID_NOT_EXPECT_TAG(ctrl, tag); ++ return NULL; ++ } ++ ++ /* Fixup will skip over any OPTIONAL manufacturer extension information */ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the Q.SIG AocComplete invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_qsig_AocComplete_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, union rose_msg_invoke_args *args) ++{ ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ const unsigned char *save_pos; ++ struct roseQsigAocCompleteArg_ARG *aoc_complete; ++ ++ aoc_complete = &args->qsig.AocComplete; ++ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " AocComplete %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CALL(pos, rose_dec_PartyNumber(ctrl, "chargedUser", tag, pos, seq_end, ++ &aoc_complete->charged_user_number)); ++ ++ /* ++ * A sequence specifies an ordered list of component types. ++ * However, for simplicity we are not checking the order of ++ * the remaining optional components. ++ */ ++ aoc_complete->charging_association_present = 0; ++ while (pos < seq_end && *pos != ASN1_INDEF_TERM) { ++ save_pos = pos; ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ switch (tag) { ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 0: ++ case ASN1_TYPE_INTEGER: ++ ASN1_CALL(pos, rose_dec_qsig_AOC_ChargingAssociation(ctrl, ++ "chargingAssociation", tag, pos, seq_end, ++ &aoc_complete->charging_association)); ++ aoc_complete->charging_association_present = 1; ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 1: ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 2: ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " completeArgExtension %s\n", asn1_tag2str(tag)); ++ } ++ /* Fixup will skip over the manufacturer extension information */ ++ default: ++ pos = save_pos; ++ goto cancel_options; ++ } ++ } ++cancel_options:; ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the Q.SIG AocComplete result argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_qsig_AocComplete_RES(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, union rose_msg_result_args *args) ++{ ++ int32_t value; ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ struct roseQsigAocCompleteRes_RES *aoc_complete; ++ ++ aoc_complete = &args->qsig.AocComplete; ++ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " AocComplete %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_ENUMERATED); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "chargingOption", tag, pos, seq_end, &value)); ++ aoc_complete->charging_option = value; ++ ++ /* Fixup will skip over any OPTIONAL manufacturer extension information */ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the Q.SIG AocDivChargeReq invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_qsig_AocDivChargeReq_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, union rose_msg_invoke_args *args) ++{ ++ int32_t value; ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ struct roseQsigAocDivChargeReqArg_ARG *aoc_div_charge_req; ++ ++ aoc_div_charge_req = &args->qsig.AocDivChargeReq; ++ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " AocDivChargeReq %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CALL(pos, rose_dec_PartyNumber(ctrl, "divertingUser", tag, pos, seq_end, ++ &aoc_div_charge_req->diverting_user_number)); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ switch (tag) { ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 0: ++ case ASN1_TYPE_INTEGER: ++ ASN1_CALL(pos, rose_dec_qsig_AOC_ChargingAssociation(ctrl, "chargingAssociation", ++ tag, pos, seq_end, &aoc_div_charge_req->charging_association)); ++ aoc_div_charge_req->charging_association_present = 1; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ break; ++ default: ++ aoc_div_charge_req->charging_association_present = 0; ++ break; ++ } ++ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_ENUMERATED); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "diversionType", tag, pos, seq_end, &value)); ++ aoc_div_charge_req->diversion_type = value; ++ ++ /* Fixup will skip over any OPTIONAL manufacturer extension information */ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/* ------------------------------------------------------------------- */ ++/* end rose_qsig_aoc.c */ + +Property changes on: rose_qsig_aoc.c +___________________________________________________________________ +Added: svn:eol-style + + native +Added: svn:mime-type + + text/plain +Added: svn:keywords + + 'Author Date Id Revision' + +Index: rose_etsi_aoc.c +=================================================================== +--- a/rose_etsi_aoc.c (.../tags/1.4.10.2) (revision 0) ++++ b/rose_etsi_aoc.c (.../branches/1.4) (revision 1357) +@@ -0,0 +1,1929 @@ ++/* ++ * libpri: An implementation of Primary Rate ISDN ++ * ++ * Copyright (C) 2009 Digium, Inc. ++ * ++ * Richard Mudgett <rmudgett@digium.com> ++ * ++ * See http://www.asterisk.org for more information about ++ * the Asterisk project. Please do not directly contact ++ * any of the maintainers of this project for assistance; ++ * the project provides a web site, mailing lists and IRC ++ * channels for your use. ++ * ++ * This program is free software, distributed under the terms of ++ * the GNU General Public License Version 2 as published by the ++ * Free Software Foundation. See the LICENSE file included with ++ * this program for more details. ++ * ++ * In addition, when this program is distributed with Asterisk in ++ * any form that would qualify as a 'combined work' or as a ++ * 'derivative work' (but not mere aggregation), you can redistribute ++ * and/or modify the combination under the terms of the license ++ * provided with that copy of Asterisk, instead of the license ++ * terms granted here. ++ */ ++ ++/*! ++ * \file ++ * \brief ROSE Advice Of Charge (AOC) operations ++ * ++ * Advice of Charge (AOC) supplementary service EN 300 182-1 ++ * ++ * \author Richard Mudgett <rmudgett@digium.com> ++ */ ++ ++ ++#include "compat.h" ++#include "libpri.h" ++#include "pri_internal.h" ++#include "rose.h" ++#include "rose_internal.h" ++#include "asn1.h" ++ ++ ++/* ------------------------------------------------------------------- */ ++ ++/*! ++ * \internal ++ * \brief Encode the Time type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param tag Component tag to identify the encoded component. ++ * The tag should be ASN1_TAG_SEQUENCE unless the caller implicitly ++ * tags it otherwise. ++ * \param time Time information to encode. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *rose_enc_etsi_AOC_Time(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, unsigned tag, const struct roseEtsiAOCTime *time) ++{ ++ unsigned char *seq_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, tag); ++ ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 1, ++ time->length)); ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 2, time->scale)); ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Encode the Amount type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param tag Component tag to identify the encoded component. ++ * The tag should be ASN1_TAG_SEQUENCE unless the caller implicitly ++ * tags it otherwise. ++ * \param amount Amount information to encode. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *rose_enc_etsi_AOC_Amount(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, unsigned tag, const struct roseEtsiAOCAmount *amount) ++{ ++ unsigned char *seq_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, tag); ++ ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 1, ++ amount->currency)); ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 2, ++ amount->multiplier)); ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Encode the RecordedCurrency type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param tag Component tag to identify the encoded component. ++ * The tag should be ASN1_TAG_SEQUENCE unless the caller implicitly ++ * tags it otherwise. ++ * \param recorded Recorded currency information to encode. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *rose_enc_etsi_AOC_RecordedCurrency(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, unsigned tag, ++ const struct roseEtsiAOCRecordedCurrency *recorded) ++{ ++ unsigned char *seq_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, tag); ++ ++ ASN1_CALL(pos, asn1_enc_string_max(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 1, ++ recorded->currency, sizeof(recorded->currency) - 1)); ++ ASN1_CALL(pos, rose_enc_etsi_AOC_Amount(ctrl, pos, end, ++ ASN1_CLASS_CONTEXT_SPECIFIC | 2, &recorded->amount)); ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Encode the DurationCurrency type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param tag Component tag to identify the encoded component. ++ * The tag should be ASN1_TAG_SEQUENCE unless the caller implicitly ++ * tags it otherwise. ++ * \param duration Duration currency information to encode. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *rose_enc_etsi_AOC_DurationCurrency(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, unsigned tag, ++ const struct roseEtsiAOCDurationCurrency *duration) ++{ ++ unsigned char *seq_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, tag); ++ ++ ASN1_CALL(pos, asn1_enc_string_max(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 1, ++ duration->currency, sizeof(duration->currency) - 1)); ++ ASN1_CALL(pos, rose_enc_etsi_AOC_Amount(ctrl, pos, end, ++ ASN1_CLASS_CONTEXT_SPECIFIC | 2, &duration->amount)); ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 3, ++ duration->charging_type)); ++ ASN1_CALL(pos, rose_enc_etsi_AOC_Time(ctrl, pos, end, ++ ASN1_CLASS_CONTEXT_SPECIFIC | 4, &duration->time)); ++ if (duration->granularity_present) { ++ ASN1_CALL(pos, rose_enc_etsi_AOC_Time(ctrl, pos, end, ++ ASN1_CLASS_CONTEXT_SPECIFIC | 5, &duration->granularity)); ++ } ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Encode the FlatRateCurrency type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param tag Component tag to identify the encoded component. ++ * The tag should be ASN1_TAG_SEQUENCE unless the caller implicitly ++ * tags it otherwise. ++ * \param flat_rate Flat rate currency information to encode. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *rose_enc_etsi_AOC_FlatRateCurrency(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, unsigned tag, ++ const struct roseEtsiAOCFlatRateCurrency *flat_rate) ++{ ++ unsigned char *seq_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, tag); ++ ++ ASN1_CALL(pos, asn1_enc_string_max(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 1, ++ flat_rate->currency, sizeof(flat_rate->currency) - 1)); ++ ASN1_CALL(pos, rose_enc_etsi_AOC_Amount(ctrl, pos, end, ++ ASN1_CLASS_CONTEXT_SPECIFIC | 2, &flat_rate->amount)); ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Encode the VolumeRateCurrency type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param tag Component tag to identify the encoded component. ++ * The tag should be ASN1_TAG_SEQUENCE unless the caller implicitly ++ * tags it otherwise. ++ * \param volume_rate Volume rate currency information to encode. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *rose_enc_etsi_AOC_VolumeRateCurrency(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, unsigned tag, ++ const struct roseEtsiAOCVolumeRateCurrency *volume_rate) ++{ ++ unsigned char *seq_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, tag); ++ ++ ASN1_CALL(pos, asn1_enc_string_max(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 1, ++ volume_rate->currency, sizeof(volume_rate->currency) - 1)); ++ ASN1_CALL(pos, rose_enc_etsi_AOC_Amount(ctrl, pos, end, ++ ASN1_CLASS_CONTEXT_SPECIFIC | 2, &volume_rate->amount)); ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 3, ++ volume_rate->unit)); ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Encode the AOCSCurrencyInfo type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param tag Component tag to identify the encoded component. ++ * The tag should be ASN1_TAG_SEQUENCE unless the caller implicitly ++ * tags it otherwise. ++ * \param currency_info Currency information record to encode. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *rose_enc_etsi_AOCSCurrencyInfo(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, unsigned tag, ++ const struct roseEtsiAOCSCurrencyInfo *currency_info) ++{ ++ unsigned char *seq_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, tag); ++ ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, ++ currency_info->charged_item)); ++ ++ switch (currency_info->currency_type) { ++ case 0: /* specialChargingCode */ ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_INTEGER, ++ currency_info->u.special_charging_code)); ++ break; ++ case 1: /* durationCurrency */ ++ ASN1_CALL(pos, rose_enc_etsi_AOC_DurationCurrency(ctrl, pos, end, ++ ASN1_CLASS_CONTEXT_SPECIFIC | 1, ¤cy_info->u.duration)); ++ break; ++ case 2: /* flatRateCurrency */ ++ ASN1_CALL(pos, rose_enc_etsi_AOC_FlatRateCurrency(ctrl, pos, end, ++ ASN1_CLASS_CONTEXT_SPECIFIC | 2, ¤cy_info->u.flat_rate)); ++ break; ++ case 3: /* volumeRateCurrency */ ++ ASN1_CALL(pos, rose_enc_etsi_AOC_VolumeRateCurrency(ctrl, pos, end, ++ ASN1_CLASS_CONTEXT_SPECIFIC | 3, ¤cy_info->u.volume_rate)); ++ break; ++ case 4: /* freeOfCharge */ ++ ASN1_CALL(pos, asn1_enc_null(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 4)); ++ break; ++ case 5: /* currencyInfoNotAvailable */ ++ ASN1_CALL(pos, asn1_enc_null(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 5)); ++ break; ++ default: ++ ASN1_ENC_ERROR(ctrl, "Unknown currency type"); ++ return NULL; ++ } ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Encode the AOCSCurrencyInfoList type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param tag Component tag to identify the encoded component. ++ * The tag should be ASN1_TAG_SEQUENCE unless the caller implicitly ++ * tags it otherwise. ++ * \param currency_info Currency information list to encode. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *rose_enc_etsi_AOCSCurrencyInfoList(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, unsigned tag, ++ const struct roseEtsiAOCSCurrencyInfoList *currency_info) ++{ ++ unsigned index; ++ unsigned char *seq_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, tag); ++ ++ for (index = 0; index < currency_info->num_records; ++index) { ++ ASN1_CALL(pos, rose_enc_etsi_AOCSCurrencyInfo(ctrl, pos, end, ASN1_TAG_SEQUENCE, ++ ¤cy_info->list[index])); ++ } ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Encode the RecordedUnits type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param tag Component tag to identify the encoded component. ++ * The tag should be ASN1_TAG_SEQUENCE unless the caller implicitly ++ * tags it otherwise. ++ * \param recorded Recorded units information record to encode. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *rose_enc_etsi_AOC_RecordedUnits(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, unsigned tag, ++ const struct roseEtsiAOCRecordedUnits *recorded) ++{ ++ unsigned char *seq_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, tag); ++ ++ if (recorded->not_available) { ++ ASN1_CALL(pos, asn1_enc_null(pos, end, ASN1_TYPE_NULL)); ++ } else { ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_INTEGER, ++ recorded->number_of_units)); ++ } ++ ++ if (recorded->type_of_unit_present) { ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_INTEGER, ++ recorded->type_of_unit)); ++ } ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Encode the RecordedUnitsList type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param tag Component tag to identify the encoded component. ++ * The tag should be ASN1_TAG_SEQUENCE unless the caller implicitly ++ * tags it otherwise. ++ * \param recorded_info Recorded units information list to encode. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *rose_enc_etsi_AOC_RecordedUnitsList(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, unsigned tag, ++ const struct roseEtsiAOCRecordedUnitsList *recorded_info) ++{ ++ unsigned index; ++ unsigned char *seq_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, tag); ++ ++ for (index = 0; index < recorded_info->num_records; ++index) { ++ ASN1_CALL(pos, rose_enc_etsi_AOC_RecordedUnits(ctrl, pos, end, ASN1_TAG_SEQUENCE, ++ &recorded_info->list[index])); ++ } ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Encode the ChargingAssociation type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param charging Charging association information to encode. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *rose_enc_etsi_AOC_ChargingAssociation(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, ++ const struct roseEtsiAOCChargingAssociation *charging) ++{ ++ unsigned char *explicit_len; ++ ++ switch (charging->type) { ++ case 0: /* charge_identifier */ ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_INTEGER, charging->id)); ++ break; ++ case 1: /* charged_number */ ++ /* EXPLICIT tag */ ++ ASN1_CONSTRUCTED_BEGIN(explicit_len, pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 0); ++ ASN1_CALL(pos, rose_enc_PartyNumber(ctrl, pos, end, &charging->number)); ++ ASN1_CONSTRUCTED_END(explicit_len, pos, end); ++ break; ++ default: ++ ASN1_ENC_ERROR(ctrl, "Unknown ChargingAssociation type"); ++ return NULL; ++ } ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Encode the AOCECurrencyInfo type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param tag Component tag to identify the encoded component. ++ * The tag should be ASN1_TAG_SEQUENCE unless the caller implicitly ++ * tags it otherwise. ++ * \param currency_info Currency information to encode. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *rose_enc_etsi_AOCECurrencyInfo(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, unsigned tag, ++ const struct roseEtsiAOCECurrencyInfo *currency_info) ++{ ++ unsigned char *seq_len; ++ unsigned char *specific_seq_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, tag); ++ ++ if (currency_info->free_of_charge) { ++ ASN1_CALL(pos, asn1_enc_null(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 1)); ++ } else { ++ ASN1_CONSTRUCTED_BEGIN(specific_seq_len, pos, end, ASN1_TAG_SEQUENCE); ++ ++ ASN1_CALL(pos, rose_enc_etsi_AOC_RecordedCurrency(ctrl, pos, end, ++ ASN1_CLASS_CONTEXT_SPECIFIC | 1, ¤cy_info->specific.recorded)); ++ ++ if (currency_info->specific.billing_id_present) { ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 2, ++ currency_info->specific.billing_id)); ++ } ++ ++ ASN1_CONSTRUCTED_END(specific_seq_len, pos, end); ++ } ++ ++ if (currency_info->charging_association_present) { ++ ASN1_CALL(pos, rose_enc_etsi_AOC_ChargingAssociation(ctrl, pos, end, ++ ¤cy_info->charging_association)); ++ } ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Encode the AOCEChargingUnitInfo type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param tag Component tag to identify the encoded component. ++ * The tag should be ASN1_TAG_SEQUENCE unless the caller implicitly ++ * tags it otherwise. ++ * \param charging_unit Charging unit information to encode. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *rose_enc_etsi_AOCEChargingUnitInfo(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, unsigned tag, ++ const struct roseEtsiAOCEChargingUnitInfo *charging_unit) ++{ ++ unsigned char *seq_len; ++ unsigned char *specific_seq_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, tag); ++ ++ if (charging_unit->free_of_charge) { ++ ASN1_CALL(pos, asn1_enc_null(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 1)); ++ } else { ++ ASN1_CONSTRUCTED_BEGIN(specific_seq_len, pos, end, ASN1_TAG_SEQUENCE); ++ ++ ASN1_CALL(pos, rose_enc_etsi_AOC_RecordedUnitsList(ctrl, pos, end, ++ ASN1_CLASS_CONTEXT_SPECIFIC | 1, &charging_unit->specific.recorded)); ++ ++ if (charging_unit->specific.billing_id_present) { ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 2, ++ charging_unit->specific.billing_id)); ++ } ++ ++ ASN1_CONSTRUCTED_END(specific_seq_len, pos, end); ++ } ++ ++ if (charging_unit->charging_association_present) { ++ ASN1_CALL(pos, rose_enc_etsi_AOC_ChargingAssociation(ctrl, pos, end, ++ &charging_unit->charging_association)); ++ } ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the ChargingRequest invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_etsi_ChargingRequest_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ return asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, ++ args->etsi.ChargingRequest.charging_case); ++} ++ ++/*! ++ * \brief Encode the ChargingRequest result facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_etsi_ChargingRequest_RES(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_result_args *args) ++{ ++ switch (args->etsi.ChargingRequest.type) { ++ case 0: /* currency_info_list */ ++ ASN1_CALL(pos, rose_enc_etsi_AOCSCurrencyInfoList(ctrl, pos, end, ++ ASN1_TAG_SEQUENCE, &args->etsi.ChargingRequest.u.currency_info)); ++ break; ++ case 1: /* special_arrangement_info */ ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_INTEGER, ++ args->etsi.ChargingRequest.u.special_arrangement)); ++ break; ++ case 2: /* charging_info_follows */ ++ ASN1_CALL(pos, asn1_enc_null(pos, end, ASN1_TYPE_NULL)); ++ break; ++ default: ++ ASN1_ENC_ERROR(ctrl, "Unknown ChargingRequst type"); ++ return NULL; ++ } ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the AOCSCurrency invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_etsi_AOCSCurrency_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ switch (args->etsi.AOCSCurrency.type) { ++ case 0: /* charge_not_available */ ++ ASN1_CALL(pos, asn1_enc_null(pos, end, ASN1_TYPE_NULL)); ++ break; ++ case 1: /* currency_info_list */ ++ ASN1_CALL(pos, rose_enc_etsi_AOCSCurrencyInfoList(ctrl, pos, end, ++ ASN1_TAG_SEQUENCE, &args->etsi.AOCSCurrency.currency_info)); ++ break; ++ default: ++ ASN1_ENC_ERROR(ctrl, "Unknown AOCSCurrency type"); ++ return NULL; ++ } ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the AOCSSpecialArr invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_etsi_AOCSSpecialArr_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ switch (args->etsi.AOCSSpecialArr.type) { ++ case 0: /* charge_not_available */ ++ ASN1_CALL(pos, asn1_enc_null(pos, end, ASN1_TYPE_NULL)); ++ break; ++ case 1: /* special_arrangement_info */ ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_INTEGER, ++ args->etsi.AOCSSpecialArr.special_arrangement)); ++ break; ++ default: ++ ASN1_ENC_ERROR(ctrl, "Unknown AOCSSpecialArr type"); ++ return NULL; ++ } ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the AOCDCurrency invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_etsi_AOCDCurrency_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ const struct roseEtsiAOCDCurrency_ARG *aoc_d; ++ unsigned char *seq_len; ++ ++ aoc_d = &args->etsi.AOCDCurrency; ++ switch (aoc_d->type) { ++ case 0: /* charge_not_available */ ++ ASN1_CALL(pos, asn1_enc_null(pos, end, ASN1_TYPE_NULL)); ++ break; ++ case 1: /* free_of_charge */ ++ ASN1_CALL(pos, asn1_enc_null(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 1)); ++ break; ++ case 2: /* specific_currency */ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ASN1_TAG_SEQUENCE); ++ ++ ASN1_CALL(pos, rose_enc_etsi_AOC_RecordedCurrency(ctrl, pos, end, ++ ASN1_CLASS_CONTEXT_SPECIFIC | 1, &aoc_d->specific.recorded)); ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 2, ++ aoc_d->specific.type_of_charging_info)); ++ if (aoc_d->specific.billing_id_present) { ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 3, ++ aoc_d->specific.billing_id)); ++ } ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ break; ++ default: ++ ASN1_ENC_ERROR(ctrl, "Unknown AOCDCurrency type"); ++ return NULL; ++ } ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the AOCDChargingUnit invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_etsi_AOCDChargingUnit_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ const struct roseEtsiAOCDChargingUnit_ARG *aoc_d; ++ unsigned char *seq_len; ++ ++ aoc_d = &args->etsi.AOCDChargingUnit; ++ switch (aoc_d->type) { ++ case 0: /* charge_not_available */ ++ ASN1_CALL(pos, asn1_enc_null(pos, end, ASN1_TYPE_NULL)); ++ break; ++ case 1: /* free_of_charge */ ++ ASN1_CALL(pos, asn1_enc_null(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 1)); ++ break; ++ case 2: /* specific_charging_units */ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ASN1_TAG_SEQUENCE); ++ ++ ASN1_CALL(pos, rose_enc_etsi_AOC_RecordedUnitsList(ctrl, pos, end, ++ ASN1_CLASS_CONTEXT_SPECIFIC | 1, &aoc_d->specific.recorded)); ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 2, ++ aoc_d->specific.type_of_charging_info)); ++ if (aoc_d->specific.billing_id_present) { ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 3, ++ aoc_d->specific.billing_id)); ++ } ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ break; ++ default: ++ ASN1_ENC_ERROR(ctrl, "Unknown AOCDChargingUnit type"); ++ return NULL; ++ } ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the AOCECurrency invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_etsi_AOCECurrency_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ switch (args->etsi.AOCECurrency.type) { ++ case 0: /* charge_not_available */ ++ ASN1_CALL(pos, asn1_enc_null(pos, end, ASN1_TYPE_NULL)); ++ break; ++ case 1: /* currency_info */ ++ ASN1_CALL(pos, rose_enc_etsi_AOCECurrencyInfo(ctrl, pos, end, ASN1_TAG_SEQUENCE, ++ &args->etsi.AOCECurrency.currency_info)); ++ break; ++ default: ++ ASN1_ENC_ERROR(ctrl, "Unknown AOCECurrency type"); ++ return NULL; ++ } ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the AOCEChargingUnit invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_etsi_AOCEChargingUnit_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ switch (args->etsi.AOCEChargingUnit.type) { ++ case 0: /* charge_not_available */ ++ ASN1_CALL(pos, asn1_enc_null(pos, end, ASN1_TYPE_NULL)); ++ break; ++ case 1: /* charging_unit */ ++ ASN1_CALL(pos, rose_enc_etsi_AOCEChargingUnitInfo(ctrl, pos, end, ++ ASN1_TAG_SEQUENCE, &args->etsi.AOCEChargingUnit.charging_unit)); ++ break; ++ default: ++ ASN1_ENC_ERROR(ctrl, "Unknown AOCEChargingUnit type"); ++ return NULL; ++ } ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Decode the Time type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param name Field name ++ * \param tag Component tag that identified this production. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param time Parameter storage to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++static const unsigned char *rose_dec_etsi_AOC_Time(struct pri *ctrl, const char *name, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct roseEtsiAOCTime *time) ++{ ++ int32_t value; ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s Time %s\n", name, asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_CLASS_CONTEXT_SPECIFIC | 1); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "lengthOfTimeUnit", tag, pos, seq_end, &value)); ++ time->length = value; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_CLASS_CONTEXT_SPECIFIC | 2); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "scale", tag, pos, seq_end, &value)); ++ time->scale = value; ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Decode the Amount type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param name Field name ++ * \param tag Component tag that identified this production. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param amount Parameter storage to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++static const unsigned char *rose_dec_etsi_AOC_Amount(struct pri *ctrl, const char *name, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct roseEtsiAOCAmount *amount) ++{ ++ int32_t value; ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s Amount %s\n", name, asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_CLASS_CONTEXT_SPECIFIC | 1); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "currencyAmount", tag, pos, seq_end, &value)); ++ amount->currency = value; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_CLASS_CONTEXT_SPECIFIC | 2); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "multiplier", tag, pos, seq_end, &value)); ++ amount->multiplier = value; ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Decode the RecordedCurrency type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param name Field name ++ * \param tag Component tag that identified this production. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param recorded Parameter storage to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++static const unsigned char *rose_dec_etsi_AOC_RecordedCurrency(struct pri *ctrl, ++ const char *name, unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct roseEtsiAOCRecordedCurrency *recorded) ++{ ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ size_t str_len; ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s RecordedCurrency %s\n", name, asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag & ~ASN1_PC_MASK, tag, ASN1_CLASS_CONTEXT_SPECIFIC | 1); ++ ASN1_CALL(pos, asn1_dec_string_max(ctrl, "rCurrency", tag, pos, seq_end, ++ sizeof(recorded->currency), recorded->currency, &str_len)); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ++ ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 2); ++ ASN1_CALL(pos, rose_dec_etsi_AOC_Amount(ctrl, "rAmount", tag, pos, seq_end, ++ &recorded->amount)); ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Decode the DurationCurrency type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param name Field name ++ * \param tag Component tag that identified this production. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param duration Parameter storage to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++static const unsigned char *rose_dec_etsi_AOC_DurationCurrency(struct pri *ctrl, ++ const char *name, unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct roseEtsiAOCDurationCurrency *duration) ++{ ++ int32_t value; ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ size_t str_len; ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s DurationCurrency %s\n", name, asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag & ~ASN1_PC_MASK, tag, ASN1_CLASS_CONTEXT_SPECIFIC | 1); ++ ASN1_CALL(pos, asn1_dec_string_max(ctrl, "dCurrency", tag, pos, seq_end, ++ sizeof(duration->currency), duration->currency, &str_len)); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ++ ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 2); ++ ASN1_CALL(pos, rose_dec_etsi_AOC_Amount(ctrl, "dAmount", tag, pos, seq_end, ++ &duration->amount)); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_CLASS_CONTEXT_SPECIFIC | 3); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "dChargingType", tag, pos, seq_end, &value)); ++ duration->charging_type = value; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ++ ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 4); ++ ASN1_CALL(pos, rose_dec_etsi_AOC_Time(ctrl, "dTime", tag, pos, seq_end, ++ &duration->time)); ++ ++ if (pos < seq_end && *pos != ASN1_INDEF_TERM) { ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ++ ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 5); ++ ASN1_CALL(pos, rose_dec_etsi_AOC_Time(ctrl, "dGranularity", tag, pos, seq_end, ++ &duration->granularity)); ++ duration->granularity_present = 1; ++ } else { ++ duration->granularity_present = 0; ++ } ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Decode the FlatRateCurrency type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param name Field name ++ * \param tag Component tag that identified this production. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param flat_rate Parameter storage to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++static const unsigned char *rose_dec_etsi_AOC_FlatRateCurrency(struct pri *ctrl, ++ const char *name, unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct roseEtsiAOCFlatRateCurrency *flat_rate) ++{ ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ size_t str_len; ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s FlatRateCurrency %s\n", name, asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag & ~ASN1_PC_MASK, tag, ASN1_CLASS_CONTEXT_SPECIFIC | 1); ++ ASN1_CALL(pos, asn1_dec_string_max(ctrl, "fRCurrency", tag, pos, seq_end, ++ sizeof(flat_rate->currency), flat_rate->currency, &str_len)); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ++ ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 2); ++ ASN1_CALL(pos, rose_dec_etsi_AOC_Amount(ctrl, "fRAmount", tag, pos, seq_end, ++ &flat_rate->amount)); ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Decode the VolumeRateCurrency type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param name Field name ++ * \param tag Component tag that identified this production. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param volume_rate Parameter storage to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++static const unsigned char *rose_dec_etsi_AOC_VolumeRateCurrency(struct pri *ctrl, ++ const char *name, unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct roseEtsiAOCVolumeRateCurrency *volume_rate) ++{ ++ int32_t value; ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ size_t str_len; ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s VolumeRateCurrency %s\n", name, asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag & ~ASN1_PC_MASK, tag, ASN1_CLASS_CONTEXT_SPECIFIC | 1); ++ ASN1_CALL(pos, asn1_dec_string_max(ctrl, "vRCurrency", tag, pos, seq_end, ++ sizeof(volume_rate->currency), volume_rate->currency, &str_len)); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ++ ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 2); ++ ASN1_CALL(pos, rose_dec_etsi_AOC_Amount(ctrl, "vRAmount", tag, pos, seq_end, ++ &volume_rate->amount)); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_CLASS_CONTEXT_SPECIFIC | 3); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "vRVolumeUnit", tag, pos, seq_end, &value)); ++ volume_rate->unit = value; ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Decode the AOCSCurrencyInfo type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param name Field name ++ * \param tag Component tag that identified this production. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param currency_info Parameter storage to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++static const unsigned char *rose_dec_etsi_AOCSCurrencyInfo(struct pri *ctrl, ++ const char *name, unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct roseEtsiAOCSCurrencyInfo *currency_info) ++{ ++ int32_t value; ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s AOCSCurrencyInfo %s\n", name, asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_ENUMERATED); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "chargedItem", tag, pos, seq_end, &value)); ++ currency_info->charged_item = value; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ switch (tag) { ++ case ASN1_TYPE_INTEGER: ++ currency_info->currency_type = 0; /* specialChargingCode */ ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "specialChargingCode", tag, pos, seq_end, ++ &value)); ++ currency_info->u.special_charging_code = value; ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 1: ++ currency_info->currency_type = 1; /* durationCurrency */ ++ ASN1_CALL(pos, rose_dec_etsi_AOC_DurationCurrency(ctrl, "durationCurrency", tag, ++ pos, seq_end, ¤cy_info->u.duration)); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 2: ++ currency_info->currency_type = 2; /* flatRateCurrency */ ++ ASN1_CALL(pos, rose_dec_etsi_AOC_FlatRateCurrency(ctrl, "flatRateCurrency", tag, ++ pos, seq_end, ¤cy_info->u.flat_rate)); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 3: ++ currency_info->currency_type = 3; /* volumeRateCurrency */ ++ ASN1_CALL(pos, rose_dec_etsi_AOC_VolumeRateCurrency(ctrl, "volumeRateCurrency", ++ tag, pos, seq_end, ¤cy_info->u.volume_rate)); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 4: ++ currency_info->currency_type = 4; /* freeOfCharge */ ++ ASN1_CALL(pos, asn1_dec_null(ctrl, "freeOfCharge", tag, pos, seq_end)); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 5: ++ currency_info->currency_type = 5; /* currencyInfoNotAvailable */ ++ ASN1_CALL(pos, asn1_dec_null(ctrl, "currencyInfoNotAvailable", tag, pos, ++ seq_end)); ++ break; ++ default: ++ ASN1_DID_NOT_EXPECT_TAG(ctrl, tag); ++ return NULL; ++ } ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Decode the AOCSCurrencyInfoList type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param name Field name ++ * \param tag Component tag that identified this production. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param currency_info Parameter storage to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++static const unsigned char *rose_dec_etsi_AOCSCurrencyInfoList(struct pri *ctrl, ++ const char *name, unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct roseEtsiAOCSCurrencyInfoList *currency_info) ++{ ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s AOCSCurrencyInfoList %s\n", name, asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ currency_info->num_records = 0; ++ while (pos < seq_end && *pos != ASN1_INDEF_TERM) { ++ if (currency_info->num_records < ARRAY_LEN(currency_info->list)) { ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ ASN1_CALL(pos, rose_dec_etsi_AOCSCurrencyInfo(ctrl, "listEntry", tag, pos, ++ seq_end, ¤cy_info->list[currency_info->num_records])); ++ ++currency_info->num_records; ++ } else { ++ /* Too many records */ ++ return NULL; ++ } ++ } ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Decode the RecordedUnits type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param name Field name ++ * \param tag Component tag that identified this production. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param recorded Parameter storage to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++static const unsigned char *rose_dec_etsi_AOC_RecordedUnits(struct pri *ctrl, ++ const char *name, unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct roseEtsiAOCRecordedUnits *recorded) ++{ ++ int32_t value; ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s RecordedUnits %s\n", name, asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ switch (tag) { ++ case ASN1_TYPE_INTEGER: ++ recorded->not_available = 0; ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "recordedNumberOfUnits", tag, pos, seq_end, ++ &value)); ++ recorded->number_of_units = value; ++ break; ++ case ASN1_TYPE_NULL: ++ recorded->not_available = 1; ++ recorded->number_of_units = 0; ++ ASN1_CALL(pos, asn1_dec_null(ctrl, "notAvailable", tag, pos, seq_end)); ++ break; ++ default: ++ ASN1_DID_NOT_EXPECT_TAG(ctrl, tag); ++ return NULL; ++ } ++ ++ if (pos < seq_end && *pos != ASN1_INDEF_TERM) { ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_INTEGER); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "recordedTypeOfUnits", tag, pos, seq_end, ++ &value)); ++ recorded->type_of_unit = value; ++ recorded->type_of_unit_present = 1; ++ } else { ++ recorded->type_of_unit_present = 0; ++ } ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Decode the RecordedUnitsList type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param name Field name ++ * \param tag Component tag that identified this production. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param recorded_info Parameter storage to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++static const unsigned char *rose_dec_etsi_AOC_RecordedUnitsList(struct pri *ctrl, ++ const char *name, unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct roseEtsiAOCRecordedUnitsList *recorded_info) ++{ ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s RecordedUnitsList %s\n", name, asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ recorded_info->num_records = 0; ++ while (pos < seq_end && *pos != ASN1_INDEF_TERM) { ++ if (recorded_info->num_records < ARRAY_LEN(recorded_info->list)) { ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ ASN1_CALL(pos, rose_dec_etsi_AOC_RecordedUnits(ctrl, "listEntry", tag, pos, ++ seq_end, &recorded_info->list[recorded_info->num_records])); ++ ++recorded_info->num_records; ++ } else { ++ /* Too many records */ ++ return NULL; ++ } ++ } ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Decode the ChargingAssociation type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param name Field name ++ * \param tag Component tag that identified this production. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param charging Parameter storage to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++static const unsigned char *rose_dec_etsi_AOC_ChargingAssociation(struct pri *ctrl, ++ const char *name, unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct roseEtsiAOCChargingAssociation *charging) ++{ ++ int32_t value; ++ int length; ++ int explicit_offset; ++ const unsigned char *explicit_end; ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s ChargingAssociation\n", name); ++ } ++ switch (tag) { ++ case ASN1_TYPE_INTEGER: ++ charging->type = 0; /* charge_identifier */ ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "chargeIdentifier", tag, pos, end, &value)); ++ charging->id = value; ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 0: ++ charging->type = 1; /* charged_number */ ++ ++ /* Remove EXPLICIT tag */ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " Explicit %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(explicit_end, explicit_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, explicit_end, &tag)); ++ ASN1_CALL(pos, rose_dec_PartyNumber(ctrl, "chargedNumber", tag, pos, ++ explicit_end, &charging->number)); ++ ++ ASN1_END_FIXUP(ctrl, pos, explicit_offset, explicit_end, end); ++ break; ++ default: ++ ASN1_DID_NOT_EXPECT_TAG(ctrl, tag); ++ return NULL; ++ } ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Decode the AOCECurrencyInfo type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param name Field name ++ * \param tag Component tag that identified this production. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param currency_info Parameter storage to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++static const unsigned char *rose_dec_etsi_AOCECurrencyInfo(struct pri *ctrl, ++ const char *name, unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct roseEtsiAOCECurrencyInfo *currency_info) ++{ ++ int32_t value; ++ int length; ++ int seq_offset; ++ int specific_offset; ++ const unsigned char *seq_end; ++ const unsigned char *specific_end; ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s AOCECurrencyInfo %s\n", name, asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ switch (tag) { ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 1: ++ currency_info->free_of_charge = 1; ++ ASN1_CALL(pos, asn1_dec_null(ctrl, "freeOfCharge", tag, pos, seq_end)); ++ break; ++ case ASN1_TAG_SEQUENCE: ++ currency_info->free_of_charge = 0; ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " specificCurrency %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(specific_end, specific_offset, length, pos, seq_end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, specific_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ++ ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 1); ++ ASN1_CALL(pos, rose_dec_etsi_AOC_RecordedCurrency(ctrl, "recordedCurrency", tag, ++ pos, specific_end, ¤cy_info->specific.recorded)); ++ ++ if (pos < specific_end && *pos != ASN1_INDEF_TERM) { ++ ASN1_CALL(pos, asn1_dec_tag(pos, specific_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_CLASS_CONTEXT_SPECIFIC | 2); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "billingId", tag, pos, specific_end, ++ &value)); ++ currency_info->specific.billing_id = value; ++ currency_info->specific.billing_id_present = 1; ++ } else { ++ currency_info->specific.billing_id_present = 0; ++ } ++ ++ ASN1_END_FIXUP(ctrl, pos, specific_offset, specific_end, seq_end); ++ break; ++ default: ++ ASN1_DID_NOT_EXPECT_TAG(ctrl, tag); ++ return NULL; ++ } ++ ++ if (pos < seq_end && *pos != ASN1_INDEF_TERM) { ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CALL(pos, rose_dec_etsi_AOC_ChargingAssociation(ctrl, "chargingAssociation", ++ tag, pos, seq_end, ¤cy_info->charging_association)); ++ currency_info->charging_association_present = 1; ++ } else { ++ currency_info->charging_association_present = 0; ++ } ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Decode the AOCEChargingUnitInfo type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param name Field name ++ * \param tag Component tag that identified this production. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param charging_unit Parameter storage to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++static const unsigned char *rose_dec_etsi_AOCEChargingUnitInfo(struct pri *ctrl, ++ const char *name, unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct roseEtsiAOCEChargingUnitInfo *charging_unit) ++{ ++ int32_t value; ++ int length; ++ int seq_offset; ++ int specific_offset; ++ const unsigned char *seq_end; ++ const unsigned char *specific_end; ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s AOCEChargingUnitInfo %s\n", name, asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ switch (tag) { ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 1: ++ charging_unit->free_of_charge = 1; ++ ASN1_CALL(pos, asn1_dec_null(ctrl, "freeOfCharge", tag, pos, seq_end)); ++ break; ++ case ASN1_TAG_SEQUENCE: ++ charging_unit->free_of_charge = 0; ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " specificChargingUnits %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(specific_end, specific_offset, length, pos, seq_end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, specific_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ++ ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 1); ++ ASN1_CALL(pos, rose_dec_etsi_AOC_RecordedUnitsList(ctrl, "recordedUnitsList", ++ tag, pos, specific_end, &charging_unit->specific.recorded)); ++ ++ if (pos < specific_end && *pos != ASN1_INDEF_TERM) { ++ ASN1_CALL(pos, asn1_dec_tag(pos, specific_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_CLASS_CONTEXT_SPECIFIC | 2); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "billingId", tag, pos, specific_end, ++ &value)); ++ charging_unit->specific.billing_id = value; ++ charging_unit->specific.billing_id_present = 1; ++ } else { ++ charging_unit->specific.billing_id_present = 0; ++ } ++ ++ ASN1_END_FIXUP(ctrl, pos, specific_offset, specific_end, seq_end); ++ break; ++ default: ++ ASN1_DID_NOT_EXPECT_TAG(ctrl, tag); ++ return NULL; ++ } ++ ++ if (pos < seq_end && *pos != ASN1_INDEF_TERM) { ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CALL(pos, rose_dec_etsi_AOC_ChargingAssociation(ctrl, "chargingAssociation", ++ tag, pos, seq_end, &charging_unit->charging_association)); ++ charging_unit->charging_association_present = 1; ++ } else { ++ charging_unit->charging_association_present = 0; ++ } ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the ChargingRequest invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_etsi_ChargingRequest_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, union rose_msg_invoke_args *args) ++{ ++ int32_t value; ++ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_ENUMERATED); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "chargingCase", tag, pos, end, &value)); ++ args->etsi.ChargingRequest.charging_case = value; ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the ChargingRequest result parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_etsi_ChargingRequest_RES(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, union rose_msg_result_args *args) ++{ ++ struct roseEtsiChargingRequest_RES *charging_request; ++ int32_t value; ++ ++ charging_request = &args->etsi.ChargingRequest; ++ switch (tag) { ++ case ASN1_TAG_SEQUENCE: ++ charging_request->type = 0; /* currency_info_list */ ++ ASN1_CALL(pos, rose_dec_etsi_AOCSCurrencyInfoList(ctrl, "currencyList", tag, pos, ++ end, &charging_request->u.currency_info)); ++ break; ++ case ASN1_TYPE_INTEGER: ++ charging_request->type = 1; /* special_arrangement_info */ ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "specialArrangement", tag, pos, end, &value)); ++ charging_request->u.special_arrangement = value; ++ break; ++ case ASN1_TYPE_NULL: ++ charging_request->type = 2; /* charging_info_follows */ ++ ASN1_CALL(pos, asn1_dec_null(ctrl, "chargingInfoFollows", tag, pos, end)); ++ break; ++ default: ++ ASN1_DID_NOT_EXPECT_TAG(ctrl, tag); ++ return NULL; ++ } ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the AOCSCurrency invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_etsi_AOCSCurrency_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, union rose_msg_invoke_args *args) ++{ ++ struct roseEtsiAOCSCurrency_ARG *aoc_s; ++ ++ aoc_s = &args->etsi.AOCSCurrency; ++ switch (tag) { ++ case ASN1_TYPE_NULL: ++ aoc_s->type = 0; /* charge_not_available */ ++ ASN1_CALL(pos, asn1_dec_null(ctrl, "chargeNotAvailable", tag, pos, end)); ++ break; ++ case ASN1_TAG_SEQUENCE: ++ aoc_s->type = 1; /* currency_info_list */ ++ ASN1_CALL(pos, rose_dec_etsi_AOCSCurrencyInfoList(ctrl, "currencyInfo", tag, pos, ++ end, &aoc_s->currency_info)); ++ break; ++ default: ++ ASN1_DID_NOT_EXPECT_TAG(ctrl, tag); ++ return NULL; ++ } ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the AOCSSpecialArr invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_etsi_AOCSSpecialArr_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, union rose_msg_invoke_args *args) ++{ ++ struct roseEtsiAOCSSpecialArr_ARG *aoc_s; ++ int32_t value; ++ ++ aoc_s = &args->etsi.AOCSSpecialArr; ++ switch (tag) { ++ case ASN1_TYPE_NULL: ++ aoc_s->type = 0; /* charge_not_available */ ++ ASN1_CALL(pos, asn1_dec_null(ctrl, "chargeNotAvailable", tag, pos, end)); ++ break; ++ case ASN1_TYPE_INTEGER: ++ aoc_s->type = 1; /* special_arrangement_info */ ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "specialArrangement", tag, pos, end, &value)); ++ aoc_s->special_arrangement = value; ++ break; ++ default: ++ ASN1_DID_NOT_EXPECT_TAG(ctrl, tag); ++ return NULL; ++ } ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the AOCDCurrency invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_etsi_AOCDCurrency_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, union rose_msg_invoke_args *args) ++{ ++ struct roseEtsiAOCDCurrency_ARG *aoc_d; ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ int32_t value; ++ ++ aoc_d = &args->etsi.AOCDCurrency; ++ switch (tag) { ++ case ASN1_TYPE_NULL: ++ aoc_d->type = 0; /* charge_not_available */ ++ ASN1_CALL(pos, asn1_dec_null(ctrl, "chargeNotAvailable", tag, pos, end)); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 1: ++ aoc_d->type = 1; /* free_of_charge */ ++ ASN1_CALL(pos, asn1_dec_null(ctrl, "freeOfCharge", tag, pos, end)); ++ break; ++ case ASN1_TAG_SEQUENCE: ++ aoc_d->type = 2; /* specific_currency */ ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " specificCurrency %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ++ ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 1); ++ ASN1_CALL(pos, rose_dec_etsi_AOC_RecordedCurrency(ctrl, "recordedCurrency", tag, ++ pos, seq_end, &aoc_d->specific.recorded)); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_CLASS_CONTEXT_SPECIFIC | 2); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "typeOfChargingInfo", tag, pos, seq_end, ++ &value)); ++ aoc_d->specific.type_of_charging_info = value; ++ ++ if (pos < seq_end && *pos != ASN1_INDEF_TERM) { ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_CLASS_CONTEXT_SPECIFIC | 3); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "billingId", tag, pos, seq_end, &value)); ++ aoc_d->specific.billing_id = value; ++ aoc_d->specific.billing_id_present = 1; ++ } else { ++ aoc_d->specific.billing_id_present = 0; ++ } ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ break; ++ default: ++ ASN1_DID_NOT_EXPECT_TAG(ctrl, tag); ++ return NULL; ++ } ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the AOCDChargingUnit invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_etsi_AOCDChargingUnit_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, union rose_msg_invoke_args *args) ++{ ++ struct roseEtsiAOCDChargingUnit_ARG *aoc_d; ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ int32_t value; ++ ++ aoc_d = &args->etsi.AOCDChargingUnit; ++ switch (tag) { ++ case ASN1_TYPE_NULL: ++ aoc_d->type = 0; /* charge_not_available */ ++ ASN1_CALL(pos, asn1_dec_null(ctrl, "chargeNotAvailable", tag, pos, end)); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 1: ++ aoc_d->type = 1; /* free_of_charge */ ++ ASN1_CALL(pos, asn1_dec_null(ctrl, "freeOfCharge", tag, pos, end)); ++ break; ++ case ASN1_TAG_SEQUENCE: ++ aoc_d->type = 2; /* specific_charging_units */ ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " specificChargingUnits %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ++ ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 1); ++ ASN1_CALL(pos, rose_dec_etsi_AOC_RecordedUnitsList(ctrl, "recordedUnitsList", ++ tag, pos, seq_end, &aoc_d->specific.recorded)); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_CLASS_CONTEXT_SPECIFIC | 2); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "typeOfChargingInfo", tag, pos, seq_end, ++ &value)); ++ aoc_d->specific.type_of_charging_info = value; ++ ++ if (pos < seq_end && *pos != ASN1_INDEF_TERM) { ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_CLASS_CONTEXT_SPECIFIC | 3); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "billingId", tag, pos, seq_end, &value)); ++ aoc_d->specific.billing_id = value; ++ aoc_d->specific.billing_id_present = 1; ++ } else { ++ aoc_d->specific.billing_id_present = 0; ++ } ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ break; ++ default: ++ ASN1_DID_NOT_EXPECT_TAG(ctrl, tag); ++ return NULL; ++ } ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the AOCECurrency invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_etsi_AOCECurrency_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, union rose_msg_invoke_args *args) ++{ ++ struct roseEtsiAOCECurrency_ARG *aoc_e; ++ ++ aoc_e = &args->etsi.AOCECurrency; ++ switch (tag) { ++ case ASN1_TYPE_NULL: ++ aoc_e->type = 0; /* charge_not_available */ ++ ASN1_CALL(pos, asn1_dec_null(ctrl, "chargeNotAvailable", tag, pos, end)); ++ break; ++ case ASN1_TAG_SEQUENCE: ++ aoc_e->type = 1; /* currency_info */ ++ ASN1_CALL(pos, rose_dec_etsi_AOCECurrencyInfo(ctrl, "currencyInfo", tag, pos, ++ end, &aoc_e->currency_info)); ++ break; ++ default: ++ ASN1_DID_NOT_EXPECT_TAG(ctrl, tag); ++ return NULL; ++ } ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the AOCEChargingUnit invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_etsi_AOCEChargingUnit_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, union rose_msg_invoke_args *args) ++{ ++ struct roseEtsiAOCEChargingUnit_ARG *aoc_e; ++ ++ aoc_e = &args->etsi.AOCEChargingUnit; ++ switch (tag) { ++ case ASN1_TYPE_NULL: ++ aoc_e->type = 0; /* charge_not_available */ ++ ASN1_CALL(pos, asn1_dec_null(ctrl, "chargeNotAvailable", tag, pos, end)); ++ break; ++ case ASN1_TAG_SEQUENCE: ++ aoc_e->type = 1; /* charging_unit */ ++ ASN1_CALL(pos, rose_dec_etsi_AOCEChargingUnitInfo(ctrl, "chargingUnitInfo", tag, ++ pos, end, &aoc_e->charging_unit)); ++ break; ++ default: ++ ASN1_DID_NOT_EXPECT_TAG(ctrl, tag); ++ return NULL; ++ } ++ ++ return pos; ++} ++ ++/* ------------------------------------------------------------------- */ ++/* end rose_etsi_aoc.c */ + +Property changes on: rose_etsi_aoc.c +___________________________________________________________________ +Added: svn:eol-style + + native +Added: svn:mime-type + + text/plain +Added: svn:keywords + + 'Author Date Id Revision' + +Index: rose_etsi_diversion.c +=================================================================== +--- a/rose_etsi_diversion.c (.../tags/1.4.10.2) (revision 0) ++++ b/rose_etsi_diversion.c (.../branches/1.4) (revision 1357) +@@ -0,0 +1,1623 @@ ++/* ++ * libpri: An implementation of Primary Rate ISDN ++ * ++ * Copyright (C) 2009 Digium, Inc. ++ * ++ * Richard Mudgett <rmudgett@digium.com> ++ * ++ * See http://www.asterisk.org for more information about ++ * the Asterisk project. Please do not directly contact ++ * any of the maintainers of this project for assistance; ++ * the project provides a web site, mailing lists and IRC ++ * channels for your use. ++ * ++ * This program is free software, distributed under the terms of ++ * the GNU General Public License Version 2 as published by the ++ * Free Software Foundation. See the LICENSE file included with ++ * this program for more details. ++ * ++ * In addition, when this program is distributed with Asterisk in ++ * any form that would qualify as a 'combined work' or as a ++ * 'derivative work' (but not mere aggregation), you can redistribute ++ * and/or modify the combination under the terms of the license ++ * provided with that copy of Asterisk, instead of the license ++ * terms granted here. ++ */ ++ ++/*! ++ * \file ++ * \brief ROSE Call diversion operations ++ * ++ * Diversion Supplementary Services ETS 300 207-1 Table 3 ++ * ++ * \author Richard Mudgett <rmudgett@digium.com> ++ */ ++ ++ ++#include "compat.h" ++#include "libpri.h" ++#include "pri_internal.h" ++#include "rose.h" ++#include "rose_internal.h" ++#include "asn1.h" ++ ++ ++/* ------------------------------------------------------------------- */ ++ ++/*! ++ * \internal ++ * \brief Encode the ServedUserNr type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param served_user_number Served user number information to encode. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *rose_enc_etsi_ServedUserNumber(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, ++ const struct rosePartyNumber *served_user_number) ++{ ++ if (served_user_number->length) { ++ /* Forward this number */ ++ pos = rose_enc_PartyNumber(ctrl, pos, end, served_user_number); ++ } else { ++ /* Forward all numbers */ ++ pos = asn1_enc_null(pos, end, ASN1_TYPE_NULL); ++ } ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Encode the IntResult type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param tag Component tag to identify the encoded component. ++ * The tag should be ASN1_TAG_SEQUENCE unless the caller implicitly ++ * tags it otherwise. ++ * \param int_result Forwarding record information to encode. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *rose_enc_etsi_IntResult(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, unsigned tag, const struct roseEtsiForwardingRecord *int_result) ++{ ++ unsigned char *seq_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, tag); ++ ++ ASN1_CALL(pos, rose_enc_etsi_ServedUserNumber(ctrl, pos, end, ++ &int_result->served_user_number)); ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, ++ int_result->basic_service)); ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, int_result->procedure)); ++ ASN1_CALL(pos, rose_enc_Address(ctrl, pos, end, ASN1_TAG_SEQUENCE, ++ &int_result->forwarded_to)); ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Encode the IntResultList type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param tag Component tag to identify the encoded component. ++ * The tag should be ASN1_TAG_SET unless the caller implicitly ++ * tags it otherwise. ++ * \param int_result_list Forwarding record list information to encode. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *rose_enc_etsi_IntResultList(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, unsigned tag, ++ const struct roseEtsiForwardingList *int_result_list) ++{ ++ unsigned index; ++ unsigned char *set_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(set_len, pos, end, tag); ++ ++ for (index = 0; index < int_result_list->num_records; ++index) { ++ ASN1_CALL(pos, rose_enc_etsi_IntResult(ctrl, pos, end, ASN1_TAG_SEQUENCE, ++ &int_result_list->list[index])); ++ } ++ ++ ASN1_CONSTRUCTED_END(set_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Encode the ServedUserNumberList type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param tag Component tag to identify the encoded component. ++ * The tag should be ASN1_TAG_SET unless the caller implicitly ++ * tags it otherwise. ++ * \param served_user_number_list Served user record list information to encode. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *rose_enc_etsi_ServedUserNumberList(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, unsigned tag, ++ const struct roseEtsiServedUserNumberList *served_user_number_list) ++{ ++ unsigned index; ++ unsigned char *set_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(set_len, pos, end, tag); ++ ++ for (index = 0; index < served_user_number_list->num_records; ++index) { ++ ASN1_CALL(pos, rose_enc_PartyNumber(ctrl, pos, end, ++ &served_user_number_list->number[index])); ++ } ++ ++ ASN1_CONSTRUCTED_END(set_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the ActivationDiversion invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_etsi_ActivationDiversion_ARG(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ const struct roseEtsiActivationDiversion_ARG *activation_diversion; ++ unsigned char *seq_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ASN1_TAG_SEQUENCE); ++ ++ activation_diversion = &args->etsi.ActivationDiversion; ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, ++ activation_diversion->procedure)); ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, ++ activation_diversion->basic_service)); ++ ASN1_CALL(pos, rose_enc_Address(ctrl, pos, end, ASN1_TAG_SEQUENCE, ++ &activation_diversion->forwarded_to)); ++ ASN1_CALL(pos, rose_enc_etsi_ServedUserNumber(ctrl, pos, end, ++ &activation_diversion->served_user_number)); ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the DeactivationDiversion invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_etsi_DeactivationDiversion_ARG(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ const struct roseEtsiDeactivationDiversion_ARG *deactivation_diversion; ++ unsigned char *seq_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ASN1_TAG_SEQUENCE); ++ ++ deactivation_diversion = &args->etsi.DeactivationDiversion; ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, ++ deactivation_diversion->procedure)); ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, ++ deactivation_diversion->basic_service)); ++ ASN1_CALL(pos, rose_enc_etsi_ServedUserNumber(ctrl, pos, end, ++ &deactivation_diversion->served_user_number)); ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the ActivationStatusNotificationDiv invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_etsi_ActivationStatusNotificationDiv_ARG(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ const struct roseEtsiActivationStatusNotificationDiv_ARG ++ *activation_status_notification_div; ++ unsigned char *seq_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ASN1_TAG_SEQUENCE); ++ ++ activation_status_notification_div = &args->etsi.ActivationStatusNotificationDiv; ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, ++ activation_status_notification_div->procedure)); ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, ++ activation_status_notification_div->basic_service)); ++ ASN1_CALL(pos, rose_enc_Address(ctrl, pos, end, ASN1_TAG_SEQUENCE, ++ &activation_status_notification_div->forwarded_to)); ++ ASN1_CALL(pos, rose_enc_etsi_ServedUserNumber(ctrl, pos, end, ++ &activation_status_notification_div->served_user_number)); ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the DeactivationStatusNotificationDiv invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_etsi_DeactivationStatusNotificationDiv_ARG(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ const struct roseEtsiDeactivationStatusNotificationDiv_ARG ++ *deactivation_status_notification_div; ++ unsigned char *seq_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ASN1_TAG_SEQUENCE); ++ ++ deactivation_status_notification_div = &args->etsi.DeactivationStatusNotificationDiv; ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, ++ deactivation_status_notification_div->procedure)); ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, ++ deactivation_status_notification_div->basic_service)); ++ ASN1_CALL(pos, rose_enc_etsi_ServedUserNumber(ctrl, pos, end, ++ &deactivation_status_notification_div->served_user_number)); ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the InterrogationDiversion invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_etsi_InterrogationDiversion_ARG(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ const struct roseEtsiInterrogationDiversion_ARG *interrogation_diversion; ++ unsigned char *seq_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ASN1_TAG_SEQUENCE); ++ ++ interrogation_diversion = &args->etsi.InterrogationDiversion; ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, ++ interrogation_diversion->procedure)); ++ if (interrogation_diversion->basic_service) { ++ /* Not the DEFAULT value */ ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, ++ interrogation_diversion->basic_service)); ++ } ++ ASN1_CALL(pos, rose_enc_etsi_ServedUserNumber(ctrl, pos, end, ++ &interrogation_diversion->served_user_number)); ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the InterrogationDiversion result facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_etsi_InterrogationDiversion_RES(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, const union rose_msg_result_args *args) ++{ ++ return rose_enc_etsi_IntResultList(ctrl, pos, end, ASN1_TAG_SET, ++ &args->etsi.InterrogationDiversion); ++} ++ ++/*! ++ * \brief Encode the DiversionInformation invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_etsi_DiversionInformation_ARG(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ const struct roseEtsiDiversionInformation_ARG *diversion_information; ++ unsigned char *seq_len; ++ unsigned char *explicit_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ASN1_TAG_SEQUENCE); ++ ++ diversion_information = &args->etsi.DiversionInformation; ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, ++ diversion_information->diversion_reason)); ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, ++ diversion_information->basic_service)); ++ if (diversion_information->served_user_subaddress.length) { ++ ASN1_CALL(pos, rose_enc_PartySubaddress(ctrl, pos, end, ++ &diversion_information->served_user_subaddress)); ++ } ++ if (diversion_information->calling_present) { ++ /* EXPLICIT tag */ ++ ASN1_CONSTRUCTED_BEGIN(explicit_len, pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 0); ++ ASN1_CALL(pos, rose_enc_PresentedAddressScreened(ctrl, pos, end, ++ &diversion_information->calling)); ++ ASN1_CONSTRUCTED_END(explicit_len, pos, end); ++ } ++ if (diversion_information->original_called_present) { ++ /* EXPLICIT tag */ ++ ASN1_CONSTRUCTED_BEGIN(explicit_len, pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 1); ++ ASN1_CALL(pos, rose_enc_PresentedNumberUnscreened(ctrl, pos, end, ++ &diversion_information->original_called)); ++ ASN1_CONSTRUCTED_END(explicit_len, pos, end); ++ } ++ if (diversion_information->last_diverting_present) { ++ /* EXPLICIT tag */ ++ ASN1_CONSTRUCTED_BEGIN(explicit_len, pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 2); ++ ASN1_CALL(pos, rose_enc_PresentedNumberUnscreened(ctrl, pos, end, ++ &diversion_information->last_diverting)); ++ ASN1_CONSTRUCTED_END(explicit_len, pos, end); ++ } ++ if (diversion_information->last_diverting_reason_present) { ++ /* EXPLICIT tag */ ++ ASN1_CONSTRUCTED_BEGIN(explicit_len, pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 3); ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, ++ diversion_information->last_diverting_reason)); ++ ASN1_CONSTRUCTED_END(explicit_len, pos, end); ++ } ++ if (diversion_information->q931ie.length) { ++ ASN1_CALL(pos, rose_enc_Q931ie(ctrl, pos, end, ASN1_CLASS_APPLICATION | 0, ++ &diversion_information->q931ie)); ++ } ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the CallDeflection invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_etsi_CallDeflection_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ const struct roseEtsiCallDeflection_ARG *call_deflection; ++ unsigned char *seq_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ASN1_TAG_SEQUENCE); ++ ++ call_deflection = &args->etsi.CallDeflection; ++ ASN1_CALL(pos, rose_enc_Address(ctrl, pos, end, ASN1_TAG_SEQUENCE, ++ &call_deflection->deflection)); ++ if (call_deflection->presentation_allowed_to_diverted_to_user_present) { ++ ASN1_CALL(pos, asn1_enc_boolean(pos, end, ASN1_TYPE_BOOLEAN, ++ call_deflection->presentation_allowed_to_diverted_to_user)); ++ } ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the CallRerouting invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_etsi_CallRerouting_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ const struct roseEtsiCallRerouting_ARG *call_rerouting; ++ unsigned char *seq_len; ++ unsigned char *explicit_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ASN1_TAG_SEQUENCE); ++ ++ call_rerouting = &args->etsi.CallRerouting; ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, ++ call_rerouting->rerouting_reason)); ++ ASN1_CALL(pos, rose_enc_Address(ctrl, pos, end, ASN1_TAG_SEQUENCE, ++ &call_rerouting->called_address)); ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_INTEGER, ++ call_rerouting->rerouting_counter)); ++ ASN1_CALL(pos, rose_enc_Q931ie(ctrl, pos, end, ASN1_CLASS_APPLICATION | 0, ++ &call_rerouting->q931ie)); ++ ++ /* EXPLICIT tag */ ++ ASN1_CONSTRUCTED_BEGIN(explicit_len, pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 1); ++ ASN1_CALL(pos, rose_enc_PresentedNumberUnscreened(ctrl, pos, end, ++ &call_rerouting->last_rerouting)); ++ ASN1_CONSTRUCTED_END(explicit_len, pos, end); ++ ++ if (call_rerouting->subscription_option) { ++ /* Not the DEFAULT value */ ++ /* EXPLICIT tag */ ++ ASN1_CONSTRUCTED_BEGIN(explicit_len, pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 2); ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, ++ call_rerouting->subscription_option)); ++ ASN1_CONSTRUCTED_END(explicit_len, pos, end); ++ } ++ ++ if (call_rerouting->calling_subaddress.length) { ++ /* EXPLICIT tag */ ++ ASN1_CONSTRUCTED_BEGIN(explicit_len, pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 3); ++ ASN1_CALL(pos, rose_enc_PartySubaddress(ctrl, pos, end, ++ &call_rerouting->calling_subaddress)); ++ ASN1_CONSTRUCTED_END(explicit_len, pos, end); ++ } ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the InterrogateServedUserNumbers result facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_etsi_InterrogateServedUserNumbers_RES(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, const union rose_msg_result_args *args) ++{ ++ return rose_enc_etsi_ServedUserNumberList(ctrl, pos, end, ASN1_TAG_SET, ++ &args->etsi.InterrogateServedUserNumbers); ++} ++ ++/*! ++ * \brief Encode the DivertingLegInformation1 invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_etsi_DivertingLegInformation1_ARG(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ const struct roseEtsiDivertingLegInformation1_ARG *diverting_leg_information_1; ++ unsigned char *seq_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ASN1_TAG_SEQUENCE); ++ ++ diverting_leg_information_1 = &args->etsi.DivertingLegInformation1; ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, ++ diverting_leg_information_1->diversion_reason)); ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, ++ diverting_leg_information_1->subscription_option)); ++ if (diverting_leg_information_1->diverted_to_present) { ++ ASN1_CALL(pos, rose_enc_PresentedNumberUnscreened(ctrl, pos, end, ++ &diverting_leg_information_1->diverted_to)); ++ } ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the DivertingLegInformation2 invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_etsi_DivertingLegInformation2_ARG(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ const struct roseEtsiDivertingLegInformation2_ARG *diverting_leg_information_2; ++ unsigned char *seq_len; ++ unsigned char *explicit_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ASN1_TAG_SEQUENCE); ++ ++ diverting_leg_information_2 = &args->etsi.DivertingLegInformation2; ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_INTEGER, ++ diverting_leg_information_2->diversion_counter)); ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, ++ diverting_leg_information_2->diversion_reason)); ++ ++ if (diverting_leg_information_2->diverting_present) { ++ /* EXPLICIT tag */ ++ ASN1_CONSTRUCTED_BEGIN(explicit_len, pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 1); ++ ASN1_CALL(pos, rose_enc_PresentedNumberUnscreened(ctrl, pos, end, ++ &diverting_leg_information_2->diverting)); ++ ASN1_CONSTRUCTED_END(explicit_len, pos, end); ++ } ++ ++ if (diverting_leg_information_2->original_called_present) { ++ /* EXPLICIT tag */ ++ ASN1_CONSTRUCTED_BEGIN(explicit_len, pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 2); ++ ASN1_CALL(pos, rose_enc_PresentedNumberUnscreened(ctrl, pos, end, ++ &diverting_leg_information_2->original_called)); ++ ASN1_CONSTRUCTED_END(explicit_len, pos, end); ++ } ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the DivertingLegInformation3 invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_etsi_DivertingLegInformation3_ARG(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ return asn1_enc_boolean(pos, end, ASN1_TYPE_BOOLEAN, ++ args->etsi.DivertingLegInformation3.presentation_allowed_indicator); ++} ++ ++/*! ++ * \internal ++ * \brief Decode the ServedUserNr argument parameters. ++ * ++ * \param ctrl D channel controller for any diagnostic messages. ++ * \param name Field name ++ * \param tag Component tag that identified this production. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param served_user_number Parameter storage to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++static const unsigned char *rose_dec_etsi_ServedUserNumber(struct pri *ctrl, ++ const char *name, unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct rosePartyNumber *served_user_number) ++{ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s ServedUserNumber\n", name); ++ } ++ if (tag == ASN1_TYPE_NULL) { ++ served_user_number->length = 0; ++ pos = asn1_dec_null(ctrl, "allNumbers", tag, pos, end); ++ } else { ++ /* Must be a PartyNumber (Which is itself a CHOICE) */ ++ pos = ++ rose_dec_PartyNumber(ctrl, "individualNumber", tag, pos, end, ++ served_user_number); ++ } ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Decode the IntResult argument parameters. ++ * ++ * \param ctrl D channel controller for any diagnostic messages. ++ * \param name Field name ++ * \param tag Component tag that identified this production. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param int_result Parameter storage to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++static const unsigned char *rose_dec_etsi_IntResult(struct pri *ctrl, const char *name, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct roseEtsiForwardingRecord *int_result) ++{ ++ int32_t value; ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s IntResult %s\n", name, asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CALL(pos, rose_dec_etsi_ServedUserNumber(ctrl, "servedUserNr", tag, pos, ++ seq_end, &int_result->served_user_number)); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_ENUMERATED); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "basicService", tag, pos, seq_end, &value)); ++ int_result->basic_service = value; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_ENUMERATED); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "procedure", tag, pos, seq_end, &value)); ++ int_result->procedure = value; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ ASN1_CALL(pos, rose_dec_Address(ctrl, "forwardedToAddress", tag, pos, seq_end, ++ &int_result->forwarded_to)); ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Decode the IntResultList argument parameters. ++ * ++ * \param ctrl D channel controller for any diagnostic messages. ++ * \param name Field name ++ * \param tag Component tag that identified this production. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param int_result_list Parameter storage to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++static const unsigned char *rose_dec_etsi_IntResultList(struct pri *ctrl, ++ const char *name, unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct roseEtsiForwardingList *int_result_list) ++{ ++ int length; ++ int set_offset; ++ const unsigned char *set_end; ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s IntResultList %s\n", name, asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(set_end, set_offset, length, pos, end); ++ ++ int_result_list->num_records = 0; ++ while (pos < set_end && *pos != ASN1_INDEF_TERM) { ++ if (int_result_list->num_records < ARRAY_LEN(int_result_list->list)) { ++ ASN1_CALL(pos, asn1_dec_tag(pos, set_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ ASN1_CALL(pos, rose_dec_etsi_IntResult(ctrl, "listEntry", tag, pos, set_end, ++ &int_result_list->list[int_result_list->num_records])); ++ ++int_result_list->num_records; ++ } else { ++ /* Too many records */ ++ return NULL; ++ } ++ } ++ ++ ASN1_END_FIXUP(ctrl, pos, set_offset, set_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Decode the ServedUserNumberList argument parameters. ++ * ++ * \param ctrl D channel controller for any diagnostic messages. ++ * \param name Field name ++ * \param tag Component tag that identified this production. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param served_user_number_list Parameter storage to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++static const unsigned char *rose_dec_etsi_ServedUserNumberList(struct pri *ctrl, ++ const char *name, unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct roseEtsiServedUserNumberList *served_user_number_list) ++{ ++ int length; ++ int set_offset; ++ const unsigned char *set_end; ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s ServedUserNumberList %s\n", name, asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(set_end, set_offset, length, pos, end); ++ ++ served_user_number_list->num_records = 0; ++ while (pos < set_end && *pos != ASN1_INDEF_TERM) { ++ if (served_user_number_list->num_records < ++ ARRAY_LEN(served_user_number_list->number)) { ++ ASN1_CALL(pos, asn1_dec_tag(pos, set_end, &tag)); ++ ASN1_CALL(pos, rose_dec_PartyNumber(ctrl, "listEntry", tag, pos, set_end, ++ &served_user_number_list->number[served_user_number_list->num_records])); ++ ++served_user_number_list->num_records; ++ } else { ++ /* Too many records */ ++ return NULL; ++ } ++ } ++ ++ ASN1_END_FIXUP(ctrl, pos, set_offset, set_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the ActivationDiversion invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_etsi_ActivationDiversion_ARG(struct pri *ctrl, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args) ++{ ++ struct roseEtsiActivationDiversion_ARG *activation_diversion; ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ int32_t value; ++ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " ActivationDiversion %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ activation_diversion = &args->etsi.ActivationDiversion; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_ENUMERATED); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "procedure", tag, pos, seq_end, &value)); ++ activation_diversion->procedure = value; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_ENUMERATED); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "basicService", tag, pos, seq_end, &value)); ++ activation_diversion->basic_service = value; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ ASN1_CALL(pos, rose_dec_Address(ctrl, "forwardedToAddress", tag, pos, seq_end, ++ &activation_diversion->forwarded_to)); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CALL(pos, rose_dec_etsi_ServedUserNumber(ctrl, "servedUserNr", tag, pos, ++ seq_end, &activation_diversion->served_user_number)); ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the DeactivationDiversion invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_etsi_DeactivationDiversion_ARG(struct pri *ctrl, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args) ++{ ++ struct roseEtsiDeactivationDiversion_ARG *deactivation_diversion; ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ int32_t value; ++ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " DeactivationDiversion %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ deactivation_diversion = &args->etsi.DeactivationDiversion; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_ENUMERATED); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "procedure", tag, pos, seq_end, &value)); ++ deactivation_diversion->procedure = value; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_ENUMERATED); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "basicService", tag, pos, seq_end, &value)); ++ deactivation_diversion->basic_service = value; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CALL(pos, rose_dec_etsi_ServedUserNumber(ctrl, "servedUserNr", tag, pos, ++ seq_end, &deactivation_diversion->served_user_number)); ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the ActivationStatusNotificationDiv invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_etsi_ActivationStatusNotificationDiv_ARG(struct pri *ctrl, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args) ++{ ++ struct roseEtsiActivationStatusNotificationDiv_ARG ++ *activation_status_notification_div; ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ int32_t value; ++ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " ActivationStatusNotificationDiv %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ activation_status_notification_div = &args->etsi.ActivationStatusNotificationDiv; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_ENUMERATED); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "procedure", tag, pos, seq_end, &value)); ++ activation_status_notification_div->procedure = value; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_ENUMERATED); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "basicService", tag, pos, seq_end, &value)); ++ activation_status_notification_div->basic_service = value; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ ASN1_CALL(pos, rose_dec_Address(ctrl, "forwardedToAddress", tag, pos, seq_end, ++ &activation_status_notification_div->forwarded_to)); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CALL(pos, rose_dec_etsi_ServedUserNumber(ctrl, "servedUserNr", tag, pos, ++ seq_end, &activation_status_notification_div->served_user_number)); ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the DeactivationStatusNotificationDiv invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_etsi_DeactivationStatusNotificationDiv_ARG(struct pri ++ *ctrl, unsigned tag, const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args) ++{ ++ struct roseEtsiDeactivationStatusNotificationDiv_ARG ++ *deactivation_status_notification_div; ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ int32_t value; ++ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " DeactivationStatusNotificationDiv %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ deactivation_status_notification_div = &args->etsi.DeactivationStatusNotificationDiv; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_ENUMERATED); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "procedure", tag, pos, seq_end, &value)); ++ deactivation_status_notification_div->procedure = value; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_ENUMERATED); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "basicService", tag, pos, seq_end, &value)); ++ deactivation_status_notification_div->basic_service = value; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CALL(pos, rose_dec_etsi_ServedUserNumber(ctrl, "forwardedToAddress", tag, pos, ++ seq_end, &deactivation_status_notification_div->served_user_number)); ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the InterrogationDiversion invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_etsi_InterrogationDiversion_ARG(struct pri *ctrl, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args) ++{ ++ struct roseEtsiInterrogationDiversion_ARG *interrogation_diversion; ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ int32_t value; ++ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " InterrogationDiversion %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ interrogation_diversion = &args->etsi.InterrogationDiversion; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_ENUMERATED); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "procedure", tag, pos, seq_end, &value)); ++ interrogation_diversion->procedure = value; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ if (tag == ASN1_TYPE_ENUMERATED) { ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "basicService", tag, pos, seq_end, &value)); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ } else { ++ value = 0; /* DEFAULT BasicService value (allServices) */ ++ } ++ interrogation_diversion->basic_service = value; ++ ++ ASN1_CALL(pos, rose_dec_etsi_ServedUserNumber(ctrl, "servedUserNr", tag, pos, ++ seq_end, &interrogation_diversion->served_user_number)); ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the InterrogationDiversion result parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_etsi_InterrogationDiversion_RES(struct pri *ctrl, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ union rose_msg_result_args *args) ++{ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SET); ++ return rose_dec_etsi_IntResultList(ctrl, "diversionList", tag, pos, end, ++ &args->etsi.InterrogationDiversion); ++} ++ ++/*! ++ * \brief Decode the DiversionInformation invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_etsi_DiversionInformation_ARG(struct pri *ctrl, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args) ++{ ++ struct roseEtsiDiversionInformation_ARG *diversion_information; ++ int length; ++ int seq_offset; ++ int explicit_offset; ++ const unsigned char *seq_end; ++ const unsigned char *explicit_end; ++ const unsigned char *save_pos; ++ int32_t value; ++ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " DiversionInformation %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ diversion_information = &args->etsi.DiversionInformation; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_ENUMERATED); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "diversionReason", tag, pos, seq_end, &value)); ++ diversion_information->diversion_reason = value; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_ENUMERATED); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "basicService", tag, pos, seq_end, &value)); ++ diversion_information->basic_service = value; ++ ++ /* ++ * A sequence specifies an ordered list of component types. ++ * However, for simplicity we are not checking the order of ++ * the remaining optional components. ++ */ ++ diversion_information->served_user_subaddress.length = 0; ++ diversion_information->calling_present = 0; ++ diversion_information->original_called_present = 0; ++ diversion_information->last_diverting_present = 0; ++ diversion_information->last_diverting_reason_present = 0; ++ diversion_information->q931ie.length = 0; ++ while (pos < seq_end && *pos != ASN1_INDEF_TERM) { ++ save_pos = pos; ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ switch (tag) { ++ case ASN1_TAG_SEQUENCE: ++ case ASN1_TYPE_OCTET_STRING: ++ case ASN1_TYPE_OCTET_STRING | ASN1_PC_CONSTRUCTED: ++ ASN1_CALL(pos, rose_dec_PartySubaddress(ctrl, "servedUserSubaddress", tag, ++ pos, seq_end, &diversion_information->served_user_subaddress)); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 0: ++ /* Remove EXPLICIT tag */ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " Explicit %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, seq_end, &length)); ++ ASN1_END_SETUP(explicit_end, explicit_offset, length, pos, seq_end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, explicit_end, &tag)); ++ ASN1_CALL(pos, rose_dec_PresentedAddressScreened(ctrl, "callingAddress", tag, ++ pos, explicit_end, &diversion_information->calling)); ++ diversion_information->calling_present = 1; ++ ++ ASN1_END_FIXUP(ctrl, pos, explicit_offset, explicit_end, seq_end); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 1: ++ /* Remove EXPLICIT tag */ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " Explicit %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, seq_end, &length)); ++ ASN1_END_SETUP(explicit_end, explicit_offset, length, pos, seq_end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, explicit_end, &tag)); ++ ASN1_CALL(pos, rose_dec_PresentedNumberUnscreened(ctrl, "originalCalledNr", ++ tag, pos, explicit_end, &diversion_information->original_called)); ++ diversion_information->original_called_present = 1; ++ ++ ASN1_END_FIXUP(ctrl, pos, explicit_offset, explicit_end, seq_end); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 2: ++ /* Remove EXPLICIT tag */ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " Explicit %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, seq_end, &length)); ++ ASN1_END_SETUP(explicit_end, explicit_offset, length, pos, seq_end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, explicit_end, &tag)); ++ ASN1_CALL(pos, rose_dec_PresentedNumberUnscreened(ctrl, "lastDivertingNr", ++ tag, pos, explicit_end, &diversion_information->last_diverting)); ++ diversion_information->last_diverting_present = 1; ++ ++ ASN1_END_FIXUP(ctrl, pos, explicit_offset, explicit_end, seq_end); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 3: ++ /* Remove EXPLICIT tag */ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " Explicit %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, seq_end, &length)); ++ ASN1_END_SETUP(explicit_end, explicit_offset, length, pos, seq_end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, explicit_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_ENUMERATED); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "lastDivertingReason", tag, pos, ++ explicit_end, &value)); ++ diversion_information->last_diverting_reason = value; ++ diversion_information->last_diverting_reason_present = 1; ++ ++ ASN1_END_FIXUP(ctrl, pos, explicit_offset, explicit_end, seq_end); ++ break; ++ case ASN1_CLASS_APPLICATION | 0: ++ case ASN1_CLASS_APPLICATION | ASN1_PC_CONSTRUCTED | 0: ++ ASN1_CALL(pos, rose_dec_Q931ie(ctrl, "userInfo", tag, pos, seq_end, ++ &diversion_information->q931ie, ++ sizeof(diversion_information->q931ie_contents))); ++ break; ++ default: ++ pos = save_pos; ++ goto cancel_options; ++ } ++ } ++cancel_options:; ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the CallDeflection invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_etsi_CallDeflection_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, union rose_msg_invoke_args *args) ++{ ++ struct roseEtsiCallDeflection_ARG *call_deflection; ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ int32_t value; ++ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " CallDeflection %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ call_deflection = &args->etsi.CallDeflection; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ ASN1_CALL(pos, rose_dec_Address(ctrl, "deflectionAddress", tag, pos, seq_end, ++ &call_deflection->deflection)); ++ ++ if (pos < seq_end && *pos != ASN1_INDEF_TERM) { ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_BOOLEAN); ++ ASN1_CALL(pos, asn1_dec_boolean(ctrl, "presentationAllowedDivertedToUser", tag, ++ pos, seq_end, &value)); ++ call_deflection->presentation_allowed_to_diverted_to_user = value; ++ call_deflection->presentation_allowed_to_diverted_to_user_present = 1; ++ } else { ++ call_deflection->presentation_allowed_to_diverted_to_user_present = 0; ++ } ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the CallRerouting invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_etsi_CallRerouting_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, union rose_msg_invoke_args *args) ++{ ++ struct roseEtsiCallRerouting_ARG *call_rerouting; ++ int length; ++ int seq_offset; ++ int explicit_offset; ++ const unsigned char *seq_end; ++ const unsigned char *explicit_end; ++ const unsigned char *save_pos; ++ int32_t value; ++ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " CallRerouting %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ call_rerouting = &args->etsi.CallRerouting; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_ENUMERATED); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "reroutingReason", tag, pos, seq_end, &value)); ++ call_rerouting->rerouting_reason = value; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ ASN1_CALL(pos, rose_dec_Address(ctrl, "calledAddress", tag, pos, seq_end, ++ &call_rerouting->called_address)); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_INTEGER); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "reroutingCounter", tag, pos, seq_end, &value)); ++ call_rerouting->rerouting_counter = value; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag & ~ASN1_PC_MASK, ASN1_CLASS_APPLICATION | 0); ++ ASN1_CALL(pos, rose_dec_Q931ie(ctrl, "q931ie", tag, pos, seq_end, ++ &call_rerouting->q931ie, sizeof(call_rerouting->q931ie_contents))); ++ ++ /* Remove EXPLICIT tag */ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ++ ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 1); ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " Explicit %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, seq_end, &length)); ++ ASN1_END_SETUP(explicit_end, explicit_offset, length, pos, seq_end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, explicit_end, &tag)); ++ ASN1_CALL(pos, rose_dec_PresentedNumberUnscreened(ctrl, "lastReroutingNr", tag, pos, ++ explicit_end, &call_rerouting->last_rerouting)); ++ ++ ASN1_END_FIXUP(ctrl, pos, explicit_offset, explicit_end, seq_end); ++ ++ /* ++ * A sequence specifies an ordered list of component types. ++ * However, for simplicity we are not checking the order of ++ * the remaining optional components. ++ */ ++ call_rerouting->subscription_option = 0; /* DEFAULT value noNotification */ ++ call_rerouting->calling_subaddress.length = 0; ++ while (pos < seq_end && *pos != ASN1_INDEF_TERM) { ++ save_pos = pos; ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ switch (tag) { ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 2: ++ /* Remove EXPLICIT tag */ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " Explicit %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, seq_end, &length)); ++ ASN1_END_SETUP(explicit_end, explicit_offset, length, pos, seq_end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, explicit_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_ENUMERATED); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "subscriptionOption", tag, pos, ++ explicit_end, &value)); ++ call_rerouting->subscription_option = value; ++ ++ ASN1_END_FIXUP(ctrl, pos, explicit_offset, explicit_end, seq_end); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 3: ++ /* Remove EXPLICIT tag */ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " Explicit %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, seq_end, &length)); ++ ASN1_END_SETUP(explicit_end, explicit_offset, length, pos, seq_end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, explicit_end, &tag)); ++ ASN1_CALL(pos, rose_dec_PartySubaddress(ctrl, "callingPartySubaddress", tag, ++ pos, explicit_end, &call_rerouting->calling_subaddress)); ++ ++ ASN1_END_FIXUP(ctrl, pos, explicit_offset, explicit_end, seq_end); ++ break; ++ default: ++ pos = save_pos; ++ goto cancel_options; ++ } ++ } ++cancel_options:; ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the InterrogateServedUserNumbers result parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_etsi_InterrogateServedUserNumbers_RES(struct pri *ctrl, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ union rose_msg_result_args *args) ++{ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SET); ++ return rose_dec_etsi_ServedUserNumberList(ctrl, "interrogateServedUserNumbers", tag, ++ pos, end, &args->etsi.InterrogateServedUserNumbers); ++} ++ ++/*! ++ * \brief Decode the DivertingLegInformation1 invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_etsi_DivertingLegInformation1_ARG(struct pri *ctrl, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args) ++{ ++ struct roseEtsiDivertingLegInformation1_ARG *diverting_leg_informtion_1; ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ int32_t value; ++ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " DivertingLegInformation1 %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ diverting_leg_informtion_1 = &args->etsi.DivertingLegInformation1; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_ENUMERATED); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "diversionReason", tag, pos, seq_end, &value)); ++ diverting_leg_informtion_1->diversion_reason = value; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_ENUMERATED); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "subscriptionOption", tag, pos, seq_end, &value)); ++ diverting_leg_informtion_1->subscription_option = value; ++ ++ if (pos < seq_end && *pos != ASN1_INDEF_TERM) { ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CALL(pos, rose_dec_PresentedNumberUnscreened(ctrl, "divertedToNumber", tag, ++ pos, seq_end, &diverting_leg_informtion_1->diverted_to)); ++ diverting_leg_informtion_1->diverted_to_present = 1; ++ } else { ++ diverting_leg_informtion_1->diverted_to_present = 0; ++ } ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the DivertingLegInformation2 invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_etsi_DivertingLegInformation2_ARG(struct pri *ctrl, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args) ++{ ++ struct roseEtsiDivertingLegInformation2_ARG *diverting_leg_information_2; ++ int length; ++ int seq_offset; ++ int explicit_offset; ++ const unsigned char *seq_end; ++ const unsigned char *explicit_end; ++ const unsigned char *save_pos; ++ int32_t value; ++ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " DivertingLegInformation2 %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ diverting_leg_information_2 = &args->etsi.DivertingLegInformation2; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_INTEGER); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "diversionCounter", tag, pos, seq_end, &value)); ++ diverting_leg_information_2->diversion_counter = value; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_ENUMERATED); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "diversionReason", tag, pos, seq_end, &value)); ++ diverting_leg_information_2->diversion_reason = value; ++ ++ /* ++ * A sequence specifies an ordered list of component types. ++ * However, for simplicity we are not checking the order of ++ * the remaining optional components. ++ */ ++ diverting_leg_information_2->diverting_present = 0; ++ diverting_leg_information_2->original_called_present = 0; ++ while (pos < seq_end && *pos != ASN1_INDEF_TERM) { ++ save_pos = pos; ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ switch (tag) { ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 1: ++ /* Remove EXPLICIT tag */ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " Explicit %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, seq_end, &length)); ++ ASN1_END_SETUP(explicit_end, explicit_offset, length, pos, seq_end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, explicit_end, &tag)); ++ ASN1_CALL(pos, rose_dec_PresentedNumberUnscreened(ctrl, "divertingNr", tag, ++ pos, explicit_end, &diverting_leg_information_2->diverting)); ++ diverting_leg_information_2->diverting_present = 1; ++ ++ ASN1_END_FIXUP(ctrl, pos, explicit_offset, explicit_end, seq_end); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 2: ++ /* Remove EXPLICIT tag */ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " Explicit %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, seq_end, &length)); ++ ASN1_END_SETUP(explicit_end, explicit_offset, length, pos, seq_end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, explicit_end, &tag)); ++ ASN1_CALL(pos, rose_dec_PresentedNumberUnscreened(ctrl, "originalCalledNr", ++ tag, pos, explicit_end, &diverting_leg_information_2->original_called)); ++ diverting_leg_information_2->original_called_present = 1; ++ ++ ASN1_END_FIXUP(ctrl, pos, explicit_offset, explicit_end, seq_end); ++ break; ++ default: ++ pos = save_pos; ++ goto cancel_options; ++ } ++ } ++cancel_options:; ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the DivertingLegInformation3 invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_etsi_DivertingLegInformation3_ARG(struct pri *ctrl, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args) ++{ ++ int32_t value; ++ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_BOOLEAN); ++ ASN1_CALL(pos, asn1_dec_boolean(ctrl, "presentationAllowedIndicator", tag, pos, end, ++ &value)); ++ args->etsi.DivertingLegInformation3.presentation_allowed_indicator = value; ++ ++ return pos; ++} ++ ++/* ------------------------------------------------------------------- */ ++/* end rose_etsi_diversion.c */ + +Property changes on: rose_etsi_diversion.c +___________________________________________________________________ +Added: svn:eol-style + + native +Added: svn:mime-type + + text/plain +Added: svn:keywords + + 'Author Date Id Revision' + +Index: rose_qsig_ct.c +=================================================================== +--- a/rose_qsig_ct.c (.../tags/1.4.10.2) (revision 0) ++++ b/rose_qsig_ct.c (.../branches/1.4) (revision 1357) +@@ -0,0 +1,883 @@ ++/* ++ * libpri: An implementation of Primary Rate ISDN ++ * ++ * Copyright (C) 2009 Digium, Inc. ++ * ++ * Richard Mudgett <rmudgett@digium.com> ++ * ++ * See http://www.asterisk.org for more information about ++ * the Asterisk project. Please do not directly contact ++ * any of the maintainers of this project for assistance; ++ * the project provides a web site, mailing lists and IRC ++ * channels for your use. ++ * ++ * This program is free software, distributed under the terms of ++ * the GNU General Public License Version 2 as published by the ++ * Free Software Foundation. See the LICENSE file included with ++ * this program for more details. ++ * ++ * In addition, when this program is distributed with Asterisk in ++ * any form that would qualify as a 'combined work' or as a ++ * 'derivative work' (but not mere aggregation), you can redistribute ++ * and/or modify the combination under the terms of the license ++ * provided with that copy of Asterisk, instead of the license ++ * terms granted here. ++ */ ++ ++/*! ++ * \file ++ * \brief Q.SIG ROSE Call-Transfer-Operations (CT) ++ * ++ * Call-Transfer-Operations ECMA-178 Annex F Table F.1 ++ * ++ * \author Richard Mudgett <rmudgett@digium.com> ++ */ ++ ++ ++#include "compat.h" ++#include "libpri.h" ++#include "pri_internal.h" ++#include "rose.h" ++#include "rose_internal.h" ++#include "asn1.h" ++ ++ ++/* ------------------------------------------------------------------- */ ++ ++/*! ++ * \brief Encode the Q.SIG CallTransferIdentify result facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_qsig_CallTransferIdentify_RES(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, const union rose_msg_result_args *args) ++{ ++ unsigned char *seq_len; ++ const struct roseQsigCTIdentifyRes_RES *call_transfer_identify; ++ ++ call_transfer_identify = &args->qsig.CallTransferIdentify; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ASN1_TAG_SEQUENCE); ++ ++ ASN1_CALL(pos, asn1_enc_string_max(pos, end, ASN1_TYPE_NUMERIC_STRING, ++ call_transfer_identify->call_id, sizeof(call_transfer_identify->call_id) - 1)); ++ ASN1_CALL(pos, rose_enc_PartyNumber(ctrl, pos, end, ++ &call_transfer_identify->rerouting_number)); ++ ++ /* No extension to encode */ ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the Q.SIG CallTransferInitiate invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_qsig_CallTransferInitiate_ARG(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ unsigned char *seq_len; ++ const struct roseQsigCTInitiateArg_ARG *call_transfer_initiate; ++ ++ call_transfer_initiate = &args->qsig.CallTransferInitiate; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ASN1_TAG_SEQUENCE); ++ ++ ASN1_CALL(pos, asn1_enc_string_max(pos, end, ASN1_TYPE_NUMERIC_STRING, ++ call_transfer_initiate->call_id, sizeof(call_transfer_initiate->call_id) - 1)); ++ ASN1_CALL(pos, rose_enc_PartyNumber(ctrl, pos, end, ++ &call_transfer_initiate->rerouting_number)); ++ ++ /* No extension to encode */ ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the Q.SIG CallTransferSetup invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_qsig_CallTransferSetup_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ unsigned char *seq_len; ++ const struct roseQsigCTSetupArg_ARG *call_transfer_setup; ++ ++ call_transfer_setup = &args->qsig.CallTransferSetup; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ASN1_TAG_SEQUENCE); ++ ++ ASN1_CALL(pos, asn1_enc_string_max(pos, end, ASN1_TYPE_NUMERIC_STRING, ++ call_transfer_setup->call_id, sizeof(call_transfer_setup->call_id) - 1)); ++ ++ /* No extension to encode */ ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the Q.SIG CallTransferActive invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_qsig_CallTransferActive_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ unsigned char *seq_len; ++ const struct roseQsigCTActiveArg_ARG *call_transfer_active; ++ ++ call_transfer_active = &args->qsig.CallTransferActive; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ASN1_TAG_SEQUENCE); ++ ++ ASN1_CALL(pos, rose_enc_PresentedAddressScreened(ctrl, pos, end, ++ &call_transfer_active->connected)); ++ ++ if (call_transfer_active->q931ie.length) { ++ ASN1_CALL(pos, rose_enc_Q931ie(ctrl, pos, end, ASN1_CLASS_APPLICATION | 0, ++ &call_transfer_active->q931ie)); ++ } ++ ++ if (call_transfer_active->connected_name_present) { ++ ASN1_CALL(pos, rose_enc_qsig_Name(ctrl, pos, end, ++ &call_transfer_active->connected_name)); ++ } ++ ++ /* No extension to encode */ ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the Q.SIG CallTransferComplete invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_qsig_CallTransferComplete_ARG(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ unsigned char *seq_len; ++ const struct roseQsigCTCompleteArg_ARG *call_transfer_complete; ++ ++ call_transfer_complete = &args->qsig.CallTransferComplete; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ASN1_TAG_SEQUENCE); ++ ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, ++ call_transfer_complete->end_designation)); ++ ++ ASN1_CALL(pos, rose_enc_PresentedNumberScreened(ctrl, pos, end, ++ &call_transfer_complete->redirection)); ++ ++ if (call_transfer_complete->q931ie.length) { ++ ASN1_CALL(pos, rose_enc_Q931ie(ctrl, pos, end, ASN1_CLASS_APPLICATION | 0, ++ &call_transfer_complete->q931ie)); ++ } ++ ++ if (call_transfer_complete->redirection_name_present) { ++ ASN1_CALL(pos, rose_enc_qsig_Name(ctrl, pos, end, ++ &call_transfer_complete->redirection_name)); ++ } ++ ++ if (call_transfer_complete->call_status) { ++ /* Not the DEFAULT value */ ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, ++ call_transfer_complete->call_status)); ++ } ++ ++ /* No extension to encode */ ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the Q.SIG CallTransferUpdate invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_qsig_CallTransferUpdate_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ unsigned char *seq_len; ++ const struct roseQsigCTUpdateArg_ARG *call_transfer_update; ++ ++ call_transfer_update = &args->qsig.CallTransferUpdate; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ASN1_TAG_SEQUENCE); ++ ++ ASN1_CALL(pos, rose_enc_PresentedNumberScreened(ctrl, pos, end, ++ &call_transfer_update->redirection)); ++ ++ if (call_transfer_update->redirection_name_present) { ++ ASN1_CALL(pos, rose_enc_qsig_Name(ctrl, pos, end, ++ &call_transfer_update->redirection_name)); ++ } ++ ++ if (call_transfer_update->q931ie.length) { ++ ASN1_CALL(pos, rose_enc_Q931ie(ctrl, pos, end, ASN1_CLASS_APPLICATION | 0, ++ &call_transfer_update->q931ie)); ++ } ++ ++ /* No extension to encode */ ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the Q.SIG SubaddressTransfer invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_qsig_SubaddressTransfer_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ unsigned char *seq_len; ++ const struct roseQsigSubaddressTransferArg_ARG *subaddress_transfer; ++ ++ subaddress_transfer = &args->qsig.SubaddressTransfer; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ASN1_TAG_SEQUENCE); ++ ++ ASN1_CALL(pos, rose_enc_PartySubaddress(ctrl, pos, end, ++ &subaddress_transfer->redirection_subaddress)); ++ ++ /* No extension to encode */ ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the Q.SIG DummyArg invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ * ++ * \details ++ * DummyArg ::= CHOICE { ++ * none NULL, ++ * extension [1] IMPLICIT Extension, ++ * multipleExtension [2] IMPLICIT SEQUENCE OF Extension ++ * } ++ */ ++unsigned char *rose_enc_qsig_DummyArg_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ return asn1_enc_null(pos, end, ASN1_TYPE_NULL); ++} ++ ++/*! ++ * \brief Encode the Q.SIG DummyRes result facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ * ++ * \details ++ * DummyRes ::= CHOICE { ++ * none NULL, ++ * extension [1] IMPLICIT Extension, ++ * multipleExtension [2] IMPLICIT SEQUENCE OF Extension ++ * } ++ */ ++unsigned char *rose_enc_qsig_DummyRes_RES(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_result_args *args) ++{ ++ return asn1_enc_null(pos, end, ASN1_TYPE_NULL); ++} ++ ++/*! ++ * \brief Decode the Q.SIG CallTransferIdentify result argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_qsig_CallTransferIdentify_RES(struct pri *ctrl, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ union rose_msg_result_args *args) ++{ ++ size_t str_len; ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ struct roseQsigCTIdentifyRes_RES *call_transfer_identify; ++ ++ call_transfer_identify = &args->qsig.CallTransferIdentify; ++ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " CallTransferIdentify %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag & ~ASN1_PC_MASK, tag, ASN1_TYPE_NUMERIC_STRING); ++ ASN1_CALL(pos, asn1_dec_string_max(ctrl, "callIdentity", tag, pos, seq_end, ++ sizeof(call_transfer_identify->call_id), call_transfer_identify->call_id, ++ &str_len)); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CALL(pos, rose_dec_PartyNumber(ctrl, "reroutingNumber", tag, pos, seq_end, ++ &call_transfer_identify->rerouting_number)); ++ ++ /* Fixup will skip over any OPTIONAL manufacturer extension information */ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the Q.SIG CallTransferInitiate invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_qsig_CallTransferInitiate_ARG(struct pri *ctrl, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args) ++{ ++ size_t str_len; ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ struct roseQsigCTInitiateArg_ARG *call_transfer_initiate; ++ ++ call_transfer_initiate = &args->qsig.CallTransferInitiate; ++ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " CallTransferInitiate %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag & ~ASN1_PC_MASK, tag, ASN1_TYPE_NUMERIC_STRING); ++ ASN1_CALL(pos, asn1_dec_string_max(ctrl, "callIdentity", tag, pos, seq_end, ++ sizeof(call_transfer_initiate->call_id), call_transfer_initiate->call_id, ++ &str_len)); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CALL(pos, rose_dec_PartyNumber(ctrl, "reroutingNumber", tag, pos, seq_end, ++ &call_transfer_initiate->rerouting_number)); ++ ++ /* Fixup will skip over any OPTIONAL manufacturer extension information */ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the Q.SIG CallTransferSetup invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_qsig_CallTransferSetup_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, union rose_msg_invoke_args *args) ++{ ++ size_t str_len; ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ struct roseQsigCTSetupArg_ARG *call_transfer_setup; ++ ++ call_transfer_setup = &args->qsig.CallTransferSetup; ++ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " CallTransferSetup %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag & ~ASN1_PC_MASK, tag, ASN1_TYPE_NUMERIC_STRING); ++ ASN1_CALL(pos, asn1_dec_string_max(ctrl, "callIdentity", tag, pos, seq_end, ++ sizeof(call_transfer_setup->call_id), call_transfer_setup->call_id, &str_len)); ++ ++ /* Fixup will skip over any OPTIONAL manufacturer extension information */ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the Q.SIG CallTransferActive invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_qsig_CallTransferActive_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, union rose_msg_invoke_args *args) ++{ ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ const unsigned char *save_pos; ++ struct roseQsigCTActiveArg_ARG *call_transfer_active; ++ ++ call_transfer_active = &args->qsig.CallTransferActive; ++ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " CallTransferActive %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CALL(pos, rose_dec_PresentedAddressScreened(ctrl, "connectedAddress", tag, pos, ++ seq_end, &call_transfer_active->connected)); ++ ++ /* ++ * A sequence specifies an ordered list of component types. ++ * However, for simplicity we are not checking the order of ++ * the remaining optional components. ++ */ ++ call_transfer_active->q931ie.length = 0; ++ call_transfer_active->connected_name_present = 0; ++ while (pos < seq_end && *pos != ASN1_INDEF_TERM) { ++ save_pos = pos; ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ switch (tag & ~ASN1_PC_MASK) { ++ case ASN1_CLASS_APPLICATION | 0: ++ ASN1_CALL(pos, rose_dec_Q931ie(ctrl, "basicCallInfoElements", tag, pos, ++ seq_end, &call_transfer_active->q931ie, ++ sizeof(call_transfer_active->q931ie_contents))); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 0: ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 1: ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 2: ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 3: ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 4: ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 7: ++ ASN1_CALL(pos, rose_dec_qsig_Name(ctrl, "connectedName", tag, pos, seq_end, ++ &call_transfer_active->connected_name)); ++ call_transfer_active->connected_name_present = 1; ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 9: ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 10: ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " argumentExtension %s\n", asn1_tag2str(tag)); ++ } ++ /* Fixup will skip over the manufacturer extension information */ ++ default: ++ pos = save_pos; ++ goto cancel_options; ++ } ++ } ++cancel_options:; ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the Q.SIG CallTransferComplete invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_qsig_CallTransferComplete_ARG(struct pri *ctrl, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ union rose_msg_invoke_args *args) ++{ ++ int32_t value; ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ const unsigned char *save_pos; ++ struct roseQsigCTCompleteArg_ARG *call_transfer_complete; ++ ++ call_transfer_complete = &args->qsig.CallTransferComplete; ++ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " CallTransferComplete %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_ENUMERATED); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "endDesignation", tag, pos, seq_end, &value)); ++ call_transfer_complete->end_designation = value; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CALL(pos, rose_dec_PresentedNumberScreened(ctrl, "redirectionNumber", tag, pos, ++ seq_end, &call_transfer_complete->redirection)); ++ ++ /* ++ * A sequence specifies an ordered list of component types. ++ * However, for simplicity we are not checking the order of ++ * the remaining optional components. ++ */ ++ call_transfer_complete->q931ie.length = 0; ++ call_transfer_complete->redirection_name_present = 0; ++ call_transfer_complete->call_status = 0; /* DEFAULT answered */ ++ while (pos < seq_end && *pos != ASN1_INDEF_TERM) { ++ save_pos = pos; ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ switch (tag & ~ASN1_PC_MASK) { ++ case ASN1_CLASS_APPLICATION | 0: ++ ASN1_CALL(pos, rose_dec_Q931ie(ctrl, "basicCallInfoElements", tag, pos, ++ seq_end, &call_transfer_complete->q931ie, ++ sizeof(call_transfer_complete->q931ie_contents))); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 0: ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 1: ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 2: ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 3: ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 4: ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 7: ++ ASN1_CALL(pos, rose_dec_qsig_Name(ctrl, "redirectionName", tag, pos, seq_end, ++ &call_transfer_complete->redirection_name)); ++ call_transfer_complete->redirection_name_present = 1; ++ break; ++ case ASN1_TYPE_ENUMERATED: ++ /* Must not be constructed but we will not check for it for simplicity. */ ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "callStatus", tag, pos, seq_end, &value)); ++ call_transfer_complete->call_status = value; ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 9: ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 10: ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " argumentExtension %s\n", asn1_tag2str(tag)); ++ } ++ /* Fixup will skip over the manufacturer extension information */ ++ default: ++ pos = save_pos; ++ goto cancel_options; ++ } ++ } ++cancel_options:; ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the Q.SIG CallTransferUpdate invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_qsig_CallTransferUpdate_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, union rose_msg_invoke_args *args) ++{ ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ const unsigned char *save_pos; ++ struct roseQsigCTUpdateArg_ARG *call_transfer_update; ++ ++ call_transfer_update = &args->qsig.CallTransferUpdate; ++ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " CallTransferUpdate %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CALL(pos, rose_dec_PresentedNumberScreened(ctrl, "redirectionNumber", tag, pos, ++ seq_end, &call_transfer_update->redirection)); ++ ++ /* ++ * A sequence specifies an ordered list of component types. ++ * However, for simplicity we are not checking the order of ++ * the remaining optional components. ++ */ ++ call_transfer_update->redirection_name_present = 0; ++ call_transfer_update->q931ie.length = 0; ++ while (pos < seq_end && *pos != ASN1_INDEF_TERM) { ++ save_pos = pos; ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ switch (tag & ~ASN1_PC_MASK) { ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 0: ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 1: ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 2: ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 3: ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 4: ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 7: ++ ASN1_CALL(pos, rose_dec_qsig_Name(ctrl, "redirectionName", tag, pos, seq_end, ++ &call_transfer_update->redirection_name)); ++ call_transfer_update->redirection_name_present = 1; ++ break; ++ case ASN1_CLASS_APPLICATION | 0: ++ ASN1_CALL(pos, rose_dec_Q931ie(ctrl, "basicCallInfoElements", tag, pos, ++ seq_end, &call_transfer_update->q931ie, ++ sizeof(call_transfer_update->q931ie_contents))); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 9: ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 10: ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " argumentExtension %s\n", asn1_tag2str(tag)); ++ } ++ /* Fixup will skip over the manufacturer extension information */ ++ default: ++ pos = save_pos; ++ goto cancel_options; ++ } ++ } ++cancel_options:; ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the Q.SIG SubaddressTransfer invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_qsig_SubaddressTransfer_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, union rose_msg_invoke_args *args) ++{ ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ struct roseQsigSubaddressTransferArg_ARG *subaddress_transfer; ++ ++ subaddress_transfer = &args->qsig.SubaddressTransfer; ++ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TAG_SEQUENCE); ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " SubaddressTransfer %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CALL(pos, rose_dec_PartySubaddress(ctrl, "redirectionSubaddress", tag, pos, ++ seq_end, &subaddress_transfer->redirection_subaddress)); ++ ++ /* Fixup will skip over any OPTIONAL manufacturer extension information */ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the Q.SIG DummyArg invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ * ++ * \details ++ * DummyArg ::= CHOICE { ++ * none NULL, ++ * extension [1] IMPLICIT Extension, ++ * multipleExtension [2] IMPLICIT SEQUENCE OF Extension ++ * } ++ */ ++const unsigned char *rose_dec_qsig_DummyArg_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, union rose_msg_invoke_args *args) ++{ ++ const char *name; ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ ++ switch (tag) { ++ case ASN1_TYPE_NULL: ++ return asn1_dec_null(ctrl, "none", tag, pos, end); ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 1: ++ name = "extension Extension"; ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 2: ++ name = "multipleExtension SEQUENCE OF Extension"; ++ break; ++ default: ++ ASN1_DID_NOT_EXPECT_TAG(ctrl, tag); ++ return NULL; ++ } ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s %s\n", name, asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ /* Fixup will skip over the manufacturer extension information */ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the Q.SIG DummyRes result argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ * ++ * \details ++ * DummyRes ::= CHOICE { ++ * none NULL, ++ * extension [1] IMPLICIT Extension, ++ * multipleExtension [2] IMPLICIT SEQUENCE OF Extension ++ * } ++ */ ++const unsigned char *rose_dec_qsig_DummyRes_RES(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, union rose_msg_result_args *args) ++{ ++ const char *name; ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ ++ switch (tag) { ++ case ASN1_TYPE_NULL: ++ return asn1_dec_null(ctrl, "none", tag, pos, end); ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 1: ++ name = "extension Extension"; ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | ASN1_PC_CONSTRUCTED | 2: ++ name = "multipleExtension SEQUENCE OF Extension"; ++ break; ++ default: ++ ASN1_DID_NOT_EXPECT_TAG(ctrl, tag); ++ return NULL; ++ } ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s %s\n", name, asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ /* Fixup will skip over the manufacturer extension information */ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/* ------------------------------------------------------------------- */ ++/* end rose_qsig_ct.c */ + +Property changes on: rose_qsig_ct.c +___________________________________________________________________ +Added: svn:eol-style + + native +Added: svn:mime-type + + text/plain +Added: svn:keywords + + 'Author Date Id Revision' + +Index: rose_q931.c +=================================================================== +--- a/rose_q931.c (.../tags/1.4.10.2) (revision 0) ++++ b/rose_q931.c (.../branches/1.4) (revision 1357) +@@ -0,0 +1,100 @@ ++/* ++ * libpri: An implementation of Primary Rate ISDN ++ * ++ * Copyright (C) 2009 Digium, Inc. ++ * ++ * Richard Mudgett <rmudgett@digium.com> ++ * ++ * See http://www.asterisk.org for more information about ++ * the Asterisk project. Please do not directly contact ++ * any of the maintainers of this project for assistance; ++ * the project provides a web site, mailing lists and IRC ++ * channels for your use. ++ * ++ * This program is free software, distributed under the terms of ++ * the GNU General Public License Version 2 as published by the ++ * Free Software Foundation. See the LICENSE file included with ++ * this program for more details. ++ * ++ * In addition, when this program is distributed with Asterisk in ++ * any form that would qualify as a 'combined work' or as a ++ * 'derivative work' (but not mere aggregation), you can redistribute ++ * and/or modify the combination under the terms of the license ++ * provided with that copy of Asterisk, instead of the license ++ * terms granted here. ++ */ ++ ++/*! ++ * \file ++ * \brief ROSE Q.931 ie encode/decode functions ++ * ++ * \author Richard Mudgett <rmudgett@digium.com> ++ */ ++ ++ ++#include "compat.h" ++#include "libpri.h" ++#include "pri_internal.h" ++#include "rose.h" ++#include "rose_internal.h" ++#include "asn1.h" ++ ++ ++/* ------------------------------------------------------------------- */ ++ ++/*! ++ * \brief Encode the Q.931 ie value. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param tag Component tag to identify the encoded component. ++ * The tag should be ASN1_CLASS_APPLICATION | 0 unless the caller ++ * implicitly tags it otherwise. ++ * \param q931ie Q931 ie information to encode. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_Q931ie(struct pri *ctrl, unsigned char *pos, unsigned char *end, ++ unsigned tag, const struct roseQ931ie *q931ie) ++{ ++ return asn1_enc_string_bin(pos, end, tag, q931ie->contents, q931ie->length); ++} ++ ++/*! ++ * \brief Decode the Q.931 ie value. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param name Field name ++ * \param tag Component tag that identified this production. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param q931ie Parameter storage to fill. ++ * \param contents_size Amount of space "allocated" for the q931ie->contents ++ * element. Must have enough room for a null terminator. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_Q931ie(struct pri *ctrl, const char *name, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, struct roseQ931ie *q931ie, ++ size_t contents_size) ++{ ++ size_t str_len; ++ ++ /* NOTE: The q931ie->contents memory is "allocated" after the struct. */ ++ ASN1_CALL(pos, asn1_dec_string_bin(ctrl, name, tag, pos, end, contents_size, ++ q931ie->contents, &str_len)); ++ q931ie->length = str_len; ++ ++ /* ++ * NOTE: We may want to do some basic decoding of the Q.931 ie list ++ * for debug purposes. ++ */ ++ ++ return pos; ++} ++ ++/* ------------------------------------------------------------------- */ ++/* end rose_q931.c */ + +Property changes on: rose_q931.c +___________________________________________________________________ +Added: svn:eol-style + + native +Added: svn:mime-type + + text/plain +Added: svn:keywords + + 'Author Date Id Revision' + +Index: rose_etsi_ect.c +=================================================================== +--- a/rose_etsi_ect.c (.../tags/1.4.10.2) (revision 0) ++++ b/rose_etsi_ect.c (.../branches/1.4) (revision 1357) +@@ -0,0 +1,332 @@ ++/* ++ * libpri: An implementation of Primary Rate ISDN ++ * ++ * Copyright (C) 2009 Digium, Inc. ++ * ++ * Richard Mudgett <rmudgett@digium.com> ++ * ++ * See http://www.asterisk.org for more information about ++ * the Asterisk project. Please do not directly contact ++ * any of the maintainers of this project for assistance; ++ * the project provides a web site, mailing lists and IRC ++ * channels for your use. ++ * ++ * This program is free software, distributed under the terms of ++ * the GNU General Public License Version 2 as published by the ++ * Free Software Foundation. See the LICENSE file included with ++ * this program for more details. ++ * ++ * In addition, when this program is distributed with Asterisk in ++ * any form that would qualify as a 'combined work' or as a ++ * 'derivative work' (but not mere aggregation), you can redistribute ++ * and/or modify the combination under the terms of the license ++ * provided with that copy of Asterisk, instead of the license ++ * terms granted here. ++ */ ++ ++/*! ++ * \file ++ * \brief ROSE Explicit Call Transfer operations. ++ * ++ * Explicit Call Transfer (ECT) Supplementary Services ETS 300 369-1 ++ * ++ * \author Richard Mudgett <rmudgett@digium.com> ++ */ ++ ++ ++#include "compat.h" ++#include "libpri.h" ++#include "pri_internal.h" ++#include "rose.h" ++#include "rose_internal.h" ++#include "asn1.h" ++ ++ ++/* ------------------------------------------------------------------- */ ++ ++/*! ++ * \brief Encode the ExplicitEctExecute invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_etsi_ExplicitEctExecute_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ return asn1_enc_int(pos, end, ASN1_TYPE_INTEGER, ++ args->etsi.ExplicitEctExecute.link_id); ++} ++ ++/*! ++ * \brief Encode the SubaddressTransfer invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_etsi_SubaddressTransfer_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ return rose_enc_PartySubaddress(ctrl, pos, end, ++ &args->etsi.SubaddressTransfer.subaddress); ++} ++ ++/*! ++ * \brief Encode the EctLinkIdRequest result facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_etsi_EctLinkIdRequest_RES(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_result_args *args) ++{ ++ return asn1_enc_int(pos, end, ASN1_TYPE_INTEGER, ++ args->etsi.EctLinkIdRequest.link_id); ++} ++ ++/*! ++ * \brief Encode the EctInform invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_etsi_EctInform_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ const struct roseEtsiEctInform_ARG *ect_inform; ++ unsigned char *seq_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, ASN1_TAG_SEQUENCE); ++ ++ ect_inform = &args->etsi.EctInform; ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, ect_inform->status)); ++ if (ect_inform->redirection_present) { ++ ASN1_CALL(pos, rose_enc_PresentedNumberUnscreened(ctrl, pos, end, ++ &ect_inform->redirection)); ++ } ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the EctLoopTest invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_etsi_EctLoopTest_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ return asn1_enc_int(pos, end, ASN1_TYPE_INTEGER, ++ args->etsi.EctLoopTest.call_transfer_id); ++} ++ ++/*! ++ * \brief Encode the EctLoopTest result facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_etsi_EctLoopTest_RES(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_result_args *args) ++{ ++ return asn1_enc_int(pos, end, ASN1_TYPE_ENUMERATED, ++ args->etsi.EctLoopTest.loop_result); ++} ++ ++/*! ++ * \brief Decode the ExplicitEctExecute invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_etsi_ExplicitEctExecute_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, union rose_msg_invoke_args *args) ++{ ++ int32_t value; ++ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_INTEGER); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "linkId", tag, pos, end, &value)); ++ args->etsi.ExplicitEctExecute.link_id = value; ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the SubaddressTransfer invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_etsi_SubaddressTransfer_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, union rose_msg_invoke_args *args) ++{ ++ return rose_dec_PartySubaddress(ctrl, "transferredToSubaddress", tag, pos, end, ++ &args->etsi.SubaddressTransfer.subaddress); ++} ++ ++/*! ++ * \brief Decode the EctLinkIdRequest result argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_etsi_EctLinkIdRequest_RES(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, union rose_msg_result_args *args) ++{ ++ int32_t value; ++ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_INTEGER); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "linkId", tag, pos, end, &value)); ++ args->etsi.EctLinkIdRequest.link_id = value; ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the EctInform invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_etsi_EctInform_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, union rose_msg_invoke_args *args) ++{ ++ struct roseEtsiEctInform_ARG *ect_inform; ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ int32_t value; ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " EctInform %s\n", asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ect_inform = &args->etsi.EctInform; ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_ENUMERATED); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "callStatus", tag, pos, seq_end, &value)); ++ ect_inform->status = value; ++ ++ if (pos < seq_end && *pos != ASN1_INDEF_TERM) { ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CALL(pos, rose_dec_PresentedNumberUnscreened(ctrl, "redirectionNumber", tag, ++ pos, seq_end, &ect_inform->redirection)); ++ ect_inform->redirection_present = 1; ++ } else { ++ ect_inform->redirection_present = 0; ++ } ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the EctLoopTest invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_etsi_EctLoopTest_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, union rose_msg_invoke_args *args) ++{ ++ int32_t value; ++ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_INTEGER); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "callTransferId", tag, pos, end, &value)); ++ args->etsi.EctLoopTest.call_transfer_id = value; ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the EctLoopTest result argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_etsi_EctLoopTest_RES(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, union rose_msg_result_args *args) ++{ ++ int32_t value; ++ ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_ENUMERATED); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "loopResult", tag, pos, end, &value)); ++ args->etsi.EctLoopTest.loop_result = value; ++ ++ return pos; ++} ++ ++/* ------------------------------------------------------------------- */ ++/* end rose_etsi_ect.c */ + +Property changes on: rose_etsi_ect.c +___________________________________________________________________ +Added: svn:eol-style + + native +Added: svn:mime-type + + text/plain +Added: svn:keywords + + 'Author Date Id Revision' + +Index: pri_internal.h +=================================================================== +--- a/pri_internal.h (.../tags/1.4.10.2) (revision 1357) ++++ b/pri_internal.h (.../branches/1.4) (revision 1357) +@@ -32,30 +32,45 @@ + + #include <stddef.h> + #include <sys/time.h> ++#include "pri_q921.h" ++#include "pri_q931.h" + ++#define ARRAY_LEN(arr) (sizeof(arr) / sizeof((arr)[0])) ++ + #define DBGHEAD __FILE__ ":%d %s: " + #define DBGINFO __LINE__,__PRETTY_FUNCTION__ + ++/* Forward declare some structs */ ++struct apdu_event; ++ + struct pri_sched { + struct timeval when; + void (*callback)(void *data); + void *data; + }; + +-struct q921_frame; +-enum q931_state; +-enum q931_mode; +- +-/* No more than 128 scheduled events */ ++/*! Maximum number of scheduled events active at the same time. */ + #define MAX_SCHED 128 + +-#define MAX_TIMERS 32 ++/*! Maximum number of facility ie's to handle per incoming message. */ ++#define MAX_FACILITY_IES 8 + ++/*! Accumulated pri_message() line until a '\n' is seen on the end. */ ++struct pri_msg_line { ++ /*! Accumulated buffer used. */ ++ unsigned length; ++ /*! Accumulated pri_message() contents. */ ++ char str[2048]; ++}; ++ ++/*! \brief D channel controller structure */ + struct pri { + int fd; /* File descriptor for D-Channel */ + pri_io_cb read_func; /* Read data callback */ + pri_io_cb write_func; /* Write data callback */ + void *userdata; ++ /*! Accumulated pri_message() line. (Valid in master record only) */ ++ struct pri_msg_line *msg_line; + struct pri *subchannel; /* Sub-channel if appropriate */ + struct pri *master; /* Master channel if appropriate */ + struct pri_sched pri_sched[MAX_SCHED]; /* Scheduled events */ +@@ -71,7 +86,9 @@ + int protodisc; + unsigned int bri:1; + unsigned int acceptinbanddisconnect:1; /* Should we allow inband progress after DISCONNECT? */ +- ++ unsigned int hold_support:1;/* TRUE if upper layer supports call hold. */ ++ unsigned int deflection_support:1;/* TRUE if upper layer supports call deflection/rerouting. */ ++ + /* Q.921 State */ + int q921_state; + int window; /* Max window size */ +@@ -97,12 +114,14 @@ + int ri; + int t200_timer; /* T-200 retransmission timer */ + /* All ISDN Timer values */ +- int timers[MAX_TIMERS]; ++ int timers[PRI_MAX_TIMERS]; + + /* Used by scheduler */ + struct timeval tv; + int schedev; + pri_event ev; /* Static event thingy */ ++ /*! Subcommands for static event thingy. */ ++ struct pri_subcommands subcmds; + + /* Q.921 Re-transmission queue */ + struct q921_frame *txqueue; +@@ -111,9 +130,19 @@ + q931_call **callpool; + q931_call *localpool; + ++ /*! ++ * \brief Q.931 Dummy call reference call associated with this TEI. ++ * \note If present then this call is allocated as part of the ++ * D channel control structure. ++ */ ++ q931_call *dummy_call; ++ + /* do we do overlap dialing */ + int overlapdial; + ++ /* do we support SERVICE messages */ ++ int service_message_support; ++ + /* do not skip channel 16 */ + int chan_mapping_logical; + +@@ -125,52 +154,227 @@ + unsigned int q931_rxcount; + #endif + +- unsigned char last_invoke; /* Last ROSE invoke ID */ ++ short last_invoke; /* Last ROSE invoke ID (Valid in master record only) */ + unsigned char sendfacility; ++ ++ /*! For delayed processing of facility ie's. */ ++ struct { ++ /*! Array of facility ie locations in the current received message. */ ++ q931_ie *ie[MAX_FACILITY_IES]; ++ /*! Codeset facility ie found within. */ ++ unsigned char codeset[MAX_FACILITY_IES]; ++ /*! Number of facility ie's in the array from the current received message. */ ++ unsigned char count; ++ } facility; + }; + ++/*! \brief Maximum name length plus null terminator (From ECMA-164) */ ++#define PRI_MAX_NAME_LEN (50 + 1) ++ ++/*! \brief Q.SIG name information. */ ++struct q931_party_name { ++ /*! \brief TRUE if name data is valid */ ++ unsigned char valid; ++ /*! ++ * \brief Q.931 presentation-indicator encoded field ++ * \note Must tollerate the Q.931 screening-indicator field values being present. ++ */ ++ unsigned char presentation; ++ /*! ++ * \brief Character set the name is using. ++ * \details ++ * unknown(0), ++ * iso8859-1(1), ++ * enum-value-withdrawn-by-ITU-T(2) ++ * iso8859-2(3), ++ * iso8859-3(4), ++ * iso8859-4(5), ++ * iso8859-5(6), ++ * iso8859-7(7), ++ * iso10646-BmpString(8), ++ * iso10646-utf-8String(9) ++ */ ++ unsigned char char_set; ++ /*! \brief Name data with null terminator. */ ++ char str[PRI_MAX_NAME_LEN]; ++}; ++ ++/*! \brief Maximum phone number (address) length plus null terminator */ ++#define PRI_MAX_NUMBER_LEN (31 + 1) ++ ++struct q931_party_number { ++ /*! \brief TRUE if number data is valid */ ++ unsigned char valid; ++ /*! \brief Q.931 presentation-indicator and screening-indicator encoded fields */ ++ unsigned char presentation; ++ /*! \brief Q.931 Type-Of-Number and numbering-plan encoded fields */ ++ unsigned char plan; ++ /*! \brief Number data with terminator. */ ++ char str[PRI_MAX_NUMBER_LEN]; ++}; ++ ++/*! \brief Maximum subaddress length plus null terminator */ ++#define PRI_MAX_SUBADDRESS_LEN (20 + 1) ++ ++struct q931_party_subaddress { ++ /*! \brief TRUE if the subaddress information is valid/present */ ++ unsigned char valid; ++ /*! ++ * \brief Subaddress type. ++ * \details ++ * nsap(0), ++ * user_specified(2) ++ */ ++ unsigned char type; ++ /*! ++ * \brief TRUE if odd number of address signals ++ * \note The odd/even indicator is used when the type of subaddress is ++ * user_specified and the coding is BCD. ++ */ ++ unsigned char odd_even_indicator; ++ /*! \brief Length of the subaddress data */ ++ unsigned char length; ++ /*! ++ * \brief Subaddress data with null terminator. ++ * \note The null terminator is a convenience only since the data could be ++ * BCD/binary and thus have a null byte as part of the contents. ++ */ ++ unsigned char data[PRI_MAX_SUBADDRESS_LEN]; ++}; ++ ++struct q931_party_address { ++ /*! \brief Subscriber phone number */ ++ struct q931_party_number number; ++ /*! \brief Subscriber subaddress */ ++ struct q931_party_subaddress subaddress; ++}; ++ ++/*! \brief Information needed to identify an endpoint in a call. */ ++struct q931_party_id { ++ /*! \brief Subscriber name */ ++ struct q931_party_name name; ++ /*! \brief Subscriber phone number */ ++ struct q931_party_number number; ++ /*! \brief Subscriber subaddress */ ++ struct q931_party_subaddress subaddress; ++}; ++ ++enum Q931_REDIRECTING_STATE { ++ /*! ++ * \details ++ * CDO-Idle/CDF-Inv-Idle ++ */ ++ Q931_REDIRECTING_STATE_IDLE, ++ /*! ++ * \details ++ * CDF-Inv-Wait - A DivLeg2 has been received and ++ * we are waiting for valid presentation restriction information to send. ++ */ ++ Q931_REDIRECTING_STATE_PENDING_TX_DIV_LEG_3, ++ /*! ++ * \details ++ * CDO-Divert - A DivLeg1 has been received and ++ * we are waiting for the presentation restriction information to come in. ++ */ ++ Q931_REDIRECTING_STATE_EXPECTING_RX_DIV_LEG_3, ++}; ++ ++/*! ++ * \brief Do not increment above this count. ++ * \details ++ * It is not our responsibility to enforce the maximum number of redirects. ++ * However, we cannot allow an increment past this number without breaking things. ++ * Besides, more than 255 redirects is probably not a good thing. ++ */ ++#define PRI_MAX_REDIRECTS 0xFF ++ ++/*! \brief Redirecting information struct */ ++struct q931_party_redirecting { ++ enum Q931_REDIRECTING_STATE state; ++ /*! \brief Who is redirecting the call (Sent to the party the call is redirected toward) */ ++ struct q931_party_id from; ++ /*! \brief Call is redirecting to a new party (Sent to the caller) */ ++ struct q931_party_id to; ++ /*! Originally called party (in cases of multiple redirects) */ ++ struct q931_party_id orig_called; ++ /*! ++ * \brief Number of times the call was redirected ++ * \note The call is being redirected if the count is non-zero. ++ */ ++ unsigned char count; ++ /*! Original reason for redirect (in cases of multiple redirects) */ ++ unsigned char orig_reason; ++ /*! \brief Redirection reasons */ ++ unsigned char reason; ++}; ++ ++/*! \brief New call setup parameter structure */ + struct pri_sr { + int transmode; + int channel; + int exclusive; + int nonisdn; +- char *caller; +- int callerplan; +- char *callername; +- int callerpres; +- char *called; +- int calledplan; ++ struct q931_party_redirecting redirecting; ++ struct q931_party_id caller; ++ struct q931_party_address called; + int userl1; + int numcomplete; +- char *redirectingnum; +- int redirectingplan; +- int redirectingpres; +- int redirectingreason; +- int justsignalling; ++ int cis_call; ++ int cis_auto_disconnect; + const char *useruserinfo; ++ const char *keypad_digits; + int transferable; ++ int reversecharge; + }; + + /* Internal switch types */ + #define PRI_SWITCH_GR303_EOC_PATH 19 + #define PRI_SWITCH_GR303_TMC_SWITCHING 20 + +-struct apdu_event { +- int message; /* What message to send the ADPU in */ +- void (*callback)(void *data); /* Callback function for when response is received */ +- void *data; /* Data to callback */ +- unsigned char apdu[255]; /* ADPU to send */ +- int apdu_len; /* Length of ADPU */ +- int sent; /* Have we been sent already? */ +- struct apdu_event *next; /* Linked list pointer */ ++#define Q931_MAX_TEI 8 ++ ++/*! \brief Incoming call transfer states. */ ++enum INCOMING_CT_STATE { ++ /*! ++ * \details ++ * Incoming call transfer is not active. ++ */ ++ INCOMING_CT_STATE_IDLE, ++ /*! ++ * \details ++ * We have seen an incoming CallTransferComplete(alerting) ++ * so we are waiting for the expected CallTransferActive ++ * before updating the connected line about the remote party id. ++ */ ++ INCOMING_CT_STATE_EXPECT_CT_ACTIVE, ++ /*! ++ * \details ++ * A call transfer message came in that updated the remote party id ++ * that we need to post a connected line update. ++ */ ++ INCOMING_CT_STATE_POST_CONNECTED_LINE + }; + ++/*! Call hold supplementary states. */ ++enum Q931_HOLD_STATE { ++ /*! \brief No call hold activity. */ ++ Q931_HOLD_STATE_IDLE, ++ /*! \brief Request made to hold call. */ ++ Q931_HOLD_STATE_HOLD_REQ, ++ /*! \brief Request received to hold call. */ ++ Q931_HOLD_STATE_HOLD_IND, ++ /*! \brief Call is held. */ ++ Q931_HOLD_STATE_CALL_HELD, ++ /*! \brief Request made to retrieve call. */ ++ Q931_HOLD_STATE_RETRIEVE_REQ, ++ /*! \brief Request received to retrieve call. */ ++ Q931_HOLD_STATE_RETRIEVE_IND, ++}; ++ + /* q931_call datastructure */ +- + struct q931_call { + struct pri *pri; /* PRI */ + int cr; /* Call Reference */ +- int forceinvert; /* Force inversion of call number even if 0 */ + q931_call *next; + /* Slotmap specified (bitmap of channels 31/24-1) (Channel Identifier IE) (-1 means not specified) */ + int slotmap; +@@ -199,73 +403,144 @@ + int userl2; + int userl3; + int rateadaption; +- +- int sentchannel; +- int justsignalling; /* for a signalling-only connection */ + ++ /*! ++ * \brief TRUE if the call is a Call Independent Signalling connection. ++ * \note The call has no B channel associated with it. (Just signalling) ++ */ ++ int cis_call; ++ /*! \brief TRUE if we will auto disconnect the cis_call we originated. */ ++ int cis_auto_disconnect; ++ + int progcode; /* Progress coding */ + int progloc; /* Progress Location */ + int progress; /* Progress indicator */ + int progressmask; /* Progress Indicator bitmask */ + +- int notify; /* Notification */ ++ int notify; /* Notification indicator. */ + + int causecode; /* Cause Coding */ + int causeloc; /* Cause Location */ + int cause; /* Cause of clearing */ + +- int peercallstate; /* Call state of peer as reported */ +- int ourcallstate; /* Our call state */ +- int sugcallstate; /* Status call state */ +- +- int callerplan; +- int callerplanani; +- int callerpres; /* Caller presentation */ +- char callerani[256]; /* Caller */ +- char callernum[256]; +- char callername[256]; ++ enum Q931_CALL_STATE peercallstate; /* Call state of peer as reported */ ++ enum Q931_CALL_STATE ourcallstate; /* Our call state */ ++ enum Q931_CALL_STATE sugcallstate; /* Status call state */ + +- char keypad_digits[64]; /* Buffer for digits that come in KEYPAD_FACILITY */ ++ int ani2; /* ANI II */ + +- int ani2; /* ANI II */ +- +- int calledplan; ++ /*! Buffer for digits that come in KEYPAD_FACILITY */ ++ char keypad_digits[32 + 1]; ++ ++ /*! Current dialed digits to be sent or just received. */ ++ char overlap_digits[PRI_MAX_NUMBER_LEN]; ++ ++ /*! ++ * \brief Local party ID ++ * \details ++ * The Caller-ID and connected-line ID are just roles the local and remote party ++ * play while a call is being established. Which roll depends upon the direction ++ * of the call. ++ * Outgoing party info is to identify the local party to the other end. ++ * (Caller-ID for originated or connected-line for answered calls.) ++ * Incoming party info is to identify the remote party to us. ++ * (Caller-ID for answered or connected-line for originated calls.) ++ */ ++ struct q931_party_id local_id; ++ /*! ++ * \brief Remote party ID ++ * \details ++ * The Caller-ID and connected-line ID are just roles the local and remote party ++ * play while a call is being established. Which roll depends upon the direction ++ * of the call. ++ * Outgoing party info is to identify the local party to the other end. ++ * (Caller-ID for originated or connected-line for answered calls.) ++ * Incoming party info is to identify the remote party to us. ++ * (Caller-ID for answered or connected-line for originated calls.) ++ */ ++ struct q931_party_id remote_id; ++ ++ /*! ++ * \brief Staging place for the Q.931 redirection number ie. ++ * \note ++ * The number could be the remote_id.number or redirecting.to.number ++ * depending upon the notification indicator. ++ */ ++ struct q931_party_number redirection_number; ++ ++ /*! ++ * \brief Called party address. ++ * \note The called.number.str is the accumulated overlap dial digits ++ * and enbloc digits. ++ * \note The called.number.presentation value is not used. ++ */ ++ struct q931_party_address called; + int nonisdn; +- char callednum[256]; /* Called Number */ + int complete; /* no more digits coming */ + int newcall; /* if the received message has a new call reference value */ + + int retranstimer; /* Timer for retransmitting DISC */ + int t308_timedout; /* Whether t308 timed out once */ + +- int redirectingplan; +- int redirectingpres; +- int redirectingreason; +- char redirectingnum[256]; /* Number of redirecting party */ +- char redirectingname[256]; /* Name of redirecting party */ ++ struct q931_party_redirecting redirecting; + +- /* Filled in cases of multiple diversions */ +- int origcalledplan; +- int origcalledpres; +- int origredirectingreason; /* Original reason for redirect (in cases of multiple redirects) */ +- char origcalledname[256]; /* Original name of person being called */ +- char origcallednum[256]; /* Orignal number of person being called */ ++ /*! \brief Incoming call transfer state. */ ++ enum INCOMING_CT_STATE incoming_ct_state; ++ /*! Call hold supplementary state. */ ++ enum Q931_HOLD_STATE hold_state; ++ /*! Call hold event timer */ ++ int hold_timer; + ++ int deflection_in_progress; /*!< CallDeflection for NT PTMP in progress. */ ++ /*! TRUE if the connected number ie was in the current received message. */ ++ int connected_number_in_message; ++ /*! TRUE if the redirecting number ie was in the current received message. */ ++ int redirecting_number_in_message; ++ + int useruserprotocoldisc; + char useruserinfo[256]; +- char callingsubaddr[256]; /* Calling parties sub address */ + + long aoc_units; /* Advice of Charge Units */ + + struct apdu_event *apdus; /* APDU queue for call */ + +- int transferable; ++ int transferable; /* RLT call is transferable */ + unsigned int rlt_call_id; /* RLT call id */ + + /* Bridged call info */ +- q931_call *bridged_call; /* Pointer to other leg of bridged call */ ++ q931_call *bridged_call; /* Pointer to other leg of bridged call (Used by Q.SIG when eliminating tromboned calls) */ ++ ++ int changestatus; /* SERVICE message changestatus */ ++ int reversecharge; /* Reverse charging indication: ++ -1 - No reverse charging ++ 1 - Reverse charging ++ 0,2-7 - Reserved for future use */ ++ int t303_timer; ++ int t303_expirycnt; ++ ++ int hangupinitiated; ++ /*! \brief TRUE if we broadcast this call's SETUP message. */ ++ int outboundbroadcast; ++ int performing_fake_clearing; ++ /*! ++ * \brief Master call controlling this call. ++ * \note Always valid. Master and normal calls point to self. ++ */ ++ struct q931_call *master_call; ++ ++ /* These valid in master call only */ ++ struct q931_call *subcalls[Q931_MAX_TEI]; ++ int pri_winner; + }; + ++/*! D channel control structure with associated dummy call reference record. */ ++struct d_ctrl_dummy { ++ /*! D channel control structure. Must be first in the structure. */ ++ struct pri ctrl; ++ /*! Dummy call reference call record. */ ++ struct q931_call dummy_call; ++}; ++ + extern int pri_schedule_event(struct pri *pri, int ms, void (*function)(void *data), void *data); + + extern pri_event *pri_schedule_run(struct pri *pri); +@@ -274,14 +549,104 @@ + + extern pri_event *pri_mkerror(struct pri *pri, char *errstr); + +-extern void pri_message(struct pri *pri, char *fmt, ...); ++void pri_message(struct pri *ctrl, const char *fmt, ...) __attribute__((format(printf, 2, 3))); ++void pri_error(struct pri *ctrl, const char *fmt, ...) __attribute__((format(printf, 2, 3))); + +-extern void pri_error(struct pri *pri, char *fmt, ...); +- + void libpri_copy_string(char *dst, const char *src, size_t size); + + struct pri *__pri_new_tei(int fd, int node, int switchtype, struct pri *master, pri_io_cb rd, pri_io_cb wr, void *userdata, int tei, int bri); + + void __pri_free_tei(struct pri *p); + ++void q931_init_call_record(struct pri *ctrl, struct q931_call *call, int cr); ++ ++void q931_party_name_init(struct q931_party_name *name); ++void q931_party_number_init(struct q931_party_number *number); ++void q931_party_subaddress_init(struct q931_party_subaddress *subaddr); ++void q931_party_address_init(struct q931_party_address *address); ++void q931_party_id_init(struct q931_party_id *id); ++void q931_party_redirecting_init(struct q931_party_redirecting *redirecting); ++ ++static inline void q931_party_address_to_id(struct q931_party_id *id, struct q931_party_address *address) ++{ ++ id->number = address->number; ++ id->subaddress = address->subaddress; ++} ++ ++int q931_party_name_cmp(const struct q931_party_name *left, const struct q931_party_name *right); ++int q931_party_number_cmp(const struct q931_party_number *left, const struct q931_party_number *right); ++int q931_party_subaddress_cmp(const struct q931_party_subaddress *left, const struct q931_party_subaddress *right); ++int q931_party_id_cmp(const struct q931_party_id *left, const struct q931_party_id *right); ++ ++void q931_party_name_copy_to_pri(struct pri_party_name *pri_name, const struct q931_party_name *q931_name); ++void q931_party_number_copy_to_pri(struct pri_party_number *pri_number, const struct q931_party_number *q931_number); ++void q931_party_subaddress_copy_to_pri(struct pri_party_subaddress *pri_subaddress, const struct q931_party_subaddress *q931_subaddress); ++void q931_party_id_copy_to_pri(struct pri_party_id *pri_id, const struct q931_party_id *q931_id); ++void q931_party_redirecting_copy_to_pri(struct pri_party_redirecting *pri_redirecting, const struct q931_party_redirecting *q931_redirecting); ++ ++void q931_party_id_fixup(const struct pri *ctrl, struct q931_party_id *id); ++int q931_party_id_presentation(const struct q931_party_id *id); ++ ++const char *q931_call_state_str(enum Q931_CALL_STATE callstate); ++const char *msg2str(int msg); ++ ++int q931_is_ptmp(const struct pri *ctrl); ++int q931_master_pass_event(struct pri *ctrl, struct q931_call *subcall, int msg_type); ++struct pri_subcommand *q931_alloc_subcommand(struct pri *ctrl); ++ ++int q931_notify_redirection(struct pri *ctrl, q931_call *call, int notify, const struct q931_party_number *number); ++ ++static inline struct pri * PRI_MASTER(struct pri *mypri) ++{ ++ struct pri *pri = mypri; ++ ++ if (!pri) ++ return NULL; ++ ++ while (pri->master) ++ pri = pri->master; ++ ++ return pri; ++} ++ ++static inline int BRI_NT_PTMP(struct pri *mypri) ++{ ++ struct pri *pri; ++ ++ pri = PRI_MASTER(mypri); ++ ++ return pri->bri && (((pri)->localtype == PRI_NETWORK) && ((pri)->tei == Q921_TEI_GROUP)); ++} ++ ++static inline int BRI_TE_PTMP(struct pri *mypri) ++{ ++ struct pri *pri; ++ ++ pri = PRI_MASTER(mypri); ++ ++ return pri->bri && (((pri)->localtype == PRI_CPE) && ((pri)->tei == Q921_TEI_GROUP)); ++} ++ ++static inline int PRI_PTP(struct pri *mypri) ++{ ++ struct pri *pri; ++ ++ pri = PRI_MASTER(mypri); ++ ++ return !pri->bri; ++} ++ ++#define Q931_DUMMY_CALL_REFERENCE -1 ++ ++/*! ++ * \brief Deterimine if the given call control pointer is a dummy call. ++ * ++ * \retval TRUE if given call is a dummy call. ++ * \retval FALSE otherwise. ++ */ ++static inline int q931_is_dummy_call(const q931_call *call) ++{ ++ return (call->cr == Q931_DUMMY_CALL_REFERENCE) ? 1 : 0; ++} ++ + #endif +Index: rose_qsig_name.c +=================================================================== +--- a/rose_qsig_name.c (.../tags/1.4.10.2) (revision 0) ++++ b/rose_qsig_name.c (.../branches/1.4) (revision 1357) +@@ -0,0 +1,474 @@ ++/* ++ * libpri: An implementation of Primary Rate ISDN ++ * ++ * Copyright (C) 2009 Digium, Inc. ++ * ++ * Richard Mudgett <rmudgett@digium.com> ++ * ++ * See http://www.asterisk.org for more information about ++ * the Asterisk project. Please do not directly contact ++ * any of the maintainers of this project for assistance; ++ * the project provides a web site, mailing lists and IRC ++ * channels for your use. ++ * ++ * This program is free software, distributed under the terms of ++ * the GNU General Public License Version 2 as published by the ++ * Free Software Foundation. See the LICENSE file included with ++ * this program for more details. ++ * ++ * In addition, when this program is distributed with Asterisk in ++ * any form that would qualify as a 'combined work' or as a ++ * 'derivative work' (but not mere aggregation), you can redistribute ++ * and/or modify the combination under the terms of the license ++ * provided with that copy of Asterisk, instead of the license ++ * terms granted here. ++ */ ++ ++/*! ++ * \file ++ * \brief Q.SIG ROSE Name operations and elements ++ * ++ * Name-Operations ECMA-164 Annex C ++ * ++ * \author Richard Mudgett <rmudgett@digium.com> ++ */ ++ ++ ++#include "compat.h" ++#include "libpri.h" ++#include "pri_internal.h" ++#include "rose.h" ++#include "rose_internal.h" ++#include "asn1.h" ++ ++ ++/* ------------------------------------------------------------------- */ ++ ++/*! ++ * \internal ++ * \brief Encode the Q.SIG NameSet type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param tag Component tag to identify the encoded component. ++ * The tag should be ASN1_TAG_SEQUENCE unless the caller ++ * implicitly tags it otherwise. ++ * \param name ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *rose_enc_qsig_NameSet(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, unsigned tag, const struct roseQsigName *name) ++{ ++ unsigned char *seq_len; ++ ++ ASN1_CONSTRUCTED_BEGIN(seq_len, pos, end, tag); ++ ++ ASN1_CALL(pos, asn1_enc_string_bin(pos, end, ASN1_TYPE_OCTET_STRING, name->data, ++ name->length)); ++ ASN1_CALL(pos, asn1_enc_int(pos, end, ASN1_TYPE_INTEGER, name->char_set)); ++ ++ ASN1_CONSTRUCTED_END(seq_len, pos, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Encode the Q.SIG Name type. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param name ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_qsig_Name(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const struct roseQsigName *name) ++{ ++ switch (name->presentation) { ++ case 0: /* optional_name_not_present */ ++ /* Do not encode anything */ ++ break; ++ case 1: /* presentation_allowed */ ++ if (name->char_set == 1) { ++ ASN1_CALL(pos, asn1_enc_string_bin(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 0, ++ name->data, name->length)); ++ } else { ++ ASN1_CALL(pos, rose_enc_qsig_NameSet(ctrl, pos, end, ++ ASN1_CLASS_CONTEXT_SPECIFIC | 1, name)); ++ } ++ break; ++ case 2: /* presentation_restricted */ ++ if (name->char_set == 1) { ++ ASN1_CALL(pos, asn1_enc_string_bin(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 2, ++ name->data, name->length)); ++ } else { ++ ASN1_CALL(pos, rose_enc_qsig_NameSet(ctrl, pos, end, ++ ASN1_CLASS_CONTEXT_SPECIFIC | 3, name)); ++ } ++ break; ++ case 3: /* presentation_restricted_null */ ++ ASN1_CALL(pos, asn1_enc_null(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 7)); ++ break; ++ case 4: /* name_not_available */ ++ ASN1_CALL(pos, asn1_enc_null(pos, end, ASN1_CLASS_CONTEXT_SPECIFIC | 4)); ++ break; ++ default: ++ ASN1_ENC_ERROR(ctrl, "Unknown name presentation"); ++ return NULL; ++ } ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Encode the Q.SIG party-Name invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param party Information to encode. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++static unsigned char *rose_enc_qsig_PartyName_ARG_Backend(struct pri *ctrl, ++ unsigned char *pos, unsigned char *end, const struct roseQsigPartyName_ARG *party) ++{ ++ return rose_enc_qsig_Name(ctrl, pos, end, &party->name); ++} ++ ++/*! ++ * \brief Encode the Q.SIG CallingName invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_qsig_CallingName_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ return rose_enc_qsig_PartyName_ARG_Backend(ctrl, pos, end, &args->qsig.CallingName); ++} ++ ++/*! ++ * \brief Encode the Q.SIG CalledName invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_qsig_CalledName_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ return rose_enc_qsig_PartyName_ARG_Backend(ctrl, pos, end, &args->qsig.CalledName); ++} ++ ++/*! ++ * \brief Encode the Q.SIG ConnectedName invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_qsig_ConnectedName_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ return rose_enc_qsig_PartyName_ARG_Backend(ctrl, pos, end, ++ &args->qsig.ConnectedName); ++} ++ ++/*! ++ * \brief Encode the Q.SIG BusyName invoke facility ie arguments. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param pos Starting position to encode ASN.1 component. ++ * \param end End of ASN.1 encoding data buffer. ++ * \param args Arguments to encode in the buffer. ++ * ++ * \retval Start of the next ASN.1 component to encode on success. ++ * \retval NULL on error. ++ */ ++unsigned char *rose_enc_qsig_BusyName_ARG(struct pri *ctrl, unsigned char *pos, ++ unsigned char *end, const union rose_msg_invoke_args *args) ++{ ++ return rose_enc_qsig_PartyName_ARG_Backend(ctrl, pos, end, &args->qsig.BusyName); ++} ++ ++/*! ++ * \internal ++ * \brief Decode the Q.SIG NameData Name argument parameters. ++ * ++ * \param ctrl D channel controller for any diagnostic messages. ++ * \param fname Field name ++ * \param tag Component tag that identified this production. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param name Parameter storage to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++static const unsigned char *rose_dec_qsig_NameData(struct pri *ctrl, const char *fname, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct roseQsigName *name) ++{ ++ size_t str_len; ++ ++ ASN1_CALL(pos, asn1_dec_string_bin(ctrl, fname, tag, pos, end, sizeof(name->data), ++ name->data, &str_len)); ++ name->length = str_len; ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Decode the Q.SIG NameSet Name argument parameters. ++ * ++ * \param ctrl D channel controller for any diagnostic messages. ++ * \param fname Field name ++ * \param tag Component tag that identified this production. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param name Parameter storage to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++static const unsigned char *rose_dec_qsig_NameSet(struct pri *ctrl, const char *fname, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct roseQsigName *name) ++{ ++ int32_t value; ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s NameSet %s\n", fname, asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag & ~ASN1_PC_MASK, ASN1_TYPE_OCTET_STRING); ++ ASN1_CALL(pos, rose_dec_qsig_NameData(ctrl, "nameData", tag, pos, seq_end, name)); ++ ++ if (pos < end && *pos != ASN1_INDEF_TERM) { ++ ASN1_CALL(pos, asn1_dec_tag(pos, seq_end, &tag)); ++ ASN1_CHECK_TAG(ctrl, tag, tag, ASN1_TYPE_INTEGER); ++ ASN1_CALL(pos, asn1_dec_int(ctrl, "characterSet", tag, pos, seq_end, &value)); ++ name->char_set = value; ++ } else { ++ name->char_set = 1; /* default to iso8859-1 */ ++ } ++ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the Q.SIG Name argument parameters. ++ * ++ * \param ctrl D channel controller for any diagnostic messages. ++ * \param fname Field name ++ * \param tag Component tag that identified this production. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param name Parameter storage to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_qsig_Name(struct pri *ctrl, const char *fname, ++ unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct roseQsigName *name) ++{ ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s Name\n", fname); ++ } ++ name->char_set = 1; /* default to iso8859-1 */ ++ switch (tag & ~ASN1_PC_MASK) { ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 0: ++ name->presentation = 1; /* presentation_allowed */ ++ ASN1_CALL(pos, rose_dec_qsig_NameData(ctrl, "namePresentationAllowedSimple", tag, ++ pos, end, name)); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 1: ++ /* Must be constructed but we will not check for it for simplicity. */ ++ name->presentation = 1; /* presentation_allowed */ ++ ASN1_CALL(pos, rose_dec_qsig_NameSet(ctrl, "namePresentationAllowedExtended", ++ tag, pos, end, name)); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 2: ++ name->presentation = 2; /* presentation_restricted */ ++ ASN1_CALL(pos, rose_dec_qsig_NameData(ctrl, "namePresentationRestrictedSimple", ++ tag, pos, end, name)); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 3: ++ /* Must be constructed but we will not check for it for simplicity. */ ++ name->presentation = 2; /* presentation_restricted */ ++ ASN1_CALL(pos, rose_dec_qsig_NameSet(ctrl, "namePresentationRestrictedExtended", ++ tag, pos, end, name)); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 4: ++ /* Must not be constructed but we will not check for it for simplicity. */ ++ name->presentation = 4; /* name_not_available */ ++ name->length = 0; ++ name->data[0] = 0; ++ ASN1_CALL(pos, asn1_dec_null(ctrl, "nameNotAvailable", tag, pos, end)); ++ break; ++ case ASN1_CLASS_CONTEXT_SPECIFIC | 7: ++ /* Must not be constructed but we will not check for it for simplicity. */ ++ name->presentation = 3; /* presentation_restricted_null */ ++ name->length = 0; ++ name->data[0] = 0; ++ ASN1_CALL(pos, asn1_dec_null(ctrl, "namePresentationRestrictedNull", tag, pos, ++ end)); ++ break; ++ default: ++ ASN1_DID_NOT_EXPECT_TAG(ctrl, tag); ++ return NULL; ++ } ++ ++ return pos; ++} ++ ++/*! ++ * \internal ++ * \brief Decode the Q.SIG party-Name invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param name Field name ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param party Parameter storage to fill. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++static const unsigned char *rose_dec_qsig_PartyName_ARG_Backend(struct pri *ctrl, ++ const char *name, unsigned tag, const unsigned char *pos, const unsigned char *end, ++ struct roseQsigPartyName_ARG *party) ++{ ++ int length; ++ int seq_offset; ++ const unsigned char *seq_end; ++ ++ if (tag == ASN1_TAG_SEQUENCE) { ++ if (ctrl->debug & PRI_DEBUG_APDU) { ++ pri_message(ctrl, " %s %s\n", name, asn1_tag2str(tag)); ++ } ++ ASN1_CALL(pos, asn1_dec_length(pos, end, &length)); ++ ASN1_END_SETUP(seq_end, seq_offset, length, pos, end); ++ ++ ASN1_CALL(pos, rose_dec_qsig_Name(ctrl, "name", tag, pos, seq_end, ++ &party->name)); ++ ++ /* Fixup will skip over any OPTIONAL manufacturer extension information */ ++ ASN1_END_FIXUP(ctrl, pos, seq_offset, seq_end, end); ++ } else { ++ ASN1_CALL(pos, rose_dec_qsig_Name(ctrl, name, tag, pos, end, &party->name)); ++ } ++ ++ return pos; ++} ++ ++/*! ++ * \brief Decode the Q.SIG CallingName invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_qsig_CallingName_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, union rose_msg_invoke_args *args) ++{ ++ return rose_dec_qsig_PartyName_ARG_Backend(ctrl, "callingName", tag, pos, end, ++ &args->qsig.CallingName); ++} ++ ++/*! ++ * \brief Decode the Q.SIG CalledName invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_qsig_CalledName_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, union rose_msg_invoke_args *args) ++{ ++ return rose_dec_qsig_PartyName_ARG_Backend(ctrl, "calledName", tag, pos, end, ++ &args->qsig.CalledName); ++} ++ ++/*! ++ * \brief Decode the Q.SIG ConnectedName invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_qsig_ConnectedName_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, union rose_msg_invoke_args *args) ++{ ++ return rose_dec_qsig_PartyName_ARG_Backend(ctrl, "connectedName", tag, pos, end, ++ &args->qsig.ConnectedName); ++} ++ ++/*! ++ * \brief Decode the Q.SIG BusyName invoke argument parameters. ++ * ++ * \param ctrl D channel controller for diagnostic messages or global options. ++ * \param tag Component tag that identified this structure. ++ * \param pos Starting position of the ASN.1 component length. ++ * \param end End of ASN.1 decoding data buffer. ++ * \param args Arguments to fill in from the decoded buffer. ++ * ++ * \retval Start of the next ASN.1 component on success. ++ * \retval NULL on error. ++ */ ++const unsigned char *rose_dec_qsig_BusyName_ARG(struct pri *ctrl, unsigned tag, ++ const unsigned char *pos, const unsigned char *end, union rose_msg_invoke_args *args) ++{ ++ return rose_dec_qsig_PartyName_ARG_Backend(ctrl, "busyName", tag, pos, end, ++ &args->qsig.BusyName); ++} ++ ++/* ------------------------------------------------------------------- */ ++/* end rose_qsig_name.c */ + +Property changes on: rose_qsig_name.c +___________________________________________________________________ +Added: svn:eol-style + + native +Added: svn:mime-type + + text/plain +Added: svn:keywords + + 'Author Date Id Revision' + +Index: q921.c +=================================================================== +--- a/q921.c (.../tags/1.4.10.2) (revision 1357) ++++ b/q921.c (.../branches/1.4) (revision 1357) +@@ -27,6 +27,7 @@ + * terms granted here. + */ + ++#include <stdint.h> + #include <stdio.h> + #include <string.h> + #include <stdlib.h> +@@ -139,8 +140,7 @@ + #endif + pri->ri = random() % 65535; + q921_send_tei(pri, Q921_TEI_IDENTITY_REQUEST, pri->ri, Q921_TEI_GROUP, 1); +- if (pri->t202_timer) +- pri_schedule_del(pri, pri->t202_timer); ++ pri_schedule_del(pri, pri->t202_timer); + pri->t202_timer = pri_schedule_event(pri, pri->timers[PRI_TIMER_T202], q921_tei_request, pri); + } + +@@ -174,8 +174,8 @@ + { + struct pri *pri = vpri; + q921_h h; ++ + pri_schedule_del(pri, pri->sabme_timer); +- pri->sabme_timer = 0; + pri->sabme_timer = pri_schedule_event(pri, pri->timers[PRI_TIMER_T200], q921_send_sabme_now, pri); + if (!now) + return; +@@ -248,14 +248,12 @@ + + static void t203_expire(void *); + static void t200_expire(void *); +-static pri_event *q921_dchannel_down(struct pri *pri); + + static void reschedule_t200(struct pri *pri) + { + if (pri->debug & PRI_DEBUG_Q921_DUMP) + pri_message(pri, "-- Restarting T200 timer\n"); +- if (pri->t200_timer) +- pri_schedule_del(pri, pri->t200_timer); ++ pri_schedule_del(pri, pri->t200_timer); + pri->t200_timer = pri_schedule_event(pri, pri->timers[PRI_TIMER_T200], t200_expire, pri); + } + +@@ -263,8 +261,7 @@ + { + if (pri->debug & PRI_DEBUG_Q921_DUMP) + pri_message(pri, "-- Restarting T203 timer\n"); +- if (pri->t203_timer) +- pri_schedule_del(pri, pri->t203_timer); ++ pri_schedule_del(pri, pri->t203_timer); + pri->t203_timer = pri_schedule_event(pri, pri->timers[PRI_TIMER_T203], t203_expire, pri); + } + +@@ -312,10 +309,8 @@ + if (pri->debug & PRI_DEBUG_Q921_DUMP) + pri_message(pri, "-- Since there was nothing left, stopping T200 counter\n"); + /* Something was ACK'd. Stop T200 counter */ +- if (pri->t200_timer) { +- pri_schedule_del(pri, pri->t200_timer); +- pri->t200_timer = 0; +- } ++ pri_schedule_del(pri, pri->t200_timer); ++ pri->t200_timer = 0; + } + if (pri->t203_timer) { + if (pri->debug & PRI_DEBUG_Q921_DUMP) +@@ -483,6 +478,45 @@ + } + } + ++int q921_transmit_uiframe(struct pri *pri, void *buf, int len) ++{ ++ uint8_t ubuf[512]; ++ q921_h *h = (void *)&ubuf[0]; ++ ++ if (len >= 512) { ++ pri_error(pri, "Requested to send UI frame larger than 512 bytes!\n"); ++ return -1; ++ } ++ ++ memset(ubuf, 0, sizeof(ubuf)); ++ h->h.sapi = 0; ++ h->h.ea1 = 0; ++ h->h.ea2 = 1; ++ h->h.tei = pri->tei; ++ h->u.m3 = 0; ++ h->u.m2 = 0; ++ h->u.p_f = 0; /* Poll bit set */ ++ h->u.ft = Q921_FRAMETYPE_U; ++ ++ switch(pri->localtype) { ++ case PRI_NETWORK: ++ h->h.c_r = 1; ++ break; ++ case PRI_CPE: ++ h->h.c_r = 0; ++ break; ++ default: ++ pri_error(pri, "Don't know how to U/A on a type %d node\n", pri->localtype); ++ return -1; ++ } ++ ++ memcpy(h->u.data, buf, len); ++ ++ q921_transmit(pri, h, len + 3); ++ ++ return 0; ++} ++ + int q921_transmit_iframe(struct pri *pri, void *buf, int len, int cr) + { + q921_frame *f, *prev=NULL; +@@ -521,6 +555,10 @@ + pri->txqueue = f; + /* Immediately transmit unless we're in a recovery state, or the window + size is too big */ ++ if (pri->debug & PRI_DEBUG_Q921_DUMP) { ++ pri_message(pri, "TEI/SAPI: %d/%d state %d retran %d busy %d\n", ++ pri->tei, pri->sapi, pri->q921_state, pri->retrans, pri->busy); ++ } + if ((pri->q921_state == Q921_LINK_CONNECTION_ESTABLISHED) && (!pri->retrans && !pri->busy)) { + if (pri->windowlen < pri->window) { + q921_send_queued_iframes(pri); +@@ -564,11 +602,15 @@ + /* Start timer T200 to resend our RR if we don't get it */ + pri->t203_timer = pri_schedule_event(pri, pri->timers[PRI_TIMER_T200], t200_expire, pri); + } else { +- if (pri->debug & PRI_DEBUG_Q921_DUMP) +- pri_message(pri, "T203 counter expired in weird state %d\n", pri->q921_state); ++ if (pri->debug & PRI_DEBUG_Q921_DUMP) { ++ pri_message(pri, ++ "T203 counter expired in weird state %d on pri with SAPI/TEI of %d/%d\n", ++ pri->q921_state, pri->sapi, pri->tei); ++ } + pri->t203_timer = 0; + } + } ++ + static pri_event *q921_handle_iframe(struct pri *pri, q921_i *i, int len) + { + int res; +@@ -622,33 +664,34 @@ + char *buf = malloc(len * 3 + 1); + int buflen = 0; + if (buf) { ++ pri_message(pri, "\n"); + for (x=0;x<len;x++) + buflen += sprintf(buf + buflen, "%02x ", h->raw[x]); +- pri_message(pri, "\n%c [ %s]\n", direction_tag, buf); ++ pri_message(pri, "%c [ %s]\n", direction_tag, buf); + free(buf); + } + } + ++ pri_message(pri, "\n"); + switch (h->h.data[0] & Q921_FRAMETYPE_MASK) { + case 0: + case 2: +- pri_message(pri, "\n%c Informational frame:\n", direction_tag); ++ pri_message(pri, "%c Informational frame:\n", direction_tag); + break; + case 1: +- pri_message(pri, "\n%c Supervisory frame:\n", direction_tag); ++ pri_message(pri, "%c Supervisory frame:\n", direction_tag); + break; + case 3: +- pri_message(pri, "\n%c Unnumbered frame:\n", direction_tag); ++ pri_message(pri, "%c Unnumbered frame:\n", direction_tag); + break; + } + +- pri_message(pri, +-"%c SAPI: %02d C/R: %d EA: %d\n" +-"%c TEI: %03d EA: %d\n", ++ pri_message(pri, "%c SAPI: %02d C/R: %d EA: %d\n", + direction_tag, + h->h.sapi, + h->h.c_r, +- h->h.ea1, ++ h->h.ea1); ++ pri_message(pri, "%c TEI: %03d EA: %d\n", + direction_tag, + h->h.tei, + h->h.ea2); +@@ -658,15 +701,17 @@ + case 2: + /* Informational frame */ + pri_message(pri, +-"%c N(S): %03d 0: %d\n" +-"%c N(R): %03d P: %d\n" +-"%c %d bytes of data\n", ++"%c N(S): %03d 0: %d\n", + direction_tag, + h->i.n_s, +- h->i.ft, ++ h->i.ft); ++ pri_message(pri, ++"%c N(R): %03d P: %d\n", + direction_tag, + h->i.n_r, +- h->i.p_f, ++ h->i.p_f); ++ pri_message(pri, ++"%c %d bytes of data\n", + direction_tag, + len - 4); + break; +@@ -685,17 +730,19 @@ + break; + } + pri_message(pri, +-"%c Zero: %d S: %d 01: %d [ %s ]\n" +-"%c N(R): %03d P/F: %d\n" +-"%c %d bytes of data\n", ++"%c Zero: %d S: %d 01: %d [ %s ]\n", + direction_tag, + h->s.x0, + h->s.ss, + h->s.ft, +- type, ++ type); ++ pri_message(pri, ++"%c N(R): %03d P/F: %d\n", + direction_tag, + h->s.n_r, +- h->s.p_f, ++ h->s.p_f); ++ pri_message(pri, ++"%c %d bytes of data\n", + direction_tag, + len - 4); + break; +@@ -731,14 +778,15 @@ + } + } + pri_message(pri, +-"%c M3: %d P/F: %d M2: %d 11: %d [ %s ]\n" +-"%c %d bytes of data\n", ++"%c M3: %d P/F: %d M2: %d 11: %d [ %s ]\n", + direction_tag, + h->u.m3, + h->u.p_f, + h->u.m2, + h->u.ft, +- type, ++ type); ++ pri_message(pri, ++"%c %d bytes of data\n", + direction_tag, + len - 3); + break; +@@ -783,7 +831,7 @@ + } + } + +-static pri_event *q921_dchannel_up(struct pri *pri) ++pri_event *q921_dchannel_up(struct pri *pri) + { + if (pri->tei == Q921_TEI_PRI) { + q921_reset(pri, 1); +@@ -792,10 +840,8 @@ + } + + /* Stop any SABME retransmissions */ +- if (pri->sabme_timer) { +- pri_schedule_del(pri, pri->sabme_timer); +- pri->sabme_timer = 0; +- } ++ pri_schedule_del(pri, pri->sabme_timer); ++ pri->sabme_timer = 0; + + /* Reset any rejects */ + pri->sentrej = 0; +@@ -805,7 +851,12 @@ + pri_message(pri, DBGHEAD "q921_state now is Q921_LINK_CONNECTION_ESTABLISHED\n", DBGINFO); + pri->q921_state = Q921_LINK_CONNECTION_ESTABLISHED; + ++ /* Ensure that we do not have T200 or T203 running when the link comes up */ ++ pri_schedule_del(pri, pri->t200_timer); ++ pri->t200_timer = 0; ++ + /* Start the T203 timer */ ++ pri_schedule_del(pri, pri->t203_timer); + pri->t203_timer = pri_schedule_event(pri, pri->timers[PRI_TIMER_T203], t203_expire, pri); + + /* Notify Layer 3 */ +@@ -818,7 +869,7 @@ + return &pri->ev; + } + +-static pri_event *q921_dchannel_down(struct pri *pri) ++pri_event *q921_dchannel_down(struct pri *pri) + { + /* Reset counters, reset sabme timer etc */ + q921_reset(pri, 1); +@@ -826,7 +877,7 @@ + /* Notify Layer 3 */ + q931_dl_indication(pri, PRI_EVENT_DCHAN_DOWN); + +- /* Report event that D-Channel is now up */ ++ /* Report event that D-Channel is now down */ + pri->ev.gen.e = PRI_EVENT_DCHAN_DOWN; + return &pri->ev; + } +@@ -842,12 +893,9 @@ + pri->v_na = 0; + pri->window = pri->timers[PRI_TIMER_K]; + pri->windowlen = 0; +- if (pri->sabme_timer) +- pri_schedule_del(pri, pri->sabme_timer); +- if (pri->t203_timer) +- pri_schedule_del(pri, pri->t203_timer); +- if (pri->t200_timer) +- pri_schedule_del(pri, pri->t200_timer); ++ pri_schedule_del(pri, pri->sabme_timer); ++ pri_schedule_del(pri, pri->t203_timer); ++ pri_schedule_del(pri, pri->t200_timer); + pri->sabme_timer = 0; + pri->sabme_count = 0; + pri->t203_timer = 0; +@@ -879,8 +927,14 @@ + static pri_event *q921_receive_MDL(struct pri *pri, q921_u *h, int len) + { + int ri; +- struct pri *sub; ++ struct pri *sub = pri; + int tei; ++ ++ if (!BRI_NT_PTMP(pri) && !BRI_TE_PTMP(pri)) { ++ pri_error(pri, "Received MDL/TEI managemement message, but configured for mode other than PTMP!\n"); ++ return NULL; ++ } ++ + if (pri->debug & PRI_DEBUG_Q921_STATE) + pri_message(pri, "Received MDL message\n"); + if (h->data[0] != 0x0f) { +@@ -895,16 +949,19 @@ + tei = (h->data[4] >> 1); + switch(h->data[3]) { + case Q921_TEI_IDENTITY_REQUEST: ++ if (!BRI_NT_PTMP(pri)) { ++ return NULL; ++ } ++ + if (tei != 127) { + pri_error(pri, "Received TEI identity request with invalid TEI %d\n", tei); + q921_send_tei(pri, Q921_TEI_IDENTITY_DENIED, ri, tei, 1); + } +- /* Go to master */ +- for (sub = pri; sub->master; sub = sub->master); + tei = 64; +- while(sub->subchannel) { +- if(sub->subchannel->tei == tei) ++ while (sub->subchannel) { ++ if (sub->subchannel->tei == tei) + ++tei; ++ sub = sub->subchannel; + } + sub->subchannel = __pri_new_tei(-1, pri->localtype, pri->switchtype, pri, NULL, NULL, NULL, tei, 1); + if (!sub->subchannel) { +@@ -914,14 +971,15 @@ + q921_send_tei(pri, Q921_TEI_IDENTITY_ASSIGNED, ri, tei, 1); + break; + case Q921_TEI_IDENTITY_ASSIGNED: ++ if (!BRI_TE_PTMP(pri)) ++ return NULL; ++ + if (ri != pri->ri) { + pri_message(pri, "TEI assignment received for invalid Ri %02x (our is %02x)\n", ri, pri->ri); + return NULL; + } +- if (pri->t202_timer) { +- pri_schedule_del(pri, pri->t202_timer); +- pri->t202_timer = 0; +- } ++ pri_schedule_del(pri, pri->t202_timer); ++ pri->t202_timer = 0; + if (pri->subchannel && (pri->subchannel->tei == tei)) { + pri_error(pri, "TEI already assigned (new is %d, current is %d)\n", tei, pri->subchannel->tei); + q921_tei_release_and_reacquire(pri); +@@ -937,6 +995,8 @@ + pri->q921_state = Q921_TEI_ASSIGNED; + break; + case Q921_TEI_IDENTITY_CHECK_REQUEST: ++ if (!BRI_TE_PTMP(pri)) ++ return NULL; + /* We're assuming one TEI per PRI in TE PTMP mode */ + + /* If no subchannel (TEI) ignore */ +@@ -949,6 +1009,8 @@ + + break; + case Q921_TEI_IDENTITY_REMOVE: ++ if (!BRI_TE_PTMP(pri)) ++ return NULL; + /* XXX: Assuming multiframe mode has been disconnected already */ + if (!pri->subchannel) + return NULL; +@@ -990,6 +1052,10 @@ + pri_error(pri, "!! Received short I-frame (expected 4, got %d)\n", len); + break; + } ++ ++ /* T203 is rescheduled only on reception of I frames or S frames */ ++ reschedule_t203(pri); ++ + return q921_handle_iframe(pri, &h->i, len); + break; + case 1: +@@ -1001,6 +1067,10 @@ + pri_error(pri, "!! Received short S-frame (expected 4, got %d)\n", len); + break; + } ++ ++ /* T203 is rescheduled only on reception of I frames or S frames */ ++ reschedule_t203(pri); ++ + switch(h->s.ss) { + case 0: + /* Receiver Ready */ +@@ -1091,10 +1161,8 @@ + } + } + /* Reset t200 timer if it was somehow going */ +- if (pri->t200_timer) { +- pri_schedule_del(pri, pri->t200_timer); +- pri->t200_timer = 0; +- } ++ pri_schedule_del(pri, pri->t200_timer); ++ pri->t200_timer = 0; + /* Reset and restart t203 timer */ + reschedule_t203(pri); + } +@@ -1152,7 +1220,9 @@ + /* Acknowledge */ + q921_send_ua(pri, h->u.p_f); + ev = q921_dchannel_down(pri); +- q921_restart(pri, 0); ++ if (BRI_TE_PTMP(pri) || PRI_PTP(pri)) { ++ q921_restart(pri, PRI_PTP(pri) ? 1 : 0); ++ } + return ev; + case 3: + if (h->u.m2 == 3) { +@@ -1212,6 +1282,34 @@ + return NULL; + } + ++static pri_event *q921_handle_unmatched_frame(struct pri *pri, q921_h *h, int len) ++{ ++ pri = PRI_MASTER(pri); ++ ++ if (h->h.tei < 64) { ++ pri_error(pri, "Do not support manual TEI range. Discarding\n"); ++ return NULL; ++ } ++ ++ if (h->h.sapi != Q921_SAPI_CALL_CTRL) { ++ pri_error(pri, "Message with SAPI other than CALL CTRL is discarded\n"); ++ return NULL; ++ } ++ ++ if (pri->debug & PRI_DEBUG_Q921_DUMP) { ++ pri_message(pri, ++ "Could not find candidate subchannel for received frame with SAPI/TEI of %d/%d.\n", ++ h->h.sapi, h->h.tei); ++ pri_message(pri, "Sending TEI release, in order to re-establish TEI state\n"); ++ } ++ ++ /* Q.921 says we should send the remove message twice, in case of link corruption */ ++ q921_send_tei(pri, Q921_TEI_IDENTITY_REMOVE, 0, h->h.tei, 1); ++ q921_send_tei(pri, Q921_TEI_IDENTITY_REMOVE, 0, h->h.tei, 1); ++ ++ return NULL; ++} ++ + static pri_event *__q921_receive(struct pri *pri, q921_h *h, int len) + { + pri_event *ev; +@@ -1225,26 +1323,22 @@ + if (h->h.ea1 || !(h->h.ea2)) + return NULL; + +-#if 0 /* Will be rejected by subchannel analyzis */ +- /* Check for broadcasts - not yet handled */ +- if (h->h.tei == Q921_TEI_GROUP) +- return NULL; +-#endif +- + if (!((h->h.sapi == pri->sapi) && ((h->h.tei == pri->tei) || (h->h.tei == Q921_TEI_GROUP)))) { + /* Check for SAPIs we don't yet handle */ + /* If it's not us, try any subchannels we have */ + if (pri->subchannel) + return q921_receive(pri->subchannel, h, len + 2); + else { +- return NULL; ++ /* This means we couldn't find a candidate subchannel for it... ++ * Time for some corrective action */ ++ ++ return q921_handle_unmatched_frame(pri, h, len); + } + + } + if (pri->debug & PRI_DEBUG_Q921_DUMP) + pri_message(pri, "Handling message for SAPI/TEI=%d/%d\n", h->h.sapi, h->h.tei); + ev = __q921_receive_qualified(pri, h, len); +- reschedule_t203(pri); + return ev; + } + + +Property changes on: . +___________________________________________________________________ +Added: reviewboard:url + + https://reviewboard.asterisk.org + |