mirror of
				https://github.com/asterisk/asterisk.git
				synced 2025-11-03 20:38:59 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			697 lines
		
	
	
		
			24 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			697 lines
		
	
	
		
			24 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/*
 | 
						|
 * Copyright (C) 2004-2005 by Objective Systems, Inc.
 | 
						|
 *
 | 
						|
 * This software is furnished under an open source license and may be
 | 
						|
 * used and copied only in accordance with the terms of this license.
 | 
						|
 * The text of the license may generally be found in the root
 | 
						|
 * directory of this installation in the COPYING file.  It
 | 
						|
 * can also be viewed online at the following URL:
 | 
						|
 *
 | 
						|
 *   http://www.obj-sys.com/open/license.html
 | 
						|
 *
 | 
						|
 * Any redistributions of this file including modified versions must
 | 
						|
 * maintain this copyright notice.
 | 
						|
 *
 | 
						|
 *****************************************************************************/
 | 
						|
 | 
						|
#include "ooh323cDriver.h"
 | 
						|
 | 
						|
#include "asterisk.h"
 | 
						|
#include "asterisk/lock.h"
 | 
						|
 | 
						|
#include "asterisk/pbx.h"
 | 
						|
#include "asterisk/logger.h"
 | 
						|
 | 
						|
#undef AST_BACKGROUND_STACKSIZE
 | 
						|
#define AST_BACKGROUND_STACKSIZE 768 * 1024
 | 
						|
 | 
						|
#define SEC_TO_HOLD_THREAD 24
 | 
						|
 | 
						|
extern struct ast_module *myself;
 | 
						|
extern OOBOOL gH323Debug;
 | 
						|
extern OOH323EndPoint gH323ep;
 | 
						|
/* ooh323c stack thread. */
 | 
						|
static pthread_t ooh323c_thread = AST_PTHREADT_NULL;
 | 
						|
static pthread_t ooh323cmd_thread = AST_PTHREADT_NULL;
 | 
						|
static int grxframes = 240;
 | 
						|
 | 
						|
static int gtxframes = 20;
 | 
						|
 | 
						|
static struct callthread {
 | 
						|
	ast_mutex_t		lock;
 | 
						|
	int			thePipe[2];
 | 
						|
	OOBOOL			inUse;
 | 
						|
	ooCallData*		call;
 | 
						|
	struct callthread	*next, *prev;
 | 
						|
} *callThreads = NULL;
 | 
						|
 | 
						|
AST_MUTEX_DEFINE_STATIC(callThreadsLock);
 | 
						|
 | 
						|
 | 
						|
int ooh323c_start_receive_channel(ooCallData *call, ooLogicalChannel *pChannel);
 | 
						|
int ooh323c_start_transmit_channel(ooCallData *call, ooLogicalChannel *pChannel);
 | 
						|
int ooh323c_stop_receive_channel(ooCallData *call, ooLogicalChannel *pChannel);
 | 
						|
int ooh323c_stop_transmit_channel(ooCallData *call, ooLogicalChannel *pChannel);
 | 
						|
 | 
						|
int ooh323c_start_receive_datachannel(ooCallData *call, ooLogicalChannel *pChannel);
 | 
						|
int ooh323c_start_transmit_datachannel(ooCallData *call, ooLogicalChannel *pChannel);
 | 
						|
int ooh323c_stop_receive_datachannel(ooCallData *call, ooLogicalChannel *pChannel);
 | 
						|
int ooh323c_stop_transmit_datachannel(ooCallData *call, ooLogicalChannel *pChannel);
 | 
						|
 | 
						|
void* ooh323c_stack_thread(void* dummy);
 | 
						|
void* ooh323c_cmd_thread(void* dummy);
 | 
						|
void* ooh323c_call_thread(void* dummy);
 | 
						|
int ooh323c_set_aliases(ooAliases * aliases);
 | 
						|
 | 
						|
void* ooh323c_stack_thread(void* dummy)
 | 
						|
{
 | 
						|
 | 
						|
  ooMonitorChannels();
 | 
						|
  return dummy;
 | 
						|
}
 | 
						|
 | 
						|
void* ooh323c_cmd_thread(void* dummy)
 | 
						|
{
 | 
						|
 | 
						|
  ooMonitorCmdChannels();
 | 
						|
  return dummy;
 | 
						|
}
 | 
						|
 | 
						|
