CAF: Leg Creation Samples

From TBwiki
(Difference between revisions)
Jump to: navigation, search
(Creating a Media-only Leg)
m (really old article. Removed the needs revising category tag.)
 
(79 intermediate revisions by 4 users not shown)
Line 1: Line 1:
==== Creating a Standalone Outgoing Call ====
+
== Creating a Standalone Outgoing Call Leg ==
First you must build the outgoing leg attribute:
+
 
     PTRCTBCMC_CALL_LEG_ATTRIBUTE ptrOutgoingLegAttribute;
 
     PTRCTBCMC_CALL_LEG_ATTRIBUTE ptrOutgoingLegAttribute;
 
 
 
     ptrOutgoingLegAttribute = tbnew CTBCMC_CALL_LEG_ATTRIBUTE();
 
     ptrOutgoingLegAttribute = tbnew CTBCMC_CALL_LEG_ATTRIBUTE();
 
    
 
    
     ptrOutgoingLegAttribute->GetCalledNumber()           = "123-4567";
+
     ptrOutgoingLegAttribute->GetCalledNumber()         = "123-4567";
     ptrOutgoingLegAttribute->GetCallingNumber()           = "987-6543";
+
     ptrOutgoingLegAttribute->GetCallingNumber()       = "987-6543";
     ptrOutgoingLegAttribute->GetNetworkAccessPoint()     = "NAP_SS7_MONTREAL";
+
     ptrOutgoingLegAttribute->GetNetworkAccessPoint()   = "NAP_SS7_MONTREAL";
 +
 
 +
    PCTBCMCLeg pCallLeg = tbnew CTBCMCLeg( ++mun32LegId, ptrOutgoingLegAttribute, this, 0, &mLegMutex );
 +
    pCallLeg->CreateCall();
  
Once you have all your parameters set, you can now create the outgoing call leg like this:
+
== Bridging an Incoming Call Leg (using CTBCAFBridge )==
  
    PTRCTBCAFCallLeg ptrOutgoingCallLeg = CreateOutgoingCallLeg(ptrOutgoingLegAttribute);
+
TBX_VOID CTBS2GWSimpleCall::OnCallLegPresent
 +
