2007-05-22 22:07:05 +00:00
/*
* Copyright ( c ) 2007 , Anthony Minessale II
* All rights reserved .
2010-11-01 00:47:43 +01:00
*
2007-05-22 22:07:05 +00:00
* Redistribution and use in source and binary forms , with or without
* modification , are permitted provided that the following conditions
* are met :
2010-11-01 00:47:43 +01:00
*
2007-05-22 22:07:05 +00:00
* * Redistributions of source code must retain the above copyright
* notice , this list of conditions and the following disclaimer .
2010-11-01 00:47:43 +01:00
*
2007-05-22 22:07:05 +00:00
* * 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 .
2010-11-01 00:47:43 +01:00
*
2007-05-22 22:07:05 +00:00
* * 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 .
2010-11-01 00:47:43 +01:00
*
*
2007-05-22 22:07:05 +00:00
* 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 .
*/
2010-11-01 00:47:43 +01:00
# ifdef HAVE_CONFIG_H
# include "config.h"
2010-01-13 17:53:22 +00:00
# endif
2010-11-01 00:47:43 +01:00
# include <private/ftdm_core.h>
# include <libisdn/Q931.h>
# include <libisdn/Q921.h>
2010-01-13 17:53:22 +00:00
2007-05-23 19:47:35 +00:00
# ifdef WIN32
# include <windows.h>
# else
# include <sys/time.h>
# endif
2010-11-01 00:47:43 +01:00
# include "ftmod_isdn.h"
2009-04-08 21:58:51 +00:00
2007-05-23 18:25:23 +00:00
# define LINE "--------------------------------------------------------------------------------"
2007-05-22 22:07:05 +00:00
2008-08-22 16:55:01 +00:00
/* helper macros */
2010-01-15 19:22:49 +00:00
# define FTDM_SPAN_IS_BRI(x) ((x)->trunk_type == FTDM_TRUNK_BRI || (x)->trunk_type == FTDM_TRUNK_BRI_PTMP)
# define FTDM_SPAN_IS_NT(x) (((ftdm_isdn_data_t *)(x)->signal_data)->mode == Q921_NT)
2008-08-22 16:55:01 +00:00
2010-11-01 00:47:43 +01:00
# define DEFAULT_NATIONAL_PREFIX "0"
# define DEFAULT_INTERNATIONAL_PREFIX "00"
2008-10-12 22:07:45 +00:00
2010-11-01 00:47:43 +01:00
/*****************************************************************************************
* PCAP
* Based on Helmut Kuper ' s ( < helmut . kuper @ ewetel . de > ) implementation ,
* but using a different approach ( needs a recent libpcap + wireshark )
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
# ifdef HAVE_PCAP
# include <arpa/inet.h> /* htons() */
2009-01-30 11:37:11 +00:00
# include <pcap.h>
2010-11-01 00:47:43 +01:00
# define PCAP_SNAPLEN 1500
2009-01-30 11:37:11 +00:00
2010-11-01 00:47:43 +01:00
struct pcap_context {
pcap_dumper_t * dump ; /*!< pcap file handle */
pcap_t * handle ; /*!< pcap lib context */
char * filename ; /*!< capture file name */
} ;
static inline ftdm_status_t isdn_pcap_is_open ( struct ftdm_isdn_data * isdn )
2009-01-30 11:37:11 +00:00
{
2010-11-01 00:47:43 +01:00
return ( isdn - > pcap ) ? 1 : 0 ;
}
2009-01-30 11:37:11 +00:00
2010-11-01 00:47:43 +01:00
static inline ftdm_status_t isdn_pcap_capture_both ( struct ftdm_isdn_data * isdn )
{
return ( ( isdn - > flags & ( FTDM_ISDN_CAPTURE | FTDM_ISDN_CAPTURE_L3ONLY ) ) = = FTDM_ISDN_CAPTURE ) ? 1 : 0 ;
}
2009-01-30 11:37:11 +00:00
2010-11-01 00:47:43 +01:00
static inline ftdm_status_t isdn_pcap_capture_l3only ( struct ftdm_isdn_data * isdn )
{
return ( ( isdn - > flags & FTDM_ISDN_CAPTURE ) & & ( isdn - > flags & FTDM_ISDN_CAPTURE_L3ONLY ) ) ? 1 : 0 ;
}
2009-01-30 11:37:11 +00:00
2010-11-01 00:47:43 +01:00
static ftdm_status_t isdn_pcap_open ( struct ftdm_isdn_data * isdn , char * filename )
{
struct pcap_context * pcap = NULL ;
2009-01-30 11:37:11 +00:00
2010-11-01 00:47:43 +01:00
if ( ! isdn ) {
return FTDM_FAIL ;
}
if ( ftdm_strlen_zero ( filename ) ) {
return FTDM_FAIL ;
}
pcap = malloc ( sizeof ( struct pcap_context ) ) ;
if ( ! pcap ) {
ftdm_log ( FTDM_LOG_ERROR , " Failed to allocate isdn pcap context \n " ) ;
return FTDM_FAIL ;
}
memset ( pcap , 0 , sizeof ( struct pcap_context ) ) ;
pcap - > filename = strdup ( filename ) ;
pcap - > handle = pcap_open_dead ( DLT_LINUX_LAPD , PCAP_SNAPLEN ) ;
if ( ! pcap - > handle ) {
ftdm_log ( FTDM_LOG_ERROR , " Failed to open pcap handle \n " ) ;
goto error ;
}
pcap - > dump = pcap_dump_open ( pcap - > handle , pcap - > filename ) ;
if ( ! pcap - > dump ) {
ftdm_log ( FTDM_LOG_ERROR , " Failed to open capture file: %s \n " , pcap_geterr ( pcap - > handle ) ) ;
goto error ;
}
ftdm_log ( FTDM_LOG_INFO , " Capture file \" %s \" opened \n " , pcap - > filename ) ;
isdn - > pcap = pcap ;
return FTDM_SUCCESS ;
error :
if ( pcap - > handle ) {
pcap_close ( pcap - > handle ) ;
}
if ( pcap - > filename ) {
free ( pcap - > filename ) ;
}
free ( pcap ) ;
return FTDM_FAIL ;
2009-01-30 11:37:11 +00:00
}
2010-11-01 00:47:43 +01:00
static ftdm_status_t isdn_pcap_close ( struct ftdm_isdn_data * isdn )
2009-01-30 11:37:11 +00:00
{
2010-11-01 00:47:43 +01:00
struct pcap_context * pcap = NULL ;
long size ;
2009-01-30 11:37:11 +00:00
2010-11-01 00:47:43 +01:00
if ( ! isdn | | ! isdn - > pcap ) {
return FTDM_FAIL ;
2009-01-30 11:37:11 +00:00
}
2010-11-01 00:47:43 +01:00
pcap = isdn - > pcap ;
isdn - > flags & = ~ ( FTDM_ISDN_CAPTURE | FTDM_ISDN_CAPTURE_L3ONLY ) ;
isdn - > pcap = NULL ;
pcap_dump_flush ( pcap - > dump ) ;
size = pcap_dump_ftell ( pcap - > dump ) ;
ftdm_log ( FTDM_LOG_INFO , " File \" %s \" captured %ld bytes of data \n " , pcap - > filename , size ) ;
pcap_dump_close ( pcap - > dump ) ;
pcap_close ( pcap - > handle ) ;
free ( pcap - > filename ) ;
free ( pcap ) ;
2009-01-30 11:37:11 +00:00
2010-01-15 19:22:49 +00:00
return FTDM_SUCCESS ;
2009-01-30 11:37:11 +00:00
}
2010-11-01 00:47:43 +01:00
static inline void isdn_pcap_start ( struct ftdm_isdn_data * isdn )
2009-01-30 11:37:11 +00:00
{
2010-11-01 00:47:43 +01:00
if ( ! isdn - > pcap )
return ;
2009-01-30 11:37:11 +00:00
2010-11-01 00:47:43 +01:00
isdn - > flags | = FTDM_ISDN_CAPTURE ;
}
2009-01-30 11:37:11 +00:00
2010-11-01 00:47:43 +01:00
static inline void isdn_pcap_stop ( struct ftdm_isdn_data * isdn )
{
isdn - > flags & = ~ FTDM_ISDN_CAPTURE ;
}
# ifndef ETH_P_LAPD
# define ETH_P_LAPD 0x0030
# endif
2009-01-30 11:37:11 +00:00
2010-11-01 00:47:43 +01:00
struct isdn_sll_hdr {
uint16_t slltype ;
uint16_t sllhatype ;
uint16_t slladdrlen ;
uint8_t slladdr [ 8 ] ;
uint16_t sllproto ;
} ;
2009-01-30 11:37:11 +00:00
2010-11-01 00:47:43 +01:00
/* Fake Q.921 I-frame */
//static const char q921_fake_frame[] = { 0x00, 0x00, 0x00, 0x00 };
2009-01-30 11:37:11 +00:00
2010-11-01 00:47:43 +01:00
enum {
ISDN_PCAP_INCOMING = 0 ,
ISDN_PCAP_INCOMING_BCAST = 1 ,
ISDN_PCAP_OUTGOING = 4 ,
} ;
2009-01-30 11:37:11 +00:00
2010-11-01 00:47:43 +01:00
static ftdm_status_t isdn_pcap_write ( struct ftdm_isdn_data * isdn , unsigned char * buf , ftdm_ssize_t len , int direction )
{
unsigned char frame [ PCAP_SNAPLEN ] ;
struct pcap_context * pcap ;
struct isdn_sll_hdr * sll_hdr = ( struct isdn_sll_hdr * ) frame ;
struct pcap_pkthdr hdr ;
int offset = sizeof ( struct isdn_sll_hdr ) ;
int nbytes ;
if ( ! isdn | | ! isdn - > pcap | | ! buf | | ! len ) {
return FTDM_FAIL ;
}
2009-01-30 11:37:11 +00:00
2010-11-01 00:47:43 +01:00
pcap = isdn - > pcap ;
/* Update SLL header */
sll_hdr - > slltype = htons ( direction ) ;
sll_hdr - > sllhatype = 0 ;
sll_hdr - > slladdrlen = 1 ;
sll_hdr - > slladdr [ 0 ] = ( isdn - > mode = = Q921_NT ) ? 1 : 0 ; /* TODO: NT/TE */
sll_hdr - > sllproto = htons ( ETH_P_LAPD ) ;
#if 0
/* Q.931-only mode: copy fake Q.921 header */
if ( isdn - > flags & FTDM_ISDN_CAPTURE_L3ONLY ) {
/* copy fake q921 header */
memcpy ( frame + offset , q921_fake_frame , sizeof ( q921_fake_frame ) ) ;
offset + = sizeof ( q921_fake_frame ) ;
}
# endif
2009-02-25 09:49:54 +00:00
2010-11-01 00:47:43 +01:00
/* Copy data */
nbytes = ( len > ( PCAP_SNAPLEN - offset ) ) ? ( PCAP_SNAPLEN - offset ) : len ;
memcpy ( frame + offset , buf , nbytes ) ;
2009-01-30 11:37:11 +00:00
2010-11-01 00:47:43 +01:00
/* Update timestamp */
memset ( & hdr , 0 , sizeof ( struct pcap_pkthdr ) ) ;
gettimeofday ( & hdr . ts , NULL ) ;
hdr . caplen = offset + nbytes ;
hdr . len = hdr . caplen ;
2009-01-30 11:37:11 +00:00
2010-11-01 00:47:43 +01:00
/* Write packet */
pcap_dump ( ( unsigned char * ) pcap - > dump , & hdr , frame ) ;
2009-01-30 11:37:11 +00:00
2010-11-01 00:47:43 +01:00
return FTDM_SUCCESS ;
2009-01-30 11:37:11 +00:00
}
2010-11-01 00:47:43 +01:00
# endif /* HAVE_PCAP */
2009-01-30 11:37:11 +00:00
2010-11-01 00:47:43 +01:00
static L2ULONG ftdm_time_now ( void )
{
return ( L2ULONG ) ftdm_current_time_in_ms ( ) ;
}
2009-06-08 20:01:29 +00:00
/**
2010-11-01 00:47:43 +01:00
* \ brief Returns the signalling status on a channel
* \ param ftdmchan Channel to get status on
* \ param status Pointer to set signalling status
* \ return Success or failure
2009-06-08 20:01:29 +00:00
*/
2010-11-01 00:47:43 +01:00
static FIO_CHANNEL_GET_SIG_STATUS_FUNCTION ( isdn_get_channel_sig_status )
2009-01-30 11:37:11 +00:00
{
2010-11-01 00:47:43 +01:00
* status = FTDM_SIG_STATE_DOWN ;
ftdm_isdn_data_t * isdn_data = ftdmchan - > span - > signal_data ;
if ( ftdm_test_flag ( isdn_data , FTDM_ISDN_RUNNING ) ) {
* status = FTDM_SIG_STATE_UP ;
}
2010-01-15 19:22:49 +00:00
return FTDM_SUCCESS ;
2009-01-30 11:37:11 +00:00
}
2009-06-08 20:01:29 +00:00
/**
2010-11-01 00:47:43 +01:00
* \ brief Returns the signalling status on a span
* \ param span Span to get status on
* \ param status Pointer to set signalling status
* \ return Success or failure
2009-06-08 20:01:29 +00:00
*/
2010-11-01 00:47:43 +01:00
static FIO_SPAN_GET_SIG_STATUS_FUNCTION ( isdn_get_span_sig_status )
2007-05-23 19:47:35 +00:00
{
2010-11-01 00:47:43 +01:00
* status = FTDM_SIG_STATE_DOWN ;
ftdm_isdn_data_t * isdn_data = span - > signal_data ;
if ( ftdm_test_flag ( isdn_data , FTDM_ISDN_RUNNING ) ) {
* status = FTDM_SIG_STATE_UP ;
}
return FTDM_SUCCESS ;
2007-05-23 19:47:35 +00:00
}
2009-06-08 20:01:29 +00:00
/**
2010-11-01 00:47:43 +01:00
* \ brief Create outgoing channel
* \ param ftdmchan Channel to create outgoing call on
* \ return Success or failure
2009-06-08 20:01:29 +00:00
*/
2010-01-15 20:35:11 +00:00
static FIO_CHANNEL_OUTGOING_CALL_FUNCTION ( isdn_outgoing_call )
2007-06-14 04:44:44 +00:00
{
2010-01-15 19:22:49 +00:00
ftdm_status_t status = FTDM_SUCCESS ;
ftdm_set_flag ( ftdmchan , FTDM_CHANNEL_OUTBOUND ) ;
ftdm_set_state_locked ( ftdmchan , FTDM_CHANNEL_STATE_DIALING ) ;
2007-06-14 04:44:44 +00:00
return status ;
}
2009-06-08 20:01:29 +00:00
/**
2010-11-01 00:47:43 +01:00
* \ brief Create outgoing channel , let module select the channel to use
* \ param span Span to create outgoing call on
* \ param caller_data
* \ return Success or failure
2009-06-08 20:01:29 +00:00
*/
2010-11-01 00:47:43 +01:00
# ifdef __TODO__
2010-01-15 20:35:11 +00:00
static FIO_CHANNEL_REQUEST_FUNCTION ( isdn_channel_request )
2008-05-05 22:54:40 +00:00
{
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_ProgInd Progress ; /* Progress Indicator IE */
2010-01-15 19:22:49 +00:00
ftdm_status_t status = FTDM_FAIL ;
ftdm_isdn_data_t * isdn_data = span - > signal_data ;
2008-05-05 22:54:40 +00:00
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
*/
2010-01-15 19:22:49 +00:00
ftdm_channel_command ( span - > channels [ chan_id ] , FTDM_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 ;
2010-01-15 19:22:49 +00:00
BearerCap . UIL1Prot = ( codec = = FTDM_CODEC_ALAW ) ? Q931_UIL1P_G711A : Q931_UIL1P_G711U ; /* U-law = 2, A-law = 3 */
2010-11-01 00:47:43 +01:00
gen - > BearerCap = Q931AppendIE ( gen , ( L3UCHAR * ) & BearerCap ) ;
2008-05-05 22:54:40 +00:00
2008-08-22 16:55:01 +00:00
/*
* Channel ID IE
*/
Q931InitIEChanID ( & ChanID ) ;
2010-01-15 19:22:49 +00:00
ChanID . IntType = FTDM_SPAN_IS_BRI ( span ) ? 0 : 1 ; /* PRI = 1, BRI = 0 */
2008-08-22 16:55:01 +00:00
2010-11-01 00:47:43 +01:00
if ( ! FTDM_SPAN_IS_NT ( span ) ) {
2010-01-15 19:22:49 +00:00
ChanID . PrefExcl = ( isdn_data - > opts & FTDM_ISDN_OPT_SUGGEST_CHANNEL ) ? 0 : 1 ; /* 0 = preferred, 1 exclusive */
2008-08-22 16:55:01 +00:00
} else {
ChanID . PrefExcl = 1 ; /* always exclusive in NT-mode */
}
2010-11-01 00:47:43 +01:00
if ( ChanID . IntType ) {
2008-08-22 16:55:01 +00:00
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 */
}
2010-11-01 00:47:43 +01:00
gen - > ChanID = Q931AppendIE ( 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 */
2010-11-01 00:47:43 +01:00
gen - > ProgInd = Q931AppendIE ( gen , ( L3UCHAR * ) & Progress ) ;
2008-08-22 16:55:01 +00:00
/*
* Display IE
*/
2010-11-01 00:47:43 +01:00
if ( ! ( isdn_data - > opts & FTDM_ISDN_OPT_OMIT_DISPLAY_IE ) & & FTDM_SPAN_IS_NT ( span ) ) {
2009-03-19 23:06:04 +00:00
Q931InitIEDisplay ( & Display ) ;
Display . Size = Display . Size + ( unsigned char ) strlen ( caller_data - > cid_name ) ;
2010-11-01 00:47:43 +01:00
gen - > Display = Q931AppendIE ( gen , ( L3UCHAR * ) & Display ) ;
2009-03-19 23:06:04 +00:00
ptrDisplay = Q931GetIEPtr ( gen - > Display , gen - > buf ) ;
2010-01-15 19:22:49 +00:00
ftdm_copy_string ( ( char * ) ptrDisplay - > Display , caller_data - > cid_name , strlen ( caller_data - > cid_name ) + 1 ) ;
2009-03-19 23:06:04 +00:00
}
2008-05-05 22:54:40 +00:00
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 ) ;
2010-11-01 00:47:43 +01:00
gen - > CallingNum = Q931AppendIE ( gen , ( L3UCHAR * ) & CallingNum ) ;
2008-05-05 22:54:40 +00:00
ptrCallingNum = Q931GetIEPtr ( gen - > CallingNum , gen - > buf ) ;
2010-01-15 19:22:49 +00:00
ftdm_copy_string ( ( char * ) ptrCallingNum - > Digit , caller_data - > cid_num . digits , strlen ( caller_data - > cid_num . digits ) + 1 ) ;
2008-05-05 22:54:40 +00:00
2008-08-22 16:55:01 +00:00
/*
* Called number IE
*/
Q931InitIECalledNum ( & CalledNum ) ;
CalledNum . TypNum = Q931_TON_UNKNOWN ;
CalledNum . NumPlanID = Q931_NUMPLAN_E164 ;
2010-11-01 00:47:43 +01:00
CalledNum . Size = CalledNum . Size + ( unsigned char ) strlen ( caller_data - > ani . digits ) ;
gen - > CalledNum = Q931AppendIE ( gen , ( L3UCHAR * ) & CalledNum ) ;
2008-05-05 22:54:40 +00:00
ptrCalledNum = Q931GetIEPtr ( gen - > CalledNum , gen - > buf ) ;
2010-11-01 00:47:43 +01:00
ftdm_copy_string ( ( char * ) ptrCalledNum - > Digit , caller_data - > ani . digits , strlen ( caller_data - > ani . digits ) + 1 ) ;
2008-05-05 22:54:40 +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 = 1 ; /* Telephony = 1, Fax G2+3 = 4, Fax G4 = 65 (Class I)/ 68 (Class II or III) */
2010-11-01 00:47:43 +01:00
gen - > HLComp = Q931AppendIE ( gen , ( L3UCHAR * ) & HLComp ) ;
2008-08-22 16:55:01 +00:00
2010-01-15 19:22:49 +00:00
caller_data - > call_state = FTDM_CALLER_STATE_DIALING ;
2010-11-01 00:47:43 +01:00
Q931Rx43 ( & isdn_data - > q931 , gen , gen - > Size ) ;
2008-05-05 22:54:40 +00:00
isdn_data - > outbound_crv [ gen - > CRV ] = caller_data ;
2010-01-15 19:22:49 +00:00
//isdn_data->channels_local_crv[gen->CRV] = ftdmchan;
2008-05-05 22:54:40 +00:00
2010-01-15 19:22:49 +00:00
while ( ftdm_running ( ) & & caller_data - > call_state = = FTDM_CALLER_STATE_DIALING ) {
ftdm_sleep ( 1 ) ;
2010-11-01 00:47:43 +01:00
2008-05-05 22:54:40 +00:00
if ( ! - - sanity ) {
2010-01-15 19:22:49 +00:00
caller_data - > call_state = FTDM_CALLER_STATE_FAIL ;
2008-05-05 22:54:40 +00:00
break ;
}
}
isdn_data - > outbound_crv [ gen - > CRV ] = NULL ;
2010-11-01 00:47:43 +01:00
2010-01-15 19:22:49 +00:00
if ( caller_data - > call_state = = FTDM_CALLER_STATE_SUCCESS ) {
ftdm_channel_t * new_chan = NULL ;
2008-05-05 22:54:40 +00:00
int fail = 1 ;
2010-11-01 00:47:43 +01:00
2008-05-05 22:54:40 +00:00
new_chan = NULL ;
2010-01-15 19:22:49 +00:00
if ( caller_data - > chan_id < FTDM_MAX_CHANNELS_SPAN & & caller_data - > chan_id < = span - > chan_count ) {
2008-09-10 15:25:02 +00:00
new_chan = span - > channels [ caller_data - > chan_id ] ;
2008-05-05 22:54:40 +00:00
}
2010-01-15 19:22:49 +00:00
if ( new_chan & & ( status = ftdm_channel_open_chan ( new_chan ) = = FTDM_SUCCESS ) ) {
if ( ftdm_test_flag ( new_chan , FTDM_CHANNEL_INUSE ) | | new_chan - > state ! = FTDM_CHANNEL_STATE_DOWN ) {
if ( new_chan - > state = = FTDM_CHANNEL_STATE_DOWN | | new_chan - > state > = FTDM_CHANNEL_STATE_TERMINATING ) {
2008-05-05 22:54:40 +00:00
int x = 0 ;
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_LOG_WARNING , " Channel %d:%d ~ %d:%d is already in use waiting for it to become available. \n " ) ;
2010-11-01 00:47:43 +01:00
2008-05-05 22:54:40 +00:00
for ( x = 0 ; x < 200 ; x + + ) {
2010-01-15 19:22:49 +00:00
if ( ! ftdm_test_flag ( new_chan , FTDM_CHANNEL_INUSE ) ) {
2008-05-05 22:54:40 +00:00
break ;
}
2010-01-15 19:22:49 +00:00
ftdm_sleep ( 5 ) ;
2008-05-05 22:54:40 +00:00
}
}
2010-01-15 19:22:49 +00:00
if ( ftdm_test_flag ( new_chan , FTDM_CHANNEL_INUSE ) ) {
ftdm_log ( FTDM_LOG_ERROR , " Channel %d:%d ~ %d:%d is already in use. \n " ,
2008-05-05 22:54:40 +00:00
new_chan - > span_id ,
new_chan - > chan_id ,
new_chan - > physical_span_id ,
new_chan - > physical_chan_id
) ;
new_chan = NULL ;
}
}
2010-01-15 19:22:49 +00:00
if ( new_chan & & new_chan - > state = = FTDM_CHANNEL_STATE_DOWN ) {
2008-05-05 22:54:40 +00:00
isdn_data - > channels_local_crv [ gen - > CRV ] = new_chan ;
memset ( & new_chan - > caller_data , 0 , sizeof ( new_chan - > caller_data ) ) ;
2010-01-15 19:22:49 +00:00
ftdm_set_flag ( new_chan , FTDM_CHANNEL_OUTBOUND ) ;
ftdm_set_state_locked ( new_chan , FTDM_CHANNEL_STATE_DIALING ) ;
2008-05-05 22:54:40 +00:00
switch ( gen - > MesType ) {
case Q931mes_ALERTING :
2010-01-15 19:22:49 +00:00
new_chan - > init_state = FTDM_CHANNEL_STATE_PROGRESS_MEDIA ;
2008-05-05 22:54:40 +00:00
break ;
case Q931mes_CONNECT :
2010-01-15 19:22:49 +00:00
new_chan - > init_state = FTDM_CHANNEL_STATE_UP ;
2008-05-05 22:54:40 +00:00
break ;
default :
2010-01-15 19:22:49 +00:00
new_chan - > init_state = FTDM_CHANNEL_STATE_PROGRESS ;
2008-05-05 22:54:40 +00:00
break ;
}
fail = 0 ;
2010-11-01 00:47:43 +01:00
}
2008-05-05 22:54:40 +00:00
}
2010-11-01 00:47:43 +01:00
2008-05-05 22:54:40 +00:00
if ( ! fail ) {
2010-01-15 19:22:49 +00:00
* ftdmchan = new_chan ;
return FTDM_SUCCESS ;
2008-05-05 22:54:40 +00:00
} 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?
2010-01-15 19:22:49 +00:00
cause . Value = ( unsigned char ) FTDM_CAUSE_WRONG_CALL_STATE ;
2008-05-05 22:54:40 +00:00
* cause . Diag = ' \0 ' ;
2010-11-01 00:47:43 +01:00
gen - > Cause = Q931AppendIE ( gen , ( L3UCHAR * ) & cause ) ;
Q931Rx43 ( & isdn_data - > q931 , gen , gen - > Size ) ;
2008-05-05 22:54:40 +00:00
if ( gen - > CRV ) {
Q931ReleaseCRV ( & isdn_data - > q931 , gen - > CRV ) ;
}
2010-11-01 00:47:43 +01:00
2008-05-05 22:54:40 +00:00
if ( new_chan ) {
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_LOG_CRIT , " Channel is busy \n " ) ;
2008-05-05 22:54:40 +00:00
} else {
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_LOG_CRIT , " Failed to open channel for new setup message \n " ) ;
2008-05-05 22:54:40 +00:00
}
}
}
2010-11-01 00:47:43 +01:00
2010-01-15 19:22:49 +00:00
* ftdmchan = NULL ;
return FTDM_FAIL ;
2008-05-05 22:54:40 +00:00
}
2010-11-01 00:47:43 +01:00
# endif /* __TODO__ */
2008-05-05 22:54:40 +00:00
2010-01-15 19:22:49 +00:00
static L3INT ftdm_isdn_931_err ( void * pvt , L3INT id , L3INT p1 , L3INT p2 )
2007-05-22 22:07:05 +00:00
{
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_LOG_ERROR , " ERROR: [%s] [%d] [%d] \n " , q931_error_to_name ( id ) , p1 , p2 ) ;
2007-05-22 22:07:05 +00:00
return 0 ;
}
2009-06-08 20:01:29 +00:00
/**
2010-11-01 00:47:43 +01:00
* \ brief The new call event handler
* \ note W000t ! ! ! \ o / ; D
* \ todo A lot
*/
static void ftdm_isdn_call_event ( struct Q931_Call * call , struct Q931_CallEvent * event , void * priv )
{
Q931_TrunkInfo_t * trunk = NULL ;
ftdm_isdn_data_t * isdn_data = NULL ;
ftdm_span_t * zspan = NULL ;
assert ( call ) ;
assert ( event ) ;
trunk = Q931CallGetTrunk ( call ) ;
assert ( trunk ) ;
zspan = Q931CallGetPrivate ( call ) ;
if ( ! zspan ) {
zspan = priv ;
Q931CallSetPrivate ( call , zspan ) ;
}
assert ( zspan ) ;
isdn_data = zspan - > signal_data ;
if ( Q931CallIsGlobal ( call ) ) {
/*
* Global event
*/
ftdm_log ( FTDM_LOG_DEBUG , " Received global event from Q.931 \n " ) ;
}
else {
ftdm_channel_t * ftdmchan = NULL ;
ftdm_sigmsg_t sig ;
int call_crv = Q931CallGetCRV ( call ) ;
int type ;
/*
* Call - specific event
*/
ftdm_log ( FTDM_LOG_DEBUG , " Received call-specific event from Q.931 for call %d [%hu] \n " , Q931CallGetCRV ( call ) , Q931CallGetCRV ( call ) ) ;
/*
* Try to get associated zap channel
* and init sigmsg struct if there is one
*/
ftdmchan = Q931CallIsOutgoing ( call ) ? isdn_data - > channels_local_crv [ call_crv ] : isdn_data - > channels_remote_crv [ call_crv ] ;
if ( ftdmchan ) {
memset ( & sig , 0 , sizeof ( ftdm_sigmsg_t ) ) ;
sig . chan_id = ftdmchan - > chan_id ;
sig . span_id = ftdmchan - > span_id ;
sig . channel = ftdmchan ;
}
type = Q931CallEventGetType ( event ) ;
if ( type = = Q931_EVENT_TYPE_CRV ) {
ftdm_log ( FTDM_LOG_DEBUG , " \t CRV event \n " ) ;
switch ( Q931CallEventGetId ( event ) ) {
case Q931_EVENT_RELEASE_CRV :
{
/* WARNING contains old interface code, yuck! */
if ( ! ftdmchan ) {
ftdm_log ( FTDM_LOG_DEBUG , " Call %d [0x%x] not associated to zap channel \n " , call_crv , call_crv ) ;
return ;
}
if ( ftdmchan - > state ! = FTDM_CHANNEL_STATE_DOWN ) {
ftdm_log ( FTDM_LOG_DEBUG , " Channel %d:%d not in DOWN state, cleaning up \n " ,
ftdmchan - > span_id , ftdmchan - > chan_id ) ;
/*
* Send hangup signal to mod_openzap
*/
if ( ! sig . channel - > caller_data . hangup_cause ) {
sig . channel - > caller_data . hangup_cause = FTDM_CAUSE_NORMAL_CLEARING ;
}
sig . event_id = FTDM_SIGEVENT_STOP ;
isdn_data - > sig_cb ( & sig ) ;
/* Release zap channel */
ftdm_set_state_locked ( ftdmchan , FTDM_CHANNEL_STATE_DOWN ) ;
}
return ;
}
break ;
default :
ftdm_log ( FTDM_LOG_ERROR , " Unknown CRV event: %d \n " , Q931CallEventGetId ( event ) ) ;
return ;
}
}
else if ( type = = Q931_EVENT_TYPE_TIMER ) {
struct Q931_CallTimerEvent * timer_evt = Q931CallEventGetData ( event ) ;
ftdm_log ( FTDM_LOG_DEBUG , " \t Timer event \n " ) ;
assert ( timer_evt - > id ) ;
switch ( timer_evt - > id ) {
case Q931_TIMER_T303 :
/*
* SETUP timeout
*
* TE - mode : Q931_EVENT_SETUP_CONFIRM ( error )
* NT - mode : Q931_EVENT_RELEASE_INDICATION
*/
{
/* WARNING contains old interface code, yuck! */
if ( ! ftdmchan ) {
ftdm_log ( FTDM_LOG_ERROR , " Call %d [0x%x] not associated to zap channel \n " , call_crv , call_crv ) ;
return ;
}
ftdm_log ( FTDM_LOG_DEBUG , " Call setup failed on channel %d:%d \n " , ftdmchan - > span_id , ftdmchan - > chan_id ) ;
/*
* Send signal to mod_openzap
*/
sig . channel - > caller_data . hangup_cause = FTDM_CAUSE_NETWORK_OUT_OF_ORDER ;
sig . event_id = FTDM_SIGEVENT_STOP ;
isdn_data - > sig_cb ( & sig ) ;
/* Release zap channel */
ftdm_set_state_locked ( ftdmchan , FTDM_CHANNEL_STATE_DOWN ) ;
return ;
}
break ;
default :
ftdm_log ( FTDM_LOG_ERROR , " Unhandled timer event %d \n " , timer_evt - > id ) ;
}
}
else if ( type = = Q931_EVENT_TYPE_MESSAGE ) {
struct Q931_CallMessageEvent * msg_evt = Q931CallEventGetData ( event ) ;
ftdm_log ( FTDM_LOG_DEBUG , " \t Message event \n " ) ;
assert ( msg_evt ) ;
/*
* Slowly move stuff from the old event handler into this part . . .
*/
switch ( Q931CallEventGetId ( event ) ) {
case Q931_EVENT_SETUP_CONFIRM :
case Q931_EVENT_SETUP_COMPLETE_INDICATION : /* CONNECT */
{
/* WARNING contains old interface code, yuck! */
if ( ! ftdmchan ) {
ftdm_log ( FTDM_LOG_ERROR , " Call %d [0x%x] not associated to zap channel \n " , call_crv , call_crv ) ;
return ;
}
ftdm_set_state_locked ( ftdmchan , FTDM_CHANNEL_STATE_UP ) ;
}
break ;
default :
ftdm_log ( FTDM_LOG_DEBUG , " Not yet handled message event %d \n " , Q931CallEventGetId ( event ) ) ;
}
}
else {
ftdm_log ( FTDM_LOG_ERROR , " Unknown event type %d \n " , type ) ;
}
}
}
/**
* Copy callednum , readding prefix as needed
2009-06-08 20:01:29 +00:00
*/
2010-11-01 00:47:43 +01:00
static void __isdn_get_number ( const char * digits , const int ton , char * buf , int size )
{
int offset = 0 ;
if ( ! digits | | ! buf | | size < = 0 ) {
return ;
}
switch ( ton ) {
case Q931_TON_NATIONAL :
offset = strlen ( DEFAULT_NATIONAL_PREFIX ) ;
memcpy ( buf , DEFAULT_NATIONAL_PREFIX , offset ) ;
break ;
case Q931_TON_INTERNATIONAL :
offset = strlen ( DEFAULT_INTERNATIONAL_PREFIX ) ;
memcpy ( buf , DEFAULT_INTERNATIONAL_PREFIX , offset ) ;
break ;
default :
break ;
}
strncpy ( & buf [ offset ] , digits , size - ( offset + 1 ) ) ;
buf [ size - 1 ] = ' \0 ' ;
}
# define isdn_get_number(num, buf) \
__isdn_get_number ( ( const char * ) ( num ) - > Digit , ( num ) - > TypNum , ( char * ) buf , sizeof ( buf ) )
/**
* \ brief The old call event handler ( err , call message handler )
* \ todo This one must die !
*/
static L3INT ftdm_isdn_931_34 ( void * pvt , struct Q931_Call * call , Q931mes_Generic * msg , int mlen )
2007-05-22 22:07:05 +00:00
{
2010-01-15 19:22:49 +00:00
ftdm_span_t * span = ( ftdm_span_t * ) pvt ;
ftdm_isdn_data_t * isdn_data = span - > signal_data ;
2007-05-22 22:07:05 +00:00
Q931mes_Generic * gen = ( Q931mes_Generic * ) msg ;
2010-11-01 00:47:43 +01:00
int chan_id = 0 ;
2008-09-07 17:39:36 +00:00
int chan_hunt = 0 ;
2010-01-15 19:22:49 +00:00
ftdm_channel_t * ftdmchan = NULL ;
2010-11-01 00:47:43 +01:00
// ftdm_caller_data_t *caller_data = NULL;
2008-05-05 22:54:40 +00:00
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-09-07 17:39:36 +00:00
/* "any" channel specified */
if ( chanid - > InfoChanSel = = 3 ) {
chan_hunt + + ;
}
2010-01-15 19:22:49 +00:00
} else if ( FTDM_SPAN_IS_NT ( span ) ) {
2008-09-07 17:39:36 +00:00
/* no channel ie */
chan_hunt + + ;
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 ) ;
2010-11-01 00:47:43 +01:00
/*
* Support code for the new event handling system
* Remove this as soon as we have the new api to set up calls
*/
# ifdef __OLD__
if ( gen - > CRV ) {
struct Q931_Call * call ;
call = Q931GetCallByCRV ( & isdn_data - > q931 , gen - > CRV ) ;
if ( call & & ! Q931CallGetPrivate ( call ) ) {
ftdm_log ( FTDM_LOG_DEBUG , " Storing reference to current span in call %d [0x%x] \n " , gen - > CRV , gen - > CRV ) ;
Q931CallSetPrivate ( call , span ) ;
}
}
# else
if ( call & & ! Q931CallGetPrivate ( call ) ) {
ftdm_log ( FTDM_LOG_DEBUG , " Storing reference to current span in call %d [0x%x] \n " , gen - > CRV , gen - > CRV ) ;
Q931CallSetPrivate ( call , span ) ;
}
# endif
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_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
2010-11-01 00:47:43 +01:00
# ifdef __TODO_OR_REMOVE__
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 :
{
2010-01-15 19:22:49 +00:00
caller_data - > call_state = FTDM_CALLER_STATE_SUCCESS ;
2008-05-05 22:54:40 +00:00
}
break ;
default :
2010-01-15 19:22:49 +00:00
caller_data - > call_state = FTDM_CALLER_STATE_FAIL ;
2008-05-05 22:54:40 +00:00
break ;
}
2010-11-01 00:47:43 +01:00
2008-05-05 22:54:40 +00:00
return 0 ;
2007-06-14 03:54:02 +00:00
}
2010-11-01 00:47:43 +01:00
# endif
2008-02-04 20:27:54 +00:00
if ( gen - > CRVFlag ) {
2010-01-15 19:22:49 +00:00
ftdmchan = isdn_data - > channels_local_crv [ gen - > CRV ] ;
2008-02-04 20:27:54 +00:00
} else {
2010-01-15 19:22:49 +00:00
ftdmchan = isdn_data - > channels_remote_crv [ gen - > CRV ] ;
2008-02-04 20:27:54 +00:00
}
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_LOG_DEBUG , " ftdmchan %x (%d:%d) source isdn_data->channels_%s_crv[%#hx] \n " , ftdmchan , ftdmchan ? ftdmchan - > span_id : - 1 , ftdmchan ? ftdmchan - > chan_id : - 1 , 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 ) ;
2010-01-15 19:22:49 +00:00
if ( ftdmchan ) {
2007-06-25 16:00:24 +00:00
switch ( changestatus - > NewStatus ) {
case 0 : /* change status to "in service" */
2007-06-26 01:02:36 +00:00
{
2010-01-15 19:22:49 +00:00
ftdm_clear_flag_locked ( ftdmchan , FTDM_CHANNEL_SUSPENDED ) ;
ftdm_log ( FTDM_LOG_DEBUG , " Channel %d:%d in service \n " , ftdmchan - > span_id , ftdmchan - > chan_id ) ;
ftdm_set_state_locked ( ftdmchan , FTDM_CHANNEL_STATE_RESTART ) ;
2007-06-26 00:47:05 +00:00
}
2007-06-25 16:00:24 +00:00
break ;
2010-11-01 00:47:43 +01:00
case 1 :
2007-06-26 01:02:36 +00:00
{ /* change status to "maintenance" */
2010-01-15 19:22:49 +00:00
ftdm_set_flag_locked ( ftdmchan , FTDM_CHANNEL_SUSPENDED ) ;
ftdm_set_state_locked ( ftdmchan , FTDM_CHANNEL_STATE_SUSPENDED ) ;
2007-06-26 01:02:36 +00:00
}
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" */
2010-01-15 19:22:49 +00:00
ftdm_set_flag_locked ( ftdmchan , FTDM_CHANNEL_SUSPENDED ) ;
ftdm_set_state_locked ( ftdmchan , FTDM_CHANNEL_STATE_SUSPENDED ) ;
2007-06-26 01:02:36 +00:00
}
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 ) {
2010-01-15 19:22:49 +00:00
ftdmchan = span - > channels [ chan_id ] ;
2008-02-27 17:37:29 +00:00
}
2010-01-15 19:22:49 +00:00
if ( ftdmchan ) {
ftdm_set_state_locked ( ftdmchan , FTDM_CHANNEL_STATE_RESTART ) ;
2007-06-25 16:00:24 +00:00
} else {
uint32_t i ;
2010-11-01 00:47:43 +01:00
2008-09-10 19:43:55 +00:00
for ( i = 1 ; i < span - > chan_count ; i + + ) {
2010-11-01 00:47:43 +01:00
/* Skip channels that are down and D-Channels (#OpenZAP-39) */
if ( span - > channels [ i ] - > state = = FTDM_CHANNEL_STATE_DOWN | | span - > channels [ i ] - > type = = FTDM_CHAN_TYPE_DQ921 )
continue ;
2010-01-15 19:22:49 +00:00
ftdm_set_state_locked ( ( span - > channels [ i ] ) , FTDM_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 " ;
2010-01-15 19:22:49 +00:00
if ( ftdmchan ) {
if ( ftdmchan - > state = = FTDM_CHANNEL_STATE_TERMINATING | | ftdmchan - > state = = FTDM_CHANNEL_STATE_HANGUP ) {
2008-08-22 16:55:01 +00:00
if ( gen - > MesType = = Q931mes_RELEASE ) {
2010-01-15 19:22:49 +00:00
ftdm_set_state_locked ( ftdmchan , FTDM_CHANNEL_STATE_HANGUP_COMPLETE ) ;
2008-08-22 16:55:01 +00:00
} else {
2010-01-15 19:22:49 +00:00
ftdm_set_state_locked ( ftdmchan , FTDM_CHANNEL_STATE_DOWN ) ;
2008-08-22 16:55:01 +00:00
}
}
2010-01-15 19:22:49 +00:00
else if ( ( gen - > MesType = = Q931mes_RELEASE & & ftdmchan - > state < = FTDM_CHANNEL_STATE_UP ) | |
( gen - > MesType = = Q931mes_RELEASE_COMPLETE & & ftdmchan - > state = = FTDM_CHANNEL_STATE_DIALING ) ) {
2008-08-22 16:55:01 +00:00
/*
* Don ' t keep inbound channels open if the remote side hangs up before we answered
*/
Q931ie_Cause * cause = Q931GetIEPtr ( gen - > Cause , gen - > buf ) ;
2010-01-15 19:22:49 +00:00
ftdm_sigmsg_t sig ;
ftdm_status_t status ;
2008-08-22 16:55:01 +00:00
memset ( & sig , 0 , sizeof ( sig ) ) ;
2010-01-15 19:22:49 +00:00
sig . chan_id = ftdmchan - > chan_id ;
sig . span_id = ftdmchan - > span_id ;
sig . channel = ftdmchan ;
sig . channel - > caller_data . hangup_cause = ( cause ) ? cause - > Value : FTDM_CAUSE_NORMAL_UNSPECIFIED ;
2008-08-22 16:55:01 +00:00
2010-01-15 19:22:49 +00:00
sig . event_id = FTDM_SIGEVENT_STOP ;
2010-11-01 00:47:43 +01:00
status = isdn_data - > sig_cb ( & sig ) ;
2008-08-22 16:55:01 +00:00
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_LOG_DEBUG , " Received %s in state %s, requested hangup for channel %d:%d \n " , what , ftdm_channel_state2str ( ftdmchan - > state ) , ftdmchan - > span_id , chan_id ) ;
2008-08-22 16:55:01 +00:00
}
else {
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_LOG_DEBUG , " Ignoring %s on channel %d \n " , what , chan_id ) ;
2008-04-19 15:57:19 +00:00
}
2008-01-28 22:17:34 +00:00
} else {
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_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 :
{
2010-01-15 19:22:49 +00:00
if ( ftdmchan ) {
2008-02-28 17:23:10 +00:00
Q931ie_Cause * cause = Q931GetIEPtr ( gen - > Cause , gen - > buf ) ;
2010-01-15 19:22:49 +00:00
ftdmchan - > caller_data . hangup_cause = cause - > Value ;
ftdm_set_state_locked ( ftdmchan , FTDM_CHANNEL_STATE_TERMINATING ) ;
2008-02-28 17:23:10 +00:00
} else {
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_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 :
{
2010-01-15 19:22:49 +00:00
if ( ftdmchan ) {
ftdm_set_state_locked ( ftdmchan , FTDM_CHANNEL_STATE_PROGRESS_MEDIA ) ;
2008-01-28 22:17:34 +00:00
} else {
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_LOG_CRIT , " Received Alerting with no matching channel %d \n " , chan_id ) ;
2008-01-28 22:17:34 +00:00
}
2007-06-25 16:00:24 +00:00
}
break ;
case Q931mes_PROGRESS :
{
2010-01-15 19:22:49 +00:00
if ( ftdmchan ) {
ftdm_set_state_locked ( ftdmchan , FTDM_CHANNEL_STATE_PROGRESS ) ;
2008-01-28 22:17:34 +00:00
} else {
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_LOG_CRIT , " Received Progress with no matching channel %d \n " , chan_id ) ;
2008-01-28 22:17:34 +00:00
}
2007-06-25 16:00:24 +00:00
}
break ;
case Q931mes_CONNECT :
2010-11-01 00:47:43 +01:00
#if 0 /* Handled by new event code */
2007-06-25 16:00:24 +00:00
{
2010-01-15 19:22:49 +00:00
if ( ftdmchan ) {
ftdm_set_state_locked ( ftdmchan , FTDM_CHANNEL_STATE_UP ) ;
2008-08-22 16:55:01 +00:00
2010-11-01 00:47:43 +01:00
#if 0 /* Auto-Ack is enabled, we actually don't need this */
2008-08-22 16:55:01 +00:00
gen - > MesType = Q931mes_CONNECT_ACKNOWLEDGE ;
gen - > CRVFlag = 0 ; /* outbound */
2010-11-01 00:47:43 +01:00
Q931Rx43 ( & isdn_data - > q931 , gen , gen - > Size ) ;
# endif
2008-01-28 22:17:34 +00:00
} else {
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_LOG_CRIT , " Received Connect with no matching channel %d \n " , chan_id ) ;
2008-01-28 22:17:34 +00:00
}
2007-06-25 16:00:24 +00:00
}
2010-11-01 00:47:43 +01:00
# endif
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 ;
2008-09-07 17:39:36 +00:00
int fail_cause = 0 ;
int overlap_dial = 0 ;
2007-06-25 16:00:24 +00:00
uint32_t cplen = mlen ;
2008-08-22 16:55:01 +00:00
2010-01-15 19:22:49 +00:00
if ( ftdmchan & & ftdmchan = = isdn_data - > channels_remote_crv [ gen - > CRV ] ) {
ftdm_log ( FTDM_LOG_INFO , " Duplicate SETUP message(?) for Channel %d:%d ~ %d:%d in state %s [ignoring] \n " ,
ftdmchan - > span_id ,
ftdmchan - > chan_id ,
ftdmchan - > physical_span_id ,
ftdmchan - > physical_chan_id ,
ftdm_channel_state2str ( ftdmchan - > state ) ) ;
2008-08-22 16:55:01 +00:00
break ;
}
2010-11-01 00:47:43 +01:00
2010-01-15 19:22:49 +00:00
ftdmchan = NULL ;
2008-09-07 17:39:36 +00:00
/*
* Channel selection for incoming calls :
*/
2010-01-15 19:22:49 +00:00
if ( FTDM_SPAN_IS_NT ( span ) & & chan_hunt ) {
2010-11-01 00:47:43 +01:00
int x ;
2008-09-07 17:39:36 +00:00
/*
* In NT - mode with channel selection " any " ,
* try to find a free channel
*/
for ( x = 1 ; x < = span - > chan_count ; x + + ) {
2010-01-15 19:22:49 +00:00
ftdm_channel_t * zc = span - > channels [ x ] ;
2008-09-07 17:39:36 +00:00
2010-01-15 19:22:49 +00:00
if ( ! ftdm_test_flag ( zc , FTDM_CHANNEL_INUSE ) & & zc - > state = = FTDM_CHANNEL_STATE_DOWN ) {
ftdmchan = zc ;
2008-09-07 17:39:36 +00:00
break ;
}
}
}
2010-01-15 19:22:49 +00:00
else if ( ! FTDM_SPAN_IS_NT ( span ) & & chan_hunt ) {
2008-09-07 17:39:36 +00:00
/*
* In TE - mode this ( " any " ) is invalid
*/
2010-01-15 19:22:49 +00:00
fail_cause = FTDM_CAUSE_CHANNEL_UNACCEPTABLE ;
2008-09-07 17:39:36 +00:00
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_LOG_ERROR , " Invalid channel selection in incoming call (network side didn't specify a channel) \n " ) ;
2008-09-07 17:39:36 +00:00
}
else {
/*
* Otherwise simply try to select the channel we ' ve been told
*
* TODO : NT mode is abled to select a different channel if the one chosen
* by the TE side is already in use
*/
2010-01-15 19:22:49 +00:00
if ( chan_id > 0 & & chan_id < FTDM_MAX_CHANNELS_SPAN & & chan_id < = span - > chan_count ) {
ftdmchan = span - > channels [ chan_id ] ;
2008-09-07 17:39:36 +00:00
}
else {
/* invalid channel id */
2010-01-15 19:22:49 +00:00
fail_cause = FTDM_CAUSE_CHANNEL_UNACCEPTABLE ;
2008-09-07 17:39:36 +00:00
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_LOG_ERROR , " Invalid channel selection in incoming call (none selected or out of bounds) \n " ) ;
2008-09-07 17:39:36 +00:00
}
}
if ( ! callednum | | ! strlen ( ( char * ) callednum - > Digit ) ) {
2010-01-15 19:22:49 +00:00
if ( FTDM_SPAN_IS_NT ( span ) ) {
ftdm_log ( FTDM_LOG_NOTICE , " No destination number found, assuming overlap dial \n " ) ;
2008-09-07 17:39:36 +00:00
overlap_dial + + ;
}
else {
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_LOG_ERROR , " No destination number found \n " ) ;
ftdmchan = NULL ;
2008-09-07 17:39:36 +00:00
}
2008-04-01 21:49:35 +00:00
}
2007-06-25 16:00:24 +00:00
2010-01-15 19:22:49 +00:00
if ( ftdmchan ) {
if ( ftdm_test_flag ( ftdmchan , FTDM_CHANNEL_INUSE ) | | ftdmchan - > state ! = FTDM_CHANNEL_STATE_DOWN ) {
if ( ftdmchan - > state = = FTDM_CHANNEL_STATE_DOWN | | ftdmchan - > state > = FTDM_CHANNEL_STATE_TERMINATING ) {
2008-04-01 21:49:35 +00:00
int x = 0 ;
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_LOG_WARNING , " Channel %d:%d ~ %d:%d is already in use waiting for it to become available. \n " ,
ftdmchan - > span_id ,
ftdmchan - > chan_id ,
ftdmchan - > physical_span_id ,
ftdmchan - > physical_chan_id ) ;
2008-04-01 21:49:35 +00:00
for ( x = 0 ; x < 200 ; x + + ) {
2010-01-15 19:22:49 +00:00
if ( ! ftdm_test_flag ( ftdmchan , FTDM_CHANNEL_INUSE ) ) {
2008-04-01 21:49:35 +00:00
break ;
}
2010-01-15 19:22:49 +00:00
ftdm_sleep ( 5 ) ;
2008-04-01 21:49:35 +00:00
}
}
2010-01-15 19:22:49 +00:00
if ( ftdm_test_flag ( ftdmchan , FTDM_CHANNEL_INUSE ) ) {
ftdm_log ( FTDM_LOG_ERROR , " Channel %d:%d ~ %d:%d is already in use. \n " ,
ftdmchan - > span_id ,
ftdmchan - > chan_id ,
ftdmchan - > physical_span_id ,
ftdmchan - > physical_chan_id
2008-04-01 21:49:35 +00:00
) ;
2010-01-15 19:22:49 +00:00
ftdmchan = NULL ;
2008-04-01 21:49:35 +00:00
}
}
2007-06-25 16:00:24 +00:00
2010-01-15 19:22:49 +00:00
if ( ftdmchan & & ftdmchan - > state = = FTDM_CHANNEL_STATE_DOWN ) {
isdn_data - > channels_remote_crv [ gen - > CRV ] = ftdmchan ;
memset ( & ftdmchan - > caller_data , 0 , sizeof ( ftdmchan - > caller_data ) ) ;
2007-06-25 16:00:24 +00:00
2010-01-15 19:22:49 +00:00
if ( ftdmchan - > mod_data ) {
memset ( ftdmchan - > mod_data , 0 , sizeof ( ftdm_isdn_bchan_data_t ) ) ;
2008-10-12 22:07:45 +00:00
}
2010-11-01 00:47:43 +01:00
/* copy number readd prefix as needed */
isdn_get_number ( callingnum , ftdmchan - > caller_data . cid_num . digits ) ;
isdn_get_number ( callingnum , ftdmchan - > caller_data . cid_name ) ;
isdn_get_number ( callingnum , ftdmchan - > caller_data . ani . digits ) ;
// ftdm_set_string(ftdmchan->caller_data.cid_num.digits, (char *)callingnum->Digit);
// ftdm_set_string(ftdmchan->caller_data.cid_name, (char *)callingnum->Digit);
// ftdm_set_string(ftdmchan->caller_data.ani.digits, (char *)callingnum->Digit);
2008-10-12 22:07:45 +00:00
if ( ! overlap_dial ) {
2010-11-01 00:47:43 +01:00
// ftdm_set_string(ftdmchan->caller_data.dnis.digits, (char *)callednum->Digit);
isdn_get_number ( callednum , ftdmchan - > caller_data . dnis . digits ) ;
2008-10-12 22:07:45 +00:00
}
2010-11-01 00:47:43 +01:00
# ifdef __TODO_OR_REMOVE__
2010-01-15 19:22:49 +00:00
ftdmchan - > caller_data . CRV = gen - > CRV ;
2010-11-01 00:47:43 +01:00
# endif
2010-01-15 19:22:49 +00:00
if ( cplen > sizeof ( ftdmchan - > caller_data . raw_data ) ) {
cplen = sizeof ( ftdmchan - > caller_data . raw_data ) ;
2007-06-25 16:00:24 +00:00
}
gen - > CRVFlag = ! ( gen - > CRVFlag ) ;
2010-01-15 19:22:49 +00:00
memcpy ( ftdmchan - > caller_data . raw_data , msg , cplen ) ;
ftdmchan - > caller_data . raw_data_len = cplen ;
2007-06-25 16:00:24 +00:00
fail = 0 ;
2010-11-01 00:47:43 +01: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?
2010-01-15 19:22:49 +00:00
cause . Value = ( unsigned char ) ( ( fail_cause ) ? fail_cause : FTDM_CAUSE_WRONG_CALL_STATE ) ;
2008-04-01 21:49:35 +00:00
* cause . Diag = ' \0 ' ;
2010-11-01 00:47:43 +01:00
gen - > Cause = Q931AppendIE ( gen , ( L3UCHAR * ) & cause ) ;
Q931Rx43 ( & isdn_data - > q931 , gen , gen - > Size ) ;
2008-04-01 21:49:35 +00:00
2008-04-02 20:24:54 +00:00
if ( gen - > CRV ) {
Q931ReleaseCRV ( & isdn_data - > q931 , gen - > CRV ) ;
}
2010-01-15 19:22:49 +00:00
if ( ftdmchan ) {
ftdm_log ( FTDM_LOG_CRIT , " Channel is busy \n " ) ;
2008-02-29 17:55:29 +00:00
} else {
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_LOG_CRIT , " Failed to open channel for new setup message \n " ) ;
2008-02-29 17:55:29 +00:00
}
2010-11-01 00:47:43 +01:00
2008-08-22 16:55:01 +00:00
} else {
2008-09-07 17:39:36 +00:00
Q931ie_ChanID ChanID ;
2008-08-22 16:55:01 +00:00
/*
2008-09-07 17:39:36 +00:00
* Update Channel ID IE
2008-08-22 16:55:01 +00:00
*/
2008-09-07 17:39:36 +00:00
Q931InitIEChanID ( & ChanID ) ;
2010-01-15 19:22:49 +00:00
ChanID . IntType = FTDM_SPAN_IS_BRI ( ftdmchan - > span ) ? 0 : 1 ; /* PRI = 1, BRI = 0 */
ChanID . PrefExcl = FTDM_SPAN_IS_NT ( ftdmchan - > span ) ? 1 : 0 ; /* Exclusive in NT-mode = 1, Preferred otherwise = 0 */
2008-09-07 17:39:36 +00:00
if ( ChanID . IntType ) {
ChanID . InfoChanSel = 1 ; /* None = 0, See Slot = 1, Any = 3 */
ChanID . ChanMapType = 3 ; /* B-Chan */
2010-01-15 19:22:49 +00:00
ChanID . ChanSlot = ( unsigned char ) ftdmchan - > chan_id ;
2008-09-07 17:39:36 +00:00
} else {
2010-01-15 19:22:49 +00:00
ChanID . InfoChanSel = ( unsigned char ) ftdmchan - > chan_id & 0x03 ; /* None = 0, B1 = 1, B2 = 2, Any = 3 */
2008-09-07 17:39:36 +00:00
}
2010-11-01 00:47:43 +01:00
gen - > ChanID = Q931AppendIE ( gen , ( L3UCHAR * ) & ChanID ) ;
2008-09-07 17:39:36 +00:00
if ( overlap_dial ) {
Q931ie_ProgInd progress ;
/*
* Setup Progress indicator
*/
progress . IEId = Q931ie_PROGRESS_INDICATOR ;
progress . Size = sizeof ( Q931ie_ProgInd ) ;
2010-11-01 00:47:43 +01:00
progress . CodStand = Q931_CODING_ITU ; /* ITU */
2008-09-07 17:39:36 +00:00
progress . Location = 1 ; /* private network serving the local user */
progress . ProgDesc = 8 ; /* call is not end-to-end isdn = 1, in-band information available = 8 */
2010-11-01 00:47:43 +01:00
gen - > ProgInd = Q931AppendIE ( gen , ( L3UCHAR * ) & progress ) ;
2008-09-07 17:39:36 +00:00
/*
* Send SETUP ACK
*/
gen - > MesType = Q931mes_SETUP_ACKNOWLEDGE ;
gen - > CRVFlag = 1 ; /* inbound call */
2010-11-01 00:47:43 +01:00
Q931Rx43 ( & isdn_data - > q931 , gen , gen - > Size ) ;
2008-09-07 17:39:36 +00:00
2010-01-15 19:22:49 +00:00
ftdm_set_state_locked ( ftdmchan , FTDM_CHANNEL_STATE_DIALTONE ) ;
2008-09-07 17:39:36 +00:00
} else {
/*
* Advance to RING state
*/
2010-01-15 19:22:49 +00:00
ftdm_set_state_locked ( ftdmchan , FTDM_CHANNEL_STATE_RING ) ;
2008-09-07 17:39:36 +00:00
}
2008-08-22 16:55:01 +00:00
}
}
break ;
case Q931mes_CALL_PROCEEDING :
{
2010-01-15 19:22:49 +00:00
if ( ftdmchan ) {
ftdm_log ( FTDM_LOG_CRIT , " Received CALL PROCEEDING message for channel %d \n " , chan_id ) ;
ftdm_set_state_locked ( ftdmchan , FTDM_CHANNEL_STATE_PROGRESS ) ;
2008-08-22 16:55:01 +00:00
} else {
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_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-09-26 18:28:28 +00:00
case Q931mes_CONNECT_ACKNOWLEDGE :
{
2010-01-15 19:22:49 +00:00
if ( ftdmchan ) {
ftdm_log ( FTDM_LOG_DEBUG , " Received CONNECT_ACK message for channel %d \n " , chan_id ) ;
2008-09-26 18:28:28 +00:00
} else {
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_LOG_DEBUG , " Received CONNECT_ACK with no matching channel %d \n " , chan_id ) ;
2008-09-26 18:28:28 +00:00
}
}
break ;
2008-08-22 16:55:01 +00:00
2008-09-07 17:39:36 +00:00
case Q931mes_INFORMATION :
{
2010-01-15 19:22:49 +00:00
if ( ftdmchan ) {
ftdm_log ( FTDM_LOG_CRIT , " Received INFORMATION message for channel %d \n " , ftdmchan - > chan_id ) ;
2008-09-07 17:39:36 +00:00
2010-01-15 19:22:49 +00:00
if ( ftdmchan - > state = = FTDM_CHANNEL_STATE_DIALTONE ) {
2008-09-07 17:39:36 +00:00
char digit = ' \0 ' ;
/*
* overlap dial digit indication
*/
if ( Q931IsIEPresent ( gen - > CalledNum ) ) {
2010-01-15 19:22:49 +00:00
ftdm_isdn_bchan_data_t * data = ( ftdm_isdn_bchan_data_t * ) ftdmchan - > mod_data ;
2008-09-07 17:39:36 +00:00
Q931ie_CalledNum * callednum = Q931GetIEPtr ( gen - > CalledNum , gen - > buf ) ;
int pos ;
digit = callednum - > Digit [ strlen ( ( char * ) callednum - > Digit ) - 1 ] ;
if ( digit = = ' # ' ) {
callednum - > Digit [ strlen ( ( char * ) callednum - > Digit ) - 1 ] = ' \0 ' ;
}
/* TODO: make this more safe with strncat() */
2010-11-01 00:47:43 +01:00
pos = strlen ( ftdmchan - > caller_data . dnis . digits ) ;
2010-01-15 19:22:49 +00:00
strcat ( & ftdmchan - > caller_data . dnis . digits [ pos ] , ( char * ) callednum - > Digit ) ;
2008-09-07 17:39:36 +00:00
2008-10-12 22:07:45 +00:00
/* update timer */
2010-01-15 19:22:49 +00:00
data - > digit_timeout = ftdm_time_now ( ) + isdn_data - > digit_timeout ;
2008-10-12 22:07:45 +00:00
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_LOG_DEBUG , " Received new overlap digit (%s), destination number: %s \n " , callednum - > Digit , ftdmchan - > caller_data . dnis . digits ) ;
2008-09-07 17:39:36 +00:00
}
if ( Q931IsIEPresent ( gen - > SendComplete ) | | digit = = ' # ' ) {
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_LOG_DEBUG , " Leaving overlap dial mode \n " ) ;
2008-09-07 17:39:36 +00:00
2010-01-15 19:22:49 +00:00
ftdm_set_state_locked ( ftdmchan , FTDM_CHANNEL_STATE_RING ) ;
2008-09-07 17:39:36 +00:00
}
}
} else {
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_LOG_CRIT , " Received INFORMATION message with no matching channel \n " ) ;
2008-09-07 17:39:36 +00:00
}
}
break ;
2007-06-25 16:00:24 +00:00
default :
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_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 ;
}
2010-01-15 19:22:49 +00:00
static int ftdm_isdn_921_23 ( void * pvt , Q921DLMsg_t ind , L2UCHAR tei , L2UCHAR * msg , L2INT mlen )
2007-05-24 01:55:26 +00:00
{
2010-11-01 00:47:43 +01:00
ftdm_span_t * span = pvt ;
ftdm_isdn_data_t * isdn_data = span - > signal_data ;
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 ) ) ;
2010-11-01 00:47:43 +01:00
ftdm_log ( FTDM_LOG_DEBUG , " READ %d \n %s \n %s \n \n " , ( int ) mlen - offset , LINE , bb ) ;
# ifdef HAVE_PCAP
if ( isdn_pcap_capture_l3only ( isdn_data ) ) {
isdn_pcap_write ( isdn_data , msg , mlen , ( ind = = Q921_DL_UNIT_DATA ) ? ISDN_PCAP_INCOMING_BCAST : ISDN_PCAP_INCOMING ) ;
}
2009-02-03 17:00:13 +00:00
# endif
2008-08-22 16:55:01 +00:00
default :
2010-11-01 00:47:43 +01:00
ret = Q931Rx23 ( & isdn_data - > q931 , ind , tei , msg , mlen ) ;
2008-08-22 16:55:01 +00:00
if ( ret ! = 0 )
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_LOG_DEBUG , " 931 parse error [%d] [%s] \n " , ret , q931_error_to_name ( ret ) ) ;
2008-08-22 16:55:01 +00:00
break ;
}
2007-05-24 01:59:26 +00:00
return ( ( ret > = 0 ) ? 1 : 0 ) ;
2007-05-24 01:55:26 +00:00
}
2010-01-15 19:22:49 +00:00
static int ftdm_isdn_921_21 ( void * pvt , L2UCHAR * msg , L2INT mlen )
2007-05-22 22:07:05 +00:00
{
2010-01-15 19:22:49 +00:00
ftdm_span_t * span = ( ftdm_span_t * ) pvt ;
ftdm_size_t len = ( ftdm_size_t ) mlen ;
ftdm_isdn_data_t * isdn_data = span - > signal_data ;
2007-06-25 05:26:37 +00:00
2010-11-01 00:47:43 +01:00
assert ( span ! = NULL ) ;
2007-05-23 18:25:23 +00:00
2010-11-01 00:47:43 +01:00
# ifdef HAVE_PCAP
if ( isdn_pcap_capture_both ( isdn_data ) ) {
isdn_pcap_write ( isdn_data , msg , mlen , ISDN_PCAP_OUTGOING ) ;
}
2007-05-23 18:25:23 +00:00
# endif
2010-01-15 19:22:49 +00:00
return ftdm_channel_write ( isdn_data - > dchan , msg , len , & len ) = = FTDM_SUCCESS ? 0 : - 1 ;
2007-05-22 22:07:05 +00:00
}
2010-01-15 19:22:49 +00:00
static __inline__ void state_advance ( ftdm_channel_t * ftdmchan )
2007-06-13 03:37:55 +00:00
{
2010-01-15 19:22:49 +00:00
Q931mes_Generic * gen = ( Q931mes_Generic * ) ftdmchan - > caller_data . raw_data ;
ftdm_isdn_data_t * isdn_data = ftdmchan - > span - > signal_data ;
ftdm_sigmsg_t sig ;
ftdm_status_t status ;
2007-06-13 03:37:55 +00:00
2010-11-01 00:47:43 +01:00
ftdm_log ( FTDM_LOG_DEBUG , " %d:%d STATE [%s] \n " ,
2010-01-15 19:22:49 +00:00
ftdmchan - > span_id , ftdmchan - > chan_id , ftdm_channel_state2str ( ftdmchan - > state ) ) ;
2007-06-13 03:37:55 +00:00
2007-06-14 03:54:02 +00:00
memset ( & sig , 0 , sizeof ( sig ) ) ;
2010-01-15 19:22:49 +00:00
sig . chan_id = ftdmchan - > chan_id ;
sig . span_id = ftdmchan - > span_id ;
sig . channel = ftdmchan ;
2007-06-13 03:37:55 +00:00
2010-01-15 19:22:49 +00:00
switch ( ftdmchan - > state ) {
case FTDM_CHANNEL_STATE_DOWN :
2007-06-16 04:39:15 +00:00
{
2008-04-02 20:24:54 +00:00
if ( gen - > CRV ) {
2010-01-15 19:22:49 +00:00
if ( ftdm_test_flag ( ftdmchan , FTDM_CHANNEL_OUTBOUND ) ) {
2008-08-22 16:55:01 +00:00
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 ) ;
}
2010-01-15 19:22:49 +00:00
ftdm_channel_done ( ftdmchan ) ;
2007-06-16 04:39:15 +00:00
}
break ;
2010-01-15 19:22:49 +00:00
case FTDM_CHANNEL_STATE_PROGRESS :
2007-06-16 04:39:15 +00:00
{
2010-01-15 19:22:49 +00:00
if ( ftdm_test_flag ( ftdmchan , FTDM_CHANNEL_OUTBOUND ) ) {
sig . event_id = FTDM_SIGEVENT_PROGRESS ;
2010-11-01 00:47:43 +01:00
if ( ( status = isdn_data - > sig_cb ( & sig ) ! = FTDM_SUCCESS ) ) {
2010-01-15 19:22:49 +00:00
ftdm_set_state_locked ( ftdmchan , FTDM_CHANNEL_STATE_HANGUP ) ;
2007-06-16 04:39:15 +00:00
}
} else {
2010-11-01 00:47:43 +01:00
int crv = gen - > CRV ;
Q931InitMesGeneric ( gen ) ;
2008-07-24 20:03:37 +00:00
gen - > MesType = Q931mes_CALL_PROCEEDING ;
2010-11-01 00:47:43 +01:00
gen - > CRV = crv ;
2008-08-22 16:55:01 +00:00
gen - > CRVFlag = 1 ; /* inbound */
2010-01-15 19:22:49 +00:00
if ( FTDM_SPAN_IS_NT ( ftdmchan - > span ) ) {
2008-09-07 17:39:36 +00:00
Q931ie_ChanID ChanID ;
/*
* Set new Channel ID
*/
Q931InitIEChanID ( & ChanID ) ;
2010-01-15 19:22:49 +00:00
ChanID . IntType = FTDM_SPAN_IS_BRI ( ftdmchan - > span ) ? 0 : 1 ; /* PRI = 1, BRI = 0 */
2008-09-07 17:39:36 +00:00
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 */
2010-01-15 19:22:49 +00:00
ChanID . ChanSlot = ( unsigned char ) ftdmchan - > chan_id ;
2008-09-07 17:39:36 +00:00
} else {
2010-01-15 19:22:49 +00:00
ChanID . InfoChanSel = ( unsigned char ) ftdmchan - > chan_id & 0x03 ; /* None = 0, B1 = 1, B2 = 2, Any = 3 */
2008-09-07 17:39:36 +00:00
}
2010-11-01 00:47:43 +01:00
gen - > ChanID = Q931AppendIE ( gen , ( L3UCHAR * ) & ChanID ) ;
2008-09-07 17:39:36 +00:00
}
2010-11-01 00:47:43 +01:00
Q931Rx43 ( & isdn_data - > q931 , gen , gen - > Size ) ;
2007-06-16 04:39:15 +00:00
}
}
break ;
2010-01-15 19:22:49 +00:00
case FTDM_CHANNEL_STATE_DIALTONE :
2008-09-07 17:39:36 +00:00
{
2010-01-15 19:22:49 +00:00
ftdm_isdn_bchan_data_t * data = ( ftdm_isdn_bchan_data_t * ) ftdmchan - > mod_data ;
2008-10-12 22:07:45 +00:00
if ( data ) {
2010-01-15 19:22:49 +00:00
data - > digit_timeout = ftdm_time_now ( ) + isdn_data - > digit_timeout ;
2008-09-07 17:39:36 +00:00
}
}
break ;
2010-01-15 19:22:49 +00:00
case FTDM_CHANNEL_STATE_RING :
2007-06-14 03:54:02 +00:00
{
2010-01-15 19:22:49 +00:00
if ( ! ftdm_test_flag ( ftdmchan , FTDM_CHANNEL_OUTBOUND ) ) {
sig . event_id = FTDM_SIGEVENT_START ;
2010-11-01 00:47:43 +01:00
if ( ( status = isdn_data - > sig_cb ( & sig ) ! = FTDM_SUCCESS ) ) {
2010-01-15 19:22:49 +00:00
ftdm_set_state_locked ( ftdmchan , FTDM_CHANNEL_STATE_HANGUP ) ;
2007-06-16 04:39:15 +00:00
}
2007-06-14 03:54:02 +00:00
}
}
break ;
2010-01-15 19:22:49 +00:00
case FTDM_CHANNEL_STATE_RESTART :
2007-06-14 03:54:02 +00:00
{
2010-01-15 19:22:49 +00:00
ftdmchan - > caller_data . hangup_cause = FTDM_CAUSE_NORMAL_UNSPECIFIED ;
sig . event_id = FTDM_SIGEVENT_RESTART ;
2010-11-01 00:47:43 +01:00
status = isdn_data - > sig_cb ( & sig ) ;
2010-01-15 19:22:49 +00:00
ftdm_set_state_locked ( ftdmchan , FTDM_CHANNEL_STATE_DOWN ) ;
2007-06-14 03:54:02 +00:00
}
break ;
2010-01-15 19:22:49 +00:00
case FTDM_CHANNEL_STATE_PROGRESS_MEDIA :
2007-06-16 04:39:15 +00:00
{
2010-01-15 19:22:49 +00:00
if ( ftdm_test_flag ( ftdmchan , FTDM_CHANNEL_OUTBOUND ) ) {
sig . event_id = FTDM_SIGEVENT_PROGRESS_MEDIA ;
2010-11-01 00:47:43 +01:00
if ( ( status = isdn_data - > sig_cb ( & sig ) ! = FTDM_SUCCESS ) ) {
2010-01-15 19:22:49 +00:00
ftdm_set_state_locked ( ftdmchan , FTDM_CHANNEL_STATE_HANGUP ) ;
2007-06-16 04:39:15 +00:00
}
} else {
2010-01-15 19:22:49 +00:00
if ( ! ftdm_test_flag ( ftdmchan , FTDM_CHANNEL_OPEN ) ) {
if ( ftdm_channel_open_chan ( ftdmchan ) ! = FTDM_SUCCESS ) {
ftdm_set_state_locked ( ftdmchan , FTDM_CHANNEL_STATE_HANGUP ) ;
2008-08-22 16:55:01 +00:00
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 */
2010-11-01 00:47:43 +01:00
Q931Rx43 ( & isdn_data - > q931 , gen , gen - > Size ) ;
2007-06-16 04:39:15 +00:00
}
}
break ;
2010-01-15 19:22:49 +00:00
case FTDM_CHANNEL_STATE_UP :
2007-06-13 03:37:55 +00:00
{
2010-01-15 19:22:49 +00:00
if ( ftdm_test_flag ( ftdmchan , FTDM_CHANNEL_OUTBOUND ) ) {
sig . event_id = FTDM_SIGEVENT_UP ;
2010-11-01 00:47:43 +01:00
if ( ( status = isdn_data - > sig_cb ( & sig ) ! = FTDM_SUCCESS ) ) {
2010-01-15 19:22:49 +00:00
ftdm_set_state_locked ( ftdmchan , FTDM_CHANNEL_STATE_HANGUP ) ;
2007-06-16 04:39:15 +00:00
}
} else {
2010-01-15 19:22:49 +00:00
if ( ! ftdm_test_flag ( ftdmchan , FTDM_CHANNEL_OPEN ) ) {
if ( ftdm_channel_open_chan ( ftdmchan ) ! = FTDM_SUCCESS ) {
ftdm_set_state_locked ( ftdmchan , FTDM_CHANNEL_STATE_HANGUP ) ;
2008-08-22 16:55:01 +00:00
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 */
2010-11-01 00:47:43 +01:00
Q931Rx43 ( & isdn_data - > q931 , gen , ftdmchan - > caller_data . raw_data_len ) ;
2007-06-16 04:39:15 +00:00
}
2007-06-13 03:37:55 +00:00
}
break ;
2010-01-15 19:22:49 +00:00
case FTDM_CHANNEL_STATE_DIALING :
if ( ! ( isdn_data - > opts & FTDM_ISDN_OPT_SUGGEST_CHANNEL ) ) {
2008-05-21 18:58:14 +00:00
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_ProgInd Progress ; /* Progress Indicator IE */
int codec = 0 ;
/*
* get codec type
*/
2010-01-15 19:22:49 +00:00
ftdm_channel_command ( ftdmchan - > span - > channels [ ftdmchan - > chan_id ] , FTDM_COMMAND_GET_NATIVE_CODEC , & codec ) ;
2008-08-22 16:55:01 +00:00
/*
* Q .931 Setup Message
2010-11-01 00:47:43 +01:00
*/
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 ;
2010-01-15 19:22:49 +00:00
BearerCap . UIL1Prot = ( codec = = FTDM_CODEC_ALAW ) ? 3 : 2 ; /* U-law = 2, A-law = 3 */
2010-11-01 00:47:43 +01:00
gen - > BearerCap = Q931AppendIE ( gen , ( L3UCHAR * ) & BearerCap ) ;
2008-05-21 18:58:14 +00:00
2008-08-22 16:55:01 +00:00
/*
* ChannelID IE
*/
Q931InitIEChanID ( & ChanID ) ;
2010-01-15 19:22:49 +00:00
ChanID . IntType = FTDM_SPAN_IS_BRI ( ftdmchan - > span ) ? 0 : 1 ; /* PRI = 1, BRI = 0 */
ChanID . PrefExcl = FTDM_SPAN_IS_NT ( ftdmchan - > span ) ? 1 : 0 ; /* Exclusive in NT-mode = 1, Preferred otherwise = 0 */
2008-08-22 16:55:01 +00:00
if ( ChanID . IntType ) {
ChanID . InfoChanSel = 1 ; /* None = 0, See Slot = 1, Any = 3 */
ChanID . ChanMapType = 3 ; /* B-Chan */
2010-01-15 19:22:49 +00:00
ChanID . ChanSlot = ( unsigned char ) ftdmchan - > chan_id ;
2008-08-22 16:55:01 +00:00
} else {
2010-01-15 19:22:49 +00:00
ChanID . InfoChanSel = ( unsigned char ) ftdmchan - > chan_id & 0x03 ; /* None = 0, B1 = 1, B2 = 2, Any = 3 */
2008-08-22 16:55:01 +00:00
}
2010-11-01 00:47:43 +01:00
gen - > ChanID = Q931AppendIE ( 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 */
2010-11-01 00:47:43 +01:00
gen - > ProgInd = Q931AppendIE ( gen , ( L3UCHAR * ) & Progress ) ;
2008-08-22 16:55:01 +00:00
/*
* Display IE
2009-03-19 23:06:04 +00:00
*/
2010-11-01 00:47:43 +01:00
if ( ! ( isdn_data - > opts & FTDM_ISDN_OPT_OMIT_DISPLAY_IE ) & & FTDM_SPAN_IS_NT ( ftdmchan - > span ) ) {
2009-03-19 23:06:04 +00:00
Q931InitIEDisplay ( & Display ) ;
2010-01-15 19:22:49 +00:00
Display . Size = Display . Size + ( unsigned char ) strlen ( ftdmchan - > caller_data . cid_name ) ;
2010-11-01 00:47:43 +01:00
gen - > Display = Q931AppendIE ( gen , ( L3UCHAR * ) & Display ) ;
2009-03-19 23:06:04 +00:00
ptrDisplay = Q931GetIEPtr ( gen - > Display , gen - > buf ) ;
2010-01-15 19:22:49 +00:00
ftdm_copy_string ( ( char * ) ptrDisplay - > Display , ftdmchan - > caller_data . cid_name , strlen ( ftdmchan - > caller_data . cid_name ) + 1 ) ;
2009-03-19 23:06:04 +00:00
}
2008-05-21 18:58:14 +00:00
2008-08-22 16:55:01 +00:00
/*
* CallingNum IE
2010-11-01 00:47:43 +01:00
*/
2008-08-22 16:55:01 +00:00
Q931InitIECallingNum ( & CallingNum ) ;
2010-01-15 19:22:49 +00:00
CallingNum . TypNum = ftdmchan - > caller_data . ani . type ;
2008-08-22 16:55:01 +00:00
CallingNum . NumPlanID = Q931_NUMPLAN_E164 ;
CallingNum . PresInd = Q931_PRES_ALLOWED ;
CallingNum . ScreenInd = Q931_SCREEN_USER_NOT_SCREENED ;
2010-01-15 19:22:49 +00:00
CallingNum . Size = CallingNum . Size + ( unsigned char ) strlen ( ftdmchan - > caller_data . cid_num . digits ) ;
2010-11-01 00:47:43 +01:00
gen - > CallingNum = Q931AppendIE ( gen , ( L3UCHAR * ) & CallingNum ) ;
2008-05-21 18:58:14 +00:00
ptrCallingNum = Q931GetIEPtr ( gen - > CallingNum , gen - > buf ) ;
2010-01-15 19:22:49 +00:00
ftdm_copy_string ( ( char * ) ptrCallingNum - > Digit , ftdmchan - > caller_data . cid_num . digits , strlen ( ftdmchan - > caller_data . cid_num . digits ) + 1 ) ;
2008-05-21 18:58:14 +00:00
2008-08-22 16:55:01 +00:00
/*
* CalledNum IE
*/
Q931InitIECalledNum ( & CalledNum ) ;
2010-11-01 00:47:43 +01:00
CalledNum . TypNum = Q931_TON_UNKNOWN ;
2008-08-22 16:55:01 +00:00
CalledNum . NumPlanID = Q931_NUMPLAN_E164 ;
2010-11-01 00:47:43 +01:00
CalledNum . Size = CalledNum . Size + ( unsigned char ) strlen ( ftdmchan - > caller_data . ani . digits ) ;
gen - > CalledNum = Q931AppendIE ( gen , ( L3UCHAR * ) & CalledNum ) ;
2008-05-21 18:58:14 +00:00
ptrCalledNum = Q931GetIEPtr ( gen - > CalledNum , gen - > buf ) ;
2010-11-01 00:47:43 +01:00
ftdm_copy_string ( ( char * ) ptrCalledNum - > Digit , ftdmchan - > caller_data . ani . digits , strlen ( ftdmchan - > 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 */
2010-11-01 00:47:43 +01:00
gen - > HLComp = Q931AppendIE ( gen , ( L3UCHAR * ) & HLComp ) ;
2008-08-22 16:55:01 +00:00
2010-11-01 00:47:43 +01:00
Q931Rx43 ( & isdn_data - > q931 , gen , gen - > Size ) ;
2010-01-15 19:22:49 +00:00
isdn_data - > channels_local_crv [ gen - > CRV ] = ftdmchan ;
2010-11-01 00:47:43 +01:00
/*
* Support code for the new event handling system
* Remove this as soon as we have the new api to set up calls
*/
if ( gen - > CRV ) {
struct Q931_Call * call ;
call = Q931GetCallByCRV ( & isdn_data - > q931 , gen - > CRV ) ;
if ( call ) {
ftdm_log ( FTDM_LOG_DEBUG , " Storing reference to current span in call %d [0x%x] \n " , gen - > CRV , gen - > CRV ) ;
Q931CallSetPrivate ( call , ftdmchan - > span ) ;
}
}
2008-05-21 18:58:14 +00:00
}
break ;
2010-01-15 19:22:49 +00:00
case FTDM_CHANNEL_STATE_HANGUP_COMPLETE :
2008-05-21 18:58:14 +00:00
{
2008-08-22 16:55:01 +00:00
/* reply RELEASE with RELEASE_COMPLETE message */
2010-01-15 19:22:49 +00:00
if ( ftdmchan - > last_state = = FTDM_CHANNEL_STATE_HANGUP ) {
2008-08-22 16:55:01 +00:00
gen - > MesType = Q931mes_RELEASE_COMPLETE ;
2010-11-01 00:47:43 +01:00
Q931Rx43 ( & isdn_data - > q931 , gen , gen - > Size ) ;
2008-08-22 16:55:01 +00:00
}
2010-01-15 19:22:49 +00:00
ftdm_set_state_locked ( ftdmchan , FTDM_CHANNEL_STATE_DOWN ) ;
2007-06-14 04:44:44 +00:00
}
break ;
2010-01-15 19:22:49 +00:00
case FTDM_CHANNEL_STATE_HANGUP :
2007-06-14 03:54:02 +00:00
{
Q931ie_Cause cause ;
2008-08-22 16:55:01 +00:00
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_LOG_DEBUG , " Hangup: Direction %s \n " , ftdm_test_flag ( ftdmchan , FTDM_CHANNEL_OUTBOUND ) ? " Outbound " : " Inbound " ) ;
2008-08-22 16:55:01 +00:00
2010-01-15 19:22:49 +00:00
gen - > CRVFlag = ftdm_test_flag ( ftdmchan , FTDM_CHANNEL_OUTBOUND ) ? 0 : 1 ;
2008-08-22 16:55:01 +00:00
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 )
*/
2010-01-15 19:22:49 +00:00
if ( ftdmchan - > last_state = = FTDM_CHANNEL_STATE_RING ) {
2008-08-22 16:55:01 +00:00
/*
* inbound call [ was : number unknown ( = not found in routing state ) ]
* ( in Q .931 spec terms : Reject request )
*/
gen - > MesType = Q931mes_RELEASE_COMPLETE ;
2010-01-15 19:22:49 +00:00
//cause.Value = (unsigned char) FTDM_CAUSE_UNALLOCATED;
cause . Value = ( unsigned char ) ftdmchan - > caller_data . hangup_cause ;
2008-08-22 16:55:01 +00:00
* cause . Diag = ' \0 ' ;
2010-11-01 00:47:43 +01:00
gen - > Cause = Q931AppendIE ( gen , ( L3UCHAR * ) & cause ) ;
Q931Rx43 ( & isdn_data - > q931 , gen , gen - > Size ) ;
2008-08-22 16:55:01 +00:00
/* we're done, release channel */
2010-01-15 19:22:49 +00:00
//ftdm_set_state_locked(ftdmchan, FTDM_CHANNEL_STATE_HANGUP_COMPLETE);
ftdm_set_state_locked ( ftdmchan , FTDM_CHANNEL_STATE_DOWN ) ;
2008-08-22 16:55:01 +00:00
}
2010-01-15 19:22:49 +00:00
else if ( ftdmchan - > last_state < = FTDM_CHANNEL_STATE_PROGRESS ) {
2008-08-22 16:55:01 +00:00
/*
* just release all unanswered calls [ was : inbound call , remote side hung up before we answered ]
*/
gen - > MesType = Q931mes_RELEASE ;
2010-01-15 19:22:49 +00:00
cause . Value = ( unsigned char ) ftdmchan - > caller_data . hangup_cause ;
2008-08-22 16:55:01 +00:00
* cause . Diag = ' \0 ' ;
2010-11-01 00:47:43 +01:00
gen - > Cause = Q931AppendIE ( gen , ( L3UCHAR * ) & cause ) ;
Q931Rx43 ( & isdn_data - > q931 , gen , gen - > Size ) ;
2008-08-22 16:55:01 +00:00
/* this will be triggered by the RELEASE_COMPLETE reply */
2010-01-15 19:22:49 +00:00
/* ftdm_set_state_locked(ftdmchan, FTDM_CHANNEL_STATE_HANGUP_COMPLETE); */
2008-08-22 16:55:01 +00:00
}
else {
/*
* call connected , hangup
*/
gen - > MesType = Q931mes_DISCONNECT ;
2010-01-15 19:22:49 +00:00
cause . Value = ( unsigned char ) ftdmchan - > caller_data . hangup_cause ;
2008-08-22 16:55:01 +00:00
* cause . Diag = ' \0 ' ;
2010-11-01 00:47:43 +01:00
gen - > Cause = Q931AppendIE ( gen , ( L3UCHAR * ) & cause ) ;
Q931Rx43 ( & isdn_data - > q931 , gen , gen - > Size ) ;
2008-08-22 16:55:01 +00:00
}
2007-06-14 03:54:02 +00:00
}
break ;
2010-01-15 19:22:49 +00:00
case FTDM_CHANNEL_STATE_TERMINATING :
2007-06-14 03:54:02 +00:00
{
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_LOG_DEBUG , " Terminating: Direction %s \n " , ftdm_test_flag ( ftdmchan , FTDM_CHANNEL_OUTBOUND ) ? " Outbound " : " Inbound " ) ;
2008-08-22 16:55:01 +00:00
2010-01-15 19:22:49 +00:00
sig . event_id = FTDM_SIGEVENT_STOP ;
2010-11-01 00:47:43 +01:00
status = isdn_data - > sig_cb ( & sig ) ;
2007-06-14 03:54:02 +00:00
gen - > MesType = Q931mes_RELEASE ;
2010-01-15 19:22:49 +00:00
gen - > CRVFlag = ftdm_test_flag ( ftdmchan , FTDM_CHANNEL_OUTBOUND ) ? 0 : 1 ;
2010-11-01 00:47:43 +01:00
Q931Rx43 ( & isdn_data - > q931 , gen , gen - > Size ) ;
2007-06-14 03:54:02 +00:00
}
2007-06-13 03:37:55 +00:00
default :
break ;
}
}
2010-01-15 19:22:49 +00:00
static __inline__ void check_state ( ftdm_span_t * span )
2007-06-13 03:37:55 +00:00
{
2010-01-15 19:22:49 +00:00
if ( ftdm_test_flag ( span , FTDM_SPAN_STATE_CHANGE ) ) {
2008-03-28 23:22:21 +00:00
uint32_t j ;
2010-01-15 19:22:49 +00:00
ftdm_clear_flag_locked ( span , FTDM_SPAN_STATE_CHANGE ) ;
2008-03-28 23:22:21 +00:00
for ( j = 1 ; j < = span - > chan_count ; j + + ) {
2010-01-15 19:22:49 +00:00
if ( ftdm_test_flag ( ( span - > channels [ j ] ) , FTDM_CHANNEL_STATE_CHANGE ) ) {
ftdm_mutex_lock ( span - > channels [ j ] - > mutex ) ;
ftdm_clear_flag ( ( span - > channels [ j ] ) , FTDM_CHANNEL_STATE_CHANGE ) ;
2008-09-10 15:25:02 +00:00
state_advance ( span - > channels [ j ] ) ;
2010-01-15 19:22:49 +00:00
ftdm_channel_complete_state ( span - > channels [ j ] ) ;
ftdm_mutex_unlock ( span - > channels [ j ] - > mutex ) ;
2008-03-28 23:22:21 +00:00
}
}
}
2007-06-13 03:37:55 +00:00
}
2010-11-01 00:47:43 +01:00
2010-01-15 19:22:49 +00:00
static __inline__ ftdm_status_t process_event ( ftdm_span_t * span , ftdm_event_t * event )
2007-06-23 18:51:10 +00:00
{
2010-11-01 00:47:43 +01:00
ftdm_isdn_data_t * isdn_data = span - > signal_data ;
ftdm_alarm_flag_t alarmbits ;
ftdm_sigmsg_t sig ;
memset ( & sig , 0 , sizeof ( sig ) ) ;
sig . chan_id = event - > channel - > chan_id ;
sig . span_id = event - > channel - > span_id ;
sig . channel = event - > channel ;
ftdm_log ( FTDM_LOG_DEBUG , " EVENT [%s][%d:%d] STATE [%s] \n " ,
2010-01-15 19:22:49 +00:00
ftdm_oob_event2str ( event - > enum_id ) , event - > channel - > span_id , event - > channel - > chan_id , ftdm_channel_state2str ( event - > channel - > state ) ) ;
2007-06-23 18:51:10 +00:00
switch ( event - > enum_id ) {
2010-01-15 19:22:49 +00:00
case FTDM_OOB_ALARM_TRAP :
2007-06-23 18:51:10 +00:00
{
2010-11-01 00:47:43 +01:00
sig . event_id = FTDM_OOB_ALARM_TRAP ;
2010-01-15 19:22:49 +00:00
if ( event - > channel - > state ! = FTDM_CHANNEL_STATE_DOWN ) {
2010-11-01 00:47:43 +01:00
ftdm_set_state_locked ( event - > channel , FTDM_CHANNEL_STATE_RESTART ) ;
2009-03-12 17:22:03 +00:00
}
2010-01-15 19:22:49 +00:00
ftdm_set_flag ( event - > channel , FTDM_CHANNEL_SUSPENDED ) ;
2010-11-01 00:47:43 +01:00
ftdm_channel_get_alarms ( event - > channel , & alarmbits ) ;
isdn_data - > sig_cb ( & sig ) ;
ftdm_log ( FTDM_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 ,
2007-06-23 18:51:10 +00:00
event - > channel - > last_error ) ;
}
break ;
2010-01-15 19:22:49 +00:00
case FTDM_OOB_ALARM_CLEAR :
2007-06-23 18:51:10 +00:00
{
2010-11-01 00:47:43 +01:00
sig . event_id = FTDM_OOB_ALARM_CLEAR ;
2010-01-15 19:22:49 +00:00
ftdm_clear_flag ( event - > channel , FTDM_CHANNEL_SUSPENDED ) ;
2010-11-01 00:47:43 +01:00
ftdm_channel_get_alarms ( event - > channel , & alarmbits ) ;
isdn_data - > sig_cb ( & sig ) ;
}
break ;
# ifdef __BROKEN_BY_FREETDM_CONVERSION__
case FTDM_OOB_DTMF : /* Taken from ozmod_analog, minus the CALLWAITING state handling */
{
const char * digit_str = ( const char * ) event - > data ;
if ( digit_str ) {
fio_event_cb_t event_callback = NULL ;
ftdm_channel_queue_dtmf ( event - > channel , digit_str ) ;
if ( span - > event_callback ) {
event_callback = span - > event_callback ;
} else if ( event - > channel - > event_callback ) {
event_callback = event - > channel - > event_callback ;
}
if ( event_callback ) {
event - > channel - > event_header . channel = event - > channel ;
event - > channel - > event_header . e_type = FTDM_EVENT_DTMF ;
event - > channel - > event_header . data = ( void * ) digit_str ;
event_callback ( event - > channel , & event - > channel - > event_header ) ;
event - > channel - > event_header . e_type = FTDM_EVENT_NONE ;
event - > channel - > event_header . data = NULL ;
}
ftdm_safe_free ( event - > data ) ;
}
2007-06-23 18:51:10 +00:00
}
break ;
2010-11-01 00:47:43 +01:00
# endif
2007-06-23 18:51:10 +00:00
}
2010-01-15 19:22:49 +00:00
return FTDM_SUCCESS ;
2007-06-23 18:51:10 +00:00
}
2010-11-01 00:47:43 +01:00
2010-01-15 19:22:49 +00:00
static __inline__ void check_events ( ftdm_span_t * span )
2007-06-23 18:51:10 +00:00
{
2010-01-15 19:22:49 +00:00
ftdm_status_t status ;
2007-06-23 18:51:10 +00:00
2010-09-23 17:51:45 -03:00
status = ftdm_span_poll_event ( span , 5 , NULL ) ;
2007-06-23 18:51:10 +00:00
switch ( status ) {
2010-01-15 19:22:49 +00:00
case FTDM_SUCCESS :
2007-06-23 18:51:10 +00:00
{
2010-01-15 19:22:49 +00:00
ftdm_event_t * event ;
while ( ftdm_span_next_event ( span , & event ) = = FTDM_SUCCESS ) {
if ( event - > enum_id = = FTDM_OOB_NOOP ) {
2007-06-23 18:51:10 +00:00
continue ;
}
2010-01-15 19:22:49 +00:00
if ( process_event ( span , event ) ! = FTDM_SUCCESS ) {
2007-06-23 18:51:10 +00:00
break ;
}
}
}
break ;
2010-01-15 19:22:49 +00:00
case FTDM_FAIL :
2007-06-23 18:51:10 +00:00
{
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_LOG_DEBUG , " Event Failure! %d \n " , ftdm_running ( ) ) ;
2007-06-23 18:51:10 +00:00
}
break ;
default :
break ;
}
}
2010-11-01 00:47:43 +01:00
2008-09-07 17:39:36 +00:00
static int teletone_handler ( teletone_generation_session_t * ts , teletone_tone_map_t * map )
{
2010-01-15 19:22:49 +00:00
ftdm_buffer_t * dt_buffer = ts - > user_data ;
2008-09-07 17:39:36 +00:00
int wrote ;
if ( ! dt_buffer ) {
return - 1 ;
}
wrote = teletone_mux_tones ( ts , map ) ;
2010-01-15 19:22:49 +00:00
ftdm_buffer_write ( dt_buffer , ts - > buffer , wrote * 2 ) ;
2008-09-07 17:39:36 +00:00
return 0 ;
}
2010-01-15 19:22:49 +00:00
static void * ftdm_isdn_tones_run ( ftdm_thread_t * me , void * obj )
2008-09-07 17:39:36 +00:00
{
2010-01-15 19:22:49 +00:00
ftdm_span_t * span = ( ftdm_span_t * ) obj ;
ftdm_isdn_data_t * isdn_data = span - > signal_data ;
ftdm_buffer_t * dt_buffer = NULL ;
2010-11-01 00:47:43 +01:00
teletone_generation_session_t ts = { { { { 0 } } } } ; ;
2008-09-07 17:39:36 +00:00
unsigned char frame [ 1024 ] ;
2010-11-01 00:47:43 +01:00
int x , interval ;
2008-09-07 17:39:36 +00:00
int offset = 0 ;
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_LOG_DEBUG , " ISDN tones thread starting. \n " ) ;
ftdm_set_flag ( isdn_data , FTDM_ISDN_TONES_RUNNING ) ;
2008-09-07 17:39:36 +00:00
2010-01-15 19:22:49 +00:00
if ( ftdm_buffer_create ( & dt_buffer , 1024 , 1024 , 0 ) ! = FTDM_SUCCESS ) {
2008-09-07 17:39:36 +00:00
snprintf ( isdn_data - > dchan - > last_error , sizeof ( isdn_data - > dchan - > last_error ) , " memory error! " ) ;
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_LOG_ERROR , " MEM ERROR \n " ) ;
2008-09-07 17:39:36 +00:00
goto done ;
}
2010-01-15 19:22:49 +00:00
ftdm_buffer_set_loops ( dt_buffer , - 1 ) ;
2008-09-07 17:39:36 +00:00
/* get a tone generation friendly interval to avoid distortions */
for ( x = 1 ; x < = span - > chan_count ; x + + ) {
2010-01-15 19:22:49 +00:00
if ( span - > channels [ x ] - > type ! = FTDM_CHAN_TYPE_DQ921 ) {
ftdm_channel_command ( span - > channels [ x ] , FTDM_COMMAND_GET_INTERVAL , & interval ) ;
2008-09-07 17:39:36 +00:00
break ;
}
}
2009-03-16 21:04:22 +00:00
if ( ! interval ) {
interval = 20 ;
}
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_LOG_NOTICE , " Tone generating interval %d \n " , interval ) ;
2008-09-07 17:39:36 +00:00
/* init teletone */
teletone_init_session ( & ts , 0 , teletone_handler , dt_buffer ) ;
ts . rate = 8000 ;
ts . duration = ts . rate ;
/* main loop */
2010-11-01 00:47:43 +01:00
while ( ftdm_running ( ) & & ftdm_test_flag ( isdn_data , FTDM_ISDN_RUNNING ) ) {
2010-01-15 19:22:49 +00:00
ftdm_wait_flag_t flags ;
ftdm_status_t status ;
2008-09-07 17:39:36 +00:00
int last_chan_state = 0 ;
int gated = 0 ;
2010-01-15 19:22:49 +00:00
L2ULONG now = ftdm_time_now ( ) ;
2008-09-07 17:39:36 +00:00
/*
* check b - channel states and generate & send tones if neccessary
*/
for ( x = 1 ; x < = span - > chan_count ; x + + ) {
2010-01-15 19:22:49 +00:00
ftdm_channel_t * ftdmchan = span - > channels [ x ] ;
ftdm_size_t len = sizeof ( frame ) , rlen ;
2008-09-07 17:39:36 +00:00
2010-01-15 19:22:49 +00:00
if ( ftdmchan - > type = = FTDM_CHAN_TYPE_DQ921 ) {
2008-09-07 17:39:36 +00:00
continue ;
}
/*
* Generate tones based on current bchan state
* ( Recycle buffer content if succeeding channels share the
* same state , this saves some cpu cycles )
*/
2010-01-15 19:22:49 +00:00
switch ( ftdmchan - > state ) {
case FTDM_CHANNEL_STATE_DIALTONE :
2008-09-07 17:39:36 +00:00
{
2010-01-15 19:22:49 +00:00
ftdm_isdn_bchan_data_t * data = ( ftdm_isdn_bchan_data_t * ) ftdmchan - > mod_data ;
2008-10-12 22:07:45 +00:00
/* check overlap dial timeout first before generating tone */
if ( data & & data - > digit_timeout & & data - > digit_timeout < = now ) {
2010-01-15 19:22:49 +00:00
if ( strlen ( ftdmchan - > caller_data . dnis . digits ) > 0 ) {
ftdm_log ( FTDM_LOG_DEBUG , " Overlap dial timeout, advancing to RING state \n " ) ;
ftdm_set_state_locked ( ftdmchan , FTDM_CHANNEL_STATE_RING ) ;
2008-10-12 22:07:45 +00:00
} else {
/* no digits received, hangup */
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_LOG_DEBUG , " Overlap dial timeout, no digits received, going to HANGUP state \n " ) ;
ftdmchan - > caller_data . hangup_cause = FTDM_CAUSE_RECOVERY_ON_TIMER_EXPIRE ; /* TODO: probably wrong cause value */
ftdm_set_state_locked ( ftdmchan , FTDM_CHANNEL_STATE_HANGUP ) ;
2008-10-12 22:07:45 +00:00
}
data - > digit_timeout = 0 ;
continue ;
}
2010-01-15 19:22:49 +00:00
if ( last_chan_state ! = ftdmchan - > state ) {
ftdm_buffer_zero ( dt_buffer ) ;
teletone_run ( & ts , ftdmchan - > span - > tone_map [ FTDM_TONEMAP_DIAL ] ) ;
last_chan_state = ftdmchan - > state ;
2008-09-07 17:39:36 +00:00
}
}
break ;
2010-01-15 19:22:49 +00:00
case FTDM_CHANNEL_STATE_RING :
2008-09-07 17:39:36 +00:00
{
2010-01-15 19:22:49 +00:00
if ( last_chan_state ! = ftdmchan - > state ) {
ftdm_buffer_zero ( dt_buffer ) ;
teletone_run ( & ts , ftdmchan - > span - > tone_map [ FTDM_TONEMAP_RING ] ) ;
last_chan_state = ftdmchan - > state ;
2008-09-07 17:39:36 +00:00
}
}
break ;
default : /* Not in a tone generating state, go to next round */
continue ;
}
2010-01-15 19:22:49 +00:00
if ( ! ftdm_test_flag ( ftdmchan , FTDM_CHANNEL_OPEN ) ) {
if ( ftdm_channel_open_chan ( ftdmchan ) ! = FTDM_SUCCESS ) {
ftdm_set_state_locked ( ftdmchan , FTDM_CHANNEL_STATE_HANGUP ) ;
2008-09-07 17:39:36 +00:00
continue ;
}
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_LOG_NOTICE , " Successfully opened channel %d:%d \n " , ftdmchan - > span_id , ftdmchan - > chan_id ) ;
2008-09-07 17:39:36 +00:00
}
2010-01-15 19:22:49 +00:00
flags = FTDM_READ ;
2008-09-07 17:39:36 +00:00
2010-01-15 19:22:49 +00:00
status = ftdm_channel_wait ( ftdmchan , & flags , ( gated ) ? 0 : interval ) ;
2008-09-07 17:39:36 +00:00
switch ( status ) {
2010-01-15 19:22:49 +00:00
case FTDM_FAIL :
2008-09-07 17:39:36 +00:00
continue ;
2010-01-15 19:22:49 +00:00
case FTDM_TIMEOUT :
2008-09-07 17:39:36 +00:00
gated = 1 ;
continue ;
default :
2010-01-15 19:22:49 +00:00
if ( ! ( flags & FTDM_READ ) ) {
2008-09-07 17:39:36 +00:00
continue ;
}
}
gated = 1 ;
2010-01-15 19:22:49 +00:00
status = ftdm_channel_read ( ftdmchan , frame , & len ) ;
if ( status ! = FTDM_SUCCESS | | len < = 0 ) {
2008-09-07 17:39:36 +00:00
continue ;
}
2010-01-15 19:22:49 +00:00
if ( ftdmchan - > effective_codec ! = FTDM_CODEC_SLIN ) {
2008-09-07 17:39:36 +00:00
len * = 2 ;
}
/* seek to current offset */
2010-01-15 19:22:49 +00:00
ftdm_buffer_seek ( dt_buffer , offset ) ;
2008-09-07 17:39:36 +00:00
2010-01-15 19:22:49 +00:00
rlen = ftdm_buffer_read_loop ( dt_buffer , frame , len ) ;
2008-09-07 17:39:36 +00:00
2010-01-15 19:22:49 +00:00
if ( ftdmchan - > effective_codec ! = FTDM_CODEC_SLIN ) {
2010-01-15 20:35:11 +00:00
fio_codec_t codec_func = NULL ;
2008-09-07 17:39:36 +00:00
2010-01-15 19:22:49 +00:00
if ( ftdmchan - > native_codec = = FTDM_CODEC_ULAW ) {
2010-01-15 20:35:11 +00:00
codec_func = fio_slin2ulaw ;
2010-01-15 19:22:49 +00:00
} else if ( ftdmchan - > native_codec = = FTDM_CODEC_ALAW ) {
2010-01-15 20:35:11 +00:00
codec_func = fio_slin2alaw ;
2008-09-07 17:39:36 +00:00
}
if ( codec_func ) {
status = codec_func ( frame , sizeof ( frame ) , & rlen ) ;
} else {
2010-01-15 19:22:49 +00:00
snprintf ( ftdmchan - > last_error , sizeof ( ftdmchan - > last_error ) , " codec error! " ) ;
2008-09-07 17:39:36 +00:00
goto done ;
}
}
2010-01-15 19:22:49 +00:00
ftdm_channel_write ( ftdmchan , frame , sizeof ( frame ) , & rlen ) ;
2008-09-07 17:39:36 +00:00
}
/*
* sleep a bit if there was nothing to do
*/
if ( ! gated ) {
2010-01-15 19:22:49 +00:00
ftdm_sleep ( interval ) ;
2008-09-07 17:39:36 +00:00
}
offset + = ( ts . rate / ( 1000 / interval ) ) < < 1 ;
if ( offset > = ts . rate ) {
offset = 0 ;
}
}
done :
if ( ts . buffer ) {
teletone_destroy_session ( & ts ) ;
}
if ( dt_buffer ) {
2010-01-15 19:22:49 +00:00
ftdm_buffer_destroy ( & dt_buffer ) ;
2008-09-07 17:39:36 +00:00
}
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_LOG_DEBUG , " ISDN tone thread ended. \n " ) ;
ftdm_clear_flag ( isdn_data , FTDM_ISDN_TONES_RUNNING ) ;
2008-09-10 15:25:02 +00:00
2008-09-07 17:39:36 +00:00
return NULL ;
}
2010-01-15 19:22:49 +00:00
static void * ftdm_isdn_run ( ftdm_thread_t * me , void * obj )
2007-05-22 22:07:05 +00:00
{
2010-01-15 19:22:49 +00:00
ftdm_span_t * span = ( ftdm_span_t * ) obj ;
ftdm_isdn_data_t * isdn_data = span - > signal_data ;
2008-09-07 17:39:36 +00:00
unsigned char frame [ 1024 ] ;
2010-01-15 19:22:49 +00:00
ftdm_size_t len = sizeof ( frame ) ;
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
2010-11-01 00:47:43 +01:00
timeBeginPeriod ( 1 ) ;
2007-05-23 19:47:35 +00:00
# endif
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_LOG_DEBUG , " ISDN thread starting. \n " ) ;
ftdm_set_flag ( isdn_data , FTDM_ISDN_RUNNING ) ;
2007-05-22 22:07:05 +00:00
2007-06-25 05:26:37 +00:00
Q921Start ( & isdn_data - > q921 ) ;
2010-11-01 00:47:43 +01:00
Q931Start ( & isdn_data - > q931 ) ;
2007-05-22 22:07:05 +00:00
2010-11-01 00:47:43 +01:00
while ( ftdm_running ( ) & & ftdm_test_flag ( isdn_data , FTDM_ISDN_RUNNING ) ) {
2010-01-15 19:22:49 +00:00
ftdm_wait_flag_t flags = FTDM_READ ;
ftdm_status_t status = ftdm_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 ) ;
2008-09-07 17:39:36 +00:00
/*
*
*/
2007-05-22 22:07:05 +00:00
switch ( status ) {
2010-01-15 19:22:49 +00:00
case FTDM_FAIL :
2007-05-22 22:07:05 +00:00
{
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_LOG_ERROR , " D-Chan Read Error! \n " ) ;
2007-05-22 22:07:05 +00:00
snprintf ( span - > last_error , sizeof ( span - > last_error ) , " D-Chan Read Error! " ) ;
2007-06-13 03:37:55 +00:00
if ( + + errs = = 10 ) {
2010-01-15 19:22:49 +00:00
isdn_data - > dchan - > state = FTDM_CHANNEL_STATE_UP ;
2007-06-13 03:37:55 +00:00
goto done ;
}
2007-05-22 22:07:05 +00:00
}
break ;
2010-01-15 19:22:49 +00:00
case FTDM_TIMEOUT :
2007-05-22 22:07:05 +00:00
{
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 ;
2010-01-15 19:22:49 +00:00
if ( flags & FTDM_READ ) {
2008-09-07 17:39:36 +00:00
len = sizeof ( frame ) ;
2010-01-15 19:22:49 +00:00
if ( ftdm_channel_read ( isdn_data - > dchan , frame , & len ) = = FTDM_SUCCESS ) {
2010-11-01 00:47:43 +01:00
# ifdef HAVE_PCAP
if ( isdn_pcap_capture_both ( isdn_data ) ) {
isdn_pcap_write ( isdn_data , frame , len , ISDN_PCAP_INCOMING ) ;
}
2007-06-20 20:30:10 +00:00
# endif
2008-09-07 17:39:36 +00:00
Q921QueueHDLCFrame ( & isdn_data - > q921 , frame , ( int ) len ) ;
2007-06-25 05:26:37 +00:00
Q921Rx12 ( & isdn_data - > q921 ) ;
2007-05-22 22:07:05 +00:00
}
} else {
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_LOG_DEBUG , " No Read FLAG! \n " ) ;
2007-05-22 22:07:05 +00:00
}
}
break ;
}
}
2010-11-01 00:47:43 +01:00
done :
2010-01-15 19:22:49 +00:00
ftdm_channel_close ( & isdn_data - > dchans [ 0 ] ) ;
ftdm_channel_close ( & isdn_data - > dchans [ 1 ] ) ;
ftdm_clear_flag ( isdn_data , FTDM_ISDN_RUNNING ) ;
2007-05-22 22:07:05 +00:00
2007-05-23 19:47:35 +00:00
# ifdef WIN32
2010-11-01 00:47:43 +01:00
timeEndPeriod ( 1 ) ;
# endif
# ifdef HAVE_PCAP
if ( isdn_pcap_is_open ( isdn_data ) ) {
isdn_pcap_close ( isdn_data ) ;
}
2007-05-23 19:47:35 +00:00
# endif
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_LOG_DEBUG , " ISDN thread ended. \n " ) ;
2007-05-22 22:07:05 +00:00
return NULL ;
}
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
{
2010-11-01 00:47:43 +01:00
ftdm_span_t * span = pvt ;
ftdm_isdn_data_t * isdn_data = span - > signal_data ;
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 ) ) ;
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_LOG_DEBUG , " WRITE %d \n %s \n %s \n \n " , ( int ) mlen - offset , LINE , bb ) ;
2008-08-22 16:55:01 +00:00
break ;
default :
break ;
}
2010-11-01 00:47:43 +01:00
# ifdef HAVE_PCAP
if ( isdn_pcap_capture_l3only ( isdn_data ) ) {
isdn_pcap_write ( isdn_data , msg , mlen , ISDN_PCAP_OUTGOING ) ;
}
# endif
return Q921Rx32 ( & isdn_data - > q921 , ind , tei , msg , mlen ) ;
2008-08-22 16:55:01 +00:00
}
2010-01-15 19:22:49 +00:00
static int ftdm_isdn_q921_log ( void * pvt , Q921LogLevel_t level , char * msg , L2INT size )
2008-08-22 16:55:01 +00:00
{
2010-01-15 19:22:49 +00:00
ftdm_span_t * span = ( ftdm_span_t * ) pvt ;
2008-08-22 16:55:01 +00:00
2010-01-15 19:22:49 +00:00
ftdm_log ( " Span " , " Q.921 " , span - > span_id , ( int ) level , " %s " , msg ) ;
2008-08-22 16:55:01 +00:00
return 0 ;
2007-06-16 04:39:15 +00:00
}
2010-01-15 19:22:49 +00:00
static L3INT ftdm_isdn_q931_log ( void * pvt , Q931LogLevel_t level , char * msg , L3INT size )
2008-08-22 16:55:01 +00:00
{
2010-01-15 19:22:49 +00:00
ftdm_span_t * span = ( ftdm_span_t * ) pvt ;
2008-09-16 18:22:00 +00:00
2010-01-15 19:22:49 +00:00
ftdm_log ( " Span " , " Q.931 " , span - > span_id , ( int ) level , " %s " , msg ) ;
2008-08-22 16:55:01 +00:00
return 0 ;
}
2010-11-01 00:47:43 +01:00
2010-01-15 19:22:49 +00:00
static ftdm_state_map_t isdn_state_map = {
2008-05-21 18:58:14 +00:00
{
{
ZSD_OUTBOUND ,
ZSM_UNACCEPTABLE ,
2010-01-15 19:22:49 +00:00
{ FTDM_ANY_STATE } ,
{ FTDM_CHANNEL_STATE_RESTART , FTDM_END }
2008-05-21 18:58:14 +00:00
} ,
{
ZSD_OUTBOUND ,
ZSM_UNACCEPTABLE ,
2010-01-15 19:22:49 +00:00
{ FTDM_CHANNEL_STATE_RESTART , FTDM_END } ,
{ FTDM_CHANNEL_STATE_DOWN , FTDM_END }
2008-05-21 18:58:14 +00:00
} ,
{
ZSD_OUTBOUND ,
ZSM_UNACCEPTABLE ,
2010-01-15 19:22:49 +00:00
{ FTDM_CHANNEL_STATE_DOWN , FTDM_END } ,
{ FTDM_CHANNEL_STATE_DIALING , FTDM_END }
2008-05-21 18:58:14 +00:00
} ,
{
ZSD_OUTBOUND ,
ZSM_UNACCEPTABLE ,
2010-01-15 19:22:49 +00:00
{ FTDM_CHANNEL_STATE_DIALING , FTDM_END } ,
2010-11-01 00:47:43 +01:00
{ FTDM_CHANNEL_STATE_PROGRESS_MEDIA , FTDM_CHANNEL_STATE_PROGRESS , FTDM_CHANNEL_STATE_UP , FTDM_CHANNEL_STATE_HANGUP , FTDM_CHANNEL_STATE_TERMINATING , FTDM_CHANNEL_STATE_DOWN , FTDM_END }
2008-05-21 18:58:14 +00:00
} ,
{
ZSD_OUTBOUND ,
ZSM_UNACCEPTABLE ,
2010-01-15 19:22:49 +00:00
{ FTDM_CHANNEL_STATE_PROGRESS_MEDIA , FTDM_CHANNEL_STATE_PROGRESS , FTDM_END } ,
{ FTDM_CHANNEL_STATE_HANGUP , FTDM_CHANNEL_STATE_PROGRESS_MEDIA , FTDM_CHANNEL_STATE_TERMINATING , FTDM_CHANNEL_STATE_UP , FTDM_END }
2008-05-21 18:58:14 +00:00
} ,
{
ZSD_OUTBOUND ,
ZSM_UNACCEPTABLE ,
2010-01-15 19:22:49 +00:00
{ FTDM_CHANNEL_STATE_HANGUP , FTDM_CHANNEL_STATE_TERMINATING , FTDM_END } ,
{ FTDM_CHANNEL_STATE_HANGUP_COMPLETE , FTDM_CHANNEL_STATE_DOWN , FTDM_END }
2008-05-21 18:58:14 +00:00
} ,
{
ZSD_OUTBOUND ,
ZSM_UNACCEPTABLE ,
2010-01-15 19:22:49 +00:00
{ FTDM_CHANNEL_STATE_HANGUP_COMPLETE , FTDM_END } ,
{ FTDM_CHANNEL_STATE_DOWN , FTDM_END } ,
2008-05-21 18:58:14 +00:00
} ,
{
ZSD_OUTBOUND ,
ZSM_UNACCEPTABLE ,
2010-01-15 19:22:49 +00:00
{ FTDM_CHANNEL_STATE_UP , FTDM_END } ,
{ FTDM_CHANNEL_STATE_HANGUP , FTDM_CHANNEL_STATE_TERMINATING , FTDM_END }
2008-05-21 18:58:14 +00:00
} ,
/****************************************/
{
ZSD_INBOUND ,
ZSM_UNACCEPTABLE ,
2010-01-15 19:22:49 +00:00
{ FTDM_ANY_STATE } ,
{ FTDM_CHANNEL_STATE_RESTART , FTDM_END }
2008-05-21 18:58:14 +00:00
} ,
{
ZSD_INBOUND ,
ZSM_UNACCEPTABLE ,
2010-01-15 19:22:49 +00:00
{ FTDM_CHANNEL_STATE_RESTART , FTDM_END } ,
{ FTDM_CHANNEL_STATE_DOWN , FTDM_END }
2008-05-21 18:58:14 +00:00
} ,
{
ZSD_INBOUND ,
ZSM_UNACCEPTABLE ,
2010-01-15 19:22:49 +00:00
{ FTDM_CHANNEL_STATE_DOWN , FTDM_END } ,
{ FTDM_CHANNEL_STATE_DIALTONE , FTDM_CHANNEL_STATE_RING , FTDM_END }
2008-09-07 17:39:36 +00:00
} ,
{
ZSD_INBOUND ,
ZSM_UNACCEPTABLE ,
2010-01-15 19:22:49 +00:00
{ FTDM_CHANNEL_STATE_DIALTONE , FTDM_END } ,
{ FTDM_CHANNEL_STATE_RING , FTDM_CHANNEL_STATE_HANGUP , FTDM_CHANNEL_STATE_TERMINATING , FTDM_END }
2008-05-21 18:58:14 +00:00
} ,
{
ZSD_INBOUND ,
ZSM_UNACCEPTABLE ,
2010-01-15 19:22:49 +00:00
{ FTDM_CHANNEL_STATE_RING , FTDM_END } ,
{ FTDM_CHANNEL_STATE_HANGUP , FTDM_CHANNEL_STATE_TERMINATING , FTDM_CHANNEL_STATE_PROGRESS , FTDM_CHANNEL_STATE_PROGRESS_MEDIA , FTDM_CHANNEL_STATE_UP , FTDM_END }
2008-05-21 18:58:14 +00:00
} ,
{
ZSD_INBOUND ,
ZSM_UNACCEPTABLE ,
2010-01-15 19:22:49 +00:00
{ FTDM_CHANNEL_STATE_HANGUP , FTDM_CHANNEL_STATE_TERMINATING , FTDM_END } ,
{ FTDM_CHANNEL_STATE_HANGUP_COMPLETE , FTDM_CHANNEL_STATE_DOWN , FTDM_END } ,
2008-05-21 18:58:14 +00:00
} ,
{
ZSD_INBOUND ,
ZSM_UNACCEPTABLE ,
2010-01-15 19:22:49 +00:00
{ FTDM_CHANNEL_STATE_HANGUP_COMPLETE , FTDM_END } ,
{ FTDM_CHANNEL_STATE_DOWN , FTDM_END } ,
2008-05-21 18:58:14 +00:00
} ,
{
ZSD_INBOUND ,
ZSM_UNACCEPTABLE ,
2010-01-15 19:22:49 +00:00
{ FTDM_CHANNEL_STATE_PROGRESS , FTDM_CHANNEL_STATE_PROGRESS_MEDIA , FTDM_END } ,
2010-11-01 00:47:43 +01:00
{ FTDM_CHANNEL_STATE_HANGUP , FTDM_CHANNEL_STATE_TERMINATING , FTDM_CHANNEL_STATE_PROGRESS_MEDIA ,
2010-01-15 19:22:49 +00:00
FTDM_CHANNEL_STATE_CANCEL , FTDM_CHANNEL_STATE_UP , FTDM_END } ,
2008-05-21 18:58:14 +00:00
} ,
{
ZSD_INBOUND ,
ZSM_UNACCEPTABLE ,
2010-01-15 19:22:49 +00:00
{ FTDM_CHANNEL_STATE_UP , FTDM_END } ,
{ FTDM_CHANNEL_STATE_HANGUP , FTDM_CHANNEL_STATE_TERMINATING , FTDM_END } ,
2008-05-21 18:58:14 +00:00
} ,
}
} ;
2009-06-08 20:01:29 +00:00
/**
2010-11-01 00:47:43 +01:00
* \ brief Stop signalling on span
2009-06-08 20:01:29 +00:00
*/
2010-01-15 19:22:49 +00:00
static ftdm_status_t ftdm_isdn_stop ( ftdm_span_t * span )
2009-03-12 17:22:03 +00:00
{
2010-01-15 19:22:49 +00:00
ftdm_isdn_data_t * isdn_data = span - > signal_data ;
2009-03-12 17:22:03 +00:00
2010-01-15 19:22:49 +00:00
if ( ! ftdm_test_flag ( isdn_data , FTDM_ISDN_RUNNING ) ) {
return FTDM_FAIL ;
2009-03-12 17:22:03 +00:00
}
2010-11-01 00:47:43 +01:00
2010-01-15 19:22:49 +00:00
ftdm_set_flag ( isdn_data , FTDM_ISDN_STOP ) ;
2010-11-01 00:47:43 +01:00
2010-01-15 19:22:49 +00:00
while ( ftdm_test_flag ( isdn_data , FTDM_ISDN_RUNNING ) ) {
ftdm_sleep ( 100 ) ;
2009-03-12 17:22:03 +00:00
}
2010-01-15 19:22:49 +00:00
while ( ftdm_test_flag ( isdn_data , FTDM_ISDN_TONES_RUNNING ) ) {
ftdm_sleep ( 100 ) ;
2009-03-12 17:22:03 +00:00
}
2010-01-15 19:22:49 +00:00
return FTDM_SUCCESS ;
2009-03-12 17:22:03 +00:00
}
2009-06-08 20:01:29 +00:00
/**
2010-11-01 00:47:43 +01:00
* \ brief Start signalling on span
2009-06-08 20:01:29 +00:00
*/
2010-01-15 19:22:49 +00:00
static ftdm_status_t ftdm_isdn_start ( ftdm_span_t * span )
2008-08-29 15:58:59 +00:00
{
2010-01-15 19:22:49 +00:00
ftdm_status_t ret ;
ftdm_isdn_data_t * isdn_data = span - > signal_data ;
2008-09-07 17:39:36 +00:00
2010-01-15 19:22:49 +00:00
if ( ftdm_test_flag ( isdn_data , FTDM_ISDN_RUNNING ) ) {
return FTDM_FAIL ;
2009-03-12 17:22:03 +00:00
}
2010-01-15 19:22:49 +00:00
ftdm_clear_flag ( isdn_data , FTDM_ISDN_STOP ) ;
ret = ftdm_thread_create_detached ( ftdm_isdn_run , span ) ;
2009-03-12 17:22:03 +00:00
2010-01-15 19:22:49 +00:00
if ( ret ! = FTDM_SUCCESS ) {
2008-09-07 17:39:36 +00:00
return ret ;
}
2010-01-15 19:22:49 +00:00
if ( FTDM_SPAN_IS_NT ( span ) & & ! ( isdn_data - > opts & FTDM_ISDN_OPT_DISABLE_TONES ) ) {
ret = ftdm_thread_create_detached ( ftdm_isdn_tones_run , span ) ;
2008-09-07 17:39:36 +00:00
}
return ret ;
2008-08-29 15:58:59 +00:00
}
2010-11-01 00:47:43 +01:00
/*###################################################################*
* ( text ) value parsing / translation
* # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # */
static int32_t parse_loglevel ( const char * level )
{
if ( ! level ) {
return - 1 ;
}
if ( ! strcasecmp ( level , " debug " ) ) {
return FTDM_LOG_LEVEL_DEBUG ;
} else if ( ! strcasecmp ( level , " info " ) ) {
return FTDM_LOG_LEVEL_INFO ;
} else if ( ! strcasecmp ( level , " notice " ) ) {
return FTDM_LOG_LEVEL_NOTICE ;
} else if ( ! strcasecmp ( level , " warning " ) ) {
return FTDM_LOG_LEVEL_WARNING ;
} else if ( ! strcasecmp ( level , " error " ) ) {
return FTDM_LOG_LEVEL_ERROR ;
} else if ( ! strcasecmp ( level , " alert " ) ) {
return FTDM_LOG_LEVEL_ALERT ;
} else if ( ! strcasecmp ( level , " crit " ) ) {
return FTDM_LOG_LEVEL_CRIT ;
} else if ( ! strcasecmp ( level , " emerg " ) ) {
return FTDM_LOG_LEVEL_EMERG ;
} else {
return - 1 ;
}
}
2009-03-19 23:06:04 +00:00
static uint32_t parse_opts ( const char * in )
{
uint32_t flags = 0 ;
2010-11-01 00:47:43 +01:00
2009-03-19 23:06:04 +00:00
if ( ! in ) {
return 0 ;
}
2010-11-01 00:47:43 +01:00
2009-03-19 23:06:04 +00:00
if ( strstr ( in , " suggest_channel " ) ) {
2010-01-15 19:22:49 +00:00
flags | = FTDM_ISDN_OPT_SUGGEST_CHANNEL ;
2009-03-19 23:06:04 +00:00
}
2008-08-29 15:58:59 +00:00
2009-03-19 23:06:04 +00:00
if ( strstr ( in , " omit_display " ) ) {
2010-01-15 19:22:49 +00:00
flags | = FTDM_ISDN_OPT_OMIT_DISPLAY_IE ;
2009-03-19 23:06:04 +00:00
}
if ( strstr ( in , " disable_tones " ) ) {
2010-01-15 19:22:49 +00:00
flags | = FTDM_ISDN_OPT_DISABLE_TONES ;
2009-03-19 23:06:04 +00:00
}
return flags ;
}
2008-08-29 15:58:59 +00:00
2010-11-01 00:47:43 +01:00
static uint32_t parse_dialect ( const char * in )
{
if ( ! in ) {
return Q931_Dialect_Count ;
}
# if __UNSUPPORTED__
if ( ! strcasecmp ( in , " national " ) ) {
return Q931_Dialect_National ;
}
if ( ! strcasecmp ( in , " dms " ) ) {
return Q931_Dialect_DMS ;
}
# endif
if ( ! strcasecmp ( in , " 5ess " ) ) {
return Q931_Dialect_5ESS ;
}
if ( ! strcasecmp ( in , " dss1 " ) ) {
return Q931_Dialect_DSS1 ;
}
if ( ! strcasecmp ( in , " q931 " ) ) {
return Q931_Dialect_Q931 ;
}
return Q931_Dialect_Count ;
}
/*###################################################################*
* API commands
* # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # */
static const char isdn_api_usage [ ] =
# ifdef HAVE_PCAP
" isdn capture <span> <start> <filename> [q931only] \n "
" isdn capture <span> <stop|suspend|resume> \n "
# endif
" isdn loglevel <span> <q921|q931|all> <loglevel> \n "
" isdn dump <span> calls \n "
" isdn help " ;
2009-06-08 20:01:29 +00:00
/**
2010-11-01 00:47:43 +01:00
* isdn_api
2009-06-08 20:01:29 +00:00
*/
2010-11-01 00:47:43 +01:00
static FIO_API_FUNCTION ( isdn_api )
{
char * mycmd = NULL , * argv [ 10 ] = { 0 } ;
int argc = 0 ;
if ( data ) {
mycmd = strdup ( data ) ;
argc = ftdm_separate_string ( mycmd , ' ' , argv , ( sizeof ( argv ) / sizeof ( argv [ 0 ] ) ) ) ;
}
if ( ! argc | | ! strcasecmp ( argv [ 0 ] , " help " ) ) {
stream - > write_function ( stream , " %s \n " , isdn_api_usage ) ;
goto done ;
}
else if ( ! strcasecmp ( argv [ 0 ] , " dump " ) ) {
ftdm_isdn_data_t * isdn_data = NULL ;
ftdm_span_t * span = NULL ;
int span_id = 0 ;
/* dump <span> calls */
if ( argc < 3 ) {
stream - > write_function ( stream , " -ERR not enough arguments. \n " ) ;
goto done ;
}
span_id = atoi ( argv [ 1 ] ) ;
if ( ftdm_span_find_by_name ( argv [ 1 ] , & span ) = = FTDM_SUCCESS | | ftdm_span_find ( span_id , & span ) = = FTDM_SUCCESS ) {
isdn_data = span - > signal_data ;
} else {
stream - > write_function ( stream , " -ERR invalid span. \n " ) ;
goto done ;
}
if ( ! strcasecmp ( argv [ 2 ] , " calls " ) ) {
/* dump all calls to log */
Q931DumpAllCalls ( & isdn_data - > q931 ) ;
stream - > write_function ( stream , " +OK call information dumped to log \n " ) ;
goto done ;
}
}
else if ( ! strcasecmp ( argv [ 0 ] , " loglevel " ) ) {
ftdm_isdn_data_t * isdn_data = NULL ;
ftdm_span_t * span = NULL ;
int span_id = 0 ;
int layer = 0 ;
int level = 0 ;
/* loglevel <span> <q921|q931|all> [level] */
if ( argc < 3 ) {
stream - > write_function ( stream , " -ERR not enough arguments. \n " ) ;
goto done ;
}
span_id = atoi ( argv [ 1 ] ) ;
if ( ftdm_span_find_by_name ( argv [ 1 ] , & span ) = = FTDM_SUCCESS | | ftdm_span_find ( span_id , & span ) = = FTDM_SUCCESS ) {
isdn_data = span - > signal_data ;
} else {
stream - > write_function ( stream , " -ERR invalid span. \n " ) ;
goto done ;
}
if ( ! strcasecmp ( argv [ 2 ] , " q921 " ) ) {
layer = 0x01 ;
} else if ( ! strcasecmp ( argv [ 2 ] , " q931 " ) ) {
layer = 0x02 ;
} else if ( ! strcasecmp ( argv [ 2 ] , " all " ) ) {
layer = 0x03 ;
} else {
stream - > write_function ( stream , " -ERR invalid layer \n " ) ;
goto done ;
}
if ( argc > 3 ) {
/* set loglevel */
if ( ( level = parse_loglevel ( argv [ 3 ] ) ) < 0 ) {
stream - > write_function ( stream , " -ERR invalid loglevel \n " ) ;
goto done ;
}
if ( layer & 0x01 ) { /* q921 */
Q921SetLogLevel ( & isdn_data - > q921 , ( Q921LogLevel_t ) level ) ;
}
if ( layer & 0x02 ) { /* q931 */
Q931SetLogLevel ( & isdn_data - > q931 , ( Q931LogLevel_t ) level ) ;
}
stream - > write_function ( stream , " +OK loglevel set " ) ;
} else {
/* get loglevel */
if ( layer & 0x01 ) {
stream - > write_function ( stream , " Q.921 loglevel: %s \n " ,
Q921GetLogLevelName ( & isdn_data - > q921 ) ) ;
}
if ( layer & 0x02 ) {
stream - > write_function ( stream , " Q.931 loglevel: %s \n " ,
Q931GetLogLevelName ( & isdn_data - > q931 ) ) ;
}
stream - > write_function ( stream , " +OK " ) ;
}
goto done ;
}
# ifdef HAVE_PCAP
else if ( ! strcasecmp ( argv [ 0 ] , " capture " ) ) {
ftdm_isdn_data_t * isdn_data = NULL ;
ftdm_span_t * span = NULL ;
int span_id = 0 ;
/* capture <span> <start> <filename> [q931only] */
/* capture <span> <stop|suspend|resume> */
if ( argc < 3 ) {
stream - > write_function ( stream , " -ERR not enough arguments. \n " ) ;
goto done ;
}
span_id = atoi ( argv [ 1 ] ) ;
if ( ftdm_span_find_by_name ( argv [ 1 ] , & span ) = = FTDM_SUCCESS | | ftdm_span_find ( span_id , & span ) = = FTDM_SUCCESS ) {
isdn_data = span - > signal_data ;
} else {
stream - > write_function ( stream , " -ERR invalid span. \n " ) ;
goto done ;
}
if ( ! strcasecmp ( argv [ 2 ] , " start " ) ) {
char * filename = NULL ;
if ( argc < 4 ) {
stream - > write_function ( stream , " -ERR not enough parameters. \n " ) ;
goto done ;
}
if ( isdn_pcap_is_open ( isdn_data ) ) {
stream - > write_function ( stream , " -ERR capture is already running. \n " ) ;
goto done ;
}
filename = argv [ 3 ] ;
if ( isdn_pcap_open ( isdn_data , filename ) ! = FTDM_SUCCESS ) {
stream - > write_function ( stream , " -ERR failed to open capture file. \n " ) ;
goto done ;
}
if ( argc > 4 & & ! strcasecmp ( argv [ 4 ] , " q931only " ) ) {
isdn_data - > flags | = FTDM_ISDN_CAPTURE_L3ONLY ;
}
isdn_pcap_start ( isdn_data ) ;
stream - > write_function ( stream , " +OK capture started. \n " ) ;
goto done ;
}
else if ( ! strcasecmp ( argv [ 2 ] , " stop " ) ) {
if ( ! isdn_pcap_is_open ( isdn_data ) ) {
stream - > write_function ( stream , " -ERR capture is not running. \n " ) ;
goto done ;
}
isdn_pcap_stop ( isdn_data ) ;
isdn_pcap_close ( isdn_data ) ;
stream - > write_function ( stream , " +OK capture stopped. \n " ) ;
goto done ;
}
else if ( ! strcasecmp ( argv [ 2 ] , " suspend " ) ) {
if ( ! isdn_pcap_is_open ( isdn_data ) ) {
stream - > write_function ( stream , " -ERR capture is not running. \n " ) ;
goto done ;
}
isdn_pcap_stop ( isdn_data ) ;
stream - > write_function ( stream , " +OK capture suspended. \n " ) ;
goto done ;
}
else if ( ! strcasecmp ( argv [ 2 ] , " resume " ) ) {
if ( ! isdn_pcap_is_open ( isdn_data ) ) {
stream - > write_function ( stream , " -ERR capture is not running. \n " ) ;
goto done ;
}
isdn_pcap_start ( isdn_data ) ;
stream - > write_function ( stream , " +OK capture resumed. \n " ) ;
goto done ;
}
else {
stream - > write_function ( stream , " -ERR wrong action. \n " ) ;
goto done ;
}
}
# endif
else {
stream - > write_function ( stream , " -ERR invalid command. \n " ) ;
}
done :
ftdm_safe_free ( mycmd ) ;
return FTDM_SUCCESS ;
}
static FIO_SIG_CONFIGURE_FUNCTION ( isdn_configure_span )
2007-05-22 22:07:05 +00:00
{
2007-05-24 01:01:50 +00:00
uint32_t i , x = 0 ;
2010-11-01 00:47:43 +01:00
ftdm_channel_t * dchans [ 2 ] = { 0 } ;
2010-01-15 19:22:49 +00:00
ftdm_isdn_data_t * isdn_data ;
2008-09-07 17:39:36 +00:00
const char * tonemap = " us " ;
2008-08-29 15:58:59 +00:00
char * var , * val ;
Q931Dialect_t dialect = Q931_Dialect_National ;
2008-10-12 22:07:45 +00:00
int32_t digit_timeout = 0 ;
int q921loglevel = - 1 ;
int q931loglevel = - 1 ;
2008-08-29 15:58:59 +00:00
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 ) ;
2010-01-15 19:22:49 +00:00
return FTDM_FAIL ;
2007-05-22 22:07:05 +00:00
}
2010-01-15 19:22:49 +00:00
if ( span - > trunk_type > = FTDM_TRUNK_NONE ) {
ftdm_log ( FTDM_LOG_WARNING , " Invalid trunk type '%s' defaulting to T1. \n " , ftdm_trunk_type2str ( span - > trunk_type ) ) ;
span - > trunk_type = FTDM_TRUNK_T1 ;
2007-05-22 22:07:05 +00:00
}
2010-11-01 00:47:43 +01:00
2007-05-22 22:07:05 +00:00
for ( i = 1 ; i < = span - > chan_count ; i + + ) {
2010-01-15 19:22:49 +00:00
if ( span - > channels [ i ] - > type = = FTDM_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! " ) ;
2010-01-15 19:22:49 +00:00
return FTDM_FAIL ;
2010-11-01 00:47:43 +01:00
}
if ( ftdm_channel_open ( span - > span_id , i , & dchans [ x ] ) = = FTDM_SUCCESS ) {
ftdm_log ( FTDM_LOG_DEBUG , " opening d-channel #%d %d:%d \n " , x , dchans [ x ] - > span_id , dchans [ x ] - > chan_id ) ;
dchans [ x ] - > state = FTDM_CHANNEL_STATE_UP ;
x + + ;
2007-05-22 22:07:05 +00:00
}
}
}
if ( ! x ) {
snprintf ( span - > last_error , sizeof ( span - > last_error ) , " Span has no D-Channels! " ) ;
2010-01-15 19:22:49 +00:00
return FTDM_FAIL ;
2007-05-22 22:07:05 +00:00
}
2010-11-01 00:47:43 +01:00
isdn_data = malloc ( sizeof ( * isdn_data ) ) ;
2007-06-25 05:26:37 +00:00
assert ( isdn_data ! = NULL ) ;
memset ( isdn_data , 0 , sizeof ( * isdn_data ) ) ;
2010-11-01 00:47:43 +01:00
2008-08-29 15:58:59 +00:00
isdn_data - > mode = Q931_TE ;
2010-11-01 00:47:43 +01:00
dialect = Q931_Dialect_Q931 ;
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 ;
}
2010-11-01 00:47:43 +01:00
dialect = parse_dialect ( val ) ;
2008-08-29 15:58:59 +00:00
if ( dialect = = Q931_Dialect_Count ) {
2010-11-01 00:47:43 +01:00
snprintf ( span - > last_error , sizeof ( span - > last_error ) , " Invalid/unknown dialect [%s]! " , val ) ;
2010-01-15 19:22:49 +00:00
return FTDM_FAIL ;
2008-08-29 15:58:59 +00:00
}
} else if ( ! strcasecmp ( var , " opts " ) ) {
2009-03-19 23:06:04 +00:00
if ( ! ( val = va_arg ( ap , char * ) ) ) {
break ;
2008-10-12 22:07:45 +00:00
}
2009-03-19 23:06:04 +00:00
isdn_data - > opts = parse_opts ( val ) ;
2008-09-07 17:39:36 +00:00
} else if ( ! strcasecmp ( var , " tonemap " ) ) {
if ( ! ( val = va_arg ( ap , char * ) ) ) {
break ;
}
tonemap = ( const char * ) val ;
2008-10-12 22:07:45 +00:00
} else if ( ! strcasecmp ( var , " digit_timeout " ) ) {
int * optp ;
if ( ! ( optp = va_arg ( ap , int * ) ) ) {
break ;
}
digit_timeout = * optp ;
2008-10-03 21:57:57 +00:00
} else if ( ! strcasecmp ( var , " q921loglevel " ) ) {
q921loglevel = va_arg ( ap , int ) ;
2008-10-12 22:07:45 +00:00
if ( q921loglevel < Q921_LOG_NONE ) {
q921loglevel = Q921_LOG_NONE ;
} else if ( q921loglevel > Q921_LOG_DEBUG ) {
q921loglevel = Q921_LOG_DEBUG ;
}
2008-10-03 21:57:57 +00:00
} else if ( ! strcasecmp ( var , " q931loglevel " ) ) {
q931loglevel = va_arg ( ap , int ) ;
2008-10-12 22:07:45 +00:00
if ( q931loglevel < Q931_LOG_NONE ) {
q931loglevel = Q931_LOG_NONE ;
} else if ( q931loglevel > Q931_LOG_DEBUG ) {
q931loglevel = Q931_LOG_DEBUG ;
}
2008-10-07 00:00:15 +00:00
} else {
snprintf ( span - > last_error , sizeof ( span - > last_error ) , " Unknown parameter [%s] " , var ) ;
2010-01-15 19:22:49 +00:00
return FTDM_FAIL ;
2008-08-29 15:58:59 +00:00
}
}
2008-10-12 22:07:45 +00:00
if ( ! digit_timeout ) {
digit_timeout = DEFAULT_DIGIT_TIMEOUT ;
}
else if ( digit_timeout < 3000 | | digit_timeout > 30000 ) {
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_LOG_WARNING , " Digit timeout %d ms outside of range (3000 - 30000 ms), using default (10000 ms) \n " , digit_timeout ) ;
2008-10-12 22:07:45 +00:00
digit_timeout = DEFAULT_DIGIT_TIMEOUT ;
}
/* allocate per b-chan data */
if ( isdn_data - > mode = = Q931_NT ) {
2010-01-15 19:22:49 +00:00
ftdm_isdn_bchan_data_t * data ;
2008-10-12 22:07:45 +00:00
2010-11-01 00:47:43 +01:00
data = malloc ( span - > chan_count * sizeof ( ftdm_isdn_bchan_data_t ) ) ;
2008-10-12 22:07:45 +00:00
if ( ! data ) {
2010-01-15 19:22:49 +00:00
return FTDM_FAIL ;
2008-10-12 22:07:45 +00:00
}
for ( i = 1 ; i < = span - > chan_count ; i + + , data + + ) {
2010-01-15 19:22:49 +00:00
if ( span - > channels [ i ] - > type = = FTDM_CHAN_TYPE_B ) {
2008-10-12 22:07:45 +00:00
span - > channels [ i ] - > mod_data = data ;
2010-01-15 19:22:49 +00:00
memset ( data , 0 , sizeof ( ftdm_isdn_bchan_data_t ) ) ;
2008-10-12 22:07:45 +00:00
}
}
}
2010-11-01 00:47:43 +01:00
isdn_data - > sig_cb = sig_cb ;
2007-06-25 05:26:37 +00:00
isdn_data - > dchans [ 0 ] = dchans [ 0 ] ;
isdn_data - > dchans [ 1 ] = dchans [ 1 ] ;
2010-11-01 00:47:43 +01:00
isdn_data - > dchan = isdn_data - > dchans [ 0 ] ;
2008-10-12 22:07:45 +00:00
isdn_data - > digit_timeout = digit_timeout ;
2010-11-01 00:47:43 +01: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 ,
2010-01-15 19:22:49 +00:00
span - > trunk_type = = FTDM_TRUNK_BRI_PTMP ? Q921_PTMP : Q921_PTP ,
2007-05-22 22:07:05 +00:00
0 ,
2010-01-15 19:22:49 +00:00
ftdm_isdn_921_21 ,
( Q921Tx23CB_t ) ftdm_isdn_921_23 ,
2007-05-22 22:07:05 +00:00
span ,
2010-11-01 00:47:43 +01:00
span ) ;
2008-08-22 16:55:01 +00:00
2010-11-01 00:47:43 +01:00
Q921SetLogCB ( & isdn_data - > q921 , & ftdm_isdn_q921_log , span ) ;
2008-10-03 21:57:57 +00:00
Q921SetLogLevel ( & isdn_data - > q921 , ( Q921LogLevel_t ) q921loglevel ) ;
2010-11-01 00:47:43 +01:00
Q931InitTrunk ( & 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 ,
2010-01-15 19:22:49 +00:00
ftdm_isdn_931_34 ,
2008-08-22 16:55:01 +00:00
( Q931Tx32CB_t ) q931_rx_32 ,
2010-01-15 19:22:49 +00:00
ftdm_isdn_931_err ,
2010-11-01 00:47:43 +01:00
span ,
2007-05-22 22:07:05 +00:00
span ) ;
2007-05-29 16:30:20 +00:00
2010-11-01 00:47:43 +01:00
Q931SetLogCB ( & isdn_data - > q931 , & ftdm_isdn_q931_log , span ) ;
2008-10-03 21:57:57 +00:00
Q931SetLogLevel ( & isdn_data - > q931 , ( Q931LogLevel_t ) q931loglevel ) ;
2008-08-22 16:55:01 +00:00
2010-11-01 00:47:43 +01:00
/* Register new event hander CB */
Q931SetCallEventCB ( & isdn_data - > q931 , ftdm_isdn_call_event , span ) ;
/* TODO: hmm, maybe drop the "Trunk" prefix */
Q931TrunkSetAutoRestartAck ( & isdn_data - > q931 , 1 ) ;
Q931TrunkSetAutoConnectAck ( & isdn_data - > q931 , 1 ) ;
Q931TrunkSetAutoServiceAck ( & isdn_data - > q931 , 1 ) ;
Q931TrunkSetStatusEnquiry ( & isdn_data - > q931 , 0 ) ;
span - > state_map = & isdn_state_map ;
span - > signal_data = isdn_data ;
span - > signal_type = FTDM_SIGTYPE_ISDN ;
span - > start = ftdm_isdn_start ;
span - > stop = ftdm_isdn_stop ;
2007-06-14 04:44:44 +00:00
span - > outgoing_call = isdn_outgoing_call ;
2008-08-29 15:58:59 +00:00
2010-11-01 00:47:43 +01:00
span - > get_channel_sig_status = isdn_get_channel_sig_status ;
span - > get_span_sig_status = isdn_get_span_sig_status ;
# ifdef __TODO__
2010-01-15 19:22:49 +00:00
if ( ( isdn_data - > opts & FTDM_ISDN_OPT_SUGGEST_CHANNEL ) ) {
2008-05-21 18:58:14 +00:00
span - > channel_request = isdn_channel_request ;
2010-11-01 00:47:43 +01:00
span - > flags | = FTDM_SPAN_SUGGEST_CHAN_ID ;
2008-05-21 18:58:14 +00:00
}
2010-11-01 00:47:43 +01:00
# endif
2010-01-15 19:22:49 +00:00
ftdm_span_load_tones ( span , tonemap ) ;
2008-09-07 17:39:36 +00:00
2010-01-15 19:22:49 +00:00
return FTDM_SUCCESS ;
2007-05-22 22:07:05 +00:00
}
2007-06-04 14:53:12 +00:00
2009-06-08 20:01:29 +00:00
/**
2010-11-01 00:47:43 +01:00
* ISDN module io interface
* \ note This is really ugly . . .
2009-06-08 20:01:29 +00:00
*/
2010-11-01 00:47:43 +01:00
static ftdm_io_interface_t isdn_interface = {
. name = " isdn " ,
. api = isdn_api
2008-08-29 15:58:59 +00:00
} ;
2010-11-01 00:47:43 +01:00
/**
* \ brief ISDN module io interface init callback
*/
static FIO_IO_LOAD_FUNCTION ( isdn_io_load )
{
assert ( fio ! = NULL ) ;
* fio = & isdn_interface ;
return FTDM_SUCCESS ;
}
/**
* \ brief ISDN module load callback
*/
static FIO_SIG_LOAD_FUNCTION ( isdn_load )
{
Q931Initialize ( ) ;
Q921SetGetTimeCB ( ftdm_time_now ) ;
Q931SetGetTimeCB ( ftdm_time_now ) ;
return FTDM_SUCCESS ;
}
2008-08-29 15:58:59 +00:00
2010-11-01 00:47:43 +01:00
/**
* \ brief ISDN module shutdown callback
*/
static FIO_SIG_UNLOAD_FUNCTION ( isdn_unload )
{
return FTDM_SUCCESS ;
} ;
ftdm_module_t ftdm_module = {
. name = " isdn " ,
. io_load = isdn_io_load ,
. io_unload = NULL ,
. sig_load = isdn_load ,
. sig_unload = isdn_unload ,
. sig_configure = isdn_configure_span
} ;
2009-01-30 11:37:11 +00:00
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 :
2010-11-01 00:47:43 +01:00
* vim : set softtabstop = 4 shiftwidth = 4 tabstop = 4 expandtab :
2007-06-04 14:53:12 +00:00
*/