diff options
author | Timo Teräs <timo.teras@iki.fi> | 2011-07-07 16:14:24 +0300 |
---|---|---|
committer | Timo Teräs <timo.teras@iki.fi> | 2011-07-07 16:14:24 +0300 |
commit | 491adc1d879b67ff4095fd32eb376f9e580e97ad (patch) | |
tree | 45caf5b64a48186352e8fe7e502c9c9bd296750f /main | |
parent | 05a00bd3b1689298d2ad59c797d43424c18ea855 (diff) | |
download | aports-491adc1d879b67ff4095fd32eb376f9e580e97ad.tar.bz2 aports-491adc1d879b67ff4095fd32eb376f9e580e97ad.tar.xz |
main/libpri: upgrade to 1.4.12
* remove unneeded patches
Diffstat (limited to 'main')
-rw-r--r-- | main/libpri/APKBUILD | 7 | ||||
-rw-r--r-- | main/libpri/libpri-1.4-r1357.patch | 39961 | ||||
-rw-r--r-- | main/libpri/libpri-cflags.patch | 21 |
3 files changed, 4 insertions, 39985 deletions
diff --git a/main/libpri/APKBUILD b/main/libpri/APKBUILD index aec4f46831..fef96994fe 100644 --- a/main/libpri/APKBUILD +++ b/main/libpri/APKBUILD @@ -1,8 +1,8 @@ # Contributor: Timo Teras <timo.teras@iki.fi> # Maintainer: Timo Teras <timo.teras@iki.fi> pkgname=libpri -pkgver=1.4.12_beta3 -_pkgver=1.4.12-beta3 +pkgver=1.4.12 +_pkgver=$pkgver pkgrel=0 pkgdesc="Primary Rate ISDN (PRI) library" url="http://www.asterisk.orig" @@ -30,4 +30,5 @@ package() { make LDCONFIG="echo" INSTALL_PREFIX="$pkgdir" install } -md5sums="9cbbaa009264b76af930529e8aae8cfd libpri-1.4.12-beta3.tar.gz" +md5sums="9de39b866f049cc20a4f431cc6245166 libpri-1.4.12.tar.gz +8df2ca48ce7db4f1a8604b0904bc9394 libpri-cflags.patch" diff --git a/main/libpri/libpri-1.4-r1357.patch b/main/libpri/libpri-1.4-r1357.patch deleted file mode 100644 index be827eebb7..0000000000 --- a/main/libpri/libpri-1.4-r1357.patch +++ /dev/null @@ -1,39961 +0,0 @@ -=================================================================== ---- 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 - diff --git a/main/libpri/libpri-cflags.patch b/main/libpri/libpri-cflags.patch deleted file mode 100644 index 7bf31c7ac1..0000000000 --- a/main/libpri/libpri-cflags.patch +++ /dev/null @@ -1,21 +0,0 @@ -Index: Makefile -=================================================================== ---- a/Makefile (revision 650) -+++ b/Makefile (working copy) -@@ -43,7 +43,7 @@ - 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) -+CFLAGS+=-Wall -Werror -Wstrict-prototypes -Wmissing-prototypes -g -fPIC $(ALERTING) $(LIBPRI_COUNTERS) - INSTALL_PREFIX=$(DESTDIR) - INSTALL_BASE=/usr - libdir?=$(INSTALL_BASE)/lib -@@ -102,7 +102,6 @@ - ifneq (${OSARCH},SunOS) - install -m 644 libpri.h $(INSTALL_PREFIX)$(INSTALL_BASE)/include - install -m 755 $(DYNAMIC_LIBRARY) $(INSTALL_PREFIX)$(libdir) -- if [ -x /usr/sbin/sestatus ] && ( /usr/sbin/sestatus | grep "SELinux status:" | grep -q "enabled"); then /sbin/restorecon -v $(INSTALL_PREFIX)$(libdir)/$(DYNAMIC_LIBRARY); fi - ( cd $(INSTALL_PREFIX)$(libdir) ; ln -sf libpri.so.$(SONAME) libpri.so) - install -m 644 $(STATIC_LIBRARY) $(INSTALL_PREFIX)$(libdir) - if test $$(id -u) = 0; then $(LDCONFIG) $(LDCONFIG_FLAGS) $(INSTALL_PREFIX)$(libdir); fi |