(
 +
  IN TBCMC_LEG_ID in_LegId,
 +
  IN CTBCMC_CALL_LEG_ATTRIBUTE_COMMON & in_CallLegAttribute,
 +
  IN CTBCMC_PROTOCOL_ATTRIBUTE & in_ProtocolAttribute
 +
)
 +
{
 +
TBX_RESULT Result;
 +
PITBCAFCallFlow pBridge;
 +
PTRCTBCMC_CALL_LEG_ATTRIBUTE ptrIncomingLegAttribute;
 +
PTRCTBCMC_PROTOCOL_ATTRIBUTE ptrIncomingLegProtocolAttributes;
 +
PTRCTBCMC_PROTOCOL_ATTRIBUTE ptrAcceptCallProtAttribute;
 +
PTRCTBCMC_CALL_LEG_ATTRIBUTE ptrOutgoingLegAttribute;
 +
PTRCTBCMC_PROTOCOL_ATTRIBUTE ptrOutgoingLegProtocolAttributes;
 +
 +
 +
/*---------------------------------------------------------------------------------------------------------------------------
 +
|  Code section
 +
*--------------------------------------------------------------------------------------------------------------------------*/
 +
CAFCODE( CTBS2GWSimpleCall::OnCallLegPresent )
 +
{
 +
// Create smart pointers to incoming call leg attributes so behaviors can keep reference to it if needed
 +
ptrIncomingLegAttribute = tbnew CTBCMC_CALL_LEG_ATTRIBUTE( *(in_CallLegAttribute.GetCallLegAttribute()) );
 +
ptrIncomingLegProtocolAttributes = tbnew CTBCMC_PROTOCOL_ATTRIBUTE( in_ProtocolAttribute, TBX_TRUE );
 +
 +
// Build default protocol attributes object so behaviors can modify
 +
// attributes used to Accept the incoming call leg
 +
ptrAcceptCallProtAttribute = tbnew CTBCMC_PROTOCOL_ATTRIBUTE();
 +
 +
// Build default protocol attributes object so behaviors can modify
 +
// attributes used to create the outgoing call leg
 +
ptrOutgoingLegProtocolAttributes = tbnew CTBCMC_PROTOCOL_ATTRIBUTE();
 +
 +
// Create an empty outgoing call leg attribute object
 +
ptrOutgoingLegAttribute = tbnew CTBCMC_CALL_LEG_ATTRIBUTE();
 +
 +
// Fill some information in the outgoing call leg attributes from the incoming leg attributes
 +
// *** Warning: Don't use operator=. We dont wan't to copy EVERYTHING, but only information
 +
// common to both incoming and outgoing legs, such as calling/called number.
 +
// Function CopyLegAttributeFrom() takes care of copying relevant information.
 +
ptrOutgoingLegAttribute->CopyLegAttributeFrom( *(ptrIncomingLegAttribute.Get()) );
 +
 +
// Create the call (i.e. which contains 2 call legs)
 +
pBridge = tbnew CTBCAFBridge( 0, this );
 +
TBCAF_EXIT_ON_NULL( pBridge, TBX_RESULT_FAIL, "Failed to allocate call." );
 +
 +
/* Prepare the chain of behaviors */
 +
PITBCAFCallFlow pPreviousBehavior = pBridge;
 +
 +
/* Add the example behavior to the chain of behaviors */
 +
pPreviousBehavior = tbnew CTBS2GWCallBehaviorExample( pPreviousBehavior );
 +
 +
// Add legs information
 +
Result = pBridge->AddIncoming( in_LegId, ptrIncomingLegAttribute, ptrIncomingLegProtocolAttributes, ptrAcceptCallProtAttribute );
 +
TBCAF_EXIT_ON_ERROR(Result, "AddIncoming Failed." );
 +
 +
// Provide an uninitialized PTRCTBCMC_PROTOCOL_ATTRIBUTE
 +
Result = pBridge->AddOutgoing( ptrOutgoingLegAttribute, ptrOutgoingLegProtocolAttributes );
 +
TBCAF_EXIT_ON_ERROR(Result, "AddOutgoing Failed." );
 +
 +
 +
Result = pBridge->InitCall( &pBridge );
 +
if( pBridge == NULL )
 +
{
 +
/* Note: If this return value is NULL, this means the InitCall function
 +
has initialized no call leg's state machine (failed to) and
 +
destroyed itself immediately (it normally destroys itself later
 +
when the state machine of all call legs has terminated).
 +
The above line of code shall thus be the last of the function
 +
to avoid accessing any member of "this" here. */
 +
TBX_EXIT_CLEANUP( Result );
 +
}
 +
TBCAF_EXIT_ON_ERROR(Result, "InitCall Failed." );
 +
 +
TBX_EXIT_SUCCESS( TBX_RESULT_OK );
 +
}
 +
 +
/*---------------------------------------------------------------------------------------------------------------------------
 +
|  Exception handling section
 +
*--------------------------------------------------------------------------------------------------------------------------*/
 +
CAF_EXCEPTION_HANDLING
 +
 +
/*---------------------------------------------------------------------------------------------------------------------------
 +
|  Error handling section
 +
*--------------------------------------------------------------------------------------------------------------------------*/
 +
CAF_ERROR_HANDLING( CAF_VERBOSE )
 +
{
 +
if( pBridge == NULL )
 +
{
 +
// Try to refuse the call leg
 +
CTBCMCLeg::RefuseLeg( in_LegId, in_CallLegAttribute );
 +
}
 +
}
 +
 +
/*---------------------------------------------------------------------------------------------------------------------------
 +
|  Cleanup section
 +
*--------------------------------------------------------------------------------------------------------------------------*/
 +
CAF_CLEANUP
 +
{
 +
}
 +
 +
RETURN_VOID;
 +
}
  