void* ooh323c_call_thread(void* dummy)
 | 
						|
{
 | 
						|
 struct callthread* mycthread = (struct callthread *)dummy;
 | 
						|
 struct pollfd pfds[1];
 | 
						|
 char c;
 | 
						|
 int res = 0;
 | 
						|
 | 
						|
 do {
 | 
						|
 | 
						|
 	ooMonitorCallChannels((ooCallData*)mycthread->call);
 | 
						|
	mycthread->call = NULL;
 | 
						|
	mycthread->prev = NULL;
 | 
						|
	mycthread->inUse = FALSE;
 | 
						|
 | 
						|
	ast_mutex_lock(&callThreadsLock);
 | 
						|
	mycthread->next = callThreads;
 | 
						|
	callThreads = mycthread;
 | 
						|
	if (mycthread->next) mycthread->next->prev = mycthread;
 | 
						|
	ast_mutex_unlock(&callThreadsLock);
 | 
						|
 | 
						|
	pfds[0].fd = mycthread->thePipe[0];
 | 
						|
	pfds[0].events = POLLIN;
 | 
						|
	ooSocketPoll(pfds, 1, SEC_TO_HOLD_THREAD * 1000);
 | 
						|
	if (ooPDRead(pfds, 1, mycthread->thePipe[0]))
 | 
						|
		res = read(mycthread->thePipe[0], &c, 1);
 | 
						|
 | 
						|
 	ast_mutex_lock(&callThreadsLock);
 | 
						|
	ast_mutex_lock(&mycthread->lock);
 | 
						|
 	if (mycthread->prev)
 | 
						|
		mycthread->prev->next = mycthread->next;
 | 
						|
 	else
 | 
						|
		callThreads = mycthread->next;
 | 
						|
 	if (mycthread->next)
 | 
						|
		mycthread->next->prev = mycthread->prev;
 | 
						|
	ast_mutex_unlock(&mycthread->lock);
 | 
						|
 	ast_mutex_unlock(&callThreadsLock);
 | 
						|
 | 
						|
 } while (mycthread->call != NULL && res >= 0);
 | 
						|
 | 
						|
 | 
						|
 ast_mutex_destroy(&mycthread->lock);
 | 
						|
 | 
						|
 close(mycthread->thePipe[0]);
 | 
						|
 close(mycthread->thePipe[1]);
 | 
						|
 ast_free(mycthread);
 | 
						|
 ast_module_unref(myself);
 | 
						|
 ast_update_use_count();
 | 
						|
 return NULL;
 | 
						|
}
 | 
						|
 | 
						|
