2007-05-22 22:07:05 +00:00
/*
* Copyright ( c ) 2007 , Anthony Minessale II
* All rights reserved .
*
* Redistribution and use in source and binary forms , with or without
* modification , are permitted provided that the following conditions
* are met :
*
* * Redistributions of source code must retain the above copyright
* notice , this list of conditions and the following disclaimer .
*
* * Redistributions in binary form must reproduce the above copyright
* notice , this list of conditions and the following disclaimer in the
* documentation and / or other materials provided with the distribution .
*
* * Neither the name of the original author ; nor the names of any contributors
* may be used to endorse or promote products derived from this software
* without specific prior written permission .
*
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* " AS IS " AND ANY EXPRESS OR IMPLIED WARRANTIES , INCLUDING , BUT NOT
* LIMITED TO , THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
* A PARTICULAR PURPOSE ARE DISCLAIMED . IN NO EVENT SHALL THE COPYRIGHT OWNER
* OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT , INDIRECT , INCIDENTAL , SPECIAL ,
* EXEMPLARY , OR CONSEQUENTIAL DAMAGES ( INCLUDING , BUT NOT LIMITED TO ,
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES ; LOSS OF USE , DATA , OR
* PROFITS ; OR BUSINESS INTERRUPTION ) HOWEVER CAUSED AND ON ANY THEORY OF
* LIABILITY , WHETHER IN CONTRACT , STRICT LIABILITY , OR TORT ( INCLUDING
* NEGLIGENCE OR OTHERWISE ) ARISING IN ANY WAY OUT OF THE USE OF THIS
* SOFTWARE , EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE .
*/
2010-01-13 17:53:22 +00:00
/**
* Workaround for missing u_int / u_short types on solaris
*/
# if defined(HAVE_LIBPCAP) && defined(__SunOS)
# define __EXTENSIONS__
# endif
2010-01-15 19:22:49 +00:00
# include "freetdm.h"
2007-05-22 22:07:05 +00:00
# include "Q931.h"
# include "Q921.h"
2007-05-23 19:47:35 +00:00
# ifdef WIN32
# include <windows.h>
# else
# include <sys/time.h>
# endif
2010-01-15 19:22:49 +00:00
# include "ftdm_isdn.h"
2009-04-08 21:58:51 +00:00
2007-05-23 18:25:23 +00:00
# define LINE "--------------------------------------------------------------------------------"
2007-11-21 22:08:47 +00:00
//#define IODEBUG
2007-05-22 22:07:05 +00:00
2008-08-22 16:55:01 +00:00
/* helper macros */
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
2008-10-12 22:07:45 +00:00
2009-02-03 17:00:13 +00:00
# ifdef HAVE_LIBPCAP
2009-01-30 11:37:11 +00:00
/*-------------------------------------------------------------------------*/
/*Q931ToPcap functions*/
# include <pcap.h>
2009-02-03 17:00:13 +00:00
# endif
2009-01-30 11:37:11 +00:00
# define SNAPLEN 1522
# define MAX_ETHER_PAYLOAD_SIZE 1500
# define MIN_ETHER_PAYLOAD_SIZE 42
# define SIZE_ETHERNET 18
# define VLANID_OFFSET 15
# define SIZE_IP 20
# define SIZE_TCP 20
# define SIZE_TPKT 4
# define SIZE_ETHERNET_CRC 4
# define OVERHEAD SIZE_ETHERNET+SIZE_IP+SIZE_TCP+SIZE_TPKT
# define MAX_Q931_SIZE MAX_ETHER_PAYLOAD_SIZE-SIZE_IP-SIZE_TCP-SIZE_TPKT
# define TPKT_SIZE_OFFSET SIZE_ETHERNET+SIZE_IP+SIZE_TCP+2
# define IP_SIZE_OFFSET SIZE_ETHERNET+2
# define TCP_SEQ_OFFSET SIZE_ETHERNET+SIZE_IP+4
2009-02-03 17:00:13 +00:00
# ifdef HAVE_LIBPCAP
2009-01-30 11:37:11 +00:00
/*Some globals*/
unsigned long pcapfilesize = 0 ;
unsigned long tcp_next_seq_no_send = 0 ;
unsigned long tcp_next_seq_no_rec = 0 ;
pcap_dumper_t * pcapfile = NULL ;
struct pcap_pkthdr pcaphdr ;
pcap_t * pcaphandle = NULL ;
char * pcapfn = NULL ;
int do_q931ToPcap = 0 ;
2009-01-31 10:23:23 +00:00
/*Predefined Ethernet Frame with Q931-over-IP encapsulated - From remote TDM host to FreeSWITCH*/
L3UCHAR recFrame [ SNAPLEN ] = {
/*IEEE 802.3 VLAN 802.1q Ethernet Frame Header*/
2 , 0 , 1 , 0xAA , 0xAA , 0xAA , 2 , 0 , 1 , 0xBB , 0xBB , 0xBB , 0x81 , 0 , 0xE0 , 0 , 0x08 , 0 ,
/*IPv4 Header (minimal size; no options)*/
0x45 , 0 , 0 , 44 , 0 , 0 , 0 , 0 , 64 , 6 , 0 , 0 , 2 , 2 , 2 , 2 , 1 , 1 , 1 , 1 ,
/*TCP-Header*/
0 , 0x66 , 0 , 0x66 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0x50 , 0 , 0 , 1 , 0 , 0 , 0 , 0 ,
/*TPKT-Header RFC 1006*/
3 , 0 , 0 , 0
} ;
/*Predefined Ethernet Frame with Q931-over-IP encapsulated - Frome FreeSWITCH to remote TDM host*/
L3UCHAR sendFrame [ SNAPLEN ] = {
/*IEEE 802.3 VLAN 802.1q Ethernet Frame Header*/
2 , 0 , 1 , 0xBB , 0xBB , 0xBB , 2 , 0 , 1 , 0xAA , 0xAA , 0xAA , 0x81 , 0 , 0xE0 , 0 , 0x08 , 0 ,
/*IPv4 Header (minimal size; no options)*/
0x45 , 0 , 0 , 44 , 0 , 0 , 0 , 0 , 64 , 6 , 0 , 0 , 1 , 1 , 1 , 1 , 2 , 2 , 2 , 2 ,
/*TCP-Header*/
0 , 0x66 , 0 , 0x66 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0x50 , 0 , 0 , 1 , 0 , 0 , 0 , 0 ,
/*TPKT-Header RFC 1006*/
3 , 0 , 0 , 0
} ;
2009-01-30 11:37:11 +00:00
2009-06-08 20:01:29 +00:00
/**
* \ brief Opens a pcap file for capture
* \ return Success or failure
*/
2010-01-15 19:22:49 +00:00
static ftdm_status_t openPcapFile ( void )
2009-01-30 11:37:11 +00:00
{
if ( ! pcaphandle )
{
pcaphandle = pcap_open_dead ( DLT_EN10MB , SNAPLEN ) ;
if ( ! pcaphandle )
{
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_LOG_ERROR , " Can't open pcap session: (%s) \n " , pcap_geterr ( pcaphandle ) ) ;
return FTDM_FAIL ;
2009-01-30 11:37:11 +00:00
}
}
if ( ! pcapfile ) {
/* Open the dump file */
if ( ! ( pcapfile = pcap_dump_open ( pcaphandle , pcapfn ) ) ) {
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_LOG_ERROR , " Error opening output file (%s) \n " , pcap_geterr ( pcaphandle ) ) ;
return FTDM_FAIL ;
2009-01-30 11:37:11 +00:00
}
}
else {
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_LOG_WARNING , " Pcap file is already open! \n " ) ;
return FTDM_FAIL ;
2009-01-30 11:37:11 +00:00
}
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_LOG_DEBUG , " Pcap file '%s' successfully opened! \n " , pcapfn ) ;
2009-01-30 11:37:11 +00:00
pcaphdr . ts . tv_sec = 0 ;
pcaphdr . ts . tv_usec = 0 ;
pcapfilesize = 24 ; /*current pcap file header seems to be 24 bytes*/
tcp_next_seq_no_send = 0 ;
tcp_next_seq_no_rec = 0 ;
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
/**
* \ brief Closes a pcap file
* \ return Success
*/
2010-01-15 19:22:49 +00:00
static ftdm_status_t closePcapFile ( void )
2009-01-30 11:37:11 +00:00
{
2009-02-03 17:00:13 +00:00
if ( pcapfile ) {
2009-01-30 11:37:11 +00:00
pcap_dump_close ( pcapfile ) ;
2009-02-03 17:00:13 +00:00
if ( pcaphandle ) pcap_close ( pcaphandle ) ;
2009-01-30 11:37:11 +00:00
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_LOG_DEBUG , " Pcap file closed! File size is %lu bytes. \n " , pcapfilesize ) ;
2009-01-30 11:37:11 +00:00
2009-02-03 17:00:13 +00:00
pcaphdr . ts . tv_sec = 0 ;
pcaphdr . ts . tv_usec = 0 ;
2009-01-30 11:37:11 +00:00
pcapfile = NULL ;
pcaphandle = NULL ;
pcapfilesize = 0 ;
tcp_next_seq_no_send = 0 ;
tcp_next_seq_no_rec = 0 ;
}
/*We have allways success with this? I think so*/
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
/**
* \ brief Writes a Q931 packet to a pcap file
* \ return Success or failure
*/
2010-01-15 19:22:49 +00:00
static ftdm_status_t writeQ931PacketToPcap ( L3UCHAR * q931buf , L3USHORT q931size , L3ULONG span_id , L3USHORT direction )
2009-01-30 11:37:11 +00:00
{
2009-02-25 09:49:54 +00:00
L3UCHAR * frame = NULL ;
struct timeval ts ;
2009-01-30 11:37:11 +00:00
u_char spanid = ( u_char ) span_id ;
2009-02-25 09:49:54 +00:00
unsigned long * tcp_next_seq_no = NULL ;
2009-01-30 11:37:11 +00:00
spanid = span_id ;
/*The total length of the ethernet frame generated by this function has a min length of 66
so we don ' t have to care about padding : ) */
/*FS is sending the packet*/
if ( direction = = 0 ) {
frame = sendFrame ;
tcp_next_seq_no = & tcp_next_seq_no_send ;
}
/*FS is receiving the packet*/
else {
frame = recFrame ;
tcp_next_seq_no = & tcp_next_seq_no_rec ;
}
/*Set spanid in VLAN-ID tag*/
frame [ VLANID_OFFSET ] = spanid ;
/*** Write sent packet ***/
if ( q931size > MAX_Q931_SIZE )
{
/*WARNING*/
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_LOG_WARNING , " Q931 packet size is too big (%u)! Ignoring it! \n " , q931size ) ;
return FTDM_FAIL ;
2009-01-30 11:37:11 +00:00
}
/*Copy q931 buffer into frame*/
memcpy ( frame + OVERHEAD , q931buf , q931size ) ;
/*Store TCP sequence number in TCP header*/
frame [ TCP_SEQ_OFFSET ] = ( * tcp_next_seq_no > > 24 ) & 0xFF ;
frame [ TCP_SEQ_OFFSET + 1 ] = ( * tcp_next_seq_no > > 16 ) & 0xFF ;
frame [ TCP_SEQ_OFFSET + 2 ] = ( * tcp_next_seq_no > > 8 ) & 0xFF ;
frame [ TCP_SEQ_OFFSET + 3 ] = * tcp_next_seq_no & 0xFF ;
/*Store size of TPKT packet*/
q931size + = 4 ;
frame [ TPKT_SIZE_OFFSET ] = ( q931size > > 8 ) & 0xFF ;
frame [ TPKT_SIZE_OFFSET + 1 ] = q931size & 0xFF ;
/*Calc next TCP sequence number*/
* tcp_next_seq_no + = q931size ;
/*Store size of IP packet*/
q931size + = SIZE_IP + SIZE_TCP ;
frame [ IP_SIZE_OFFSET ] = ( q931size > > 8 ) & 0xFF ;
frame [ IP_SIZE_OFFSET + 1 ] = q931size & 0xFF ;
pcaphdr . caplen = SIZE_ETHERNET + SIZE_ETHERNET_CRC + q931size ;
pcaphdr . len = pcaphdr . caplen ;
2009-02-25 09:49:54 +00:00
/* Set Timestamp */
/* Get Time in ms. usecs would be better ... */
gettimeofday ( & ts , NULL ) ;
/*Write it into packet header*/
pcaphdr . ts . tv_sec = ts . tv_sec ;
pcaphdr . ts . tv_usec = ts . tv_usec ;
pcap_dump ( ( u_char * ) pcapfile , & pcaphdr , frame ) ;
pcap_dump_flush ( pcapfile ) ;
2009-01-30 11:37:11 +00:00
/*Maintain pcap file size*/
pcapfilesize + = pcaphdr . caplen ;
pcapfilesize + = sizeof ( struct pcap_pkthdr ) ;
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_LOG_DEBUG , " Added %u bytes to pcap file. File size is now %lu, \n " , q931size , pcapfilesize ) ;
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
}
2009-02-03 17:00:13 +00:00
# endif
2009-06-08 20:01:29 +00:00
/**
* \ brief Unloads pcap IO
* \ return Success or failure
*/
2009-01-30 11:37:11 +00:00
static ZIO_IO_UNLOAD_FUNCTION ( close_pcap )
{
2009-02-03 17:00:13 +00:00
# ifdef HAVE_LIBPCAP
2009-01-30 11:37:11 +00:00
return closePcapFile ( ) ;
2009-02-03 17:00:13 +00:00
# else
2010-01-15 19:22:49 +00:00
return FTDM_SUCCESS ;
2009-02-03 17:00:13 +00:00
# endif
2009-01-30 11:37:11 +00:00
}
/*Q931ToPcap functions DONE*/
/*-------------------------------------------------------------------------*/
2009-06-08 20:01:29 +00:00
/**
* \ brief Gets current time
* \ return Current time ( in ms )
*/
2010-01-15 19:22:49 +00:00
static L2ULONG ftdm_time_now ( void )
2007-05-23 19:47:35 +00:00
{
2010-01-15 19:22:49 +00:00
return ( L2ULONG ) ftdm_current_time_in_ms ( ) ;
2007-05-23 19:47:35 +00:00
}
2009-06-08 20:01:29 +00:00
/**
* \ brief Initialises an ISDN channel ( outgoing call )
2010-01-15 19:22:49 +00:00
* \ param ftdmchan Channel to initiate call on
2009-06-08 20:01:29 +00:00
* \ return Success or failure
*/
2007-06-14 04:44:44 +00:00
static ZIO_CHANNEL_OUTGOING_CALL_FUNCTION ( isdn_outgoing_call )
{
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
/**
* \ brief Requests an ISDN channel on a span ( outgoing call )
* \ param span Span where to get a channel
* \ param chan_id Specific channel to get ( 0 for any )
* \ param direction Call direction ( inbound , outbound )
* \ param caller_data Caller information
2010-01-15 19:22:49 +00:00
* \ param ftdmchan Channel to initialise
2009-06-08 20:01:29 +00:00
* \ return Success or failure
*/
2008-05-05 22:54:40 +00:00
static ZIO_CHANNEL_REQUEST_FUNCTION ( isdn_channel_request )
{
Q931mes_Generic * gen = ( Q931mes_Generic * ) caller_data - > raw_data ;
Q931ie_BearerCap BearerCap ;
Q931ie_ChanID ChanID = { 0 } ;
Q931ie_CallingNum CallingNum ;
Q931ie_CallingNum * ptrCallingNum ;
Q931ie_CalledNum CalledNum ;
Q931ie_CalledNum * ptrCalledNum ;
Q931ie_Display Display , * ptrDisplay ;
2008-08-22 16:55:01 +00:00
Q931ie_HLComp HLComp ; /* High-Layer Compatibility IE */
Q931ie_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 */
2008-05-05 22:54:40 +00:00
gen - > BearerCap = Q931AppendIE ( ( L3UCHAR * ) gen , ( L3UCHAR * ) & BearerCap ) ;
2008-08-22 16:55:01 +00:00
/*
* Channel ID IE
*/
Q931InitIEChanID ( & ChanID ) ;
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-01-15 19:22:49 +00:00
if ( ! FTDM_SPAN_IS_NT ( span ) ) {
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 */
}
if ( ChanID . IntType ) {
ChanID . InfoChanSel = 1 ; /* None = 0, See Slot = 1, Any = 3 */
ChanID . ChanMapType = 3 ; /* B-Chan */
ChanID . ChanSlot = ( unsigned char ) chan_id ;
} else {
ChanID . InfoChanSel = ( unsigned char ) chan_id & 0x03 ; /* None = 0, B1 = 1, B2 = 2, Any = 3 */
}
2008-05-05 22:54:40 +00:00
gen - > ChanID = Q931AppendIE ( ( L3UCHAR * ) gen , ( L3UCHAR * ) & ChanID ) ;
2008-08-22 16:55:01 +00:00
/*
* Progress IE
*/
Q931InitIEProgInd ( & Progress ) ;
Progress . CodStand = Q931_CODING_ITU ; /* 0 = ITU */
Progress . Location = 0 ; /* 0 = User, 1 = Private Network */
Progress . ProgDesc = 3 ; /* 1 = Not end-to-end ISDN */
gen - > ProgInd = Q931AppendIE ( ( L3UCHAR * ) gen , ( L3UCHAR * ) & Progress ) ;
/*
* Display IE
*/
2010-01-15 19:22:49 +00:00
if ( ! ( isdn_data - > opts & FTDM_ISDN_OPT_OMIT_DISPLAY_IE ) ) {
2009-03-19 23:06:04 +00:00
Q931InitIEDisplay ( & Display ) ;
Display . Size = Display . Size + ( unsigned char ) strlen ( caller_data - > cid_name ) ;
gen - > Display = Q931AppendIE ( ( L3UCHAR * ) gen , ( L3UCHAR * ) & Display ) ;
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 ) ;
gen - > CallingNum = Q931AppendIE ( ( L3UCHAR * ) gen , ( L3UCHAR * ) & CallingNum ) ;
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 ;
2008-05-05 22:54:40 +00:00
CalledNum . Size = CalledNum . Size + ( unsigned char ) strlen ( caller_data - > ani . digits ) ;
gen - > CalledNum = Q931AppendIE ( ( L3UCHAR * ) gen , ( L3UCHAR * ) & CalledNum ) ;
ptrCalledNum = Q931GetIEPtr ( gen - > CalledNum , gen - > buf ) ;
2010-01-15 19:22:49 +00: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) */
gen - > HLComp = Q931AppendIE ( ( L3UCHAR * ) gen , ( L3UCHAR * ) & HLComp ) ;
2010-01-15 19:22:49 +00:00
caller_data - > call_state = FTDM_CALLER_STATE_DIALING ;
2008-05-05 22:54:40 +00:00
Q931Rx43 ( & isdn_data - > q931 , ( L3UCHAR * ) gen , gen - > Size ) ;
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 ) ;
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-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 ;
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 " ) ;
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 ;
}
}
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 ' ;
gen - > Cause = Q931AppendIE ( ( L3UCHAR * ) gen , ( L3UCHAR * ) & cause ) ;
Q931Rx43 ( & isdn_data - > q931 , ( L3UCHAR * ) gen , gen - > Size ) ;
if ( gen - > CRV ) {
Q931ReleaseCRV ( & isdn_data - > q931 , gen - > CRV ) ;
}
if ( new_chan ) {
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-01-15 19:22:49 +00:00
* ftdmchan = NULL ;
return FTDM_FAIL ;
2008-05-05 22:54:40 +00:00
}
2009-06-08 20:01:29 +00:00
/**
* \ brief Handler for Q931 error
* \ param pvt Private structure ( span ? )
* \ param id Error number
* \ param p1 ? ?
* \ param p2 ? ?
* \ return 0
*/
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
/**
* \ brief Handler for Q931 event message
* \ param pvt Span to handle
* \ param msg Message string
* \ param mlen Message string length
* \ return 0
*/
2010-01-15 19:22:49 +00:00
static L3INT ftdm_isdn_931_34 ( 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_isdn_data_t * isdn_data = span - > signal_data ;
2007-05-22 22:07:05 +00:00
Q931mes_Generic * gen = ( Q931mes_Generic * ) msg ;
2009-03-10 17:42:49 +00:00
uint32_t 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 ;
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 ) ;
2007-06-14 03:54:02 +00:00
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
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 ;
}
return 0 ;
2007-06-14 03:54:02 +00:00
}
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 ;
2007-06-26 01:02:36 +00:00
case 1 :
{ /* 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 ;
2008-09-10 19:43:55 +00:00
for ( i = 1 ; i < span - > chan_count ; i + + ) {
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 ;
2008-08-22 16:55:01 +00:00
status = isdn_data - > sig_cb ( & sig ) ;
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-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
gen - > MesType = Q931mes_CONNECT_ACKNOWLEDGE ;
gen - > CRVFlag = 0 ; /* outbound */
Q931Rx43 ( & isdn_data - > q931 , ( L3UCHAR * ) gen , gen - > Size ) ;
2008-01-28 22:17:34 +00:00
} else {
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
}
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 ;
}
2008-04-01 21:49:35 +00: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 ) {
2009-03-10 17:42:49 +00:00
uint32_t 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-01-15 19:22:49 +00:00
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-01-15 19:22:49 +00:00
ftdm_set_string ( ftdmchan - > caller_data . dnis . digits , ( char * ) callednum - > Digit ) ;
2008-10-12 22:07:45 +00:00
}
2007-06-25 16:00:24 +00:00
2010-01-15 19:22:49 +00:00
ftdmchan - > caller_data . CRV = gen - > CRV ;
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 ;
}
2007-06-14 03:54:02 +00:00
}
2007-06-13 03:37:55 +00:00
2007-06-25 16:00:24 +00:00
if ( fail ) {
2008-04-01 21:49:35 +00:00
Q931ie_Cause cause ;
gen - > MesType = Q931mes_DISCONNECT ;
2008-08-22 16:55:01 +00:00
gen - > CRVFlag = 1 ; /* inbound call */
2008-04-01 21:49:35 +00:00
cause . IEId = Q931ie_CAUSE ;
cause . Size = sizeof ( Q931ie_Cause ) ;
2008-08-22 16:55:01 +00:00
cause . CodStand = Q931_CODING_ITU ;
2008-04-01 21:49:35 +00:00
cause . Location = 1 ;
cause . Recom = 1 ;
//should we be casting here.. or do we need to translate value?
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 ' ;
gen - > Cause = Q931AppendIE ( ( L3UCHAR * ) gen , ( L3UCHAR * ) & cause ) ;
Q931Rx43 ( & isdn_data - > q931 , ( L3UCHAR * ) gen , gen - > Size ) ;
2008-04-02 20:24:54 +00:00
if ( gen - > CRV ) {
Q931ReleaseCRV ( & isdn_data - > q931 , gen - > CRV ) ;
}
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
}
2008-04-01 21:49:35 +00: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
}
gen - > ChanID = Q931AppendIE ( ( L3UCHAR * ) gen , ( L3UCHAR * ) & ChanID ) ;
if ( overlap_dial ) {
Q931ie_ProgInd progress ;
/*
* Setup Progress indicator
*/
progress . IEId = Q931ie_PROGRESS_INDICATOR ;
progress . Size = sizeof ( Q931ie_ProgInd ) ;
progress . CodStand = Q931_CODING_ITU ; /* ITU */
progress . Location = 1 ; /* private network serving the local user */
progress . ProgDesc = 8 ; /* call is not end-to-end isdn = 1, in-band information available = 8 */
gen - > ProgInd = Q931AppendIE ( ( L3UCHAR * ) gen , ( L3UCHAR * ) & progress ) ;
/*
* Send SETUP ACK
*/
gen - > MesType = Q931mes_SETUP_ACKNOWLEDGE ;
gen - > CRVFlag = 1 ; /* inbound call */
Q931Rx43 ( & isdn_data - > q931 , ( L3UCHAR * ) gen , gen - > Size ) ;
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-01-15 19:22:49 +00:00
pos = ( int ) strlen ( ftdmchan - > caller_data . dnis . digits ) ;
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 ;
2009-07-03 09:24:52 +00:00
case Q931mes_STATUS_ENQUIRY :
{
/*
* ! ! HACK ALERT ! !
*
2010-01-15 19:56:54 +00:00
* Map FreeTDM channel states to Q .931 states
2009-07-03 09:24:52 +00:00
*/
Q931ie_CallState state ;
Q931ie_Cause cause ;
gen - > MesType = Q931mes_STATUS ;
gen - > CRVFlag = gen - > CRVFlag ? 0 : 1 ;
state . CodStand = Q931_CODING_ITU ; /* ITU-T */
state . CallState = Q931_U0 ; /* Default: Null */
cause . IEId = Q931ie_CAUSE ;
cause . Size = sizeof ( Q931ie_Cause ) ;
cause . CodStand = Q931_CODING_ITU ; /* ITU */
cause . Location = 1 ; /* private network */
cause . Recom = 1 ; /* */
* cause . Diag = ' \0 ' ;
2010-01-15 19:22:49 +00:00
if ( ftdmchan ) {
switch ( ftdmchan - > state ) {
case FTDM_CHANNEL_STATE_UP :
2009-07-03 09:24:52 +00:00
state . CallState = Q931_U10 ; /* Active */
break ;
2010-01-15 19:22:49 +00:00
case FTDM_CHANNEL_STATE_RING :
2009-07-03 09:24:52 +00:00
state . CallState = Q931_U6 ; /* Call present */
break ;
2010-01-15 19:22:49 +00:00
case FTDM_CHANNEL_STATE_DIALING :
2009-07-03 09:24:52 +00:00
state . CallState = Q931_U1 ; /* Call initiated */
break ;
2010-01-15 19:22:49 +00:00
case FTDM_CHANNEL_STATE_DIALTONE :
2009-07-03 09:24:52 +00:00
state . CallState = Q931_U25 ; /* Overlap receiving */
break ;
/* TODO: map missing states */
default :
state . CallState = Q931_U0 ;
}
cause . Value = 30 ; /* response to STATUS ENQUIRY */
} else {
cause . Value = 98 ; /* */
}
gen - > CallState = Q931AppendIE ( ( L3UCHAR * ) gen , ( L3UCHAR * ) & state ) ;
gen - > Cause = Q931AppendIE ( ( L3UCHAR * ) gen , ( L3UCHAR * ) & cause ) ;
Q931Rx43 ( & isdn_data - > q931 , ( L3UCHAR * ) gen , gen - > Size ) ;
}
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 ;
}
2009-06-08 20:01:29 +00:00
/**
* \ brief Handler for Q921 read event
* \ param pvt Span were message is coming from
* \ param ind Q921 indication
* \ param tei Terminal Endpoint Identifier
* \ param msg Message string
* \ param mlen Message string length
* \ return 0 on success , 1 on failure
*/
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
{
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 ) ) ;
2009-02-03 17:00:13 +00:00
# ifdef HAVE_LIBPCAP
2009-01-30 11:37:11 +00:00
/*Q931ToPcap*/
if ( do_q931ToPcap = = 1 ) {
2010-01-15 19:22:49 +00:00
ftdm_span_t * span = ( ftdm_span_t * ) pvt ;
if ( writeQ931PacketToPcap ( msg + offset , mlen - offset , span - > span_id , 1 ) ! = FTDM_SUCCESS ) {
ftdm_log ( FTDM_LOG_WARNING , " Couldn't write Q931 buffer to pcap file! \n " ) ;
2009-01-30 11:37:11 +00:00
}
}
/*Q931ToPcap done*/
2009-02-03 17:00:13 +00:00
# endif
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_LOG_DEBUG , " READ %d \n %s \n %s \n \n \n " , ( int ) mlen - offset , LINE , bb ) ;
2009-02-25 09:49:54 +00:00
2008-08-22 16:55:01 +00:00
default :
ret = Q931Rx23 ( pvt , ind , tei , msg , mlen ) ;
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
}
2009-06-08 20:01:29 +00:00
/**
* \ brief Handler for Q921 write event
* \ param pvt Span were message is coming from
* \ param msg Message string
* \ param mlen Message string length
* \ return 0 on success , - 1 on failure
*/
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
2007-05-23 18:25:23 +00:00
# ifdef IODEBUG
2007-11-21 22:08:47 +00:00
char bb [ 4096 ] = " " ;
print_hex_bytes ( msg , len , bb , sizeof ( bb ) ) ;
2010-01-15 19:22:49 +00:00
print_bits ( msg , ( int ) len , bb , sizeof ( bb ) , FTDM_ENDIAN_LITTLE , 0 ) ;
ftdm_log ( FTDM_LOG_DEBUG , " WRITE %d \n %s \n %s \n \n " , ( int ) len , LINE , bb ) ;
2007-05-23 18:25:23 +00:00
# endif
2007-05-22 22:07:05 +00:00
assert ( span ! = NULL ) ;
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
}
2009-06-08 20:01:29 +00:00
/**
* \ brief Handler for channel state change
2010-01-15 19:22:49 +00:00
* \ param ftdmchan Channel to handle
2009-06-08 20:01:29 +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-01-15 19:22:49 +00:00
ftdm_log ( FTDM_LOG_DEBUG , " %d:%d STATE [%s] \n " ,
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 ;
if ( ( status = isdn_data - > sig_cb ( & sig ) ! = FTDM_SUCCESS ) ) {
ftdm_set_state_locked ( ftdmchan , FTDM_CHANNEL_STATE_HANGUP ) ;
2007-06-16 04:39:15 +00:00
}
} else {
2008-07-24 20:03:37 +00:00
gen - > MesType = Q931mes_CALL_PROCEEDING ;
2008-08-22 16:55:01 +00:00
gen - > CRVFlag = 1 ; /* inbound */
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
}
gen - > ChanID = Q931AppendIE ( ( L3UCHAR * ) gen , ( L3UCHAR * ) & ChanID ) ;
}
2007-06-25 05:26:37 +00:00
Q931Rx43 ( & isdn_data - > q931 , ( void * ) gen , gen - > Size ) ;
2007-06-16 04:39:15 +00:00
}
}
break ;
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 ;
if ( ( status = isdn_data - > sig_cb ( & sig ) ! = FTDM_SUCCESS ) ) {
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 ;
2008-05-26 01:51:11 +00: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 ;
if ( ( status = isdn_data - > sig_cb ( & sig ) ! = FTDM_SUCCESS ) ) {
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 */
2007-06-25 05:26:37 +00:00
Q931Rx43 ( & isdn_data - > q931 , ( void * ) gen , gen - > Size ) ;
2007-06-16 04:39:15 +00:00
}
}
break ;
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 ;
if ( ( status = isdn_data - > sig_cb ( & sig ) ! = FTDM_SUCCESS ) ) {
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-01-15 19:22:49 +00:00
Q931Rx43 ( & isdn_data - > q931 , ( void * ) 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
*/
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 */
2008-05-21 18:58:14 +00:00
gen - > BearerCap = Q931AppendIE ( ( L3UCHAR * ) gen , ( L3UCHAR * ) & BearerCap ) ;
2008-08-22 16:55:01 +00:00
/*
* ChannelID IE
*/
Q931InitIEChanID ( & ChanID ) ;
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
}
2008-05-21 18:58:14 +00:00
gen - > ChanID = Q931AppendIE ( ( L3UCHAR * ) gen , ( L3UCHAR * ) & ChanID ) ;
2008-08-22 16:55:01 +00:00
/*
* Progress IE
*/
Q931InitIEProgInd ( & Progress ) ;
Progress . CodStand = Q931_CODING_ITU ; /* 0 = ITU */
Progress . Location = 0 ; /* 0 = User, 1 = Private Network */
Progress . ProgDesc = 3 ; /* 1 = Not end-to-end ISDN */
gen - > ProgInd = Q931AppendIE ( ( L3UCHAR * ) gen , ( L3UCHAR * ) & Progress ) ;
/*
* Display IE
2009-03-19 23:06:04 +00:00
*/
2010-01-15 19:22:49 +00:00
if ( ! ( isdn_data - > opts & FTDM_ISDN_OPT_OMIT_DISPLAY_IE ) ) {
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 ) ;
2009-03-19 23:06:04 +00:00
gen - > Display = Q931AppendIE ( ( L3UCHAR * ) gen , ( L3UCHAR * ) & Display ) ;
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
*/
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 ) ;
2008-05-21 18:58:14 +00:00
gen - > CallingNum = Q931AppendIE ( ( L3UCHAR * ) gen , ( L3UCHAR * ) & CallingNum ) ;
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 ) ;
CalledNum . TypNum = Q931_TON_UNKNOWN ;
CalledNum . NumPlanID = Q931_NUMPLAN_E164 ;
2010-01-15 19:22:49 +00:00
CalledNum . Size = CalledNum . Size + ( unsigned char ) strlen ( ftdmchan - > caller_data . ani . digits ) ;
2008-05-21 18:58:14 +00:00
gen - > CalledNum = Q931AppendIE ( ( L3UCHAR * ) gen , ( L3UCHAR * ) & CalledNum ) ;
ptrCalledNum = Q931GetIEPtr ( gen - > CalledNum , gen - > buf ) ;
2010-01-15 19:22:49 +00: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 */
gen - > HLComp = Q931AppendIE ( ( L3UCHAR * ) gen , ( L3UCHAR * ) & HLComp ) ;
2008-05-21 18:58:14 +00:00
Q931Rx43 ( & isdn_data - > q931 , ( L3UCHAR * ) gen , gen - > Size ) ;
2010-01-15 19:22:49 +00:00
isdn_data - > channels_local_crv [ gen - > CRV ] = ftdmchan ;
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 ;
Q931Rx43 ( & isdn_data - > q931 , ( L3UCHAR * ) gen , gen - > Size ) ;
}
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 ' ;
gen - > Cause = Q931AppendIE ( ( L3UCHAR * ) gen , ( L3UCHAR * ) & cause ) ;
Q931Rx43 ( & isdn_data - > q931 , ( L3UCHAR * ) gen , gen - > Size ) ;
/* 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 ' ;
gen - > Cause = Q931AppendIE ( ( L3UCHAR * ) gen , ( L3UCHAR * ) & cause ) ;
Q931Rx43 ( & isdn_data - > q931 , ( void * ) gen , gen - > Size ) ;
/* 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 ' ;
gen - > Cause = Q931AppendIE ( ( L3UCHAR * ) gen , ( L3UCHAR * ) & cause ) ;
Q931Rx43 ( & isdn_data - > q931 , ( L3UCHAR * ) gen , gen - > Size ) ;
}
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 ;
2007-06-25 05:26:37 +00:00
status = isdn_data - > sig_cb ( & sig ) ;
2007-06-14 03:54:02 +00:00
gen - > MesType = Q931mes_RELEASE ;
2010-01-15 19:22:49 +00:00
gen - > CRVFlag = ftdm_test_flag ( ftdmchan , FTDM_CHANNEL_OUTBOUND ) ? 0 : 1 ;
2007-06-25 05:26:37 +00:00
Q931Rx43 ( & isdn_data - > q931 , ( void * ) gen , gen - > Size ) ;
2007-06-14 03:54:02 +00:00
}
2007-06-13 03:37:55 +00:00
default :
break ;
}
}
2009-06-08 20:01:29 +00:00
/**
* \ brief Checks current state on a span
* \ param span Span to check status on
*/
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
}
2009-06-08 20:01:29 +00:00
/**
2010-01-15 19:22:49 +00:00
* \ brief Processes Openftdm event on a span
2009-06-08 20:01:29 +00:00
* \ param span Span to process event on
* \ param event Event to process
* \ return Success or failure
*/
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-01-15 19:22:49 +00:00
ftdm_sigmsg_t sig ;
ftdm_isdn_data_t * isdn_data = span - > signal_data ;
2007-06-23 18:51:10 +00:00
memset ( & sig , 0 , sizeof ( sig ) ) ;
sig . chan_id = event - > channel - > chan_id ;
sig . span_id = event - > channel - > span_id ;
sig . channel = event - > channel ;
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_LOG_DEBUG , " EVENT [%s][%d:%d] STATE [%s] \n " ,
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-01-15 19:22:49 +00:00
sig . event_id = FTDM_SIGEVENT_HWSTATUS_CHANGED ;
sig . raw_data = ( void * ) FTDM_HW_LINK_DISCONNECTED ;
if ( event - > channel - > state ! = FTDM_CHANNEL_STATE_DOWN ) {
if ( event - > channel - > type = = FTDM_CHAN_TYPE_B ) {
ftdm_set_state_locked ( event - > channel , FTDM_CHANNEL_STATE_RESTART ) ;
2009-03-12 17:22:03 +00:00
}
}
2009-03-13 18:52:35 +00:00
2009-03-12 17:22:03 +00:00
2010-01-15 19:22:49 +00:00
ftdm_set_flag ( event - > channel , FTDM_CHANNEL_SUSPENDED ) ;
2009-03-13 18:52:35 +00:00
2010-01-15 19:22:49 +00:00
ftdm_channel_get_alarms ( event - > channel ) ;
2007-06-25 05:26:37 +00:00
isdn_data - > sig_cb ( & sig ) ;
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_LOG_WARNING , " channel %d:%d (%d:%d) has alarms! [%s] \n " ,
2007-06-23 18:51:10 +00:00
event - > channel - > span_id , event - > channel - > chan_id ,
event - > channel - > physical_span_id , event - > channel - > physical_chan_id ,
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
{
2009-03-13 18:52:35 +00:00
2010-01-15 19:22:49 +00:00
ftdm_log ( FTDM_LOG_WARNING , " channel %d:%d (%d:%d) alarms Cleared! \n " , event - > channel - > span_id , event - > channel - > chan_id ,
2009-03-13 18:52:35 +00:00
event - > channel - > physical_span_id , event - > channel - > physical_chan_id ) ;
2009-03-12 17:22:03 +00:00
2010-01-15 19:22:49 +00:00
sig . event_id = FTDM_SIGEVENT_HWSTATUS_CHANGED ;
sig . raw_data = ( void * ) FTDM_HW_LINK_CONNECTED ;
ftdm_clear_flag ( event - > channel , FTDM_CHANNEL_SUSPENDED ) ;
ftdm_channel_get_alarms ( event - > channel ) ;
2007-06-25 05:26:37 +00:00
isdn_data - > sig_cb ( & sig ) ;
2007-06-23 18:51:10 +00:00
}
break ;
}
2010-01-15 19:22:49 +00:00
return FTDM_SUCCESS ;
2007-06-23 18:51:10 +00:00
}
2009-06-08 20:01:29 +00:00
/**
* \ brief Checks for events on a span
* \ param span Span to check for events
*/
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-01-15 19:22:49 +00:00
status = ftdm_span_poll_event ( span , 5 ) ;
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 ;
}
}
2009-06-08 20:01:29 +00:00
/**
* \ brief Retrieves tone generation output to be sent
* \ param ts Teletone generator
* \ param map Tone map
* \ return - 1 on error , 0 on success
*/
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 ;
}
2009-06-08 20:01:29 +00:00
/**
* \ brief Main thread function for tone generation on a span
* \ param me Current thread
* \ param obj Span to generate tones on
*/
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 ;
2009-03-10 20:09:50 +00:00
teletone_generation_session_t ts = { { { { 0 } } } } ;
2008-09-07 17:39:36 +00:00
unsigned char frame [ 1024 ] ;
2009-03-10 17:42:49 +00:00
uint32_t x ;
int interval = 0 ;
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-01-15 19:22:49 +00:00
while ( ftdm_running ( ) & & ftdm_test_flag ( isdn_data , FTDM_ISDN_TONES_RUNNING ) & & ! ftdm_test_flag ( isdn_data , FTDM_ISDN_STOP ) ) {
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 ) {
2008-09-07 17:39:36 +00:00
zio_codec_t codec_func = NULL ;
2010-01-15 19:22:49 +00:00
if ( ftdmchan - > native_codec = = FTDM_CODEC_ULAW ) {
2008-09-07 17:39:36 +00:00
codec_func = zio_slin2ulaw ;
2010-01-15 19:22:49 +00:00
} else if ( ftdmchan - > native_codec = = FTDM_CODEC_ALAW ) {
2008-09-07 17:39:36 +00:00
codec_func = zio_slin2alaw ;
}
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 ;
}
2009-06-08 20:01:29 +00:00
/**
* \ brief Main thread function for an ISDN span
* \ param me Current thread
* \ param obj Span to monitor
*/
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
timeBeginPeriod ( 1 ) ;
# 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 ) ;
2007-05-22 22:07:05 +00:00
2010-01-15 19:22:49 +00:00
while ( ftdm_running ( ) & & ftdm_test_flag ( isdn_data , FTDM_ISDN_RUNNING ) & & ! ftdm_test_flag ( isdn_data , FTDM_ISDN_STOP ) ) {
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 ) {
2009-03-13 18:52:35 +00:00
2010-01-15 19:22:49 +00:00
if ( ftdm_test_flag ( isdn_data - > dchan , FTDM_CHANNEL_SUSPENDED ) ) {
ftdm_clear_flag_all ( span , FTDM_CHANNEL_SUSPENDED ) ;
2009-03-13 18:52:35 +00:00
}
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 ) {
2007-06-20 20:30:10 +00:00
# ifdef IODEBUG
2007-11-21 22:08:47 +00:00
char bb [ 4096 ] = " " ;
2008-09-07 17:39:36 +00:00
print_hex_bytes ( frame , len , bb , sizeof ( bb ) ) ;
2007-05-22 22:07:05 +00:00
2010-01-15 19:22:49 +00:00
print_bits ( frame , ( int ) len , bb , sizeof ( bb ) , FTDM_ENDIAN_LITTLE , 0 ) ;
ftdm_log ( FTDM_LOG_DEBUG , " READ %d \n %s \n %s \n \n " , ( int ) len , LINE , bb ) ;
2007-06-20 20:30:10 +00:00
# endif
2008-09-16 18:22:00 +00:00
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 ;
}
}
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
timeEndPeriod ( 1 ) ;
# 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 ;
}
2009-06-08 20:01:29 +00:00
/**
2010-01-15 19:22:49 +00:00
* \ brief Openftdm ISDN signaling module initialisation
2009-06-08 20:01:29 +00:00
* \ return Success
*/
2010-01-15 19:22:49 +00:00
static ZIO_SIG_LOAD_FUNCTION ( ftdm_isdn_init )
2007-05-22 22:07:05 +00:00
{
Q931Initialize ( ) ;
2007-05-23 19:47:35 +00:00
2010-01-15 19:22:49 +00:00
Q921SetGetTimeCB ( ftdm_time_now ) ;
Q931SetGetTimeCB ( ftdm_time_now ) ;
2008-08-22 16:55:01 +00:00
2010-01-15 19:22:49 +00:00
return FTDM_SUCCESS ;
2007-05-22 22:07:05 +00:00
}
2009-06-08 20:01:29 +00:00
/**
* \ brief Receives a Q931 indication message
* \ param pvt Span were message is coming from
* \ param ind Q931 indication
* \ param tei Terminal Endpoint Identifier
* \ param msg Message string
* \ param mlen Message string length
* \ return 0 on success
*/
2008-08-22 16:55:01 +00:00
static int q931_rx_32 ( void * pvt , Q921DLMsg_t ind , L3UCHAR tei , L3UCHAR * msg , L3INT mlen )
2007-06-16 04:39:15 +00:00
{
2008-08-22 16:55:01 +00:00
int offset = 4 ;
2007-06-16 04:39:15 +00:00
char bb [ 4096 ] = " " ;
2008-08-22 16:55:01 +00:00
switch ( ind ) {
case Q921_DL_UNIT_DATA :
offset = 3 ;
case Q921_DL_DATA :
print_hex_bytes ( msg + offset , mlen - offset , bb , sizeof ( bb ) ) ;
2009-02-03 17:00:13 +00:00
# ifdef HAVE_LIBPCAP
2009-01-30 11:37:11 +00:00
/*Q931ToPcap*/
if ( do_q931ToPcap = = 1 ) {
2010-01-15 19:22:49 +00:00
ftdm_span_t * span = ( ftdm_span_t * ) pvt ;
if ( writeQ931PacketToPcap ( msg + offset , mlen - offset , span - > span_id , 0 ) ! = FTDM_SUCCESS ) {
ftdm_log ( FTDM_LOG_WARNING , " Couldn't write Q931 buffer to pcap file! \n " ) ;
2009-01-30 11:37:11 +00:00
}
}
/*Q931ToPcap done*/
2009-02-03 17:00:13 +00:00
# endif
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 ;
}
return Q921Rx32 ( pvt , ind , tei , msg , mlen ) ;
}
2009-06-08 20:01:29 +00:00
/**
* \ brief Logs Q921 message
* \ param pvt Span were message is coming from
* \ param level Q921 log level
* \ param msg Message string
* \ param size Message string length
* \ return 0
*/
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
}
2009-06-08 20:01:29 +00:00
/**
* \ brief Logs Q931 message
* \ param pvt Span were message is coming from
* \ param level Q931 log level
* \ param msg Message string
* \ param size Message string length
* \ return 0
*/
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 ;
}
2009-06-08 20:01:29 +00:00
/**
* \ brief ISDN state map
*/
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 } ,
{ FTDM_CHANNEL_STATE_PROGRESS_MEDIA , FTDM_CHANNEL_STATE_PROGRESS , FTDM_CHANNEL_STATE_UP , FTDM_CHANNEL_STATE_HANGUP , FTDM_CHANNEL_STATE_TERMINATING , 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 } ,
{ FTDM_CHANNEL_STATE_HANGUP , FTDM_CHANNEL_STATE_TERMINATING , FTDM_CHANNEL_STATE_PROGRESS_MEDIA ,
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
/**
* \ brief Stops an ISDN span
* \ param span Span to halt
* \ return Success
*
* Sets a stop flag and waits for the threads to end
*/
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-01-15 19:22:49 +00:00
ftdm_set_flag ( isdn_data , FTDM_ISDN_STOP ) ;
2009-03-12 17:22:03 +00: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
/**
* \ brief Starts an ISDN span
* \ param span Span to halt
* \ return Success or failure
*
* Launches a thread to monitor the span and a thread to generate tones on the span
*/
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
}
2009-06-08 20:01:29 +00:00
/**
* \ brief Parses an option string to flags
* \ param in String to parse for configuration options
* \ return Flags
*/
2009-03-19 23:06:04 +00:00
static uint32_t parse_opts ( const char * in )
{
uint32_t flags = 0 ;
if ( ! in ) {
return 0 ;
}
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
2009-06-08 20:01:29 +00:00
/**
* \ brief Initialises an ISDN span from configuration variables
* \ param span Span to configure
* \ param sig_cb Callback function for event signals
* \ param ap List of configuration variables
* \ return Success or failure
*/
2010-01-15 19:22:49 +00:00
static ZIO_SIG_CONFIGURE_FUNCTION ( ftdm_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-01-15 19:22:49 +00:00
ftdm_channel_t * dchans [ 2 ] = { 0 } ;
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 ;
2009-02-03 17:00:13 +00:00
# ifdef HAVE_LIBPCAP
2009-01-30 11:37:11 +00:00
int q931topcap = - 1 ; /*Q931ToPcap*/
int openPcap = 0 ; /*Flag: open Pcap file please*/
2009-02-03 17:00:13 +00:00
# endif
2008-08-29 15:58:59 +00:00
2007-05-22 22:07:05 +00:00
if ( span - > signal_type ) {
2009-02-03 17:00:13 +00:00
# ifdef HAVE_LIBPCAP
2010-01-15 19:22:49 +00:00
/*Q931ToPcap: Get the content of the q931topcap and pcapfilename args given by mod_freetdm */
2009-01-30 11:37:11 +00:00
while ( ( var = va_arg ( ap , char * ) ) ) {
2009-02-03 17:00:13 +00:00
if ( ! strcasecmp ( var , " q931topcap " ) ) {
q931topcap = va_arg ( ap , int ) ;
if ( q931topcap = = 1 ) {
/*PCAP on*/ ;
2009-01-30 11:37:11 +00:00
openPcap = 1 ;
2009-02-03 17:00:13 +00:00
} else if ( q931topcap = = 0 ) {
/*PCAP off*/
2010-01-15 19:22:49 +00:00
if ( closePcapFile ( ) ! = FTDM_SUCCESS ) return FTDM_FAIL ;
2009-01-30 11:37:11 +00:00
do_q931ToPcap = 0 ;
2010-01-15 19:22:49 +00:00
return FTDM_SUCCESS ;
2009-02-03 17:00:13 +00:00
}
2009-01-30 11:37:11 +00:00
}
if ( ! strcasecmp ( var , " pcapfilename " ) ) {
/*Put filename into global var*/
2009-02-03 17:00:13 +00:00
pcapfn = va_arg ( ap , char * ) ;
}
2009-01-30 11:37:11 +00:00
}
/*We know now, that user wants to enable Q931ToPcap and what file name he wants, so open it please*/
if ( openPcap = = 1 ) {
2010-01-15 19:22:49 +00:00
if ( openPcapFile ( ) ! = FTDM_SUCCESS ) return FTDM_FAIL ;
2009-01-30 11:37:11 +00:00
do_q931ToPcap = 1 ;
2010-01-15 19:22:49 +00:00
return FTDM_SUCCESS ;
2009-01-30 11:37:11 +00:00
}
/*Q931ToPcap done*/
2009-02-03 17:00:13 +00:00
# endif
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
}
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 ;
2008-02-28 00:19:07 +00:00
} else {
2010-01-15 19:22:49 +00: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 ;
2008-02-28 00:19:07 +00:00
x + + ;
}
2007-05-22 22:07:05 +00:00
}
}
}
if ( ! x ) {
snprintf ( span - > last_error , sizeof ( span - > last_error ) , " Span has no D-Channels! " ) ;
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
isdn_data = ftdm_malloc ( sizeof ( * isdn_data ) ) ;
2007-06-25 05:26:37 +00:00
assert ( isdn_data ! = NULL ) ;
memset ( isdn_data , 0 , sizeof ( * isdn_data ) ) ;
2007-05-22 22:07:05 +00:00
2008-08-29 15:58:59 +00:00
isdn_data - > mode = Q931_TE ;
dialect = Q931_Dialect_National ;
2008-09-05 03:53:32 +00:00
while ( ( var = va_arg ( ap , char * ) ) ) {
2008-08-29 15:58:59 +00:00
if ( ! strcasecmp ( var , " mode " ) ) {
if ( ! ( val = va_arg ( ap , char * ) ) ) {
break ;
}
isdn_data - > mode = strcasecmp ( val , " net " ) ? Q931_TE : Q931_NT ;
} else if ( ! strcasecmp ( var , " dialect " ) ) {
if ( ! ( val = va_arg ( ap , char * ) ) ) {
break ;
}
dialect = q931_str2Q931Dialect_type ( val ) ;
if ( dialect = = Q931_Dialect_Count ) {
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
2009-01-09 22:11:04 +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-01-15 19:22:49 +00:00
data = ftdm_malloc ( ( span - > chan_count - 1 ) * 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
}
}
}
2008-10-03 21:57:57 +00:00
2010-01-15 19:22:49 +00:00
span - > start = ftdm_isdn_start ;
span - > stop = ftdm_isdn_stop ;
2007-06-25 05:26:37 +00:00
isdn_data - > sig_cb = sig_cb ;
isdn_data - > dchans [ 0 ] = dchans [ 0 ] ;
isdn_data - > dchans [ 1 ] = dchans [ 1 ] ;
isdn_data - > dchan = isdn_data - > dchans [ 0 ] ;
2008-10-12 22:07:45 +00:00
isdn_data - > digit_timeout = digit_timeout ;
2007-05-22 22:07:05 +00:00
2007-06-25 05:26:37 +00:00
Q921_InitTrunk ( & isdn_data - > q921 ,
2007-05-22 22:07:05 +00:00
0 ,
0 ,
2008-08-29 15:58:59 +00:00
isdn_data - > mode ,
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 ,
2007-06-25 05:26:37 +00:00
& isdn_data - > q931 ) ;
2008-08-22 16:55:01 +00:00
2010-01-15 19:22:49 +00:00
Q921SetLogCB ( & isdn_data - > q921 , & ftdm_isdn_q921_log , isdn_data ) ;
2008-10-03 21:57:57 +00:00
Q921SetLogLevel ( & isdn_data - > q921 , ( Q921LogLevel_t ) q921loglevel ) ;
2007-05-22 22:07:05 +00:00
2007-06-25 05:26:37 +00:00
Q931Api_InitTrunk ( & isdn_data - > q931 ,
2007-05-22 22:07:05 +00:00
dialect ,
2008-08-29 15:58:59 +00:00
isdn_data - > mode ,
2007-05-22 22:07:05 +00:00
span - > trunk_type ,
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 ,
2007-06-25 05:26:37 +00:00
& isdn_data - > q921 ,
2007-05-22 22:07:05 +00:00
span ) ;
2007-05-29 16:30:20 +00:00
2010-01-15 19:22:49 +00:00
Q931SetLogCB ( & isdn_data - > q931 , & ftdm_isdn_q931_log , isdn_data ) ;
2008-10-03 21:57:57 +00:00
Q931SetLogLevel ( & isdn_data - > q931 , ( Q931LogLevel_t ) q931loglevel ) ;
2008-08-22 16:55:01 +00:00
2007-06-25 05:26:37 +00:00
isdn_data - > q931 . autoRestartAck = 1 ;
2008-08-22 16:55:01 +00:00
isdn_data - > q931 . autoConnectAck = 0 ;
2007-06-25 16:00:24 +00:00
isdn_data - > q931 . autoServiceAck = 1 ;
2007-06-25 05:26:37 +00:00
span - > signal_data = isdn_data ;
2010-01-15 19:22:49 +00:00
span - > signal_type = FTDM_SIGTYPE_ISDN ;
2007-06-14 04:44:44 +00:00
span - > outgoing_call = isdn_outgoing_call ;
2008-08-29 15:58:59 +00:00
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 ;
span - > suggest_chan_id = 1 ;
}
span - > state_map = & isdn_state_map ;
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-01-15 19:22:49 +00:00
* \ brief Openftdm ISDN signaling module definition
2009-06-08 20:01:29 +00:00
*/
2010-01-15 19:22:49 +00:00
EX_DECLARE_DATA ftdm_module_t ftdm_module = {
2008-08-29 15:58:59 +00:00
" isdn " ,
NULL ,
2009-01-30 11:37:11 +00:00
close_pcap ,
2010-01-15 19:22:49 +00:00
ftdm_isdn_init ,
ftdm_isdn_configure_span ,
2008-08-29 15:58:59 +00:00
NULL
} ;
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 :
2009-09-14 22:06:20 +00:00
* vim : set softtabstop = 4 shiftwidth = 4 tabstop = 4 :
2007-06-04 14:53:12 +00:00
*/