The function CreateOutgoingCallLeg() will do all the work to create the leg and will return it. You have now an active outgoing call leg.
 
  
==== Bridging an Incoming Call (manual method)====
+
InitCall() will create each leg added from AddIncoming() and AddOutgoing() and incoming and outgoing call leg will be joined when we receive the notification OnCallLegAnswered() indicating that the outgoing call has been answered by the remote peer.
  TODO ADD SAMPLE CODE
+
 
==== Bridging an Incoming Call (using CTBCAFBridge )====
+
 
  TODO ADD SAMPLE CODE
+
== Pitfall ==
==== Creating a Media-only Leg ====
+
It's mandatory that the call leg and protocol attribute objects passed to AddIncoming() and AddOutgoing() remain valid until function InitCall() returns.
You must first create a media only leg attribute:
+
 
 +
== Creating a Media-only Leg ==
 +
 
 +
===If you are using a nap of type NAP_MEDIA_TDM:===
 +
   
 
     PTRCTBCMC_MEDIA_ONLY_LEG_ATTRIBUTE ptrLegAttribute;
 
     PTRCTBCMC_MEDIA_ONLY_LEG_ATTRIBUTE ptrLegAttribute;
 +
    PTBCMC_MEDIA_DESCRIPTION          pMediaDesc;
 
    
 
    
 
     ptrLegAttribute = tbnew CTBCMC_MEDIA_ONLY_LEG_ATTRIBUTE();
 
     ptrLegAttribute = tbnew CTBCMC_MEDIA_ONLY_LEG_ATTRIBUTE();
     ptrLegAttribute->GetNetworkAccessPoint()     = "NAP_SS7_MONTREAL";
+
     pMediaDesc   = &ptrLegAttribute->GetProfile().GetMediaDescription();  
   
+
     pMediaDesc = ptrLegAttribute->GetProfile()->MediaDescription;
+
     ptrLegAttribute->GetNetworkAccessPoint()     = "NAP_MEDIA_TDM";
 
     pMediaDesc->Type                              = TBCMC_MEDIA_TYPE_AUDIO;
 
     pMediaDesc->Type                              = TBCMC_MEDIA_TYPE_AUDIO;
     pMediaDesc->Transport                        = TBCMC_MEDIA_TRANSPORT_TDM or TBCMC_MEDIA_TRANSPORT_IP;
+
     pMediaDesc->Transport                        = TBCMC_MEDIA_TRANSPORT_TDM;
 
+
Depending on which type of NAP your using, some parameters must be set. If you are using the NAP_MEDIA_TDM:
+
 
+
 
     pMediaDesc->Settings.TdmAudio.Type            = TBCMC_MEDIA_SETTINGS_TYPE_TDM_AUDIO;
 
     pMediaDesc->Settings.TdmAudio.Type            = TBCMC_MEDIA_SETTINGS_TYPE_TDM_AUDIO;
 
     pMediaDesc->Settings.TdmAudio.un8Timeslot    = 5;
 
     pMediaDesc->Settings.TdmAudio.un8Timeslot    = 5;
Line 40: Line 144:
 
         sizeof(pMediaDesc->Settings.TdmAudio.szTrunkName)
 
         sizeof(pMediaDesc->Settings.TdmAudio.szTrunkName)
 
     );
 
     );
 +
 
 +
    PCTBCMCLeg pCallLeg = tbnew CTBCMCLeg( ++mun32LegId, ptrLegAttribute, this, 0, &mLegMutex );
 +
    pCallLeg->CreateCall();
  
If you are using the NAP_MEDIA_VOIP:
+
===If you are using a nap of type NAP_MEDIA_VOIP:===
  
     TBX_RESULT               Result;
+
     TBX_RESULT                         Result;
     TBX_SDP_INFO             SdpInfo;
