2001-05-13 00:57:53 +08:00
|
|
|
/*
|
|
|
|
* libpri: An implementation of Primary Rate ISDN
|
|
|
|
*
|
2008-08-06 06:18:12 +08:00
|
|
|
* Written by Mark Spencer <markster@digium.com>
|
2001-05-13 00:57:53 +08:00
|
|
|
*
|
2008-08-06 06:18:12 +08:00
|
|
|
* Copyright (C) 2001, Digium, Inc.
|
2001-05-13 00:57:53 +08:00
|
|
|
* All Rights Reserved.
|
2008-08-06 06:18:12 +08:00
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* 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.
|
2001-05-13 00:57:53 +08:00
|
|
|
*
|
2008-08-06 06:18:12 +08:00
|
|
|
* 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.
|
2001-05-13 00:57:53 +08:00
|
|
|
*
|
2008-08-06 06:18:12 +08:00
|
|
|
* 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.
|
2001-05-13 00:57:53 +08:00
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef _PRI_INTERNAL_H
|
|
|
|
#define _PRI_INTERNAL_H
|
|
|
|
|
2007-10-22 23:10:06 +08:00
|
|
|
#include <stddef.h>
|
2001-05-13 00:57:53 +08:00
|
|
|
#include <sys/time.h>
|
2009-09-03 04:19:40 +08:00
|
|
|
#include "pri_q921.h"
|
|
|
|
#include "pri_q931.h"
|
2001-05-13 00:57:53 +08:00
|
|
|
|
2009-04-22 06:08:45 +08:00
|
|
|
#define ARRAY_LEN(arr) (sizeof(arr) / sizeof((arr)[0]))
|
|
|
|
|
2006-06-07 06:06:52 +08:00
|
|
|
#define DBGHEAD __FILE__ ":%d %s: "
|
|
|
|
#define DBGINFO __LINE__,__PRETTY_FUNCTION__
|
|
|
|
|
2009-11-14 08:20:53 +08:00
|
|
|
/* Forward declare some structs */
|
|
|
|
struct apdu_event;
|
|
|
|
|
2001-05-13 00:57:53 +08:00
|
|
|
struct pri_sched {
|
|
|
|
struct timeval when;
|
|
|
|
void (*callback)(void *data);
|
|
|
|
void *data;
|
|
|
|
};
|
|
|
|
|
2009-06-27 03:43:13 +08:00
|
|
|
/*! Maximum number of scheduled events active at the same time. */
|
2001-05-13 00:57:53 +08:00
|
|
|
#define MAX_SCHED 128
|
|
|
|
|
2009-11-21 07:42:40 +08:00
|
|
|
/*! Maximum number of facility ie's to handle per incoming message. */
|
|
|
|
#define MAX_FACILITY_IES 8
|
|
|
|
|
2009-11-21 10:40:23 +08:00
|
|
|
/*! 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];
|
|
|
|
};
|
|
|
|
|
2009-04-22 06:08:45 +08:00
|
|
|
/*! \brief D channel controller structure */
|
2001-05-13 00:57:53 +08:00
|
|
|
struct pri {
|
|
|
|
int fd; /* File descriptor for D-Channel */
|
2005-03-17 23:46:23 +08:00
|
|
|
pri_io_cb read_func; /* Read data callback */
|
|
|
|
pri_io_cb write_func; /* Write data callback */
|
|
|
|
void *userdata;
|
2009-11-21 10:40:23 +08:00
|
|
|
/*! Accumulated pri_message() line. (Valid in master record only) */
|
|
|
|
struct pri_msg_line *msg_line;
|
2004-06-05 14:50:55 +08:00
|
|
|
struct pri *subchannel; /* Sub-channel if appropriate */
|
|
|
|
struct pri *master; /* Master channel if appropriate */
|
2001-05-13 00:57:53 +08:00
|
|
|
struct pri_sched pri_sched[MAX_SCHED]; /* Scheduled events */
|
|
|
|
int debug; /* Debug stuff */
|
|
|
|
int state; /* State of D-channel */
|
|
|
|
int switchtype; /* Switch type */
|
2004-06-26 12:37:09 +08:00
|
|
|
int nsf; /* Network-Specific Facility (if any) */
|
2001-05-13 00:57:53 +08:00
|
|
|
int localtype; /* Local network type (unknown, network, cpe) */
|
|
|
|
int remotetype; /* Remote network type (unknown, network, cpe) */
|
2004-06-05 14:50:55 +08:00
|
|
|
|
|
|
|
int sapi;
|
|
|
|
int tei;
|
|
|
|
int protodisc;
|
2008-07-12 00:24:01 +08:00
|
|
|
unsigned int bri:1;
|
2008-07-23 06:36:25 +08:00
|
|
|
unsigned int acceptinbanddisconnect:1; /* Should we allow inband progress after DISCONNECT? */
|
2009-10-24 03:50:45 +08:00
|
|
|
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. */
|
|
|
|
|
2010-01-14 03:37:59 +08:00
|
|
|
/* MDL variables */
|
|
|
|
int mdl_error;
|
|
|
|
int mdl_error_state;
|
|
|
|
int mdl_timer;
|
|
|
|
int mdl_free_me;
|
|
|
|
|
2001-05-13 00:57:53 +08:00
|
|
|
/* Q.921 State */
|
|
|
|
int q921_state;
|
2010-01-14 03:37:59 +08:00
|
|
|
int k;
|
|
|
|
int RC;
|
|
|
|
int peer_rx_busy:1;
|
|
|
|
int own_rx_busy:1;
|
|
|
|
int acknowledge_pending:1;
|
|
|
|
int reject_exception:1;
|
|
|
|
|
2001-05-13 00:57:53 +08:00
|
|
|
int v_s; /* Next N(S) for transmission */
|
|
|
|
int v_a; /* Last acknowledged frame */
|
|
|
|
int v_r; /* Next frame expected to be received */
|
|
|
|
|
|
|
|
int cref; /* Next call reference value */
|
|
|
|
|
2010-01-14 03:37:59 +08:00
|
|
|
int l3initiated;
|
2001-05-13 00:57:53 +08:00
|
|
|
|
|
|
|
/* Various timers */
|
|
|
|
int t203_timer; /* Max idle time */
|
2008-05-08 03:51:44 +08:00
|
|
|
int t202_timer;
|
|
|
|
int n202_counter;
|
|
|
|
int ri;
|
2001-05-13 00:57:53 +08:00
|
|
|
int t200_timer; /* T-200 retransmission timer */
|
2004-10-02 22:55:20 +08:00
|
|
|
/* All ISDN Timer values */
|
2009-06-27 03:43:13 +08:00
|
|
|
int timers[PRI_MAX_TIMERS];
|
2004-10-02 22:55:20 +08:00
|
|
|
|
2001-05-13 00:57:53 +08:00
|
|
|
/* Used by scheduler */
|
|
|
|
struct timeval tv;
|
2001-11-26 10:54:32 +08:00
|
|
|
int schedev;
|
2001-05-13 00:57:53 +08:00
|
|
|
pri_event ev; /* Static event thingy */
|
2009-08-19 07:53:32 +08:00
|
|
|
/*! Subcommands for static event thingy. */
|
|
|
|
struct pri_subcommands subcmds;
|
2001-05-13 00:57:53 +08:00
|
|
|
|
|
|
|
/* Q.921 Re-transmission queue */
|
|
|
|
struct q921_frame *txqueue;
|
|
|
|
|
|
|
|
/* Q.931 calls */
|
2004-06-07 11:33:51 +08:00
|
|
|
q931_call **callpool;
|
|
|
|
q931_call *localpool;
|
2003-05-16 03:35:05 +08:00
|
|
|
|
2009-11-11 03:27:34 +08:00
|
|
|
/*!
|
|
|
|
* \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;
|
|
|
|
|
2003-05-16 03:35:05 +08:00
|
|
|
/* do we do overlap dialing */
|
|
|
|
int overlapdial;
|
2004-03-15 13:53:25 +08:00
|
|
|
|
2009-04-14 23:05:21 +08:00
|
|
|
/* do we support SERVICE messages */
|
|
|
|
int service_message_support;
|
|
|
|
|
2008-10-18 00:13:42 +08:00
|
|
|
/* do not skip channel 16 */
|
|
|
|
int chan_mapping_logical;
|
|
|
|
|
2004-03-15 13:53:25 +08:00
|
|
|
#ifdef LIBPRI_COUNTERS
|
|
|
|
/* q921/q931 packet counters */
|
|
|
|
unsigned int q921_txcount;
|
|
|
|
unsigned int q921_rxcount;
|
|
|
|
unsigned int q931_txcount;
|
|
|
|
unsigned int q931_rxcount;
|
|
|
|
#endif
|
2005-02-04 06:14:44 +08:00
|
|
|
|
2009-11-14 06:59:35 +08:00
|
|
|
short last_invoke; /* Last ROSE invoke ID (Valid in master record only) */
|
2005-05-12 22:48:00 +08:00
|
|
|
unsigned char sendfacility;
|
2009-11-21 07:42:40 +08:00
|
|
|
|
|
|
|
/*! 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;
|
2001-05-13 00:57:53 +08:00
|
|
|
};
|
|
|
|
|
2009-08-19 07:53:32 +08:00
|
|
|
/*! \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.
|
|
|
|
*/
|
2009-10-23 00:16:50 +08:00
|
|
|
unsigned char data[PRI_MAX_SUBADDRESS_LEN];
|
2009-08-19 07:53:32 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
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;
|
|
|
|
};
|
|
|
|
|
2009-04-22 06:08:45 +08:00
|
|
|
/*! \brief New call setup parameter structure */
|
2004-06-16 23:33:58 +08:00
|
|
|
struct pri_sr {
|
|
|
|
int transmode;
|
|
|
|
int channel;
|
|
|
|
int exclusive;
|
|
|
|
int nonisdn;
|
2009-08-19 07:53:32 +08:00
|
|
|
struct q931_party_redirecting redirecting;
|
|
|
|
struct q931_party_id caller;
|
|
|
|
struct q931_party_address called;
|
2004-06-16 23:33:58 +08:00
|
|
|
int userl1;
|
|
|
|
int numcomplete;
|
2009-09-01 01:24:19 +08:00
|
|
|
int cis_call;
|
|
|
|
int cis_auto_disconnect;
|
2005-12-04 23:32:46 +08:00
|
|
|
const char *useruserinfo;
|
2009-10-24 03:50:45 +08:00
|
|
|
const char *keypad_digits;
|
2006-01-20 02:22:06 +08:00
|
|
|
int transferable;
|
2009-06-26 02:53:38 +08:00
|
|
|
int reversecharge;
|
2004-06-16 23:33:58 +08:00
|
|
|
};
|
|
|
|
|
2004-06-05 14:50:55 +08:00
|
|
|
/* Internal switch types */
|
2005-03-02 23:56:11 +08:00
|
|
|
#define PRI_SWITCH_GR303_EOC_PATH 19
|
|
|
|
#define PRI_SWITCH_GR303_TMC_SWITCHING 20
|
|
|
|
|
2009-10-24 03:50:45 +08:00
|
|
|
#define Q931_MAX_TEI 8
|
|
|
|
|
2009-08-19 07:53:32 +08:00
|
|
|
/*! \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
|
|
|
|
};
|
2004-10-28 04:43:23 +08:00
|
|
|
|
2009-10-24 03:50:45 +08:00
|
|
|
/*! 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,
|
|
|
|
};
|
|
|
|
|
2009-08-19 07:53:32 +08:00
|
|
|
/* q931_call datastructure */
|
2004-10-28 04:43:23 +08:00
|
|
|
struct q931_call {
|
|
|
|
struct pri *pri; /* PRI */
|
2005-01-17 20:58:05 +08:00
|
|
|
int cr; /* Call Reference */
|
2004-10-28 04:43:23 +08:00
|
|
|
q931_call *next;
|
|
|
|
/* Slotmap specified (bitmap of channels 31/24-1) (Channel Identifier IE) (-1 means not specified) */
|
|
|
|
int slotmap;
|
|
|
|
/* An explicit channel (Channel Identifier IE) (-1 means not specified) */
|
|
|
|
int channelno;
|
|
|
|
/* An explicit DS1 (-1 means not specified) */
|
|
|
|
int ds1no;
|
2005-05-19 22:06:43 +08:00
|
|
|
/* Whether or not the ds1 is explicitly identified or implicit. If implicit
|
|
|
|
the bchan is on the same span as the current active dchan (NFAS) */
|
|
|
|
int ds1explicit;
|
2004-10-28 04:43:23 +08:00
|
|
|
/* Channel flags (0 means none retrieved) */
|
|
|
|
int chanflags;
|
|
|
|
|
|
|
|
int alive; /* Whether or not the call is alive */
|
|
|
|
int acked; /* Whether setup has been acked or not */
|
2005-01-17 20:58:05 +08:00
|
|
|
int sendhangupack; /* Whether or not to send a hangup ack */
|
2004-10-28 04:43:23 +08:00
|
|
|
int proc; /* Whether we've sent a call proceeding / alerting */
|
|
|
|
|
2005-01-17 20:58:05 +08:00
|
|
|
int ri; /* Restart Indicator (Restart Indicator IE) */
|
2004-10-28 04:43:23 +08:00
|
|
|
|
|
|
|
/* Bearer Capability */
|
|
|
|
int transcapability;
|
|
|
|
int transmoderate;
|
|
|
|
int transmultiple;
|
|
|
|
int userl1;
|
|
|
|
int userl2;
|
|
|
|
int userl3;
|
|
|
|
int rateadaption;
|
2009-08-19 07:53:32 +08:00
|
|
|
|
2009-09-01 01:24:19 +08:00
|
|
|
/*!
|
|
|
|
* \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;
|
2004-10-28 04:43:23 +08:00
|
|
|
|
|
|
|
int progcode; /* Progress coding */
|
|
|
|
int progloc; /* Progress Location */
|
|
|
|
int progress; /* Progress indicator */
|
2005-01-17 20:58:05 +08:00
|
|
|
int progressmask; /* Progress Indicator bitmask */
|
2004-10-28 04:43:23 +08:00
|
|
|
|
2009-08-19 07:53:32 +08:00
|
|
|
int notify; /* Notification indicator. */
|
2004-10-28 04:43:23 +08:00
|
|
|
|
|
|
|
int causecode; /* Cause Coding */
|
|
|
|
int causeloc; /* Cause Location */
|
|
|
|
int cause; /* Cause of clearing */
|
|
|
|
|
2009-09-03 04:19:40 +08:00
|
|
|
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 */
|
2005-03-02 23:56:11 +08:00
|
|
|
|
2004-10-28 04:43:23 +08:00
|
|
|
int ani2; /* ANI II */
|
2009-08-19 07:53:32 +08:00
|
|
|
|
|
|
|
/*! 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;
|
2004-10-28 04:43:23 +08:00
|
|
|
int nonisdn;
|
|
|
|
int complete; /* no more digits coming */
|
2005-01-17 20:58:05 +08:00
|
|
|
int newcall; /* if the received message has a new call reference value */
|
2004-10-28 04:43:23 +08:00
|
|
|
|
|
|
|
int retranstimer; /* Timer for retransmitting DISC */
|
|
|
|
int t308_timedout; /* Whether t308 timed out once */
|
2005-04-07 03:42:41 +08:00
|
|
|
|
2009-08-19 07:53:32 +08:00
|
|
|
struct q931_party_redirecting redirecting;
|
2005-04-07 03:42:41 +08:00
|
|
|
|
2009-08-19 07:53:32 +08:00
|
|
|
/*! \brief Incoming call transfer state. */
|
|
|
|
enum INCOMING_CT_STATE incoming_ct_state;
|
2009-10-24 03:50:45 +08:00
|
|
|
/*! 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. */
|
2009-11-21 07:42:40 +08:00
|
|
|
/*! 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;
|
2004-10-28 04:43:23 +08:00
|
|
|
|
2005-07-12 10:33:25 +08:00
|
|
|
int useruserprotocoldisc;
|
2004-10-28 04:43:23 +08:00
|
|
|
char useruserinfo[256];
|
2005-04-05 11:55:58 +08:00
|
|
|
|
|
|
|
long aoc_units; /* Advice of Charge Units */
|
2005-03-02 23:56:11 +08:00
|
|
|
|
|
|
|
struct apdu_event *apdus; /* APDU queue for call */
|
2006-01-20 02:22:06 +08:00
|
|
|
|
2009-08-19 07:53:32 +08:00
|
|
|
int transferable; /* RLT call is transferable */
|
2006-01-20 02:22:06 +08:00
|
|
|
unsigned int rlt_call_id; /* RLT call id */
|
2008-05-08 03:51:44 +08:00
|
|
|
|
|
|
|
/* Bridged call info */
|
2009-05-08 00:21:24 +08:00
|
|
|
q931_call *bridged_call; /* Pointer to other leg of bridged call (Used by Q.SIG when eliminating tromboned calls) */
|
2009-04-14 23:05:21 +08:00
|
|
|
|
|
|
|
int changestatus; /* SERVICE message changestatus */
|
2009-06-26 02:53:38 +08:00
|
|
|
int reversecharge; /* Reverse charging indication:
|
|
|
|
-1 - No reverse charging
|
|
|
|
1 - Reverse charging
|
|
|
|
0,2-7 - Reserved for future use */
|
2010-01-14 03:37:59 +08:00
|
|
|
/*! \brief TEI associated with call */
|
|
|
|
int tei;
|
2009-10-24 03:50:45 +08:00
|
|
|
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;
|
2004-10-28 04:43:23 +08:00
|
|
|
};
|
|
|
|
|
2009-11-11 03:27:34 +08:00
|
|
|
/*! D channel control structure with associated dummy call reference record. */
|
|
|
|
struct d_ctrl_dummy {
|
2009-11-21 10:40:23 +08:00
|
|
|
/*! D channel control structure. Must be first in the structure. */
|
2009-11-11 03:27:34 +08:00
|
|
|
struct pri ctrl;
|
|
|
|
/*! Dummy call reference call record. */
|
|
|
|
struct q931_call dummy_call;
|
|
|
|
};
|
|
|
|
|
2001-05-13 00:57:53 +08:00
|
|
|
extern int pri_schedule_event(struct pri *pri, int ms, void (*function)(void *data), void *data);
|
|
|
|
|
2001-11-26 10:54:32 +08:00
|
|
|
extern pri_event *pri_schedule_run(struct pri *pri);
|
2001-05-13 00:57:53 +08:00
|
|
|
|
|
|
|
extern void pri_schedule_del(struct pri *pri, int ev);
|
|
|
|
|
|
|
|
extern pri_event *pri_mkerror(struct pri *pri, char *errstr);
|
|
|
|
|
2009-11-21 10:40:23 +08:00
|
|
|
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)));
|
2003-02-12 21:59:23 +08:00
|
|
|
|
2005-06-22 06:47:39 +08:00
|
|
|
void libpri_copy_string(char *dst, const char *src, size_t size);
|
|
|
|
|
2008-05-08 03:51:44 +08:00
|
|
|
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);
|
|
|
|
|
2009-11-11 03:27:34 +08:00
|
|
|
void q931_init_call_record(struct pri *ctrl, struct q931_call *call, int cr);
|
|
|
|
|
2009-08-19 07:53:32 +08:00
|
|
|
void q931_party_name_init(struct q931_party_name *name);
|
|
|
|
void q931_party_number_init(struct q931_party_number *number);
|
2009-10-23 00:16:50 +08:00
|
|
|
void q931_party_subaddress_init(struct q931_party_subaddress *subaddr);
|
2009-08-19 07:53:32 +08:00
|
|
|
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);
|
|
|
|
|
2009-10-24 07:47:58 +08:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2009-08-19 07:53:32 +08:00
|
|
|
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);
|
2009-10-23 00:16:50 +08:00
|
|
|
int q931_party_subaddress_cmp(const struct q931_party_subaddress *left, const struct q931_party_subaddress *right);
|
2009-08-19 07:53:32 +08:00
|
|
|
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);
|
2009-10-23 00:16:50 +08:00
|
|
|
void q931_party_subaddress_copy_to_pri(struct pri_party_subaddress *pri_subaddress, const struct q931_party_subaddress *q931_subaddress);
|
2009-08-19 07:53:32 +08:00
|
|
|
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);
|
|
|
|
|
2009-10-13 01:17:48 +08:00
|
|
|
const char *q931_call_state_str(enum Q931_CALL_STATE callstate);
|
2009-11-11 05:51:26 +08:00
|
|
|
const char *msg2str(int msg);
|
2009-08-19 07:53:32 +08:00
|
|
|
|
2009-10-13 01:17:48 +08:00
|
|
|
int q931_is_ptmp(const struct pri *ctrl);
|
2009-10-24 03:50:45 +08:00
|
|
|
int q931_master_pass_event(struct pri *ctrl, struct q931_call *subcall, int msg_type);
|
2009-08-19 07:53:32 +08:00
|
|
|
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);
|
|
|
|
|
2009-10-24 03:50:45 +08:00
|
|
|
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));
|
|
|
|
}
|
|
|
|
|
2010-01-14 03:37:59 +08:00
|
|
|
static inline int NT_MODE(struct pri *mypri)
|
|
|
|
{
|
|
|
|
struct pri *pri;
|
|
|
|
|
|
|
|
pri = PRI_MASTER(mypri);
|
|
|
|
|
|
|
|
return pri->localtype == PRI_NETWORK;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline int TE_MODE(struct pri *mypri)
|
|
|
|
{
|
|
|
|
struct pri *pri;
|
|
|
|
|
|
|
|
pri = PRI_MASTER(mypri);
|
|
|
|
|
|
|
|
return pri->localtype == PRI_CPE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline int PTP_MODE(struct pri *mypri)
|
|
|
|
{
|
|
|
|
struct pri *pri;
|
|
|
|
|
|
|
|
pri = PRI_MASTER(mypri);
|
|
|
|
|
|
|
|
return pri->tei == Q921_TEI_PRI;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline int PTMP_MODE(struct pri *mypri)
|
2009-11-11 08:22:47 +08:00
|
|
|
{
|
|
|
|
struct pri *pri;
|
|
|
|
|
|
|
|
pri = PRI_MASTER(mypri);
|
|
|
|
|
2010-01-14 03:37:59 +08:00
|
|
|
return pri->tei == Q921_TEI_GROUP;
|
2009-11-11 08:22:47 +08:00
|
|
|
}
|
|
|
|
|
2009-11-11 03:27:34 +08:00
|
|
|
#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;
|
|
|
|
}
|
|
|
|
|
2001-05-13 00:57:53 +08:00
|
|
|
#endif
|