|
|
/****************************************************************************
* * $Archive: S:/STURGEON/SRC/CALLCONT/VCS/h245man.c_v $ * * INTEL Corporation Prorietary Information * * This listing is supplied under the terms of a license agreement * with INTEL Corporation and may not be copied nor disclosed except * in accordance with the terms of that agreement. * * Copyright (c) 1993-1994 Intel Corporation. * * $Revision: 1.225 $ * $Date: 03 Mar 1997 09:08:10 $ * $Author: MANDREWS $ * * Deliverable: * * Abstract: * * * Notes: * ***************************************************************************/
#include "precomp.h"
#include "apierror.h"
#include "incommon.h"
#include "callcont.h"
#include "q931.h"
#include "ccmain.h"
#include "listman.h"
#include "q931man.h"
#include "userman.h"
#include "callman.h"
#include "confman.h"
#include "h245man.h"
#include "chanman.h"
#include "hangman.h"
#include "ccutils.h"
#include "linkapi.h"
#include "h245com.h"
extern CALL_CONTROL_STATE CallControlState; extern THREADCOUNT ThreadCount;
static BOOL bH245ManagerInited = FALSE;
static struct { DWORD dwPhysicalID; LOCK Lock; } PhysicalID;
HRESULT InitH245Manager() { ASSERT(bH245ManagerInited == FALSE);
// Note -- don't use a physical ID of 0; the physical ID gets mapped
// to an H245 instance of the same value, and an H245 instance of
// 0 is invalid
PhysicalID.dwPhysicalID = 1; InitializeLock(&PhysicalID.Lock); bH245ManagerInited = H245SysInit(); return CC_OK; }
HRESULT DeInitH245Manager() { if (bH245ManagerInited == FALSE) return CC_OK;
H245SysDeInit(); H245WSShutdown(); DeleteLock(&PhysicalID.Lock); bH245ManagerInited = FALSE; return CC_OK; }
HRESULT MakeH245PhysicalID( DWORD *pdwH245PhysicalID) { AcquireLock(&PhysicalID.Lock); *pdwH245PhysicalID = PhysicalID.dwPhysicalID++; RelinquishLock(&PhysicalID.Lock); return CC_OK; }
HRESULT _ConstructTermCapList( PCC_TERMCAPLIST *ppTermCapList, PCC_TERMCAP *ppH2250MuxCap, PCC_TERMCAPDESCRIPTORS *ppTermCapDescriptors, PCALL pCall) { #define MAX_TERM_CAPS 257
#define MAX_TERM_CAP_DESC 255
H245_TOTCAP_T * pTermCapArray[MAX_TERM_CAPS]; H245_TOTCAPDESC_T * pTermCapDescriptorArray[MAX_TERM_CAP_DESC]; unsigned long CapArrayLength; unsigned long CapDescriptorArrayLength; unsigned long i, j; HRESULT status;
ASSERT(ppTermCapList != NULL); ASSERT(*ppTermCapList == NULL); ASSERT(ppH2250MuxCap != NULL); ASSERT(*ppH2250MuxCap == NULL); ASSERT(ppTermCapDescriptors != NULL); ASSERT(*ppTermCapDescriptors == NULL); ASSERT(pCall != NULL);
CapArrayLength = MAX_TERM_CAPS; CapDescriptorArrayLength = MAX_TERM_CAP_DESC;
status = H245GetCaps(pCall->H245Instance, H245_CAPDIR_RMTRXTX, H245_DATA_DONTCARE, H245_CLIENT_DONTCARE, pTermCapArray, &CapArrayLength, pTermCapDescriptorArray, &CapDescriptorArrayLength); if (status != H245_ERROR_OK) { *ppTermCapList = NULL; *ppH2250MuxCap = NULL; *ppTermCapDescriptors = NULL; return status; }
// Check the term cap list to see if an H.225.0 mux capability is present;
// this capability is treated as a special case
*ppH2250MuxCap = NULL; for (i = 0; i < CapArrayLength; i++) { ASSERT(pTermCapArray[i] != NULL); if (pTermCapArray[i]->CapId == 0) { *ppH2250MuxCap = pTermCapArray[i]; --CapArrayLength; for (j = i; j < CapArrayLength; j++) pTermCapArray[j] = pTermCapArray[j+1]; break; } }
if (CapArrayLength == 0) *ppTermCapList = NULL; else { *ppTermCapList = (PCC_TERMCAPLIST)MemAlloc(sizeof(CC_TERMCAPLIST)); if (*ppTermCapList == NULL) { for (i = 0; i < CapArrayLength; i++) H245FreeCap(pTermCapArray[i]); if (*ppH2250MuxCap != NULL) H245FreeCap(*ppH2250MuxCap); for (i = 0; i < CapDescriptorArrayLength; i++) H245FreeCapDescriptor(pTermCapDescriptorArray[i]); return CC_NO_MEMORY; }
(*ppTermCapList)->wLength = (WORD)CapArrayLength; (*ppTermCapList)->pTermCapArray = (H245_TOTCAP_T **)MemAlloc(sizeof(H245_TOTCAP_T *) * CapArrayLength); if ((*ppTermCapList)->pTermCapArray == NULL) { MemFree(*ppTermCapList); for (i = 0; i < CapArrayLength; i++) H245FreeCap(pTermCapArray[i]); if (*ppH2250MuxCap != NULL) H245FreeCap(*ppH2250MuxCap); for (i = 0; i < CapDescriptorArrayLength; i++) H245FreeCapDescriptor(pTermCapDescriptorArray[i]); *ppTermCapList = NULL; *ppH2250MuxCap = NULL; *ppTermCapDescriptors = NULL; return CC_NO_MEMORY; }
for (i = 0; i < CapArrayLength; i++) (*ppTermCapList)->pTermCapArray[i] = pTermCapArray[i]; }
if (CapDescriptorArrayLength == 0) *ppTermCapDescriptors = NULL; else { *ppTermCapDescriptors = (PCC_TERMCAPDESCRIPTORS)MemAlloc(sizeof(CC_TERMCAPDESCRIPTORS)); if (*ppTermCapDescriptors == NULL) { for (i = 0; i < CapArrayLength; i++) H245FreeCap(pTermCapArray[i]); if (*ppH2250MuxCap != NULL) H245FreeCap(*ppH2250MuxCap); for (i = 0; i < CapDescriptorArrayLength; i++) H245FreeCapDescriptor(pTermCapDescriptorArray[i]); if (*ppTermCapList != NULL) { MemFree((*ppTermCapList)->pTermCapArray); MemFree(*ppTermCapList); } *ppTermCapList = NULL; *ppH2250MuxCap = NULL; *ppTermCapDescriptors = NULL; return CC_NO_MEMORY; }
(*ppTermCapDescriptors)->wLength = (WORD)CapDescriptorArrayLength; (*ppTermCapDescriptors)->pTermCapDescriptorArray = (H245_TOTCAPDESC_T **)MemAlloc(sizeof(H245_TOTCAPDESC_T *) * CapDescriptorArrayLength); if ((*ppTermCapDescriptors)->pTermCapDescriptorArray == NULL) { for (i = 0; i < CapArrayLength; i++) H245FreeCap(pTermCapArray[i]); if (*ppH2250MuxCap != NULL) H245FreeCap(*ppH2250MuxCap); for (i = 0; i < CapDescriptorArrayLength; i++) H245FreeCapDescriptor(pTermCapDescriptorArray[i]); if (*ppTermCapList != NULL) { MemFree((*ppTermCapList)->pTermCapArray); MemFree(*ppTermCapList); } MemFree(*ppTermCapDescriptors); *ppTermCapList = NULL; *ppH2250MuxCap = NULL; *ppTermCapDescriptors = NULL; return CC_NO_MEMORY; }
for (i = 0; i < CapDescriptorArrayLength; i++) (*ppTermCapDescriptors)->pTermCapDescriptorArray[i] = pTermCapDescriptorArray[i]; } return CC_OK; }
HRESULT _ProcessConnectionComplete( PCONFERENCE pConference, PCALL pCall) { CC_HCONFERENCE hConference; CC_HCALL hCall; HQ931CALL hQ931Call; HQ931CALL hQ931CallInvitor; HRESULT status; CC_CONNECT_CALLBACK_PARAMS ConnectCallbackParams; CC_MULTIPOINT_CALLBACK_PARAMS MultipointCallbackParams; CC_PEER_CHANGE_CAP_CALLBACK_PARAMS PeerChangeCapCallbackParams; CC_PEER_ADD_CALLBACK_PARAMS PeerAddCallbackParams; WORD i; BOOL bMultipointConference; H245_TRANSPORT_ADDRESS_T Q931Address; PDU_T Pdu; CALLTYPE CallType; WORD wNumCalls; PCC_HCALL CallList; WORD wNumChannels; PCC_HCHANNEL ChannelList; PCHANNEL pChannel; PCALL pOldCall; CC_HCALL hOldCall; BYTE bNewTerminalNumber; BYTE bNewMCUNumber; CC_ENDPOINTTYPE DestinationEndpointType; H245_COMM_MODE_ENTRY_T *pH245CommunicationTable; BYTE bCommunicationTableCount; BOOL bSessionTableChanged; CONFMODE PreviousConferenceMode; CC_ADDR MCAddress; BOOL bConferenceTermCapsChanged; H245_INST_T H245Instance; PCC_TERMCAP pTxTermCap; PCC_TERMCAP pRxTermCap; H245_MUX_T *pTxMuxTable; H245_MUX_T *pRxMuxTable;
// caution: the size of PDU_T is ~70K because of the size of
// OpenLogicalChannel, because of struct EncryptionSync
// If there is a way to tweak the ASN to make this a pointer,
// then it needs to be done
ASSERT(pConference != NULL); ASSERT(pCall != NULL); ASSERT(pCall->hConference == pConference->hConference); hConference = pConference->hConference; hCall = pCall->hCall; hQ931Call = pCall->hQ931Call; hQ931CallInvitor = pCall->hQ931CallInvitor; H245Instance = pCall->H245Instance; CallType = pCall->CallType;
// Note that pConference->ConferenceMode refers to the conference mode BEFORE
// this connection attempt completes. If the current conference mode is
// point-to-point, this connection (if successful) will result in a multipoint
// conference. We want to reflect in the CONNECT callback the connection mode
// that would exist if the connect attempt is successful.
if ((pConference->ConferenceMode == POINT_TO_POINT_MODE) || (pConference->ConferenceMode == MULTIPOINT_MODE) || (pCall->bCallerIsMC)) bMultipointConference = TRUE; else bMultipointConference = FALSE;
// Initialize all fields of ConnectCallbackParams now
ConnectCallbackParams.pNonStandardData = pCall->pPeerNonStandardData; ConnectCallbackParams.pszPeerDisplay = pCall->pszPeerDisplay; ConnectCallbackParams.bRejectReason = CC_REJECT_UNDEFINED_REASON; ConnectCallbackParams.pTermCapList = pCall->pPeerH245TermCapList; ConnectCallbackParams.pH2250MuxCapability = pCall->pPeerH245H2250MuxCapability; ConnectCallbackParams.pTermCapDescriptors = pCall->pPeerH245TermCapDescriptors; ConnectCallbackParams.pLocalAddr = pCall->pQ931LocalConnectAddr; if (pCall->pQ931DestinationAddr == NULL) ConnectCallbackParams.pPeerAddr = pCall->pQ931PeerConnectAddr; else ConnectCallbackParams.pPeerAddr = pCall->pQ931DestinationAddr; ConnectCallbackParams.pVendorInfo = pCall->pPeerVendorInfo; ConnectCallbackParams.bMultipointConference = bMultipointConference; ConnectCallbackParams.pConferenceID = &pConference->ConferenceID; ConnectCallbackParams.pMCAddress = pConference->pMultipointControllerAddr; ConnectCallbackParams.pAlternateAddress = NULL; ConnectCallbackParams.dwUserToken = pCall->dwUserToken;
status = AddEstablishedCallToConference(pCall, pConference); if (status != CC_OK) { MarkCallForDeletion(pCall);
if (CallType == THIRD_PARTY_INTERMEDIARY) Q931RejectCall(hQ931CallInvitor, CC_REJECT_UNDEFINED_REASON, &pCall->ConferenceID, NULL, // alternate address
pCall->pPeerNonStandardData);
if ((CallType == CALLER) || (CallType == THIRD_PARTY_INVITOR) || ((CallType == CALLEE) && (pConference->LocalEndpointAttached == NEVER_ATTACHED))) InvokeUserConferenceCallback(pConference, CC_CONNECT_INDICATION, status, &ConnectCallbackParams);
if (ValidateCallMarkedForDeletion(hCall) == CC_OK) FreeCall(pCall); H245ShutDown(H245Instance); Q931Hangup(hQ931Call, CC_REJECT_UNDEFINED_REASON); if (ValidateConference(hConference) == CC_OK) UnlockConference(pConference);
return status; }
if (((pConference->ConferenceMode == POINT_TO_POINT_MODE) || (pConference->ConferenceMode == MULTIPOINT_MODE)) && (pConference->tsMultipointController == TS_TRUE)) status = CreateConferenceTermCaps(pConference, &bConferenceTermCapsChanged); else { status = CC_OK; bConferenceTermCapsChanged = FALSE; } if (status != CC_OK) { MarkCallForDeletion(pCall);
if (CallType == THIRD_PARTY_INTERMEDIARY) Q931RejectCall(pCall->hQ931CallInvitor, CC_REJECT_UNDEFINED_REASON, &pCall->ConferenceID, NULL, // alternate address
pCall->pPeerNonStandardData);
if ((CallType == CALLER) || (CallType == THIRD_PARTY_INVITOR) || ((CallType == CALLEE) && (pConference->LocalEndpointAttached == NEVER_ATTACHED))) InvokeUserConferenceCallback(pConference, CC_CONNECT_INDICATION, status, &ConnectCallbackParams);
if (ValidateCallMarkedForDeletion(hCall) == CC_OK) FreeCall(pCall); H245ShutDown(H245Instance); Q931Hangup(hQ931Call, CC_REJECT_UNDEFINED_REASON); if (ValidateConference(hConference) == CC_OK) UnlockConference(pConference);
return status; }
if (pConference->tsMultipointController == TS_TRUE) { // Send MCLocationIndication
status = GetLastListenAddress(&MCAddress); if (status == CC_OK) { ASSERT(MCAddress.nAddrType == CC_IP_BINARY); Q931Address.type = H245_IP_UNICAST; Q931Address.u.ip.tsapIdentifier = MCAddress.Addr.IP_Binary.wPort; HostToH245IPNetwork(Q931Address.u.ip.network, MCAddress.Addr.IP_Binary.dwAddr); H245MCLocationIndication(pCall->H245Instance, &Q931Address); } }
EnumerateCallsInConference(&wNumCalls, &CallList, pConference, ESTABLISHED_CALL);
if (pConference->ConferenceMode == UNCONNECTED_MODE) { ASSERT(pConference->pSessionTable == NULL); ASSERT(wNumCalls == 1);
pConference->ConferenceMode = POINT_TO_POINT_MODE; } else { // we're currently in point-to-point mode or multipoint mode
if (pConference->tsMultipointController == TS_TRUE) { PreviousConferenceMode = pConference->ConferenceMode; pConference->ConferenceMode = MULTIPOINT_MODE;
// In the future, we may want to construct a new session table
// each time a new peer is added to the conference
if (PreviousConferenceMode == POINT_TO_POINT_MODE) { // Assign a terminal label to ourselves
// Note that we reserve a terminal number of 0 for ourselves
// if we're the MC
ASSERT(pConference->LocalParticipantInfo.ParticipantInfo.TerminalLabel.bTerminalNumber == 255); pConference->LocalParticipantInfo.ParticipantInfo.TerminalLabel.bTerminalNumber = 0;
// Create a new session table
CreateConferenceSessionTable( pConference, &bSessionTableChanged); } else // For the current implementation, don't cause a new
// CommunicationModeCommand to be issued when a new peer is added
// unless we're switching from point-to-point to multipoint mode
// (in which case bSessionTableChanged is ignored)
bSessionTableChanged = FALSE;
if (bSessionTableChanged) SessionTableToH245CommunicationTable(pConference->pSessionTable, &pH245CommunicationTable, &bCommunicationTableCount); else pH245CommunicationTable = NULL;
// Send MultipointModeCommand to new call
Pdu.choice = MSCMg_cmmnd_chosen; Pdu.u.MSCMg_cmmnd.choice = miscellaneousCommand_chosen; // logical channel number is irrelavent but needs to be filled in
Pdu.u.MSCMg_cmmnd.u.miscellaneousCommand.logicalChannelNumber = 1; Pdu.u.MSCMg_cmmnd.u.miscellaneousCommand.type.choice = multipointModeCommand_chosen; H245SendPDU(pCall->H245Instance, &Pdu);
status = AllocatePeerParticipantInfo(pConference, &pCall->pPeerParticipantInfo); if (status == CC_OK) { bNewMCUNumber = pCall->pPeerParticipantInfo->ParticipantInfo.TerminalLabel.bMCUNumber; bNewTerminalNumber = pCall->pPeerParticipantInfo->ParticipantInfo.TerminalLabel.bTerminalNumber; // Send TerminalNumberAssign to new call
H245ConferenceIndication(pCall->H245Instance, H245_IND_TERMINAL_NUMBER_ASSIGN,// Indication Type
0, // SBE number; ignored here
pCall->pPeerParticipantInfo->ParticipantInfo.TerminalLabel.bMCUNumber, // MCU number
pCall->pPeerParticipantInfo->ParticipantInfo.TerminalLabel.bTerminalNumber); // terminal number
// Send EnterH243TerminalID to new call
H245ConferenceRequest(pCall->H245Instance, H245_REQ_ENTER_H243_TERMINAL_ID, pCall->pPeerParticipantInfo->ParticipantInfo.TerminalLabel.bMCUNumber, pCall->pPeerParticipantInfo->ParticipantInfo.TerminalLabel.bTerminalNumber); pCall->pPeerParticipantInfo->TerminalIDState = TERMINAL_ID_REQUESTED; } else { // Unable to assign a terminal number to the new call
bNewMCUNumber = 0; bNewTerminalNumber = 0; }
if (pH245CommunicationTable != NULL) { // Send CommunicationModeCommand to new call
status = H245CommunicationModeCommand(pCall->H245Instance, pH245CommunicationTable, bCommunicationTableCount); }
if (PreviousConferenceMode == POINT_TO_POINT_MODE) { // Generate MULTIPOINT callback
MultipointCallbackParams.pTerminalInfo = &pConference->LocalParticipantInfo.ParticipantInfo; MultipointCallbackParams.pSessionTable = pConference->pSessionTable; InvokeUserConferenceCallback(pConference, CC_MULTIPOINT_INDICATION, CC_OK, &MultipointCallbackParams); if (ValidateConference(hConference) != CC_OK) { if (ValidateCall(hCall) == CC_OK) { pCall->CallState = CALL_COMPLETE; UnlockCall(pCall); } MemFree(CallList); return CC_OK; }
// Generate CC_PEER_CHANGE_CAP callback
PeerChangeCapCallbackParams.pTermCapList = pConference->pConferenceTermCapList; PeerChangeCapCallbackParams.pH2250MuxCapability = pConference->pConferenceH245H2250MuxCapability; PeerChangeCapCallbackParams.pTermCapDescriptors = pConference->pConferenceTermCapDescriptors; InvokeUserConferenceCallback(pConference, CC_PEER_CHANGE_CAP_INDICATION, CC_OK, &PeerChangeCapCallbackParams); if (ValidateConference(hConference) != CC_OK) { if (ValidateCall(hCall) == CC_OK) { pCall->CallState = CALL_COMPLETE; UnlockCall(pCall); } MemFree(CallList); return CC_OK; }
ASSERT(wNumCalls == 2); // one existing call and new call
if (CallList[0] == hCall) hOldCall = CallList[1]; else hOldCall = CallList[0];
if (LockCall(hOldCall, &pOldCall) == CC_OK) { // Send MultipointModeCommand to old call
Pdu.choice = MSCMg_cmmnd_chosen; Pdu.u.MSCMg_cmmnd.choice = miscellaneousCommand_chosen; // logical channel number is irrelavent but needs to be filled in
Pdu.u.MSCMg_cmmnd.u.miscellaneousCommand.logicalChannelNumber = 1; Pdu.u.MSCMg_cmmnd.u.miscellaneousCommand.type.choice = multipointModeCommand_chosen; H245SendPDU(pOldCall->H245Instance, &Pdu);
status = AllocatePeerParticipantInfo(pConference, &pOldCall->pPeerParticipantInfo); if (status == CC_OK) { // Send TerminalNumberAssign to old call
H245ConferenceIndication(pOldCall->H245Instance, H245_IND_TERMINAL_NUMBER_ASSIGN,// Indication Type
0, // SBE number; ignored here
pOldCall->pPeerParticipantInfo->ParticipantInfo.TerminalLabel.bMCUNumber, // MCU number
pOldCall->pPeerParticipantInfo->ParticipantInfo.TerminalLabel.bTerminalNumber); // terminal number
// Send EnterH243TerminalID to old call
H245ConferenceRequest(pOldCall->H245Instance, H245_REQ_ENTER_H243_TERMINAL_ID, pOldCall->pPeerParticipantInfo->ParticipantInfo.TerminalLabel.bMCUNumber, pOldCall->pPeerParticipantInfo->ParticipantInfo.TerminalLabel.bTerminalNumber); pOldCall->pPeerParticipantInfo->TerminalIDState = TERMINAL_ID_REQUESTED; }
if (pH245CommunicationTable != NULL) { // Send CommunicationModeCommand to old call
status = H245CommunicationModeCommand(pOldCall->H245Instance, pH245CommunicationTable, bCommunicationTableCount); FreeH245CommunicationTable(pH245CommunicationTable, bCommunicationTableCount); }
// Send TerminalJoinedConference (this call) to old call
H245ConferenceIndication(pOldCall->H245Instance, H245_IND_TERMINAL_JOINED, // Indication Type
0, // SBE number; ignored here
pConference->LocalParticipantInfo.ParticipantInfo.TerminalLabel.bMCUNumber, // MCU number
pConference->LocalParticipantInfo.ParticipantInfo.TerminalLabel.bTerminalNumber); // terminal number // terminal number of MC
if (bNewTerminalNumber != 0) { // Send TerminalJoinedConference (new call) to old call
H245ConferenceIndication(pOldCall->H245Instance, H245_IND_TERMINAL_JOINED, // Indication Type
0, // SBE number; ignored here
bNewMCUNumber, // MCU number
bNewTerminalNumber); // terminal number
// Generate PEER_ADD callback for old call
PeerAddCallbackParams.hCall = pOldCall->hCall; PeerAddCallbackParams.TerminalLabel = pOldCall->pPeerParticipantInfo->ParticipantInfo.TerminalLabel; PeerAddCallbackParams.pPeerTerminalID = NULL; InvokeUserConferenceCallback(pConference, CC_PEER_ADD_INDICATION, CC_OK, &PeerAddCallbackParams); if (ValidateConference(hConference) != CC_OK) { if (ValidateCall(hOldCall) == CC_OK) UnlockCall(pCall); if (ValidateCall(hCall) == CC_OK) { pCall->CallState = CALL_COMPLETE; UnlockCall(pCall); } MemFree(CallList); return CC_OK; } }
// Send new term caps to old call
SendTermCaps(pOldCall, pConference);
UnlockCall(pOldCall); } } else { // we're currently in multipoint mode
EnumerateChannelsInConference(&wNumChannels, &ChannelList, pConference, TX_CHANNEL | PROXY_CHANNEL | TXRX_CHANNEL); for (i = 0; i < wNumChannels; i++) { if (LockChannel(ChannelList[i], &pChannel) == CC_OK) { if (pChannel->bMultipointChannel) { if ((pChannel->bChannelType == TX_CHANNEL) || ((pChannel->bChannelType == TXRX_CHANNEL) && (pChannel->bLocallyOpened == TRUE))) { pTxTermCap = pChannel->pTxH245TermCap; pTxMuxTable = pChannel->pTxMuxTable; pRxTermCap = pChannel->pRxH245TermCap; pRxMuxTable = pChannel->pRxMuxTable; } else { // Note: since this is a proxy or remotely-opened
// bi-directional channel, RxTermCap and RxMuxTable
// contain the channel's term cap and mux table,
// and must be sent to other endpoints as the
// Tx term cap and mux table;
// TxTermCap and TxMuxTable should be NULL
pTxTermCap = pChannel->pRxH245TermCap; pTxMuxTable = pChannel->pRxMuxTable; pRxTermCap = pChannel->pTxH245TermCap; pRxMuxTable = pChannel->pTxMuxTable; } status = H245OpenChannel( pCall->H245Instance, pChannel->hChannel, // dwTransId
pChannel->wLocalChannelNumber, pTxTermCap, // TxMode
pTxMuxTable, // TxMux
H245_INVALID_PORT_NUMBER, // TxPort
pRxTermCap, // RxMode
pRxMuxTable, // RxMux
pChannel->pSeparateStack); if ((status == CC_OK) && (pChannel->wNumOutstandingRequests != 0)) (pChannel->wNumOutstandingRequests)++; } UnlockChannel(pChannel); } } MemFree(ChannelList);
for (i = 0; i < wNumCalls; i++) { // Don't send a message to the endpoint that just joined the conference!
if (CallList[i] != hCall) { if (LockCall(CallList[i], &pOldCall) == CC_OK) { if (bNewTerminalNumber != 0) // Send TerminalJoinedConference (new call) to old call
H245ConferenceIndication(pOldCall->H245Instance, H245_IND_TERMINAL_JOINED, // Indication Type
0, // SBE number; ignored here
bNewMCUNumber, // MCU number
bNewTerminalNumber); // terminal number
// Send CommunicationModeCommand, if necessary
if (pH245CommunicationTable != NULL) status = H245CommunicationModeCommand(pOldCall->H245Instance, pH245CommunicationTable, bCommunicationTableCount); if (bConferenceTermCapsChanged) // Send new term caps
SendTermCaps(pOldCall, pConference);
UnlockCall(pOldCall); } } } if (bConferenceTermCapsChanged) { // Generate CC_PEER_CHANGE_CAP callback
PeerChangeCapCallbackParams.pTermCapList = pConference->pConferenceTermCapList; PeerChangeCapCallbackParams.pH2250MuxCapability = pConference->pConferenceH245H2250MuxCapability; PeerChangeCapCallbackParams.pTermCapDescriptors = pConference->pConferenceTermCapDescriptors; InvokeUserConferenceCallback(pConference, CC_PEER_CHANGE_CAP_INDICATION, CC_OK, &PeerChangeCapCallbackParams); if (ValidateConference(hConference) != CC_OK) { if (ValidateCall(hCall) == CC_OK) { pCall->CallState = CALL_COMPLETE; UnlockCall(pCall); } MemFree(CallList); return CC_OK; } } }
// Generate PEER_ADD callback
PeerAddCallbackParams.hCall = pCall->hCall; PeerAddCallbackParams.TerminalLabel = pCall->pPeerParticipantInfo->ParticipantInfo.TerminalLabel; PeerAddCallbackParams.pPeerTerminalID = NULL; InvokeUserConferenceCallback(pConference, CC_PEER_ADD_INDICATION, CC_OK, &PeerAddCallbackParams); if (ValidateConference(hConference) != CC_OK) { if (ValidateCall(hCall) == CC_OK) { pCall->CallState = CALL_COMPLETE; UnlockCall(pCall); MemFree(CallList); return CC_OK; } }
if (CallType == THIRD_PARTY_INTERMEDIARY) { DestinationEndpointType.pVendorInfo = pCall->pPeerVendorInfo; DestinationEndpointType.bIsTerminal = TRUE; DestinationEndpointType.bIsGateway = FALSE;
status = Q931AcceptCall(pCall->hQ931CallInvitor, pCall->pszPeerDisplay, pCall->pPeerNonStandardData, &DestinationEndpointType, NULL, pCall->hCall); Q931Hangup(pCall->hQ931CallInvitor, CC_REJECT_NORMAL_CALL_CLEARING); } } // if (pConference->tsMultipointController == TS_TRUE)
}
MemFree(CallList);
if (ValidateConference(hConference) == CC_OK) if ((CallType == CALLER) || (CallType == THIRD_PARTY_INVITOR) || ((CallType == CALLEE) && (pConference->LocalEndpointAttached == NEVER_ATTACHED))) { // This CONNECT must apply to the local endpoint
pConference->LocalEndpointAttached = ATTACHED; InvokeUserConferenceCallback(pConference, CC_CONNECT_INDICATION, CC_OK, &ConnectCallbackParams); } // Need to validate the conference and call handles; the associated
// objects may have been deleted during user callback on this thread
if (ValidateConference(hConference) == CC_OK) UnlockConference(pConference);
if (ValidateCall(hCall) == CC_OK) { pCall->CallState = CALL_COMPLETE; UnlockCall(pCall); } return status; }
HRESULT _IndUnimplemented( H245_CONF_IND_T *pH245ConfIndData) { return H245_ERROR_NOSUP; }
HRESULT _IndFlowControl( H245_CONF_IND_T *pH245ConfIndData) { HRESULT status; CC_HCALL hCall; PCALL pCall; PCONFERENCE pConference; CC_HCONFERENCE hConference; CC_HCHANNEL hChannel; PCHANNEL pChannel; CC_FLOW_CONTROL_CALLBACK_PARAMS FlowControlCallbackParams;
if (pH245ConfIndData->u.Indication.u.IndFlowControl.Scope != H245_SCOPE_CHANNEL_NUMBER) return H245_ERROR_NOSUP;
hCall = pH245ConfIndData->u.Indication.dwPreserved; status = LockCallAndConference(hCall, &pCall, &pConference); if (status != CC_OK) { // This may be OK, if the call was cancelled while
// call setup was in progress.
return H245_ERROR_OK; }
hConference = pCall->hConference; UnlockCall(pCall);
if (FindChannelInConference(pH245ConfIndData->u.Indication.u.IndFlowControl.Channel, TRUE, // local channel number
TX_CHANNEL | PROXY_CHANNEL, CC_INVALID_HANDLE, &hChannel, pConference) != CC_OK) { UnlockConference(pConference); return H245_ERROR_OK; }
if (LockChannel(hChannel, &pChannel) != CC_OK) { UnlockConference(pConference); return H245_ERROR_OK; } if (pChannel->bChannelType == TX_CHANNEL) { UnlockChannel(pChannel); FlowControlCallbackParams.hChannel = hChannel; FlowControlCallbackParams.dwRate = pH245ConfIndData->u.Indication.u.IndFlowControl.dwRestriction; InvokeUserConferenceCallback(pConference, CC_FLOW_CONTROL_INDICATION, CC_OK, &FlowControlCallbackParams); if (ValidateConference(hConference) == CC_OK) UnlockConference(pConference); } else { // pChannel->bChannelType == PROXY_CHANNEL
if (LockCall(pChannel->hCall, &pCall) == CC_OK) { H245FlowControl(pCall->H245Instance, pH245ConfIndData->u.Indication.u.IndFlowControl.Scope, pChannel->wRemoteChannelNumber, pH245ConfIndData->u.Indication.u.IndFlowControl.wResourceID, pH245ConfIndData->u.Indication.u.IndFlowControl.dwRestriction); UnlockCall(pCall); } UnlockChannel(pChannel); UnlockConference(pConference); } return H245_ERROR_OK; }
HRESULT _IndEndSession( H245_CONF_IND_T *pH245ConfIndData) { CC_HCALL hCall;
hCall = pH245ConfIndData->u.Indication.dwPreserved; if (hCall != CC_INVALID_HANDLE) ProcessRemoteHangup(hCall, CC_INVALID_HANDLE, CC_REJECT_NORMAL_CALL_CLEARING); return H245_ERROR_OK; }
HRESULT _IndCapability( H245_CONF_IND_T *pH245ConfIndData) { CC_HCALL hCall; PCALL pCall; HRESULT status; PCONFERENCE pConference; CC_HCONFERENCE hConference; CC_PEER_CHANGE_CAP_CALLBACK_PARAMS PeerChangeCapCallbackParams; BOOL bConferenceTermCapsChanged; WORD wNumCalls; PCC_HCALL CallList; PCALL pOldCall; WORD i;
// We received a TerminalCapabilitySet message from a peer
hCall = pH245ConfIndData->u.Indication.dwPreserved; status = LockCallAndConference(hCall, &pCall, &pConference); if (status != CC_OK) { // This may be OK, if the call was cancelled while
// call setup was in progress.
return H245_ERROR_OK; }
hConference = pCall->hConference;
if ((pConference->ConferenceMode == MULTIPOINT_MODE) && (pConference->tsMultipointController == TS_TRUE)) CreateConferenceTermCaps(pConference, &bConferenceTermCapsChanged); else bConferenceTermCapsChanged = FALSE;
pCall->bLinkEstablished = TRUE;
pCall->IncomingTermCapState = TERMCAP_COMPLETE;
if (pCall->CallState == TERMCAP) { ASSERT(pCall->pPeerH245TermCapList == NULL); ASSERT(pCall->pPeerH245H2250MuxCapability == NULL); ASSERT(pCall->pPeerH245TermCapDescriptors == NULL); } else { DestroyH245TermCapList(&pCall->pPeerH245TermCapList); DestroyH245TermCap(&pCall->pPeerH245H2250MuxCapability); DestroyH245TermCapDescriptors(&pCall->pPeerH245TermCapDescriptors); }
_ConstructTermCapList(&(pCall->pPeerH245TermCapList), &(pCall->pPeerH245H2250MuxCapability), &(pCall->pPeerH245TermCapDescriptors), pCall);
if ((pCall->OutgoingTermCapState == TERMCAP_COMPLETE) && (pCall->IncomingTermCapState == TERMCAP_COMPLETE) && (pCall->CallState == TERMCAP) && (pCall->MasterSlaveState == MASTER_SLAVE_COMPLETE)) { // Note that _ProcessConnectionComplete() returns with pConference and pCall unlocked
_ProcessConnectionComplete(pConference, pCall); return H245_ERROR_OK; }
if (pCall->CallState == CALL_COMPLETE) { if ((pConference->ConferenceMode == MULTIPOINT_MODE) && (pConference->tsMultipointController == TS_TRUE)) { CreateConferenceTermCaps(pConference, &bConferenceTermCapsChanged); if (bConferenceTermCapsChanged) { EnumerateCallsInConference(&wNumCalls, &CallList, pConference, ESTABLISHED_CALL); for (i = 0; i < wNumCalls; i++) { // Don't send a message to the endpoint that just joined the conference!
if (CallList[i] != hCall) { if (LockCall(CallList[i], &pOldCall) == CC_OK) { // Send new term caps
SendTermCaps(pOldCall, pConference); UnlockCall(pOldCall); } } } if (CallList != NULL) MemFree(CallList);
// Generate CC_PEER_CHANGE_CAP callback
PeerChangeCapCallbackParams.pTermCapList = pConference->pConferenceTermCapList; PeerChangeCapCallbackParams.pH2250MuxCapability = pConference->pConferenceH245H2250MuxCapability; PeerChangeCapCallbackParams.pTermCapDescriptors = pConference->pConferenceTermCapDescriptors; InvokeUserConferenceCallback(pConference, CC_PEER_CHANGE_CAP_INDICATION, CC_OK, &PeerChangeCapCallbackParams); } } else { PeerChangeCapCallbackParams.pTermCapList = pCall->pPeerH245TermCapList; PeerChangeCapCallbackParams.pH2250MuxCapability = pCall->pPeerH245H2250MuxCapability; PeerChangeCapCallbackParams.pTermCapDescriptors = pCall->pPeerH245TermCapDescriptors; InvokeUserConferenceCallback(pConference, CC_PEER_CHANGE_CAP_INDICATION, CC_OK, &PeerChangeCapCallbackParams); } if (ValidateConference(hConference) == CC_OK) UnlockConference(pConference); if (ValidateCall(hCall) == CC_OK) UnlockCall(pCall); return H245_ERROR_OK; }
UnlockCall(pCall); UnlockConference(pConference); return H245_ERROR_OK; }
HRESULT _IndOpenT120( H245_CONF_IND_T *pH245ConfIndData) { BOOL bFailed; CC_T120_CHANNEL_REQUEST_CALLBACK_PARAMS T120ChannelRequestCallbackParams; CC_HCALL hCall; PCALL pCall; CC_HCONFERENCE hConference; PCONFERENCE pConference; CC_HCHANNEL hChannel; PCHANNEL pChannel; CC_TERMCAP RxTermCap; CC_TERMCAP TxTermCap; H245_MUX_T RxH245MuxTable; H245_MUX_T TxH245MuxTable; CC_ADDR T120Addr; CC_OCTETSTRING ExternalReference;
hCall = pH245ConfIndData->u.Indication.dwPreserved; if (LockCallAndConference(hCall, &pCall, &pConference) != CC_OK) { // Can't cancel with H245, because we don't have the H245 instance
return H245_ERROR_OK; }
hConference = pCall->hConference; if (pH245ConfIndData->u.Indication.u.IndOpen.RxDataType != H245_DATA_DATA || pH245ConfIndData->u.Indication.u.IndOpen.RxClientType != H245_CLIENT_DAT_T120 || pH245ConfIndData->u.Indication.u.IndOpen.pRxCap == NULL || pH245ConfIndData->u.Indication.u.IndOpen.pRxCap->H245Dat_T120.application.choice != DACy_applctn_t120_chosen || pH245ConfIndData->u.Indication.u.IndOpen.pRxCap->H245Dat_T120.application.u.DACy_applctn_t120.choice != separateLANStack_chosen || pH245ConfIndData->u.Indication.u.IndOpen.TxDataType != H245_DATA_DATA || pH245ConfIndData->u.Indication.u.IndOpen.TxClientType != H245_CLIENT_DAT_T120 || pH245ConfIndData->u.Indication.u.IndOpen.pTxCap == NULL || pH245ConfIndData->u.Indication.u.IndOpen.pTxCap->H245Dat_T120.application.choice != DACy_applctn_t120_chosen || pH245ConfIndData->u.Indication.u.IndOpen.pTxCap->H245Dat_T120.application.u.DACy_applctn_t120.choice != separateLANStack_chosen) { bFailed = TRUE; } else { bFailed = FALSE; }
if (pH245ConfIndData->u.Indication.u.IndOpen.pSeparateStack) { if ((pH245ConfIndData->u.Indication.u.IndOpen.pSeparateStack->networkAddress.choice == localAreaAddress_chosen) && (pH245ConfIndData->u.Indication.u.IndOpen.pSeparateStack->networkAddress.u.localAreaAddress.choice == unicastAddress_chosen) && (pH245ConfIndData->u.Indication.u.IndOpen.pSeparateStack->networkAddress.u.localAreaAddress.u.unicastAddress.choice == UnicastAddress_iPAddress_chosen)) { T120Addr.nAddrType = CC_IP_BINARY; T120Addr.bMulticast = FALSE; T120Addr.Addr.IP_Binary.wPort = pH245ConfIndData->u.Indication.u.IndOpen.pSeparateStack->networkAddress.u.localAreaAddress.u.unicastAddress.u.UnicastAddress_iPAddress.tsapIdentifier; H245IPNetworkToHost(&T120Addr.Addr.IP_Binary.dwAddr, pH245ConfIndData->u.Indication.u.IndOpen.pSeparateStack->networkAddress.u.localAreaAddress.u.unicastAddress.u.UnicastAddress_iPAddress.network.value); } else { bFailed = TRUE; } }
if (bFailed) { H245OpenChannelReject(pCall->H245Instance, // H245 instance
pH245ConfIndData->u.Indication.u.IndOpen.RxChannel, H245_REJ); // rejection reason
UnlockConference(pConference); UnlockCall(pCall); return H245_ERROR_OK; }
RxTermCap.Dir = H245_CAPDIR_RMTTX; RxTermCap.DataType = pH245ConfIndData->u.Indication.u.IndOpen.RxDataType; RxTermCap.ClientType = pH245ConfIndData->u.Indication.u.IndOpen.RxClientType; RxTermCap.CapId = 0; // not used for channels
RxTermCap.Cap = *pH245ConfIndData->u.Indication.u.IndOpen.pRxCap;
TxTermCap.Dir = H245_CAPDIR_RMTTX; TxTermCap.DataType = pH245ConfIndData->u.Indication.u.IndOpen.TxDataType; TxTermCap.ClientType = pH245ConfIndData->u.Indication.u.IndOpen.TxClientType; TxTermCap.CapId = 0; // not used for channels
TxTermCap.Cap = *pH245ConfIndData->u.Indication.u.IndOpen.pTxCap;
RxH245MuxTable = *pH245ConfIndData->u.Indication.u.IndOpen.pRxMux; if ((pCall->pPeerParticipantInfo != NULL) && (pCall->pPeerParticipantInfo->TerminalIDState == TERMINAL_ID_VALID)) { RxH245MuxTable.u.H2250.destinationPresent = TRUE; RxH245MuxTable.u.H2250.destination.mcuNumber = pCall->pPeerParticipantInfo->ParticipantInfo.TerminalLabel.bMCUNumber; RxH245MuxTable.u.H2250.destination.terminalNumber = pCall->pPeerParticipantInfo->ParticipantInfo.TerminalLabel.bTerminalNumber; } else RxH245MuxTable.u.H2250.destinationPresent = FALSE;
if(pH245ConfIndData->u.Indication.u.IndOpen.pTxMux) { TxH245MuxTable = *pH245ConfIndData->u.Indication.u.IndOpen.pTxMux; TxH245MuxTable.u.H2250.destinationPresent = FALSE; }
if (AllocAndLockChannel(&hChannel, pConference, hCall, &TxTermCap, // Tx terminal capability
&RxTermCap, // Rx terminal capability
(pH245ConfIndData->u.Indication.u.IndOpen.pTxMux)? &TxH245MuxTable: NULL, // Tx H245 mux table
&RxH245MuxTable, // Rx H245 mux table
pH245ConfIndData->u.Indication.u.IndOpen.pSeparateStack, // separate stack
0, // user token
TXRX_CHANNEL, // channel type
0, // session ID
0, // associated session ID
pH245ConfIndData->u.Indication.u.IndOpen.RxChannel, // remote bi-dir channel number
NULL, // pLocalRTPAddr
NULL, // pLocalRTCPAddr
NULL, // pPeerRTPAddr
NULL, // pPeerRTCPAddr
FALSE, // locally opened
&pChannel) != CC_OK) {
H245OpenChannelReject(pCall->H245Instance, // H245 instance
pH245ConfIndData->u.Indication.u.IndOpen.RxChannel, H245_REJ); // rejection reason
UnlockConference(pConference); UnlockCall(pCall); return H245_ERROR_OK; }
if (AddChannelToConference(pChannel, pConference) != CC_OK) { H245OpenChannelReject(pCall->H245Instance, // H245 instance
pH245ConfIndData->u.Indication.u.IndOpen.RxChannel, H245_REJ); // rejection reason
UnlockConference(pConference); UnlockCall(pCall); FreeChannel(pChannel); return H245_ERROR_OK; }
T120ChannelRequestCallbackParams.hChannel = hChannel; if (pH245ConfIndData->u.Indication.u.IndOpen.pSeparateStack == NULL) { T120ChannelRequestCallbackParams.bAssociateConference = FALSE; T120ChannelRequestCallbackParams.pExternalReference = NULL; T120ChannelRequestCallbackParams.pAddr = NULL; } else { T120ChannelRequestCallbackParams.bAssociateConference = pH245ConfIndData->u.Indication.u.IndOpen.pSeparateStack->associateConference; if (pH245ConfIndData->u.Indication.u.IndOpen.pSeparateStack->bit_mask & externalReference_present) { ExternalReference.wOctetStringLength = (WORD) pH245ConfIndData->u.Indication.u.IndOpen.pSeparateStack->externalReference.length; ExternalReference.pOctetString = pH245ConfIndData->u.Indication.u.IndOpen.pSeparateStack->externalReference.value; T120ChannelRequestCallbackParams.pExternalReference = &ExternalReference; } else T120ChannelRequestCallbackParams.pExternalReference = NULL; T120ChannelRequestCallbackParams.pAddr = &T120Addr; } if ((pConference->ConferenceMode == MULTIPOINT_MODE) && (pConference->tsMultipointController == TS_TRUE)) T120ChannelRequestCallbackParams.bMultipointController = TRUE; else T120ChannelRequestCallbackParams.bMultipointController = FALSE; if (pH245ConfIndData->u.Indication.u.IndOpen.pRxMux->u.H2250.destinationPresent) { T120ChannelRequestCallbackParams.TerminalLabel.bMCUNumber = (BYTE)pH245ConfIndData->u.Indication.u.IndOpen.pRxMux->u.H2250.destination.mcuNumber; T120ChannelRequestCallbackParams.TerminalLabel.bTerminalNumber = (BYTE)pH245ConfIndData->u.Indication.u.IndOpen.pRxMux->u.H2250.destination.terminalNumber; } else { T120ChannelRequestCallbackParams.TerminalLabel.bMCUNumber = 255; T120ChannelRequestCallbackParams.TerminalLabel.bTerminalNumber = 255; }
pChannel->wNumOutstandingRequests = 1;
InvokeUserConferenceCallback(pConference, CC_T120_CHANNEL_REQUEST_INDICATION, CC_OK, &T120ChannelRequestCallbackParams);
if (ValidateChannel(hChannel) == CC_OK) UnlockChannel(pChannel); if (ValidateCall(hCall) == CC_OK) UnlockCall(pCall); if (ValidateConference(hConference) == CC_OK) UnlockConference(pConference);
return H245_ERROR_OK; }
HRESULT _IndOpen( H245_CONF_IND_T *pH245ConfIndData) { CC_HCALL hCall; PCALL pCall; PCALL pOldCall; CC_HCONFERENCE hConference; PCONFERENCE pConference; WORD wNumCalls; PCC_HCALL CallList; CC_HCHANNEL hChannel; PCHANNEL pChannel; CC_TERMCAP TermCap; CC_ADDR PeerRTPAddr; CC_ADDR PeerRTCPAddr; CC_RX_CHANNEL_REQUEST_CALLBACK_PARAMS RxChannelRequestCallbackParams; BYTE bChannelType; WORD i; H245_MUX_T H245MuxTable; PCC_ADDR pLocalRTPAddr; PCC_ADDR pLocalRTCPAddr; PCC_ADDR pPeerRTPAddr; PCC_ADDR pPeerRTCPAddr; BOOL bFoundSession; HRESULT status;
// First check to see if this is a T.120 channel request,
// as T.120 channels are handled differently then other channels
if (pH245ConfIndData->u.Indication.u.IndOpen.RxClientType == H245_CLIENT_DAT_T120) { status = _IndOpenT120(pH245ConfIndData); return status; }
hCall = pH245ConfIndData->u.Indication.dwPreserved; if (LockCallAndConference(hCall, &pCall, &pConference) != CC_OK) { // Can't cancel with H245, because we don't have the H245 instance
return H245_ERROR_OK; }
// Make sure that this is not a bi-directional channel
if (pH245ConfIndData->u.Indication.u.IndOpen.pTxMux != NULL) { H245OpenChannelReject(pCall->H245Instance, // H245 instance
pH245ConfIndData->u.Indication.u.IndOpen.RxChannel, H245_REJ); // rejection reason
UnlockConference(pConference); UnlockCall(pCall); return H245_ERROR_OK; }
hConference = pCall->hConference; TermCap.Dir = H245_CAPDIR_RMTTX; TermCap.DataType = pH245ConfIndData->u.Indication.u.IndOpen.RxDataType; TermCap.ClientType = pH245ConfIndData->u.Indication.u.IndOpen.RxClientType; TermCap.CapId = 0; // not used for Rx channels
TermCap.Cap = *pH245ConfIndData->u.Indication.u.IndOpen.pRxCap; RxChannelRequestCallbackParams.pChannelCapability = &TermCap;
if ((pH245ConfIndData->u.Indication.u.IndOpen.pRxMux != NULL) && (pH245ConfIndData->u.Indication.u.IndOpen.pRxMux->Kind == H245_H2250)) { RxChannelRequestCallbackParams.bSessionID = pH245ConfIndData->u.Indication.u.IndOpen.pRxMux->u.H2250.sessionID; if (pH245ConfIndData->u.Indication.u.IndOpen.pRxMux->u.H2250.associatedSessionIDPresent) RxChannelRequestCallbackParams.bAssociatedSessionID = pH245ConfIndData->u.Indication.u.IndOpen.pRxMux->u.H2250.associatedSessionID; else RxChannelRequestCallbackParams.bAssociatedSessionID = 0; if (pH245ConfIndData->u.Indication.u.IndOpen.pRxMux->u.H2250.silenceSuppressionPresent) RxChannelRequestCallbackParams.bSilenceSuppression = pH245ConfIndData->u.Indication.u.IndOpen.pRxMux->u.H2250.silenceSuppression; else RxChannelRequestCallbackParams.bSilenceSuppression = FALSE; if ((pH245ConfIndData->u.Indication.u.IndOpen.pRxMux->u.H2250.mediaChannelPresent) && ((pH245ConfIndData->u.Indication.u.IndOpen.pRxMux->u.H2250.mediaChannel.type == H245_IP_MULTICAST) || (pH245ConfIndData->u.Indication.u.IndOpen.pRxMux->u.H2250.mediaChannel.type == H245_IP_UNICAST))) { RxChannelRequestCallbackParams.pPeerRTPAddr = &PeerRTPAddr; PeerRTPAddr.nAddrType = CC_IP_BINARY; if (pH245ConfIndData->u.Indication.u.IndOpen.pRxMux->u.H2250.mediaChannel.type == H245_IP_MULTICAST) PeerRTPAddr.bMulticast = TRUE; else PeerRTPAddr.bMulticast = FALSE; PeerRTPAddr.Addr.IP_Binary.wPort = pH245ConfIndData->u.Indication.u.IndOpen.pRxMux->u.H2250.mediaChannel.u.ip.tsapIdentifier; H245IPNetworkToHost(&PeerRTPAddr.Addr.IP_Binary.dwAddr, pH245ConfIndData->u.Indication.u.IndOpen.pRxMux->u.H2250.mediaChannel.u.ip.network); } else RxChannelRequestCallbackParams.pPeerRTPAddr = NULL;
if ((pH245ConfIndData->u.Indication.u.IndOpen.pRxMux->u.H2250.mediaControlChannelPresent) && ((pH245ConfIndData->u.Indication.u.IndOpen.pRxMux->u.H2250.mediaControlChannel.type == H245_IP_MULTICAST) || (pH245ConfIndData->u.Indication.u.IndOpen.pRxMux->u.H2250.mediaControlChannel.type == H245_IP_UNICAST))) { RxChannelRequestCallbackParams.pPeerRTCPAddr = &PeerRTCPAddr; PeerRTCPAddr.nAddrType = CC_IP_BINARY; if (pH245ConfIndData->u.Indication.u.IndOpen.pRxMux->u.H2250.mediaControlChannel.type == H245_IP_MULTICAST) PeerRTCPAddr.bMulticast = TRUE; else PeerRTCPAddr.bMulticast = FALSE; PeerRTCPAddr.Addr.IP_Binary.wPort = pH245ConfIndData->u.Indication.u.IndOpen.pRxMux->u.H2250.mediaControlChannel.u.ip.tsapIdentifier; H245IPNetworkToHost(&PeerRTCPAddr.Addr.IP_Binary.dwAddr, pH245ConfIndData->u.Indication.u.IndOpen.pRxMux->u.H2250.mediaControlChannel.u.ip.network); } else RxChannelRequestCallbackParams.pPeerRTCPAddr = NULL;
if (pH245ConfIndData->u.Indication.u.IndOpen.pRxMux->u.H2250.destinationPresent) { RxChannelRequestCallbackParams.TerminalLabel.bMCUNumber = (BYTE)pH245ConfIndData->u.Indication.u.IndOpen.pRxMux->u.H2250.destination.mcuNumber; RxChannelRequestCallbackParams.TerminalLabel.bTerminalNumber = (BYTE)pH245ConfIndData->u.Indication.u.IndOpen.pRxMux->u.H2250.destination.terminalNumber; } else { RxChannelRequestCallbackParams.TerminalLabel.bMCUNumber = 255; RxChannelRequestCallbackParams.TerminalLabel.bTerminalNumber = 255; }
if (pH245ConfIndData->u.Indication.u.IndOpen.pRxMux->u.H2250.dynamicRTPPayloadTypePresent) RxChannelRequestCallbackParams.bRTPPayloadType = pH245ConfIndData->u.Indication.u.IndOpen.pRxMux->u.H2250.dynamicRTPPayloadType; else RxChannelRequestCallbackParams.bRTPPayloadType = 0; } else { H245OpenChannelReject(pCall->H245Instance, // H245 instance
pH245ConfIndData->u.Indication.u.IndOpen.RxChannel, H245_REJ); // rejection reason
UnlockConference(pConference); UnlockCall(pCall); return H245_ERROR_OK; }
// XXX -- someday we should allow dynamic sessions to be created on the MC
if (pH245ConfIndData->u.Indication.u.IndOpen.pRxMux->u.H2250.sessionID == 0) { H245OpenChannelReject(pCall->H245Instance, // H245 instance
pH245ConfIndData->u.Indication.u.IndOpen.RxChannel, H245_REJ); // rejection reason
UnlockConference(pConference); UnlockCall(pCall); return H245_ERROR_OK; }
if (pConference->ConferenceMode == MULTIPOINT_MODE) { if ((pConference->tsMultipointController == TS_TRUE) && ((RxChannelRequestCallbackParams.pPeerRTPAddr != NULL) || (RxChannelRequestCallbackParams.pPeerRTCPAddr != NULL)) || ((pConference->tsMultipointController == TS_FALSE) && ((RxChannelRequestCallbackParams.pPeerRTPAddr == NULL) || (RxChannelRequestCallbackParams.pPeerRTCPAddr == NULL) || (RxChannelRequestCallbackParams.bSessionID == 0)))) { H245OpenChannelReject(pCall->H245Instance, // H245 instance
pH245ConfIndData->u.Indication.u.IndOpen.RxChannel, H245_REJ); // rejection reason
UnlockConference(pConference); UnlockCall(pCall); return H245_ERROR_OK; }
// Validate session ID
pLocalRTPAddr = NULL; pLocalRTCPAddr = NULL; bFoundSession = FALSE; if (pConference->pSessionTable != NULL) { for (i = 0; i < pConference->pSessionTable->wLength; i++) { if (pH245ConfIndData->u.Indication.u.IndOpen.pRxMux->u.H2250.sessionID == pConference->pSessionTable->SessionInfoArray[i].bSessionID) { bFoundSession = TRUE; pLocalRTPAddr = pConference->pSessionTable->SessionInfoArray[i].pRTPAddr; pLocalRTCPAddr = pConference->pSessionTable->SessionInfoArray[i].pRTCPAddr; break; } } } if (bFoundSession == FALSE) { H245OpenChannelReject(pCall->H245Instance, // H245 instance
pH245ConfIndData->u.Indication.u.IndOpen.RxChannel, H245_REJ); // rejection reason
UnlockConference(pConference); UnlockCall(pCall); return H245_ERROR_OK; }
ASSERT(pLocalRTPAddr != NULL); ASSERT(pLocalRTCPAddr != NULL);
if (pConference->tsMultipointController == TS_TRUE) { pPeerRTPAddr = pLocalRTPAddr; pPeerRTCPAddr = pLocalRTCPAddr; RxChannelRequestCallbackParams.pPeerRTPAddr = pLocalRTPAddr; RxChannelRequestCallbackParams.pPeerRTCPAddr = pLocalRTCPAddr; bChannelType = PROXY_CHANNEL; } else { // multipoint mode, not MC
pLocalRTPAddr = RxChannelRequestCallbackParams.pPeerRTPAddr; pLocalRTCPAddr = RxChannelRequestCallbackParams.pPeerRTCPAddr; pPeerRTPAddr = RxChannelRequestCallbackParams.pPeerRTPAddr; pPeerRTCPAddr = RxChannelRequestCallbackParams.pPeerRTCPAddr; bChannelType = RX_CHANNEL; } } else { // not multipoint mode
pLocalRTPAddr = NULL; pLocalRTCPAddr = NULL; pPeerRTPAddr = RxChannelRequestCallbackParams.pPeerRTPAddr; pPeerRTCPAddr = RxChannelRequestCallbackParams.pPeerRTCPAddr; bChannelType = RX_CHANNEL; }
H245MuxTable = *pH245ConfIndData->u.Indication.u.IndOpen.pRxMux; if ((pCall->pPeerParticipantInfo != NULL) && (pCall->pPeerParticipantInfo->TerminalIDState == TERMINAL_ID_VALID)) { H245MuxTable.u.H2250.destinationPresent = TRUE; H245MuxTable.u.H2250.destination.mcuNumber = pCall->pPeerParticipantInfo->ParticipantInfo.TerminalLabel.bMCUNumber; H245MuxTable.u.H2250.destination.terminalNumber = pCall->pPeerParticipantInfo->ParticipantInfo.TerminalLabel.bTerminalNumber; } else H245MuxTable.u.H2250.destinationPresent = FALSE; if (pLocalRTPAddr != NULL) { if (pLocalRTPAddr->bMulticast) H245MuxTable.u.H2250.mediaChannel.type = H245_IP_MULTICAST; else H245MuxTable.u.H2250.mediaChannel.type = H245_IP_UNICAST; H245MuxTable.u.H2250.mediaChannel.u.ip.tsapIdentifier = pLocalRTPAddr->Addr.IP_Binary.wPort; HostToH245IPNetwork(H245MuxTable.u.H2250.mediaChannel.u.ip.network, pLocalRTPAddr->Addr.IP_Binary.dwAddr); H245MuxTable.u.H2250.mediaChannelPresent = TRUE; } else H245MuxTable.u.H2250.mediaChannelPresent = FALSE; if (pLocalRTCPAddr != NULL) { if (pLocalRTCPAddr->bMulticast) H245MuxTable.u.H2250.mediaControlChannel.type = H245_IP_MULTICAST; else H245MuxTable.u.H2250.mediaControlChannel.type = H245_IP_UNICAST; H245MuxTable.u.H2250.mediaControlChannel.u.ip.tsapIdentifier = pLocalRTCPAddr->Addr.IP_Binary.wPort; HostToH245IPNetwork(H245MuxTable.u.H2250.mediaControlChannel.u.ip.network, pLocalRTCPAddr->Addr.IP_Binary.dwAddr); H245MuxTable.u.H2250.mediaControlChannelPresent = TRUE; } else H245MuxTable.u.H2250.mediaControlChannelPresent = FALSE;
if (AllocAndLockChannel(&hChannel, pConference, hCall, NULL, // Tx terminal capability
&TermCap, // Rx terminal capability
NULL, // Tx H245 mux table
&H245MuxTable, // Rx H245 mux table
NULL, // separate stack
0, // user token
bChannelType, pH245ConfIndData->u.Indication.u.IndOpen.pRxMux->u.H2250.sessionID, RxChannelRequestCallbackParams.bAssociatedSessionID, pH245ConfIndData->u.Indication.u.IndOpen.RxChannel, pLocalRTPAddr, // pLocalRTPAddr
pLocalRTCPAddr, // pLocalRTCPAddr
pPeerRTPAddr, // pPeerRTPAddr
pPeerRTCPAddr, // pPeerRTCPAddr
FALSE, // locally opened
&pChannel) != CC_OK) {
H245OpenChannelReject(pCall->H245Instance, // H245 instance
pH245ConfIndData->u.Indication.u.IndOpen.RxChannel, H245_REJ); // rejection reason
UnlockConference(pConference); UnlockCall(pCall); return H245_ERROR_OK; }
if (AddChannelToConference(pChannel, pConference) != CC_OK) { H245OpenChannelReject(pCall->H245Instance, // H245 instance
pH245ConfIndData->u.Indication.u.IndOpen.RxChannel, H245_REJ); // rejection reason
UnlockConference(pConference); UnlockCall(pCall); FreeChannel(pChannel); return H245_ERROR_OK; }
RxChannelRequestCallbackParams.hChannel = hChannel; if ((pConference->ConferenceMode == MULTIPOINT_MODE) && (pConference->tsMultipointController == TS_TRUE)) { // Open this channel to each peer in the conference (except the peer
// that requested this channel)
EnumerateCallsInConference(&wNumCalls, &CallList, pConference, ESTABLISHED_CALL); for (i = 0; i < wNumCalls; i++) { if (CallList[i] != hCall) { if (LockCall(CallList[i], &pOldCall) == CC_OK) { ASSERT(pChannel->bChannelType == PROXY_CHANNEL); // Note: since this is a proxy channel, RxTermCap and RxMuxTable
// contain the channel's term cap and mux table, and must be sent
// to other endpoints as the Tx term cap and mux table;
// TxTermCap and TxMuxTable should be NULL
if (H245OpenChannel(pOldCall->H245Instance, pChannel->hChannel, // dwTransId
pChannel->wLocalChannelNumber, pChannel->pRxH245TermCap, // TxMode
pChannel->pRxMuxTable, // TxMux
H245_INVALID_PORT_NUMBER, // TxPort
pChannel->pTxH245TermCap, // RxMode
pChannel->pTxMuxTable, // RxMux
pChannel->pSeparateStack) == CC_OK) (pChannel->wNumOutstandingRequests)++; UnlockCall(pOldCall); } } } MemFree(CallList); if (pConference->LocalEndpointAttached == ATTACHED) (pChannel->wNumOutstandingRequests)++; if (pChannel->wNumOutstandingRequests == 0) { H245OpenChannelReject(pCall->H245Instance, // H245 instance
pH245ConfIndData->u.Indication.u.IndOpen.RxChannel, H245_REJ); // rejection reason
UnlockConference(pConference); UnlockCall(pCall); FreeChannel(pChannel); return H245_ERROR_OK; } } else pChannel->wNumOutstandingRequests = 1; InvokeUserConferenceCallback(pConference, CC_RX_CHANNEL_REQUEST_INDICATION, CC_OK, &RxChannelRequestCallbackParams);
if (ValidateChannel(hChannel) == CC_OK) UnlockChannel(pChannel); if (ValidateCall(hCall) == CC_OK) UnlockCall(pCall); if (ValidateConference(hConference) == CC_OK) UnlockConference(pConference);
return H245_ERROR_OK; }
HRESULT _IndOpenConf( H245_CONF_IND_T *pH245ConfIndData) { CC_HCALL hCall; PCALL pCall; CC_HCONFERENCE hConference; PCONFERENCE pConference; CC_HCHANNEL hChannel; CC_ACCEPT_CHANNEL_CALLBACK_PARAMS AcceptChannelCallbackParams;
// Bi-directional channel open initiated by remote peer is now complete.
// Local peer may now send data over this channel.
hCall = pH245ConfIndData->u.Indication.dwPreserved; if (LockCallAndConference(hCall, &pCall, &pConference) != CC_OK) return H245_ERROR_OK;
hConference = pConference->hConference; UnlockCall(pCall);
if (FindChannelInConference(pH245ConfIndData->u.Indication.u.IndOpenConf.TxChannel, FALSE, // remote channel number
TXRX_CHANNEL, hCall, &hChannel, pConference) != CC_OK) { UnlockConference(pConference); return H245_ERROR_OK; }
AcceptChannelCallbackParams.hChannel = hChannel; InvokeUserConferenceCallback(pConference, CC_ACCEPT_CHANNEL_INDICATION, CC_OK, &AcceptChannelCallbackParams);
if (ValidateConference(hConference) == CC_OK) UnlockConference(pConference);
return H245_ERROR_OK; }
HRESULT _IndMstslv( H245_CONF_IND_T *pH245ConfIndData) { CC_HCALL hCall; PCALL pCall; PCONFERENCE pConference; CC_CONNECT_CALLBACK_PARAMS ConnectCallbackParams; CC_HCALL hEnqueuedCall; PCALL pEnqueuedCall; CC_HCONFERENCE hConference; HRESULT status;
hCall = pH245ConfIndData->u.Indication.dwPreserved; if (LockCallAndConference(hCall, &pCall, &pConference) != CC_OK) { // Can't cancel with H245, because we don't have the H245 instance
return H245_ERROR_OK; }
ASSERT(pCall->MasterSlaveState != MASTER_SLAVE_COMPLETE);
switch (pH245ConfIndData->u.Indication.u.IndMstSlv) { case H245_MASTER: pConference->tsMaster = TS_TRUE; if (pConference->tsMultipointController == TS_UNKNOWN) { ASSERT(pConference->bMultipointCapable == TRUE); pConference->tsMultipointController = TS_TRUE;
// place all calls enqueued on this conference object
for ( ; ; ) { // Start up all enqueued calls, if any exist
status = RemoveEnqueuedCallFromConference(pConference, &hEnqueuedCall); if ((status != CC_OK) || (hEnqueuedCall == CC_INVALID_HANDLE)) break;
status = LockCall(hEnqueuedCall, &pEnqueuedCall); if (status == CC_OK) { pEnqueuedCall->CallState = PLACED;
status = PlaceCall(pEnqueuedCall, pConference); UnlockCall(pEnqueuedCall); } } } break;
case H245_SLAVE: ASSERT(pConference->tsMaster != TS_TRUE); ASSERT(pConference->tsMultipointController != TS_TRUE); pConference->tsMaster = TS_FALSE; pConference->tsMultipointController = TS_FALSE;
// XXX -- we may eventually want to re-enqueue these requests
// and set an expiration timer
hConference = pConference->hConference; for ( ; ; ) { status = RemoveEnqueuedCallFromConference(pConference, &hEnqueuedCall); if ((status != CC_OK) || (hEnqueuedCall == CC_INVALID_HANDLE)) break;
status = LockCall(hEnqueuedCall, &pEnqueuedCall); if (status == CC_OK) { MarkCallForDeletion(pEnqueuedCall); ConnectCallbackParams.pNonStandardData = pEnqueuedCall->pPeerNonStandardData; ConnectCallbackParams.pszPeerDisplay = pEnqueuedCall->pszPeerDisplay; ConnectCallbackParams.bRejectReason = CC_REJECT_UNDEFINED_REASON; ConnectCallbackParams.pTermCapList = pEnqueuedCall->pPeerH245TermCapList; ConnectCallbackParams.pH2250MuxCapability = pEnqueuedCall->pPeerH245H2250MuxCapability; ConnectCallbackParams.pTermCapDescriptors = pEnqueuedCall->pPeerH245TermCapDescriptors; ConnectCallbackParams.pLocalAddr = pEnqueuedCall->pQ931LocalConnectAddr; if (pEnqueuedCall->pQ931DestinationAddr == NULL) ConnectCallbackParams.pPeerAddr = pEnqueuedCall->pQ931PeerConnectAddr; else ConnectCallbackParams.pPeerAddr = pEnqueuedCall->pQ931DestinationAddr; ConnectCallbackParams.pVendorInfo = pEnqueuedCall->pPeerVendorInfo; ConnectCallbackParams.bMultipointConference = TRUE; ConnectCallbackParams.pConferenceID = &pConference->ConferenceID; ConnectCallbackParams.pMCAddress = pConference->pMultipointControllerAddr; ConnectCallbackParams.pAlternateAddress = NULL; ConnectCallbackParams.dwUserToken = pEnqueuedCall->dwUserToken;
InvokeUserConferenceCallback(pConference, CC_CONNECT_INDICATION, CC_NOT_MULTIPOINT_CAPABLE, &ConnectCallbackParams); if (ValidateCallMarkedForDeletion(hEnqueuedCall) == CC_OK) FreeCall(pEnqueuedCall); if (ValidateConference(hConference) != CC_OK) { if (ValidateCall(hCall) == CC_OK) UnlockCall(pCall); return H245_ERROR_OK; } } } break;
default: // H245_INDETERMINATE
UnlockConference(pConference); if (++pCall->wMasterSlaveRetry < MASTER_SLAVE_RETRY_MAX) { H245InitMasterSlave(pCall->H245Instance, pCall->H245Instance); UnlockCall(pCall); } else { UnlockCall(pCall); ProcessRemoteHangup(hCall, CC_INVALID_HANDLE, CC_REJECT_UNDEFINED_REASON); } return H245_ERROR_OK; } // switch
pCall->MasterSlaveState = MASTER_SLAVE_COMPLETE;
if ((pCall->OutgoingTermCapState == TERMCAP_COMPLETE) && (pCall->IncomingTermCapState == TERMCAP_COMPLETE) && (pCall->CallState == TERMCAP) && (pCall->MasterSlaveState == MASTER_SLAVE_COMPLETE)) { // Note that _ProcessConnectionComplete() returns with pConference and pCall unlocked
_ProcessConnectionComplete(pConference, pCall); return H245_ERROR_OK; } UnlockCall(pCall); UnlockConference(pConference); return H245_ERROR_OK; }
HRESULT _IndClose( H245_CONF_IND_T *pH245ConfIndData) { CC_HCALL hCall; PCALL pCall; CC_HCONFERENCE hConference; PCONFERENCE pConference; CC_HCHANNEL hChannel; PCHANNEL pChannel; WORD i; WORD wNumCalls; PCC_HCALL CallList; CC_RX_CHANNEL_CLOSE_CALLBACK_PARAMS RxChannelCloseCallbackParams; #ifdef GATEKEEPER
unsigned uBandwidth; #endif // GATEKEEPER
hCall = pH245ConfIndData->u.Indication.dwPreserved; if (LockCallAndConference(hCall, &pCall, &pConference) != CC_OK) return H245_ERROR_OK;
hConference = pCall->hConference; UnlockCall(pCall);
if (FindChannelInConference(pH245ConfIndData->u.Indication.u.IndClose.Channel, FALSE, // remote channel number
RX_CHANNEL | TXRX_CHANNEL | PROXY_CHANNEL, hCall, &hChannel, pConference) != CC_OK) { UnlockConference(pConference); return H245_ERROR_OK; }
if (LockChannel(hChannel, &pChannel) != CC_OK) return H245_ERROR_OK;
EnumerateCallsInConference(&wNumCalls, &CallList, pConference, ESTABLISHED_CALL);
#ifdef GATEKEEPER
if(GKIExists()) { if (pChannel->bChannelType != TXRX_CHANNEL) { uBandwidth = pChannel->dwChannelBitRate / 100; for (i = 0; i < wNumCalls; i++) { if (LockCall(CallList[i], &pCall) == CC_OK) { if (uBandwidth && pCall->GkiCall.uBandwidthUsed >= uBandwidth) { if (GkiCloseChannel(&pCall->GkiCall, pChannel->dwChannelBitRate, hChannel) == CC_OK) { uBandwidth = 0; UnlockCall(pCall); break; } } UnlockCall(pCall); } } // for
} } #endif // GATEKEEPER
if (pChannel->bChannelType == PROXY_CHANNEL) { ASSERT(pConference->ConferenceMode == MULTIPOINT_MODE); ASSERT(pConference->tsMultipointController == TS_TRUE); ASSERT(pChannel->bMultipointChannel == TRUE);
for (i = 0; i < wNumCalls; i++) { if (CallList[i] != hCall) { if (LockCall(CallList[i], &pCall) == CC_OK) { H245CloseChannel(pCall->H245Instance, // H245 instance
0, // dwTransId
pChannel->wLocalChannelNumber); UnlockCall(pCall); } } } }
if (CallList != NULL) MemFree(CallList);
if (pChannel->tsAccepted == TS_TRUE) { RxChannelCloseCallbackParams.hChannel = hChannel; InvokeUserConferenceCallback(pConference, CC_RX_CHANNEL_CLOSE_INDICATION, CC_OK, &RxChannelCloseCallbackParams); }
if (ValidateChannel(hChannel) == CC_OK) FreeChannel(pChannel); if (ValidateConference(hConference) == CC_OK) UnlockConference(pConference); return H245_ERROR_OK; }
HRESULT _IndRequestClose( H245_CONF_IND_T *pH245ConfIndData) { CC_HCALL hCall; PCALL pCall; CC_HCONFERENCE hConference; PCONFERENCE pConference; CC_HCHANNEL hChannel; PCHANNEL pChannel; CC_TX_CHANNEL_CLOSE_REQUEST_CALLBACK_PARAMS TxChannelCloseRequestCallbackParams;
hCall = pH245ConfIndData->u.Indication.dwPreserved; if (LockCallAndConference(hCall, &pCall, &pConference) != CC_OK) return H245_ERROR_OK;
hConference = pCall->hConference; UnlockCall(pCall);
if (FindChannelInConference(pH245ConfIndData->u.Indication.u.IndClose.Channel, TRUE, // local channel number
TX_CHANNEL | TXRX_CHANNEL | PROXY_CHANNEL, CC_INVALID_HANDLE, &hChannel, pConference) != CC_OK) { UnlockConference(pConference); return H245_ERROR_OK; }
if (LockChannel(hChannel, &pChannel) != CC_OK) { UnlockConference(pConference); return H245_ERROR_OK; } if ((pChannel->bChannelType == TX_CHANNEL) || (pChannel->bChannelType == TXRX_CHANNEL)) { EnqueueRequest(&pChannel->pCloseRequests, hCall); UnlockChannel(pChannel); TxChannelCloseRequestCallbackParams.hChannel = hChannel; InvokeUserConferenceCallback(pConference, CC_TX_CHANNEL_CLOSE_REQUEST_INDICATION, CC_OK, &TxChannelCloseRequestCallbackParams); if (ValidateConference(hConference) == CC_OK) UnlockConference(pConference); } else { // pChannel->bChannelType == PROXY_CHANNEL
if (LockCall(pChannel->hCall, &pCall) == CC_OK) { // Note that dwTransID is set to the call handle of the peer who
// initiated the close channel request. When the close channel response
// is received, the dwTransID gives us back the call handle to which
// the response must be forwarded
H245CloseChannelReq(pCall->H245Instance, hCall, // dwTransID
pChannel->wRemoteChannelNumber); UnlockCall(pCall); } UnlockChannel(pChannel); UnlockConference(pConference); } return H245_ERROR_OK; }
HRESULT _IndNonStandard( H245_CONF_IND_T *pH245ConfIndData) { CC_HCALL hCall; PCALL pCall; CC_HCONFERENCE hConference; PCONFERENCE pConference; CC_RX_NONSTANDARD_MESSAGE_CALLBACK_PARAMS RxNonStandardMessageCallbackParams;
// We only handle H221 non-standard messages; if pwObjectId is non-NULL,
// ignore the message
if (pH245ConfIndData->u.Indication.u.IndNonstandard.pwObjectId != NULL) return H245_ERROR_OK;
hCall = pH245ConfIndData->u.Indication.dwPreserved; if (LockCallAndConference(hCall, &pCall, &pConference) != CC_OK) return H245_ERROR_OK;
hConference = pCall->hConference;
switch (pH245ConfIndData->u.Indication.Indicator) { case H245_IND_NONSTANDARD_REQUEST: RxNonStandardMessageCallbackParams.bH245MessageType = CC_H245_MESSAGE_REQUEST; break; case H245_IND_NONSTANDARD_RESPONSE: RxNonStandardMessageCallbackParams.bH245MessageType = CC_H245_MESSAGE_RESPONSE; break; case H245_IND_NONSTANDARD_COMMAND: RxNonStandardMessageCallbackParams.bH245MessageType = CC_H245_MESSAGE_COMMAND; break; case H245_IND_NONSTANDARD: RxNonStandardMessageCallbackParams.bH245MessageType = CC_H245_MESSAGE_INDICATION; break; default: UnlockConference(pConference); return H245_ERROR_NOSUP; }
RxNonStandardMessageCallbackParams.NonStandardData.sData.pOctetString = pH245ConfIndData->u.Indication.u.IndNonstandard.pData; RxNonStandardMessageCallbackParams.NonStandardData.sData.wOctetStringLength = (WORD)pH245ConfIndData->u.Indication.u.IndNonstandard.dwDataLength; RxNonStandardMessageCallbackParams.NonStandardData.bCountryCode = pH245ConfIndData->u.Indication.u.IndNonstandard.byCountryCode; RxNonStandardMessageCallbackParams.NonStandardData.bExtension = pH245ConfIndData->u.Indication.u.IndNonstandard.byExtension; RxNonStandardMessageCallbackParams.NonStandardData.wManufacturerCode = pH245ConfIndData->u.Indication.u.IndNonstandard.wManufacturerCode; RxNonStandardMessageCallbackParams.hCall = pCall->hCall; if (pCall->pPeerParticipantInfo != NULL) RxNonStandardMessageCallbackParams.InitiatorTerminalLabel = pCall->pPeerParticipantInfo->ParticipantInfo.TerminalLabel; else { RxNonStandardMessageCallbackParams.InitiatorTerminalLabel.bMCUNumber = 255; RxNonStandardMessageCallbackParams.InitiatorTerminalLabel.bTerminalNumber = 255; } InvokeUserConferenceCallback(pConference, CC_RX_NONSTANDARD_MESSAGE_INDICATION, CC_OK, &RxNonStandardMessageCallbackParams);
if (ValidateConference(hConference) == CC_OK) UnlockConference(pConference); if (ValidateCall(hCall) == CC_OK) UnlockCall(pCall); return H245_ERROR_OK; }
HRESULT _IndMiscellaneous( H245_CONF_IND_T *pH245ConfIndData, MiscellaneousIndication *pMiscellaneousIndication) { HRESULT status = CC_OK; CC_HCALL hCall; PCALL pCall; PCALL pOldCall; CC_HCONFERENCE hConference; PCONFERENCE pConference; CC_HCHANNEL hChannel; PCHANNEL pChannel; WORD i; WORD wNumCalls; PCC_HCALL CallList; PDU_T Pdu; CC_MUTE_CALLBACK_PARAMS MuteCallbackParams; CC_UNMUTE_CALLBACK_PARAMS UnMuteCallbackParams; CC_H245_MISCELLANEOUS_INDICATION_CALLBACK_PARAMS H245MiscellaneousIndicationCallbackParams;
if (pMiscellaneousIndication == NULL) // Should never hit this case
return H245_ERROR_NOSUP;
hCall = pH245ConfIndData->u.Indication.dwPreserved; if (LockCallAndConference(hCall, &pCall, &pConference) != CC_OK) return H245_ERROR_OK;
hConference = pCall->hConference;
switch (pMiscellaneousIndication->type.choice) { case logicalChannelActive_chosen: case logicalChannelInactive_chosen:
UnlockCall(pCall);
if (FindChannelInConference(pMiscellaneousIndication->logicalChannelNumber, FALSE, // remote channel number
RX_CHANNEL | PROXY_CHANNEL, hCall, &hChannel, pConference) != CC_OK) { UnlockConference(pConference); return H245_ERROR_OK; }
if (LockChannel(hChannel, &pChannel) != CC_OK) { UnlockConference(pConference); return H245_ERROR_OK; }
if (pChannel->bChannelType == PROXY_CHANNEL) { ASSERT(pConference->ConferenceMode == MULTIPOINT_MODE); ASSERT(pConference->tsMultipointController == TS_TRUE); ASSERT(pChannel->bMultipointChannel == TRUE);
// Construct an H.245 PDU to hold a miscellaneous indication
// of "logical channel inactive" (mute) or "logical channel active" (unmute)
Pdu.choice = indication_chosen; Pdu.u.indication.choice = miscellaneousIndication_chosen; Pdu.u.indication.u.miscellaneousIndication.logicalChannelNumber = pChannel->wLocalChannelNumber; Pdu.u.indication.u.miscellaneousIndication.type.choice = pMiscellaneousIndication->type.choice;
EnumerateCallsInConference(&wNumCalls, &CallList, pConference, ESTABLISHED_CALL); for (i = 0; i < wNumCalls; i++) { if (CallList[i] != hCall) { if (LockCall(CallList[i], &pCall) == CC_OK) { H245SendPDU(pCall->H245Instance, &Pdu); UnlockCall(pCall); } } } MemFree(CallList); }
if (pMiscellaneousIndication->type.choice == logicalChannelActive_chosen) { if (pChannel->tsAccepted == TS_TRUE) { UnMuteCallbackParams.hChannel = hChannel; InvokeUserConferenceCallback(pConference, CC_UNMUTE_INDICATION, CC_OK, &UnMuteCallbackParams); } } else { if (pChannel->tsAccepted == TS_TRUE) { MuteCallbackParams.hChannel = hChannel; InvokeUserConferenceCallback(pConference, CC_MUTE_INDICATION, CC_OK, &MuteCallbackParams); } }
if (ValidateChannel(hChannel) == CC_OK) UnlockChannel(pChannel); if (ValidateConference(hConference) == CC_OK) UnlockConference(pConference); status = H245_ERROR_OK; break;
case multipointConference_chosen: case cnclMltpntCnfrnc_chosen: // We're required to support receipt of this indication, but I have no
// idea what we're supposed to do with it
UnlockCall(pCall); UnlockConference(pConference); status = H245_ERROR_OK; break;
case vdIndctRdyTActvt_chosen: case MIn_tp_vdTmprlSptlTrdOff_chosen: if (FindChannelInConference(pMiscellaneousIndication->logicalChannelNumber, FALSE, // remote channel number
RX_CHANNEL | PROXY_CHANNEL, hCall, &hChannel, pConference) != CC_OK) { UnlockCall(pCall); UnlockConference(pConference); return H245_ERROR_OK; }
if (LockChannel(hChannel, &pChannel) != CC_OK) { UnlockCall(pCall); UnlockConference(pConference); return H245_ERROR_OK; }
if (pChannel->bChannelType == PROXY_CHANNEL) { ASSERT(pConference->ConferenceMode == MULTIPOINT_MODE); ASSERT(pConference->tsMultipointController == TS_TRUE); ASSERT(pChannel->bMultipointChannel == TRUE);
// Construct an H.245 PDU to hold a miscellaneous indication
// of "video indicate ready to activate" or
// "video temporal spatial tradeoff"
Pdu.choice = indication_chosen; Pdu.u.indication.choice = miscellaneousIndication_chosen; Pdu.u.indication.u.miscellaneousIndication.logicalChannelNumber = pChannel->wLocalChannelNumber; Pdu.u.indication.u.miscellaneousIndication.type.choice = pMiscellaneousIndication->type.choice;
EnumerateCallsInConference(&wNumCalls, &CallList, pConference, ESTABLISHED_CALL); for (i = 0; i < wNumCalls; i++) { if (CallList[i] != hCall) { if (LockCall(CallList[i], &pOldCall) == CC_OK) { H245SendPDU(pOldCall->H245Instance, &Pdu); UnlockCall(pOldCall); } } } MemFree(CallList); }
if (pChannel->tsAccepted == TS_TRUE) { H245MiscellaneousIndicationCallbackParams.hCall = hCall; if (pCall->pPeerParticipantInfo == NULL) { H245MiscellaneousIndicationCallbackParams.InitiatorTerminalLabel.bMCUNumber = 255; H245MiscellaneousIndicationCallbackParams.InitiatorTerminalLabel.bTerminalNumber = 255; } else H245MiscellaneousIndicationCallbackParams.InitiatorTerminalLabel = pCall->pPeerParticipantInfo->ParticipantInfo.TerminalLabel; H245MiscellaneousIndicationCallbackParams.hChannel = hChannel; H245MiscellaneousIndicationCallbackParams.pMiscellaneousIndication = pMiscellaneousIndication;
status = InvokeUserConferenceCallback(pConference, CC_H245_MISCELLANEOUS_INDICATION_INDICATION, CC_OK, &H245MiscellaneousIndicationCallbackParams); if (status != CC_OK) status = H245_ERROR_NOSUP; } else status = H245_ERROR_OK;
if (ValidateChannel(hChannel) == CC_OK) UnlockChannel(pChannel); if (ValidateCall(hCall) == CC_OK) UnlockCall(pCall); if (ValidateConference(hConference) == CC_OK) UnlockConference(pConference); break;
case videoNotDecodedMBs_chosen: if (FindChannelInConference(pMiscellaneousIndication->logicalChannelNumber, TRUE, // local channel number
TX_CHANNEL | PROXY_CHANNEL, CC_INVALID_HANDLE, &hChannel, pConference) != CC_OK) { UnlockCall(pCall); UnlockConference(pConference); return H245_ERROR_OK; }
if (LockChannel(hChannel, &pChannel) != CC_OK) { UnlockCall(pCall); UnlockConference(pConference); return H245_ERROR_OK; }
if (pChannel->bChannelType == TX_CHANNEL) { H245MiscellaneousIndicationCallbackParams.hCall = hCall; if (pCall->pPeerParticipantInfo == NULL) { H245MiscellaneousIndicationCallbackParams.InitiatorTerminalLabel.bMCUNumber = 255; H245MiscellaneousIndicationCallbackParams.InitiatorTerminalLabel.bTerminalNumber = 255; } else H245MiscellaneousIndicationCallbackParams.InitiatorTerminalLabel = pCall->pPeerParticipantInfo->ParticipantInfo.TerminalLabel; H245MiscellaneousIndicationCallbackParams.hChannel = hChannel; H245MiscellaneousIndicationCallbackParams.pMiscellaneousIndication = pMiscellaneousIndication;
status = InvokeUserConferenceCallback(pConference, CC_H245_MISCELLANEOUS_INDICATION_INDICATION, CC_OK, &H245MiscellaneousIndicationCallbackParams); if (status != CC_OK) status = H245_ERROR_NOSUP;
if (ValidateChannel(hChannel) == CC_OK) UnlockChannel(pChannel); if (ValidateCall(hCall) == CC_OK) UnlockCall(pCall); if (ValidateConference(hConference) == CC_OK) UnlockConference(pConference); return H245_ERROR_OK; } else { // Proxy channel; forward the request to the transmitter
ASSERT(pConference->ConferenceMode == MULTIPOINT_MODE); ASSERT(pConference->tsMultipointController == TS_TRUE); ASSERT(pChannel->bMultipointChannel == TRUE);
// Construct an H.245 PDU to hold a miscellaneous indication
// of "video not decoded MBs"
Pdu.choice = indication_chosen; Pdu.u.indication.choice = miscellaneousIndication_chosen; Pdu.u.indication.u.miscellaneousIndication.logicalChannelNumber = pChannel->wRemoteChannelNumber; Pdu.u.indication.u.miscellaneousIndication.type.choice = pMiscellaneousIndication->type.choice;
if (LockCall(pChannel->hCall, &pOldCall) == CC_OK) { H245SendPDU(pOldCall->H245Instance, &Pdu); UnlockCall(pOldCall); } UnlockChannel(pChannel); UnlockCall(pCall); UnlockConference(pConference); return H245_ERROR_OK; } // We should never reach here
ASSERT(0);
default: // Miscellaneous indication not containing channel information
// Pass it up to the client
H245MiscellaneousIndicationCallbackParams.hCall = hCall; if (pCall->pPeerParticipantInfo == NULL) { H245MiscellaneousIndicationCallbackParams.InitiatorTerminalLabel.bMCUNumber = 255; H245MiscellaneousIndicationCallbackParams.InitiatorTerminalLabel.bTerminalNumber = 255; } else H245MiscellaneousIndicationCallbackParams.InitiatorTerminalLabel = pCall->pPeerParticipantInfo->ParticipantInfo.TerminalLabel; H245MiscellaneousIndicationCallbackParams.hChannel = CC_INVALID_HANDLE;; H245MiscellaneousIndicationCallbackParams.pMiscellaneousIndication = pMiscellaneousIndication;
status = InvokeUserConferenceCallback(pConference, CC_H245_MISCELLANEOUS_INDICATION_INDICATION, CC_OK, &H245MiscellaneousIndicationCallbackParams);
if (status != CC_OK) status = H245_ERROR_NOSUP; if (ValidateCall(hCall) == CC_OK) UnlockCall(pCall); if (ValidateConference(hConference) == CC_OK) UnlockConference(pConference); break; }
return status;
// We should never reach this point
ASSERT(0); }
HRESULT _IndMiscellaneousCommand( H245_CONF_IND_T *pH245ConfIndData, MiscellaneousCommand *pMiscellaneousCommand) { CC_HCALL hCall; PCALL pCall; PCALL pOldCall; CC_HCONFERENCE hConference; PCONFERENCE pConference; HRESULT status = CC_OK; WORD wChoice; CC_HCHANNEL hChannel; PCHANNEL pChannel; PDU_T Pdu; CC_H245_MISCELLANEOUS_COMMAND_CALLBACK_PARAMS H245MiscellaneousCommandCallbackParams;
if (pMiscellaneousCommand == NULL) // Should never hit this case
return H245_ERROR_NOSUP;
hCall = pH245ConfIndData->u.Indication.dwPreserved; if (LockCallAndConference(hCall, &pCall, &pConference) != CC_OK) return H245_ERROR_OK;
hConference = pConference->hConference;
switch (pMiscellaneousCommand->type.choice) { case multipointModeCommand_chosen: //
// from this point on, expect CommunicationModeCommand
// also, theoretically, channels shouldn't be opened until at
// least one CommunicationModeCommand is received.
// It's only by examining CommunicationModeCommand contents
// that we can determine if a conference has decentralized
// media.
// I'm commenting this out on 6/4/98 because it's bogus: all
// endpoints have centralized media distribution. Set
// pConference->ConferenceMode = MULTIPOINT_MODE; only after
// CommunicationModeCommand is received and the multiplex table has
// been examined and decentralized media is found
#if(0)
if (pConference->bMultipointCapable == FALSE) { // We can't support multipoint operation, so treat this as if
// we received a remote hangup indication
UnlockConference(pConference); UnlockCall(pCall); ProcessRemoteHangup(hCall, CC_INVALID_HANDLE, CC_REJECT_NORMAL_CALL_CLEARING); return H245_ERROR_OK; } else { pConference->ConferenceMode = MULTIPOINT_MODE;
// Send TerminalListRequest
H245ConferenceRequest(pCall->H245Instance, H245_REQ_TERMINAL_LIST, pConference->LocalParticipantInfo.ParticipantInfo.TerminalLabel.bMCUNumber, pConference->LocalParticipantInfo.ParticipantInfo.TerminalLabel.bTerminalNumber); } #else
// Send TerminalListRequest
H245ConferenceRequest(pCall->H245Instance, H245_REQ_TERMINAL_LIST, pConference->LocalParticipantInfo.ParticipantInfo.TerminalLabel.bMCUNumber, pConference->LocalParticipantInfo.ParticipantInfo.TerminalLabel.bTerminalNumber); #endif
status = H245_ERROR_OK; break;
case cnclMltpntMdCmmnd_chosen: // We're required to support receipt of this command, but I have no
// idea what we're supposed to do with it
status = H245_ERROR_OK; break;
case videoFreezePicture_chosen: case videoFastUpdatePicture_chosen: case videoFastUpdateGOB_chosen: case MCd_tp_vdTmprlSptlTrdOff_chosen: case videoSendSyncEveryGOB_chosen: case videoFastUpdateMB_chosen: case vdSndSyncEvryGOBCncl_chosen: if (FindChannelInConference(pMiscellaneousCommand->logicalChannelNumber, TRUE, // local channel number
TX_CHANNEL | PROXY_CHANNEL, CC_INVALID_HANDLE, &hChannel, pConference) != CC_OK) { UnlockCall(pCall); UnlockConference(pConference); return H245_ERROR_OK; }
if (LockChannel(hChannel, &pChannel) != CC_OK) { UnlockCall(pCall); UnlockConference(pConference); return H245_ERROR_OK; }
if (pChannel->bChannelType == TX_CHANNEL) { H245MiscellaneousCommandCallbackParams.hCall = hCall; if (pCall->pPeerParticipantInfo == NULL) { H245MiscellaneousCommandCallbackParams.InitiatorTerminalLabel.bMCUNumber = 255; H245MiscellaneousCommandCallbackParams.InitiatorTerminalLabel.bTerminalNumber = 255; } else H245MiscellaneousCommandCallbackParams.InitiatorTerminalLabel = pCall->pPeerParticipantInfo->ParticipantInfo.TerminalLabel; H245MiscellaneousCommandCallbackParams.hChannel = hChannel; wChoice = pMiscellaneousCommand->type.choice; if ((wChoice == videoFreezePicture_chosen) || (wChoice == videoFastUpdatePicture_chosen) || (wChoice == videoFastUpdateGOB_chosen) || (wChoice == videoFastUpdateMB_chosen)) H245MiscellaneousCommandCallbackParams.bH323ActionRequired = TRUE; else H245MiscellaneousCommandCallbackParams.bH323ActionRequired = FALSE; H245MiscellaneousCommandCallbackParams.pMiscellaneousCommand = pMiscellaneousCommand;
status = InvokeUserConferenceCallback(pConference, CC_H245_MISCELLANEOUS_COMMAND_INDICATION, CC_OK, &H245MiscellaneousCommandCallbackParams); if (status != CC_OK) status = H245_ERROR_NOSUP;
if (ValidateChannel(hChannel) == CC_OK) UnlockChannel(pChannel); if (ValidateCall(hCall) == CC_OK) UnlockCall(pCall); if (ValidateConference(hConference) == CC_OK) UnlockConference(pConference); return H245_ERROR_OK; } else { // Proxy channel; forward the request to the transmitter
ASSERT(pConference->ConferenceMode == MULTIPOINT_MODE); ASSERT(pConference->tsMultipointController == TS_TRUE); ASSERT(pChannel->bMultipointChannel == TRUE);
Pdu.choice = MSCMg_cmmnd_chosen; Pdu.u.MSCMg_cmmnd.choice = miscellaneousCommand_chosen; Pdu.u.MSCMg_cmmnd.u.miscellaneousCommand.logicalChannelNumber = pChannel->wRemoteChannelNumber; Pdu.u.MSCMg_cmmnd.u.miscellaneousCommand = *pMiscellaneousCommand; if (LockCall(pChannel->hCall, &pOldCall) == CC_OK) { H245SendPDU(pOldCall->H245Instance, &Pdu); UnlockCall(pOldCall); } UnlockChannel(pChannel); UnlockCall(pCall); UnlockConference(pConference); return H245_ERROR_OK; } // We should never reach here
ASSERT(0);
default: // Unrecognized miscellaneous command
// Pass it up to the client
H245MiscellaneousCommandCallbackParams.hCall = hCall; if (pCall->pPeerParticipantInfo == NULL) { H245MiscellaneousCommandCallbackParams.InitiatorTerminalLabel.bMCUNumber = 255; H245MiscellaneousCommandCallbackParams.InitiatorTerminalLabel.bTerminalNumber = 255; } else H245MiscellaneousCommandCallbackParams.InitiatorTerminalLabel = pCall->pPeerParticipantInfo->ParticipantInfo.TerminalLabel; H245MiscellaneousCommandCallbackParams.hChannel = CC_INVALID_HANDLE; H245MiscellaneousCommandCallbackParams.bH323ActionRequired = FALSE; H245MiscellaneousCommandCallbackParams.pMiscellaneousCommand = pMiscellaneousCommand; status = InvokeUserConferenceCallback(pConference, CC_H245_MISCELLANEOUS_COMMAND_INDICATION, CC_OK, &H245MiscellaneousCommandCallbackParams); if (status != CC_OK) status = H245_ERROR_NOSUP; if (ValidateCall(hCall) == CC_OK) UnlockCall(pCall); if (ValidateConference(hConference) == CC_OK) UnlockConference(pConference); return status; }
if (ValidateCall(hCall) == CC_OK) UnlockCall(pCall); if (ValidateConference(hConference) == CC_OK) UnlockConference(pConference);
return status;
// We should never reach this point
ASSERT(0); }
HRESULT _IndMCLocation( H245_CONF_IND_T *pH245ConfIndData) { CC_HCALL hCall; PCALL pCall; PCONFERENCE pConference; CC_CONNECT_CALLBACK_PARAMS ConnectCallbackParams; PCALL pEnqueuedCall; CC_HCALL hEnqueuedCall; HRESULT status; CC_HCONFERENCE hConference;
if (pH245ConfIndData->u.Indication.u.IndMcLocation.type != H245_IP_UNICAST) return H245_ERROR_OK;
hCall = pH245ConfIndData->u.Indication.dwPreserved;
if (LockCallAndConference(hCall, &pCall, &pConference) != CC_OK) return H245_ERROR_OK;
UnlockCall(pCall);
hConference = pConference->hConference;
if (pConference->tsMultipointController != TS_FALSE) { // We don't expect to receive an MCLocationIndication until master/slave
// has completed, at which time tsMultipointController will change from
// TS_UNKNOWN to either TS_TRUE or TS_FALSE.
UnlockConference(pConference); return H245_ERROR_NOSUP; }
if (pConference->pMultipointControllerAddr == NULL) { pConference->pMultipointControllerAddr = (PCC_ADDR)MemAlloc(sizeof(CC_ADDR)); if (pConference->pMultipointControllerAddr == NULL) { UnlockConference(pConference); return H245_ERROR_OK; } } pConference->pMultipointControllerAddr->nAddrType = CC_IP_BINARY; pConference->pMultipointControllerAddr->bMulticast = FALSE; pConference->pMultipointControllerAddr->Addr.IP_Binary.wPort = pH245ConfIndData->u.Indication.u.IndMcLocation.u.ip.tsapIdentifier; H245IPNetworkToHost(&pConference->pMultipointControllerAddr->Addr.IP_Binary.dwAddr, pH245ConfIndData->u.Indication.u.IndMcLocation.u.ip.network);
// place all calls enqueued on this conference object
for ( ; ; ) { // Start up all enqueued calls, if any exist
status = RemoveEnqueuedCallFromConference(pConference, &hEnqueuedCall); if ((status != CC_OK) || (hEnqueuedCall == CC_INVALID_HANDLE)) break;
status = LockCall(hEnqueuedCall, &pEnqueuedCall); if (status == CC_OK) { // Place Call to MC
pEnqueuedCall->CallState = PLACED; pEnqueuedCall->CallType = THIRD_PARTY_INVITOR; if (pEnqueuedCall->pQ931DestinationAddr == NULL) pEnqueuedCall->pQ931DestinationAddr = pEnqueuedCall->pQ931PeerConnectAddr; if (pEnqueuedCall->pQ931PeerConnectAddr == NULL) pEnqueuedCall->pQ931PeerConnectAddr = (PCC_ADDR)MemAlloc(sizeof(CC_ADDR)); if (pEnqueuedCall->pQ931PeerConnectAddr == NULL) { MarkCallForDeletion(pEnqueuedCall); ConnectCallbackParams.pNonStandardData = pEnqueuedCall->pPeerNonStandardData; ConnectCallbackParams.pszPeerDisplay = pEnqueuedCall->pszPeerDisplay; ConnectCallbackParams.bRejectReason = CC_REJECT_UNDEFINED_REASON; ConnectCallbackParams.pTermCapList = pEnqueuedCall->pPeerH245TermCapList; ConnectCallbackParams.pH2250MuxCapability = pEnqueuedCall->pPeerH245H2250MuxCapability; ConnectCallbackParams.pTermCapDescriptors = pEnqueuedCall->pPeerH245TermCapDescriptors; ConnectCallbackParams.pLocalAddr = pEnqueuedCall->pQ931LocalConnectAddr; if (pEnqueuedCall->pQ931DestinationAddr == NULL) ConnectCallbackParams.pPeerAddr = pEnqueuedCall->pQ931PeerConnectAddr; else ConnectCallbackParams.pPeerAddr = pEnqueuedCall->pQ931DestinationAddr; ConnectCallbackParams.pVendorInfo = pEnqueuedCall->pPeerVendorInfo; ConnectCallbackParams.bMultipointConference = TRUE; ConnectCallbackParams.pConferenceID = &pConference->ConferenceID; ConnectCallbackParams.pMCAddress = pConference->pMultipointControllerAddr; ConnectCallbackParams.pAlternateAddress = NULL; ConnectCallbackParams.dwUserToken = pEnqueuedCall->dwUserToken;
InvokeUserConferenceCallback(pConference, CC_CONNECT_INDICATION, CC_NO_MEMORY, &ConnectCallbackParams);
if (ValidateCallMarkedForDeletion(hEnqueuedCall) == CC_OK) FreeCall(pEnqueuedCall); if (ValidateConference(hConference) != CC_OK) return H245_ERROR_OK; } pEnqueuedCall->pQ931PeerConnectAddr = pConference->pMultipointControllerAddr;
status = PlaceCall(pEnqueuedCall, pConference); UnlockCall(pEnqueuedCall); } }
UnlockConference(pConference); return H245_ERROR_OK; }
HRESULT _IndConferenceRequest( H245_CONF_IND_T *pH245ConfIndData) { CC_HCALL hCall; PCALL pCall; PCALL pPeerCall; CC_HCONFERENCE hConference; PCONFERENCE pConference; HRESULT status; H245_TERMINAL_LABEL_T *H245TerminalLabelList; H245_TERMINAL_LABEL_T H245TerminalLabel; WORD wNumTerminalLabels; CC_H245_CONFERENCE_REQUEST_CALLBACK_PARAMS H245ConferenceRequestCallbackParams;
hCall = pH245ConfIndData->u.Indication.dwPreserved; if (LockCallAndConference(hCall, &pCall, &pConference) != CC_OK) return H245_ERROR_OK;
hConference = pConference->hConference;
switch (pH245ConfIndData->u.Indication.u.IndConferReq.RequestType) { case H245_REQ_ENTER_H243_TERMINAL_ID: switch (pConference->LocalParticipantInfo.TerminalIDState) { case TERMINAL_ID_INVALID: UnlockCall(pCall); EnqueueRequest(&pConference->LocalParticipantInfo.pEnqueuedRequestsForTerminalID, hCall); pConference->LocalParticipantInfo.TerminalIDState = TERMINAL_ID_REQUESTED; InvokeUserConferenceCallback(pConference, CC_TERMINAL_ID_REQUEST_INDICATION, CC_OK, NULL); if (ValidateConference(hConference) == CC_OK) UnlockConference(pConference); return H245_ERROR_OK;
case TERMINAL_ID_REQUESTED: UnlockCall(pCall); EnqueueRequest(&pConference->LocalParticipantInfo.pEnqueuedRequestsForTerminalID, hCall); UnlockConference(pConference); return H245_ERROR_OK;
case TERMINAL_ID_VALID: H245ConferenceResponse(pCall->H245Instance, H245_RSP_TERMINAL_ID, pConference->LocalParticipantInfo.ParticipantInfo.TerminalLabel.bMCUNumber, pConference->LocalParticipantInfo.ParticipantInfo.TerminalLabel.bTerminalNumber, pConference->LocalParticipantInfo.ParticipantInfo.TerminalID.pOctetString, (BYTE)pConference->LocalParticipantInfo.ParticipantInfo.TerminalID.wOctetStringLength, NULL, // terminal list
0); // terminal list count
UnlockCall(pCall); UnlockConference(pConference); return H245_ERROR_OK;
default: ASSERT(0); }
case H245_REQ_TERMINAL_LIST: if ((pConference->ConferenceMode == MULTIPOINT_MODE) && (pConference->tsMultipointController == TS_TRUE)) { status = EnumerateTerminalLabelsInConference(&wNumTerminalLabels, &H245TerminalLabelList, pConference); if (status == CC_OK) H245ConferenceResponse(pCall->H245Instance, H245_RSP_TERMINAL_LIST, pConference->LocalParticipantInfo.ParticipantInfo.TerminalLabel.bMCUNumber, pConference->LocalParticipantInfo.ParticipantInfo.TerminalLabel.bTerminalNumber, pConference->LocalParticipantInfo.ParticipantInfo.TerminalID.pOctetString, (BYTE)pConference->LocalParticipantInfo.ParticipantInfo.TerminalID.wOctetStringLength, H245TerminalLabelList, // terminal list
wNumTerminalLabels); // terminal list count
if (H245TerminalLabelList != NULL) MemFree(H245TerminalLabelList); status = H245_ERROR_OK; } else status = H245_ERROR_NOSUP; break;
case H245_REQ_TERMINAL_ID: if (pConference->tsMultipointController != TS_TRUE) { status = H245_ERROR_NOSUP; break; }
if (pConference->LocalParticipantInfo.ParticipantInfo.TerminalLabel.bMCUNumber != pH245ConfIndData->u.Indication.u.IndConferReq.byMcuNumber) { // This terminal ID wasn't allocated by this MC, so return without a response
status = H245_ERROR_OK; break; }
// First check to see whether the requested terminal ID is ours
if (pConference->LocalParticipantInfo.ParticipantInfo.TerminalLabel.bTerminalNumber == pH245ConfIndData->u.Indication.u.IndConferReq.byTerminalNumber) { if (pConference->LocalEndpointAttached != ATTACHED) { status = H245_ERROR_OK; break; }
switch (pConference->LocalParticipantInfo.TerminalIDState) { case TERMINAL_ID_INVALID: UnlockCall(pCall); EnqueueRequest(&pConference->LocalParticipantInfo.pEnqueuedRequestsForTerminalID, hCall); pConference->LocalParticipantInfo.TerminalIDState = TERMINAL_ID_REQUESTED; InvokeUserConferenceCallback(pConference, CC_TERMINAL_ID_REQUEST_INDICATION, CC_OK, NULL); if (ValidateConference(hConference) == CC_OK) UnlockConference(pConference); return H245_ERROR_OK;
case TERMINAL_ID_REQUESTED: UnlockCall(pCall); EnqueueRequest(&pConference->LocalParticipantInfo.pEnqueuedRequestsForTerminalID, hCall); UnlockConference(pConference); return H245_ERROR_OK;
case TERMINAL_ID_VALID: H245ConferenceResponse(pCall->H245Instance, H245_RSP_MC_TERMINAL_ID, pConference->LocalParticipantInfo.ParticipantInfo.TerminalLabel.bMCUNumber, pConference->LocalParticipantInfo.ParticipantInfo.TerminalLabel.bTerminalNumber, pConference->LocalParticipantInfo.ParticipantInfo.TerminalID.pOctetString, (BYTE)pConference->LocalParticipantInfo.ParticipantInfo.TerminalID.wOctetStringLength, NULL, // terminal list
0); // terminal list count
UnlockCall(pCall); UnlockConference(pConference); return H245_ERROR_OK;
default: ASSERT(0); } }
H245TerminalLabel.mcuNumber = pH245ConfIndData->u.Indication.u.IndConferReq.byMcuNumber; H245TerminalLabel.terminalNumber = pH245ConfIndData->u.Indication.u.IndConferReq.byTerminalNumber;
FindPeerParticipantInfo(H245TerminalLabel, pConference, ESTABLISHED_CALL, &pPeerCall); if (pPeerCall == NULL) { // We don't know about the existance of this terminal ID, so return without a response
status = H245_ERROR_OK; break; }
if (pPeerCall->pPeerParticipantInfo == NULL) { UnlockCall(pPeerCall); status = H245_ERROR_OK; break; }
switch (pPeerCall->pPeerParticipantInfo->TerminalIDState) { case TERMINAL_ID_INVALID: EnqueueRequest(&pPeerCall->pPeerParticipantInfo->pEnqueuedRequestsForTerminalID, hCall); pPeerCall->pPeerParticipantInfo->TerminalIDState = TERMINAL_ID_REQUESTED; H245ConferenceRequest(pPeerCall->H245Instance, H245_REQ_ENTER_H243_TERMINAL_ID, pPeerCall->pPeerParticipantInfo->ParticipantInfo.TerminalLabel.bMCUNumber, pPeerCall->pPeerParticipantInfo->ParticipantInfo.TerminalLabel.bTerminalNumber); break;
case TERMINAL_ID_REQUESTED: EnqueueRequest(&pPeerCall->pPeerParticipantInfo->pEnqueuedRequestsForTerminalID, hCall); break;
case TERMINAL_ID_VALID: H245ConferenceResponse(pCall->H245Instance, H245_RSP_MC_TERMINAL_ID, pPeerCall->pPeerParticipantInfo->ParticipantInfo.TerminalLabel.bMCUNumber, pPeerCall->pPeerParticipantInfo->ParticipantInfo.TerminalLabel.bTerminalNumber, pPeerCall->pPeerParticipantInfo->ParticipantInfo.TerminalID.pOctetString, (BYTE)pPeerCall->pPeerParticipantInfo->ParticipantInfo.TerminalID.wOctetStringLength, NULL, // terminal list
0); // terminal list count
break;
default: ASSERT(0); break; } UnlockCall(pPeerCall); status = H245_ERROR_OK; break;
default: H245ConferenceRequestCallbackParams.hCall = hCall; if (pCall->pPeerParticipantInfo == NULL) { H245ConferenceRequestCallbackParams.InitiatorTerminalLabel.bMCUNumber = 255; H245ConferenceRequestCallbackParams.InitiatorTerminalLabel.bTerminalNumber = 255; } else H245ConferenceRequestCallbackParams.InitiatorTerminalLabel = pCall->pPeerParticipantInfo->ParticipantInfo.TerminalLabel; H245ConferenceRequestCallbackParams.RequestType = pH245ConfIndData->u.Indication.u.IndConferReq.RequestType; H245ConferenceRequestCallbackParams.TerminalLabel.bMCUNumber = pH245ConfIndData->u.Indication.u.IndConferReq.byMcuNumber; H245ConferenceRequestCallbackParams.TerminalLabel.bTerminalNumber = pH245ConfIndData->u.Indication.u.IndConferReq.byTerminalNumber; status = InvokeUserConferenceCallback(pConference, CC_H245_CONFERENCE_REQUEST_INDICATION, CC_OK, &H245ConferenceRequestCallbackParams); if (status != CC_OK) status = H245_ERROR_NOSUP; break; }
if (ValidateCall(hCall) == CC_OK) UnlockCall(pCall); if (ValidateConference(hConference) == CC_OK) UnlockConference(pConference); return status; }
HRESULT _IndConferenceResponse( H245_CONF_IND_T *pH245ConfIndData) { CC_HCALL hCall; PCALL pCall; PCALL pPeerCall; CC_HCALL hEnqueuedCall; PCALL pEnqueuedCall; CC_HCALL hVirtualCall; CC_HCALL hPeerCall; PCALL pVirtualCall; PCONFERENCE pConference; CC_HCONFERENCE hConference; HRESULT status; WORD i; PPARTICIPANTINFO pPeerParticipantInfo; H245_TERMINAL_LABEL_T TerminalLabel; CC_PEER_ADD_CALLBACK_PARAMS PeerAddCallbackParams; CC_PEER_UPDATE_CALLBACK_PARAMS PeerUpdateCallbackParams; CC_H245_CONFERENCE_RESPONSE_CALLBACK_PARAMS H245ConferenceResponseCallbackParams; CC_OCTETSTRING OctetString;
hCall = pH245ConfIndData->u.Indication.dwPreserved; if (LockCallAndConference(hCall, &pCall, &pConference) != CC_OK) return H245_ERROR_OK;
hConference = pConference->hConference;
switch (pH245ConfIndData->u.Indication.u.IndConferRsp.ResponseType) { case H245_RSP_TERMINAL_LIST: if (pConference->tsMultipointController == TS_FALSE) { for (i = 0; i < pH245ConfIndData->u.Indication.u.IndConferRsp.wTerminalListCount; i++) { if ((pH245ConfIndData->u.Indication.u.IndConferRsp.pTerminalList[i].mcuNumber == pConference->LocalParticipantInfo.ParticipantInfo.TerminalLabel.bMCUNumber) && (pH245ConfIndData->u.Indication.u.IndConferRsp.pTerminalList[i].terminalNumber == pConference->LocalParticipantInfo.ParticipantInfo.TerminalLabel.bTerminalNumber)) // This terminal number refers to us
continue; FindPeerParticipantInfo(pH245ConfIndData->u.Indication.u.IndConferRsp.pTerminalList[i], pConference, VIRTUAL_CALL, &pPeerCall); if (pPeerCall != NULL) { // We already know this peer's terminal label, and we
// eithet know its terminal ID or we have a pending request
// to obtain it
UnlockCall(pPeerCall); continue; }
// We don't know about this peer.
// Create a virtual call object for it, and issue a request
// for its terminal ID
status = AllocAndLockCall(&hVirtualCall, pConference->hConference, CC_INVALID_HANDLE, // hQ931Call
CC_INVALID_HANDLE, // hQ931CallInvitor,
NULL, // pLocalAliasNames,
NULL, // pPeerAliasNames,
NULL, // pPeerExtraAliasNames
NULL, // pPeerExtension
NULL, // pLocalNonStandardData,
NULL, // pPeerNonStandardData,
NULL, // pszLocalDisplay,
NULL, // pszPeerDisplay,
NULL, // pPeerVendorInfo,
NULL, // pQ931LocalConnectAddr,
NULL, // pQ931PeerConnectAddr,
NULL, // pQ931DestinationAddr,
NULL, // pSourceCallSignalAddress
VIRTUAL, // CallType,
FALSE, // bCallerIsMC,
0, // dwUserToken,
CALL_COMPLETE, // InitialCallState,
NULL, // no CallIdentifier
&pConference->ConferenceID, &pVirtualCall); if (status == CC_OK) { status = AllocatePeerParticipantInfo(NULL, &pPeerParticipantInfo); if (status == CC_OK) { pVirtualCall->pPeerParticipantInfo = pPeerParticipantInfo; pPeerParticipantInfo->ParticipantInfo.TerminalLabel.bMCUNumber = (BYTE)pH245ConfIndData->u.Indication.u.IndConferRsp.pTerminalList[i].mcuNumber; pPeerParticipantInfo->ParticipantInfo.TerminalLabel.bTerminalNumber = (BYTE)pH245ConfIndData->u.Indication.u.IndConferRsp.pTerminalList[i].terminalNumber; AddVirtualCallToConference(pVirtualCall, pConference); // Send RequestTerminalID
H245ConferenceRequest(pCall->H245Instance, H245_REQ_TERMINAL_ID, (BYTE)pH245ConfIndData->u.Indication.u.IndConferRsp.pTerminalList[i].mcuNumber, (BYTE)pH245ConfIndData->u.Indication.u.IndConferRsp.pTerminalList[i].terminalNumber); pPeerParticipantInfo->TerminalIDState = TERMINAL_ID_REQUESTED;
// Generate PEER_ADD callback
PeerAddCallbackParams.hCall = hVirtualCall; PeerAddCallbackParams.TerminalLabel = pVirtualCall->pPeerParticipantInfo->ParticipantInfo.TerminalLabel; PeerAddCallbackParams.pPeerTerminalID = NULL; InvokeUserConferenceCallback(pConference, CC_PEER_ADD_INDICATION, CC_OK, &PeerAddCallbackParams); if (ValidateCall(hVirtualCall) == CC_OK) UnlockCall(pVirtualCall); } else FreeCall(pVirtualCall); } } } status = H245_ERROR_OK; break;
case H245_RSP_MC_TERMINAL_ID: if (pConference->tsMultipointController == TS_FALSE) { TerminalLabel.mcuNumber = pH245ConfIndData->u.Indication.u.IndConferRsp.byMcuNumber; TerminalLabel.terminalNumber = pH245ConfIndData->u.Indication.u.IndConferRsp.byTerminalNumber; FindPeerParticipantInfo(TerminalLabel, pConference, VIRTUAL_CALL, &pPeerCall); if (pPeerCall != NULL) { hPeerCall = pPeerCall->hCall; if (pPeerCall->pPeerParticipantInfo->TerminalIDState != TERMINAL_ID_VALID) { pPeerCall->pPeerParticipantInfo->ParticipantInfo.TerminalID.pOctetString = (BYTE *)MemAlloc(pH245ConfIndData->u.Indication.u.IndConferRsp.byOctetStringLength); if (pPeerCall->pPeerParticipantInfo->ParticipantInfo.TerminalID.pOctetString == NULL) { UnlockCall(pPeerCall); status = H245_ERROR_OK; break; } memcpy(pPeerCall->pPeerParticipantInfo->ParticipantInfo.TerminalID.pOctetString, pH245ConfIndData->u.Indication.u.IndConferRsp.pOctetString, pH245ConfIndData->u.Indication.u.IndConferRsp.byOctetStringLength); pPeerCall->pPeerParticipantInfo->ParticipantInfo.TerminalID.wOctetStringLength = pH245ConfIndData->u.Indication.u.IndConferRsp.byOctetStringLength; pPeerCall->pPeerParticipantInfo->TerminalIDState = TERMINAL_ID_VALID; PeerUpdateCallbackParams.hCall = hPeerCall; PeerUpdateCallbackParams.TerminalLabel = pPeerCall->pPeerParticipantInfo->ParticipantInfo.TerminalLabel; PeerUpdateCallbackParams.pPeerTerminalID = &pPeerCall->pPeerParticipantInfo->ParticipantInfo.TerminalID; InvokeUserConferenceCallback(pConference, CC_PEER_UPDATE_INDICATION, CC_OK, &PeerUpdateCallbackParams); } if (ValidateCall(hPeerCall) == CC_OK) UnlockCall(pPeerCall); } } status = H245_ERROR_OK; break;
case H245_RSP_TERMINAL_ID: if ((pConference->ConferenceMode == MULTIPOINT_MODE) && (pConference->tsMultipointController == TS_TRUE)) { TerminalLabel.mcuNumber = pH245ConfIndData->u.Indication.u.IndConferRsp.byMcuNumber; TerminalLabel.terminalNumber = pH245ConfIndData->u.Indication.u.IndConferRsp.byTerminalNumber; FindPeerParticipantInfo(TerminalLabel, pConference, ESTABLISHED_CALL, &pPeerCall); if (pPeerCall != NULL) { hPeerCall = pPeerCall->hCall; if (pPeerCall->pPeerParticipantInfo->TerminalIDState != TERMINAL_ID_VALID) { pPeerCall->pPeerParticipantInfo->ParticipantInfo.TerminalID.pOctetString = (BYTE *)MemAlloc(pH245ConfIndData->u.Indication.u.IndConferRsp.byOctetStringLength); if (pPeerCall->pPeerParticipantInfo->ParticipantInfo.TerminalID.pOctetString == NULL) { UnlockCall(pPeerCall); status = H245_ERROR_OK; break; } memcpy(pPeerCall->pPeerParticipantInfo->ParticipantInfo.TerminalID.pOctetString, pH245ConfIndData->u.Indication.u.IndConferRsp.pOctetString, pH245ConfIndData->u.Indication.u.IndConferRsp.byOctetStringLength); pPeerCall->pPeerParticipantInfo->ParticipantInfo.TerminalID.wOctetStringLength = pH245ConfIndData->u.Indication.u.IndConferRsp.byOctetStringLength; pPeerCall->pPeerParticipantInfo->TerminalIDState = TERMINAL_ID_VALID; // Dequeue and respond to each enqueued request for this terminal ID
while (DequeueRequest(&pPeerCall->pPeerParticipantInfo->pEnqueuedRequestsForTerminalID, &hEnqueuedCall) == CC_OK) { if (LockCall(hEnqueuedCall, &pEnqueuedCall) == CC_OK) { H245ConferenceResponse(pEnqueuedCall->H245Instance, H245_RSP_MC_TERMINAL_ID, pPeerCall->pPeerParticipantInfo->ParticipantInfo.TerminalLabel.bMCUNumber, pPeerCall->pPeerParticipantInfo->ParticipantInfo.TerminalLabel.bTerminalNumber, pPeerCall->pPeerParticipantInfo->ParticipantInfo.TerminalID.pOctetString, (BYTE)pPeerCall->pPeerParticipantInfo->ParticipantInfo.TerminalID.wOctetStringLength, NULL, // terminal list
0); // terminal list count
UnlockCall(pEnqueuedCall); } }
// Generate a CC_PEER_UPDATE_INDICATION callback
PeerUpdateCallbackParams.hCall = hPeerCall; PeerUpdateCallbackParams.TerminalLabel = pPeerCall->pPeerParticipantInfo->ParticipantInfo.TerminalLabel; PeerUpdateCallbackParams.pPeerTerminalID = &pPeerCall->pPeerParticipantInfo->ParticipantInfo.TerminalID; InvokeUserConferenceCallback(pConference, CC_PEER_UPDATE_INDICATION, CC_OK, &PeerUpdateCallbackParams); } if (ValidateCall(hPeerCall) == CC_OK) UnlockCall(pPeerCall); } } status = H245_ERROR_OK; break;
default: H245ConferenceResponseCallbackParams.hCall = hCall; if (pCall->pPeerParticipantInfo == NULL) { H245ConferenceResponseCallbackParams.InitiatorTerminalLabel.bMCUNumber = 255; H245ConferenceResponseCallbackParams.InitiatorTerminalLabel.bTerminalNumber = 255; } else H245ConferenceResponseCallbackParams.InitiatorTerminalLabel = pCall->pPeerParticipantInfo->ParticipantInfo.TerminalLabel; H245ConferenceResponseCallbackParams.ResponseType = pH245ConfIndData->u.Indication.u.IndConferRsp.ResponseType; H245ConferenceResponseCallbackParams.TerminalLabel.bMCUNumber = pH245ConfIndData->u.Indication.u.IndConferRsp.byMcuNumber; H245ConferenceResponseCallbackParams.TerminalLabel.bTerminalNumber = pH245ConfIndData->u.Indication.u.IndConferRsp.byTerminalNumber; if ((pH245ConfIndData->u.Indication.u.IndConferRsp.pOctetString == NULL) || (pH245ConfIndData->u.Indication.u.IndConferRsp.byOctetStringLength == 0)) { H245ConferenceResponseCallbackParams.pOctetString = NULL; } else { OctetString.pOctetString = pH245ConfIndData->u.Indication.u.IndConferRsp.pOctetString; OctetString.wOctetStringLength = pH245ConfIndData->u.Indication.u.IndConferRsp.byOctetStringLength; H245ConferenceResponseCallbackParams.pOctetString = &OctetString; } if (pH245ConfIndData->u.Indication.u.IndConferRsp.wTerminalListCount == 0) { H245ConferenceResponseCallbackParams.pTerminalList = NULL; H245ConferenceResponseCallbackParams.wTerminalListCount = 0; status = CC_OK; } else { H245ConferenceResponseCallbackParams.pTerminalList = (CC_TERMINAL_LABEL *)MemAlloc(sizeof(CC_TERMINAL_LABEL) * pH245ConfIndData->u.Indication.u.IndConferRsp.wTerminalListCount); if (H245ConferenceResponseCallbackParams.pTerminalList == NULL) { H245ConferenceResponseCallbackParams.wTerminalListCount = 0; status = CC_NO_MEMORY; } else { for (i = 0; i < pH245ConfIndData->u.Indication.u.IndConferRsp.wTerminalListCount; i++) { H245ConferenceResponseCallbackParams.pTerminalList[i].bMCUNumber = (BYTE)pH245ConfIndData->u.Indication.u.IndConferRsp.pTerminalList[i].mcuNumber; H245ConferenceResponseCallbackParams.pTerminalList[i].bMCUNumber = (BYTE)pH245ConfIndData->u.Indication.u.IndConferRsp.pTerminalList[i].terminalNumber; } H245ConferenceResponseCallbackParams.wTerminalListCount = pH245ConfIndData->u.Indication.u.IndConferRsp.wTerminalListCount; status = CC_OK; } } status = InvokeUserConferenceCallback(pConference, CC_H245_CONFERENCE_RESPONSE_INDICATION, status, &H245ConferenceResponseCallbackParams); if (status != CC_OK) status = H245_ERROR_NOSUP; if (H245ConferenceResponseCallbackParams.pTerminalList != NULL) MemFree(H245ConferenceResponseCallbackParams.pTerminalList); break; }
if (ValidateCall(hCall) == CC_OK) UnlockCall(pCall); if (ValidateConference(hConference) == CC_OK) UnlockConference(pConference); return status; }
HRESULT _IndConferenceCommand( H245_CONF_IND_T *pH245ConfIndData) { CC_HCALL hCall; PCALL pCall; PCALL pOldCall; PCONFERENCE pConference; CC_HCONFERENCE hConference; WORD i; WORD wNumCalls; PCC_HCALL CallList; WORD wNumChannels; PCC_HCHANNEL ChannelList; PCHANNEL pChannel; CC_HCHANNEL hChannel; CALLSTATE CallState; HQ931CALL hQ931Call; H245_INST_T H245Instance; HRESULT status = CC_OK; CC_H245_CONFERENCE_COMMAND_CALLBACK_PARAMS H245ConferenceCommandCallbackParams;
hCall = pH245ConfIndData->u.Indication.dwPreserved; if (LockCallAndConference(hCall, &pCall, &pConference) != CC_OK) return H245_ERROR_OK;
hConference = pConference->hConference;
switch (pH245ConfIndData->u.Indication.u.IndConferCmd.CommandType) { case H245_CMD_DROP_CONFERENCE: if ((pConference->ConferenceMode == MULTIPOINT_MODE) && (pConference->tsMultipointController == TS_TRUE)) { UnlockCall(pCall); EnumerateCallsInConference(&wNumCalls, &CallList, pConference, ALL_CALLS); for (i = 0; i < wNumCalls; i++) { if (LockCall(CallList[i], &pCall) == CC_OK) { hQ931Call = pCall->hQ931Call; H245Instance = pCall->H245Instance; CallState = pCall->CallState; FreeCall(pCall); switch (CallState) { case ENQUEUED: break;
case PLACED: case RINGING: Q931Hangup(hQ931Call, CC_REJECT_NORMAL_CALL_CLEARING); break;
default: H245ShutDown(H245Instance); Q931Hangup(hQ931Call, CC_REJECT_NORMAL_CALL_CLEARING); break; } } } if (CallList != NULL) MemFree(CallList);
EnumerateChannelsInConference(&wNumChannels, &ChannelList, pConference, ALL_CHANNELS); for (i = 0; i < wNumChannels; i++) { if (LockChannel(ChannelList[i], &pChannel) == CC_OK) FreeChannel(pChannel); } if (ChannelList != NULL) MemFree(ChannelList); InvokeUserConferenceCallback( pConference, CC_CONFERENCE_TERMINATION_INDICATION, CC_OK, NULL); if (ValidateConference(hConference) == CC_OK) { if (pConference->bDeferredDelete) FreeConference(pConference); else { ReInitializeConference(pConference); UnlockConference(pConference); } } return H245_ERROR_OK; } status = H245_ERROR_OK; break;
case brdcstMyLgclChnnl_chosen: case cnclBrdcstMyLgclChnnl_chosen: if (FindChannelInConference(pH245ConfIndData->u.Indication.u.IndConferCmd.Channel, FALSE, // remote channel number
RX_CHANNEL | PROXY_CHANNEL, hCall, &hChannel, pConference) != CC_OK) { UnlockCall(pCall); UnlockConference(pConference); return H245_ERROR_OK; }
if (LockChannel(hChannel, &pChannel) != CC_OK) { UnlockCall(pCall); UnlockConference(pConference); return H245_ERROR_OK; } if (pChannel->bChannelType == PROXY_CHANNEL) { ASSERT(pConference->ConferenceMode == MULTIPOINT_MODE); ASSERT(pConference->tsMultipointController == TS_TRUE); ASSERT(pChannel->bMultipointChannel == TRUE);
EnumerateCallsInConference(&wNumCalls, &CallList, pConference, ESTABLISHED_CALL); for (i = 0; i < wNumCalls; i++) { if (CallList[i] != hCall) { if (LockCall(CallList[i], &pOldCall) == CC_OK) { H245ConferenceCommand(pOldCall->H245Instance, pH245ConfIndData->u.Indication.u.IndConferCmd.CommandType, pChannel->wLocalChannelNumber, pH245ConfIndData->u.Indication.u.IndConferCmd.byMcuNumber, pH245ConfIndData->u.Indication.u.IndConferCmd.byTerminalNumber); UnlockCall(pOldCall); } } } MemFree(CallList); }
if (pChannel->tsAccepted == TS_TRUE) { H245ConferenceCommandCallbackParams.hCall = hCall; H245ConferenceCommandCallbackParams.hChannel = hChannel; if (pCall->pPeerParticipantInfo == NULL) { H245ConferenceCommandCallbackParams.InitiatorTerminalLabel.bMCUNumber = 255; H245ConferenceCommandCallbackParams.InitiatorTerminalLabel.bTerminalNumber = 255; } else H245ConferenceCommandCallbackParams.InitiatorTerminalLabel = pCall->pPeerParticipantInfo->ParticipantInfo.TerminalLabel; H245ConferenceCommandCallbackParams.CommandType = pH245ConfIndData->u.Indication.u.IndConferCmd.CommandType; H245ConferenceCommandCallbackParams.TerminalLabel.bMCUNumber = pH245ConfIndData->u.Indication.u.IndConferCmd.byMcuNumber; H245ConferenceCommandCallbackParams.TerminalLabel.bTerminalNumber = pH245ConfIndData->u.Indication.u.IndConferCmd.byTerminalNumber; status = InvokeUserConferenceCallback(pConference, CC_H245_CONFERENCE_COMMAND_INDICATION, CC_OK, &H245ConferenceCommandCallbackParams); if (status != CC_OK) status = H245_ERROR_NOSUP; } else status = H245_ERROR_OK;
if (ValidateChannel(hChannel) == CC_OK) UnlockChannel(pChannel); if (ValidateCall(hCall) == CC_OK) UnlockCall(pCall); if (ValidateConference(hConference) == CC_OK) UnlockConference(pConference); return status;
default: // Unrecognized conference command
// Pass it up to the client
H245ConferenceCommandCallbackParams.hCall = hCall; if (pCall->pPeerParticipantInfo == NULL) { H245ConferenceCommandCallbackParams.InitiatorTerminalLabel.bMCUNumber = 255; H245ConferenceCommandCallbackParams.InitiatorTerminalLabel.bTerminalNumber = 255; } else H245ConferenceCommandCallbackParams.InitiatorTerminalLabel = pCall->pPeerParticipantInfo->ParticipantInfo.TerminalLabel; H245ConferenceCommandCallbackParams.CommandType = pH245ConfIndData->u.Indication.u.IndConferCmd.CommandType; H245ConferenceCommandCallbackParams.TerminalLabel.bMCUNumber = pH245ConfIndData->u.Indication.u.IndConferCmd.byMcuNumber; H245ConferenceCommandCallbackParams.TerminalLabel.bTerminalNumber = pH245ConfIndData->u.Indication.u.IndConferCmd.byTerminalNumber; H245ConferenceCommandCallbackParams.hChannel = CC_INVALID_HANDLE; status = InvokeUserConferenceCallback(pConference, CC_H245_CONFERENCE_COMMAND_INDICATION, CC_OK, &H245ConferenceCommandCallbackParams); if (status != CC_OK) status = H245_ERROR_NOSUP; if (ValidateCall(hCall) == CC_OK) UnlockCall(pCall); if (ValidateConference(hConference) == CC_OK) UnlockConference(pConference); return status; }
if (ValidateCall(hCall) == CC_OK) UnlockCall(pCall); if (ValidateConference(hConference) == CC_OK) UnlockConference(pConference); return status; }
HRESULT _IndConference( H245_CONF_IND_T *pH245ConfIndData) { CC_HCALL hCall; PCALL pCall; PCALL pPeerCall; PCONFERENCE pConference; CC_HCONFERENCE hConference; H245_TERMINAL_LABEL_T H245TerminalLabel; PPARTICIPANTINFO pPeerParticipantInfo; HRESULT status; CC_HCALL hVirtualCall; PCALL pVirtualCall; CC_PEER_ADD_CALLBACK_PARAMS PeerAddCallbackParams; CC_PEER_DROP_CALLBACK_PARAMS PeerDropCallbackParams; CC_H245_CONFERENCE_INDICATION_CALLBACK_PARAMS H245ConferenceIndicationCallbackParams;
hCall = pH245ConfIndData->u.Indication.dwPreserved; if (LockCallAndConference(hCall, &pCall, &pConference) != CC_OK) return H245_ERROR_OK;
hConference = pConference->hConference;
switch (pH245ConfIndData->u.Indication.u.IndConfer.IndicationType) { case H245_IND_TERMINAL_NUMBER_ASSIGN: if (pConference->tsMultipointController == TS_FALSE) { pConference->LocalParticipantInfo.ParticipantInfo.TerminalLabel.bMCUNumber = pH245ConfIndData->u.Indication.u.IndConfer.byMcuNumber; pConference->LocalParticipantInfo.ParticipantInfo.TerminalLabel.bTerminalNumber = pH245ConfIndData->u.Indication.u.IndConfer.byTerminalNumber;
hConference = pConference->hConference; // Generate a CC_TERMINAL_NUMBER_ASSIGN callback
InvokeUserConferenceCallback(pConference, CC_TERMINAL_NUMBER_INDICATION, CC_OK, &pConference->LocalParticipantInfo.ParticipantInfo.TerminalLabel);
if (ValidateConference(hConference) == CC_OK) UnlockConference(pConference); UnlockCall(pCall); return H245_ERROR_OK; } status = H245_ERROR_OK; break;
case H245_IND_TERMINAL_JOINED: if (pConference->tsMultipointController == TS_FALSE) { if ((pH245ConfIndData->u.Indication.u.IndConfer.byMcuNumber == pConference->LocalParticipantInfo.ParticipantInfo.TerminalLabel.bMCUNumber) && (pH245ConfIndData->u.Indication.u.IndConfer.byTerminalNumber == pConference->LocalParticipantInfo.ParticipantInfo.TerminalLabel.bTerminalNumber)) { // This message refers to us
status = H245_ERROR_OK; break; }
H245TerminalLabel.mcuNumber = pH245ConfIndData->u.Indication.u.IndConfer.byMcuNumber; H245TerminalLabel.terminalNumber = pH245ConfIndData->u.Indication.u.IndConfer.byTerminalNumber; FindPeerParticipantInfo(H245TerminalLabel, pConference, VIRTUAL_CALL, &pPeerCall); if (pPeerCall != NULL) { // We already know this peer's terminal label, and we
// eithet know its terminal ID or we have a pending request
// to obtain it
UnlockCall(pPeerCall); status = H245_ERROR_OK; break; }
// We don't know about this peer.
// Create a virtual call object for it, and issue a request
// for its terminal ID
status = AllocAndLockCall(&hVirtualCall, pConference->hConference, CC_INVALID_HANDLE, // hQ931Call
CC_INVALID_HANDLE, // hQ931CallInvitor,
NULL, // pLocalAliasNames,
NULL, // pPeerAliasNames,
NULL, // pPeerExtraAliasNames
NULL, // pPeerExtension
NULL, // pLocalNonStandardData,
NULL, // pPeerNonStandardData,
NULL, // pszLocalDisplay,
NULL, // pszPeerDisplay,
NULL, // pPeerVendorInfo,
NULL, // pQ931LocalConnectAddr,
NULL, // pQ931PeerConnectAddr,
NULL, // pQ931DestinationAddr,
NULL, // pSourceCallSignalAddress
VIRTUAL, // CallType,
FALSE, // bCallerIsMC,
0, // dwUserToken,
CALL_COMPLETE, // InitialCallState,
NULL, // no CallIdentifier
&pConference->ConferenceID, &pVirtualCall); if (status == CC_OK) { status = AllocatePeerParticipantInfo(NULL, &pPeerParticipantInfo); if (status == CC_OK) { pVirtualCall->pPeerParticipantInfo = pPeerParticipantInfo; pPeerParticipantInfo->ParticipantInfo.TerminalLabel.bMCUNumber = (BYTE)H245TerminalLabel.mcuNumber; pPeerParticipantInfo->ParticipantInfo.TerminalLabel.bTerminalNumber = (BYTE)H245TerminalLabel.terminalNumber; AddVirtualCallToConference(pVirtualCall, pConference); // Send RequestTerminalID
H245ConferenceRequest(pCall->H245Instance, H245_REQ_TERMINAL_ID, (BYTE)H245TerminalLabel.mcuNumber, (BYTE)H245TerminalLabel.terminalNumber); pPeerParticipantInfo->TerminalIDState = TERMINAL_ID_REQUESTED;
// Generate PEER_ADD callback
PeerAddCallbackParams.hCall = hVirtualCall; PeerAddCallbackParams.TerminalLabel = pVirtualCall->pPeerParticipantInfo->ParticipantInfo.TerminalLabel; PeerAddCallbackParams.pPeerTerminalID = NULL; InvokeUserConferenceCallback(pConference, CC_PEER_ADD_INDICATION, CC_OK, &PeerAddCallbackParams); if (ValidateCall(hVirtualCall) == CC_OK) UnlockCall(pVirtualCall); if (ValidateConference(hConference) == CC_OK) UnlockConference(pConference); UnlockCall(pCall); return H245_ERROR_OK; } else FreeCall(pVirtualCall); } } status = H245_ERROR_OK; break;
case H245_IND_TERMINAL_LEFT: if (pConference->tsMultipointController == TS_FALSE) { H245TerminalLabel.mcuNumber = pH245ConfIndData->u.Indication.u.IndConfer.byMcuNumber; H245TerminalLabel.terminalNumber = pH245ConfIndData->u.Indication.u.IndConfer.byTerminalNumber; status = FindPeerParticipantInfo(H245TerminalLabel, pConference, VIRTUAL_CALL, &pVirtualCall); if (status == CC_OK) { ASSERT(pVirtualCall != NULL); ASSERT(pVirtualCall->pPeerParticipantInfo != NULL); // Save the virtual call handle; we'll need to validate the virtual
// call object after returning from the conference callback
hVirtualCall = pVirtualCall->hCall; PeerDropCallbackParams.hCall = hVirtualCall; PeerDropCallbackParams.TerminalLabel = pVirtualCall->pPeerParticipantInfo->ParticipantInfo.TerminalLabel; if (pVirtualCall->pPeerParticipantInfo->TerminalIDState == TERMINAL_ID_VALID) PeerDropCallbackParams.pPeerTerminalID = &pVirtualCall->pPeerParticipantInfo->ParticipantInfo.TerminalID; else PeerDropCallbackParams.pPeerTerminalID = NULL; } else { // Set pVirtualCall to NULL to indicate that we don't have
// a virtual call object that needs to be free'd up later
pVirtualCall = NULL; PeerDropCallbackParams.hCall = CC_INVALID_HANDLE; PeerDropCallbackParams.TerminalLabel.bMCUNumber = pH245ConfIndData->u.Indication.u.IndConfer.byMcuNumber; PeerDropCallbackParams.TerminalLabel.bTerminalNumber = pH245ConfIndData->u.Indication.u.IndConfer.byTerminalNumber; PeerDropCallbackParams.pPeerTerminalID = NULL; }
hConference = pConference->hConference;
// Generate a CC_PEER_DROP_INDICATION callback
InvokeUserConferenceCallback(pConference, CC_PEER_DROP_INDICATION, CC_OK, &PeerDropCallbackParams); if (ValidateConference(hConference) == CC_OK) UnlockConference(pConference); // Check to see if we have a virtual call object that needs to be free'd up
if (pVirtualCall != NULL) if (ValidateCall(hVirtualCall) == CC_OK) FreeCall(pVirtualCall); UnlockCall(pCall); return H245_ERROR_OK; } status = H245_ERROR_OK; break;
default: H245ConferenceIndicationCallbackParams.hCall = hCall; if (pCall->pPeerParticipantInfo == NULL) { H245ConferenceIndicationCallbackParams.InitiatorTerminalLabel.bMCUNumber = 255; H245ConferenceIndicationCallbackParams.InitiatorTerminalLabel.bTerminalNumber = 255; } else H245ConferenceIndicationCallbackParams.InitiatorTerminalLabel = pCall->pPeerParticipantInfo->ParticipantInfo.TerminalLabel; H245ConferenceIndicationCallbackParams.IndicationType = pH245ConfIndData->u.Indication.u.IndConfer.IndicationType; H245ConferenceIndicationCallbackParams.bSBENumber = pH245ConfIndData->u.Indication.u.IndConfer.bySbeNumber; H245ConferenceIndicationCallbackParams.TerminalLabel.bMCUNumber = pH245ConfIndData->u.Indication.u.IndConfer.byMcuNumber; H245ConferenceIndicationCallbackParams.TerminalLabel.bTerminalNumber = pH245ConfIndData->u.Indication.u.IndConfer.byTerminalNumber; status = InvokeUserConferenceCallback(pConference, CC_H245_CONFERENCE_INDICATION_INDICATION, CC_OK, &H245ConferenceIndicationCallbackParams); if (status != CC_OK) status = H245_ERROR_NOSUP; break; } if (ValidateCall(hCall) == CC_OK) UnlockCall(pCall); if (ValidateConference(hConference) == CC_OK) UnlockConference(pConference); return status; }
HRESULT _IndCommunicationModeCommand( H245_CONF_IND_T *pH245ConfIndData) { CC_HCALL hCall; PCALL pCall; CC_HCONFERENCE hConference; PCONFERENCE pConference; CC_MULTIPOINT_CALLBACK_PARAMS MultipointCallbackParams;
hCall = pH245ConfIndData->u.Indication.dwPreserved; if (LockCallAndConference(hCall, &pCall, &pConference) != CC_OK) return H245_ERROR_OK;
if (pConference->tsMultipointController == TS_TRUE) { UnlockCall(pCall); UnlockConference(pConference); return H245_ERROR_OK; }
hConference = pConference->hConference;
// Destroy the old session table
FreeConferenceSessionTable(pConference);
H245CommunicationTableToSessionTable( pH245ConfIndData->u.Indication.u.IndCommRsp.pTable, pH245ConfIndData->u.Indication.u.IndCommRsp.byTableCount, &pConference->pSessionTable); pConference->bSessionTableInternallyConstructed = TRUE;
// Generate MULTIPOINT callback
MultipointCallbackParams.pTerminalInfo = &pConference->LocalParticipantInfo.ParticipantInfo; MultipointCallbackParams.pSessionTable = pConference->pSessionTable; InvokeUserConferenceCallback(pConference, CC_MULTIPOINT_INDICATION, CC_OK, &MultipointCallbackParams);
if (ValidateCall(hCall) == CC_OK) UnlockCall(pCall); if (ValidateConference(hConference) == CC_OK) UnlockConference(pConference); return H245_ERROR_OK; }
HRESULT _IndVendorIdentification( H245_CONF_IND_T *pH245ConfIndData, VendorIdentification *pVendorIdentification) { CC_HCALL hCall; PCALL pCall; CC_HCONFERENCE hConference; PCONFERENCE pConference; CC_NONSTANDARDDATA NonStandardData; CC_OCTETSTRING ProductNumber; CC_OCTETSTRING VersionNumber; CC_VENDOR_ID_CALLBACK_PARAMS VendorIDCallbackParams;
hCall = pH245ConfIndData->u.Indication.dwPreserved; if (LockCallAndConference(hCall, &pCall, &pConference) != CC_OK) return H245_ERROR_OK;
hConference = pConference->hConference;
VendorIDCallbackParams.hCall = hCall; if (pCall->pPeerParticipantInfo == NULL) { VendorIDCallbackParams.InitiatorTerminalLabel.bMCUNumber = 255; VendorIDCallbackParams.InitiatorTerminalLabel.bTerminalNumber = 255; } else VendorIDCallbackParams.InitiatorTerminalLabel = pCall->pPeerParticipantInfo->ParticipantInfo.TerminalLabel;
if (pVendorIdentification->vendor.choice == h221NonStandard_chosen) { NonStandardData.sData.pOctetString = NULL; NonStandardData.sData.wOctetStringLength = 0; NonStandardData.bCountryCode = (BYTE)pVendorIdentification->vendor.u.h221NonStandard.t35CountryCode; NonStandardData.bExtension = (BYTE)pVendorIdentification->vendor.u.h221NonStandard.t35Extension; NonStandardData.wManufacturerCode = pVendorIdentification->vendor.u.h221NonStandard.manufacturerCode; VendorIDCallbackParams.pNonStandardData = &NonStandardData; } else VendorIDCallbackParams.pNonStandardData = NULL;
if (pVendorIdentification->bit_mask & productNumber_present) { ProductNumber.pOctetString = pVendorIdentification->productNumber.value; ProductNumber.wOctetStringLength = (WORD) pVendorIdentification->productNumber.length; VendorIDCallbackParams.pProductNumber = &ProductNumber; } else VendorIDCallbackParams.pProductNumber = NULL; if (pVendorIdentification->bit_mask & versionNumber_present) { VersionNumber.pOctetString = pVendorIdentification->versionNumber.value; VersionNumber.wOctetStringLength = (WORD) pVendorIdentification->versionNumber.length; VendorIDCallbackParams.pVersionNumber = &VersionNumber; } else VendorIDCallbackParams.pVersionNumber = NULL;
InvokeUserConferenceCallback(pConference, CC_VENDOR_ID_INDICATION, CC_OK, &VendorIDCallbackParams); if (ValidateCall(hCall) == CC_OK) UnlockCall(pCall); if (ValidateConference(hConference) == CC_OK) UnlockConference(pConference); return H245_ERROR_OK; }
HRESULT _IndH2250MaximumSkew( H245_CONF_IND_T *pH245ConfIndData) { HRESULT status; CC_HCONFERENCE hConference; PCONFERENCE pConference; CC_HCALL hCall; PCC_HCALL CallList; WORD wNumCalls; WORD i; PCALL pCall; PCALL pOldCall; CC_HCHANNEL hChannel1; PCHANNEL pChannel1; CC_HCHANNEL hChannel2; PCHANNEL pChannel2; CC_MAXIMUM_AUDIO_VIDEO_SKEW_CALLBACK_PARAMS MaximumAudioVideoSkewCallbackParams;
hCall = pH245ConfIndData->u.Indication.dwPreserved; status = LockCallAndConference(hCall, &pCall, &pConference); if (status != CC_OK) { // This may be OK, if the call was cancelled while
// call setup was in progress.
return H245_ERROR_OK; }
hConference = pCall->hConference; UnlockCall(pCall);
if (FindChannelInConference(pH245ConfIndData->u.Indication.u.IndH2250MaxSkew.LogicalChannelNumber1, FALSE, // remote channel number
RX_CHANNEL | PROXY_CHANNEL, hCall, &hChannel1, pConference) != CC_OK) { UnlockConference(pConference); return H245_ERROR_OK; }
if (LockChannel(hChannel1, &pChannel1) != CC_OK) { UnlockConference(pConference); return H245_ERROR_OK; } if (pChannel1->bChannelType == RX_CHANNEL) { UnlockChannel(pChannel1); if (FindChannelInConference(pH245ConfIndData->u.Indication.u.IndH2250MaxSkew.LogicalChannelNumber2, FALSE, // remote channel number
RX_CHANNEL, hCall, &hChannel2, pConference) != CC_OK) { UnlockConference(pConference); return H245_ERROR_OK; } if (LockChannel(hChannel2, &pChannel2) != CC_OK) { UnlockConference(pConference); return H245_ERROR_OK; } if (pChannel2->bChannelType != RX_CHANNEL) { UnlockChannel(pChannel2); UnlockConference(pConference); return H245_ERROR_OK; } UnlockChannel(pChannel2);
MaximumAudioVideoSkewCallbackParams.hChannel1 = hChannel1; MaximumAudioVideoSkewCallbackParams.hChannel2 = hChannel2; MaximumAudioVideoSkewCallbackParams.wMaximumSkew = pH245ConfIndData->u.Indication.u.IndH2250MaxSkew.wSkew; InvokeUserConferenceCallback(pConference, CC_MAXIMUM_AUDIO_VIDEO_SKEW_INDICATION, CC_OK, &MaximumAudioVideoSkewCallbackParams); if (ValidateConference(hConference) == CC_OK) UnlockConference(pConference); } else { // pChannel1->bChannelType == PROXY_CHANNEL
if (FindChannelInConference(pH245ConfIndData->u.Indication.u.IndH2250MaxSkew.LogicalChannelNumber2, FALSE, // remote channel number
PROXY_CHANNEL, hCall, &hChannel2, pConference) != CC_OK) { UnlockChannel(pChannel1); UnlockConference(pConference); return H245_ERROR_OK; } if (LockChannel(hChannel2, &pChannel2) != CC_OK) { UnlockChannel(pChannel1); UnlockConference(pConference); return H245_ERROR_OK; } if (pChannel1->hCall != pChannel2->hCall) { UnlockChannel(pChannel1); UnlockChannel(pChannel2); UnlockConference(pConference); return H245_ERROR_OK; }
EnumerateCallsInConference(&wNumCalls, &CallList, pConference, ESTABLISHED_CALL); for (i = 0; i < wNumCalls; i++) { if (CallList[i] != hCall) { if (LockCall(CallList[i], &pOldCall) == CC_OK) { H245H2250MaximumSkewIndication(pOldCall->H245Instance, pChannel1->wLocalChannelNumber, pChannel2->wLocalChannelNumber, pH245ConfIndData->u.Indication.u.IndH2250MaxSkew.wSkew); UnlockCall(pCall); } } }
if (CallList != NULL) MemFree(CallList);
if ((pChannel1->tsAccepted == TS_TRUE) && (pChannel2->tsAccepted == TS_TRUE)) { MaximumAudioVideoSkewCallbackParams.hChannel1 = hChannel1; MaximumAudioVideoSkewCallbackParams.hChannel2 = hChannel2; MaximumAudioVideoSkewCallbackParams.wMaximumSkew = pH245ConfIndData->u.Indication.u.IndH2250MaxSkew.wSkew; InvokeUserConferenceCallback(pConference, CC_MAXIMUM_AUDIO_VIDEO_SKEW_INDICATION, CC_OK, &MaximumAudioVideoSkewCallbackParams); }
if (ValidateChannel(hChannel1) == CC_OK) UnlockChannel(pChannel1); if (ValidateChannel(hChannel2) == CC_OK) UnlockChannel(pChannel2); if (ValidateConference(hConference) == CC_OK) UnlockConference(pConference); } return H245_ERROR_OK; }
HRESULT _IndUserInput( H245_CONF_IND_T *pH245ConfIndData) { return H245_ERROR_OK; }
HRESULT _IndSendTerminalCapabilitySet( H245_CONF_IND_T *pH245ConfIndData) { HRESULT status; CC_HCALL hCall; PCALL pCall; PCONFERENCE pConference;
hCall = pH245ConfIndData->u.Indication.dwPreserved; status = LockCallAndConference(hCall, &pCall, &pConference); if (status != CC_OK) { // This may be OK, if the call was cancelled while
// call setup was in progress.
return H245_ERROR_OK; }
SendTermCaps(pCall, pConference); UnlockCall(pCall); UnlockConference(pConference); return H245_ERROR_OK; }
HRESULT _IndModeRequest( H245_CONF_IND_T *pH245ConfIndData) { HRESULT status; CC_HCALL hCall; PCALL pCall; CC_HCONFERENCE hConference; PCONFERENCE pConference; CC_REQUEST_MODE_CALLBACK_PARAMS RequestModeCallbackParams;
hCall = pH245ConfIndData->u.Indication.dwPreserved; status = LockCallAndConference(hCall, &pCall, &pConference); if (status != CC_OK) { // This may be OK, if the call was cancelled while
// call setup was in progress.
return H245_ERROR_OK; }
hConference = pConference->hConference;
EnqueueRequest(&pConference->pEnqueuedRequestModeCalls, hCall);
RequestModeCallbackParams.hCall = hCall; if (pCall->pPeerParticipantInfo == NULL) { RequestModeCallbackParams.InitiatorTerminalLabel.bMCUNumber = 255; RequestModeCallbackParams.InitiatorTerminalLabel.bTerminalNumber = 255; } else RequestModeCallbackParams.InitiatorTerminalLabel = pCall->pPeerParticipantInfo->ParticipantInfo.TerminalLabel; RequestModeCallbackParams.pRequestedModes = pH245ConfIndData->u.Indication.u.IndMrse.pRequestedModes;
InvokeUserConferenceCallback(pConference, CC_REQUEST_MODE_INDICATION, CC_OK, &RequestModeCallbackParams); if (ValidateCall(hCall) == CC_OK) UnlockCall(pCall); if (ValidateConference(hConference) == CC_OK) UnlockConference(pConference); return H245_ERROR_OK; }
HRESULT _ConfUnimplemented( H245_CONF_IND_T *pH245ConfIndData) { return H245_ERROR_NOSUP; }
HRESULT _ConfBiDirectionalOpen( H245_CONF_IND_T *pH245ConfIndData) { CC_HCALL hCall; CC_HCHANNEL hChannel; CC_HCONFERENCE hConference; PCHANNEL pChannel; PCONFERENCE pConference; BOOL bAccept; HRESULT status; CC_ADDR T120Addr; CC_OCTETSTRING ExternalReference; CC_T120_CHANNEL_OPEN_CALLBACK_PARAMS T120ChannelOpenCallbackParams;
hCall = pH245ConfIndData->u.Confirm.dwPreserved; if (hCall == CC_INVALID_HANDLE) return H245_ERROR_OK;
hChannel = pH245ConfIndData->u.Confirm.dwTransId; if (hChannel == CC_INVALID_HANDLE) return H245_ERROR_OK;
if (LockChannelAndConference(hChannel, &pChannel, &pConference) != CC_OK) return H245_ERROR_OK;
hConference = pConference->hConference;
if (pChannel->bChannelType != TXRX_CHANNEL) { UnlockChannel(pChannel); UnlockConference(pConference); return H245_ERROR_OK; }
if ((pH245ConfIndData->u.Confirm.u.ConfOpenNeedRsp.AccRej == H245_ACC) && (pH245ConfIndData->u.Confirm.Error == H245_ERROR_OK)) { pChannel->wNumOutstandingRequests = 0; bAccept = TRUE; } else { (pChannel->wNumOutstandingRequests)--; bAccept = FALSE; }
T120ChannelOpenCallbackParams.hChannel = hChannel; T120ChannelOpenCallbackParams.hCall = hCall; T120ChannelOpenCallbackParams.dwUserToken = pChannel->dwUserToken; T120ChannelOpenCallbackParams.dwRejectReason = 0;
if (bAccept) { status = CC_OK; if (pH245ConfIndData->u.Confirm.u.ConfOpenNeedRsp.pSeparateStack) { if ((pH245ConfIndData->u.Confirm.u.ConfOpenNeedRsp.pSeparateStack->networkAddress.choice == localAreaAddress_chosen) && (pH245ConfIndData->u.Confirm.u.ConfOpenNeedRsp.pSeparateStack->networkAddress.u.localAreaAddress.choice == unicastAddress_chosen) && (pH245ConfIndData->u.Confirm.u.ConfOpenNeedRsp.pSeparateStack->networkAddress.u.localAreaAddress.u.unicastAddress.choice == UnicastAddress_iPAddress_chosen)) { T120Addr.nAddrType = CC_IP_BINARY; T120Addr.bMulticast = FALSE; T120Addr.Addr.IP_Binary.wPort = pH245ConfIndData->u.Confirm.u.ConfOpenNeedRsp.pSeparateStack->networkAddress.u.localAreaAddress.u.unicastAddress.u.UnicastAddress_iPAddress.tsapIdentifier; H245IPNetworkToHost(&T120Addr.Addr.IP_Binary.dwAddr, pH245ConfIndData->u.Confirm.u.ConfOpenNeedRsp.pSeparateStack->networkAddress.u.localAreaAddress.u.unicastAddress.u.UnicastAddress_iPAddress.network.value); T120ChannelOpenCallbackParams.pAddr = &T120Addr; } else { T120ChannelOpenCallbackParams.pAddr = NULL; } T120ChannelOpenCallbackParams.bAssociateConference = pH245ConfIndData->u.Confirm.u.ConfOpenNeedRsp.pSeparateStack->associateConference; if (pH245ConfIndData->u.Confirm.u.ConfOpenNeedRsp.pSeparateStack->bit_mask & externalReference_present) { ExternalReference.wOctetStringLength = (WORD) pH245ConfIndData->u.Confirm.u.ConfOpenNeedRsp.pSeparateStack->externalReference.length; ExternalReference.pOctetString = pH245ConfIndData->u.Confirm.u.ConfOpenNeedRsp.pSeparateStack->externalReference.value; T120ChannelOpenCallbackParams.pExternalReference = &ExternalReference; } else T120ChannelOpenCallbackParams.pExternalReference = NULL; } else { T120ChannelOpenCallbackParams.pAddr = NULL; T120ChannelOpenCallbackParams.bAssociateConference = FALSE; T120ChannelOpenCallbackParams.pExternalReference = NULL; } } else { // bAccept == FALSE
if (pH245ConfIndData->u.Confirm.Error == H245_ERROR_OK) status = CC_PEER_REJECT; else status = pH245ConfIndData->u.Confirm.Error; T120ChannelOpenCallbackParams.pAddr = NULL; T120ChannelOpenCallbackParams.bAssociateConference = FALSE; T120ChannelOpenCallbackParams.pExternalReference = NULL; T120ChannelOpenCallbackParams.dwRejectReason = pH245ConfIndData->u.Confirm.u.ConfOpenNeedRsp.AccRej; }
InvokeUserConferenceCallback(pConference, CC_T120_CHANNEL_OPEN_INDICATION, status, &T120ChannelOpenCallbackParams);
if (ValidateChannel(hChannel) == CC_OK) if (bAccept) UnlockChannel(pChannel); else FreeChannel(pChannel); if (ValidateConference(hConference) == CC_OK) UnlockConference(pConference);
return H245_ERROR_OK; }
HRESULT _ConfOpenT120( H245_CONF_IND_T *pH245ConfIndData) { CC_HCALL hCall; CC_HCHANNEL hChannel; CC_HCONFERENCE hConference; PCHANNEL pChannel; PCONFERENCE pConference; HRESULT status; CC_T120_CHANNEL_OPEN_CALLBACK_PARAMS T120ChannelOpenCallbackParams;
hCall = pH245ConfIndData->u.Confirm.dwPreserved; if (hCall == CC_INVALID_HANDLE) return H245_ERROR_OK;
hChannel = pH245ConfIndData->u.Confirm.dwTransId; if (hChannel == CC_INVALID_HANDLE) return H245_ERROR_OK;
if (LockChannelAndConference(hChannel, &pChannel, &pConference) != CC_OK) return H245_ERROR_OK;
hConference = pConference->hConference;
if (pChannel->bChannelType != TXRX_CHANNEL) { UnlockChannel(pChannel); UnlockConference(pConference); return H245_ERROR_OK; }
if ((pH245ConfIndData->u.Confirm.u.ConfOpen.AccRej == H245_ACC) && (pH245ConfIndData->u.Confirm.Error == H245_ERROR_OK)) { // We expect to get a ConfOpenNeedRsp callback for this case;
// Since we're not sure how we got here, just bail out
UnlockChannel(pChannel); UnlockConference(pConference); return H245_ERROR_OK; }
T120ChannelOpenCallbackParams.hChannel = hChannel; T120ChannelOpenCallbackParams.hCall = hCall; T120ChannelOpenCallbackParams.dwUserToken = pChannel->dwUserToken;
if (pH245ConfIndData->u.Confirm.Error == H245_ERROR_OK) status = CC_PEER_REJECT; else status = pH245ConfIndData->u.Confirm.Error; T120ChannelOpenCallbackParams.pAddr = NULL; T120ChannelOpenCallbackParams.bAssociateConference = FALSE; T120ChannelOpenCallbackParams.pExternalReference = NULL; T120ChannelOpenCallbackParams.dwRejectReason = pH245ConfIndData->u.Confirm.u.ConfOpenNeedRsp.AccRej;
InvokeUserConferenceCallback(pConference, CC_T120_CHANNEL_OPEN_INDICATION, status, &T120ChannelOpenCallbackParams);
if (ValidateChannel(hChannel) == CC_OK) FreeChannel(pChannel); if (ValidateConference(hConference) == CC_OK) UnlockConference(pConference);
return H245_ERROR_OK; }
HRESULT _ConfOpen( H245_CONF_IND_T *pH245ConfIndData) { HRESULT status; CC_ADDR PeerRTPAddr; PCC_ADDR pPeerRTPAddr; CC_ADDR PeerRTCPAddr; PCC_ADDR pPeerRTCPAddr; CC_HCHANNEL hChannel; PCHANNEL pChannel; CC_HCONFERENCE hConference; PCONFERENCE pConference; CC_TX_CHANNEL_OPEN_CALLBACK_PARAMS TxChannelOpenCallbackParams; PCALL pCall; BOOL bAccept; H245_MUX_T H245MuxTable; WORD i; #ifdef GATEKEEPER
unsigned uBandwidth; WORD wNumCalls; PCC_HCALL CallList; #endif // GATEKEEPER
// a channel was opened
hChannel = pH245ConfIndData->u.Confirm.dwTransId; if (hChannel == CC_INVALID_HANDLE) return H245_ERROR_OK;
if (LockChannelAndConference(hChannel, &pChannel, &pConference) != CC_OK) return H245_ERROR_OK;
if (pChannel->bChannelType == TXRX_CHANNEL) { UnlockChannel(pChannel); UnlockConference(pConference); return _ConfOpenT120(pH245ConfIndData); }
hConference = pConference->hConference;
if (pChannel->wNumOutstandingRequests == 0) { UnlockChannel(pChannel); UnlockConference(pConference); return H245_ERROR_OK; }
if ((pH245ConfIndData->u.Confirm.u.ConfOpen.AccRej == H245_ACC) && (pH245ConfIndData->u.Confirm.Error == H245_ERROR_OK)) { pChannel->wNumOutstandingRequests = 0; bAccept = TRUE; } else { (pChannel->wNumOutstandingRequests)--; bAccept = FALSE; #ifdef GATEKEEPER
if(GKIExists()) { uBandwidth = pChannel->dwChannelBitRate / 100; if (uBandwidth != 0 && pChannel->bChannelType != TXRX_CHANNEL) { EnumerateCallsInConference(&wNumCalls, &CallList, pConference, ESTABLISHED_CALL); for (i = 0; i < wNumCalls; ++i) { if (LockCall(CallList[i], &pCall) == CC_OK) { if (pCall->GkiCall.uBandwidthUsed >= uBandwidth) { if (GkiCloseChannel(&pCall->GkiCall, pChannel->dwChannelBitRate, hChannel) == CC_OK) { UnlockCall(pCall); break; } } UnlockCall(pCall); } } // for
if (CallList != NULL) MemFree(CallList); } } #endif // GATEKEEPER
} if (pChannel->wNumOutstandingRequests == 0) {
if (pH245ConfIndData->u.Confirm.u.ConfOpen.pTxMux == NULL) { pPeerRTPAddr = NULL; pPeerRTCPAddr = NULL; } else { ASSERT(pH245ConfIndData->u.Confirm.u.ConfOpen.pTxMux->Kind == H245_H2250ACK); if ((pH245ConfIndData->u.Confirm.u.ConfOpen.pTxMux->u.H2250ACK.mediaChannelPresent) && ((pH245ConfIndData->u.Confirm.u.ConfOpen.pTxMux->u.H2250ACK.mediaChannel.type == H245_IP_MULTICAST) || (pH245ConfIndData->u.Confirm.u.ConfOpen.pTxMux->u.H2250ACK.mediaChannel.type == H245_IP_UNICAST))) { pPeerRTPAddr = &PeerRTPAddr; PeerRTPAddr.nAddrType = CC_IP_BINARY; if (pH245ConfIndData->u.Confirm.u.ConfOpen.pTxMux->u.H2250ACK.mediaChannel.type == H245_IP_MULTICAST) PeerRTPAddr.bMulticast = TRUE; else PeerRTPAddr.bMulticast = FALSE; H245IPNetworkToHost(&PeerRTPAddr.Addr.IP_Binary.dwAddr, pH245ConfIndData->u.Confirm.u.ConfOpen.pTxMux->u.H2250ACK.mediaChannel.u.ip.network); PeerRTPAddr.Addr.IP_Binary.wPort = pH245ConfIndData->u.Confirm.u.ConfOpen.pTxMux->u.H2250ACK.mediaChannel.u.ip.tsapIdentifier; } else pPeerRTPAddr = NULL;
if ((pH245ConfIndData->u.Confirm.u.ConfOpen.pTxMux->u.H2250ACK.mediaControlChannelPresent) && ((pH245ConfIndData->u.Confirm.u.ConfOpen.pTxMux->u.H2250ACK.mediaControlChannel.type == H245_IP_MULTICAST) || (pH245ConfIndData->u.Confirm.u.ConfOpen.pTxMux->u.H2250ACK.mediaControlChannel.type == H245_IP_UNICAST))) { pPeerRTCPAddr = &PeerRTCPAddr; PeerRTCPAddr.nAddrType = CC_IP_BINARY; if (pH245ConfIndData->u.Confirm.u.ConfOpen.pTxMux->u.H2250ACK.mediaControlChannel.type == H245_IP_MULTICAST) PeerRTCPAddr.bMulticast = TRUE; else PeerRTCPAddr.bMulticast = FALSE; H245IPNetworkToHost(&PeerRTCPAddr.Addr.IP_Binary.dwAddr, pH245ConfIndData->u.Confirm.u.ConfOpen.pTxMux->u.H2250ACK.mediaControlChannel.u.ip.network); PeerRTCPAddr.Addr.IP_Binary.wPort = pH245ConfIndData->u.Confirm.u.ConfOpen.pTxMux->u.H2250ACK.mediaControlChannel.u.ip.tsapIdentifier; } else pPeerRTCPAddr = NULL; }
if ((pPeerRTPAddr == NULL) || (pPeerRTCPAddr == NULL)) { if (pConference->pSessionTable != NULL) { for (i = 0; i < pConference->pSessionTable->wLength; i++) { if (pConference->pSessionTable->SessionInfoArray[i].bSessionID == pChannel->bSessionID) { if (pPeerRTPAddr == NULL) pPeerRTPAddr = pConference->pSessionTable->SessionInfoArray[i].pRTPAddr; if (pPeerRTCPAddr == NULL) pPeerRTCPAddr = pConference->pSessionTable->SessionInfoArray[i].pRTCPAddr; break; } } } }
if ((pChannel->pPeerRTPAddr == NULL) && (pPeerRTPAddr != NULL)) CopyAddr(&pChannel->pPeerRTPAddr, pPeerRTPAddr); if ((pChannel->pPeerRTCPAddr == NULL) && (pPeerRTCPAddr != NULL)) CopyAddr(&pChannel->pPeerRTCPAddr, pPeerRTCPAddr);
if (pChannel->bChannelType == PROXY_CHANNEL) { if (LockCall(pChannel->hCall, &pCall) == CC_OK) { if (bAccept) { H245MuxTable.Kind = H245_H2250ACK; H245MuxTable.u.H2250ACK.nonStandardList = NULL;
if (pPeerRTPAddr != NULL) { if (pPeerRTPAddr->bMulticast) H245MuxTable.u.H2250ACK.mediaChannel.type = H245_IP_MULTICAST; else H245MuxTable.u.H2250ACK.mediaChannel.type = H245_IP_UNICAST; H245MuxTable.u.H2250ACK.mediaChannel.u.ip.tsapIdentifier = pPeerRTPAddr->Addr.IP_Binary.wPort; HostToH245IPNetwork(H245MuxTable.u.H2250ACK.mediaChannel.u.ip.network, pPeerRTPAddr->Addr.IP_Binary.dwAddr); H245MuxTable.u.H2250ACK.mediaChannelPresent = TRUE; } else H245MuxTable.u.H2250ACK.mediaChannelPresent = FALSE;
if (pPeerRTCPAddr != NULL) { if (pPeerRTCPAddr->bMulticast) H245MuxTable.u.H2250ACK.mediaControlChannel.type = H245_IP_MULTICAST; else H245MuxTable.u.H2250ACK.mediaControlChannel.type = H245_IP_UNICAST; H245MuxTable.u.H2250ACK.mediaControlChannel.u.ip.tsapIdentifier = pPeerRTCPAddr->Addr.IP_Binary.wPort; HostToH245IPNetwork(H245MuxTable.u.H2250ACK.mediaControlChannel.u.ip.network, pPeerRTCPAddr->Addr.IP_Binary.dwAddr); H245MuxTable.u.H2250ACK.mediaControlChannelPresent = TRUE; } else H245MuxTable.u.H2250ACK.mediaControlChannelPresent = FALSE;
H245MuxTable.u.H2250ACK.dynamicRTPPayloadTypePresent = FALSE; H245MuxTable.u.H2250ACK.sessionIDPresent = TRUE; H245MuxTable.u.H2250ACK.sessionID = pChannel->bSessionID; status = H245OpenChannelAccept(pCall->H245Instance, 0, // dwTransId
pChannel->wRemoteChannelNumber, // Rx channel
&H245MuxTable, 0, // Tx channel
NULL, // Tx mux
H245_INVALID_PORT_NUMBER,// Port
NULL); } else { // bAccept == FALSE
status = H245OpenChannelReject(pCall->H245Instance, pChannel->wRemoteChannelNumber, // Rx channel
(unsigned short)pH245ConfIndData->u.Confirm.u.ConfOpen.AccRej); // rejection reason
} UnlockCall(pCall); } }
TxChannelOpenCallbackParams.hChannel = hChannel; TxChannelOpenCallbackParams.pPeerRTPAddr = pPeerRTPAddr; TxChannelOpenCallbackParams.pPeerRTCPAddr = pPeerRTCPAddr; TxChannelOpenCallbackParams.dwUserToken = pChannel->dwUserToken;
if (bAccept) { status = CC_OK; TxChannelOpenCallbackParams.dwRejectReason = H245_ACC; } else { // bAccept = FALSE
if (pH245ConfIndData->u.Confirm.Error == H245_ERROR_OK) status = CC_PEER_REJECT; else status = pH245ConfIndData->u.Confirm.Error; TxChannelOpenCallbackParams.dwRejectReason = pH245ConfIndData->u.Confirm.u.ConfOpen.AccRej; }
if ((pChannel->bCallbackInvoked == FALSE) && ((pChannel->bChannelType == TX_CHANNEL) || ((pChannel->bChannelType == TXRX_CHANNEL) && (pChannel->bLocallyOpened == TRUE)))) { pChannel->bCallbackInvoked = TRUE;
InvokeUserConferenceCallback(pConference, CC_TX_CHANNEL_OPEN_INDICATION, status, &TxChannelOpenCallbackParams); }
if (ValidateChannel(hChannel) == CC_OK) if (bAccept) UnlockChannel(pChannel); else FreeChannel(pChannel); } else UnlockChannel(pChannel);
if (ValidateConference(hConference) == CC_OK) UnlockConference(pConference); return H245_ERROR_OK; }
HRESULT _ConfClose( H245_CONF_IND_T *pH245ConfIndData) {
CC_HCALL hCall; CC_HCHANNEL hChannel; PCHANNEL pChannel; CC_HCONFERENCE hConference; PCONFERENCE pConference; PCALL pCall; H245_ACC_REJ_T AccRej;
hCall = pH245ConfIndData->u.Confirm.dwPreserved; if (LockCallAndConference(hCall, &pCall, &pConference) != CC_OK) return H245_ERROR_OK;
hConference = pCall->hConference; UnlockCall(pCall);
if (pH245ConfIndData->u.Confirm.Error != H245_ERROR_OK) { // TBD - Report error to Call Control client
// but wait! CC_CloseChannel() is a synchronous API! Until/unless that
// changes, the buck stops here.
if (FindChannelInConference(pH245ConfIndData->u.Confirm.u.ConfReqClose.Channel, TRUE, // local channel number
TX_CHANNEL | PROXY_CHANNEL, hCall, &hChannel, pConference) != CC_OK) { UnlockConference(pConference); return H245_ERROR_OK; } if (LockChannel(hChannel, &pChannel) != CC_OK) { UnlockConference(pConference); return H245_ERROR_OK; } // NOTE STOPGAP MEASURE : short term intentional "leak" of channel number.
// The channel number is actually a bit in a per-conference bitmap, so there
// is no real memory leak.
// This case is rare. The most likely error that leads here is a timeout.
// Calling FreeChannel() will normally recycle the logical channel
// number, and a new channel could reuse this number very quickly. If the error
// is a timeout, chances are that a late CloseLogicalChannelAck is on its
// way up the wire. We don't want that late CloseLogicalChannelAck to be
// associated with a completely new unrelated channel.
// set channel number to zero so that FreeChannel() does not recycle the number
pChannel->wLocalChannelNumber = 0;
FreeChannel(pChannel); UnlockConference(pConference);
} else { if(pH245ConfIndData->u.Confirm.u.ConfClose.AccRej == H245_ACC) { if (FindChannelInConference(pH245ConfIndData->u.Confirm.u.ConfReqClose.Channel, TRUE, // local channel number
TX_CHANNEL | PROXY_CHANNEL, hCall, &hChannel, pConference) != CC_OK) { UnlockConference(pConference); return H245_ERROR_OK; } if (LockChannel(hChannel, &pChannel) != CC_OK) { UnlockConference(pConference); return H245_ERROR_OK; } FreeChannel(pChannel); UnlockConference(pConference); } else { // At the time the ASSERT(0) was added here, the path that leads here
// always set pH245ConfIndData->u.Confirm.u.ConfClose.AccRej = H245_ACC
// at the same point it set ConfInd.u.Confirm.Error = H245_ERROR_OK;
// if that is ever changed, this also needs to change.
// see ..\h245\src\api_up.c, function H245FsmConfirm(), case H245_CONF_CLOSE:
ASSERT(0); }
} return H245_ERROR_OK; }
HRESULT _ConfRequestClose( H245_CONF_IND_T *pH245ConfIndData) { CC_HCALL hCall; CC_HCHANNEL hChannel; PCHANNEL pChannel; CC_HCONFERENCE hConference; PCONFERENCE pConference; PCALL pCall; H245_ACC_REJ_T AccRej;
hCall = pH245ConfIndData->u.Confirm.dwPreserved; if (LockCallAndConference(hCall, &pCall, &pConference) != CC_OK) return H245_ERROR_OK;
hConference = pCall->hConference; UnlockCall(pCall);
if (pH245ConfIndData->u.Confirm.Error == H245_ERROR_OK) AccRej = pH245ConfIndData->u.Confirm.u.ConfReqClose.AccRej; else AccRej = H245_REJ;
// Note: the only time we need to take any real action is when the channel
// is a proxy channel, and the local endpoint is not the one which requested
// the channel closure; in this case, we simply forward the closure response
// on to the endpoint which initiated the request.
// If the channel is an RX or TXRX channel, the channel object was deleted
// when our client requested the channel closure, so there's no real work to
// be done.
// If the channel is a proxy channel which our client requested be closed,
// the channel object will remain around until closed by the TX side, but we
// don't need (nor do we have a mechanism) to inform our client of receipt
// of this channel closure response.
if (FindChannelInConference(pH245ConfIndData->u.Confirm.u.ConfReqClose.Channel, FALSE, // remote channel number
PROXY_CHANNEL, hCall, &hChannel, pConference) != CC_OK) { UnlockConference(pConference); return H245_ERROR_OK; }
// Set hCall to the peer which initiated the close channel request
hCall = pH245ConfIndData->u.Confirm.dwTransId; if (hCall == CC_INVALID_HANDLE) { // The local endpoint was the one who requested the channel closure,
// so there's no one to forwards this response onto. We don't provide
// a callback for informing our client of receipt of this response,
// so we can simply clean up and return
UnlockConference(pConference); return H245_ERROR_OK; }
if (LockChannel(hChannel, &pChannel) != CC_OK) { UnlockConference(pConference); return H245_ERROR_OK; }
// Forward this response onto the endpoint which requested the channel closure
if (LockCall(hCall, &pCall) == CC_OK) { H245CloseChannelReqResp(pCall->H245Instance, AccRej, pChannel->wLocalChannelNumber); UnlockCall(pCall); }
UnlockChannel(pChannel); UnlockConference(pConference); return H245_ERROR_OK; }
#if 0
HRESULT _ConfShutdown( H245_CONF_IND_T *pH245ConfIndData) { CC_HCALL hCall; PCALL pCall; CC_HCONFERENCE hConference; PCONFERENCE pConference; HRESULT status; HQ931CALL hQ931Call; H245_INST_T H245Instance;
#if 1
// Sync 2 - specific code
hCall = pH245ConfIndData->u.Confirm.dwPreserved; if (LockCallAndConference(hCall, &pCall, &pConference) != CC_OK) return H245_ERROR_OK;
hConference = pCall->hConference;
if (pConference->tsMultipointController == TS_TRUE) { // XXX -- invoke user callback with "peer drop indication"
} else { H245Instance = pCall->H245Instance; hQ931Call = pCall->hQ931Call; FreeCall(pCall);
if (H245Instance != H245_INVALID_ID) status = H245ShutDown(H245Instance); else status = H245_ERROR_OK;
if (status == H245_ERROR_OK) { status = Q931Hangup(hQ931Call, CC_REJECT_UNDEFINED_REASON); // Q931Hangup may legitimately return CS_BAD_PARAM, because the Q.931 call object
// may have been deleted at this point
if (status == CS_BAD_PARAM) status = CC_OK; } else Q931Hangup(hQ931Call, CC_REJECT_UNDEFINED_REASON);
InvokeUserConferenceCallback(pConference, CC_CONFERENCE_TERMINATION_INDICATION, status, NULL);
if (ValidateConference(hConference) == CC_OK) UnlockConference(pConference);
return H245_ERROR_OK; } #else
// Probably sync 3 code
HHANGUP hHangup; PHANGUP pHangup; CC_HANGUP_CALLBACK_PARAMS HangupCallbackParams;
hHangup = pH245ConfIndData->u.Confirm.dwTransId; if (hHangup == CC_INVALID_HANDLE) return H245_ERROR_OK;
if (LockHangup(hHangup, &pHangup) != CC_OK) return H245_ERROR_OK;
pHangup->wNumCalls--; if (pHangup->wNumCalls == 0) { hConference = pHangup->hConference; if (LockConference(hConference, &pConference) != CC_OK) { UnlockHangup(pHangup); return H245_ERROR_OK; } HangupCallbackParams.dwUserToken = pHangup->dwUserToken; InvokeUserConferenceCallback(pConference->ConferenceCallback, CC_HANGUP_INDICATION, CC_OK, hConference, pConference->dwConferenceToken, &HangupCallbackParams); if (ValidateConference(hConference) == CC_OK) UnlockConference(pConference); if (ValidateHangup(hHangup) == CC_OK) FreeHangup(pHangup); return H245_ERROR_OK; } else UnlockHangup(pHangup); return H245_ERROR_OK; #endif // Sync 3 code
}
#endif
HRESULT _ConfInitMstslv( H245_CONF_IND_T *pH245ConfIndData) { CC_HCALL hCall; PCALL pCall; PCONFERENCE pConference; CC_CONNECT_CALLBACK_PARAMS ConnectCallbackParams; CC_HCALL hEnqueuedCall; PCALL pEnqueuedCall; CC_HCONFERENCE hConference; HRESULT status;
hCall = pH245ConfIndData->u.Confirm.dwPreserved; if (LockCallAndConference(hCall, &pCall, &pConference) != CC_OK) return H245_ERROR_OK;
ASSERT(pCall->MasterSlaveState != MASTER_SLAVE_COMPLETE);
switch (pH245ConfIndData->u.Confirm.u.ConfMstSlv) { case H245_MASTER: pConference->tsMaster = TS_TRUE; if (pConference->tsMultipointController == TS_UNKNOWN) { ASSERT(pConference->bMultipointCapable == TRUE); pConference->tsMultipointController = TS_TRUE;
// place all calls enqueued on this conference object
for ( ; ; ) { // Start up all enqueued calls, if any exist
status = RemoveEnqueuedCallFromConference(pConference, &hEnqueuedCall); if ((status != CC_OK) || (hEnqueuedCall == CC_INVALID_HANDLE)) break;
status = LockCall(hEnqueuedCall, &pEnqueuedCall); if (status == CC_OK) { pEnqueuedCall->CallState = PLACED;
status = PlaceCall(pEnqueuedCall, pConference); UnlockCall(pEnqueuedCall); } } } break;
case H245_SLAVE: ASSERT(pConference->tsMaster != TS_TRUE); ASSERT(pConference->tsMultipointController != TS_TRUE); pConference->tsMaster = TS_FALSE; pConference->tsMultipointController = TS_FALSE;
// XXX -- we may eventually want to re-enqueue these requests
// and set an expiration timer
hConference = pConference->hConference; for ( ; ; ) { status = RemoveEnqueuedCallFromConference(pConference, &hEnqueuedCall); if ((status != CC_OK) || (hEnqueuedCall == CC_INVALID_HANDLE)) break;
status = LockCall(hEnqueuedCall, &pEnqueuedCall); if (status == CC_OK) { MarkCallForDeletion(pEnqueuedCall); ConnectCallbackParams.pNonStandardData = pEnqueuedCall->pPeerNonStandardData; ConnectCallbackParams.pszPeerDisplay = pEnqueuedCall->pszPeerDisplay; ConnectCallbackParams.bRejectReason = CC_REJECT_UNDEFINED_REASON; ConnectCallbackParams.pTermCapList = pEnqueuedCall->pPeerH245TermCapList; ConnectCallbackParams.pH2250MuxCapability = pEnqueuedCall->pPeerH245H2250MuxCapability; ConnectCallbackParams.pTermCapDescriptors = pEnqueuedCall->pPeerH245TermCapDescriptors; ConnectCallbackParams.pLocalAddr = pEnqueuedCall->pQ931LocalConnectAddr; if (pEnqueuedCall->pQ931DestinationAddr == NULL) ConnectCallbackParams.pPeerAddr = pEnqueuedCall->pQ931PeerConnectAddr; else ConnectCallbackParams.pPeerAddr = pEnqueuedCall->pQ931DestinationAddr; ConnectCallbackParams.pVendorInfo = pEnqueuedCall->pPeerVendorInfo; ConnectCallbackParams.bMultipointConference = TRUE; ConnectCallbackParams.pConferenceID = &pConference->ConferenceID; ConnectCallbackParams.pMCAddress = pConference->pMultipointControllerAddr; ConnectCallbackParams.pAlternateAddress = NULL; ConnectCallbackParams.dwUserToken = pEnqueuedCall->dwUserToken;
InvokeUserConferenceCallback(pConference, CC_CONNECT_INDICATION, CC_NOT_MULTIPOINT_CAPABLE, &ConnectCallbackParams); if (ValidateCallMarkedForDeletion(hEnqueuedCall) == CC_OK) FreeCall(pEnqueuedCall); if (ValidateConference(hConference) != CC_OK) { if (ValidateCall(hCall) == CC_OK) UnlockCall(pCall); return H245_ERROR_OK; } } } break;
default: // H245_INDETERMINATE
UnlockConference(pConference); if (++pCall->wMasterSlaveRetry < MASTER_SLAVE_RETRY_MAX) { H245InitMasterSlave(pCall->H245Instance, pCall->H245Instance); UnlockCall(pCall); } else { UnlockCall(pCall); ProcessRemoteHangup(hCall, CC_INVALID_HANDLE, CC_REJECT_UNDEFINED_REASON); } return H245_ERROR_OK; } // switch
pCall->MasterSlaveState = MASTER_SLAVE_COMPLETE;
if ((pCall->OutgoingTermCapState == TERMCAP_COMPLETE) && (pCall->IncomingTermCapState == TERMCAP_COMPLETE) && (pCall->CallState == TERMCAP) && (pCall->MasterSlaveState == MASTER_SLAVE_COMPLETE)) { // Note that _ProcessConnectionComplete() returns with pConference and pCall unlocked
_ProcessConnectionComplete(pConference, pCall); return H245_ERROR_OK; }
UnlockCall(pCall); UnlockConference(pConference); return H245_ERROR_OK; }
HRESULT _ConfSendTermCap( H245_CONF_IND_T *pH245ConfIndData) { CC_HCALL hCall; PCALL pCall; HRESULT status; PCONFERENCE pConference;
// A TerminalCapabilitySet message was successfully sent from this endpoint
hCall = pH245ConfIndData->u.Confirm.dwPreserved; status = LockCallAndConference(hCall, &pCall, &pConference); if (status != CC_OK) { // This may be OK, if the call was cancelled while
// call setup was in progress.
return H245_ERROR_OK; }
if (pH245ConfIndData->u.Confirm.Error == H245_ERROR_OK && pH245ConfIndData->u.Confirm.u.ConfSndTcap.AccRej == H245_ACC) { pCall->OutgoingTermCapState = TERMCAP_COMPLETE; if ((pCall->IncomingTermCapState == TERMCAP_COMPLETE) && (pCall->CallState == TERMCAP) && (pCall->MasterSlaveState == MASTER_SLAVE_COMPLETE)) { // Note that _ProcessConnectionComplete() returns with pConference and pCall unlocked
_ProcessConnectionComplete(pConference, pCall); return H245_ERROR_OK; } } else if (pCall->CallState == TERMCAP) { // Report error to Call Control client
UnlockConference(pConference); UnlockCall(pCall); ProcessRemoteHangup(hCall, CC_INVALID_HANDLE, CC_REJECT_UNDEFINED_REASON); return H245_ERROR_OK; }
UnlockConference(pConference); UnlockCall(pCall); return H245_ERROR_OK; }
HRESULT _ConfRequestMode( H245_CONF_IND_T *pH245ConfIndData) { HRESULT status; CC_HCALL hCall; PCALL pCall; CC_HCONFERENCE hConference; PCONFERENCE pConference; CC_REQUEST_MODE_RESPONSE_CALLBACK_PARAMS RequestModeResponseCallbackParams;
hCall = pH245ConfIndData->u.Confirm.dwPreserved; status = LockCallAndConference(hCall, &pCall, &pConference); if (status != CC_OK) { // This may be OK, if the call was cancelled while
// call setup was in progress.
return H245_ERROR_OK; } hConference = pConference->hConference;
RequestModeResponseCallbackParams.hCall = hCall; if (pCall->pPeerParticipantInfo == NULL) { RequestModeResponseCallbackParams.TerminalLabel.bMCUNumber = 255; RequestModeResponseCallbackParams.TerminalLabel.bTerminalNumber = 255; } else RequestModeResponseCallbackParams.TerminalLabel = pCall->pPeerParticipantInfo->ParticipantInfo.TerminalLabel; switch (pH245ConfIndData->u.Confirm.u.ConfMrse) { case wllTrnsmtMstPrfrrdMd_chosen: RequestModeResponseCallbackParams.RequestModeResponse = CC_WILL_TRANSMIT_PREFERRED_MODE; break; case wllTrnsmtLssPrfrrdMd_chosen: RequestModeResponseCallbackParams.RequestModeResponse = CC_WILL_TRANSMIT_LESS_PREFERRED_MODE; break; default: RequestModeResponseCallbackParams.RequestModeResponse = CC_REQUEST_DENIED; break; } InvokeUserConferenceCallback(pConference, CC_REQUEST_MODE_RESPONSE_INDICATION, pH245ConfIndData->u.Confirm.Error, &RequestModeResponseCallbackParams); if (ValidateCall(hCall) == CC_OK) UnlockCall(pCall); if (ValidateConference(hConference) == CC_OK) UnlockConference(pConference); return H245_ERROR_OK; }
HRESULT _ConfRequestModeReject( H245_CONF_IND_T *pH245ConfIndData) { HRESULT status; CC_HCALL hCall; PCALL pCall; CC_HCONFERENCE hConference; PCONFERENCE pConference; CC_REQUEST_MODE_RESPONSE_CALLBACK_PARAMS RequestModeResponseCallbackParams;
hCall = pH245ConfIndData->u.Confirm.dwPreserved; status = LockCallAndConference(hCall, &pCall, &pConference); if (status != CC_OK) { // This may be OK, if the call was cancelled while
// call setup was in progress.
return H245_ERROR_OK; } hConference = pConference->hConference;
RequestModeResponseCallbackParams.hCall = hCall; if (pCall->pPeerParticipantInfo == NULL) { RequestModeResponseCallbackParams.TerminalLabel.bMCUNumber = 255; RequestModeResponseCallbackParams.TerminalLabel.bTerminalNumber = 255; } else RequestModeResponseCallbackParams.TerminalLabel = pCall->pPeerParticipantInfo->ParticipantInfo.TerminalLabel; switch (pH245ConfIndData->u.Confirm.u.ConfMrseReject) { case H245_REJ_UNAVAILABLE: RequestModeResponseCallbackParams.RequestModeResponse = CC_MODE_UNAVAILABLE; break; case H245_REJ_MULTIPOINT: RequestModeResponseCallbackParams.RequestModeResponse = CC_MULTIPOINT_CONSTRAINT; break; case H245_REJ_DENIED: RequestModeResponseCallbackParams.RequestModeResponse = CC_REQUEST_DENIED; break; default: RequestModeResponseCallbackParams.RequestModeResponse = CC_REQUEST_DENIED; break; } InvokeUserConferenceCallback(pConference, CC_REQUEST_MODE_RESPONSE_INDICATION, pH245ConfIndData->u.Confirm.Error, &RequestModeResponseCallbackParams); if (ValidateCall(hCall) == CC_OK) UnlockCall(pCall); if (ValidateConference(hConference) == CC_OK) UnlockConference(pConference); return H245_ERROR_OK; }
HRESULT _ConfRequestModeExpired( H245_CONF_IND_T *pH245ConfIndData) { HRESULT status; CC_HCALL hCall; PCALL pCall; CC_HCONFERENCE hConference; PCONFERENCE pConference; CC_REQUEST_MODE_RESPONSE_CALLBACK_PARAMS RequestModeResponseCallbackParams;
hCall = pH245ConfIndData->u.Confirm.dwPreserved; status = LockCallAndConference(hCall, &pCall, &pConference); if (status != CC_OK) { // This may be OK, if the call was cancelled while
// call setup was in progress.
return H245_ERROR_OK; } hConference = pConference->hConference;
RequestModeResponseCallbackParams.hCall = hCall; if (pCall->pPeerParticipantInfo == NULL) { RequestModeResponseCallbackParams.TerminalLabel.bMCUNumber = 255; RequestModeResponseCallbackParams.TerminalLabel.bTerminalNumber = 255; } else RequestModeResponseCallbackParams.TerminalLabel = pCall->pPeerParticipantInfo->ParticipantInfo.TerminalLabel; RequestModeResponseCallbackParams.RequestModeResponse = CC_REQUEST_DENIED; InvokeUserConferenceCallback(pConference, CC_REQUEST_MODE_RESPONSE_INDICATION, pH245ConfIndData->u.Confirm.Error, &RequestModeResponseCallbackParams); if (ValidateCall(hCall) == CC_OK) UnlockCall(pCall); if (ValidateConference(hConference) == CC_OK) UnlockConference(pConference); return H245_ERROR_OK; }
HRESULT _ConfRoundTrip( H245_CONF_IND_T *pH245ConfIndData) { CC_HCALL hCall; PCALL pCall; CC_HCONFERENCE hConference; PCONFERENCE pConference; HRESULT status; CC_PING_RESPONSE_CALLBACK_PARAMS PingCallbackParams;
hCall = pH245ConfIndData->u.Confirm.dwPreserved; status = LockCallAndConference(hCall, &pCall, &pConference); if (status != CC_OK) { // This may be OK, if the call was cancelled while
// call setup was in progress.
return H245_ERROR_OK; } hConference = pConference->hConference;
PingCallbackParams.hCall = hCall; if (pCall->pPeerParticipantInfo == NULL) { PingCallbackParams.TerminalLabel.bMCUNumber = 255; PingCallbackParams.TerminalLabel.bTerminalNumber = 255; } else PingCallbackParams.TerminalLabel = pCall->pPeerParticipantInfo->ParticipantInfo.TerminalLabel; PingCallbackParams.bResponse = TRUE; InvokeUserConferenceCallback(pConference, CC_PING_RESPONSE_INDICATION, pH245ConfIndData->u.Confirm.Error, &PingCallbackParams); if (ValidateCall(hCall) == CC_OK) UnlockCall(pCall); if (ValidateConference(hConference) == CC_OK) UnlockConference(pConference); return H245_ERROR_OK; }
HRESULT _ConfRoundTripExpired( H245_CONF_IND_T *pH245ConfIndData) { CC_HCALL hCall; PCALL pCall; CC_HCONFERENCE hConference; PCONFERENCE pConference; HRESULT status; CC_PING_RESPONSE_CALLBACK_PARAMS PingCallbackParams;
hCall = pH245ConfIndData->u.Confirm.dwPreserved; status = LockCallAndConference(hCall, &pCall, &pConference); if (status != CC_OK) { // This may be OK, if the call was cancelled while
// call setup was in progress.
return H245_ERROR_OK; } hConference = pConference->hConference;
PingCallbackParams.hCall = hCall; if (pCall->pPeerParticipantInfo == NULL) { PingCallbackParams.TerminalLabel.bMCUNumber = 255; PingCallbackParams.TerminalLabel.bTerminalNumber = 255; } else PingCallbackParams.TerminalLabel = pCall->pPeerParticipantInfo->ParticipantInfo.TerminalLabel; PingCallbackParams.bResponse = FALSE; InvokeUserConferenceCallback(pConference, CC_PING_RESPONSE_INDICATION, pH245ConfIndData->u.Confirm.Error, &PingCallbackParams); if (ValidateCall(hCall) == CC_OK) UnlockCall(pCall); if (ValidateConference(hConference) == CC_OK) UnlockConference(pConference); return H245_ERROR_OK; }
HRESULT H245Callback( H245_CONF_IND_T *pH245ConfIndData, void *pMisc) { HRESULT status = H245_ERROR_OK;
EnterCallControl();
if (CallControlState != OPERATIONAL_STATE) HResultLeaveCallControl(H245_ERROR_OK);
if (pH245ConfIndData == NULL) HResultLeaveCallControl(H245_ERROR_OK);
if (pH245ConfIndData->Kind == H245_CONF) { switch (pH245ConfIndData->u.Confirm.Confirm) { case H245_CONF_INIT_MSTSLV: status = _ConfInitMstslv(pH245ConfIndData); break;
case H245_CONF_SEND_TERMCAP: status = _ConfSendTermCap(pH245ConfIndData); break;
case H245_CONF_OPEN: status = _ConfOpen(pH245ConfIndData); break;
case H245_CONF_NEEDRSP_OPEN: status = _ConfBiDirectionalOpen(pH245ConfIndData); break;
case H245_CONF_CLOSE: status = _ConfClose(pH245ConfIndData); break;
case H245_CONF_REQ_CLOSE: status = _ConfRequestClose(pH245ConfIndData); break;
// case H245_CONF_MUXTBL_SND: not valid for H.323 MuliplexEntrySend
// case H245_CONF_RMESE: not valid for H.323 RequestMultiplexEntry
// case H245_CONF_RMESE_REJECT: not valid for H.323 RequestMultiplexEntryReject
// case H245_CONF_RMESE_EXPIRED: not valid for H.323
case H245_CONF_MRSE: status = _ConfRequestMode(pH245ConfIndData); break;
case H245_CONF_MRSE_REJECT: status = _ConfRequestModeReject(pH245ConfIndData); break;
case H245_CONF_MRSE_EXPIRED: status = _ConfRequestModeExpired(pH245ConfIndData); break;
case H245_CONF_RTDSE: status = _ConfRoundTrip(pH245ConfIndData); break;
case H245_CONF_RTDSE_EXPIRED: status = _ConfRoundTripExpired(pH245ConfIndData); break;
default: status = _ConfUnimplemented(pH245ConfIndData); break; } } else if (pH245ConfIndData->Kind == H245_IND) { switch (pH245ConfIndData->u.Indication.Indicator) { case H245_IND_MSTSLV: status = _IndMstslv(pH245ConfIndData); break;
case H245_IND_CAP: status = _IndCapability(pH245ConfIndData); break;
case H245_IND_CESE_RELEASE: // Remote has abandoned TerminalCapabilitySet
// No longer need to send TerminalCapabilitySetAck
// We can probably get away with ignoring this,
// but we should NOT return FunctionNotSupported!
break;
case H245_IND_OPEN: status = _IndOpen(pH245ConfIndData); break;
case H245_IND_OPEN_CONF: // Bi-directionl channel open complete
status = _IndOpenConf(pH245ConfIndData); break;
case H245_IND_CLOSE: status = _IndClose(pH245ConfIndData); break;
case H245_IND_REQ_CLOSE: status = _IndRequestClose(pH245ConfIndData); break;
case H245_IND_CLCSE_RELEASE: // Remote has abandoned RequestChannelClose
// No longer need to send RequestChannelCloseAck and CloseLogicalChannel
// We can probably get away with ignoring this,
// but we should NOT return FunctionNotSupported!
break;
// case H245_IND_MUX_TBL: not valid in H.323 MuliplexEntrySend
// case H245_IND_MTSE_RELEASE not valid in H.323 MuliplexEntrySendRelease
// case H245_IND_RMESE not valid in H.323 RequestMuliplexEntry
// case H245_IND_RMESE_RELEASE not valid in H.323 RequestMuliplexEntryRelease
case H245_IND_MRSE: status = _IndModeRequest(pH245ConfIndData); break;
case H245_IND_MRSE_RELEASE: // Remote has abandoned RequestMode
// No longer need to send RequestModeAck or RequestModeReject
// We can probably get away with ignoring this,
// but we should NOT return FunctionNotSupported!
break;
// case H245_IND_MLSE: We don't support looping back data
case H245_IND_MLSE_RELEASE: // Required to accept this message
break;
case H245_IND_NONSTANDARD_REQUEST: case H245_IND_NONSTANDARD_RESPONSE: case H245_IND_NONSTANDARD_COMMAND: case H245_IND_NONSTANDARD: status = _IndNonStandard(pH245ConfIndData); break;
case H245_IND_MISC_COMMAND: status = _IndMiscellaneousCommand(pH245ConfIndData, pMisc); break;
case H245_IND_MISC: status = _IndMiscellaneous(pH245ConfIndData, pMisc); break; case H245_IND_COMM_MODE_REQUEST: status = _IndUnimplemented(pH245ConfIndData); // TBD
break;
// case H245_IND_COMM_MODE_RESPONSE: We never send request!
case H245_IND_COMM_MODE_COMMAND: status = _IndCommunicationModeCommand(pH245ConfIndData); break;
case H245_IND_CONFERENCE_REQUEST: status = _IndConferenceRequest(pH245ConfIndData); break;
case H245_IND_CONFERENCE_RESPONSE: status = _IndConferenceResponse(pH245ConfIndData); break;
case H245_IND_CONFERENCE_COMMAND: status = _IndConferenceCommand(pH245ConfIndData); break;
case H245_IND_CONFERENCE: status = _IndConference(pH245ConfIndData); break; case H245_IND_SEND_TERMCAP: status = _IndSendTerminalCapabilitySet(pH245ConfIndData); break;
// case H245_IND_ENCRYPTION: Not valid in H.323
case H245_IND_FLOW_CONTROL: status = _IndFlowControl(pH245ConfIndData); break;
case H245_IND_ENDSESSION: status = _IndEndSession(pH245ConfIndData); break;
case H245_IND_FUNCTION_NOT_UNDERSTOOD: // We don't do anything with this but we still want to
// return H245_ERROR_OK so H.245 does not sent
// FunctionNotSupported back to remote peer!
break;
case H245_IND_JITTER: // It is ok to ignore this; no response is expected
break;
// case H245_IND_H223_SKEW: Not valid in H.323
// case H245_IND_NEW_ATM_VC: Not valid in H.323
case H245_IND_USERINPUT: status = _IndUserInput(pH245ConfIndData); break;
case H245_IND_H2250_MAX_SKEW: status = _IndH2250MaximumSkew(pH245ConfIndData); break;
case H245_IND_MC_LOCATION: status = _IndMCLocation(pH245ConfIndData); break;
case H245_IND_VENDOR_ID: status = _IndVendorIdentification(pH245ConfIndData, pMisc); break;
case H245_IND_FUNCTION_NOT_SUPPORTED: // We don't do anything with this but we still want to
// return H245_ERROR_OK so H.245 does not sent
// FunctionNotSupported back to remote peer!
break;
// case H245_IND_H223_RECONFIG: Not valid in H.323
// case H245_IND_H223_RECONFIG_ACK: Not valid in H.323
// case H245_IND_H223_RECONFIG_REJECT: Not valid in H.323
default: status = _IndUnimplemented(pH245ConfIndData); break; } } HResultLeaveCallControl(status); }
|