+
     TBX_SDP_INFO                       SdpInfo;
      
+
     PTRCTBCMC_MEDIA_ONLY_LEG_ATTRIBUTE ptrLegAttribute;
 +
    PTBCMC_MEDIA_DESCRIPTION          pMediaDesc;
 +
 
 +
    ptrLegAttribute = tbnew CTBCMC_MEDIA_ONLY_LEG_ATTRIBUTE();
 +
    pMediaDesc   = &ptrLegAttribute->GetProfile().GetMediaDescription();
 +
 
 +
    ptrLegAttribute->GetNetworkAccessPoint()      = "NAP_MEDIA_VOIP";
 +
    pMediaDesc->Type                              = TBCMC_MEDIA_TYPE_AUDIO;
 +
    pMediaDesc->Transport                        = TBCMC_MEDIA_TRANSPORT_IP;
 
     pMediaDesc->Settings.PacketAudio.Type        = TBCMC_MEDIA_SETTINGS_TYPE_PACKET_AUDIO;
 
     pMediaDesc->Settings.PacketAudio.Type        = TBCMC_MEDIA_SETTINGS_TYPE_PACKET_AUDIO;
 
    
 
    
Line 51: Line 166:
 
     Result = BuildSdpInfo("", 0, SdpInfo);                // No IP specified, Toolpack will choose one
 
     Result = BuildSdpInfo("", 0, SdpInfo);                // No IP specified, Toolpack will choose one
 
     TBCAF_EXIT_ON_ERROR( Result, "BuildSdpInfo failed." );
 
     TBCAF_EXIT_ON_ERROR( Result, "BuildSdpInfo failed." );
     ptrLegAttribute->SetLocalSDP(&SdpInfo);
+
     ptrLegAttribute->GetProfile().SetLocalSDP(SdpInfo);
 
      
 
      
 
     // Set Peer SDP
 
     // Set Peer SDP
 
     Result = BuildSdpInfo("10.0.0.15", 5000, SdpInfo);    // Using peer IP address and port
 
     Result = BuildSdpInfo("10.0.0.15", 5000, SdpInfo);    // Using peer IP address and port
 
     TBCAF_EXIT_ON_ERROR( Result, "BuildSdpInfo failed." );
 
     TBCAF_EXIT_ON_ERROR( Result, "BuildSdpInfo failed." );
     ptrLegAttribute->SetPeerSDP(&SdpInfo);
+
     ptrLegAttribute->GetProfile().SetPeerSDP(SdpInfo);
 +
 
 +
    PCTBCMCLeg pCallLeg = tbnew CTBCMCLeg( ++mun32LegId, ptrLegAttribute, this, 0, &mLegMutex );
 +
    pCallLeg->CreateCall();
  
Here is the implementation of BuildSdpInfo, it's a good example of how to build a sdp info from scratch:
+
Note that you have two possibilities to build a TBX_SDP_INFO structure. The first one is like in the example (BuildSdpInfo()) where you can build manually your TBX_SDP_INFO structure and the second one is when you have an sdp info in text format. There is a function that will parse the text and build a TBX_SDP_INFO structure from it.
   TBX_RESULT CTBS2GWSimpleMediaCall::BuildSdpInfo
