mirror of https://github.com/tongzx/nt5src
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
2824 lines
79 KiB
2824 lines
79 KiB
/****************************************************************************
|
|
*
|
|
* $Archive: S:/STURGEON/SRC/CALLCONT/VCS/ccutils.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.107.1.0 $
|
|
* $Date: 20 Jun 1997 14:19:02 $
|
|
* $Author: MANDREWS $
|
|
*
|
|
* Deliverable:
|
|
*
|
|
* Abstract:
|
|
*
|
|
*
|
|
* Notes:
|
|
*
|
|
***************************************************************************/
|
|
|
|
#pragma warning ( disable : 4057 4100 4115 4201 4214 4244)
|
|
#include <nt.h>
|
|
#include <ntrtl.h>
|
|
#include <nturtl.h>
|
|
#include <windows.h>
|
|
#include <winreg.h>
|
|
#pragma warning ( default : 4115 4201 4214 )
|
|
#include "incommon.h"
|
|
#include "callcont.h"
|
|
#include "q931.h"
|
|
#include "apierror.h"
|
|
#include "ccmain.h"
|
|
#include "chanman.h"
|
|
#include "confman.h"
|
|
#include "callman.h"
|
|
#include "q931man.h"
|
|
#include "userman.h"
|
|
#include "ccutils.h"
|
|
#include "linkapi.h"
|
|
#include "h245man.h"
|
|
#include "provider.h"
|
|
|
|
extern CC_CONFERENCEID InvalidConferenceID;
|
|
|
|
|
|
|
|
// The following two procedures are currently obsolete. They're kept
|
|
// around in case we need them in the future
|
|
#if 0
|
|
void * MallocDbg( DWORD dwSize,
|
|
PSTR pszFile,
|
|
int nLine)
|
|
{
|
|
void *pMem;
|
|
|
|
ASSERT(dwSize != 0);
|
|
|
|
pMem = GlobalAlloc(GMEM_FIXED, dwSize);
|
|
|
|
#ifdef MEM_TRACE
|
|
printf("MALLOC %s %d %lu %08x\n",
|
|
pszFile, nLine, dwSize, pMem);
|
|
#endif // MEM_TRACE
|
|
|
|
return pMem;
|
|
}
|
|
|
|
|
|
|
|
void FreeDbg( void * pMem,
|
|
PSTR pszFile,
|
|
int nLine)
|
|
{
|
|
ASSERT(pMem != NULL);
|
|
|
|
GlobalFree(pMem);
|
|
|
|
#ifdef MEM_TRACE
|
|
printf("FREE %s %d %08x\n",
|
|
pszFile, nLine, pMem);
|
|
#endif // MEM_TRACE
|
|
}
|
|
#endif // 0
|
|
|
|
|
|
|
|
HRESULT InitializeLock( PLOCK pLock)
|
|
{
|
|
ASSERT(pLock != NULL);
|
|
|
|
#ifdef DBG
|
|
|
|
__try {
|
|
|
|
// initialize lock (and allocate event immediately)
|
|
InitializeCriticalSectionAndSpinCount(&pLock->LockInfoLock,H323_SPIN_COUNT);
|
|
|
|
} __except ((GetExceptionCode() == STATUS_NO_MEMORY)
|
|
? EXCEPTION_EXECUTE_HANDLER
|
|
: EXCEPTION_CONTINUE_SEARCH
|
|
) {
|
|
|
|
// failure
|
|
return CS_NO_MEMORY;
|
|
}
|
|
|
|
pLock->wLockCount = 0;
|
|
pLock->wNumQueuedThreads = 0;
|
|
pLock->hOwningThread = 0;
|
|
#endif
|
|
|
|
pLock->Lock = CreateMutex(NULL, // security attributes
|
|
FALSE, // initial owner
|
|
NULL); // name
|
|
|
|
if (pLock->Lock == NULL) {
|
|
#ifdef DBG
|
|
DeleteCriticalSection(&pLock->LockInfoLock);
|
|
#endif
|
|
return CC_INTERNAL_ERROR;
|
|
} else
|
|
return CC_OK;
|
|
}
|
|
|
|
|
|
|
|
HRESULT DeleteLock( PLOCK pLock)
|
|
{
|
|
ASSERT(pLock != NULL);
|
|
|
|
#ifdef DBG
|
|
DeleteCriticalSection(&pLock->LockInfoLock);
|
|
#endif
|
|
|
|
if (CloseHandle(pLock->Lock) == TRUE)
|
|
return CC_OK;
|
|
else
|
|
return CC_INTERNAL_ERROR;
|
|
}
|
|
|
|
|
|
|
|
HRESULT AcquireLock( PLOCK pLock)
|
|
{
|
|
HRESULT status;
|
|
|
|
ASSERT(pLock != NULL);
|
|
|
|
status = AcquireTimedLock(pLock, INFINITE, NULL);
|
|
return status;
|
|
}
|
|
|
|
|
|
|
|
HRESULT AcquireTimedLock( PLOCK pLock,
|
|
DWORD dwTimeout,
|
|
BOOL *pbTimedOut)
|
|
{
|
|
DWORD dwStatus;
|
|
|
|
ASSERT(pLock != NULL);
|
|
|
|
#ifdef DBG
|
|
EnterCriticalSection(&pLock->LockInfoLock);
|
|
(pLock->wNumQueuedThreads)++;
|
|
LeaveCriticalSection(&pLock->LockInfoLock);
|
|
#endif
|
|
|
|
dwStatus = WaitForSingleObject(pLock->Lock, dwTimeout);
|
|
|
|
#ifdef DBG
|
|
EnterCriticalSection(&pLock->LockInfoLock);
|
|
(pLock->wNumQueuedThreads)--;
|
|
(pLock->wLockCount)++;
|
|
pLock->hOwningThread = GetCurrentThread();
|
|
LeaveCriticalSection(&pLock->LockInfoLock);
|
|
#endif
|
|
|
|
if ((dwStatus != WAIT_OBJECT_0) && (dwStatus != WAIT_TIMEOUT))
|
|
return CC_INTERNAL_ERROR;
|
|
|
|
if (dwStatus == WAIT_TIMEOUT) {
|
|
if (pbTimedOut != NULL) {
|
|
*pbTimedOut = TRUE;
|
|
}
|
|
} else {
|
|
if (pbTimedOut != NULL) {
|
|
*pbTimedOut = FALSE;
|
|
}
|
|
}
|
|
return CC_OK;
|
|
}
|
|
|
|
|
|
|
|
HRESULT RelinquishLock( PLOCK pLock)
|
|
{
|
|
ASSERT(pLock != NULL);
|
|
|
|
#ifdef DBG
|
|
EnterCriticalSection(&pLock->LockInfoLock);
|
|
(pLock->wLockCount)--;
|
|
if (pLock->wLockCount == 0)
|
|
pLock->hOwningThread = 0;
|
|
LeaveCriticalSection(&pLock->LockInfoLock);
|
|
#endif
|
|
|
|
if (ReleaseMutex(pLock->Lock) == TRUE)
|
|
return CC_OK;
|
|
else
|
|
return CC_INTERNAL_ERROR;
|
|
}
|
|
|
|
|
|
|
|
HRESULT ValidateOctetString( PCC_OCTETSTRING pOctetString)
|
|
{
|
|
if (pOctetString == NULL)
|
|
return CC_OK;
|
|
if ((pOctetString->wOctetStringLength > 0) &&
|
|
(pOctetString->pOctetString == NULL))
|
|
return CC_BAD_PARAM;
|
|
return CC_OK;
|
|
}
|
|
|
|
|
|
|
|
HRESULT CopyOctetString( PCC_OCTETSTRING *ppDest,
|
|
PCC_OCTETSTRING pSource)
|
|
{
|
|
ASSERT(ppDest != NULL);
|
|
|
|
if (pSource == NULL) {
|
|
*ppDest = NULL;
|
|
return CC_OK;
|
|
}
|
|
*ppDest = (PCC_OCTETSTRING)Malloc(sizeof(CC_OCTETSTRING));
|
|
if (*ppDest == NULL)
|
|
return CC_NO_MEMORY;
|
|
(*ppDest)->wOctetStringLength = pSource->wOctetStringLength;
|
|
if ((pSource->wOctetStringLength == 0) ||
|
|
(pSource->pOctetString == NULL)) {
|
|
pSource->wOctetStringLength = 0;
|
|
(*ppDest)->pOctetString = NULL;
|
|
} else {
|
|
(*ppDest)->pOctetString = (BYTE *)Malloc(pSource->wOctetStringLength);
|
|
if ((*ppDest)->pOctetString == NULL) {
|
|
Free(*ppDest);
|
|
*ppDest = NULL;
|
|
return CC_NO_MEMORY;
|
|
}
|
|
memcpy((*ppDest)->pOctetString, pSource->pOctetString, pSource->wOctetStringLength);
|
|
}
|
|
return CC_OK;
|
|
}
|
|
|
|
|
|
|
|
HRESULT FreeOctetString( PCC_OCTETSTRING pOctetString)
|
|
{
|
|
if (pOctetString == NULL)
|
|
return CC_OK;
|
|
if ((pOctetString->wOctetStringLength > 0) &&
|
|
(pOctetString->pOctetString != NULL))
|
|
Free(pOctetString->pOctetString);
|
|
Free(pOctetString);
|
|
return CC_OK;
|
|
}
|
|
|
|
|
|
|
|
HRESULT CopySeparateStack( H245_ACCESS_T **ppDest,
|
|
H245_ACCESS_T *pSource)
|
|
{
|
|
ASSERT(ppDest != NULL);
|
|
|
|
if (pSource == NULL) {
|
|
*ppDest = NULL;
|
|
return CC_OK;
|
|
}
|
|
|
|
// We currently can't handle IP source route addresses,
|
|
// since this address format contains embedded pointers
|
|
// that cannot simply be copied
|
|
if ((pSource->networkAddress.choice == localAreaAddress_chosen) &&
|
|
(pSource->networkAddress.u.localAreaAddress.choice == unicastAddress_chosen) &&
|
|
(pSource->networkAddress.u.localAreaAddress.u.unicastAddress.choice == iPSourceRouteAddress_chosen))
|
|
return CC_NOT_IMPLEMENTED;
|
|
|
|
*ppDest = (H245_ACCESS_T *)Malloc(sizeof(H245_ACCESS_T));
|
|
if (*ppDest == NULL)
|
|
return CC_NO_MEMORY;
|
|
|
|
**ppDest = *pSource;
|
|
return CC_OK;
|
|
}
|
|
|
|
|
|
|
|
HRESULT FreeSeparateStack( H245_ACCESS_T *pSeparateStack)
|
|
{
|
|
if (pSeparateStack == NULL)
|
|
return CC_OK;
|
|
Free(pSeparateStack);
|
|
return CC_OK;
|
|
}
|
|
|
|
|
|
|
|
HRESULT ValidateNonStandardData( PCC_NONSTANDARDDATA pNonStandardData)
|
|
{
|
|
if (pNonStandardData == NULL)
|
|
return CC_OK;
|
|
return ValidateOctetString(&pNonStandardData->sData);
|
|
}
|
|
|
|
|
|
|
|
HRESULT CopyNonStandardData( PCC_NONSTANDARDDATA *ppDest,
|
|
PCC_NONSTANDARDDATA pSource)
|
|
{
|
|
ASSERT(ppDest != NULL);
|
|
|
|
if (pSource == NULL) {
|
|
*ppDest = NULL;
|
|
return CC_OK;
|
|
}
|
|
*ppDest = (PCC_NONSTANDARDDATA)Malloc(sizeof(CC_NONSTANDARDDATA));
|
|
if (*ppDest == NULL)
|
|
return CC_NO_MEMORY;
|
|
**ppDest = *pSource;
|
|
if ((pSource->sData.wOctetStringLength == 0) ||
|
|
(pSource->sData.pOctetString == NULL)) {
|
|
(*ppDest)->sData.wOctetStringLength = 0;
|
|
(*ppDest)->sData.pOctetString = NULL;
|
|
} else {
|
|
(*ppDest)->sData.pOctetString = (BYTE *)Malloc(pSource->sData.wOctetStringLength);
|
|
if ((*ppDest)->sData.pOctetString == NULL) {
|
|
Free(*ppDest);
|
|
return CC_NO_MEMORY;
|
|
}
|
|
memcpy((*ppDest)->sData.pOctetString,
|
|
pSource->sData.pOctetString,
|
|
pSource->sData.wOctetStringLength);
|
|
}
|
|
return CC_OK;
|
|
}
|
|
|
|
|
|
|
|
HRESULT FreeNonStandardData( PCC_NONSTANDARDDATA pNonStandardData)
|
|
{
|
|
if (pNonStandardData == NULL)
|
|
return CC_OK;
|
|
if ((pNonStandardData->sData.wOctetStringLength > 0) &&
|
|
(pNonStandardData->sData.pOctetString != NULL))
|
|
Free(pNonStandardData->sData.pOctetString);
|
|
Free(pNonStandardData);
|
|
return CC_OK;
|
|
}
|
|
|
|
|
|
|
|
HRESULT ValidateVendorInfo( PCC_VENDORINFO pVendorInfo)
|
|
{
|
|
HRESULT status;
|
|
|
|
if (pVendorInfo == NULL)
|
|
return CC_OK;
|
|
status = ValidateOctetString(pVendorInfo->pProductNumber);
|
|
if (status != CC_OK)
|
|
return status;
|
|
status = ValidateOctetString(pVendorInfo->pVersionNumber);
|
|
return status;
|
|
}
|
|
|
|
|
|
|
|
HRESULT CopyVendorInfo( PCC_VENDORINFO *ppDest,
|
|
PCC_VENDORINFO pSource)
|
|
{
|
|
HRESULT status;
|
|
|
|
ASSERT(ppDest != NULL);
|
|
|
|
if (pSource == NULL) {
|
|
*ppDest = NULL;
|
|
return CC_OK;
|
|
}
|
|
*ppDest = (PCC_VENDORINFO)Malloc(sizeof(CC_VENDORINFO));
|
|
if (*ppDest == NULL)
|
|
return CC_NO_MEMORY;
|
|
**ppDest = *pSource;
|
|
status = CopyOctetString(&(*ppDest)->pProductNumber, pSource->pProductNumber);
|
|
if (status != CC_OK) {
|
|
Free(*ppDest);
|
|
return status;
|
|
}
|
|
status = CopyOctetString(&(*ppDest)->pVersionNumber, pSource->pVersionNumber);
|
|
if (status != CC_OK) {
|
|
FreeOctetString((*ppDest)->pProductNumber);
|
|
Free(*ppDest);
|
|
return status;
|
|
}
|
|
return CC_OK;
|
|
}
|
|
|
|
|
|
|
|
HRESULT FreeVendorInfo( PCC_VENDORINFO pVendorInfo)
|
|
{
|
|
if (pVendorInfo == NULL)
|
|
return CC_OK;
|
|
FreeOctetString(pVendorInfo->pProductNumber);
|
|
FreeOctetString(pVendorInfo->pVersionNumber);
|
|
Free(pVendorInfo);
|
|
return CC_OK;
|
|
}
|
|
|
|
|
|
|
|
BOOL EqualConferenceIDs( PCC_CONFERENCEID pConferenceID1,
|
|
PCC_CONFERENCEID pConferenceID2)
|
|
{
|
|
ASSERT(pConferenceID1 != NULL);
|
|
ASSERT(pConferenceID2 != NULL);
|
|
|
|
if (memcmp(pConferenceID1->buffer,
|
|
pConferenceID2->buffer,
|
|
sizeof(pConferenceID1->buffer)) == 0)
|
|
return TRUE;
|
|
else
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
|
|
BOOL EqualAddrs( PCC_ADDR pAddr1,
|
|
PCC_ADDR pAddr2)
|
|
{
|
|
ASSERT(pAddr1 != NULL);
|
|
ASSERT(pAddr2 != NULL);
|
|
|
|
if (pAddr1->nAddrType != pAddr2->nAddrType)
|
|
return FALSE;
|
|
|
|
if (pAddr1->bMulticast != pAddr2->bMulticast)
|
|
return FALSE;
|
|
|
|
switch (pAddr1->nAddrType) {
|
|
case CC_IP_DOMAIN_NAME:
|
|
if ((pAddr1->Addr.IP_DomainName.wPort == pAddr2->Addr.IP_DomainName.wPort) &&
|
|
(wcscmp(pAddr1->Addr.IP_DomainName.cAddr, pAddr2->Addr.IP_DomainName.cAddr) == 0))
|
|
return TRUE;
|
|
else
|
|
return FALSE;
|
|
case CC_IP_DOT:
|
|
if ((pAddr1->Addr.IP_Dot.wPort == pAddr2->Addr.IP_Dot.wPort) &&
|
|
(wcscmp(pAddr1->Addr.IP_Dot.cAddr, pAddr2->Addr.IP_Dot.cAddr) == 0))
|
|
return TRUE;
|
|
else
|
|
return FALSE;
|
|
case CC_IP_BINARY:
|
|
if ((pAddr1->Addr.IP_Binary.wPort == pAddr2->Addr.IP_Binary.wPort) &&
|
|
(pAddr1->Addr.IP_Binary.dwAddr == pAddr2->Addr.IP_Binary.dwAddr))
|
|
return TRUE;
|
|
else
|
|
return FALSE;
|
|
default:
|
|
ASSERT(0);
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
HRESULT ValidateTermCapList( PCC_TERMCAPLIST pTermCapList)
|
|
{
|
|
unsigned i, j;
|
|
|
|
if (pTermCapList == NULL)
|
|
return CC_OK;
|
|
|
|
for (i = 0; i < pTermCapList->wLength; i++)
|
|
if (pTermCapList->pTermCapArray[i] == NULL)
|
|
return CC_BAD_PARAM;
|
|
|
|
// make sure that all capability IDs are unique
|
|
for (i = 0; i < pTermCapList->wLength; i++) {
|
|
for (j = i + 1; j < pTermCapList->wLength; j++) {
|
|
if (pTermCapList->pTermCapArray[i]->CapId == pTermCapList->pTermCapArray[j]->CapId)
|
|
return CC_BAD_PARAM;
|
|
}
|
|
if ((pTermCapList->pTermCapArray[i]->CapId == H245_INVALID_CAPID) ||
|
|
(pTermCapList->pTermCapArray[i]->CapId == 0))
|
|
return CC_BAD_PARAM;
|
|
}
|
|
return CC_OK;
|
|
}
|
|
|
|
|
|
|
|
HRESULT ValidateTermCapDescriptors( PCC_TERMCAPDESCRIPTORS pTermCapDescriptors,
|
|
PCC_TERMCAPLIST pTermCapList)
|
|
{
|
|
WORD i, j, k, l;
|
|
H245_TOTCAPDESC_T *pTermCapDescriptor;
|
|
H245_SIMCAP_T *pSimCaps;
|
|
|
|
if (pTermCapDescriptors == NULL)
|
|
return CC_OK;
|
|
|
|
for (i = 0; i < pTermCapDescriptors->wLength; i++) {
|
|
pTermCapDescriptor = pTermCapDescriptors->pTermCapDescriptorArray[i];
|
|
if ((pTermCapDescriptor->CapDescId > 255) ||
|
|
(pTermCapDescriptor->CapDesc.Length == 0) ||
|
|
(pTermCapDescriptor->CapDesc.Length > H245_MAX_SIMCAPS))
|
|
return CC_BAD_PARAM;
|
|
for (j = i + 1; j < pTermCapDescriptors->wLength; j++) {
|
|
if (pTermCapDescriptor->CapDescId ==
|
|
pTermCapDescriptors->pTermCapDescriptorArray[j]->CapDescId) {
|
|
return CC_BAD_PARAM;
|
|
}
|
|
}
|
|
for (j = 0; j < pTermCapDescriptor->CapDesc.Length; j++) {
|
|
pSimCaps = &(pTermCapDescriptor->CapDesc.SimCapArray[j]);
|
|
if ((pSimCaps->Length == 0) ||
|
|
(pSimCaps->Length > H245_MAX_ALTCAPS))
|
|
return CC_BAD_PARAM;
|
|
for (k = 0; k < pSimCaps->Length; k++) {
|
|
for (l = 0; l < pTermCapList->wLength; l++) {
|
|
if (pSimCaps->AltCaps[k] ==
|
|
pTermCapList->pTermCapArray[l]->CapId)
|
|
break;
|
|
}
|
|
if (l == pTermCapList->wLength)
|
|
// the capability descriptor contains a capability ID
|
|
// which is not present in the capability table
|
|
return CC_BAD_PARAM;
|
|
}
|
|
}
|
|
}
|
|
return CC_OK;
|
|
}
|
|
|
|
|
|
|
|
HRESULT ValidateAddr( PCC_ADDR pAddr)
|
|
{
|
|
if (pAddr == NULL)
|
|
return CC_OK;
|
|
if ((pAddr->nAddrType != CC_IP_DOMAIN_NAME) &&
|
|
(pAddr->nAddrType != CC_IP_DOT) &&
|
|
(pAddr->nAddrType != CC_IP_BINARY))
|
|
return CC_BAD_PARAM;
|
|
return CC_OK;
|
|
}
|
|
|
|
|
|
|
|
HRESULT CopyAddr( PCC_ADDR *ppDest,
|
|
PCC_ADDR pSource)
|
|
{
|
|
ASSERT(ppDest != NULL);
|
|
|
|
if (pSource == NULL) {
|
|
*ppDest = NULL;
|
|
return CC_OK;
|
|
}
|
|
*ppDest = (PCC_ADDR)Malloc(sizeof(CC_ADDR));
|
|
if (*ppDest == NULL)
|
|
return CC_NO_MEMORY;
|
|
**ppDest = *pSource;
|
|
return CC_OK;
|
|
}
|
|
|
|
|
|
|
|
HRESULT FreeAddr( PCC_ADDR pAddr)
|
|
{
|
|
if (pAddr == NULL)
|
|
return CC_OK;
|
|
Free(pAddr);
|
|
return CC_OK;
|
|
}
|
|
|
|
|
|
|
|
HRESULT SetQ931Port( PCC_ADDR pAddr)
|
|
{
|
|
if (pAddr == NULL)
|
|
return CC_OK;
|
|
switch (pAddr->nAddrType) {
|
|
case CC_IP_DOMAIN_NAME:
|
|
if (pAddr->Addr.IP_DomainName.wPort == 0)
|
|
pAddr->Addr.IP_DomainName.wPort = CC_H323_HOST_CALL;
|
|
return CC_OK;
|
|
case CC_IP_DOT:
|
|
if (pAddr->Addr.IP_Dot.wPort == 0)
|
|
pAddr->Addr.IP_Dot.wPort = CC_H323_HOST_CALL;
|
|
return CC_OK;
|
|
case CC_IP_BINARY:
|
|
if (pAddr->Addr.IP_Binary.wPort == 0)
|
|
pAddr->Addr.IP_Binary.wPort = CC_H323_HOST_CALL;
|
|
return CC_OK;
|
|
}
|
|
|
|
ASSERT(0);
|
|
return CC_INTERNAL_ERROR;
|
|
}
|
|
|
|
|
|
|
|
HRESULT ValidateDisplay( PWSTR pszDisplay)
|
|
{
|
|
return CC_OK;
|
|
}
|
|
|
|
|
|
|
|
HRESULT CopyDisplay( PWSTR *ppDest,
|
|
PWSTR pSource)
|
|
{
|
|
ASSERT(ppDest != NULL);
|
|
|
|
if (pSource == NULL) {
|
|
*ppDest = NULL;
|
|
return CC_OK;
|
|
}
|
|
*ppDest = (WCHAR *)Malloc((wcslen(pSource)+1)*sizeof(WCHAR));
|
|
if (*ppDest == NULL)
|
|
return CC_NO_MEMORY;
|
|
wcscpy(*ppDest, pSource);
|
|
return CC_OK;
|
|
}
|
|
|
|
|
|
|
|
HRESULT FreeDisplay( PWSTR pszDisplay)
|
|
{
|
|
Free(pszDisplay);
|
|
return CC_OK;
|
|
}
|
|
|
|
|
|
|
|
HRESULT ValidateTerminalID( PCC_OCTETSTRING pTerminalID)
|
|
{
|
|
return ValidateOctetString(pTerminalID);
|
|
}
|
|
|
|
|
|
|
|
HRESULT CopyTerminalID( PCC_OCTETSTRING *ppDest,
|
|
PCC_OCTETSTRING pSource)
|
|
{
|
|
ASSERT(ppDest != NULL);
|
|
|
|
return CopyOctetString(ppDest, pSource);
|
|
}
|
|
|
|
|
|
|
|
HRESULT FreeTerminalID( PCC_OCTETSTRING pTerminalID)
|
|
{
|
|
return FreeOctetString(pTerminalID);
|
|
}
|
|
|
|
|
|
|
|
HRESULT SetTerminalType( TRISTATE tsMultipointController,
|
|
BYTE *pbTerminalType)
|
|
{
|
|
switch (tsMultipointController) {
|
|
case TS_TRUE:
|
|
*pbTerminalType = 240;
|
|
break;
|
|
case TS_UNKNOWN:
|
|
*pbTerminalType = 70;
|
|
break;
|
|
case TS_FALSE:
|
|
*pbTerminalType = 50;
|
|
break;
|
|
default:
|
|
ASSERT(0);
|
|
*pbTerminalType = 0;
|
|
break;
|
|
}
|
|
return CC_OK;
|
|
}
|
|
|
|
|
|
|
|
HRESULT CopyH245TermCapList( PCC_TERMCAPLIST *ppDest,
|
|
PCC_TERMCAPLIST pSource)
|
|
{
|
|
WORD i;
|
|
HRESULT status;
|
|
|
|
ASSERT(ppDest != NULL);
|
|
|
|
if (pSource == NULL) {
|
|
*ppDest = NULL;
|
|
return CC_OK;
|
|
}
|
|
|
|
*ppDest = (PCC_TERMCAPLIST)Malloc(sizeof(CC_TERMCAPLIST));
|
|
if (*ppDest == NULL)
|
|
return CC_NO_MEMORY;
|
|
|
|
(*ppDest)->wLength = pSource->wLength;
|
|
(*ppDest)->pTermCapArray =
|
|
(PPCC_TERMCAP)Malloc(sizeof(PCC_TERMCAP) * pSource->wLength);
|
|
if ((*ppDest)->pTermCapArray == NULL) {
|
|
(*ppDest)->wLength = 0;
|
|
DestroyH245TermCapList(ppDest);
|
|
return CC_NO_MEMORY;
|
|
}
|
|
for (i = 0; i < pSource->wLength; i++) {
|
|
status = H245CopyCap(&((*ppDest)->pTermCapArray[i]), pSource->pTermCapArray[i]);
|
|
if (status != H245_ERROR_OK) {
|
|
(*ppDest)->wLength = i;
|
|
DestroyH245TermCapList(ppDest);
|
|
return status;
|
|
}
|
|
}
|
|
return CC_OK;
|
|
}
|
|
|
|
|
|
|
|
HRESULT CopyH245TermCapDescriptors( PCC_TERMCAPDESCRIPTORS *ppDest,
|
|
PCC_TERMCAPDESCRIPTORS pSource)
|
|
{
|
|
WORD i;
|
|
HRESULT status;
|
|
|
|
ASSERT(ppDest != NULL);
|
|
|
|
if (pSource == NULL) {
|
|
*ppDest = NULL;
|
|
return CC_OK;
|
|
}
|
|
|
|
(*ppDest) = (PCC_TERMCAPDESCRIPTORS)Malloc(sizeof(CC_TERMCAPDESCRIPTORS));
|
|
if (*ppDest == NULL)
|
|
return CC_NO_MEMORY;
|
|
|
|
(*ppDest)->wLength = pSource->wLength;
|
|
(*ppDest)->pTermCapDescriptorArray = (H245_TOTCAPDESC_T **)Malloc(sizeof(H245_TOTCAPDESC_T *) *
|
|
pSource->wLength);
|
|
if ((*ppDest)->pTermCapDescriptorArray == NULL) {
|
|
(*ppDest)->wLength = 0;
|
|
DestroyH245TermCapDescriptors(ppDest);
|
|
return CC_NO_MEMORY;
|
|
}
|
|
|
|
for (i = 0; i < pSource->wLength; i++) {
|
|
status = H245CopyCapDescriptor(&((*ppDest)->pTermCapDescriptorArray[i]),
|
|
pSource->pTermCapDescriptorArray[i]);
|
|
if (status != H245_ERROR_OK) {
|
|
(*ppDest)->wLength = i;
|
|
DestroyH245TermCapDescriptors(ppDest);
|
|
return status;
|
|
}
|
|
}
|
|
return CC_OK;
|
|
}
|
|
|
|
|
|
|
|
HRESULT CreateH245DefaultTermCapDescriptors(
|
|
PCC_TERMCAPDESCRIPTORS *ppDest,
|
|
PCC_TERMCAPLIST pTermCapList)
|
|
{
|
|
H245_TOTCAPDESC_T TermCapDescriptor;
|
|
WORD i;
|
|
HRESULT status;
|
|
|
|
ASSERT(ppDest != NULL);
|
|
|
|
if (pTermCapList == NULL) {
|
|
*ppDest = NULL;
|
|
return CC_OK;
|
|
}
|
|
|
|
*ppDest = (PCC_TERMCAPDESCRIPTORS)Malloc(sizeof(CC_TERMCAPDESCRIPTORS));
|
|
if (*ppDest == NULL)
|
|
return CC_NO_MEMORY;
|
|
|
|
if (pTermCapList->wLength == 0) {
|
|
(*ppDest)->wLength = 0;
|
|
(*ppDest)->pTermCapDescriptorArray = NULL;
|
|
return CC_OK;
|
|
}
|
|
|
|
(*ppDest)->wLength = 1;
|
|
(*ppDest)->pTermCapDescriptorArray = (H245_TOTCAPDESC_T **)Malloc(sizeof(H245_TOTCAPDESC_T *));
|
|
if ((*ppDest)->pTermCapDescriptorArray == NULL) {
|
|
(*ppDest)->wLength = 0;
|
|
DestroyH245TermCapDescriptors(ppDest);
|
|
return CC_NO_MEMORY;
|
|
}
|
|
|
|
TermCapDescriptor.CapDesc.Length = pTermCapList->wLength;
|
|
TermCapDescriptor.CapDescId = 0;
|
|
|
|
for (i = 0; i < pTermCapList->wLength; i++) {
|
|
TermCapDescriptor.CapDesc.SimCapArray[i].Length = 1;
|
|
TermCapDescriptor.CapDesc.SimCapArray[i].AltCaps[0] =
|
|
pTermCapList->pTermCapArray[i]->CapId;
|
|
}
|
|
|
|
status = H245CopyCapDescriptor(&((*ppDest)->pTermCapDescriptorArray[0]),
|
|
&TermCapDescriptor);
|
|
if (status != H245_ERROR_OK) {
|
|
(*ppDest)->wLength = 0;
|
|
DestroyH245TermCapDescriptors(ppDest);
|
|
return status;
|
|
}
|
|
return CC_OK;
|
|
}
|
|
|
|
|
|
|
|
HRESULT DestroyH245TermCap( PPCC_TERMCAP ppTermCap)
|
|
{
|
|
ASSERT(ppTermCap != NULL);
|
|
|
|
if (*ppTermCap == NULL)
|
|
return CC_OK;
|
|
|
|
H245FreeCap(*ppTermCap);
|
|
*ppTermCap = NULL;
|
|
return CC_OK;
|
|
}
|
|
|
|
|
|
|
|
HRESULT UnregisterTermCapListFromH245(
|
|
PCONFERENCE pConference,
|
|
PCC_TERMCAPLIST pTermCapList)
|
|
{
|
|
WORD i, j;
|
|
PCALL pCall;
|
|
PCC_HCALL CallList;
|
|
WORD wNumCalls;
|
|
HRESULT status;
|
|
HRESULT SaveStatus;
|
|
|
|
ASSERT(pConference != NULL);
|
|
|
|
if (pTermCapList == NULL)
|
|
return CC_OK;
|
|
|
|
EnumerateCallsInConference(&wNumCalls, &CallList, pConference, ESTABLISHED_CALL);
|
|
|
|
SaveStatus = CC_OK;
|
|
for (i = 0; i < pTermCapList->wLength; i++) {
|
|
ASSERT(pTermCapList->pTermCapArray[i] != NULL);
|
|
for (j = 0; j < wNumCalls; j++) {
|
|
if (LockCall(CallList[j], &pCall) == CC_OK) {
|
|
status = H245DelLocalCap(pCall->H245Instance,
|
|
pTermCapList->pTermCapArray[i]->CapId);
|
|
if (status != CC_OK)
|
|
SaveStatus = status;
|
|
UnlockCall(pCall);
|
|
}
|
|
}
|
|
}
|
|
if (CallList != NULL)
|
|
Free(CallList);
|
|
return SaveStatus;
|
|
}
|
|
|
|
|
|
|
|
HRESULT DestroyH245TermCapList( PCC_TERMCAPLIST *ppTermCapList)
|
|
{
|
|
WORD i;
|
|
|
|
ASSERT(ppTermCapList != NULL);
|
|
|
|
if (*ppTermCapList == NULL)
|
|
return CC_OK;
|
|
|
|
for (i = 0; i < (*ppTermCapList)->wLength; i++) {
|
|
ASSERT((*ppTermCapList)->pTermCapArray[i] != NULL);
|
|
H245FreeCap((*ppTermCapList)->pTermCapArray[i]);
|
|
}
|
|
if ((*ppTermCapList)->pTermCapArray != NULL)
|
|
Free((*ppTermCapList)->pTermCapArray);
|
|
Free(*ppTermCapList);
|
|
*ppTermCapList = NULL;
|
|
return CC_OK;
|
|
}
|
|
|
|
|
|
|
|
HRESULT UnregisterTermCapDescriptorsFromH245(
|
|
PCONFERENCE pConference,
|
|
PCC_TERMCAPDESCRIPTORS pTermCapDescriptors)
|
|
{
|
|
WORD i, j;
|
|
PCALL pCall;
|
|
PCC_HCALL CallList;
|
|
WORD wNumCalls;
|
|
HRESULT status;
|
|
HRESULT SaveStatus;
|
|
|
|
ASSERT(pConference != NULL);
|
|
|
|
if (pTermCapDescriptors == NULL)
|
|
return CC_OK;
|
|
|
|
EnumerateCallsInConference(&wNumCalls, &CallList, pConference, ESTABLISHED_CALL);
|
|
|
|
SaveStatus = CC_OK;
|
|
for (i = 0; i < pTermCapDescriptors->wLength; i++) {
|
|
ASSERT(pTermCapDescriptors->pTermCapDescriptorArray[i] != NULL);
|
|
for (j = 0; j < wNumCalls; j++) {
|
|
if (LockCall(CallList[j], &pCall) == CC_OK) {
|
|
status = H245DelCapDescriptor(pCall->H245Instance,
|
|
pTermCapDescriptors->pTermCapDescriptorArray[i]->CapDescId);
|
|
if (status != CC_OK)
|
|
SaveStatus = status;
|
|
UnlockCall(pCall);
|
|
}
|
|
}
|
|
}
|
|
if (CallList != NULL)
|
|
Free(CallList);
|
|
return SaveStatus;
|
|
}
|
|
|
|
|
|
|
|
HRESULT DestroyH245TermCapDescriptors( PCC_TERMCAPDESCRIPTORS *ppTermCapDescriptors)
|
|
{
|
|
WORD i;
|
|
|
|
ASSERT(ppTermCapDescriptors != NULL);
|
|
|
|
if (*ppTermCapDescriptors == NULL)
|
|
return CC_OK;
|
|
|
|
for (i = 0; i < (*ppTermCapDescriptors)->wLength; i++) {
|
|
ASSERT((*ppTermCapDescriptors)->pTermCapDescriptorArray[i] != NULL);
|
|
H245FreeCapDescriptor((*ppTermCapDescriptors)->pTermCapDescriptorArray[i]);
|
|
}
|
|
if ((*ppTermCapDescriptors)->pTermCapDescriptorArray != NULL)
|
|
Free((*ppTermCapDescriptors)->pTermCapDescriptorArray);
|
|
Free(*ppTermCapDescriptors);
|
|
*ppTermCapDescriptors = NULL;
|
|
return CC_OK;
|
|
}
|
|
|
|
|
|
|
|
HRESULT HostToH245IPNetwork( BYTE *NetworkArray,
|
|
DWORD dwAddr)
|
|
{
|
|
if (NetworkArray == NULL) {
|
|
ASSERT(0);
|
|
return CC_BAD_PARAM;
|
|
}
|
|
|
|
NetworkArray[0] = HIBYTE(HIWORD(dwAddr));
|
|
NetworkArray[1] = LOBYTE(HIWORD(dwAddr));
|
|
NetworkArray[2] = HIBYTE(LOWORD(dwAddr));
|
|
NetworkArray[3] = LOBYTE(LOWORD(dwAddr));
|
|
|
|
return CC_OK;
|
|
}
|
|
|
|
|
|
|
|
HRESULT H245IPNetworkToHost( DWORD *pdwAddr,
|
|
BYTE *NetworkArray)
|
|
{
|
|
if ((pdwAddr == NULL) || (NetworkArray == NULL)) {
|
|
ASSERT(0);
|
|
return CC_BAD_PARAM;
|
|
}
|
|
|
|
*pdwAddr = NetworkArray[0] * 0x01000000 +
|
|
NetworkArray[1] * 0x00010000 +
|
|
NetworkArray[2] * 0x00000100 +
|
|
NetworkArray[3] * 0x00000001;
|
|
|
|
return CC_OK;
|
|
}
|
|
|
|
|
|
|
|
HRESULT ProcessRemoteHangup( CC_HCALL hCall,
|
|
HQ931CALL hQ931Initiator,
|
|
BYTE bHangupReason)
|
|
{
|
|
PCALL pCall;
|
|
CC_HCONFERENCE hConference;
|
|
PCONFERENCE pConference;
|
|
HRESULT status;
|
|
HQ931CALL hQ931Call;
|
|
H245_INST_T H245Instance;
|
|
PCHANNEL pChannel;
|
|
WORD wNumChannels;
|
|
PCC_HCHANNEL ChannelList;
|
|
WORD i;
|
|
WORD wNumCalls;
|
|
PCC_HCALL CallList;
|
|
PCALL pOldCall;
|
|
CC_CONNECT_CALLBACK_PARAMS ConnectCallbackParams;
|
|
CC_PEER_DROP_CALLBACK_PARAMS PeerDropCallbackParams;
|
|
CC_PEER_CHANGE_CAP_CALLBACK_PARAMS PeerChangeCapCallbackParams;
|
|
BOOL bConferenceTermCapsChanged;
|
|
HRESULT CallbackStatus;
|
|
|
|
if (hCall == CC_INVALID_HANDLE)
|
|
return CC_BAD_PARAM;
|
|
|
|
status = LockCallAndConference(hCall, &pCall, &pConference);
|
|
if (status != CC_OK)
|
|
return CC_BAD_PARAM;
|
|
|
|
hConference = pCall->hConference;
|
|
hQ931Call = pCall->hQ931Call;
|
|
H245Instance = pCall->H245Instance;
|
|
PeerDropCallbackParams.hCall = pCall->hCall;
|
|
if (pCall->pPeerParticipantInfo == NULL) {
|
|
PeerDropCallbackParams.TerminalLabel.bMCUNumber = 255;
|
|
PeerDropCallbackParams.TerminalLabel.bTerminalNumber = 255;
|
|
PeerDropCallbackParams.pPeerTerminalID = NULL;
|
|
} else {
|
|
PeerDropCallbackParams.TerminalLabel = pCall->pPeerParticipantInfo->ParticipantInfo.TerminalLabel;
|
|
if (pCall->pPeerParticipantInfo->TerminalIDState == TERMINAL_ID_VALID)
|
|
PeerDropCallbackParams.pPeerTerminalID = &pCall->pPeerParticipantInfo->ParticipantInfo.TerminalID;
|
|
else
|
|
PeerDropCallbackParams.pPeerTerminalID = NULL;
|
|
}
|
|
|
|
if ((pConference->ConferenceMode == MULTIPOINT_MODE) &&
|
|
(pConference->tsMultipointController == TS_TRUE))
|
|
CreateConferenceTermCaps(pConference, &bConferenceTermCapsChanged);
|
|
else
|
|
bConferenceTermCapsChanged = FALSE;
|
|
|
|
// Remove all TX, RX and PROXY channels associated with this call
|
|
EnumerateChannelsInConference(&wNumChannels,
|
|
&ChannelList,
|
|
pConference,
|
|
TX_CHANNEL | RX_CHANNEL | PROXY_CHANNEL);
|
|
for (i = 0; i < wNumChannels; i++) {
|
|
if (LockChannel(ChannelList[i], &pChannel) == CC_OK) {
|
|
if (pChannel->hCall == hCall)
|
|
FreeChannel(pChannel);
|
|
else
|
|
UnlockChannel(pChannel);
|
|
}
|
|
}
|
|
if (ChannelList != NULL)
|
|
Free(ChannelList);
|
|
|
|
switch (bHangupReason)
|
|
{
|
|
case CC_REJECT_NORMAL_CALL_CLEARING:
|
|
CallbackStatus = CC_OK;
|
|
break;
|
|
case CC_REJECT_GATEKEEPER_TERMINATED:
|
|
CallbackStatus = CC_GATEKEEPER_REFUSED;
|
|
bHangupReason = CC_REJECT_NORMAL_CALL_CLEARING;
|
|
break;
|
|
default:
|
|
CallbackStatus = CC_PEER_REJECT;
|
|
} // switch
|
|
|
|
if (pCall->CallType == THIRD_PARTY_INVITOR) {
|
|
MarkCallForDeletion(pCall);
|
|
|
|
ConnectCallbackParams.pNonStandardData = pCall->pPeerNonStandardData;
|
|
ConnectCallbackParams.pszPeerDisplay = pCall->pszPeerDisplay;
|
|
ConnectCallbackParams.bRejectReason = bHangupReason;
|
|
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 = TRUE;
|
|
ConnectCallbackParams.pConferenceID = &pConference->ConferenceID;
|
|
ConnectCallbackParams.pMCAddress = pConference->pMultipointControllerAddr;
|
|
ConnectCallbackParams.pAlternateAddress = NULL;
|
|
ConnectCallbackParams.dwUserToken = pCall->dwUserToken;
|
|
InvokeUserConferenceCallback(pConference,
|
|
CC_CONNECT_INDICATION,
|
|
CallbackStatus,
|
|
&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 (ValidateCallMarkedForDeletion(hCall) == CC_OK)
|
|
FreeCall(pCall);
|
|
Q931Hangup(hQ931Call, bHangupReason);
|
|
return CC_OK;
|
|
}
|
|
|
|
if (pCall->CallType == THIRD_PARTY_INTERMEDIARY) {
|
|
if ((hQ931Initiator == pCall->hQ931CallInvitor) &&
|
|
(hQ931Initiator != CC_INVALID_HANDLE)) {
|
|
pCall->hQ931CallInvitor = CC_INVALID_HANDLE;
|
|
UnlockCall(pCall);
|
|
UnlockConference(pConference);
|
|
return CC_OK;
|
|
} else {
|
|
if (pCall->CallState != CALL_COMPLETE) {
|
|
if (pCall->hQ931CallInvitor != CC_INVALID_HANDLE)
|
|
Q931Hangup(pCall->hQ931CallInvitor, CC_REJECT_UNDEFINED_REASON);
|
|
if (ValidateCall(hCall) == CC_OK)
|
|
Q931Hangup(pCall->hQ931Call, bHangupReason);
|
|
}
|
|
}
|
|
}
|
|
|
|
if ((pConference->ConferenceMode == MULTIPOINT_MODE) &&
|
|
(pConference->tsMultipointController == TS_TRUE)) {
|
|
if (pCall->pPeerParticipantInfo != NULL) {
|
|
EnumerateCallsInConference(&wNumCalls,
|
|
&CallList,
|
|
pConference,
|
|
ESTABLISHED_CALL);
|
|
for (i = 0; i < wNumCalls; i++) {
|
|
if (CallList[i] != hCall) {
|
|
if (LockCall(CallList[i], &pOldCall) == CC_OK) {
|
|
if (pCall->pPeerParticipantInfo != NULL)
|
|
H245ConferenceIndication(
|
|
pOldCall->H245Instance,
|
|
H245_IND_TERMINAL_LEFT, // Indication Type
|
|
0, // SBE number; ignored here
|
|
pCall->pPeerParticipantInfo->ParticipantInfo.TerminalLabel.bMCUNumber, // MCU number
|
|
pCall->pPeerParticipantInfo->ParticipantInfo.TerminalLabel.bTerminalNumber);
|
|
if (bConferenceTermCapsChanged)
|
|
// Send new term caps
|
|
SendTermCaps(pOldCall, pConference);
|
|
UnlockCall(pOldCall);
|
|
}
|
|
}
|
|
}
|
|
if (CallList != NULL)
|
|
Free(CallList);
|
|
}
|
|
|
|
InvokeUserConferenceCallback(pConference,
|
|
CC_PEER_DROP_INDICATION,
|
|
CC_OK,
|
|
&PeerDropCallbackParams);
|
|
|
|
if (ValidateCall(hCall) == CC_OK)
|
|
FreeCall(pCall);
|
|
|
|
if (ValidateConference(hConference) == CC_OK) {
|
|
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 (pConference->bDeferredDelete) {
|
|
ASSERT(pConference->LocalEndpointAttached == DETACHED);
|
|
EnumerateCallsInConference(&wNumCalls, NULL, pConference, ALL_CALLS);
|
|
if (wNumCalls == 0) {
|
|
FreeConference(pConference);
|
|
return CC_OK;
|
|
}
|
|
}
|
|
UnlockConference(pConference);
|
|
}
|
|
return CC_OK;
|
|
} else {
|
|
status = EnumerateChannelsInConference(&wNumChannels,
|
|
&ChannelList,
|
|
pConference,
|
|
ALL_CHANNELS);
|
|
if (status == CC_OK) {
|
|
// free all the channels
|
|
for (i = 0; i < wNumChannels; i++) {
|
|
if (LockChannel(ChannelList[i], &pChannel) == CC_OK)
|
|
// Notice that since we're going to hangup, we don't need to
|
|
// close any channels
|
|
FreeChannel(pChannel);
|
|
}
|
|
if (ChannelList != NULL)
|
|
Free(ChannelList);
|
|
}
|
|
|
|
if (H245Instance != H245_INVALID_ID)
|
|
status = H245ShutDown(H245Instance);
|
|
else
|
|
status = H245_ERROR_OK;
|
|
|
|
if (status == H245_ERROR_OK) {
|
|
status = Q931Hangup(hQ931Call, CC_REJECT_NORMAL_CALL_CLEARING);
|
|
// 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);
|
|
|
|
if (pConference->bDeferredDelete) {
|
|
ASSERT(pConference->LocalEndpointAttached == DETACHED);
|
|
FreeConference(pConference);
|
|
} else {
|
|
ReInitializeConference(pConference);
|
|
|
|
InvokeUserConferenceCallback(pConference,
|
|
CC_CONFERENCE_TERMINATION_INDICATION,
|
|
CallbackStatus,
|
|
NULL);
|
|
|
|
if (ValidateConference(hConference) == CC_OK)
|
|
UnlockConference(pConference);
|
|
}
|
|
return CC_OK;
|
|
}
|
|
// We should never reach this point
|
|
ASSERT(0);
|
|
}
|
|
|
|
|
|
|
|
HRESULT DefaultSessionTableConstructor(
|
|
CC_HCONFERENCE hConference,
|
|
DWORD dwConferenceToken,
|
|
BOOL bCreate,
|
|
BOOL *pbSessionTableChanged,
|
|
WORD wListCount,
|
|
PCC_TERMCAPLIST pTermCapList[],
|
|
PCC_TERMCAPDESCRIPTORS pTermCapDescriptors[],
|
|
PCC_SESSIONTABLE *ppSessionTable)
|
|
{
|
|
WORD i;
|
|
HRESULT status;
|
|
PCONFERENCE pConference;
|
|
WORD wNumChannels;
|
|
PCC_HCHANNEL ChannelList;
|
|
PCHANNEL pChannel;
|
|
WORD wNumCalls;
|
|
PCC_HCALL CallList;
|
|
PCALL pCall;
|
|
BYTE bSessionID;
|
|
WORD wPort;
|
|
DWORD dwAddr;
|
|
WCHAR szSessionDescription[100];
|
|
WCHAR ss[10];
|
|
|
|
ASSERT(hConference != CC_INVALID_HANDLE);
|
|
ASSERT(ppSessionTable != NULL);
|
|
|
|
if (*ppSessionTable != NULL) {
|
|
for (i = 0; i < (*ppSessionTable)->wLength; i++) {
|
|
if ((*ppSessionTable)->SessionInfoArray[i].pTermCap != NULL)
|
|
H245FreeCap((*ppSessionTable)->SessionInfoArray[i].pTermCap);
|
|
FreeAddr((*ppSessionTable)->SessionInfoArray[i].pRTPAddr);
|
|
FreeAddr((*ppSessionTable)->SessionInfoArray[i].pRTCPAddr);
|
|
}
|
|
if ((*ppSessionTable)->SessionInfoArray != NULL)
|
|
Free((*ppSessionTable)->SessionInfoArray);
|
|
Free(*ppSessionTable);
|
|
*ppSessionTable = NULL;
|
|
}
|
|
|
|
if (bCreate == FALSE)
|
|
return CC_OK;
|
|
|
|
*ppSessionTable = NULL;
|
|
if (pbSessionTableChanged != NULL)
|
|
*pbSessionTableChanged = FALSE;
|
|
|
|
status = LockConference(hConference, &pConference);
|
|
if (status != CC_OK)
|
|
return status;
|
|
|
|
if ((pConference->ConferenceMode == UNCONNECTED_MODE) ||
|
|
(pConference->ConferenceMode == POINT_TO_POINT_MODE)) {
|
|
UnlockConference(pConference);
|
|
return CC_BAD_PARAM;
|
|
}
|
|
|
|
// pConference->ConferenceMode == MULTIPOINT_MODE
|
|
// Create one session entry for each open channel on this conference
|
|
|
|
bSessionID = 1;
|
|
wPort = 2050;
|
|
|
|
// Set dwAddr
|
|
EnumerateCallsInConference(&wNumCalls, &CallList, pConference, ESTABLISHED_CALL);
|
|
if (wNumCalls == 0) {
|
|
UnlockConference(pConference);
|
|
return CC_INTERNAL_ERROR;
|
|
}
|
|
|
|
status = LockCall(CallList[0], &pCall);
|
|
if (status != CC_OK) {
|
|
Free(CallList);
|
|
UnlockConference(pConference);
|
|
return status;
|
|
}
|
|
|
|
if (pCall->pQ931LocalConnectAddr == NULL) {
|
|
Free(CallList);
|
|
UnlockCall(pCall);
|
|
UnlockConference(pConference);
|
|
return CC_INTERNAL_ERROR;
|
|
}
|
|
|
|
if (pCall->pQ931LocalConnectAddr->nAddrType != CC_IP_BINARY) {
|
|
Free(CallList);
|
|
UnlockCall(pCall);
|
|
UnlockConference(pConference);
|
|
return CC_INTERNAL_ERROR;
|
|
}
|
|
|
|
// Construct dwAddr from one of the unicast Q.931 addresses by setting the high
|
|
// nibble of the Q.931 address to 0xE
|
|
dwAddr = (pCall->pQ931LocalConnectAddr->Addr.IP_Binary.dwAddr & 0xEFFFFFFF) | 0xE0000000;
|
|
|
|
UnlockCall(pCall);
|
|
Free(CallList);
|
|
|
|
EnumerateChannelsInConference(&wNumChannels, &ChannelList, pConference, TX_CHANNEL);
|
|
|
|
*ppSessionTable = (PCC_SESSIONTABLE)Malloc(sizeof(CC_SESSIONTABLE));
|
|
if (*ppSessionTable == NULL) {
|
|
Free(ChannelList);
|
|
UnlockConference(pConference);
|
|
return CC_NO_MEMORY;
|
|
}
|
|
(*ppSessionTable)->wLength = wNumChannels;
|
|
if (wNumChannels == 0)
|
|
(*ppSessionTable)->SessionInfoArray = NULL;
|
|
else {
|
|
(*ppSessionTable)->SessionInfoArray =
|
|
(PCC_SESSIONINFO)Malloc(sizeof(CC_SESSIONINFO) * wNumChannels);
|
|
if ((*ppSessionTable)->SessionInfoArray == NULL) {
|
|
Free(ChannelList);
|
|
UnlockConference(pConference);
|
|
(*ppSessionTable)->wLength = 0;
|
|
DefaultSessionTableConstructor(
|
|
hConference,
|
|
dwConferenceToken,
|
|
FALSE, // bCreate,
|
|
NULL, // pbSessionTableChanged,
|
|
0, // wListCount,
|
|
NULL, // pTermCapList[],
|
|
NULL, // pTermCapDescriptors[],
|
|
ppSessionTable);
|
|
return CC_NO_MEMORY;
|
|
}
|
|
for (i = 0; i < wNumChannels; i++) {
|
|
(*ppSessionTable)->SessionInfoArray[i].bSessionID = bSessionID++;
|
|
|
|
(*ppSessionTable)->SessionInfoArray[i].bAssociatedSessionID = 0;
|
|
|
|
wcscpy(szSessionDescription, L"Session ");
|
|
_itow((int)(*ppSessionTable)->SessionInfoArray[i].bSessionID,
|
|
ss, 10);
|
|
wcscat(szSessionDescription, ss);
|
|
|
|
(*ppSessionTable)->SessionInfoArray[i].SessionDescription.wOctetStringLength =
|
|
(WORD)((wcslen(szSessionDescription)+1)*sizeof(WCHAR));
|
|
(*ppSessionTable)->SessionInfoArray[i].SessionDescription.pOctetString =
|
|
(BYTE *)Malloc((*ppSessionTable)->SessionInfoArray[i].SessionDescription.wOctetStringLength);
|
|
if ((*ppSessionTable)->SessionInfoArray[i].SessionDescription.pOctetString == NULL) {
|
|
Free(ChannelList);
|
|
UnlockConference(pConference);
|
|
(*ppSessionTable)->wLength = i;
|
|
DefaultSessionTableConstructor(
|
|
hConference,
|
|
dwConferenceToken,
|
|
FALSE, // bCreate,
|
|
NULL, // pbSessionTableChanged,
|
|
0, // wListCount,
|
|
NULL, // pTermCapList[],
|
|
NULL, // pTermCapDescriptors[],
|
|
ppSessionTable);
|
|
return CC_NO_MEMORY;
|
|
}
|
|
memcpy((*ppSessionTable)->SessionInfoArray[i].SessionDescription.pOctetString,
|
|
szSessionDescription,
|
|
(*ppSessionTable)->SessionInfoArray[i].SessionDescription.wOctetStringLength);
|
|
|
|
status = LockChannel(ChannelList[i], &pChannel);
|
|
if (status != CC_OK) {
|
|
Free((*ppSessionTable)->SessionInfoArray[i].SessionDescription.pOctetString);
|
|
Free(ChannelList);
|
|
UnlockConference(pConference);
|
|
(*ppSessionTable)->wLength = i;
|
|
DefaultSessionTableConstructor(
|
|
hConference,
|
|
dwConferenceToken,
|
|
FALSE, // bCreate,
|
|
NULL, // pbSessionTableChanged,
|
|
0, // wListCount,
|
|
NULL, // pTermCapList[],
|
|
NULL, // pTermCapDescriptors[],
|
|
ppSessionTable);
|
|
return CC_NO_MEMORY;
|
|
}
|
|
status = H245CopyCap(&(*ppSessionTable)->SessionInfoArray[i].pTermCap,
|
|
pChannel->pTxH245TermCap);
|
|
UnlockChannel(pChannel);
|
|
if (status != H245_ERROR_OK) {
|
|
Free((*ppSessionTable)->SessionInfoArray[i].SessionDescription.pOctetString);
|
|
Free(ChannelList);
|
|
UnlockConference(pConference);
|
|
(*ppSessionTable)->wLength = i;
|
|
DefaultSessionTableConstructor(
|
|
hConference,
|
|
dwConferenceToken,
|
|
FALSE, // bCreate,
|
|
NULL, // pbSessionTableChanged,
|
|
0, // wListCount,
|
|
NULL, // pTermCapList[],
|
|
NULL, // pTermCapDescriptors[],
|
|
ppSessionTable);
|
|
return status;
|
|
}
|
|
|
|
(*ppSessionTable)->SessionInfoArray[i].pRTPAddr =
|
|
(PCC_ADDR)Malloc(sizeof(CC_ADDR));
|
|
if ((*ppSessionTable)->SessionInfoArray[i].pRTPAddr == NULL) {
|
|
H245FreeCap((*ppSessionTable)->SessionInfoArray[i].pTermCap);
|
|
Free((*ppSessionTable)->SessionInfoArray[i].SessionDescription.pOctetString);
|
|
Free(ChannelList);
|
|
UnlockConference(pConference);
|
|
(*ppSessionTable)->wLength = i;
|
|
DefaultSessionTableConstructor(
|
|
hConference,
|
|
dwConferenceToken,
|
|
FALSE, // bCreate,
|
|
NULL, // pbSessionTableChanged,
|
|
0, // wListCount,
|
|
NULL, // pTermCapList[],
|
|
NULL, // pTermCapDescriptors[],
|
|
ppSessionTable);
|
|
return CC_NO_MEMORY;
|
|
}
|
|
(*ppSessionTable)->SessionInfoArray[i].pRTPAddr->nAddrType = CC_IP_BINARY;
|
|
(*ppSessionTable)->SessionInfoArray[i].pRTPAddr->bMulticast = TRUE;
|
|
(*ppSessionTable)->SessionInfoArray[i].pRTPAddr->Addr.IP_Binary.wPort = wPort++;
|
|
(*ppSessionTable)->SessionInfoArray[i].pRTPAddr->Addr.IP_Binary.dwAddr = dwAddr;
|
|
|
|
(*ppSessionTable)->SessionInfoArray[i].pRTCPAddr =
|
|
(PCC_ADDR)Malloc(sizeof(CC_ADDR));
|
|
if ((*ppSessionTable)->SessionInfoArray[i].pRTCPAddr == NULL) {
|
|
H245FreeCap((*ppSessionTable)->SessionInfoArray[i].pTermCap);
|
|
FreeAddr((*ppSessionTable)->SessionInfoArray[i].pRTPAddr);
|
|
Free((*ppSessionTable)->SessionInfoArray[i].SessionDescription.pOctetString);
|
|
Free(ChannelList);
|
|
UnlockConference(pConference);
|
|
(*ppSessionTable)->wLength = i;
|
|
DefaultSessionTableConstructor(
|
|
hConference,
|
|
dwConferenceToken,
|
|
FALSE, // bCreate,
|
|
NULL, // pbSessionTableChanged,
|
|
0, // wListCount,
|
|
NULL, // pTermCapList[],
|
|
NULL, // pTermCapDescriptors[],
|
|
ppSessionTable);
|
|
return CC_NO_MEMORY;
|
|
}
|
|
(*ppSessionTable)->SessionInfoArray[i].pRTCPAddr->nAddrType = CC_IP_BINARY;
|
|
(*ppSessionTable)->SessionInfoArray[i].pRTCPAddr->bMulticast = TRUE;
|
|
(*ppSessionTable)->SessionInfoArray[i].pRTCPAddr->Addr.IP_Binary.wPort = wPort++;
|
|
(*ppSessionTable)->SessionInfoArray[i].pRTCPAddr->Addr.IP_Binary.dwAddr = dwAddr;
|
|
}
|
|
}
|
|
|
|
Free(ChannelList);
|
|
UnlockConference(pConference);
|
|
if (pbSessionTableChanged != NULL)
|
|
*pbSessionTableChanged = TRUE;
|
|
|
|
return CC_OK;
|
|
}
|
|
|
|
|
|
|
|
HRESULT DefaultTermCapConstructor( CC_HCONFERENCE hConference,
|
|
DWORD dwConferenceToken,
|
|
BOOL bCreate,
|
|
BOOL *pbTermCapsChanged,
|
|
WORD wListCount,
|
|
PCC_TERMCAPLIST pInTermCapList[],
|
|
PCC_TERMCAPDESCRIPTORS pInTermCapDescriptors[],
|
|
PCC_TERMCAPLIST *ppOutTermCapList,
|
|
PCC_TERMCAPDESCRIPTORS *ppOutTermCapDescriptors)
|
|
{
|
|
HRESULT status;
|
|
PCONFERENCE pConference;
|
|
WORD wNumChannels;
|
|
PCC_HCHANNEL ChannelList;
|
|
WORD i;
|
|
PCHANNEL pChannel;
|
|
|
|
ASSERT(hConference != CC_INVALID_HANDLE);
|
|
ASSERT(ppOutTermCapList != NULL);
|
|
ASSERT(ppOutTermCapDescriptors != NULL);
|
|
|
|
if (*ppOutTermCapList != NULL) {
|
|
DestroyH245TermCapList(ppOutTermCapList);
|
|
*ppOutTermCapList = NULL;
|
|
}
|
|
|
|
if (*ppOutTermCapDescriptors != NULL) {
|
|
DestroyH245TermCapDescriptors(ppOutTermCapDescriptors);
|
|
*ppOutTermCapDescriptors = NULL;
|
|
}
|
|
|
|
if (bCreate == FALSE)
|
|
return CC_OK;
|
|
|
|
*ppOutTermCapList = NULL;
|
|
*ppOutTermCapDescriptors = NULL;
|
|
if (pbTermCapsChanged != NULL)
|
|
*pbTermCapsChanged = FALSE;
|
|
|
|
status = LockConference(hConference, &pConference);
|
|
if (status != CC_OK)
|
|
return status;
|
|
|
|
if (pConference->LocalEndpointAttached == NEVER_ATTACHED) {
|
|
// Copy the local term caps to the conference term caps
|
|
status = CopyH245TermCapList(ppOutTermCapList, pConference->pLocalH245TermCapList);
|
|
if (status != CC_OK) {
|
|
UnlockConference(pConference);
|
|
return CC_NO_MEMORY;
|
|
}
|
|
|
|
// Copy the local term cap descriptors to the conference term cap descriptors
|
|
status = CopyH245TermCapDescriptors(ppOutTermCapDescriptors, pConference->pLocalH245TermCapDescriptors);
|
|
if (status != CC_OK) {
|
|
UnlockConference(pConference);
|
|
return CC_NO_MEMORY;
|
|
}
|
|
} else { // pConference->LocalEndpointAttached != NEVER_ATTACHED
|
|
// Create one term cap entry for each open channel on this conference
|
|
|
|
EnumerateChannelsInConference(&wNumChannels, &ChannelList, pConference, TX_CHANNEL);
|
|
|
|
*ppOutTermCapList = (PCC_TERMCAPLIST)Malloc(sizeof(CC_TERMCAPLIST));
|
|
if (*ppOutTermCapList == NULL) {
|
|
Free(ChannelList);
|
|
UnlockConference(pConference);
|
|
return CC_NO_MEMORY;
|
|
}
|
|
(*ppOutTermCapList)->wLength = wNumChannels;
|
|
if (wNumChannels == 0)
|
|
(*ppOutTermCapList)->pTermCapArray = NULL;
|
|
else {
|
|
(*ppOutTermCapList)->pTermCapArray =
|
|
(PPCC_TERMCAP)Malloc(sizeof(PCC_TERMCAP) * wNumChannels);
|
|
if ((*ppOutTermCapList)->pTermCapArray == NULL) {
|
|
Free(ChannelList);
|
|
UnlockConference(pConference);
|
|
(*ppOutTermCapList)->wLength = 0;
|
|
DefaultTermCapConstructor(
|
|
hConference,
|
|
dwConferenceToken,
|
|
FALSE, // bCreate
|
|
NULL, // pbTermCapsChanged
|
|
0, // wListCount
|
|
NULL, // pInTermCapList[]
|
|
NULL, // pInTermCapDescriptors[]
|
|
ppOutTermCapList,
|
|
ppOutTermCapDescriptors);
|
|
return CC_NO_MEMORY;
|
|
}
|
|
for (i = 0; i < wNumChannels; i++) {
|
|
status = LockChannel(ChannelList[i], &pChannel);
|
|
if (status != CC_OK) {
|
|
Free(ChannelList);
|
|
UnlockConference(pConference);
|
|
(*ppOutTermCapList)->wLength = i;
|
|
DefaultTermCapConstructor(
|
|
hConference,
|
|
dwConferenceToken,
|
|
FALSE, // bCreate
|
|
NULL, // pbTermCapsChanged
|
|
0, // wListCount
|
|
NULL, // pInTermCapList[]
|
|
NULL, // pInTermCapDescriptors[]
|
|
ppOutTermCapList,
|
|
ppOutTermCapDescriptors);
|
|
return CC_NO_MEMORY;
|
|
}
|
|
status = H245CopyCap(&((*ppOutTermCapList)->pTermCapArray[i]),
|
|
pChannel->pTxH245TermCap);
|
|
UnlockChannel(pChannel);
|
|
if (status != H245_ERROR_OK) {
|
|
Free(ChannelList);
|
|
UnlockConference(pConference);
|
|
(*ppOutTermCapList)->wLength = i;
|
|
DefaultTermCapConstructor(
|
|
hConference,
|
|
dwConferenceToken,
|
|
FALSE, // bCreate
|
|
NULL, // pbTermCapsChanged
|
|
0, // wListCount
|
|
NULL, // pInTermCapList[]
|
|
NULL, // pInTermCapDescriptors[]
|
|
ppOutTermCapList,
|
|
ppOutTermCapDescriptors);
|
|
return status;
|
|
}
|
|
|
|
(*ppOutTermCapList)->pTermCapArray[i]->Dir = H245_CAPDIR_LCLRXTX;
|
|
(*ppOutTermCapList)->pTermCapArray[i]->CapId = (WORD)(i+1);
|
|
}
|
|
}
|
|
|
|
Free(ChannelList);
|
|
UnlockConference(pConference);
|
|
|
|
// create a new descriptor list
|
|
status = CreateH245DefaultTermCapDescriptors(ppOutTermCapDescriptors,
|
|
*ppOutTermCapList);
|
|
if (status != CC_OK) {
|
|
DefaultTermCapConstructor(
|
|
hConference,
|
|
dwConferenceToken,
|
|
FALSE, // bCreate
|
|
NULL, // pbTermCapsChanged
|
|
0, // wListCount
|
|
NULL, // pInTermCapList[]
|
|
NULL, // pInTermCapDescriptors[]
|
|
ppOutTermCapList,
|
|
ppOutTermCapDescriptors);
|
|
return CC_NO_MEMORY;
|
|
}
|
|
} // pConference->LocalEndpointAttached != NEVER_ATTACHED
|
|
|
|
if (pbTermCapsChanged != NULL)
|
|
*pbTermCapsChanged = TRUE;
|
|
|
|
return CC_OK;
|
|
}
|
|
|
|
|
|
|
|
HRESULT AcceptCall( PCALL pCall,
|
|
PCONFERENCE pConference)
|
|
{
|
|
HRESULT status;
|
|
CC_HCALL hCall;
|
|
CC_HCONFERENCE hConference;
|
|
HQ931CALL hQ931Call;
|
|
CC_CONFERENCEID ConferenceID;
|
|
BYTE bTerminalType;
|
|
CC_ADDR H245Addr;
|
|
H245_INST_T H245Instance;
|
|
PCC_VENDORINFO pVendorInfo;
|
|
PCC_NONSTANDARDDATA pNonStandardData;
|
|
PWSTR pszDisplay;
|
|
CC_ENDPOINTTYPE DestinationEndpointType;
|
|
TRISTATE tsMultipointController;
|
|
DWORD dwLinkLayerPhysicalId;
|
|
DWORD dwBandwidth;
|
|
|
|
ASSERT(pCall != NULL);
|
|
ASSERT(pConference != NULL);
|
|
|
|
hCall = pCall->hCall;
|
|
hConference = pConference->hConference;
|
|
hQ931Call = pCall->hQ931Call;
|
|
ConferenceID = pCall->ConferenceID;
|
|
pCall->hConference = pConference->hConference;
|
|
|
|
status = CopyNonStandardData(&pNonStandardData, pCall->pLocalNonStandardData);
|
|
if (status != CC_OK) {
|
|
UnlockConference(pConference);
|
|
FreeCall(pCall);
|
|
Q931RejectCall(hQ931Call, // Q931 call handle
|
|
CC_REJECT_UNDEFINED_REASON, // reject reason
|
|
&ConferenceID,
|
|
NULL, // alternate address
|
|
pNonStandardData); // non-standard data
|
|
return status;
|
|
}
|
|
|
|
status = CopyVendorInfo(&pVendorInfo, pConference->pVendorInfo);
|
|
if (status != CC_OK) {
|
|
UnlockConference(pConference);
|
|
FreeCall(pCall);
|
|
Q931RejectCall(hQ931Call, // Q931 call handle
|
|
CC_REJECT_UNDEFINED_REASON, // reject reason
|
|
&ConferenceID,
|
|
NULL, // alternate address
|
|
pNonStandardData); // non-standard data
|
|
FreeNonStandardData(pNonStandardData);
|
|
return status;
|
|
}
|
|
|
|
status = CopyDisplay(&pszDisplay, pCall->pszLocalDisplay);
|
|
if (status != CC_OK) {
|
|
UnlockConference(pConference);
|
|
FreeCall(pCall);
|
|
Q931RejectCall(hQ931Call, // Q931 call handle
|
|
CC_REJECT_UNDEFINED_REASON, // reject reason
|
|
&ConferenceID,
|
|
NULL, // alternate address
|
|
pNonStandardData); // non-standard data
|
|
FreeNonStandardData(pNonStandardData);
|
|
FreeVendorInfo(pVendorInfo);
|
|
return status;
|
|
}
|
|
|
|
status = MakeH245PhysicalID(&pCall->dwH245PhysicalID);
|
|
if (status != CC_OK) {
|
|
UnlockConference(pConference);
|
|
FreeCall(pCall);
|
|
Q931RejectCall(hQ931Call, // Q931 call handle
|
|
CC_REJECT_UNDEFINED_REASON, // reject reason
|
|
&ConferenceID,
|
|
NULL, // alternate address
|
|
pNonStandardData); // non-standard data
|
|
FreeNonStandardData(pNonStandardData);
|
|
FreeVendorInfo(pVendorInfo);
|
|
FreeDisplay(pszDisplay);
|
|
return status;
|
|
}
|
|
|
|
if (pCall->bCallerIsMC) {
|
|
ASSERT(pConference->tsMultipointController != TS_TRUE);
|
|
ASSERT(pConference->bMultipointCapable == TRUE);
|
|
tsMultipointController = TS_FALSE;
|
|
} else
|
|
tsMultipointController = pConference->tsMultipointController;
|
|
|
|
//MULTITHREAD
|
|
//Use a tmp ID so we don't clobber the chosen H245Id.
|
|
// H245Id=>
|
|
// <= linkLayerId
|
|
dwLinkLayerPhysicalId = INVALID_PHYS_ID;
|
|
|
|
SetTerminalType(tsMultipointController, &bTerminalType);
|
|
pCall->H245Instance = H245Init(H245_CONF_H323, // configuration
|
|
pCall->dwH245PhysicalID, // H245 physical ID
|
|
&dwLinkLayerPhysicalId, // the link layer ID is returned
|
|
hCall, // dwPreserved
|
|
(H245_CONF_IND_CALLBACK_T)H245Callback, // callback
|
|
bTerminalType);
|
|
if (pCall->H245Instance == H245_INVALID_ID) {
|
|
// H245 initialization failure
|
|
UnlockConference(pConference);
|
|
FreeCall(pCall);
|
|
Q931RejectCall(hQ931Call, // Q931 call handle
|
|
CC_REJECT_UNDEFINED_REASON, // reject reason
|
|
&ConferenceID,
|
|
NULL, // alternate address
|
|
pNonStandardData); // non-standard data
|
|
FreeNonStandardData(pNonStandardData);
|
|
FreeVendorInfo(pVendorInfo);
|
|
FreeDisplay(pszDisplay);
|
|
return CC_INTERNAL_ERROR;
|
|
}
|
|
|
|
H245Instance = pCall->H245Instance;
|
|
|
|
// Set the H.245 TCP/IP address to the same IP address on which
|
|
// the Q.931 connection was made; this ensures that if the host
|
|
// is multi-homed, the H.245 will be made on the same IP address
|
|
// as the Q.931 connection. Set the initial H.245 port to zero,
|
|
// so that it will be dynamically determined.
|
|
ASSERT(pCall->pQ931LocalConnectAddr != NULL);
|
|
H245Addr = *pCall->pQ931LocalConnectAddr;
|
|
|
|
switch (pCall->pQ931LocalConnectAddr->nAddrType) {
|
|
case CC_IP_DOMAIN_NAME:
|
|
H245Addr.Addr.IP_DomainName.wPort = 0;
|
|
break;
|
|
case CC_IP_DOT:
|
|
H245Addr.Addr.IP_Dot.wPort = 0;
|
|
break;
|
|
case CC_IP_BINARY:
|
|
H245Addr.Addr.IP_Binary.wPort = 0;
|
|
break;
|
|
default:
|
|
ASSERT(0);
|
|
UnlockConference(pConference);
|
|
FreeCall(pCall);
|
|
H245ShutDown(H245Instance);
|
|
Q931RejectCall(hQ931Call, // Q931 call handle
|
|
CC_REJECT_UNDEFINED_REASON, // reject reason
|
|
&ConferenceID,
|
|
NULL, // alternate address
|
|
pNonStandardData); // non-standard data
|
|
FreeNonStandardData(pNonStandardData);
|
|
FreeVendorInfo(pVendorInfo);
|
|
FreeDisplay(pszDisplay);
|
|
return CC_INTERNAL_ERROR;
|
|
}
|
|
|
|
status = linkLayerListen(&dwLinkLayerPhysicalId,
|
|
H245Instance,
|
|
&H245Addr,
|
|
NULL);
|
|
if (status != NOERROR) {
|
|
UnlockConference(pConference);
|
|
FreeCall(pCall);
|
|
H245ShutDown(H245Instance);
|
|
Q931RejectCall(hQ931Call, // Q931 call handle
|
|
CC_REJECT_UNDEFINED_REASON, // reject reason
|
|
&ConferenceID,
|
|
NULL, // alternate address
|
|
pNonStandardData); // non-standard data
|
|
FreeNonStandardData(pNonStandardData);
|
|
FreeVendorInfo(pVendorInfo);
|
|
FreeDisplay(pszDisplay);
|
|
return status;
|
|
}
|
|
|
|
dwBandwidth = pCall->dwBandwidth;
|
|
|
|
UnlockConference(pConference);
|
|
UnlockCall(pCall);
|
|
|
|
DestinationEndpointType.pVendorInfo = pVendorInfo;
|
|
DestinationEndpointType.bIsTerminal = TRUE;
|
|
DestinationEndpointType.bIsGateway = FALSE;
|
|
|
|
status = Q931AcceptCall(hQ931Call,
|
|
pszDisplay,
|
|
pNonStandardData, // non-standard data
|
|
&DestinationEndpointType,
|
|
&H245Addr, // H245 address
|
|
dwBandwidth,
|
|
hCall); // user token
|
|
FreeNonStandardData(pNonStandardData);
|
|
FreeVendorInfo(pVendorInfo);
|
|
FreeDisplay(pszDisplay);
|
|
if (status != CS_OK) {
|
|
if (LockCall(hCall, &pCall) == CC_OK)
|
|
FreeCall(pCall);
|
|
H245ShutDown(H245Instance);
|
|
return status;
|
|
}
|
|
|
|
status = LockCallAndConference(hCall, &pCall, &pConference);
|
|
if (status != CC_OK) {
|
|
if (LockCall(hCall, &pCall) == CC_OK)
|
|
FreeCall(pCall);
|
|
H245ShutDown(H245Instance);
|
|
Q931Hangup(hQ931Call, CC_REJECT_UNDEFINED_REASON);
|
|
return status;
|
|
}
|
|
|
|
pCall->CallState = TERMCAP;
|
|
pConference->ConferenceID = pCall->ConferenceID;
|
|
|
|
status = AddPlacedCallToConference(pCall, pConference);
|
|
if (status != CC_OK) {
|
|
UnlockConference(pConference);
|
|
FreeCall(pCall);
|
|
H245ShutDown(H245Instance);
|
|
Q931Hangup(hQ931Call, CC_REJECT_UNDEFINED_REASON);
|
|
return status;
|
|
}
|
|
|
|
status = SendTermCaps(pCall, pConference);
|
|
if (status != CC_OK) {
|
|
UnlockConference(pConference);
|
|
FreeCall(pCall);
|
|
H245ShutDown(H245Instance);
|
|
Q931Hangup(hQ931Call, CC_REJECT_UNDEFINED_REASON);
|
|
return status;
|
|
}
|
|
|
|
pCall->OutgoingTermCapState = AWAITING_ACK;
|
|
|
|
if (pCall->MasterSlaveState == MASTER_SLAVE_NOT_STARTED) {
|
|
status = H245InitMasterSlave(H245Instance,
|
|
H245Instance); // returned as dwTransId in the callback
|
|
if (status != H245_ERROR_OK) {
|
|
UnlockConference(pConference);
|
|
FreeCall(pCall);
|
|
H245ShutDown(H245Instance);
|
|
Q931Hangup(hQ931Call, CC_REJECT_UNDEFINED_REASON);
|
|
return status;
|
|
}
|
|
pCall->MasterSlaveState = MASTER_SLAVE_IN_PROGRESS;
|
|
}
|
|
|
|
if (pCall->bCallerIsMC) {
|
|
pConference->tsMultipointController = TS_FALSE;
|
|
pConference->ConferenceMode = MULTIPOINT_MODE;
|
|
}
|
|
|
|
UnlockConference(pConference);
|
|
UnlockCall(pCall);
|
|
return CC_OK;
|
|
}
|
|
|
|
|
|
|
|
HRESULT PlaceCall( PCALL pCall,
|
|
PCONFERENCE pConference)
|
|
{
|
|
CC_HCALL hCall;
|
|
HRESULT status;
|
|
WORD wGoal;
|
|
HQ931CALL hQ931Call;
|
|
PCC_ALIASNAMES pCallerAliasNames;
|
|
PCC_ALIASNAMES pCalleeAliasNames;
|
|
PCC_ALIASNAMES pCalleeExtraAliasNames;
|
|
PCC_ALIASITEM pCalleeExtension;
|
|
PCC_VENDORINFO pVendorInfo;
|
|
PWSTR pszDisplay;
|
|
PCC_NONSTANDARDDATA pNonStandardData;
|
|
WORD wNumCalls;
|
|
PCC_ADDR pConnectAddr;
|
|
PCC_ADDR pDestinationAddr;
|
|
CC_ADDR SourceAddr;
|
|
CC_ENDPOINTTYPE SourceEndpointType;
|
|
BOOL bCallerIsMC;
|
|
WORD wCallType;
|
|
|
|
ASSERT(pCall != NULL);
|
|
|
|
hCall = pCall->hCall;
|
|
|
|
if (pCall->CallState == ENQUEUED) {
|
|
// Enqueue the call on the conference object and HResultLeaveCallControl.
|
|
// There will be exactly one placed call for this conference,
|
|
// which is in the process of being placed. If this call placement
|
|
// completes successfully, all enqueued calls will then be placed.
|
|
// If this call placement fails or is terminated, one enqueued call
|
|
// will be placed
|
|
status = AddEnqueuedCallToConference(pCall, pConference);
|
|
return status;
|
|
}
|
|
|
|
// CallState == PLACED
|
|
EnumerateCallsInConference(&wNumCalls,
|
|
NULL,
|
|
pConference,
|
|
PLACED_CALL | ESTABLISHED_CALL);
|
|
if (EqualConferenceIDs(&pConference->ConferenceID, &InvalidConferenceID))
|
|
wGoal = CSG_CREATE;
|
|
else if ((wNumCalls == 0) && (pConference->tsMultipointController != TS_TRUE))
|
|
wGoal = CSG_JOIN;
|
|
else
|
|
wGoal = CSG_INVITE;
|
|
|
|
status = AddPlacedCallToConference(pCall, pConference);
|
|
if (status != CC_OK)
|
|
return status;
|
|
|
|
status = CopyAddr(&pConnectAddr, pCall->pQ931PeerConnectAddr);
|
|
if (status != CC_OK)
|
|
return status;
|
|
|
|
status = CopyAddr(&pDestinationAddr, pCall->pQ931DestinationAddr);
|
|
if (status != CC_OK) {
|
|
FreeAddr(pConnectAddr);
|
|
return status;
|
|
}
|
|
|
|
status = Q931CopyAliasNames(&pCallerAliasNames, pCall->pLocalAliasNames);
|
|
if (status != CS_OK) {
|
|
FreeAddr(pConnectAddr);
|
|
FreeAddr(pDestinationAddr);
|
|
return status;
|
|
}
|
|
|
|
status = CopyVendorInfo(&pVendorInfo, pConference->pVendorInfo);
|
|
if (status != CC_OK) {
|
|
FreeAddr(pConnectAddr);
|
|
FreeAddr(pDestinationAddr);
|
|
Q931FreeAliasNames(pCallerAliasNames);
|
|
return status;
|
|
}
|
|
|
|
status = CopyDisplay(&pszDisplay, pCall->pszLocalDisplay);
|
|
if (status != CC_OK) {
|
|
FreeAddr(pConnectAddr);
|
|
FreeAddr(pDestinationAddr);
|
|
Q931FreeAliasNames(pCallerAliasNames);
|
|
FreeVendorInfo(pVendorInfo);
|
|
return status;
|
|
}
|
|
|
|
status = Q931CopyAliasNames(&pCalleeAliasNames, pCall->pPeerAliasNames);
|
|
if (status != CS_OK) {
|
|
FreeAddr(pConnectAddr);
|
|
FreeAddr(pDestinationAddr);
|
|
Q931FreeAliasNames(pCallerAliasNames);
|
|
FreeVendorInfo(pVendorInfo);
|
|
FreeDisplay(pszDisplay);
|
|
return status;
|
|
}
|
|
|
|
status = Q931CopyAliasNames(&pCalleeExtraAliasNames,
|
|
pCall->pPeerExtraAliasNames);
|
|
if (status != CS_OK) {
|
|
FreeAddr(pConnectAddr);
|
|
FreeAddr(pDestinationAddr);
|
|
Q931FreeAliasNames(pCallerAliasNames);
|
|
FreeVendorInfo(pVendorInfo);
|
|
FreeDisplay(pszDisplay);
|
|
Q931FreeAliasNames(pCalleeAliasNames);
|
|
return status;
|
|
}
|
|
|
|
status = Q931CopyAliasItem(&pCalleeExtension,
|
|
pCall->pPeerExtension);
|
|
if (status != CS_OK) {
|
|
FreeAddr(pConnectAddr);
|
|
FreeAddr(pDestinationAddr);
|
|
Q931FreeAliasNames(pCallerAliasNames);
|
|
FreeVendorInfo(pVendorInfo);
|
|
FreeDisplay(pszDisplay);
|
|
Q931FreeAliasNames(pCalleeAliasNames);
|
|
Q931FreeAliasNames(pCalleeExtraAliasNames);
|
|
return status;
|
|
}
|
|
|
|
status = CopyNonStandardData(&pNonStandardData, pCall->pLocalNonStandardData);
|
|
if (status != CC_OK) {
|
|
FreeAddr(pConnectAddr);
|
|
FreeAddr(pDestinationAddr);
|
|
Q931FreeAliasNames(pCallerAliasNames);
|
|
FreeVendorInfo(pVendorInfo);
|
|
FreeDisplay(pszDisplay);
|
|
Q931FreeAliasNames(pCalleeAliasNames);
|
|
Q931FreeAliasNames(pCalleeExtraAliasNames);
|
|
Q931FreeAliasItem(pCalleeExtension);
|
|
return status;
|
|
}
|
|
|
|
bCallerIsMC = (pConference->tsMultipointController == TS_TRUE ? TRUE : FALSE);
|
|
// Note that if pConference->ConferenceMode == POINT_TO_POINT_MODE, this call attempt
|
|
// will result in a multipoint call if successful, so set the wCallType accordingly
|
|
wCallType = (WORD)((pConference->ConferenceMode == UNCONNECTED_MODE) ? CC_CALLTYPE_PT_PT : CC_CALLTYPE_N_N);
|
|
|
|
SourceEndpointType.pVendorInfo = pVendorInfo;
|
|
SourceEndpointType.bIsTerminal = TRUE;
|
|
SourceEndpointType.bIsGateway = FALSE;
|
|
|
|
// Cause our local Q.931 connect address to be placed in the
|
|
// Q.931 setup-UUIE sourceAddress field
|
|
SourceAddr.nAddrType = CC_IP_BINARY;
|
|
SourceAddr.bMulticast = FALSE;
|
|
SourceAddr.Addr.IP_Binary.dwAddr = 0;
|
|
SourceAddr.Addr.IP_Binary.wPort = 0;
|
|
|
|
status = Q931PlaceCall(&hQ931Call, // Q931 call handle
|
|
pszDisplay,
|
|
pCallerAliasNames,
|
|
pCalleeAliasNames,
|
|
pCalleeExtraAliasNames, // pCalleeExtraAliasNames
|
|
pCalleeExtension, // pCalleeExtension
|
|
pNonStandardData, // non-standard data
|
|
&SourceEndpointType,
|
|
NULL, // pszCalledPartyNumber
|
|
pConnectAddr,
|
|
pDestinationAddr,
|
|
&SourceAddr, // source address
|
|
bCallerIsMC,
|
|
&pCall->ConferenceID, // conference ID
|
|
wGoal,
|
|
wCallType,
|
|
hCall, // user token
|
|
(Q931_CALLBACK)Q931Callback, // callback
|
|
pCall->dwBandwidth,
|
|
#ifdef GATEKEEPER
|
|
pCall->GkiCall.usCRV); // CRV
|
|
#else
|
|
0); // CRV
|
|
#endif GATEKEEPER
|
|
FreeAddr(pConnectAddr);
|
|
FreeAddr(pDestinationAddr);
|
|
Q931FreeAliasNames(pCallerAliasNames);
|
|
FreeVendorInfo(pVendorInfo);
|
|
FreeDisplay(pszDisplay);
|
|
Q931FreeAliasNames(pCalleeAliasNames);
|
|
Q931FreeAliasNames(pCalleeExtraAliasNames);
|
|
Q931FreeAliasItem(pCalleeExtension);
|
|
FreeNonStandardData(pNonStandardData);
|
|
if (status != CS_OK)
|
|
return status;
|
|
|
|
pCall->hQ931Call = hQ931Call;
|
|
return CC_OK;
|
|
}
|
|
|
|
|
|
|
|
HRESULT SendTermCaps( PCALL pCall,
|
|
PCONFERENCE pConference)
|
|
{
|
|
HRESULT status;
|
|
WORD i;
|
|
H245_TOTCAPDESC_T *pTermCapDescriptor;
|
|
PCC_TERMCAP pH2250MuxCapability;
|
|
PCC_TERMCAPLIST pTermCapList;
|
|
PCC_TERMCAPDESCRIPTORS pTermCapDescriptors;
|
|
|
|
ASSERT(pCall != NULL);
|
|
ASSERT(pConference != NULL);
|
|
|
|
if ((pConference->ConferenceMode == MULTIPOINT_MODE) &&
|
|
(pConference->tsMultipointController == TS_TRUE)) {
|
|
pH2250MuxCapability = pConference->pConferenceH245H2250MuxCapability;
|
|
pTermCapList = pConference->pConferenceTermCapList;
|
|
pTermCapDescriptors = pConference->pConferenceTermCapDescriptors;
|
|
} else {
|
|
pH2250MuxCapability = pConference->pLocalH245H2250MuxCapability;
|
|
pTermCapList = pConference->pLocalH245TermCapList;
|
|
pTermCapDescriptors = pConference->pLocalH245TermCapDescriptors;
|
|
}
|
|
|
|
ASSERT(pH2250MuxCapability != NULL);
|
|
ASSERT(pTermCapList != NULL);
|
|
ASSERT(pTermCapDescriptors != NULL);
|
|
|
|
// First send out the H.225.0 capability
|
|
status = H245SetLocalCap(pCall->H245Instance,
|
|
pH2250MuxCapability,
|
|
&pH2250MuxCapability->CapId);
|
|
ASSERT(pH2250MuxCapability->CapId == 0);
|
|
if (status != H245_ERROR_OK)
|
|
return status;
|
|
|
|
// Now send out the terminal capabilities
|
|
for (i = 0; i < pTermCapList->wLength; i++) {
|
|
status = H245SetLocalCap(pCall->H245Instance,
|
|
pTermCapList->pTermCapArray[i],
|
|
&pTermCapList->pTermCapArray[i]->CapId);
|
|
if (status != H245_ERROR_OK)
|
|
return status;
|
|
}
|
|
|
|
// Finally send out the capability descriptors
|
|
for (i = 0; i < pTermCapDescriptors->wLength; i++) {
|
|
pTermCapDescriptor = pTermCapDescriptors->pTermCapDescriptorArray[i];
|
|
status = H245SetCapDescriptor(pCall->H245Instance,
|
|
&pTermCapDescriptor->CapDesc,
|
|
&pTermCapDescriptor->CapDescId);
|
|
if (status != H245_ERROR_OK)
|
|
return status;
|
|
}
|
|
|
|
status = H245SendTermCaps(pCall->H245Instance,
|
|
pCall->H245Instance); // returned as dwTransId in the callback
|
|
return status;
|
|
}
|
|
|
|
|
|
|
|
HRESULT SessionTableToH245CommunicationTable(
|
|
PCC_SESSIONTABLE pSessionTable,
|
|
H245_COMM_MODE_ENTRY_T *pH245CommunicationTable[],
|
|
BYTE *pbCommunicationTableCount)
|
|
{
|
|
WORD i, j;
|
|
WORD wStringLength;
|
|
|
|
ASSERT(pH245CommunicationTable != NULL);
|
|
ASSERT(pbCommunicationTableCount != NULL);
|
|
|
|
if ((pSessionTable == NULL) || (pSessionTable->wLength == 0)) {
|
|
*pH245CommunicationTable = NULL;
|
|
*pbCommunicationTableCount = 0;
|
|
return CC_OK;
|
|
}
|
|
|
|
if (pSessionTable->SessionInfoArray == NULL) {
|
|
*pH245CommunicationTable = NULL;
|
|
*pbCommunicationTableCount = 0;
|
|
return CC_BAD_PARAM;
|
|
}
|
|
|
|
*pH245CommunicationTable = (H245_COMM_MODE_ENTRY_T *)Malloc(sizeof(H245_COMM_MODE_ENTRY_T) * pSessionTable->wLength);
|
|
if (*pH245CommunicationTable == NULL) {
|
|
*pbCommunicationTableCount = 0;
|
|
return CC_NO_MEMORY;
|
|
}
|
|
|
|
*pbCommunicationTableCount = (BYTE)pSessionTable->wLength;
|
|
|
|
for (i = 0; i < pSessionTable->wLength; i++) {
|
|
(*pH245CommunicationTable)[i].pNonStandard = NULL;
|
|
(*pH245CommunicationTable)[i].sessionID = pSessionTable->SessionInfoArray[i].bSessionID;
|
|
if (pSessionTable->SessionInfoArray[i].bAssociatedSessionID == 0)
|
|
(*pH245CommunicationTable)[i].associatedSessionIDPresent = FALSE;
|
|
else {
|
|
(*pH245CommunicationTable)[i].associatedSessionIDPresent = TRUE;
|
|
(*pH245CommunicationTable)[i].associatedSessionID = pSessionTable->SessionInfoArray[i].bAssociatedSessionID;
|
|
}
|
|
(*pH245CommunicationTable)[i].terminalLabelPresent = FALSE;
|
|
wStringLength = pSessionTable->SessionInfoArray[i].SessionDescription.wOctetStringLength;
|
|
if (wStringLength > 0) {
|
|
(*pH245CommunicationTable)[i].pSessionDescription = (unsigned short *)Malloc(sizeof(unsigned short) * wStringLength);
|
|
if ((*pH245CommunicationTable)[i].pSessionDescription == NULL) {
|
|
for (j = 0; j < i; j++)
|
|
Free((*pH245CommunicationTable)[j].pSessionDescription);
|
|
Free(*pH245CommunicationTable);
|
|
*pbCommunicationTableCount = 0;
|
|
return CC_NO_MEMORY;
|
|
}
|
|
memcpy((*pH245CommunicationTable)[i].pSessionDescription,
|
|
pSessionTable->SessionInfoArray[i].SessionDescription.pOctetString,
|
|
wStringLength);
|
|
} else
|
|
(*pH245CommunicationTable)[i].pSessionDescription = NULL;
|
|
(*pH245CommunicationTable)[i].wSessionDescriptionLength = wStringLength;
|
|
(*pH245CommunicationTable)[i].dataType = *pSessionTable->SessionInfoArray[i].pTermCap;
|
|
if (pSessionTable->SessionInfoArray[i].pRTPAddr == NULL)
|
|
(*pH245CommunicationTable)[i].mediaChannelPresent = FALSE;
|
|
else {
|
|
if (pSessionTable->SessionInfoArray[i].pRTPAddr->nAddrType != CC_IP_BINARY) {
|
|
for (j = 0; j <= i; j++)
|
|
if ((*pH245CommunicationTable)[j].pSessionDescription != NULL)
|
|
Free((*pH245CommunicationTable)[j].pSessionDescription);
|
|
Free(*pH245CommunicationTable);
|
|
*pbCommunicationTableCount = 0;
|
|
return CC_BAD_PARAM;
|
|
}
|
|
if (pSessionTable->SessionInfoArray[i].pRTPAddr->bMulticast)
|
|
(*pH245CommunicationTable)[i].mediaChannel.type = H245_IP_MULTICAST;
|
|
else
|
|
(*pH245CommunicationTable)[i].mediaChannel.type = H245_IP_UNICAST;
|
|
(*pH245CommunicationTable)[i].mediaChannel.u.ip.tsapIdentifier =
|
|
pSessionTable->SessionInfoArray[i].pRTPAddr->Addr.IP_Binary.wPort;
|
|
HostToH245IPNetwork((*pH245CommunicationTable)[i].mediaChannel.u.ip.network,
|
|
pSessionTable->SessionInfoArray[i].pRTPAddr->Addr.IP_Binary.dwAddr);
|
|
(*pH245CommunicationTable)[i].mediaChannelPresent = TRUE;
|
|
}
|
|
if (pSessionTable->SessionInfoArray[i].pRTCPAddr == NULL)
|
|
(*pH245CommunicationTable)[i].mediaControlChannelPresent = FALSE;
|
|
else {
|
|
if (pSessionTable->SessionInfoArray[i].pRTCPAddr->nAddrType != CC_IP_BINARY) {
|
|
for (j = 0; j <= i; j++)
|
|
if ((*pH245CommunicationTable)[j].pSessionDescription != NULL)
|
|
Free((*pH245CommunicationTable)[j].pSessionDescription);
|
|
Free(*pH245CommunicationTable);
|
|
*pbCommunicationTableCount = 0;
|
|
return CC_BAD_PARAM;
|
|
}
|
|
if (pSessionTable->SessionInfoArray[i].pRTCPAddr->bMulticast)
|
|
(*pH245CommunicationTable)[i].mediaControlChannel.type = H245_IP_MULTICAST;
|
|
else
|
|
(*pH245CommunicationTable)[i].mediaControlChannel.type = H245_IP_UNICAST;
|
|
(*pH245CommunicationTable)[i].mediaControlChannel.u.ip.tsapIdentifier =
|
|
pSessionTable->SessionInfoArray[i].pRTCPAddr->Addr.IP_Binary.wPort;
|
|
HostToH245IPNetwork((*pH245CommunicationTable)[i].mediaControlChannel.u.ip.network,
|
|
pSessionTable->SessionInfoArray[i].pRTCPAddr->Addr.IP_Binary.dwAddr);
|
|
(*pH245CommunicationTable)[i].mediaControlChannelPresent = TRUE;
|
|
}
|
|
(*pH245CommunicationTable)[i].mediaGuaranteed = FALSE;
|
|
(*pH245CommunicationTable)[i].mediaGuaranteedPresent = TRUE;
|
|
(*pH245CommunicationTable)[i].mediaControlGuaranteed = FALSE;
|
|
(*pH245CommunicationTable)[i].mediaControlGuaranteedPresent = TRUE;
|
|
}
|
|
|
|
return CC_OK;
|
|
}
|
|
|
|
|
|
|
|
HRESULT H245CommunicationTableToSessionTable(
|
|
H245_COMM_MODE_ENTRY_T H245CommunicationTable[],
|
|
BYTE bCommunicationTableCount,
|
|
PCC_SESSIONTABLE *ppSessionTable)
|
|
{
|
|
WORD i, j;
|
|
HRESULT status;
|
|
|
|
ASSERT(ppSessionTable != NULL);
|
|
|
|
if (H245CommunicationTable == NULL)
|
|
if (bCommunicationTableCount == 0) {
|
|
*ppSessionTable = NULL;
|
|
return CC_OK;
|
|
} else
|
|
return CC_BAD_PARAM;
|
|
else
|
|
if (bCommunicationTableCount == 0)
|
|
return CC_BAD_PARAM;
|
|
|
|
*ppSessionTable = (PCC_SESSIONTABLE)Malloc(sizeof(CC_SESSIONTABLE));
|
|
if (*ppSessionTable == NULL)
|
|
return CC_NO_MEMORY;
|
|
|
|
(*ppSessionTable)->wLength = bCommunicationTableCount;
|
|
|
|
(*ppSessionTable)->SessionInfoArray = (PCC_SESSIONINFO)Malloc(sizeof(CC_SESSIONINFO) * bCommunicationTableCount);
|
|
if ((*ppSessionTable)->SessionInfoArray == NULL) {
|
|
Free(*ppSessionTable);
|
|
*ppSessionTable = NULL;
|
|
return CC_NO_MEMORY;
|
|
}
|
|
|
|
for (i = 0; i < bCommunicationTableCount; i++) {
|
|
(*ppSessionTable)->SessionInfoArray[i].bSessionID = H245CommunicationTable[i].sessionID;
|
|
if (H245CommunicationTable[i].associatedSessionIDPresent)
|
|
(*ppSessionTable)->SessionInfoArray[i].bAssociatedSessionID =
|
|
H245CommunicationTable[i].associatedSessionID;
|
|
else
|
|
(*ppSessionTable)->SessionInfoArray[i].bAssociatedSessionID = 0;
|
|
(*ppSessionTable)->SessionInfoArray[i].SessionDescription.wOctetStringLength =
|
|
H245CommunicationTable[i].wSessionDescriptionLength;
|
|
if ((*ppSessionTable)->SessionInfoArray[i].SessionDescription.wOctetStringLength == 0)
|
|
(*ppSessionTable)->SessionInfoArray[i].SessionDescription.pOctetString = NULL;
|
|
else {
|
|
(*ppSessionTable)->SessionInfoArray[i].SessionDescription.pOctetString =
|
|
(BYTE *)Malloc(H245CommunicationTable[i].wSessionDescriptionLength);
|
|
if ((*ppSessionTable)->SessionInfoArray[i].SessionDescription.pOctetString == NULL) {
|
|
for (j = 0; j < i; j++) {
|
|
H245FreeCap((*ppSessionTable)->SessionInfoArray[j].pTermCap);
|
|
if ((*ppSessionTable)->SessionInfoArray[j].pRTPAddr != NULL)
|
|
Free((*ppSessionTable)->SessionInfoArray[j].pRTPAddr);
|
|
if ((*ppSessionTable)->SessionInfoArray[j].pRTCPAddr != NULL)
|
|
Free((*ppSessionTable)->SessionInfoArray[j].pRTCPAddr);
|
|
}
|
|
Free((*ppSessionTable)->SessionInfoArray);
|
|
Free(*ppSessionTable);
|
|
return CC_NO_MEMORY;
|
|
}
|
|
memcpy((*ppSessionTable)->SessionInfoArray[i].SessionDescription.pOctetString,
|
|
H245CommunicationTable[i].pSessionDescription,
|
|
H245CommunicationTable[i].wSessionDescriptionLength);
|
|
}
|
|
status = H245CopyCap(&(*ppSessionTable)->SessionInfoArray[i].pTermCap,
|
|
&H245CommunicationTable[i].dataType);
|
|
if (status != H245_ERROR_OK) {
|
|
if ((*ppSessionTable)->SessionInfoArray[i].SessionDescription.pOctetString != NULL)
|
|
Free((*ppSessionTable)->SessionInfoArray[i].SessionDescription.pOctetString);
|
|
for (j = 0; j < i; j++) {
|
|
H245FreeCap((*ppSessionTable)->SessionInfoArray[j].pTermCap);
|
|
if ((*ppSessionTable)->SessionInfoArray[j].pRTPAddr != NULL)
|
|
Free((*ppSessionTable)->SessionInfoArray[j].pRTPAddr);
|
|
if ((*ppSessionTable)->SessionInfoArray[j].pRTCPAddr != NULL)
|
|
Free((*ppSessionTable)->SessionInfoArray[j].pRTCPAddr);
|
|
}
|
|
Free((*ppSessionTable)->SessionInfoArray);
|
|
Free(*ppSessionTable);
|
|
return status;
|
|
}
|
|
if ((H245CommunicationTable[i].mediaChannelPresent) &&
|
|
((H245CommunicationTable[i].mediaChannel.type == H245_IP_MULTICAST) ||
|
|
(H245CommunicationTable[i].mediaChannel.type == H245_IP_UNICAST))) {
|
|
(*ppSessionTable)->SessionInfoArray[i].pRTPAddr = (PCC_ADDR)Malloc(sizeof(CC_ADDR));
|
|
if ((*ppSessionTable)->SessionInfoArray[i].pRTPAddr == NULL) {
|
|
if ((*ppSessionTable)->SessionInfoArray[i].SessionDescription.pOctetString != NULL)
|
|
Free((*ppSessionTable)->SessionInfoArray[i].SessionDescription.pOctetString);
|
|
H245FreeCap((*ppSessionTable)->SessionInfoArray[i].pTermCap);
|
|
for (j = 0; j < i; j++) {
|
|
H245FreeCap((*ppSessionTable)->SessionInfoArray[j].pTermCap);
|
|
if ((*ppSessionTable)->SessionInfoArray[j].pRTPAddr != NULL)
|
|
Free((*ppSessionTable)->SessionInfoArray[j].pRTPAddr);
|
|
if ((*ppSessionTable)->SessionInfoArray[j].pRTCPAddr != NULL)
|
|
Free((*ppSessionTable)->SessionInfoArray[j].pRTCPAddr);
|
|
}
|
|
Free((*ppSessionTable)->SessionInfoArray);
|
|
Free(*ppSessionTable);
|
|
return CC_NO_MEMORY;
|
|
}
|
|
(*ppSessionTable)->SessionInfoArray[i].pRTPAddr->nAddrType = CC_IP_BINARY;
|
|
if (H245CommunicationTable[i].mediaChannel.type == H245_IP_MULTICAST)
|
|
(*ppSessionTable)->SessionInfoArray[i].pRTPAddr->bMulticast = TRUE;
|
|
else
|
|
(*ppSessionTable)->SessionInfoArray[i].pRTPAddr->bMulticast = FALSE;
|
|
(*ppSessionTable)->SessionInfoArray[i].pRTPAddr->Addr.IP_Binary.wPort =
|
|
H245CommunicationTable[i].mediaChannel.u.ip.tsapIdentifier;
|
|
H245IPNetworkToHost(&(*ppSessionTable)->SessionInfoArray[i].pRTPAddr->Addr.IP_Binary.dwAddr,
|
|
H245CommunicationTable[i].mediaChannel.u.ip.network);
|
|
} else
|
|
(*ppSessionTable)->SessionInfoArray[i].pRTPAddr = NULL;
|
|
if ((H245CommunicationTable[i].mediaControlChannelPresent) &&
|
|
((H245CommunicationTable[i].mediaControlChannel.type == H245_IP_MULTICAST) ||
|
|
(H245CommunicationTable[i].mediaControlChannel.type == H245_IP_UNICAST))) {
|
|
(*ppSessionTable)->SessionInfoArray[i].pRTCPAddr = (PCC_ADDR)Malloc(sizeof(CC_ADDR));
|
|
if ((*ppSessionTable)->SessionInfoArray[i].pRTCPAddr == NULL) {
|
|
if ((*ppSessionTable)->SessionInfoArray[i].SessionDescription.pOctetString != NULL)
|
|
Free((*ppSessionTable)->SessionInfoArray[i].SessionDescription.pOctetString);
|
|
H245FreeCap((*ppSessionTable)->SessionInfoArray[i].pTermCap);
|
|
if ((*ppSessionTable)->SessionInfoArray[i].pRTPAddr != NULL)
|
|
Free((*ppSessionTable)->SessionInfoArray[i].pRTPAddr);
|
|
for (j = 0; j < i; j++) {
|
|
H245FreeCap((*ppSessionTable)->SessionInfoArray[j].pTermCap);
|
|
if ((*ppSessionTable)->SessionInfoArray[j].pRTPAddr != NULL)
|
|
Free((*ppSessionTable)->SessionInfoArray[j].pRTPAddr);
|
|
if ((*ppSessionTable)->SessionInfoArray[j].pRTCPAddr != NULL)
|
|
Free((*ppSessionTable)->SessionInfoArray[j].pRTCPAddr);
|
|
}
|
|
Free((*ppSessionTable)->SessionInfoArray);
|
|
Free(*ppSessionTable);
|
|
return CC_NO_MEMORY;
|
|
}
|
|
(*ppSessionTable)->SessionInfoArray[i].pRTCPAddr->nAddrType = CC_IP_BINARY;
|
|
if (H245CommunicationTable[i].mediaChannel.type == H245_IP_MULTICAST)
|
|
(*ppSessionTable)->SessionInfoArray[i].pRTCPAddr->bMulticast = TRUE;
|
|
else
|
|
(*ppSessionTable)->SessionInfoArray[i].pRTCPAddr->bMulticast = FALSE;
|
|
(*ppSessionTable)->SessionInfoArray[i].pRTCPAddr->Addr.IP_Binary.wPort =
|
|
H245CommunicationTable[i].mediaControlChannel.u.ip.tsapIdentifier;
|
|
H245IPNetworkToHost(&(*ppSessionTable)->SessionInfoArray[i].pRTCPAddr->Addr.IP_Binary.dwAddr,
|
|
H245CommunicationTable[i].mediaControlChannel.u.ip.network);
|
|
} else
|
|
(*ppSessionTable)->SessionInfoArray[i].pRTCPAddr = NULL;
|
|
}
|
|
return CC_OK;
|
|
}
|
|
|
|
|
|
|
|
HRESULT FreeH245CommunicationTable( H245_COMM_MODE_ENTRY_T H245CommunicationTable[],
|
|
BYTE bCommunicationTableCount)
|
|
{
|
|
WORD i;
|
|
|
|
if (H245CommunicationTable == NULL)
|
|
if (bCommunicationTableCount == 0)
|
|
return CC_OK;
|
|
else
|
|
return CC_BAD_PARAM;
|
|
else
|
|
if (bCommunicationTableCount == 0)
|
|
return CC_BAD_PARAM;
|
|
|
|
for (i = 0; i < bCommunicationTableCount; i++)
|
|
if (H245CommunicationTable[i].pSessionDescription != NULL)
|
|
Free(H245CommunicationTable[i].pSessionDescription);
|
|
Free(H245CommunicationTable);
|
|
return CC_OK;
|
|
}
|
|
|
|
|
|
|
|
HRESULT _PrepareTermCapLists( PCONFERENCE pConference,
|
|
WORD *pwListCount,
|
|
PCC_TERMCAPLIST **ppTermCapList,
|
|
PCC_TERMCAPDESCRIPTORS **ppTermCapDescriptorList,
|
|
PCALL *pCallList[])
|
|
{
|
|
WORD i;
|
|
WORD wNumCalls;
|
|
WORD wOffset;
|
|
PCC_HCALL CallList;
|
|
PCALL pCall;
|
|
|
|
ASSERT(pConference != NULL);
|
|
ASSERT(pwListCount != NULL);
|
|
ASSERT(ppTermCapList != NULL);
|
|
ASSERT(ppTermCapDescriptorList != NULL);
|
|
ASSERT(pCallList != NULL);
|
|
|
|
EnumerateCallsInConference(&wNumCalls, &CallList, pConference, ESTABLISHED_CALL);
|
|
|
|
if ((pConference->LocalEndpointAttached == DETACHED) && (wNumCalls > 0))
|
|
wOffset = 0;
|
|
else
|
|
// LocalEndpointAttached is either UNATTACHED or ATTACHED, or there are no calls
|
|
// in the conference; in the latter case, we need to have some term caps in
|
|
// order to form the conference term cap set (which cannot be empty)
|
|
wOffset = 1;
|
|
|
|
*pwListCount = (WORD)(wNumCalls + wOffset);
|
|
|
|
*ppTermCapList = (PCC_TERMCAPLIST *)Malloc(sizeof(PCC_TERMCAPLIST) * (*pwListCount));
|
|
if (*ppTermCapList == NULL) {
|
|
Free(CallList);
|
|
return CC_NO_MEMORY;
|
|
}
|
|
|
|
*ppTermCapDescriptorList = (PCC_TERMCAPDESCRIPTORS *)Malloc(sizeof(PCC_TERMCAPDESCRIPTORS) * (*pwListCount));
|
|
if (*ppTermCapDescriptorList == NULL) {
|
|
Free(CallList);
|
|
Free(*ppTermCapList);
|
|
return CC_NO_MEMORY;
|
|
}
|
|
|
|
*pCallList = (PCALL *)Malloc(sizeof(PCALL) * (*pwListCount));
|
|
if (*pCallList == NULL) {
|
|
Free(CallList);
|
|
Free(*ppTermCapList);
|
|
Free(*ppTermCapDescriptorList);
|
|
return CC_NO_MEMORY;
|
|
}
|
|
|
|
// Fill in pTermCapList and pTermCapDescriptorList
|
|
if (wOffset == 1) {
|
|
// The local endpoint is attached to the conference, so fill in the first
|
|
// slot in both lists with the local term cap and descriptor lists
|
|
(*ppTermCapList)[0] = pConference->pLocalH245TermCapList;
|
|
(*ppTermCapDescriptorList)[0] = pConference->pLocalH245TermCapDescriptors;
|
|
}
|
|
for (i = 0; i < wNumCalls; i++) {
|
|
if (LockCall(CallList[i], &pCall) == CC_OK) {
|
|
(*ppTermCapList)[i+wOffset] = pCall->pPeerH245TermCapList;
|
|
(*ppTermCapDescriptorList)[i+wOffset] = pCall->pPeerH245TermCapDescriptors;
|
|
(*pCallList)[i] = pCall;
|
|
} else {
|
|
(*ppTermCapList)[i+wOffset] = NULL;
|
|
(*ppTermCapDescriptorList)[i+wOffset] = NULL;
|
|
(*pCallList)[i] = NULL;
|
|
}
|
|
}
|
|
for (i = 0; i < wOffset; i++)
|
|
(*pCallList)[wNumCalls+i] = NULL;
|
|
Free(CallList);
|
|
return CC_OK;
|
|
}
|
|
|
|
|
|
|
|
HRESULT _FreeTermCapLists( WORD wListCount,
|
|
PCC_TERMCAPLIST *pTermCapList,
|
|
PCC_TERMCAPDESCRIPTORS *pTermCapDescriptorList,
|
|
PCALL pCallList[])
|
|
{
|
|
WORD i;
|
|
|
|
for (i = 0; i < wListCount; i++)
|
|
if (pCallList[i] != NULL)
|
|
UnlockCall(pCallList[i]);
|
|
if (pTermCapList != NULL)
|
|
Free(pTermCapList);
|
|
if (pTermCapDescriptorList != NULL)
|
|
Free(pTermCapDescriptorList);
|
|
Free(pCallList);
|
|
return CC_OK;
|
|
}
|
|
|
|
|
|
|
|
HRESULT CreateConferenceSessionTable(
|
|
PCONFERENCE pConference,
|
|
BOOL *pbSessionTableChanged)
|
|
{
|
|
HRESULT status;
|
|
PCALL *pCallList;
|
|
PCC_TERMCAPLIST *pTermCapList;
|
|
PCC_TERMCAPDESCRIPTORS *pTermCapDescriptorList;
|
|
WORD wListCount;
|
|
|
|
ASSERT(pConference != NULL);
|
|
|
|
if (pConference->bSessionTableInternallyConstructed == TRUE) {
|
|
status = FreeConferenceSessionTable(pConference);
|
|
if (status != CC_OK)
|
|
return status;
|
|
pConference->bSessionTableInternallyConstructed = FALSE;
|
|
}
|
|
|
|
status = _PrepareTermCapLists(pConference,
|
|
&wListCount,
|
|
&pTermCapList,
|
|
&pTermCapDescriptorList,
|
|
&pCallList);
|
|
if (status != CC_OK)
|
|
return status;
|
|
|
|
status = pConference->SessionTableConstructor(
|
|
pConference->hConference,
|
|
pConference->dwConferenceToken,
|
|
TRUE, // bCreate
|
|
pbSessionTableChanged,
|
|
wListCount,
|
|
pTermCapList,
|
|
pTermCapDescriptorList,
|
|
&pConference->pSessionTable);
|
|
|
|
_FreeTermCapLists(wListCount,
|
|
pTermCapList,
|
|
pTermCapDescriptorList,
|
|
pCallList);
|
|
return status;
|
|
}
|
|
|
|
|
|
|
|
HRESULT FreeConferenceSessionTable( PCONFERENCE pConference)
|
|
{
|
|
HRESULT status;
|
|
|
|
ASSERT(pConference != NULL);
|
|
|
|
if (pConference->bSessionTableInternallyConstructed)
|
|
status = DefaultSessionTableConstructor(
|
|
pConference->hConference,
|
|
pConference->dwConferenceToken,
|
|
FALSE, // bCreate
|
|
NULL, // pbSessionTableChanged
|
|
0, // wListCount
|
|
NULL, // pTermCapList[]
|
|
NULL, // pTermCapDescriptors[]
|
|
&pConference->pSessionTable);
|
|
else
|
|
status = pConference->SessionTableConstructor(
|
|
pConference->hConference,
|
|
pConference->dwConferenceToken,
|
|
FALSE, // bCreate
|
|
NULL, // pbSessionTableChanged
|
|
0, // wListCount
|
|
NULL, // pTermCapList[]
|
|
NULL, // pTermCapDescriptors[]
|
|
&pConference->pSessionTable);
|
|
pConference->pSessionTable = NULL;
|
|
return status;
|
|
}
|
|
|
|
|
|
|
|
HRESULT CreateConferenceTermCaps( PCONFERENCE pConference,
|
|
BOOL *pbTermCapsChanged)
|
|
{
|
|
HRESULT status;
|
|
WORD wListCount;
|
|
PCALL *pCallList;
|
|
PCC_TERMCAPLIST *pInTermCapList;
|
|
PCC_TERMCAPDESCRIPTORS *pInTermCapDescriptors;
|
|
|
|
ASSERT(pConference != NULL);
|
|
|
|
if (pConference->pConferenceH245H2250MuxCapability != NULL)
|
|
H245FreeCap(pConference->pConferenceH245H2250MuxCapability);
|
|
|
|
ASSERT(pConference->pLocalH245H2250MuxCapability != NULL);
|
|
status = H245CopyCap(&pConference->pConferenceH245H2250MuxCapability,
|
|
pConference->pLocalH245H2250MuxCapability);
|
|
if (status != H245_ERROR_OK)
|
|
return status;
|
|
|
|
status = _PrepareTermCapLists(pConference,
|
|
&wListCount,
|
|
&pInTermCapList,
|
|
&pInTermCapDescriptors,
|
|
&pCallList);
|
|
if (status != CC_OK)
|
|
return status;
|
|
|
|
status = UnregisterTermCapListFromH245(pConference,
|
|
pConference->pConferenceTermCapList);
|
|
if (status != CC_OK)
|
|
return status;
|
|
|
|
status = UnregisterTermCapDescriptorsFromH245(pConference,
|
|
pConference->pConferenceTermCapDescriptors);
|
|
if (status != CC_OK)
|
|
return status;
|
|
|
|
status = pConference->TermCapConstructor(
|
|
pConference->hConference,
|
|
pConference->dwConferenceToken,
|
|
TRUE, // bCreate
|
|
pbTermCapsChanged,
|
|
wListCount,
|
|
pInTermCapList,
|
|
pInTermCapDescriptors,
|
|
&pConference->pConferenceTermCapList,
|
|
&pConference->pConferenceTermCapDescriptors);
|
|
|
|
_FreeTermCapLists(wListCount,
|
|
pInTermCapList,
|
|
pInTermCapDescriptors,
|
|
pCallList);
|
|
return status;
|
|
}
|
|
|
|
|
|
|
|
HRESULT FreeConferenceTermCaps( PCONFERENCE pConference)
|
|
{
|
|
HRESULT status;
|
|
|
|
ASSERT(pConference != NULL);
|
|
|
|
status = pConference->TermCapConstructor(
|
|
pConference->hConference,
|
|
pConference->dwConferenceToken,
|
|
FALSE, // bCreate
|
|
NULL, // pbTermCapsChanged
|
|
0, // wListCount
|
|
NULL, // pInTermCapList[]
|
|
NULL, // pInTermCapDescriptors[]
|
|
&pConference->pConferenceTermCapList,
|
|
&pConference->pConferenceTermCapDescriptors);
|
|
pConference->pConferenceTermCapList = NULL;
|
|
pConference->pConferenceTermCapDescriptors = NULL;
|
|
return status;
|
|
}
|
|
|
|
|
|
|
|
HRESULT FindEnqueuedRequest( PCALL_QUEUE pQueueHead,
|
|
CC_HCALL hEnqueuedCall)
|
|
{
|
|
PCALL_QUEUE pQueueItem;
|
|
|
|
ASSERT(hEnqueuedCall != CC_INVALID_HANDLE);
|
|
|
|
pQueueItem = pQueueHead;
|
|
|
|
while (pQueueItem != NULL) {
|
|
if (pQueueItem->hCall == hEnqueuedCall)
|
|
break;
|
|
pQueueItem = pQueueItem->pNext;
|
|
}
|
|
if (pQueueItem == NULL)
|
|
return CC_BAD_PARAM;
|
|
else
|
|
return CC_OK;
|
|
}
|
|
|
|
|
|
|
|
HRESULT EnqueueRequest( PCALL_QUEUE *ppQueueHead,
|
|
CC_HCALL hEnqueuedCall)
|
|
{
|
|
PCALL_QUEUE pQueueItem;
|
|
|
|
ASSERT(ppQueueHead != NULL);
|
|
ASSERT(hEnqueuedCall != CC_INVALID_HANDLE);
|
|
|
|
// Make sure we're not enqueuing a duplicate request
|
|
pQueueItem = *ppQueueHead;
|
|
while (pQueueItem != NULL) {
|
|
if (pQueueItem->hCall == hEnqueuedCall)
|
|
return CC_OK;
|
|
pQueueItem = pQueueItem->pNext;
|
|
}
|
|
|
|
pQueueItem = (PCALL_QUEUE)Malloc(sizeof(CALL_QUEUE));
|
|
if (pQueueItem == NULL)
|
|
return CC_NO_MEMORY;
|
|
pQueueItem->hCall = hEnqueuedCall;
|
|
pQueueItem->pPrev = NULL;
|
|
pQueueItem->pNext = *ppQueueHead;
|
|
if (*ppQueueHead != NULL)
|
|
(*ppQueueHead)->pPrev = pQueueItem;
|
|
*ppQueueHead = pQueueItem;
|
|
return CC_OK;
|
|
}
|
|
|
|
|
|
|
|
HRESULT DequeueRequest( PCALL_QUEUE *ppQueueHead,
|
|
PCC_HCALL phEnqueuedCall)
|
|
|
|
{
|
|
PCALL_QUEUE pQueueItem;
|
|
|
|
ASSERT(ppQueueHead != NULL);
|
|
|
|
if (phEnqueuedCall != NULL)
|
|
*phEnqueuedCall = CC_INVALID_HANDLE;
|
|
|
|
if (*ppQueueHead == NULL)
|
|
return CC_BAD_PARAM;
|
|
|
|
pQueueItem = *ppQueueHead;
|
|
*ppQueueHead = (*ppQueueHead)->pNext;
|
|
if (*ppQueueHead != NULL)
|
|
(*ppQueueHead)->pPrev = NULL;
|
|
|
|
if (phEnqueuedCall != NULL)
|
|
*phEnqueuedCall = pQueueItem->hCall;
|
|
Free(pQueueItem);
|
|
return CC_OK;
|
|
}
|
|
|
|
|
|
|
|
HRESULT DequeueSpecificRequest( PCALL_QUEUE *ppQueueHead,
|
|
CC_HCALL hEnqueuedCall)
|
|
{
|
|
PCALL_QUEUE pQueueItem;
|
|
|
|
ASSERT(ppQueueHead != NULL);
|
|
ASSERT(hEnqueuedCall != CC_INVALID_HANDLE);
|
|
|
|
pQueueItem = *ppQueueHead;
|
|
while (pQueueItem != NULL)
|
|
if (pQueueItem->hCall == hEnqueuedCall)
|
|
break;
|
|
else
|
|
pQueueItem = pQueueItem->pNext;
|
|
|
|
if (pQueueItem == NULL)
|
|
return CC_BAD_PARAM;
|
|
|
|
if (pQueueItem->pNext != NULL)
|
|
pQueueItem->pNext->pPrev = pQueueItem->pPrev;
|
|
if (pQueueItem->pPrev == NULL)
|
|
*ppQueueHead = pQueueItem->pNext;
|
|
else
|
|
pQueueItem->pPrev->pNext = pQueueItem->pNext;
|
|
|
|
Free(pQueueItem);
|
|
return CC_OK;
|
|
}
|