int ooh323c_start_call_thread(ooCallData *call) {
 | 
						|
 char c = 'c';
 | 
						|
 struct callthread *cur = callThreads;
 | 
						|
 | 
						|
 ast_mutex_lock(&callThreadsLock);
 | 
						|
 while (cur != NULL && (cur->inUse || ast_mutex_trylock(&cur->lock))) {
 | 
						|
	cur = cur->next;
 | 
						|
 }
 | 
						|
 ast_mutex_unlock(&callThreadsLock);
 | 
						|
 | 
						|
 if (cur != NULL) {
 | 
						|
   if (cur->inUse || write(cur->thePipe[1], &c, 1) < 0) {
 | 
						|
	ast_mutex_unlock(&cur->lock);
 | 
						|
	cur = NULL;
 | 
						|
   }
 | 
						|
 }
 | 
						|
 | 
						|
/* make new thread */
 | 
						|
 if (cur == NULL) {
 | 
						|
	if (!(cur = ast_calloc(1, sizeof(struct callthread)))) {
 | 
						|
		ast_log(LOG_ERROR, "Unable to allocate thread structure for call %s\n",
 | 
						|
							call->callToken);
 | 
						|
		return -1;
 | 
						|
	}
 | 
						|
 | 
						|
	ast_module_ref(myself);
 | 
						|
	if ((socketpair(PF_LOCAL, SOCK_STREAM, 0, cur->thePipe)) == -1) {
 | 
						|
		ast_log(LOG_ERROR, "Can't create thread pipe for call %s\n", call->callToken);
 | 
						|
		ast_free(cur);
 | 
						|
		return -1;
 | 
						|
	}
 | 
						|
	cur->inUse = TRUE;
 | 
						|
	cur->call = call;
 | 
						|
 | 
						|
	ast_mutex_init(&cur->lock);
 | 
						|
 | 
						|
	if (gH323Debug)
 | 
						|
		ast_debug(1,"new call thread created for call %s\n", call->callToken);
 | 
						|
 | 
						|
	if(ast_pthread_create_detached_background(&call->callThread, NULL, ooh323c_call_thread, cur) < 0)
 | 
						|
 	{
 | 
						|
  		ast_log(LOG_ERROR, "Unable to start ooh323c call thread for call %s\n",
 | 
						|
					call->callToken);
 | 
						|
		ast_mutex_destroy(&cur->lock);
 | 
						|
		close(cur->thePipe[0]);
 | 
						|
		close(cur->thePipe[1]);
 | 
						|
		ast_free(cur);
 | 
						|
  		return -1;
 | 
						|
 	}
 | 
						|
 | 
						|
 } else {
 | 
						|
	if (gH323Debug)
 | 
						|
		ast_debug(1,"using existing call thread for call %s\n", call->callToken);
 | 
						|
	cur->inUse = TRUE;
 | 
						|
	cur->call = call;
 | 
						|
	ast_mutex_unlock(&cur->lock);
 | 
						|
 | 
						|
 }
 | 
						|
 return 0;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
int ooh323c_stop_call_thread(ooCallData *call) {
 | 
						|
 if (call->callThread != AST_PTHREADT_NULL) {
 | 
						|
  ooStopMonitorCallChannels(call);
 | 
						|
 }
 | 
						|
 return 0;
 | 
						|
}
 | 
						|
 | 
						|
int ooh323c_start_stack_thread()
 | 
						|
{
 | 
						|
   if(ast_pthread_create_background(&ooh323c_thread, NULL, ooh323c_stack_thread, NULL) < 0)
 | 
						|
   {
 | 
						|
      ast_log(LOG_ERROR, "Unable to start ooh323c thread.\n");
 | 
						|
      return -1;
 | 
						|
   }
 | 
						|
   if(ast_pthread_create_background(&ooh323cmd_thread, NULL, ooh323c_cmd_thread, NULL) < 0)
 | 
						|
   {
 | 
						|
      ast_log(LOG_ERROR, "Unable to start ooh323cmd thread.\n");
 | 
						|
      return -1;
 | 
						|
   }
 | 
						|
   return 0;
 | 
						|
}
 | 
						|
 | 
						|
int ooh323c_stop_stack_thread(void)
 | 
						|
{
 | 
						|
   if(ooh323c_thread !=  AST_PTHREADT_NULL)
 | 
						|
   {
 | 
						|
      ooStopMonitor();
 | 
						|
      pthread_join(ooh323c_thread, NULL);
 | 
						|
      ooh323c_thread =  AST_PTHREADT_NULL;
 | 
						|
      pthread_join(ooh323cmd_thread, NULL);
 | 
						|
      ooh323cmd_thread =  AST_PTHREADT_NULL;
 | 
						|
   }
 | 
						|
   return 0;
 | 
						|
}
 | 
						|
 | 
						|
int ooh323c_set_capability
 | 
						|
   (struct ast_format_cap *cap, int dtmf, int dtmfcodec)
 | 
						|
{
 | 
						|
   int ret = 0, x;
 | 
						|
   if (gH323Debug) {
 | 
						|
     ast_verb(0, "\tAdding capabilities to H323 endpoint\n");
 | 
						|
   }
 | 
						|
 | 
						|
   for(x=0; x<ast_format_cap_count(cap); x++)
 | 
						|
   {
 | 
						|
    struct ast_format *format = ast_format_cap_get_format(cap, x);
 | 
						|
      if(ast_format_cmp(format, ast_format_ulaw) == AST_FORMAT_CMP_EQUAL)
 | 
						|
      {
 | 
						|
         if (gH323Debug) {
 | 
						|
            ast_verb(0, "\tAdding g711 ulaw capability to H323 endpoint\n");
 | 
						|
	 }
 | 
						|
         ret= ooH323EpAddG711Capability(OO_G711ULAW64K, gtxframes, grxframes,
 | 
						|
                                     OORXANDTX, &ooh323c_start_receive_channel,
 | 
						|
                                     &ooh323c_start_transmit_channel,
 | 
						|
                                     &ooh323c_stop_receive_channel,
 | 
						|
                                     &ooh323c_stop_transmit_channel);
 | 
						|
      }
 | 
						|
      if(ast_format_cmp(format, ast_format_alaw) == AST_FORMAT_CMP_EQUAL)
 | 
						|
      {
 | 
						|
         if (gH323Debug) {
 | 
						|
            ast_verb(0, "\tAdding g711 alaw capability to H323 endpoint\n");
 | 
						|
	 }
 | 
						|
         ret= ooH323EpAddG711Capability(OO_G711ALAW64K, gtxframes, grxframes,
 | 
						|
                                     OORXANDTX, &ooh323c_start_receive_channel,
 | 
						|
                                     &ooh323c_start_transmit_channel,
 | 
						|
                                     &ooh323c_stop_receive_channel,
 | 
						|
                                     &ooh323c_stop_transmit_channel);
 | 
						|
      }
 | 
						|
 | 
						|
      if(ast_format_cmp(format, ast_format_g729) == AST_FORMAT_CMP_EQUAL)
 | 
						|
      {
 | 
						|
         if (gH323Debug) {
 | 
						|
	    ast_verb(0, "\tAdding g729A capability to H323 endpoint\n");
 | 
						|
	 }
 | 
						|
         ret = ooH323EpAddG729Capability(OO_G729A, 2, 24,
 | 
						|
                                     OORXANDTX, &ooh323c_start_receive_channel,
 | 
						|
                                     &ooh323c_start_transmit_channel,
 | 
						|
                                     &ooh323c_stop_receive_channel,
 | 
						|
                                     &ooh323c_stop_transmit_channel);
 | 
						|
 | 
						|
         if (gH323Debug) {
 | 
						|
            ast_verb(0, "\tAdding g729 capability to H323 endpoint\n");
 | 
						|
	 }
 | 
						|
         ret |= ooH323EpAddG729Capability(OO_G729, 2, 24,
 | 
						|
                                     OORXANDTX, &ooh323c_start_receive_channel,
 | 
						|
                                     &ooh323c_start_transmit_channel,
 | 
						|
                                     &ooh323c_stop_receive_channel,
 | 
						|
                                     &ooh323c_stop_transmit_channel);
 | 
						|
         if (gH323Debug) {
 | 
						|
            ast_verb(0, "\tAdding g729b capability to H323 endpoint\n");
 | 
						|
	 }
 | 
						|
         ret |= ooH323EpAddG729Capability(OO_G729B, 2, 24,
 | 
						|
                                     OORXANDTX, &ooh323c_start_receive_channel,
 | 
						|
                                     &ooh323c_start_transmit_channel,
 | 
						|
                                     &ooh323c_stop_receive_channel,
 | 
						|
                                     &ooh323c_stop_transmit_channel);
 | 
						|
      }
 | 
						|
 | 
						|
      if(ast_format_cmp(format, ast_format_g723) == AST_FORMAT_CMP_EQUAL)
 | 
						|
      {
 | 
						|
         if (gH323Debug) {
 | 
						|
            ast_verb(0, "\tAdding g7231 capability to H323 endpoint\n");
 | 
						|
	 }
 | 
						|
         ret = ooH323EpAddG7231Capability(OO_G7231, 1, 1, FALSE,
 | 
						|
                                     OORXANDTX, &ooh323c_start_receive_channel,
 | 
						|
                                     &ooh323c_start_transmit_channel,
 | 
						|
                                     &ooh323c_stop_receive_channel,
 | 
						|
                                     &ooh323c_stop_transmit_channel);
 | 
						|
 | 
						|
      }
 | 
						|
 | 
						|
      if(ast_format_cmp(format, ast_format_g726) == AST_FORMAT_CMP_EQUAL)
 | 
						|
      {
 | 
						|
         if (gH323Debug) {
 | 
						|
            ast_verb(0, "\tAdding g726 capability to H323 endpoint\n");
 | 
						|
	 }
 | 
						|
         ret = ooH323EpAddG726Capability(OO_G726, gtxframes, grxframes, FALSE,
 | 
						|
                                     OORXANDTX, &ooh323c_start_receive_channel,
 | 
						|
                                     &ooh323c_start_transmit_channel,
 | 
						|
                                     &ooh323c_stop_receive_channel,
 | 
						|
                                     &ooh323c_stop_transmit_channel);
 | 
						|
 | 
						|
      }
 | 
						|
 | 
						|
      if(ast_format_cmp(format, ast_format_g726_aal2) == AST_FORMAT_CMP_EQUAL)
 | 
						|
      {
 | 
						|
         if (gH323Debug) {
 | 
						|
            ast_verb(0, "\tAdding g726aal2 capability to H323 endpoint\n");
 | 
						|
	 }
 | 
						|
         ret = ooH323EpAddG726Capability(OO_G726AAL2, gtxframes, grxframes, FALSE,
 | 
						|
                                     OORXANDTX, &ooh323c_start_receive_channel,
 | 
						|
                                     &ooh323c_start_transmit_channel,
 | 
						|
                                     &ooh323c_stop_receive_channel,
 | 
						|
                                     &ooh323c_stop_transmit_channel);
 | 
						|
 | 
						|
      }
 | 
						|
 | 
						|
      if(ast_format_cmp(format, ast_format_h263) == AST_FORMAT_CMP_EQUAL)
 | 
						|
      {
 | 
						|
         if (gH323Debug) {
 | 
						|
            ast_verb(0, "\tAdding h263 capability to H323 endpoint\n");
 | 
						|
	 }
 | 
						|
         ret = ooH323EpAddH263VideoCapability(OO_H263VIDEO, 1, 0, 0, 0, 0, 320*1024,
 | 
						|
                                     OORXANDTX, &ooh323c_start_receive_channel,
 | 
						|
                                     &ooh323c_start_transmit_channel,
 | 
						|
                                     &ooh323c_stop_receive_channel,
 | 
						|
                                     &ooh323c_stop_transmit_channel);
 | 
						|
 | 
						|
      }
 | 
						|
 | 
						|
      if(ast_format_cmp(format, ast_format_gsm) == AST_FORMAT_CMP_EQUAL)
 | 
						|
      {
 | 
						|
         if (gH323Debug) {
 | 
						|
            ast_verb(0, "\tAdding gsm capability to H323 endpoint\n");
 | 
						|
	 }
 | 
						|
         ret = ooH323EpAddGSMCapability(OO_GSMFULLRATE, 4, FALSE, FALSE,
 | 
						|
                                     OORXANDTX, &ooh323c_start_receive_channel,
 | 
						|
                                     &ooh323c_start_transmit_channel,
 | 
						|
                                     &ooh323c_stop_receive_channel,
 | 
						|
                                     &ooh323c_stop_transmit_channel);
 | 
						|
 | 
						|
      }
 | 
						|
 | 
						|
      if(ast_format_cmp(format, ast_format_speex) == AST_FORMAT_CMP_EQUAL)
 | 
						|
      {
 | 
						|
         if (gH323Debug) {
 | 
						|
            ast_verb(0, "\tAdding speex capability to H323 endpoint\n");
 | 
						|
	 }
 | 
						|
         ret = ooH323EpAddSpeexCapability(OO_SPEEX, 4, 4, FALSE,
 | 
						|
                                     OORXANDTX, &ooh323c_start_receive_channel,
 | 
						|
                                     &ooh323c_start_transmit_channel,
 | 
						|
                                     &ooh323c_stop_receive_channel,
 | 
						|
                                     &ooh323c_stop_transmit_channel);
 | 
						|
 | 
						|
      }
 | 
						|
 | 
						|
    ao2_ref(format, -1);
 | 
						|
   }
 | 
						|
 | 
						|
   if(dtmf & H323_DTMF_CISCO)
 | 
						|
      ret |= ooH323EpEnableDTMFCISCO(0);
 | 
						|
   if(dtmf & H323_DTMF_RFC2833)
 | 
						|
      ret |= ooH323EpEnableDTMFRFC2833(0);
 | 
						|
   else if(dtmf & H323_DTMF_H245ALPHANUMERIC)
 | 
						|
      ret |= ooH323EpEnableDTMFH245Alphanumeric();
 | 
						|
   else if(dtmf & H323_DTMF_H245SIGNAL)
 | 
						|
      ret |= ooH323EpEnableDTMFH245Signal();
 | 
						|
 | 
						|
   return ret;
 | 
						|
}
 | 
						|
 | 
						|
int ooh323c_set_capability_for_call
 | 
						|
   (ooCallData *call, struct ast_format_cap *cap, int dtmf, int dtmfcodec,
 | 
						|
		 int t38support, int g729onlyA)
 | 
						|
{
 | 
						|
   int ret = 0, x, txframes;
 | 
						|
   if (gH323Debug) {
 | 
						|
     ast_verb(0, "\tAdding capabilities to call(%s, %s)\n", call->callType,
 | 
						|
                                                            call->callToken);
 | 
						|
   }
 | 
						|
   if(dtmf & H323_DTMF_CISCO || 1)
 | 
						|
      ret |= ooCallEnableDTMFCISCO(call,dtmfcodec);
 | 
						|
   if(dtmf & H323_DTMF_RFC2833 || 1)
 | 
						|
      ret |= ooCallEnableDTMFRFC2833(call,dtmfcodec);
 | 
						|
   if(dtmf & H323_DTMF_H245ALPHANUMERIC || 1)
 | 
						|
      ret |= ooCallEnableDTMFH245Alphanumeric(call);
 | 
						|
   if(dtmf & H323_DTMF_H245SIGNAL || 1)
 | 
						|
      ret |= ooCallEnableDTMFH245Signal(call);
 | 
						|
 | 
						|
   if (t38support)
 | 
						|
   	ooCapabilityAddT38Capability(call, OO_T38, OORXANDTX,
 | 
						|
					&ooh323c_start_receive_datachannel,
 | 
						|
					&ooh323c_start_transmit_datachannel,
 | 
						|
					&ooh323c_stop_receive_datachannel,
 | 
						|
					&ooh323c_stop_transmit_datachannel,
 | 
						|
					0);
 | 
						|
 | 
						|
   for(x=0; x<ast_format_cap_count(cap); x++)
 | 
						|
   {
 | 
						|
    struct ast_format *format = ast_format_cap_get_format(cap, x);
 | 
						|
      if(ast_format_cmp(format, ast_format_ulaw) == AST_FORMAT_CMP_EQUAL)
 | 
						|
      {
 | 
						|
         if (gH323Debug) {
 | 
						|
            ast_verb(0, "\tAdding g711 ulaw capability to call(%s, %s)\n",
 | 
						|
                                              call->callType, call->callToken);
 | 
						|
	 }
 | 
						|
	 txframes = ast_format_cap_get_format_framing(cap, format);
 | 
						|
         ret= ooCallAddG711Capability(call, OO_G711ULAW64K, txframes,
 | 
						|
                                      txframes, OORXANDTX,
 | 
						|
                                      &ooh323c_start_receive_channel,
 | 
						|
                                      &ooh323c_start_transmit_channel,
 | 
						|
                                      &ooh323c_stop_receive_channel,
 | 
						|
                                      &ooh323c_stop_transmit_channel);
 | 
						|
      }
 | 
						|
      if(ast_format_cmp(format, ast_format_alaw) == AST_FORMAT_CMP_EQUAL)
 | 
						|
      {
 | 
						|
         if (gH323Debug) {
 | 
						|
            ast_verb(0, "\tAdding g711 alaw capability to call(%s, %s)\n",
 | 
						|
                                            call->callType, call->callToken);
 | 
						|
	 }
 | 
						|
         txframes = ast_format_cap_get_format_framing(cap, format);
 | 
						|
         ret= ooCallAddG711Capability(call, OO_G711ALAW64K, txframes,
 | 
						|
                                     txframes, OORXANDTX,
 | 
						|
                                     &ooh323c_start_receive_channel,
 | 
						|
                                     &ooh323c_start_transmit_channel,
 | 
						|
                                     &ooh323c_stop_receive_channel,
 | 
						|
                                     &ooh323c_stop_transmit_channel);
 | 
						|
      }
 | 
						|
 | 
						|
      if(ast_format_cmp(format, ast_format_g726) == AST_FORMAT_CMP_EQUAL)
 | 
						|
      {
 | 
						|
         if (gH323Debug) {
 | 
						|
            ast_verb(0, "\tAdding g726 capability to call (%s, %s)\n",
 | 
						|
                                           call->callType, call->callToken);
 | 
						|
	 }
 | 
						|
	 txframes = ast_format_cap_get_format_framing(cap, format);
 | 
						|
         ret = ooCallAddG726Capability(call, OO_G726, txframes, grxframes, FALSE,
 | 
						|
                                     OORXANDTX, &ooh323c_start_receive_channel,
 | 
						|
                                     &ooh323c_start_transmit_channel,
 | 
						|
                                     &ooh323c_stop_receive_channel,
 | 
						|
                                     &ooh323c_stop_transmit_channel);
 | 
						|
 | 
						|
      }
 | 
						|
 | 
						|
      if(ast_format_cmp(format, ast_format_g726_aal2) == AST_FORMAT_CMP_EQUAL)
 | 
						|
      {
 | 
						|
         if (gH323Debug) {
 | 
						|
            ast_verb(0, "\tAdding g726aal2 capability to call (%s, %s)\n",
 | 
						|
                                           call->callType, call->callToken);
 | 
						|
	 }
 | 
						|
	 txframes = ast_format_cap_get_format_framing(cap, format);
 | 
						|
         ret = ooCallAddG726Capability(call, OO_G726AAL2, txframes, grxframes, FALSE,
 | 
						|
                                     OORXANDTX, &ooh323c_start_receive_channel,
 | 
						|
                                     &ooh323c_start_transmit_channel,
 | 
						|
                                     &ooh323c_stop_receive_channel,
 | 
						|
                                     &ooh323c_stop_transmit_channel);
 | 
						|
 | 
						|
      }
 | 
						|
 | 
						|
      if(ast_format_cmp(format, ast_format_g729) == AST_FORMAT_CMP_EQUAL)
 | 
						|
      {
 | 
						|
 | 
						|
         txframes = (ast_format_cap_get_format_framing(cap, format))/10;
 | 
						|
         if (gH323Debug) {
 | 
						|
            ast_verb(0, "\tAdding g729A capability to call(%s, %s)\n",
 | 
						|
                                            call->callType, call->callToken);
 | 
						|
	 }
 | 
						|
         ret= ooCallAddG729Capability(call, OO_G729A, txframes, txframes,
 | 
						|
                                     OORXANDTX, &ooh323c_start_receive_channel,
 | 
						|
                                     &ooh323c_start_transmit_channel,
 | 
						|
                                     &ooh323c_stop_receive_channel,
 | 
						|
                                     &ooh323c_stop_transmit_channel);
 | 
						|
	 if (g729onlyA)
 | 
						|
		continue;
 | 
						|
         if (gH323Debug) {
 | 
						|
            ast_verb(0, "\tAdding g729 capability to call(%s, %s)\n",
 | 
						|
                                            call->callType, call->callToken);
 | 
						|
	 }
 | 
						|
         ret|= ooCallAddG729Capability(call, OO_G729, txframes, txframes,
 | 
						|
                                     OORXANDTX, &ooh323c_start_receive_channel,
 | 
						|
                                     &ooh323c_start_transmit_channel,
 | 
						|
                                     &ooh323c_stop_receive_channel,
 | 
						|
                                     &ooh323c_stop_transmit_channel);
 | 
						|
         if (gH323Debug) {
 | 
						|
            ast_verb(0, "\tAdding g729B capability to call(%s, %s)\n",
 | 
						|
                                            call->callType, call->callToken);
 | 
						|
	 }
 | 
						|
         ret|= ooCallAddG729Capability(call, OO_G729B, txframes, txframes,
 | 
						|
                                     OORXANDTX, &ooh323c_start_receive_channel,
 | 
						|
                                     &ooh323c_start_transmit_channel,
 | 
						|
                                     &ooh323c_stop_receive_channel,
 | 
						|
                                     &ooh323c_stop_transmit_channel);
 | 
						|
 | 
						|
      }
 | 
						|
 | 
						|
      if(ast_format_cmp(format, ast_format_g723) == AST_FORMAT_CMP_EQUAL)
 | 
						|
      {
 | 
						|
         if (gH323Debug) {
 | 
						|
            ast_verb(0, "\tAdding g7231 capability to call (%s, %s)\n",
 | 
						|
                                           call->callType, call->callToken);
 | 
						|
	 }
 | 
						|
         ret = ooCallAddG7231Capability(call, OO_G7231, 1, 1, FALSE,
 | 
						|
                                     OORXANDTX, &ooh323c_start_receive_channel,
 | 
						|
                                     &ooh323c_start_transmit_channel,
 | 
						|
                                     &ooh323c_stop_receive_channel,
 | 
						|
                                     &ooh323c_stop_transmit_channel);
 | 
						|
 | 
						|
      }
 | 
						|
 | 
						|
      if(ast_format_cmp(format, ast_format_h263) == AST_FORMAT_CMP_EQUAL)
 | 
						|
      {
 | 
						|
         if (gH323Debug) {
 | 
						|
            ast_verb(0, "\tAdding h263 capability to call (%s, %s)\n",
 | 
						|
                                           call->callType, call->callToken);
 | 
						|
	 }
 | 
						|
         ret = ooCallAddH263VideoCapability(call, OO_H263VIDEO, 1, 0, 0, 0, 0, 320*1024,
 | 
						|
                                     OORXANDTX, &ooh323c_start_receive_channel,
 | 
						|
                                     &ooh323c_start_transmit_channel,
 | 
						|
                                     &ooh323c_stop_receive_channel,
 | 
						|
                                     &ooh323c_stop_transmit_channel);
 | 
						|
 | 
						|
      }
 | 
						|
 | 
						|
      if(ast_format_cmp(format, ast_format_gsm) == AST_FORMAT_CMP_EQUAL)
 | 
						|
      {
 | 
						|
         if (gH323Debug) {
 | 
						|
            ast_verb(0, "\tAdding gsm capability to call(%s, %s)\n",
 | 
						|
                                             call->callType, call->callToken);
 | 
						|
	 }
 | 
						|
         ret = ooCallAddGSMCapability(call, OO_GSMFULLRATE, 4, FALSE, FALSE,
 | 
						|
                                     OORXANDTX, &ooh323c_start_receive_channel,
 | 
						|
                                     &ooh323c_start_transmit_channel,
 | 
						|
                                     &ooh323c_stop_receive_channel,
 | 
						|
                                     &ooh323c_stop_transmit_channel);
 | 
						|
      }
 | 
						|
 | 
						|
      if(ast_format_cmp(format, ast_format_speex) == AST_FORMAT_CMP_EQUAL)
 | 
						|
      {
 | 
						|
         if (gH323Debug) {
 | 
						|
            ast_verb(0, "\tAdding Speex capability to call(%s, %s)\n",
 | 
						|
                                             call->callType, call->callToken);
 | 
						|
	 }
 | 
						|
         ret = ooCallAddSpeexCapability(call, OO_SPEEX, 4, 4, FALSE,
 | 
						|
                                     OORXANDTX, &ooh323c_start_receive_channel,
 | 
						|
                                     &ooh323c_start_transmit_channel,
 | 
						|
                                     &ooh323c_stop_receive_channel,
 | 
						|
                                     &ooh323c_stop_transmit_channel);
 | 
						|
      }
 | 
						|
 | 
						|
      ao2_ref(format, -1);
 | 
						|
   }
 | 
						|
   return ret;
 | 
						|
}
 | 
						|
 | 
						|
int ooh323c_set_aliases(ooAliases * aliases)
 | 
						|
{
 | 
						|
   ooAliases *cur = aliases;
 | 
						|
   while(cur)
 | 
						|
   {
 | 
						|
      switch(cur->type)
 | 
						|
      {
 | 
						|
      case T_H225AliasAddress_dialedDigits:
 | 
						|
         ooH323EpAddAliasDialedDigits(cur->value);
 | 
						|
         break;
 | 
						|
      case T_H225AliasAddress_h323_ID:
 | 
						|
         ooH323EpAddAliasH323ID(cur->value);
 | 
						|
         break;
 | 
						|
      case T_H225AliasAddress_url_ID:
 | 
						|
         ooH323EpAddAliasURLID(cur->value);
 | 
						|
         break;
 | 
						|
      case T_H225AliasAddress_email_ID:
 | 
						|
         ooH323EpAddAliasEmailID(cur->value);
 | 
						|
         break;
 | 
						|
      default:
 | 
						|
         ast_debug(1, "Ignoring unknown alias type\n");
 | 
						|
      }
 | 
						|
      cur = cur->next;
 | 
						|
   }
 | 
						|
   return 1;
 | 
						|
}
 | 
						|
 | 
						|
int ooh323c_start_receive_channel(ooCallData *call, ooLogicalChannel *pChannel)
 | 
						|
{
 | 
						|
   struct ast_format *tmpfmt = NULL;
 | 
						|
   tmpfmt = convertH323CapToAsteriskCap(pChannel->chanCap->cap);
 | 
						|
   if(tmpfmt) {
 | 
						|
      /* ooh323_set_read_format(call, fmt); */
 | 
						|
   }else{
 | 
						|
     ast_log(LOG_ERROR, "Invalid capability type for receive channel %s\n",
 | 
						|
                                                          call->callToken);
 | 
						|
     return -1;
 | 
						|
   }
 | 
						|
   return 1;
 | 
						|
}
 | 
						|
 | 
						|
int ooh323c_start_transmit_channel(ooCallData *call, ooLogicalChannel *pChannel)
 | 
						|
{
 | 
						|
   struct ast_format *tmpfmt = NULL;
 | 
						|
   tmpfmt = convertH323CapToAsteriskCap(pChannel->chanCap->cap);
 | 
						|
 | 
						|
   if (tmpfmt) {
 | 
						|
    if ((ast_format_cmp(tmpfmt, ast_format_alaw) == AST_FORMAT_CMP_EQUAL) ||
 | 
						|
      (ast_format_cmp(tmpfmt, ast_format_ulaw) == AST_FORMAT_CMP_EQUAL)) {
 | 
						|
  	ooh323_set_write_format(call, tmpfmt, ((OOCapParams *)(pChannel->chanCap->params))->txframes);
 | 
						|
    } else if (ast_format_cmp(tmpfmt, ast_format_g729) == AST_FORMAT_CMP_EQUAL) {
 | 
						|
	 ooh323_set_write_format(call, tmpfmt, ((OOCapParams *)(pChannel->chanCap->params))->txframes*10);
 | 
						|
  } else {
 | 
						|
	ooh323_set_write_format(call, tmpfmt, 0);
 | 
						|
      }
 | 
						|
   }else{
 | 
						|
      ast_log(LOG_ERROR, "Invalid capability type for receive channel %s\n",
 | 
						|
                                                          call->callToken);
 | 
						|
      return -1;
 | 
						|
   }
 | 
						|
   setup_rtp_connection(call, pChannel->remoteIP, pChannel->remoteMediaPort);
 | 
						|
    return 1;
 | 
						|
}
 | 
						|
 | 
						|
int ooh323c_stop_receive_channel(ooCallData *call, ooLogicalChannel *pChannel)
 | 
						|
{
 | 
						|
   return 1;
 | 
						|
}
 | 
						|
 | 
						|
int ooh323c_stop_transmit_channel(ooCallData *call, ooLogicalChannel *pChannel)
 | 
						|
{
 | 
						|
   close_rtp_connection(call);
 | 
						|
   return 1;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
int ooh323c_start_receive_datachannel(ooCallData *call, ooLogicalChannel *pChannel)
 | 
						|
{
 | 
						|
   return 1;
 | 
						|
}
 | 
						|
 | 
						|
int ooh323c_start_transmit_datachannel(ooCallData *call, ooLogicalChannel *pChannel)
 | 
						|
{
 | 
						|
   setup_udptl_connection(call, pChannel->remoteIP, pChannel->remoteMediaPort);
 | 
						|
   return 1;
 | 
						|
}
 | 
						|
 | 
						|
int ooh323c_stop_receive_datachannel(ooCallData *call, ooLogicalChannel *pChannel)
 | 
						|
{
 | 
						|
   return 1;
 | 
						|
}
 | 
						|
 | 
						|
int ooh323c_stop_transmit_datachannel(ooCallData *call, ooLogicalChannel *pChannel)
 | 
						|
{
 | 
						|
   close_udptl_connection(call);
 | 
						|
   return 1;
 | 
						|
}
 | 
						|
 | 
						|
struct ast_format *convertH323CapToAsteriskCap(int cap)
 | 
						|
{
 | 
						|
   switch(cap)
 | 
						|
   {
 | 
						|
      case OO_G711ULAW64K:
 | 
						|
         return ast_format_ulaw;
 | 
						|
      case OO_G711ALAW64K:
 | 
						|
         return ast_format_alaw;
 | 
						|
      case OO_GSMFULLRATE:
 | 
						|
         return ast_format_gsm;
 | 
						|
      case OO_SPEEX:
 | 
						|
         return ast_format_speex;
 | 
						|
      case OO_G729:
 | 
						|
         return ast_format_g729;
 | 
						|
      case OO_G729A:
 | 
						|
         return ast_format_g729;
 | 
						|
      case OO_G729B:
 | 
						|
         return ast_format_g729;
 | 
						|
      case OO_G7231:
 | 
						|
         return ast_format_g723;
 | 
						|
      case OO_G726:
 | 
						|
         return ast_format_g726;
 | 
						|
      case OO_G726AAL2:
 | 
						|
         return ast_format_g726_aal2;
 | 
						|
      case OO_H263VIDEO:
 | 
						|
         return ast_format_h263;
 | 
						|
      default:
 | 
						|
         ast_debug(1, "Cap %d is not supported by driver yet\n", cap);
 | 
						|
         return NULL;
 | 
						|
   }
 | 
						|
 | 
						|
   return NULL;
 | 
						|
}
 |