+
 
 +
   TBXMediaLibSdpParse
 
   (
 
   (
  IN PTBX_CHAR   in_pszIp,
+
      IN   PTBX_CHAR                       in_pszSdp,
  IN  TBX_UINT32   in_un32Port,
+
      OUT  PTBX_SDP_INFO                    out_pSdpInfo,
  OUT TBX_SDP_INFO&  out_SdpInfo
+
      IN    TBX_MEDIA_SDP_PARSE_OPTIONS      in_Options
   )
+
   );
  {
+
 
   TBCAF_MUTEX_GET_SCOPE_BEGIN( &mMutex )
+
To see how to build a TBX_SDP_INFO structure, refer to [[Customer_Application_Framework_:_Filling_SDP_Structure|Filling an SDP Structure]].
  PTBX_MEDIA_CAPABILITY pCap;
+
 
  PTBX_SDP_MEDIA_CONNECTION pConn;
+
 
  PTBX_MEDIA_CAPABILITY_DESCRIPTION pMediaDesc;
+
 
  /*---------------------------------------------------------------------------------------------------------------------------
+
[[category:CAF]]
  |  Code section
+
  *--------------------------------------------------------------------------------------------------------------------------*/
+
  CAFCODE( CTBS2GWSimpleMediaCall::BuildSdpInfo )
+
  {
+
  memset(&out_SdpInfo, 0, sizeof(out_SdpInfo));
+
 
+
  // Set capabilities
+
  pCap = &( out_SdpInfo.Capabilities.aMediaCap[ out_SdpInfo.Capabilities.un8NbMediaCapabilities ] );
+
  {
+
  /* Reserve this capability in aMediaCap */
+
  out_SdpInfo.Capabilities.un8NbMediaCapabilities++;
+
 
+
  pCap->Encoding.MediaType = TBX_MEDIA_TYPE_AUDIO_PCMU;
+
  pCap->Direction = TBX_MEDIA_DIRECTION_RX_TX_SAME;
+
    pCap->Transport.Type = TBX_MEDIA_TRANSPORT_TYPE_RTP;
+
    pCap->Transport.Rtp.PayloadType = TBX_MEDIA_PAYLOAD_TYPE_PCMU;
+
  pCap->Transport.Rtp.un16MaxAudioFramesPerPacket = 160;
+
  pCap->Transport.Rtp.un16PacketDurationMs = 20;
+
  }
+
  pCap = &( out_SdpInfo.Capabilities.aMediaCap[ out_SdpInfo.Capabilities.un8NbMediaCapabilities ] );
+
  {
+
  /* Reserve this capability in aMediaCap */
+
  out_SdpInfo.Capabilities.un8NbMediaCapabilities++;
+
 
+
  pCap->Encoding.MediaType = TBX_MEDIA_TYPE_AUDIO_PCMA;
+
  pCap->Direction = TBX_MEDIA_DIRECTION_RX_TX_SAME;
+
  pCap->Transport.Type = TBX_MEDIA_TRANSPORT_TYPE_RTP;
+
    pCap->Transport.Rtp.PayloadType = TBX_MEDIA_PAYLOAD_TYPE_PCMA;
+
    pCap->Transport.Rtp.un16MaxAudioFramesPerPacket = 160;
+
    pCap->Transport.Rtp.un16PacketDurationMs = 40;
+
  }
+
  out_SdpInfo.Capabilities.un8NbCapabilityGroups = 1;
+
  out_SdpInfo.Capabilities.aCapGroups[0].un8NbSimultaneousCap = 1;
+
  pMediaDesc = &( out_SdpInfo.Capabilities.aCapGroups[0].aSimultaneousCap[ 0 ] );
+
  pMediaDesc->un8NbPorts = in_un32Port ? 1 : 0;
+
  pMediaDesc->un16UdpPort = in_un32Port;
+
  pMediaDesc->un8ConnectionIdx = 0;
+
  pMediaDesc->un8NbChoices = 2;
+
  pMediaDesc->aMediaCapChoices[0] = 0;
+
  pMediaDesc->aMediaCapChoices[1] = 1;
+
 
+
  // Set connection info
+
  pConn = &out_SdpInfo.aConnections[out_SdpInfo.un8NbConnections];
+
  pConn->un32NumAddr = 0; /* Not used, not multicast */
+
  pConn->un16Ttl = 0; /* Not used, not multicast */
+
  pConn->IPDescriptionType = (TBX_UINT8)TBX_SDP_MEDIA_IP_DESCRIPTION_TYPE_IPV4_ADDRESS;
+
  Strncpy
+
  (
+
  pConn->szIp,
+
  in_pszIp,
+
  sizeof( pConn->szIp )
+
  );
+
  out_SdpInfo.un8NbConnections++;
+
 
+
  TBX_EXIT_SUCCESS( TBX_RESULT_OK );
+
  }
+
 
+
  /*---------------------------------------------------------------------------------------------------------------------------
+
  |  Exception handling section
+
  *--------------------------------------------------------------------------------------------------------------------------*/
+
  CAF_EXCEPTION_HANDLING
+
 
+
  /*---------------------------------------------------------------------------------------------------------------------------
+
  |  Error handling section
+
  *--------------------------------------------------------------------------------------------------------------------------*/
+
  CAF_ERROR_HANDLING( CAF_VERBOSE )
+
  {
+
  }
+
 
+
  /*--------------------------------------------------------------------------------------------------------------------------
+
  |  Cleanup section
+
  *-------------------------------------------------------------------------------------------------------------------------*/
+
  CAF_CLEANUP
+
  {
+
  }
+
 
+
  RETURN;
+
  TBCAF_MUTEX_GET_SCOPE_END( &mMutex )
+
  }
