2007-05-22 22:07:05 +00:00
/*
* Copyright ( c ) 2007 , Anthony Minessale II
* All rights reserved .
*
* Redistribution and use in source and binary forms , with or without
* modification , are permitted provided that the following conditions
* are met :
*
* * Redistributions of source code must retain the above copyright
* notice , this list of conditions and the following disclaimer .
*
* * Redistributions in binary form must reproduce the above copyright
* notice , this list of conditions and the following disclaimer in the
* documentation and / or other materials provided with the distribution .
*
* * Neither the name of the original author ; nor the names of any contributors
* may be used to endorse or promote products derived from this software
* without specific prior written permission .
*
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* " AS IS " AND ANY EXPRESS OR IMPLIED WARRANTIES , INCLUDING , BUT NOT
* LIMITED TO , THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
* A PARTICULAR PURPOSE ARE DISCLAIMED . IN NO EVENT SHALL THE COPYRIGHT OWNER
* OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT , INDIRECT , INCIDENTAL , SPECIAL ,
* EXEMPLARY , OR CONSEQUENTIAL DAMAGES ( INCLUDING , BUT NOT LIMITED TO ,
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES ; LOSS OF USE , DATA , OR
* PROFITS ; OR BUSINESS INTERRUPTION ) HOWEVER CAUSED AND ON ANY THEORY OF
* LIABILITY , WHETHER IN CONTRACT , STRICT LIABILITY , OR TORT ( INCLUDING
* NEGLIGENCE OR OTHERWISE ) ARISING IN ANY WAY OUT OF THE USE OF THIS
* SOFTWARE , EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE .
*/
# include "openzap.h"
# include "zap_isdn.h"
# include "Q931.h"
# include "Q921.h"
2007-05-23 19:47:35 +00:00
# ifdef WIN32
# include <windows.h>
# else
# include <sys/time.h>
# endif
2007-05-23 18:25:23 +00:00
# define LINE "--------------------------------------------------------------------------------"
2007-11-21 22:08:47 +00:00
//#define IODEBUG
2007-05-22 22:07:05 +00:00
2008-08-22 16:55:01 +00:00
/* helper macros */
# define ZAP_SPAN_IS_BRI(x) ((x)->trunk_type == ZAP_TRUNK_BRI || (x)->trunk_type == ZAP_TRUNK_BRI_PTMP)
# define ZAP_SPAN_IS_NT(x) (((zap_isdn_data_t *)(x)->signal_data)->mode == Q921_NT)
2008-02-18 17:17:38 +00:00
static L2ULONG zap_time_now ( void )
2007-05-23 19:47:35 +00:00
{
2007-06-25 16:00:24 +00:00
return ( L2ULONG ) zap_current_time_in_ms ( ) ;
2007-05-23 19:47:35 +00:00
}
2007-06-14 04:44:44 +00:00
static ZIO_CHANNEL_OUTGOING_CALL_FUNCTION ( isdn_outgoing_call )
{
zap_status_t status = ZAP_SUCCESS ;
2008-05-21 18:58:14 +00:00
zap_set_flag ( zchan , ZAP_CHANNEL_OUTBOUND ) ;
2007-06-14 04:44:44 +00:00
zap_set_state_locked ( zchan , ZAP_CHANNEL_STATE_DIALING ) ;
return status ;
}
2008-05-05 22:54:40 +00:00
static ZIO_CHANNEL_REQUEST_FUNCTION ( isdn_channel_request )
{
Q931mes_Generic * gen = ( Q931mes_Generic * ) caller_data - > raw_data ;
Q931ie_BearerCap BearerCap ;
Q931ie_ChanID ChanID = { 0 } ;
Q931ie_CallingNum CallingNum ;
Q931ie_CallingNum * ptrCallingNum ;
Q931ie_CalledNum CalledNum ;
Q931ie_CalledNum * ptrCalledNum ;
Q931ie_Display Display , * ptrDisplay ;
2008-08-22 16:55:01 +00:00
Q931ie_HLComp HLComp ; /* High-Layer Compatibility IE */
Q931ie_SendComplete SComplete ; /* Sending Complete IE */
Q931ie_ProgInd Progress ; /* Progress Indicator IE */
2008-05-05 22:54:40 +00:00
zap_status_t status = ZAP_FAIL ;
zap_isdn_data_t * isdn_data = span - > signal_data ;
int sanity = 60000 ;
2008-08-22 16:55:01 +00:00
int codec = 0 ;
2008-05-05 22:54:40 +00:00
2008-08-22 16:55:01 +00:00
/*
* get codec type
*/
zap_channel_command ( & span - > channels [ chan_id ] , ZAP_COMMAND_GET_NATIVE_CODEC , & codec ) ;
2008-05-05 22:54:40 +00:00
2008-08-22 16:55:01 +00:00
/*
* Q .931 Setup Message
*/
2008-05-05 22:54:40 +00:00
Q931InitMesGeneric ( gen ) ;
gen - > MesType = Q931mes_SETUP ;
2008-08-22 16:55:01 +00:00
gen - > CRVFlag = 0 ; /* outgoing call */
2008-05-05 22:54:40 +00:00
2008-08-22 16:55:01 +00:00
/*
* Bearer Capability IE
*/
Q931InitIEBearerCap ( & BearerCap ) ;
BearerCap . CodStand = Q931_CODING_ITU ; /* ITU-T = 0, ISO/IEC = 1, National = 2, Network = 3 */
BearerCap . ITC = Q931_ITC_SPEECH ; /* Speech */
BearerCap . TransMode = 0 ; /* Circuit = 0, Packet = 1 */
BearerCap . ITR = Q931_ITR_64K ; /* 64k */
2008-05-05 22:54:40 +00:00
BearerCap . Layer1Ident = 1 ;
2008-08-22 16:55:01 +00:00
BearerCap . UIL1Prot = ( codec = = ZAP_CODEC_ALAW ) ? Q931_UIL1P_G711A : Q931_UIL1P_G711U ; /* U-law = 2, A-law = 3 */
2008-05-05 22:54:40 +00:00
gen - > BearerCap = Q931AppendIE ( ( L3UCHAR * ) gen , ( L3UCHAR * ) & BearerCap ) ;
2008-08-22 16:55:01 +00:00
/*
* Channel ID IE
*/
Q931InitIEChanID ( & ChanID ) ;
ChanID . IntType = ZAP_SPAN_IS_BRI ( span ) ? 0 : 1 ; /* PRI = 1, BRI = 0 */
if ( ! ZAP_SPAN_IS_NT ( span ) ) {
ChanID . PrefExcl = ( isdn_data - > opts & ZAP_ISDN_OPT_SUGGEST_CHANNEL ) ? 0 : 1 ; /* 0 = preferred, 1 exclusive */
} else {
ChanID . PrefExcl = 1 ; /* always exclusive in NT-mode */
}
if ( ChanID . IntType ) {
ChanID . InfoChanSel = 1 ; /* None = 0, See Slot = 1, Any = 3 */
ChanID . ChanMapType = 3 ; /* B-Chan */
ChanID . ChanSlot = ( unsigned char ) chan_id ;
} else {
ChanID . InfoChanSel = ( unsigned char ) chan_id & 0x03 ; /* None = 0, B1 = 1, B2 = 2, Any = 3 */
}
2008-05-05 22:54:40 +00:00
gen - > ChanID = Q931AppendIE ( ( L3UCHAR * ) gen , ( L3UCHAR * ) & ChanID ) ;
2008-08-22 16:55:01 +00:00
/*
* Progress IE
*/
Q931InitIEProgInd ( & Progress ) ;
Progress . CodStand = Q931_CODING_ITU ; /* 0 = ITU */
Progress . Location = 0 ; /* 0 = User, 1 = Private Network */
Progress . ProgDesc = 3 ; /* 1 = Not end-to-end ISDN */
gen - > ProgInd = Q931AppendIE ( ( L3UCHAR * ) gen , ( L3UCHAR * ) & Progress ) ;
/*
* Display IE
*/
Q931InitIEDisplay ( & Display ) ;
2008-05-05 22:54:40 +00:00
Display . Size = Display . Size + ( unsigned char ) strlen ( caller_data - > cid_name ) ;
gen - > Display = Q931AppendIE ( ( L3UCHAR * ) gen , ( L3UCHAR * ) & Display ) ;
ptrDisplay = Q931GetIEPtr ( gen - > Display , gen - > buf ) ;
zap_copy_string ( ( char * ) ptrDisplay - > Display , caller_data - > cid_name , strlen ( caller_data - > cid_name ) + 1 ) ;
2008-08-22 16:55:01 +00:00
/*
* Calling Number IE
*/
Q931InitIECallingNum ( & CallingNum ) ;
CallingNum . TypNum = Q931_TON_UNKNOWN ;
CallingNum . NumPlanID = Q931_NUMPLAN_E164 ;
CallingNum . PresInd = Q931_PRES_ALLOWED ;
CallingNum . ScreenInd = Q931_SCREEN_USER_NOT_SCREENED ;
2008-05-05 22:54:40 +00:00
CallingNum . Size = CallingNum . Size + ( unsigned char ) strlen ( caller_data - > cid_num . digits ) ;
gen - > CallingNum = Q931AppendIE ( ( L3UCHAR * ) gen , ( L3UCHAR * ) & CallingNum ) ;
ptrCallingNum = Q931GetIEPtr ( gen - > CallingNum , gen - > buf ) ;
zap_copy_string ( ( char * ) ptrCallingNum - > Digit , caller_data - > cid_num . digits , strlen ( caller_data - > cid_num . digits ) + 1 ) ;
2008-08-22 16:55:01 +00:00
/*
* Called number IE
*/
Q931InitIECalledNum ( & CalledNum ) ;
CalledNum . TypNum = Q931_TON_UNKNOWN ;
CalledNum . NumPlanID = Q931_NUMPLAN_E164 ;
2008-05-05 22:54:40 +00:00
CalledNum . Size = CalledNum . Size + ( unsigned char ) strlen ( caller_data - > ani . digits ) ;
gen - > CalledNum = Q931AppendIE ( ( L3UCHAR * ) gen , ( L3UCHAR * ) & CalledNum ) ;
ptrCalledNum = Q931GetIEPtr ( gen - > CalledNum , gen - > buf ) ;
zap_copy_string ( ( char * ) ptrCalledNum - > Digit , caller_data - > ani . digits , strlen ( caller_data - > ani . digits ) + 1 ) ;
2008-08-22 16:55:01 +00:00
/*
* High - Layer Compatibility IE ( Note : Required for AVM FritzBox )
*/
Q931InitIEHLComp ( & HLComp ) ;
HLComp . CodStand = Q931_CODING_ITU ; /* ITU */
HLComp . Interpret = 4 ; /* only possible value */
HLComp . PresMeth = 1 ; /* High-layer protocol profile */
HLComp . HLCharID = 1 ; /* Telephony = 1, Fax G2+3 = 4, Fax G4 = 65 (Class I)/ 68 (Class II or III) */
gen - > HLComp = Q931AppendIE ( ( L3UCHAR * ) gen , ( L3UCHAR * ) & HLComp ) ;
/*
* Sending complete IE ( or some NT stuff waits forever in Q .931 overlap dial state . . . )
*/
SComplete . IEId = Q931ie_SENDING_COMPLETE ;
// gen->SendComplete = Q931AppendIE((L3UCHAR *) gen, (L3UCHAR *) &SComplete);
2008-05-05 22:54:40 +00:00
caller_data - > call_state = ZAP_CALLER_STATE_DIALING ;
Q931Rx43 ( & isdn_data - > q931 , ( L3UCHAR * ) gen , gen - > Size ) ;
isdn_data - > outbound_crv [ gen - > CRV ] = caller_data ;
//isdn_data->channels_local_crv[gen->CRV] = zchan;
while ( zap_running ( ) & & caller_data - > call_state = = ZAP_CALLER_STATE_DIALING ) {
zap_sleep ( 1 ) ;
if ( ! - - sanity ) {
caller_data - > call_state = ZAP_CALLER_STATE_FAIL ;
break ;
}
}
isdn_data - > outbound_crv [ gen - > CRV ] = NULL ;
if ( caller_data - > call_state = = ZAP_CALLER_STATE_SUCCESS ) {
zap_channel_t * new_chan = NULL ;
int fail = 1 ;
new_chan = NULL ;
if ( caller_data - > chan_id < ZAP_MAX_CHANNELS_SPAN & & caller_data - > chan_id < = span - > chan_count ) {
new_chan = & span - > channels [ caller_data - > chan_id ] ;
}
if ( new_chan & & ( status = zap_channel_open_chan ( new_chan ) = = ZAP_SUCCESS ) ) {
if ( zap_test_flag ( new_chan , ZAP_CHANNEL_INUSE ) | | new_chan - > state ! = ZAP_CHANNEL_STATE_DOWN ) {
if ( new_chan - > state = = ZAP_CHANNEL_STATE_DOWN | | new_chan - > state > = ZAP_CHANNEL_STATE_TERMINATING ) {
int x = 0 ;
zap_log ( ZAP_LOG_WARNING , " Channel %d:%d ~ %d:%d is already in use waiting for it to become available. \n " ) ;
for ( x = 0 ; x < 200 ; x + + ) {
if ( ! zap_test_flag ( new_chan , ZAP_CHANNEL_INUSE ) ) {
break ;
}
zap_sleep ( 5 ) ;
}
}
if ( zap_test_flag ( new_chan , ZAP_CHANNEL_INUSE ) ) {
zap_log ( ZAP_LOG_ERROR , " Channel %d:%d ~ %d:%d is already in use. \n " ,
new_chan - > span_id ,
new_chan - > chan_id ,
new_chan - > physical_span_id ,
new_chan - > physical_chan_id
) ;
new_chan = NULL ;
}
}
if ( new_chan & & new_chan - > state = = ZAP_CHANNEL_STATE_DOWN ) {
isdn_data - > channels_local_crv [ gen - > CRV ] = new_chan ;
memset ( & new_chan - > caller_data , 0 , sizeof ( new_chan - > caller_data ) ) ;
2008-05-21 18:58:14 +00:00
zap_set_flag ( new_chan , ZAP_CHANNEL_OUTBOUND ) ;
zap_set_state_locked ( new_chan , ZAP_CHANNEL_STATE_DIALING ) ;
2008-05-05 22:54:40 +00:00
switch ( gen - > MesType ) {
case Q931mes_ALERTING :
new_chan - > init_state = ZAP_CHANNEL_STATE_PROGRESS_MEDIA ;
break ;
case Q931mes_CONNECT :
new_chan - > init_state = ZAP_CHANNEL_STATE_UP ;
break ;
default :
new_chan - > init_state = ZAP_CHANNEL_STATE_PROGRESS ;
break ;
}
fail = 0 ;
}
}
if ( ! fail ) {
* zchan = new_chan ;
return ZAP_SUCCESS ;
} else {
Q931ie_Cause cause ;
gen - > MesType = Q931mes_DISCONNECT ;
cause . IEId = Q931ie_CAUSE ;
cause . Size = sizeof ( Q931ie_Cause ) ;
cause . CodStand = 0 ;
cause . Location = 1 ;
cause . Recom = 1 ;
//should we be casting here.. or do we need to translate value?
cause . Value = ( unsigned char ) ZAP_CAUSE_WRONG_CALL_STATE ;
* cause . Diag = ' \0 ' ;
gen - > Cause = Q931AppendIE ( ( L3UCHAR * ) gen , ( L3UCHAR * ) & cause ) ;
Q931Rx43 ( & isdn_data - > q931 , ( L3UCHAR * ) gen , gen - > Size ) ;
if ( gen - > CRV ) {
Q931ReleaseCRV ( & isdn_data - > q931 , gen - > CRV ) ;
}
if ( new_chan ) {
zap_log ( ZAP_LOG_CRIT , " Channel is busy \n " ) ;
} else {
zap_log ( ZAP_LOG_CRIT , " Failed to open channel for new setup message \n " ) ;
}
}
}
* zchan = NULL ;
return ZAP_FAIL ;
}
2007-05-22 22:07:05 +00:00
static L3INT zap_isdn_931_err ( void * pvt , L3INT id , L3INT p1 , L3INT p2 )
{
2007-05-24 17:19:03 +00:00
zap_log ( ZAP_LOG_ERROR , " ERROR: [%s] [%d] [%d] \n " , q931_error_to_name ( id ) , p1 , p2 ) ;
2007-05-22 22:07:05 +00:00
return 0 ;
}
static L3INT zap_isdn_931_34 ( void * pvt , L2UCHAR * msg , L2INT mlen )
{
zap_span_t * span = ( zap_span_t * ) pvt ;
2007-06-25 05:26:37 +00:00
zap_isdn_data_t * isdn_data = span - > signal_data ;
2007-05-22 22:07:05 +00:00
Q931mes_Generic * gen = ( Q931mes_Generic * ) msg ;
2008-01-30 01:44:05 +00:00
int chan_id = 0 ;
2007-06-14 03:54:02 +00:00
zap_channel_t * zchan = NULL ;
2008-05-05 22:54:40 +00:00
zap_caller_data_t * caller_data = NULL ;
2008-02-04 20:27:54 +00:00
if ( Q931IsIEPresent ( gen - > ChanID ) ) {
Q931ie_ChanID * chanid = Q931GetIEPtr ( gen - > ChanID , gen - > buf ) ;
2008-08-22 16:55:01 +00:00
if ( chanid - > IntType )
chan_id = chanid - > ChanSlot ;
else
chan_id = chanid - > InfoChanSel ;
2008-01-30 15:29:20 +00:00
}
2008-01-30 01:44:05 +00:00
2007-05-22 22:07:05 +00:00
assert ( span ! = NULL ) ;
2007-06-25 05:26:37 +00:00
assert ( isdn_data ! = NULL ) ;
2007-06-14 03:54:02 +00:00
2008-08-22 16:55:01 +00:00
zap_log ( ZAP_LOG_DEBUG , " Yay I got an event! Type:[%02x] Size:[%d] CRV: %d (%#hx, CTX: %s) \n " , gen - > MesType , gen - > Size , gen - > CRV , gen - > CRV , gen - > CRVFlag ? " Terminator " : " Originator " ) ;
2008-05-05 22:54:40 +00:00
if ( gen - > CRVFlag & & ( caller_data = isdn_data - > outbound_crv [ gen - > CRV ] ) ) {
if ( chan_id ) {
caller_data - > chan_id = chan_id ;
}
switch ( gen - > MesType ) {
case Q931mes_STATUS :
case Q931mes_CALL_PROCEEDING :
break ;
case Q931mes_ALERTING :
case Q931mes_PROGRESS :
case Q931mes_CONNECT :
{
caller_data - > call_state = ZAP_CALLER_STATE_SUCCESS ;
}
break ;
default :
caller_data - > call_state = ZAP_CALLER_STATE_FAIL ;
break ;
}
return 0 ;
2007-06-14 03:54:02 +00:00
}
2008-02-04 20:27:54 +00:00
if ( gen - > CRVFlag ) {
2008-04-02 20:24:54 +00:00
zchan = isdn_data - > channels_local_crv [ gen - > CRV ] ;
2008-02-04 20:27:54 +00:00
} else {
2008-04-02 20:24:54 +00:00
zchan = isdn_data - > channels_remote_crv [ gen - > CRV ] ;
2008-02-04 20:27:54 +00:00
}
2008-08-22 16:55:01 +00:00
zap_log ( ZAP_LOG_DEBUG , " zchan %x source isdn_data->channels_%s_crv[%#hx] \n " , zchan , gen - > CRVFlag ? " local " : " remote " , gen - > CRV ) ;
2008-05-05 22:54:40 +00:00
2007-06-25 16:00:24 +00:00
if ( gen - > ProtDisc = = 3 ) {
switch ( gen - > MesType ) {
case Q931mes_SERVICE :
{
Q931ie_ChangeStatus * changestatus = Q931GetIEPtr ( gen - > ChangeStatus , gen - > buf ) ;
if ( zchan ) {
switch ( changestatus - > NewStatus ) {
case 0 : /* change status to "in service" */
2007-06-26 01:02:36 +00:00
{
zap_clear_flag_locked ( zchan , ZAP_CHANNEL_SUSPENDED ) ;
zap_log ( ZAP_LOG_DEBUG , " Channel %d:%d in service \n " , zchan - > span_id , zchan - > chan_id ) ;
2008-04-01 21:49:35 +00:00
zap_set_state_locked ( zchan , ZAP_CHANNEL_STATE_RESTART ) ;
2007-06-26 00:47:05 +00:00
}
2007-06-25 16:00:24 +00:00
break ;
2007-06-26 01:02:36 +00:00
case 1 :
{ /* change status to "maintenance" */
zap_set_flag_locked ( zchan , ZAP_CHANNEL_SUSPENDED ) ;
zap_set_state_locked ( zchan , ZAP_CHANNEL_STATE_SUSPENDED ) ;
}
2007-06-25 16:00:24 +00:00
break ;
2007-06-26 01:02:36 +00:00
case 2 :
{ /* change status to "out of service" */
zap_set_flag_locked ( zchan , ZAP_CHANNEL_SUSPENDED ) ;
zap_set_state_locked ( zchan , ZAP_CHANNEL_STATE_SUSPENDED ) ;
}
2007-06-25 16:00:24 +00:00
break ;
2007-06-26 01:02:36 +00:00
default : /* unknown */
{
break ;
}
2007-06-25 16:00:24 +00:00
}
2007-06-14 03:54:02 +00:00
}
}
2007-06-25 16:00:24 +00:00
break ;
default :
break ;
2007-06-14 03:54:02 +00:00
}
2007-06-25 16:00:24 +00:00
} else {
switch ( gen - > MesType ) {
case Q931mes_RESTART :
{
2008-02-27 17:37:29 +00:00
if ( chan_id ) {
zchan = & span - > channels [ chan_id ] ;
}
2007-06-25 16:00:24 +00:00
if ( zchan ) {
zap_set_state_locked ( zchan , ZAP_CHANNEL_STATE_RESTART ) ;
} else {
uint32_t i ;
for ( i = 0 ; i < span - > chan_count ; i + + ) {
zap_set_state_locked ( ( & span - > channels [ i ] ) , ZAP_CHANNEL_STATE_RESTART ) ;
2007-06-14 03:54:02 +00:00
}
2007-06-25 16:00:24 +00:00
}
}
break ;
2007-11-21 22:08:47 +00:00
case Q931mes_RELEASE :
2007-06-25 16:00:24 +00:00
case Q931mes_RELEASE_COMPLETE :
{
2008-04-19 15:57:19 +00:00
const char * what = gen - > MesType = = Q931mes_RELEASE ? " Release " : " Release Complete " ;
2008-01-28 22:17:34 +00:00
if ( zchan ) {
2008-04-19 15:57:19 +00:00
if ( zchan - > state = = ZAP_CHANNEL_STATE_TERMINATING | | zchan - > state = = ZAP_CHANNEL_STATE_HANGUP ) {
2008-08-22 16:55:01 +00:00
if ( gen - > MesType = = Q931mes_RELEASE ) {
zap_set_state_locked ( zchan , ZAP_CHANNEL_STATE_HANGUP_COMPLETE ) ;
} else {
zap_set_state_locked ( zchan , ZAP_CHANNEL_STATE_DOWN ) ;
}
}
else if ( ( gen - > MesType = = Q931mes_RELEASE & & zchan - > state < = ZAP_CHANNEL_STATE_UP ) | |
( gen - > MesType = = Q931mes_RELEASE_COMPLETE & & zchan - > state = = ZAP_CHANNEL_STATE_DIALING ) ) {
/*
* Don ' t keep inbound channels open if the remote side hangs up before we answered
*/
Q931ie_Cause * cause = Q931GetIEPtr ( gen - > Cause , gen - > buf ) ;
zap_sigmsg_t sig ;
zap_status_t status ;
memset ( & sig , 0 , sizeof ( sig ) ) ;
sig . chan_id = zchan - > chan_id ;
sig . span_id = zchan - > span_id ;
sig . channel = zchan ;
sig . channel - > caller_data . hangup_cause = ( cause ) ? cause - > Value : ZAP_CAUSE_NORMAL_UNSPECIFIED ;
sig . event_id = ZAP_SIGEVENT_STOP ;
status = isdn_data - > sig_cb ( & sig ) ;
zap_log ( ZAP_LOG_DEBUG , " Received %s in state %s, requested hangup for channel %d:%d \n " , what , zap_channel_state2str ( zchan - > state ) , zchan - > span_id , chan_id ) ;
}
else {
2008-04-19 15:57:19 +00:00
zap_log ( ZAP_LOG_DEBUG , " Ignoring %s on channel %d \n " , what , chan_id ) ;
}
2008-01-28 22:17:34 +00:00
} else {
2008-04-19 15:57:19 +00:00
zap_log ( ZAP_LOG_CRIT , " Received %s with no matching channel %d \n " , what , chan_id ) ;
2008-01-28 22:17:34 +00:00
}
2007-06-25 16:00:24 +00:00
}
break ;
case Q931mes_DISCONNECT :
{
2008-02-28 17:23:10 +00:00
if ( zchan ) {
Q931ie_Cause * cause = Q931GetIEPtr ( gen - > Cause , gen - > buf ) ;
zchan - > caller_data . hangup_cause = cause - > Value ;
zap_set_state_locked ( zchan , ZAP_CHANNEL_STATE_TERMINATING ) ;
} else {
2008-04-01 18:53:25 +00:00
zap_log ( ZAP_LOG_CRIT , " Received Disconnect with no matching channel %d \n " , chan_id ) ;
2008-02-28 17:23:10 +00:00
}
2007-06-25 16:00:24 +00:00
}
break ;
case Q931mes_ALERTING :
{
2008-01-28 22:17:34 +00:00
if ( zchan ) {
zap_set_state_locked ( zchan , ZAP_CHANNEL_STATE_PROGRESS_MEDIA ) ;
} else {
zap_log ( ZAP_LOG_CRIT , " Received Alerting with no matching channel %d \n " , chan_id ) ;
}
2007-06-25 16:00:24 +00:00
}
break ;
case Q931mes_PROGRESS :
{
2008-01-28 22:17:34 +00:00
if ( zchan ) {
zap_set_state_locked ( zchan , ZAP_CHANNEL_STATE_PROGRESS ) ;
} else {
zap_log ( ZAP_LOG_CRIT , " Received Progress with no matching channel %d \n " , chan_id ) ;
}
2007-06-25 16:00:24 +00:00
}
break ;
case Q931mes_CONNECT :
{
2008-01-28 22:17:34 +00:00
if ( zchan ) {
zap_set_state_locked ( zchan , ZAP_CHANNEL_STATE_UP ) ;
2008-08-22 16:55:01 +00:00
gen - > MesType = Q931mes_CONNECT_ACKNOWLEDGE ;
gen - > CRVFlag = 0 ; /* outbound */
Q931Rx43 ( & isdn_data - > q931 , ( L3UCHAR * ) gen , gen - > Size ) ;
2008-01-28 22:17:34 +00:00
} else {
zap_log ( ZAP_LOG_CRIT , " Received Connect with no matching channel %d \n " , chan_id ) ;
}
2007-06-25 16:00:24 +00:00
}
break ;
case Q931mes_SETUP :
{
Q931ie_CallingNum * callingnum = Q931GetIEPtr ( gen - > CallingNum , gen - > buf ) ;
Q931ie_CalledNum * callednum = Q931GetIEPtr ( gen - > CalledNum , gen - > buf ) ;
int fail = 1 ;
uint32_t cplen = mlen ;
2008-08-22 16:55:01 +00:00
if ( zchan & & zchan = = isdn_data - > channels_remote_crv [ gen - > CRV ] ) {
zap_log ( ZAP_LOG_INFO , " Duplicate SETUP message(?) for Channel %d:%d ~ %d:%d in state %s [ignoring] \n " ,
zchan - > span_id ,
zchan - > chan_id ,
zchan - > physical_span_id ,
zchan - > physical_chan_id ,
zap_channel_state2str ( zchan - > state ) ) ;
break ;
}
2008-04-01 21:49:35 +00:00
zchan = NULL ;
if ( chan_id < ZAP_MAX_CHANNELS_SPAN & & chan_id < = span - > chan_count ) {
zchan = & span - > channels [ chan_id ] ;
}
2007-06-25 16:00:24 +00:00
2008-08-22 16:55:01 +00:00
if ( zchan ) {
2008-04-01 21:49:35 +00:00
if ( zap_test_flag ( zchan , ZAP_CHANNEL_INUSE ) | | zchan - > state ! = ZAP_CHANNEL_STATE_DOWN ) {
if ( zchan - > state = = ZAP_CHANNEL_STATE_DOWN | | zchan - > state > = ZAP_CHANNEL_STATE_TERMINATING ) {
int x = 0 ;
2008-08-22 16:55:01 +00:00
zap_log ( ZAP_LOG_WARNING , " Channel %d:%d ~ %d:%d is already in use waiting for it to become available. \n " ,
zchan - > span_id ,
zchan - > chan_id ,
zchan - > physical_span_id ,
zchan - > physical_chan_id ) ;
2008-04-01 21:49:35 +00:00
for ( x = 0 ; x < 200 ; x + + ) {
if ( ! zap_test_flag ( zchan , ZAP_CHANNEL_INUSE ) ) {
break ;
}
zap_sleep ( 5 ) ;
}
}
if ( zap_test_flag ( zchan , ZAP_CHANNEL_INUSE ) ) {
zap_log ( ZAP_LOG_ERROR , " Channel %d:%d ~ %d:%d is already in use. \n " ,
zchan - > span_id ,
zchan - > chan_id ,
zchan - > physical_span_id ,
zchan - > physical_chan_id
) ;
2008-04-07 05:38:37 +00:00
zchan = NULL ;
2008-04-01 21:49:35 +00:00
}
}
2007-06-25 16:00:24 +00:00
2008-04-07 06:01:27 +00:00
if ( zchan & & zchan - > state = = ZAP_CHANNEL_STATE_DOWN ) {
2008-04-02 20:24:54 +00:00
isdn_data - > channels_remote_crv [ gen - > CRV ] = zchan ;
2007-06-25 16:00:24 +00:00
memset ( & zchan - > caller_data , 0 , sizeof ( zchan - > caller_data ) ) ;
2008-03-01 00:48:36 +00:00
zap_set_string ( zchan - > caller_data . cid_num . digits , ( char * ) callingnum - > Digit ) ;
2007-06-25 16:00:24 +00:00
zap_set_string ( zchan - > caller_data . cid_name , ( char * ) callingnum - > Digit ) ;
2008-03-01 00:48:36 +00:00
zap_set_string ( zchan - > caller_data . ani . digits , ( char * ) callingnum - > Digit ) ;
zap_set_string ( zchan - > caller_data . dnis . digits , ( char * ) callednum - > Digit ) ;
2007-06-25 16:00:24 +00:00
zchan - > caller_data . CRV = gen - > CRV ;
if ( cplen > sizeof ( zchan - > caller_data . raw_data ) ) {
cplen = sizeof ( zchan - > caller_data . raw_data ) ;
}
gen - > CRVFlag = ! ( gen - > CRVFlag ) ;
memcpy ( zchan - > caller_data . raw_data , msg , cplen ) ;
zchan - > caller_data . raw_data_len = cplen ;
zap_set_state_locked ( zchan , ZAP_CHANNEL_STATE_RING ) ;
fail = 0 ;
}
2007-06-14 03:54:02 +00:00
}
2007-06-13 03:37:55 +00:00
2007-06-25 16:00:24 +00:00
if ( fail ) {
2008-04-01 21:49:35 +00:00
Q931ie_Cause cause ;
gen - > MesType = Q931mes_DISCONNECT ;
2008-08-22 16:55:01 +00:00
gen - > CRVFlag = 1 ; /* inbound call */
2008-04-01 21:49:35 +00:00
cause . IEId = Q931ie_CAUSE ;
cause . Size = sizeof ( Q931ie_Cause ) ;
2008-08-22 16:55:01 +00:00
cause . CodStand = Q931_CODING_ITU ;
2008-04-01 21:49:35 +00:00
cause . Location = 1 ;
cause . Recom = 1 ;
//should we be casting here.. or do we need to translate value?
cause . Value = ( unsigned char ) ZAP_CAUSE_WRONG_CALL_STATE ;
* cause . Diag = ' \0 ' ;
gen - > Cause = Q931AppendIE ( ( L3UCHAR * ) gen , ( L3UCHAR * ) & cause ) ;
Q931Rx43 ( & isdn_data - > q931 , ( L3UCHAR * ) gen , gen - > Size ) ;
2008-04-02 20:24:54 +00:00
if ( gen - > CRV ) {
Q931ReleaseCRV ( & isdn_data - > q931 , gen - > CRV ) ;
}
2008-02-29 17:55:29 +00:00
if ( zchan ) {
2008-04-01 21:49:35 +00:00
zap_log ( ZAP_LOG_CRIT , " Channel is busy \n " ) ;
2008-02-29 17:55:29 +00:00
} else {
zap_log ( ZAP_LOG_CRIT , " Failed to open channel for new setup message \n " ) ;
}
2008-04-01 21:49:35 +00:00
2008-08-22 16:55:01 +00:00
} else {
Q931ie_ProgInd progress ;
/*
* Setup Progress indicator
*/
progress . IEId = Q931ie_PROGRESS_INDICATOR ;
progress . Size = sizeof ( Q931ie_ProgInd ) ;
progress . CodStand = Q931_CODING_ITU ; /* ITU */
progress . Location = 1 ; /* private network serving the local user */
progress . ProgDesc = 1 ; /* call is not end-to-end isdn = 1, in-band information available = 8 */
gen - > ProgInd = Q931AppendIE ( ( L3UCHAR * ) gen , ( L3UCHAR * ) & progress ) ;
}
}
break ;
case Q931mes_CALL_PROCEEDING :
{
if ( zchan ) {
zap_log ( ZAP_LOG_CRIT , " Received CALL PROCEEDING message for channel %d \n " , chan_id ) ;
zap_set_state_locked ( zchan , ZAP_CHANNEL_STATE_PROGRESS ) ;
} else {
zap_log ( ZAP_LOG_CRIT , " Received CALL PROCEEDING with no matching channel %d \n " , chan_id ) ;
2007-06-25 16:00:24 +00:00
}
2007-06-13 03:37:55 +00:00
}
2007-06-25 16:00:24 +00:00
break ;
2008-08-22 16:55:01 +00:00
2007-06-25 16:00:24 +00:00
default :
2008-08-22 16:55:01 +00:00
zap_log ( ZAP_LOG_CRIT , " Received unhandled message %d (%#x) \n " , ( int ) gen - > MesType , ( int ) gen - > MesType ) ;
2007-06-25 16:00:24 +00:00
break ;
2007-06-13 03:37:55 +00:00
}
}
2007-05-24 14:30:36 +00:00
2007-05-22 22:07:05 +00:00
return 0 ;
}
2008-08-22 16:55:01 +00:00
static int zap_isdn_921_23 ( void * pvt , Q921DLMsg_t ind , L2UCHAR tei , L2UCHAR * msg , L2INT mlen )
2007-05-24 01:55:26 +00:00
{
2008-08-22 16:55:01 +00:00
int ret , offset = ( ind = = Q921_DL_DATA ) ? 4 : 3 ;
2007-06-23 18:51:10 +00:00
char bb [ 4096 ] = " " ;
2007-06-16 04:39:15 +00:00
2008-08-22 16:55:01 +00:00
switch ( ind ) {
case Q921_DL_DATA :
case Q921_DL_UNIT_DATA :
print_hex_bytes ( msg + offset , mlen - offset , bb , sizeof ( bb ) ) ;
zap_log ( ZAP_LOG_DEBUG , " READ %d \n %s \n %s \n \n " , ( int ) mlen - offset , LINE , bb ) ;
default :
ret = Q931Rx23 ( pvt , ind , tei , msg , mlen ) ;
if ( ret ! = 0 )
zap_log ( ZAP_LOG_DEBUG , " 931 parse error [%d] [%s] \n " , ret , q931_error_to_name ( ret ) ) ;
break ;
}
2007-05-24 01:59:26 +00:00
return ( ( ret > = 0 ) ? 1 : 0 ) ;
2007-05-24 01:55:26 +00:00
}
2007-05-22 22:07:05 +00:00
static int zap_isdn_921_21 ( void * pvt , L2UCHAR * msg , L2INT mlen )
{
zap_span_t * span = ( zap_span_t * ) pvt ;
zap_size_t len = ( zap_size_t ) mlen ;
2007-06-25 05:26:37 +00:00
zap_isdn_data_t * isdn_data = span - > signal_data ;
2007-05-23 18:25:23 +00:00
# ifdef IODEBUG
2007-11-21 22:08:47 +00:00
char bb [ 4096 ] = " " ;
print_hex_bytes ( msg , len , bb , sizeof ( bb ) ) ;
print_bits ( msg , ( int ) len , bb , sizeof ( bb ) , ZAP_ENDIAN_LITTLE , 0 ) ;
zap_log ( ZAP_LOG_DEBUG , " WRITE %d \n %s \n %s \n \n " , ( int ) len , LINE , bb ) ;
2007-05-23 18:25:23 +00:00
# endif
2007-05-22 22:07:05 +00:00
assert ( span ! = NULL ) ;
2007-06-25 05:26:37 +00:00
return zap_channel_write ( isdn_data - > dchan , msg , len , & len ) = = ZAP_SUCCESS ? 0 : - 1 ;
2007-05-22 22:07:05 +00:00
}
2007-06-14 03:54:02 +00:00
static __inline__ void state_advance ( zap_channel_t * zchan )
2007-06-13 03:37:55 +00:00
{
Q931mes_Generic * gen = ( Q931mes_Generic * ) zchan - > caller_data . raw_data ;
2007-06-25 05:26:37 +00:00
zap_isdn_data_t * isdn_data = zchan - > span - > signal_data ;
2007-06-14 03:54:02 +00:00
zap_sigmsg_t sig ;
zap_status_t status ;
2007-06-13 03:37:55 +00:00
2008-08-29 15:58:59 +00:00
zap_log ( ZAP_LOG_DEBUG , " %d:%d STATE [%s] \n " ,
2007-06-13 03:37:55 +00:00
zchan - > span_id , zchan - > chan_id , zap_channel_state2str ( zchan - > state ) ) ;
2007-06-14 03:54:02 +00:00
memset ( & sig , 0 , sizeof ( sig ) ) ;
sig . chan_id = zchan - > chan_id ;
sig . span_id = zchan - > span_id ;
sig . channel = zchan ;
2007-06-13 03:37:55 +00:00
switch ( zchan - > state ) {
2007-06-16 04:39:15 +00:00
case ZAP_CHANNEL_STATE_DOWN :
{
2008-04-02 20:24:54 +00:00
if ( gen - > CRV ) {
2008-08-22 16:55:01 +00:00
if ( gen - > CRVFlag ) {
isdn_data - > channels_local_crv [ gen - > CRV ] = NULL ;
} else {
isdn_data - > channels_remote_crv [ gen - > CRV ] = NULL ;
}
2008-04-02 20:24:54 +00:00
Q931ReleaseCRV ( & isdn_data - > q931 , gen - > CRV ) ;
}
2007-06-16 04:39:15 +00:00
zap_channel_done ( zchan ) ;
}
break ;
case ZAP_CHANNEL_STATE_PROGRESS :
{
if ( zap_test_flag ( zchan , ZAP_CHANNEL_OUTBOUND ) ) {
sig . event_id = ZAP_SIGEVENT_PROGRESS ;
2007-06-25 05:26:37 +00:00
if ( ( status = isdn_data - > sig_cb ( & sig ) ! = ZAP_SUCCESS ) ) {
2007-06-16 04:39:15 +00:00
zap_set_state_locked ( zchan , ZAP_CHANNEL_STATE_HANGUP ) ;
}
} else {
2008-07-24 20:03:37 +00:00
gen - > MesType = Q931mes_CALL_PROCEEDING ;
2008-08-22 16:55:01 +00:00
gen - > CRVFlag = 1 ; /* inbound */
2007-06-25 05:26:37 +00:00
Q931Rx43 ( & isdn_data - > q931 , ( void * ) gen , gen - > Size ) ;
2007-06-16 04:39:15 +00:00
}
}
break ;
2007-06-14 03:54:02 +00:00
case ZAP_CHANNEL_STATE_RING :
{
2007-06-16 04:39:15 +00:00
if ( ! zap_test_flag ( zchan , ZAP_CHANNEL_OUTBOUND ) ) {
sig . event_id = ZAP_SIGEVENT_START ;
2007-06-25 05:26:37 +00:00
if ( ( status = isdn_data - > sig_cb ( & sig ) ! = ZAP_SUCCESS ) ) {
2007-06-16 04:39:15 +00:00
zap_set_state_locked ( zchan , ZAP_CHANNEL_STATE_HANGUP ) ;
}
2007-06-14 03:54:02 +00:00
}
}
break ;
case ZAP_CHANNEL_STATE_RESTART :
{
2008-05-26 01:51:11 +00:00
zchan - > caller_data . hangup_cause = ZAP_CAUSE_NORMAL_UNSPECIFIED ;
sig . event_id = ZAP_SIGEVENT_RESTART ;
status = isdn_data - > sig_cb ( & sig ) ;
zap_set_state_locked ( zchan , ZAP_CHANNEL_STATE_DOWN ) ;
2007-06-14 03:54:02 +00:00
}
break ;
2007-06-16 13:46:32 +00:00
case ZAP_CHANNEL_STATE_PROGRESS_MEDIA :
2007-06-16 04:39:15 +00:00
{
if ( zap_test_flag ( zchan , ZAP_CHANNEL_OUTBOUND ) ) {
sig . event_id = ZAP_SIGEVENT_PROGRESS_MEDIA ;
2007-06-25 05:26:37 +00:00
if ( ( status = isdn_data - > sig_cb ( & sig ) ! = ZAP_SUCCESS ) ) {
2007-06-16 04:39:15 +00:00
zap_set_state_locked ( zchan , ZAP_CHANNEL_STATE_HANGUP ) ;
}
} else {
2008-08-22 16:55:01 +00:00
if ( ! zap_test_flag ( zchan , ZAP_CHANNEL_OPEN ) ) {
if ( zap_channel_open_chan ( zchan ) ! = ZAP_SUCCESS ) {
zap_set_state_locked ( zchan , ZAP_CHANNEL_STATE_HANGUP ) ;
return ;
}
}
2007-06-16 04:39:15 +00:00
gen - > MesType = Q931mes_ALERTING ;
2008-08-22 16:55:01 +00:00
gen - > CRVFlag = 1 ; /* inbound call */
2007-06-25 05:26:37 +00:00
Q931Rx43 ( & isdn_data - > q931 , ( void * ) gen , gen - > Size ) ;
2007-06-16 04:39:15 +00:00
}
}
break ;
2007-06-13 03:37:55 +00:00
case ZAP_CHANNEL_STATE_UP :
{
2007-06-16 04:39:15 +00:00
if ( zap_test_flag ( zchan , ZAP_CHANNEL_OUTBOUND ) ) {
sig . event_id = ZAP_SIGEVENT_UP ;
2007-06-25 05:26:37 +00:00
if ( ( status = isdn_data - > sig_cb ( & sig ) ! = ZAP_SUCCESS ) ) {
2007-06-16 04:39:15 +00:00
zap_set_state_locked ( zchan , ZAP_CHANNEL_STATE_HANGUP ) ;
}
} else {
2008-08-22 16:55:01 +00:00
if ( ! zap_test_flag ( zchan , ZAP_CHANNEL_OPEN ) ) {
if ( zap_channel_open_chan ( zchan ) ! = ZAP_SUCCESS ) {
zap_set_state_locked ( zchan , ZAP_CHANNEL_STATE_HANGUP ) ;
return ;
}
}
2007-06-16 04:39:15 +00:00
gen - > MesType = Q931mes_CONNECT ;
gen - > BearerCap = 0 ;
2008-08-22 16:55:01 +00:00
gen - > CRVFlag = 1 ; /* inbound call */
2007-06-25 05:26:37 +00:00
Q931Rx43 ( & isdn_data - > q931 , ( void * ) gen , zchan - > caller_data . raw_data_len ) ;
2007-06-16 04:39:15 +00:00
}
2007-06-13 03:37:55 +00:00
}
break ;
2007-06-14 04:44:44 +00:00
case ZAP_CHANNEL_STATE_DIALING :
2008-05-21 18:58:14 +00:00
if ( ! ( isdn_data - > opts & ZAP_ISDN_OPT_SUGGEST_CHANNEL ) ) {
Q931ie_BearerCap BearerCap ;
Q931ie_ChanID ChanID ;
Q931ie_CallingNum CallingNum ;
Q931ie_CallingNum * ptrCallingNum ;
Q931ie_CalledNum CalledNum ;
Q931ie_CalledNum * ptrCalledNum ;
Q931ie_Display Display , * ptrDisplay ;
2008-08-22 16:55:01 +00:00
Q931ie_HLComp HLComp ; /* High-Layer Compatibility IE */
Q931ie_SendComplete SComplete ; /* Sending Complete IE */
Q931ie_ProgInd Progress ; /* Progress Indicator IE */
int codec = 0 ;
/*
* get codec type
*/
zap_channel_command ( & zchan - > span - > channels [ zchan - > chan_id ] , ZAP_COMMAND_GET_NATIVE_CODEC , & codec ) ;
/*
* Q .931 Setup Message
*/
2008-05-21 18:58:14 +00:00
Q931InitMesGeneric ( gen ) ;
gen - > MesType = Q931mes_SETUP ;
2008-08-22 16:55:01 +00:00
gen - > CRVFlag = 0 ; /* outbound(?) */
2008-05-21 18:58:14 +00:00
2008-08-22 16:55:01 +00:00
/*
* Bearer Capability IE
*/
Q931InitIEBearerCap ( & BearerCap ) ;
BearerCap . CodStand = Q931_CODING_ITU ; /* ITU-T = 0, ISO/IEC = 1, National = 2, Network = 3 */
BearerCap . ITC = Q931_ITC_SPEECH ; /* Speech */
BearerCap . TransMode = 0 ; /* Circuit = 0, Packet = 1 */
BearerCap . ITR = Q931_ITR_64K ; /* 64k = 16, Packet mode = 0 */
2008-05-21 18:58:14 +00:00
BearerCap . Layer1Ident = 1 ;
2008-08-22 16:55:01 +00:00
BearerCap . UIL1Prot = ( codec = = ZAP_CODEC_ALAW ) ? 3 : 2 ; /* U-law = 2, A-law = 3 */
2008-05-21 18:58:14 +00:00
gen - > BearerCap = Q931AppendIE ( ( L3UCHAR * ) gen , ( L3UCHAR * ) & BearerCap ) ;
2008-08-22 16:55:01 +00:00
/*
* ChannelID IE
*/
Q931InitIEChanID ( & ChanID ) ;
ChanID . IntType = ZAP_SPAN_IS_BRI ( zchan - > span ) ? 0 : 1 ; /* PRI = 1, BRI = 0 */
ChanID . PrefExcl = ZAP_SPAN_IS_NT ( zchan - > span ) ? 1 : 0 ; /* Exclusive in NT-mode = 1, Preferred otherwise = 0 */
if ( ChanID . IntType ) {
ChanID . InfoChanSel = 1 ; /* None = 0, See Slot = 1, Any = 3 */
ChanID . ChanMapType = 3 ; /* B-Chan */
ChanID . ChanSlot = ( unsigned char ) zchan - > chan_id ;
} else {
ChanID . InfoChanSel = ( unsigned char ) zchan - > chan_id & 0x03 ; /* None = 0, B1 = 1, B2 = 2, Any = 3 */
}
2008-05-21 18:58:14 +00:00
gen - > ChanID = Q931AppendIE ( ( L3UCHAR * ) gen , ( L3UCHAR * ) & ChanID ) ;
2008-08-22 16:55:01 +00:00
/*
* Progress IE
*/
Q931InitIEProgInd ( & Progress ) ;
Progress . CodStand = Q931_CODING_ITU ; /* 0 = ITU */
Progress . Location = 0 ; /* 0 = User, 1 = Private Network */
Progress . ProgDesc = 3 ; /* 1 = Not end-to-end ISDN */
gen - > ProgInd = Q931AppendIE ( ( L3UCHAR * ) gen , ( L3UCHAR * ) & Progress ) ;
/*
* Display IE
*/
Q931InitIEDisplay ( & Display ) ;
2008-05-21 18:58:14 +00:00
Display . Size = Display . Size + ( unsigned char ) strlen ( zchan - > caller_data . cid_name ) ;
gen - > Display = Q931AppendIE ( ( L3UCHAR * ) gen , ( L3UCHAR * ) & Display ) ;
ptrDisplay = Q931GetIEPtr ( gen - > Display , gen - > buf ) ;
zap_copy_string ( ( char * ) ptrDisplay - > Display , zchan - > caller_data . cid_name , strlen ( zchan - > caller_data . cid_name ) + 1 ) ;
2008-08-22 16:55:01 +00:00
/*
* CallingNum IE
*/
Q931InitIECallingNum ( & CallingNum ) ;
CallingNum . TypNum = Q931_TON_UNKNOWN ;
CallingNum . NumPlanID = Q931_NUMPLAN_E164 ;
CallingNum . PresInd = Q931_PRES_ALLOWED ;
CallingNum . ScreenInd = Q931_SCREEN_USER_NOT_SCREENED ;
2008-05-21 18:58:14 +00:00
CallingNum . Size = CallingNum . Size + ( unsigned char ) strlen ( zchan - > caller_data . cid_num . digits ) ;
gen - > CallingNum = Q931AppendIE ( ( L3UCHAR * ) gen , ( L3UCHAR * ) & CallingNum ) ;
ptrCallingNum = Q931GetIEPtr ( gen - > CallingNum , gen - > buf ) ;
zap_copy_string ( ( char * ) ptrCallingNum - > Digit , zchan - > caller_data . cid_num . digits , strlen ( zchan - > caller_data . cid_num . digits ) + 1 ) ;
2008-08-22 16:55:01 +00:00
/*
* CalledNum IE
*/
Q931InitIECalledNum ( & CalledNum ) ;
CalledNum . TypNum = Q931_TON_UNKNOWN ;
CalledNum . NumPlanID = Q931_NUMPLAN_E164 ;
2008-05-21 18:58:14 +00:00
CalledNum . Size = CalledNum . Size + ( unsigned char ) strlen ( zchan - > caller_data . ani . digits ) ;
gen - > CalledNum = Q931AppendIE ( ( L3UCHAR * ) gen , ( L3UCHAR * ) & CalledNum ) ;
ptrCalledNum = Q931GetIEPtr ( gen - > CalledNum , gen - > buf ) ;
zap_copy_string ( ( char * ) ptrCalledNum - > Digit , zchan - > caller_data . ani . digits , strlen ( zchan - > caller_data . ani . digits ) + 1 ) ;
2007-06-16 04:39:15 +00:00
2008-08-22 16:55:01 +00:00
/*
* High - Layer Compatibility IE ( Note : Required for AVM FritzBox )
*/
Q931InitIEHLComp ( & HLComp ) ;
HLComp . CodStand = Q931_CODING_ITU ; /* ITU */
HLComp . Interpret = 4 ; /* only possible value */
HLComp . PresMeth = 1 ; /* High-layer protocol profile */
HLComp . HLCharID = Q931_HLCHAR_TELEPHONY ; /* Telephony = 1, Fax G2+3 = 4, Fax G4 = 65 (Class I)/ 68 (Class II or III) */ /* TODO: make accessible from user layer */
gen - > HLComp = Q931AppendIE ( ( L3UCHAR * ) gen , ( L3UCHAR * ) & HLComp ) ;
/*
* Sending complete IE ( or some NT stuff waits forever in Q .931 overlap dial state . . . )
*/
SComplete . IEId = Q931ie_SENDING_COMPLETE ;
// gen->SendComplete = Q931AppendIE((L3UCHAR *) gen, (L3UCHAR *) &SComplete);
2008-05-21 18:58:14 +00:00
Q931Rx43 ( & isdn_data - > q931 , ( L3UCHAR * ) gen , gen - > Size ) ;
isdn_data - > channels_local_crv [ gen - > CRV ] = zchan ;
}
break ;
case ZAP_CHANNEL_STATE_HANGUP_COMPLETE :
{
2008-08-22 16:55:01 +00:00
/* reply RELEASE with RELEASE_COMPLETE message */
if ( zchan - > last_state = = ZAP_CHANNEL_STATE_HANGUP ) {
gen - > MesType = Q931mes_RELEASE_COMPLETE ;
Q931Rx43 ( & isdn_data - > q931 , ( L3UCHAR * ) gen , gen - > Size ) ;
}
2008-05-21 18:58:14 +00:00
zap_set_state_locked ( zchan , ZAP_CHANNEL_STATE_DOWN ) ;
2007-06-14 04:44:44 +00:00
}
break ;
2007-06-14 03:54:02 +00:00
case ZAP_CHANNEL_STATE_HANGUP :
{
Q931ie_Cause cause ;
2008-08-22 16:55:01 +00:00
zap_log ( ZAP_LOG_DEBUG , " Hangup: Direction %s \n " , zap_test_flag ( zchan , ZAP_CHANNEL_OUTBOUND ) ? " Outbound " : " Inbound " ) ;
gen - > CRVFlag = zap_test_flag ( zchan , ZAP_CHANNEL_OUTBOUND ) ? 0 : 1 ;
2007-06-14 03:54:02 +00:00
cause . IEId = Q931ie_CAUSE ;
cause . Size = sizeof ( Q931ie_Cause ) ;
2008-08-22 16:55:01 +00:00
cause . CodStand = Q931_CODING_ITU ; /* ITU */
cause . Location = 1 ; /* private network */
cause . Recom = 1 ; /* */
/*
* BRI PTMP needs special handling here . . .
* TODO : cleanup / refine ( see above )
*/
if ( zchan - > last_state = = ZAP_CHANNEL_STATE_RING ) {
/*
* inbound call [ was : number unknown ( = not found in routing state ) ]
* ( in Q .931 spec terms : Reject request )
*/
gen - > MesType = Q931mes_RELEASE_COMPLETE ;
//cause.Value = (unsigned char) ZAP_CAUSE_UNALLOCATED;
cause . Value = ( unsigned char ) zchan - > caller_data . hangup_cause ;
* cause . Diag = ' \0 ' ;
gen - > Cause = Q931AppendIE ( ( L3UCHAR * ) gen , ( L3UCHAR * ) & cause ) ;
Q931Rx43 ( & isdn_data - > q931 , ( L3UCHAR * ) gen , gen - > Size ) ;
/* we're done, release channel */
//zap_set_state_locked(zchan, ZAP_CHANNEL_STATE_HANGUP_COMPLETE);
zap_set_state_locked ( zchan , ZAP_CHANNEL_STATE_DOWN ) ;
}
else if ( zchan - > last_state < = ZAP_CHANNEL_STATE_PROGRESS ) {
/*
* just release all unanswered calls [ was : inbound call , remote side hung up before we answered ]
*/
gen - > MesType = Q931mes_RELEASE ;
cause . Value = ( unsigned char ) zchan - > caller_data . hangup_cause ;
* cause . Diag = ' \0 ' ;
gen - > Cause = Q931AppendIE ( ( L3UCHAR * ) gen , ( L3UCHAR * ) & cause ) ;
Q931Rx43 ( & isdn_data - > q931 , ( void * ) gen , gen - > Size ) ;
/* this will be triggered by the RELEASE_COMPLETE reply */
/* zap_set_state_locked(zchan, ZAP_CHANNEL_STATE_HANGUP_COMPLETE); */
}
else {
/*
* call connected , hangup
*/
gen - > MesType = Q931mes_DISCONNECT ;
cause . Value = ( unsigned char ) zchan - > caller_data . hangup_cause ;
* cause . Diag = ' \0 ' ;
gen - > Cause = Q931AppendIE ( ( L3UCHAR * ) gen , ( L3UCHAR * ) & cause ) ;
Q931Rx43 ( & isdn_data - > q931 , ( L3UCHAR * ) gen , gen - > Size ) ;
}
2007-06-14 03:54:02 +00:00
}
break ;
case ZAP_CHANNEL_STATE_TERMINATING :
{
2008-08-22 16:55:01 +00:00
zap_log ( ZAP_LOG_DEBUG , " Terminating: Direction %s \n " , zap_test_flag ( zchan , ZAP_CHANNEL_OUTBOUND ) ? " Outbound " : " Inbound " ) ;
2007-06-16 15:52:49 +00:00
sig . event_id = ZAP_SIGEVENT_STOP ;
2007-06-25 05:26:37 +00:00
status = isdn_data - > sig_cb ( & sig ) ;
2007-06-14 03:54:02 +00:00
gen - > MesType = Q931mes_RELEASE ;
2008-08-22 16:55:01 +00:00
gen - > CRVFlag = zap_test_flag ( zchan , ZAP_CHANNEL_OUTBOUND ) ? 0 : 1 ;
2007-06-25 05:26:37 +00:00
Q931Rx43 ( & isdn_data - > q931 , ( void * ) gen , gen - > Size ) ;
2007-06-14 03:54:02 +00:00
}
2007-06-13 03:37:55 +00:00
default :
break ;
}
}
2007-06-14 03:54:02 +00:00
static __inline__ void check_state ( zap_span_t * span )
2007-06-13 03:37:55 +00:00
{
2008-03-28 23:22:21 +00:00
if ( zap_test_flag ( span , ZAP_SPAN_STATE_CHANGE ) ) {
uint32_t j ;
zap_clear_flag_locked ( span , ZAP_SPAN_STATE_CHANGE ) ;
for ( j = 1 ; j < = span - > chan_count ; j + + ) {
if ( zap_test_flag ( ( & span - > channels [ j ] ) , ZAP_CHANNEL_STATE_CHANGE ) ) {
2008-05-02 22:28:45 +00:00
zap_mutex_lock ( span - > channels [ j ] . mutex ) ;
zap_clear_flag ( ( & span - > channels [ j ] ) , ZAP_CHANNEL_STATE_CHANGE ) ;
2008-03-28 23:22:21 +00:00
state_advance ( & span - > channels [ j ] ) ;
zap_channel_complete_state ( & span - > channels [ j ] ) ;
2008-05-02 22:28:45 +00:00
zap_mutex_unlock ( span - > channels [ j ] . mutex ) ;
2008-03-28 23:22:21 +00:00
}
}
}
2007-06-13 03:37:55 +00:00
}
2008-03-28 23:22:21 +00:00
2007-06-23 18:51:10 +00:00
static __inline__ zap_status_t process_event ( zap_span_t * span , zap_event_t * event )
{
zap_sigmsg_t sig ;
2007-06-25 05:26:37 +00:00
zap_isdn_data_t * isdn_data = span - > signal_data ;
2007-06-23 18:51:10 +00:00
memset ( & sig , 0 , sizeof ( sig ) ) ;
sig . chan_id = event - > channel - > chan_id ;
sig . span_id = event - > channel - > span_id ;
sig . channel = event - > channel ;
zap_log ( ZAP_LOG_DEBUG , " EVENT [%s][%d:%d] STATE [%s] \n " ,
zap_oob_event2str ( event - > enum_id ) , event - > channel - > span_id , event - > channel - > chan_id , zap_channel_state2str ( event - > channel - > state ) ) ;
switch ( event - > enum_id ) {
case ZAP_OOB_ALARM_TRAP :
{
sig . event_id = ZAP_OOB_ALARM_TRAP ;
if ( event - > channel - > state ! = ZAP_CHANNEL_STATE_DOWN ) {
2008-04-01 21:49:35 +00:00
zap_set_state_locked ( event - > channel , ZAP_CHANNEL_STATE_RESTART ) ;
2007-06-23 18:51:10 +00:00
}
zap_set_flag ( event - > channel , ZAP_CHANNEL_SUSPENDED ) ;
zap_channel_get_alarms ( event - > channel ) ;
2007-06-25 05:26:37 +00:00
isdn_data - > sig_cb ( & sig ) ;
2007-06-23 18:51:10 +00:00
zap_log ( ZAP_LOG_WARNING , " channel %d:%d (%d:%d) has alarms [%s] \n " ,
event - > channel - > span_id , event - > channel - > chan_id ,
event - > channel - > physical_span_id , event - > channel - > physical_chan_id ,
event - > channel - > last_error ) ;
}
break ;
case ZAP_OOB_ALARM_CLEAR :
{
sig . event_id = ZAP_OOB_ALARM_CLEAR ;
zap_clear_flag ( event - > channel , ZAP_CHANNEL_SUSPENDED ) ;
zap_channel_get_alarms ( event - > channel ) ;
2007-06-25 05:26:37 +00:00
isdn_data - > sig_cb ( & sig ) ;
2007-06-23 18:51:10 +00:00
}
break ;
}
return ZAP_SUCCESS ;
}
static __inline__ void check_events ( zap_span_t * span )
{
zap_status_t status ;
status = zap_span_poll_event ( span , 5 ) ;
switch ( status ) {
case ZAP_SUCCESS :
{
zap_event_t * event ;
while ( zap_span_next_event ( span , & event ) = = ZAP_SUCCESS ) {
if ( event - > enum_id = = ZAP_OOB_NOOP ) {
continue ;
}
if ( process_event ( span , event ) ! = ZAP_SUCCESS ) {
break ;
}
}
}
break ;
case ZAP_FAIL :
{
2007-11-24 18:04:14 +00:00
zap_log ( ZAP_LOG_DEBUG , " Event Failure! %d \n " , zap_running ( ) ) ;
2007-06-23 18:51:10 +00:00
}
break ;
default :
break ;
}
}
2007-05-22 22:07:05 +00:00
static void * zap_isdn_run ( zap_thread_t * me , void * obj )
{
zap_span_t * span = ( zap_span_t * ) obj ;
2007-06-25 05:26:37 +00:00
zap_isdn_data_t * isdn_data = span - > signal_data ;
2007-05-22 22:07:05 +00:00
unsigned char buf [ 1024 ] ;
zap_size_t len = sizeof ( buf ) ;
2007-06-13 03:37:55 +00:00
int errs = 0 ;
2007-05-22 22:07:05 +00:00
2007-05-23 19:47:35 +00:00
# ifdef WIN32
timeBeginPeriod ( 1 ) ;
# endif
2007-05-22 22:07:05 +00:00
zap_log ( ZAP_LOG_DEBUG , " ISDN thread starting. \n " ) ;
2007-06-25 05:26:37 +00:00
Q921Start ( & isdn_data - > q921 ) ;
2007-05-22 22:07:05 +00:00
2007-11-24 18:04:14 +00:00
while ( zap_running ( ) & & zap_test_flag ( isdn_data , ZAP_ISDN_RUNNING ) ) {
2007-05-22 22:07:05 +00:00
zap_wait_flag_t flags = ZAP_READ ;
2007-06-25 05:26:37 +00:00
zap_status_t status = zap_channel_wait ( isdn_data - > dchan , & flags , 100 ) ;
2007-05-23 19:47:35 +00:00
2007-06-25 05:26:37 +00:00
Q921TimerTick ( & isdn_data - > q921 ) ;
2008-08-22 16:55:01 +00:00
Q931TimerTick ( & isdn_data - > q931 ) ;
2007-06-13 03:37:55 +00:00
check_state ( span ) ;
2007-06-23 18:51:10 +00:00
check_events ( span ) ;
2007-05-22 22:07:05 +00:00
switch ( status ) {
case ZAP_FAIL :
{
zap_log ( ZAP_LOG_ERROR , " D-Chan Read Error! \n " ) ;
snprintf ( span - > last_error , sizeof ( span - > last_error ) , " D-Chan Read Error! " ) ;
2007-06-13 03:37:55 +00:00
if ( + + errs = = 10 ) {
2008-05-21 16:48:19 +00:00
isdn_data - > dchan - > state = ZAP_CHANNEL_STATE_UP ;
2007-06-13 03:37:55 +00:00
goto done ;
}
2007-05-22 22:07:05 +00:00
}
break ;
case ZAP_TIMEOUT :
{
2007-06-13 03:37:55 +00:00
errs = 0 ;
2007-05-22 22:07:05 +00:00
}
break ;
default :
{
2007-06-13 03:37:55 +00:00
errs = 0 ;
2007-05-22 22:07:05 +00:00
if ( flags & ZAP_READ ) {
len = sizeof ( buf ) ;
2007-06-25 05:26:37 +00:00
if ( zap_channel_read ( isdn_data - > dchan , buf , & len ) = = ZAP_SUCCESS ) {
2007-06-20 20:30:10 +00:00
# ifdef IODEBUG
2007-11-21 22:08:47 +00:00
char bb [ 4096 ] = " " ;
print_hex_bytes ( buf , len , bb , sizeof ( bb ) ) ;
2007-05-22 22:07:05 +00:00
2007-11-21 22:08:47 +00:00
print_bits ( buf , ( int ) len , bb , sizeof ( bb ) , ZAP_ENDIAN_LITTLE , 0 ) ;
zap_log ( ZAP_LOG_DEBUG , " READ %d \n %s \n %s \n \n " , ( int ) len , LINE , bb ) ;
2007-06-20 20:30:10 +00:00
# endif
2007-06-25 05:26:37 +00:00
Q921QueueHDLCFrame ( & isdn_data - > q921 , buf , ( int ) len ) ;
Q921Rx12 ( & isdn_data - > q921 ) ;
2007-05-22 22:07:05 +00:00
}
} else {
zap_log ( ZAP_LOG_DEBUG , " No Read FLAG! \n " ) ;
}
}
break ;
}
}
done :
2007-06-25 05:26:37 +00:00
zap_channel_close ( & isdn_data - > dchans [ 0 ] ) ;
zap_channel_close ( & isdn_data - > dchans [ 1 ] ) ;
zap_clear_flag ( isdn_data , ZAP_ISDN_RUNNING ) ;
2007-05-22 22:07:05 +00:00
2007-05-23 19:47:35 +00:00
# ifdef WIN32
timeEndPeriod ( 1 ) ;
# endif
2007-05-22 22:07:05 +00:00
zap_log ( ZAP_LOG_DEBUG , " ISDN thread ended. \n " ) ;
return NULL ;
}
2008-08-29 15:58:59 +00:00
static ZIO_SIG_LOAD_FUNCTION ( zap_isdn_init )
2007-05-22 22:07:05 +00:00
{
Q931Initialize ( ) ;
2007-05-23 19:47:35 +00:00
Q921SetGetTimeCB ( zap_time_now ) ;
2008-08-22 16:55:01 +00:00
Q931SetGetTimeCB ( zap_time_now ) ;
2007-05-22 22:07:05 +00:00
return ZAP_SUCCESS ;
}
2008-08-22 16:55:01 +00:00
static int q931_rx_32 ( void * pvt , Q921DLMsg_t ind , L3UCHAR tei , L3UCHAR * msg , L3INT mlen )
2007-06-16 04:39:15 +00:00
{
2008-08-22 16:55:01 +00:00
int offset = 4 ;
2007-06-16 04:39:15 +00:00
char bb [ 4096 ] = " " ;
2008-08-22 16:55:01 +00:00
switch ( ind ) {
case Q921_DL_UNIT_DATA :
offset = 3 ;
case Q921_DL_DATA :
print_hex_bytes ( msg + offset , mlen - offset , bb , sizeof ( bb ) ) ;
zap_log ( ZAP_LOG_DEBUG , " WRITE %d \n %s \n %s \n \n " , ( int ) mlen - offset , LINE , bb ) ;
break ;
default :
break ;
}
return Q921Rx32 ( pvt , ind , tei , msg , mlen ) ;
}
static int zap_isdn_q921_log ( void * pvt , Q921LogLevel_t level , char * msg , L2INT size )
{
int loglevel = ZAP_LOG_LEVEL_DEBUG ;
switch ( level ) {
case Q921_LOG_DEBUG :
loglevel = ZAP_LOG_LEVEL_DEBUG ;
break ;
case Q921_LOG_INFO :
loglevel = ZAP_LOG_LEVEL_INFO ;
break ;
case Q921_LOG_NOTICE :
loglevel = ZAP_LOG_LEVEL_NOTICE ;
break ;
case Q921_LOG_WARNING :
loglevel = ZAP_LOG_LEVEL_WARNING ;
break ;
case Q921_LOG_ERROR :
loglevel = ZAP_LOG_LEVEL_ERROR ;
break ;
default :
return 0 ;
}
zap_log ( __FILE__ , " Q.921 " , __LINE__ , loglevel , " %s " , msg ) ;
return 0 ;
2007-06-16 04:39:15 +00:00
}
2008-08-22 16:55:01 +00:00
static L3INT zap_isdn_q931_log ( void * pvt , Q931LogLevel_t level , char * msg , L3INT size )
{
zap_log ( __FILE__ , " Q.931 " , __LINE__ , ZAP_LOG_LEVEL_DEBUG , " %s " , msg ) ;
return 0 ;
}
2008-05-21 18:58:14 +00:00
static zap_state_map_t isdn_state_map = {
{
{
ZSD_OUTBOUND ,
ZSM_UNACCEPTABLE ,
{ ZAP_ANY_STATE } ,
{ ZAP_CHANNEL_STATE_RESTART , ZAP_END }
} ,
{
ZSD_OUTBOUND ,
ZSM_UNACCEPTABLE ,
{ ZAP_CHANNEL_STATE_RESTART , ZAP_END } ,
{ ZAP_CHANNEL_STATE_DOWN , ZAP_END }
} ,
{
ZSD_OUTBOUND ,
ZSM_UNACCEPTABLE ,
{ ZAP_CHANNEL_STATE_DOWN , ZAP_END } ,
{ ZAP_CHANNEL_STATE_DIALING , ZAP_END }
} ,
{
ZSD_OUTBOUND ,
ZSM_UNACCEPTABLE ,
{ ZAP_CHANNEL_STATE_DIALING , ZAP_END } ,
{ ZAP_CHANNEL_STATE_PROGRESS_MEDIA , ZAP_CHANNEL_STATE_PROGRESS , ZAP_CHANNEL_STATE_UP , ZAP_CHANNEL_STATE_HANGUP , ZAP_CHANNEL_STATE_TERMINATING , ZAP_END }
} ,
{
ZSD_OUTBOUND ,
ZSM_UNACCEPTABLE ,
{ ZAP_CHANNEL_STATE_PROGRESS_MEDIA , ZAP_CHANNEL_STATE_PROGRESS , ZAP_END } ,
{ ZAP_CHANNEL_STATE_HANGUP , ZAP_CHANNEL_STATE_PROGRESS_MEDIA , ZAP_CHANNEL_STATE_TERMINATING , ZAP_CHANNEL_STATE_UP , ZAP_END }
} ,
{
ZSD_OUTBOUND ,
ZSM_UNACCEPTABLE ,
{ ZAP_CHANNEL_STATE_HANGUP , ZAP_CHANNEL_STATE_TERMINATING , ZAP_END } ,
2008-08-22 16:55:01 +00:00
{ ZAP_CHANNEL_STATE_HANGUP_COMPLETE , ZAP_CHANNEL_STATE_DOWN , ZAP_END }
2008-05-21 18:58:14 +00:00
} ,
{
ZSD_OUTBOUND ,
ZSM_UNACCEPTABLE ,
{ ZAP_CHANNEL_STATE_HANGUP_COMPLETE , ZAP_END } ,
{ ZAP_CHANNEL_STATE_DOWN , ZAP_END } ,
} ,
{
ZSD_OUTBOUND ,
ZSM_UNACCEPTABLE ,
{ ZAP_CHANNEL_STATE_UP , ZAP_END } ,
{ ZAP_CHANNEL_STATE_HANGUP , ZAP_CHANNEL_STATE_TERMINATING , ZAP_END }
} ,
/****************************************/
{
ZSD_INBOUND ,
ZSM_UNACCEPTABLE ,
{ ZAP_ANY_STATE } ,
{ ZAP_CHANNEL_STATE_RESTART , ZAP_END }
} ,
{
ZSD_INBOUND ,
ZSM_UNACCEPTABLE ,
{ ZAP_CHANNEL_STATE_RESTART , ZAP_END } ,
{ ZAP_CHANNEL_STATE_DOWN , ZAP_END }
} ,
{
ZSD_INBOUND ,
ZSM_UNACCEPTABLE ,
{ ZAP_CHANNEL_STATE_DOWN , ZAP_END } ,
{ ZAP_CHANNEL_STATE_RING , ZAP_END }
} ,
{
ZSD_INBOUND ,
ZSM_UNACCEPTABLE ,
{ ZAP_CHANNEL_STATE_RING , ZAP_END } ,
{ ZAP_CHANNEL_STATE_HANGUP , ZAP_CHANNEL_STATE_TERMINATING , ZAP_CHANNEL_STATE_PROGRESS , ZAP_CHANNEL_STATE_PROGRESS_MEDIA , ZAP_CHANNEL_STATE_UP , ZAP_END }
} ,
{
ZSD_INBOUND ,
ZSM_UNACCEPTABLE ,
{ ZAP_CHANNEL_STATE_HANGUP , ZAP_CHANNEL_STATE_TERMINATING , ZAP_END } ,
2008-08-22 16:55:01 +00:00
{ ZAP_CHANNEL_STATE_HANGUP_COMPLETE , ZAP_CHANNEL_STATE_DOWN , ZAP_END } ,
2008-05-21 18:58:14 +00:00
} ,
{
ZSD_INBOUND ,
ZSM_UNACCEPTABLE ,
{ ZAP_CHANNEL_STATE_HANGUP_COMPLETE , ZAP_END } ,
{ ZAP_CHANNEL_STATE_DOWN , ZAP_END } ,
} ,
{
ZSD_INBOUND ,
ZSM_UNACCEPTABLE ,
{ ZAP_CHANNEL_STATE_PROGRESS , ZAP_CHANNEL_STATE_PROGRESS_MEDIA , ZAP_END } ,
{ ZAP_CHANNEL_STATE_HANGUP , ZAP_CHANNEL_STATE_PROGRESS_MEDIA , ZAP_CHANNEL_STATE_CANCEL , ZAP_CHANNEL_STATE_UP , ZAP_END } ,
} ,
{
ZSD_INBOUND ,
ZSM_UNACCEPTABLE ,
{ ZAP_CHANNEL_STATE_UP , ZAP_END } ,
{ ZAP_CHANNEL_STATE_HANGUP , ZAP_CHANNEL_STATE_TERMINATING , ZAP_END } ,
} ,
}
} ;
2008-08-29 15:58:59 +00:00
static zap_status_t zap_isdn_start ( zap_span_t * span )
{
zap_isdn_data_t * isdn_data = span - > signal_data ;
zap_set_flag ( isdn_data , ZAP_ISDN_RUNNING ) ;
return zap_thread_create_detached ( zap_isdn_run , span ) ;
}
static ZIO_SIG_CONFIGURE_FUNCTION ( zap_isdn_configure_span )
2007-05-22 22:07:05 +00:00
{
2007-05-24 01:01:50 +00:00
uint32_t i , x = 0 ;
2007-05-22 22:07:05 +00:00
zap_channel_t * dchans [ 2 ] = { 0 } ;
2007-06-25 05:26:37 +00:00
zap_isdn_data_t * isdn_data ;
2008-08-29 15:58:59 +00:00
char * var , * val ;
Q931Dialect_t dialect = Q931_Dialect_National ;
uint32_t opts = 0 ;
2007-05-22 22:07:05 +00:00
if ( span - > signal_type ) {
2007-06-13 03:37:55 +00:00
snprintf ( span - > last_error , sizeof ( span - > last_error ) , " Span is already configured for signalling [%d]. " , span - > signal_type ) ;
2007-05-22 22:07:05 +00:00
return ZAP_FAIL ;
}
if ( span - > trunk_type > = ZAP_TRUNK_NONE ) {
snprintf ( span - > last_error , sizeof ( span - > last_error ) , " Unknown trunk type! " ) ;
return ZAP_FAIL ;
}
for ( i = 1 ; i < = span - > chan_count ; i + + ) {
if ( span - > channels [ i ] . type = = ZAP_CHAN_TYPE_DQ921 ) {
2008-02-28 00:19:07 +00:00
if ( x > 1 ) {
snprintf ( span - > last_error , sizeof ( span - > last_error ) , " Span has more than 2 D-Channels! " ) ;
return ZAP_FAIL ;
} else {
if ( zap_channel_open ( span - > span_id , i , & dchans [ x ] ) = = ZAP_SUCCESS ) {
zap_log ( ZAP_LOG_DEBUG , " opening d-channel #%d %d:%d \n " , x , dchans [ x ] - > span_id , dchans [ x ] - > chan_id ) ;
2008-05-21 16:48:19 +00:00
dchans [ x ] - > state = ZAP_CHANNEL_STATE_UP ;
2008-02-28 00:19:07 +00:00
x + + ;
}
2007-05-22 22:07:05 +00:00
}
}
}
if ( ! x ) {
snprintf ( span - > last_error , sizeof ( span - > last_error ) , " Span has no D-Channels! " ) ;
return ZAP_FAIL ;
}
2007-06-25 05:26:37 +00:00
isdn_data = malloc ( sizeof ( * isdn_data ) ) ;
assert ( isdn_data ! = NULL ) ;
memset ( isdn_data , 0 , sizeof ( * isdn_data ) ) ;
2007-05-22 22:07:05 +00:00
2008-08-29 15:58:59 +00:00
isdn_data - > mode = Q931_TE ;
dialect = Q931_Dialect_National ;
2008-09-05 03:53:32 +00:00
while ( ( var = va_arg ( ap , char * ) ) ) {
2008-08-29 15:58:59 +00:00
if ( ! strcasecmp ( var , " mode " ) ) {
if ( ! ( val = va_arg ( ap , char * ) ) ) {
break ;
}
isdn_data - > mode = strcasecmp ( val , " net " ) ? Q931_TE : Q931_NT ;
} else if ( ! strcasecmp ( var , " dialect " ) ) {
if ( ! ( val = va_arg ( ap , char * ) ) ) {
break ;
}
dialect = q931_str2Q931Dialect_type ( val ) ;
if ( dialect = = Q931_Dialect_Count ) {
dialect = Q931_Dialect_National ;
}
} else if ( ! strcasecmp ( var , " opts " ) ) {
int * optp ;
if ( ! ( optp = va_arg ( ap , int * ) ) ) {
break ;
}
opts = isdn_data - > opts = * optp ;
}
}
span - > start = zap_isdn_start ;
2007-06-25 05:26:37 +00:00
isdn_data - > sig_cb = sig_cb ;
isdn_data - > dchans [ 0 ] = dchans [ 0 ] ;
isdn_data - > dchans [ 1 ] = dchans [ 1 ] ;
isdn_data - > dchan = isdn_data - > dchans [ 0 ] ;
2007-05-22 22:07:05 +00:00
2007-06-25 05:26:37 +00:00
Q921_InitTrunk ( & isdn_data - > q921 ,
2007-05-22 22:07:05 +00:00
0 ,
0 ,
2008-08-29 15:58:59 +00:00
isdn_data - > mode ,
2008-08-22 16:55:01 +00:00
span - > trunk_type = = ZAP_TRUNK_BRI_PTMP ? Q921_PTMP : Q921_PTP ,
2007-05-22 22:07:05 +00:00
0 ,
zap_isdn_921_21 ,
2008-08-22 16:55:01 +00:00
( Q921Tx23CB_t ) zap_isdn_921_23 ,
2007-05-22 22:07:05 +00:00
span ,
2007-06-25 05:26:37 +00:00
& isdn_data - > q931 ) ;
2008-08-22 16:55:01 +00:00
Q921SetLogCB ( & isdn_data - > q921 , & zap_isdn_q921_log , isdn_data ) ;
2008-09-04 04:58:00 +00:00
Q921SetLogLevel ( & isdn_data - > q921 , Q921_LOG_NONE ) ;
2007-05-22 22:07:05 +00:00
2007-06-25 05:26:37 +00:00
Q931Api_InitTrunk ( & isdn_data - > q931 ,
2007-05-22 22:07:05 +00:00
dialect ,
2008-08-29 15:58:59 +00:00
isdn_data - > mode ,
2007-05-22 22:07:05 +00:00
span - > trunk_type ,
zap_isdn_931_34 ,
2008-08-22 16:55:01 +00:00
( Q931Tx32CB_t ) q931_rx_32 ,
2007-05-22 22:07:05 +00:00
zap_isdn_931_err ,
2007-06-25 05:26:37 +00:00
& isdn_data - > q921 ,
2007-05-22 22:07:05 +00:00
span ) ;
2007-05-29 16:30:20 +00:00
2008-08-22 16:55:01 +00:00
Q931SetLogCB ( & isdn_data - > q931 , & zap_isdn_q931_log , isdn_data ) ;
Q931SetLogLevel ( & isdn_data - > q931 , Q931_LOG_DEBUG ) ;
2007-06-25 05:26:37 +00:00
isdn_data - > q931 . autoRestartAck = 1 ;
2008-08-22 16:55:01 +00:00
// isdn_data->q931.autoConnectAck = 1;
isdn_data - > q931 . autoConnectAck = 0 ;
2007-06-25 16:00:24 +00:00
isdn_data - > q931 . autoServiceAck = 1 ;
2007-06-25 05:26:37 +00:00
span - > signal_data = isdn_data ;
2007-05-22 22:07:05 +00:00
span - > signal_type = ZAP_SIGTYPE_ISDN ;
2007-06-14 04:44:44 +00:00
span - > outgoing_call = isdn_outgoing_call ;
2008-08-29 15:58:59 +00:00
2008-05-21 18:58:14 +00:00
if ( ( opts & ZAP_ISDN_OPT_SUGGEST_CHANNEL ) ) {
span - > channel_request = isdn_channel_request ;
span - > suggest_chan_id = 1 ;
}
span - > state_map = & isdn_state_map ;
2007-05-22 22:07:05 +00:00
return ZAP_SUCCESS ;
}
2007-06-04 14:53:12 +00:00
2008-08-29 15:58:59 +00:00
zap_module_t zap_module = {
" isdn " ,
NULL ,
NULL ,
zap_isdn_init ,
zap_isdn_configure_span ,
NULL
} ;
2007-06-04 14:53:12 +00:00
/* For Emacs:
* Local Variables :
* mode : c
* indent - tabs - mode : t
* tab - width : 4
* c - basic - offset : 4
* End :
* For VIM :
* vim : set softtabstop = 4 shiftwidth = 4 tabstop = 4 expandtab :
*/