+

Latest revision as of 14:53, 22 February 2018

Contents

Creating a Standalone Outgoing Call Leg

   PTRCTBCMC_CALL_LEG_ATTRIBUTE ptrOutgoingLegAttribute;
   ptrOutgoingLegAttribute = tbnew CTBCMC_CALL_LEG_ATTRIBUTE();
 
   ptrOutgoingLegAttribute->GetCalledNumber()         = "123-4567";
   ptrOutgoingLegAttribute->GetCallingNumber()        = "987-6543";
   ptrOutgoingLegAttribute->GetNetworkAccessPoint()   = "NAP_SS7_MONTREAL";
 
   PCTBCMCLeg pCallLeg = tbnew CTBCMCLeg( ++mun32LegId, ptrOutgoingLegAttribute, this, 0, &mLegMutex );
   pCallLeg->CreateCall();

Bridging an Incoming Call Leg (using CTBCAFBridge )

TBX_VOID CTBS2GWSimpleCall::OnCallLegPresent
(
  IN	TBCMC_LEG_ID						in_LegId, 
  IN	CTBCMC_CALL_LEG_ATTRIBUTE_COMMON &	in_CallLegAttribute, 
  IN	CTBCMC_PROTOCOL_ATTRIBUTE &			in_ProtocolAttribute
)
{
	TBX_RESULT								Result;
	PITBCAFCallFlow			 				pBridge;
	PTRCTBCMC_CALL_LEG_ATTRIBUTE			ptrIncomingLegAttribute;
	PTRCTBCMC_PROTOCOL_ATTRIBUTE			ptrIncomingLegProtocolAttributes;
	PTRCTBCMC_PROTOCOL_ATTRIBUTE			ptrAcceptCallProtAttribute;
	PTRCTBCMC_CALL_LEG_ATTRIBUTE			ptrOutgoingLegAttribute;
	PTRCTBCMC_PROTOCOL_ATTRIBUTE			ptrOutgoingLegProtocolAttributes;


	/*---------------------------------------------------------------------------------------------------------------------------
	 |  Code section
	 *--------------------------------------------------------------------------------------------------------------------------*/
	CAFCODE( CTBS2GWSimpleCall::OnCallLegPresent )
	{
		// Create smart pointers to incoming call leg attributes so behaviors can keep reference to it if needed
		ptrIncomingLegAttribute				= tbnew CTBCMC_CALL_LEG_ATTRIBUTE( *(in_CallLegAttribute.GetCallLegAttribute()) );
		ptrIncomingLegProtocolAttributes	= tbnew CTBCMC_PROTOCOL_ATTRIBUTE( in_ProtocolAttribute, TBX_TRUE );

		// Build default protocol attributes object so behaviors can modify
		// attributes used to Accept the incoming call leg
		ptrAcceptCallProtAttribute			= tbnew CTBCMC_PROTOCOL_ATTRIBUTE();

		// Build default protocol attributes object so behaviors can modify 
		// attributes used to create the outgoing call leg
		ptrOutgoingLegProtocolAttributes	= tbnew CTBCMC_PROTOCOL_ATTRIBUTE();

		// Create an empty outgoing call leg attribute object
		ptrOutgoingLegAttribute		= tbnew CTBCMC_CALL_LEG_ATTRIBUTE();

		// Fill some information in the outgoing call leg attributes from the incoming leg attributes
		// *** Warning: Don't use operator=. We dont wan't to copy EVERYTHING, but only information
		//				common to both incoming and outgoing legs, such as calling/called number.
		//				Function CopyLegAttributeFrom() takes care of copying relevant information.
		ptrOutgoingLegAttribute->CopyLegAttributeFrom( *(ptrIncomingLegAttribute.Get()) );

		// Create the call (i.e. which contains 2 call legs)
		pBridge = tbnew CTBCAFBridge( 0, this );
		TBCAF_EXIT_ON_NULL( pBridge, TBX_RESULT_FAIL, "Failed to allocate call." );

		/* Prepare the chain of behaviors */
		PITBCAFCallFlow pPreviousBehavior = pBridge;

		/* Add the example behavior to the chain of behaviors */
		pPreviousBehavior = tbnew CTBS2GWCallBehaviorExample( pPreviousBehavior );

		// Add legs information
		Result = pBridge->AddIncoming( in_LegId, ptrIncomingLegAttribute, ptrIncomingLegProtocolAttributes, ptrAcceptCallProtAttribute );
		TBCAF_EXIT_ON_ERROR(Result, "AddIncoming Failed." );
		
		// Provide an uninitialized PTRCTBCMC_PROTOCOL_ATTRIBUTE
		Result = pBridge->AddOutgoing( ptrOutgoingLegAttribute, ptrOutgoingLegProtocolAttributes );
		TBCAF_EXIT_ON_ERROR(Result, "AddOutgoing Failed." );


		Result = pBridge->InitCall( &pBridge );
		if( pBridge == NULL )
		{
			/* Note: If this return value is NULL, this means the InitCall function
					 has initialized no call leg's state machine (failed to) and
					 destroyed itself immediately (it normally destroys itself later
					 when the state machine of all call legs has terminated).
					 The above line of code shall thus be the last of the function
					 to avoid accessing any member of "this" here. */
			TBX_EXIT_CLEANUP( Result );
		}
		TBCAF_EXIT_ON_ERROR(Result, "InitCall Failed." );

		TBX_EXIT_SUCCESS( TBX_RESULT_OK );
	}

	/*---------------------------------------------------------------------------------------------------------------------------
	 |  Exception handling section
	 *--------------------------------------------------------------------------------------------------------------------------*/
	CAF_EXCEPTION_HANDLING

	/*---------------------------------------------------------------------------------------------------------------------------
	 |  Error handling section
	 *--------------------------------------------------------------------------------------------------------------------------*/
	CAF_ERROR_HANDLING( CAF_VERBOSE )
	{
		if( pBridge == NULL )
		{
			// Try to refuse the call leg
			CTBCMCLeg::RefuseLeg( in_LegId, in_CallLegAttribute );
		}
	}

	/*---------------------------------------------------------------------------------------------------------------------------
	 |  Cleanup section
	 *--------------------------------------------------------------------------------------------------------------------------*/
	CAF_CLEANUP
	{
	}

	RETURN_VOID;
}


InitCall() will create each leg added from AddIncoming() and AddOutgoing() and incoming and outgoing call leg will be joined when we receive the notification OnCallLegAnswered() indicating that the outgoing call has been answered by the remote peer.


Pitfall

It's mandatory that the call leg and protocol attribute objects passed to AddIncoming() and AddOutgoing() remain valid until function InitCall() returns.

Creating a Media-only Leg

If you are using a nap of type NAP_MEDIA_TDM:

   PTRCTBCMC_MEDIA_ONLY_LEG_ATTRIBUTE ptrLegAttribute;
   PTBCMC_MEDIA_DESCRIPTION           pMediaDesc;
 
   ptrLegAttribute = tbnew CTBCMC_MEDIA_ONLY_LEG_ATTRIBUTE();
   pMediaDesc	  = &ptrLegAttribute->GetProfile().GetMediaDescription(); 

   ptrLegAttribute->GetNetworkAccessPoint()      = "NAP_MEDIA_TDM";
   pMediaDesc->Type                              = TBCMC_MEDIA_TYPE_AUDIO;
   pMediaDesc->Transport                         = TBCMC_MEDIA_TRANSPORT_TDM;
   pMediaDesc->Settings.TdmAudio.Type            = TBCMC_MEDIA_SETTINGS_TYPE_TDM_AUDIO;
   pMediaDesc->Settings.TdmAudio.un8Timeslot     = 5;
   Strncpy
   (
       pMediaDesc->Settings.TdmAudio.szTrunkName, 
       "TRUNK_TORONTO_1", 
       sizeof(pMediaDesc->Settings.TdmAudio.szTrunkName)
   );
 
   PCTBCMCLeg pCallLeg = tbnew CTBCMCLeg( ++mun32LegId, ptrLegAttribute, this, 0, &mLegMutex );
   pCallLeg->CreateCall();

If you are using a nap of type NAP_MEDIA_VOIP:

   TBX_RESULT                         Result;
   TBX_SDP_INFO                       SdpInfo;
   PTRCTBCMC_MEDIA_ONLY_LEG_ATTRIBUTE ptrLegAttribute;
   PTBCMC_MEDIA_DESCRIPTION           pMediaDesc;
 
   ptrLegAttribute = tbnew CTBCMC_MEDIA_ONLY_LEG_ATTRIBUTE();
   pMediaDesc	   = &ptrLegAttribute->GetProfile().GetMediaDescription();
 
   ptrLegAttribute->GetNetworkAccessPoint()      = "NAP_MEDIA_VOIP";
   pMediaDesc->Type                              = TBCMC_MEDIA_TYPE_AUDIO;
   pMediaDesc->Transport                         = TBCMC_MEDIA_TRANSPORT_IP;
   pMediaDesc->Settings.PacketAudio.Type         = TBCMC_MEDIA_SETTINGS_TYPE_PACKET_AUDIO;
  
   // Set Local SDP
   Result = BuildSdpInfo("", 0, SdpInfo);                 // No IP specified, Toolpack will choose one
   TBCAF_EXIT_ON_ERROR( Result, "BuildSdpInfo failed." );
   ptrLegAttribute->GetProfile().SetLocalSDP(SdpInfo);
   
   // Set Peer SDP
   Result = BuildSdpInfo("10.0.0.15", 5000, SdpInfo);     // Using peer IP address and port
   TBCAF_EXIT_ON_ERROR( Result, "BuildSdpInfo failed." );
   ptrLegAttribute->GetProfile().SetPeerSDP(SdpInfo);
 
   PCTBCMCLeg pCallLeg = tbnew CTBCMCLeg( ++mun32LegId, ptrLegAttribute, this, 0, &mLegMutex );
   pCallLeg->CreateCall();

Note that you have two possibilities to build a TBX_SDP_INFO structure. The first one is like in the example (BuildSdpInfo()) where you can build manually your TBX_SDP_INFO structure and the second one is when you have an sdp info in text format. There is a function that will parse the text and build a TBX_SDP_INFO structure from it.

 TBXMediaLibSdpParse
 (
     IN    PTBX_CHAR                        in_pszSdp,
     OUT   PTBX_SDP_INFO                    out_pSdpInfo,
     IN    TBX_MEDIA_SDP_PARSE_OPTIONS      in_Options
 );

To see how to build a TBX_SDP_INFO structure, refer to Filling an SDP Structure.

Personal tools