2020-09-30 16:53:55 +02:00

1641 lines
52 KiB
C++
Raw Permalink Blame History

This file contains invisible Unicode characters

This file contains invisible Unicode characters that are indistinguishable to humans but may be processed differently by a computer. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/*
* File: capsctl.cpp
*
* capability control object implementations
*
*
* Revision History:
*
* 10/10/96 mikeg created
* 06/24/97 mikev - Added T.120 capability to serialized caps and simcaps (interim hack until a
* T120 resolver is implemented)
* - Retired ResolveEncodeFormat(Audio,Video) and implemented a data-independent
* resolution algorithm and exposed method ResolveFormats(). Added support
* routines ResolvePermutations(), TestSimultaneousCaps() and
* AreSimcaps().
*/
#include "precomp.h"
UINT g_AudioPacketDurationMs = AUDIO_PACKET_DURATION_LONG; // preferred packet duration
BOOL g_fRegAudioPacketDuration = FALSE; // AudioPacketDurationMs from registry
PCC_TERMCAPDESCRIPTORS CapsCtl::pAdvertisedSets=NULL;
DWORD CapsCtl::dwConSpeed = 0;
UINT CapsCtl::uStaticGlobalRefCount=0;
UINT CapsCtl::uAdvertizedSize=0;
extern HRESULT WINAPI CreateMediaCapability(REFGUID, LPIH323MediaCap *);
LPIH323MediaCap CapsCtl::FindHostForID(MEDIA_FORMAT_ID id)
{
if(pAudCaps && pAudCaps->IsHostForCapID(id))
{
return (pAudCaps);
}
else if (pVidCaps && pVidCaps->IsHostForCapID(id))
{
return (pVidCaps);
}
return NULL;
}
LPIH323MediaCap CapsCtl::FindHostForMediaType(PCC_TERMCAP pCapability)
{
if(pCapability->DataType == H245_DATA_AUDIO)
{
return (pAudCaps);
}
else if(pCapability->DataType == H245_DATA_VIDEO)
{
return (pVidCaps);
}
return NULL;
}
LPIH323MediaCap CapsCtl::FindHostForMediaGuid(LPGUID pMediaGuid)
{
if(MEDIA_TYPE_H323VIDEO == *pMediaGuid)
{
return (pVidCaps);
}
else if(MEDIA_TYPE_H323AUDIO == *pMediaGuid)
{
return (pAudCaps);
}
else
return NULL;
}
ULONG CapsCtl::AddRef()
{
uRef++;
return uRef;
}
ULONG CapsCtl::Release()
{
uRef--;
if(uRef == 0)
{
delete this;
return 0;
}
return uRef;
}
STDMETHODIMP CapsCtl::QueryInterface( REFIID iid, void ** ppvObject)
{
// this breaks the rules for the official COM QueryInterface because
// the interfaces that are queried for are not necessarily real COM
// interfaces. The reflexive property of QueryInterface would be broken in
// that case.
HRESULT hr = E_NOINTERFACE;
if(!ppvObject)
return hr;
*ppvObject = 0;
if(iid == IID_IDualPubCap)// satisfy symmetric property of QI
{
*ppvObject = (IDualPubCap *)this;
hr = hrSuccess;
AddRef();
}
else if(iid == IID_IAppAudioCap )
{
if(pAudCaps)
{
return pAudCaps->QueryInterface(iid, ppvObject);
}
}
else if(iid == IID_IAppVidCap )
{
if(pVidCaps)
{
return pVidCaps->QueryInterface(iid, ppvObject);
}
}
return hr;
}
CapsCtl::CapsCtl () :
uRef(1),
pVidCaps(NULL),
pAudCaps(NULL),
pACapsBuf(NULL),
pVCapsBuf(NULL),
dwNumInUse(0),
bAudioPublicize(TRUE),
bVideoPublicize(TRUE),
bT120Publicize(TRUE),
m_localT120cap(INVALID_MEDIA_FORMAT),
m_remoteT120cap(INVALID_MEDIA_FORMAT),
m_remoteT120bitrate(0),
m_pAudTermCaps(NULL),
m_pVidTermCaps(NULL),
pSetIDs(NULL),
pRemAdvSets(NULL)
{
uStaticGlobalRefCount++;
}
CapsCtl::~CapsCtl ()
{
if (pACapsBuf) {
MemFree (pACapsBuf);
}
if (pVCapsBuf) {
MemFree (pVCapsBuf);
}
if (pAudCaps) {
pAudCaps->Release();
}
if (pVidCaps) {
pVidCaps->Release();
}
uStaticGlobalRefCount--;
if (uStaticGlobalRefCount == 0) {
//Free up the sim. caps array
if (pAdvertisedSets) {
while (pAdvertisedSets->wLength) {
//wLength is Zero based
MemFree ((VOID *)pAdvertisedSets->pTermCapDescriptorArray[--pAdvertisedSets->wLength]);
}
MemFree ((VOID *)pAdvertisedSets->pTermCapDescriptorArray);
pAdvertisedSets->pTermCapDescriptorArray = NULL;
MemFree ((void *) pAdvertisedSets);
pAdvertisedSets=NULL;
dwNumInUse=0;
}
}
//And the remote array
if (pRemAdvSets) {
while (pRemAdvSets->wLength) {
MemFree ((VOID *)pRemAdvSets->pTermCapDescriptorArray[--pRemAdvSets->wLength]);
}
MemFree ((void *) pRemAdvSets->pTermCapDescriptorArray);
pRemAdvSets->pTermCapDescriptorArray = NULL;
MemFree ((void *) pRemAdvSets);
pRemAdvSets=NULL;
}
MemFree (pSetIDs);
pSetIDs=NULL;
}
BOOL CapsCtl::Init()
{
HRESULT hrLast;
int iBase = 1;
if (g_capFlags & CAPFLAGS_AV_STREAMS)
{
hrLast = ::CreateMediaCapability(MEDIA_TYPE_H323AUDIO, &pAudCaps);
if(!HR_SUCCEEDED(hrLast))
{
goto InitDone;
}
}
if (g_capFlags & CAPFLAGS_AV_STREAMS)
{
hrLast = ::CreateMediaCapability(MEDIA_TYPE_H323VIDEO, &pVidCaps);
if(!HR_SUCCEEDED(hrLast))
{
goto InitDone;
}
}
if (pAudCaps)
{
// Base the capability IDs beginning at 1 (zero is an invalid capability ID!)
pAudCaps->SetCapIDBase(iBase);
iBase += pAudCaps->GetNumCaps();
}
if (pVidCaps)
{
pVidCaps->SetCapIDBase(iBase);
iBase += pVidCaps->GetNumCaps();
}
InitDone:
m_localT120cap = iBase;
return TRUE;
}
HRESULT CapsCtl::ReInitialize()
{
HRESULT hr = hrSuccess;
int iBase = 1;
if (pAudCaps && !pAudCaps->ReInit())
{
hr = CAPS_E_SYSTEM_ERROR;
goto EXIT;
}
if (pVidCaps && !pVidCaps->ReInit())
{
hr = CAPS_E_SYSTEM_ERROR;
goto EXIT;
}
// Base the capability IDs beginning at 1 (zero is an invalid capability ID!)
if (pAudCaps)
{
pAudCaps->SetCapIDBase(iBase);
iBase += pAudCaps->GetNumCaps();
}
if (pVidCaps)
{
pVidCaps->SetCapIDBase(iBase);
iBase += pVidCaps->GetNumCaps();
}
m_localT120cap = iBase;
EXIT:
return hr;
}
const char szNMProdNum[] = "Microsoft\256 NetMeeting(TM)\0";
const char szNM20VerNum[] = "Version 2.0\0";
HRESULT CapsCtl::AddRemoteDecodeCaps(PCC_TERMCAPLIST pTermCapList,PCC_TERMCAPDESCRIPTORS pTermCapDescriptors, PCC_VENDORINFO pVendorInfo)
{
FX_ENTRY("CapsCtl::AddRemoteDecodeCaps");
HRESULT hr;
void *pData=NULL;
UINT uSize,x,y,z;
//WLength is # of capabilities, not structure length
WORD wNDesc;
LPIH323MediaCap pMediaCap;
if(!pTermCapList && !pTermCapDescriptors) // additional capability descriptors may be added
{ // at any time
return CAPS_E_INVALID_PARAM;
}
// Check for NM version 2.0
m_fNM20 = FALSE;
ASSERT(pVendorInfo);
if (pVendorInfo->bCountryCode == USA_H221_COUNTRY_CODE
&& pVendorInfo->wManufacturerCode == MICROSOFT_H_221_MFG_CODE
&& pVendorInfo->pProductNumber && pVendorInfo->pVersionNumber
&& pVendorInfo->pProductNumber->wOctetStringLength == sizeof(szNMProdNum)
&& pVendorInfo->pVersionNumber->wOctetStringLength == sizeof(szNM20VerNum)
&& memcmp(pVendorInfo->pProductNumber->pOctetString, szNMProdNum, sizeof(szNMProdNum)) == 0
&& memcmp(pVendorInfo->pVersionNumber->pOctetString, szNM20VerNum, sizeof(szNM20VerNum)) == 0
)
{
m_fNM20 = TRUE;
}
// cleanup old term caps if term caps are being added and old caps exist
if (pAudCaps)
pAudCaps->FlushRemoteCaps();
if (pVidCaps)
pVidCaps->FlushRemoteCaps();
m_remoteT120cap = INVALID_MEDIA_FORMAT; // note there is no T120 cap resolver and
// this CapsCtl holds exactly one local and remote T120 cap
// Copy pTermcapDescriptors to a local copy, (and free any old one)
if (pRemAdvSets) {
while (pRemAdvSets->wLength) {
//0 based
MemFree ((VOID *)pRemAdvSets->pTermCapDescriptorArray[--pRemAdvSets->wLength]);
}
MemFree ((VOID *)pRemAdvSets->pTermCapDescriptorArray);
pRemAdvSets->pTermCapDescriptorArray = NULL;
MemFree ((VOID *)pRemAdvSets);
pRemAdvSets=NULL;
}
//Ok, walk through the PCC_TERMCAPDESCRIPTORS list, first, allocate memory for the Master PCC_TERMCAPDESCRIPTORS
//structure, then each simcap, and the altcaps therin, then copy the data.
if (!(pRemAdvSets=(PCC_TERMCAPDESCRIPTORS) MemAlloc (sizeof (CC_TERMCAPDESCRIPTORS) ))){
return CAPS_E_SYSTEM_ERROR;
}
//How many Descriptors?
pRemAdvSets->wLength=pTermCapDescriptors->wLength;
if (!(pRemAdvSets->pTermCapDescriptorArray=((H245_TOTCAPDESC_T **)MemAlloc (sizeof (H245_TOTCAPDESC_T*)*pTermCapDescriptors->wLength))) ) {
return CAPS_E_SYSTEM_ERROR;
}
//Once per descriptor...
for (x=0;x < pTermCapDescriptors->wLength;x++) {
//Allocate memory for the descriptor entry
if (!(pRemAdvSets->pTermCapDescriptorArray[x]=(H245_TOTCAPDESC_T *)MemAlloc (sizeof (H245_TOTCAPDESC_T)))) {
return CAPS_E_SYSTEM_ERROR;
}
//BUGBUG for beta 2 Copy en masse.
memcpy (pRemAdvSets->pTermCapDescriptorArray[x],pTermCapDescriptors->pTermCapDescriptorArray[x],sizeof (H245_TOTCAPDESC_T));
/* post beta 2?
//Copy the capability ID
pRemAdvSets->pTermCapDescriptorArray[x].CapID=pTermCapDescriptors[x].CapID
//Walk the simcaps, then altcaps and copy entries */
}
for (wNDesc=0;wNDesc <pTermCapList->wLength;wNDesc++) {
pData=NULL;
pMediaCap = FindHostForMediaType(pTermCapList->pTermCapArray[wNDesc]);
if(!pMediaCap)
{
// special case: there is no T120 resolver. THIS IS A TEMPORARY
// SITUATION. We cannot track bitrate limits on multiple T120 capability
// instances because of this. As of now, we (NetMeeting) do not advertise
// more than one T.120 capability.
//This code will keep the last T.120 capability encountered.
if(((pTermCapList->pTermCapArray[wNDesc])->DataType == H245_DATA_DATA)
&& ((pTermCapList->pTermCapArray[wNDesc])->Cap.H245Dat_T120.application.choice
== DACy_applctn_t120_chosen)
&& ((pTermCapList->pTermCapArray[wNDesc])->Cap.H245Dat_T120.application.u.DACy_applctn_t120.choice
== separateLANStack_chosen))
{
// it's data data
m_remoteT120cap = (pTermCapList->pTermCapArray[wNDesc])->CapId;
m_remoteT120bitrate =
(pTermCapList->pTermCapArray[wNDesc])->Cap.H245Dat_T120.maxBitRate;
}
// continue;
// handled it in-line
}
else if(pMediaCap->IsCapabilityRecognized(pTermCapList->pTermCapArray[wNDesc]))
{
hr = pMediaCap->AddRemoteDecodeFormat(pTermCapList->pTermCapArray[wNDesc]);
#ifdef DEBUG
if(!HR_SUCCEEDED(hr))
{
ERRORMESSAGE(("%s:AddRemoteDecodeFormat returned 0x%08lx\r\n",_fx_, hr));
}
#endif // DEBUG
}
}
return (hrSuccess);
}
HRESULT CapsCtl::CreateCapList(PCC_TERMCAPLIST *ppCapBuf, PCC_TERMCAPDESCRIPTORS *ppCombinations)
{
PCC_TERMCAPLIST pTermCapList = NULL, pTermListAud=NULL, pTermListVid=NULL;
PCC_TERMCAPDESCRIPTORS pCombinations = NULL;
UINT uCount = 0, uSize = 0, uT120Size = 0;
HRESULT hr;
WORD wc;
UINT x=0,y=0,z=0,uNumAud=0,uNumVid=0;
H245_TOTCAPDESC_T *pTotCaps, **ppThisDescriptor;
PPCC_TERMCAP ppCCThisTermCap;
PCC_TERMCAP pCCT120Cap = NULL;
uCount = GetNumCaps(TRUE);
ASSERT((NULL == m_pAudTermCaps) && (NULL == m_pVidTermCaps));
// calc size of CC_TERMCAPLIST header + CC_TERMCAPDESCRIPTORS + array of PCC_TERMCAP
// allocate mem for the master CC_TERMCAPLIST, including the array of pointers to all CC_TERMCAPs
uSize = sizeof(CC_TERMCAPLIST)
+ sizeof (CC_TERMCAPDESCRIPTORS) + (uCount * sizeof(PCC_TERMCAP));
if((m_localT120cap != INVALID_MEDIA_FORMAT) && bT120Publicize)
{
uSize += sizeof(CC_TERMCAP);
}
pTermCapList = (PCC_TERMCAPLIST)MemAlloc(uSize);
if(pTermCapList == NULL)
{
hr = CAPS_E_NOMEM;
goto ERROR_EXIT;
}
// divide up the buffer, CC_TERMCAPLIST first, followed by array of PCC_TERMCAP.
// The array of PCC_TERMCAP follows fixed size CC_TERMCAPLIST structure and the fixed size
// CC_TERMCAP structure that holds the one T.120 cap.
if((m_localT120cap != INVALID_MEDIA_FORMAT) && bT120Publicize)
{
pCCT120Cap = (PCC_TERMCAP)(((BYTE *)pTermCapList) + sizeof(CC_TERMCAPLIST));
ppCCThisTermCap = (PPCC_TERMCAP) (((BYTE *)pTermCapList) + sizeof(CC_TERMCAPLIST) +
sizeof(CC_TERMCAP));
}
else
ppCCThisTermCap = (PPCC_TERMCAP) (((BYTE *)pTermCapList) + sizeof(CC_TERMCAPLIST));
// allocate mem for the simultaneous caps
// get size of cached advertised sets if it exists and more than one media
// type is enabled for publication
if(bAudioPublicize && bVideoPublicize && pAdvertisedSets)
{
// use size of cached buffer
uSize = uAdvertizedSize;
}
else if (pAdvertisedSets)
{
// This case needs to be fixed. If media types are disabled, the simultaneous capability
// descriptors in pAdvertisedSets should be rebuilt at that time. There should be no need to test
// if(bAudioPublicize && bVideoPublicize && pAdvertisedSets)
// calculate size of capability descriptors and simultaneous capability structures.
#pragma message ("Figure out the size this needs to be...")
#define NUMBER_TERMCAP_DESCRIPTORS 1
uSize = sizeof(H245_TOTCAPDESC_T) * NUMBER_TERMCAP_DESCRIPTORS+
sizeof (CC_TERMCAPDESCRIPTORS)+NUMBER_TERMCAP_DESCRIPTORS*
sizeof (H245_TOTCAPDESC_T *);
}
else
{
uSize = 0;
}
if (uSize)
{
pCombinations = (PCC_TERMCAPDESCRIPTORS)MemAlloc(uSize);
// skip the CC_TERMCAPDESCRIPTORS, which has a variable length array of (H245_TOTCAPDESC_T *) following it
// the total size of that glob is uSimCapsSize
// The actual array of [H245_TOTCAPDESC_T *] follows the CC_TERMCAPDESCRIPTORS structure
// anchor the pCombinations->pTermCapDescriptorArray to this point.
if(pCombinations == NULL)
{
hr = CAPS_E_NOMEM;
goto ERROR_EXIT;
}
ppThisDescriptor = pCombinations->pTermCapDescriptorArray
= (H245_TOTCAPDESC_T **)((BYTE *)pCombinations + sizeof(CC_TERMCAPDESCRIPTORS));
// the first H245_TOTCAPDESC_T follows the array of [H245_TOTCAPDESC_T *]
pTotCaps = (H245_TOTCAPDESC_T *)((BYTE *)ppThisDescriptor + pCombinations->wLength*sizeof(H245_TOTCAPDESC_T **));
if(pAudCaps && bAudioPublicize)
{
hr=pAudCaps->CreateCapList((LPVOID *)&pTermListAud);
if(!HR_SUCCEEDED(hr))
goto ERROR_EXIT;
ASSERT(pTermListAud != NULL);
}
if(pVidCaps && bVideoPublicize)
{
hr=pVidCaps->CreateCapList((LPVOID *)&pTermListVid);
if(!HR_SUCCEEDED(hr))
goto ERROR_EXIT;
ASSERT(pTermListVid != NULL);
}
}
else
{
pCombinations = NULL;
}
// fix pointers in the master caps list
// Now need to fixup the CC_TERMCAPLIST to refer to the individual capabilities
// Anchor the CC_TERMCAPLIST member pTermCapArray at the array of PCC_TERMCAP, and
// start partying on the array.
pTermCapList->wLength =0;
pTermCapList->pTermCapArray = ppCCThisTermCap;
if(pCCT120Cap)
{
*ppCCThisTermCap++ = pCCT120Cap;
// set T120 capability parameters
pCCT120Cap->DataType = H245_DATA_DATA;
pCCT120Cap->ClientType = H245_CLIENT_DAT_T120;
pCCT120Cap->Dir = H245_CAPDIR_LCLRXTX;
pCCT120Cap->Cap.H245Dat_T120.application.choice = DACy_applctn_t120_chosen;
pCCT120Cap->Cap.H245Dat_T120.application.u.DACy_applctn_t120.choice= separateLANStack_chosen;
pCCT120Cap->Cap.H245Dat_T120.maxBitRate = dwConSpeed;
pCCT120Cap->CapId = (H245_CAPID_T)m_localT120cap;
pTermCapList->wLength++;
}
if(pAudCaps && pTermListAud)
{
for(wc = 0; wc < pTermListAud->wLength; wc++)
{
// copy the array of "pointers to CC_TERMCAP"
*ppCCThisTermCap++ = pTermListAud->pTermCapArray[wc];
pTermCapList->wLength++;
}
}
if(pVidCaps && pTermListVid)
{
for(wc = 0; wc < pTermListVid->wLength; wc++)
{
// copy the array of "pointers to CC_TERMCAP"
*ppCCThisTermCap++ = pTermListVid->pTermCapArray[wc];
pTermCapList->wLength++;
}
}
// fixup the simultaneous capability descriptors
// Create a default set if necessary
//
if(bAudioPublicize && bVideoPublicize && pAdvertisedSets)
{
pCombinations->wLength = pAdvertisedSets->wLength;
// point pCombinations->pTermCapDescriptorArray past the header (CC_TERMCAPDESCRIPTORS)
pCombinations->pTermCapDescriptorArray
= (H245_TOTCAPDESC_T **)((BYTE *)pCombinations + sizeof(CC_TERMCAPDESCRIPTORS));
// the first H245_TOTCAPDESC_T follows the array of [H245_TOTCAPDESC_T *]
pTotCaps = (H245_TOTCAPDESC_T *)((BYTE *)pCombinations->pTermCapDescriptorArray +
pAdvertisedSets->wLength*sizeof(H245_TOTCAPDESC_T **));
for(x = 0; x < pAdvertisedSets->wLength; x++)
{
// write into the array of descriptor pointers. pointer[x] = this one
pCombinations->pTermCapDescriptorArray[x] = pTotCaps;
pTotCaps->CapDescId= pAdvertisedSets->pTermCapDescriptorArray[x]->CapDescId;
pTotCaps->CapDesc.Length=pAdvertisedSets->pTermCapDescriptorArray[x]->CapDesc.Length;
for(y = 0; y < pTotCaps->CapDesc.Length;y++)
{
//Copy the length field.
pTotCaps->CapDesc.SimCapArray[y].Length=
pAdvertisedSets->pTermCapDescriptorArray[x]->CapDesc.SimCapArray[y].Length;
for(z=0;
z < pAdvertisedSets->pTermCapDescriptorArray[x]->CapDesc.SimCapArray[y].Length;
z++)
{
pTotCaps->CapDesc.SimCapArray[y].AltCaps[z] =
pAdvertisedSets->pTermCapDescriptorArray[x]->CapDesc.SimCapArray[y].AltCaps[z];
}
}
pTotCaps++;
}
}
else if (pAdvertisedSets)
{
// descriptors in pAdvertisedSets should be rebuilt at that time. There should be no need to test
// if(bAudioPublicize && bVideoPublicize && pAdvertisedSets)
// HACK - put all audio or video caps in one AltCaps[], the T.120 cap in another AltCaps[]
// and put both of those in one single capability descriptor (H245_TOTCAPDESC_T)
// This hack will not extend past the assumption of one audio channel, one video channel, and
// one T.120 channel. If arbitrary media is supported, or multiple audio channels are supported,
// this code will be wrong
pCombinations->wLength=1;
// point pCombinations->pTermCapDescriptorArray past the header (CC_TERMCAPDESCRIPTORS)
pCombinations->pTermCapDescriptorArray
= (H245_TOTCAPDESC_T **)((BYTE *)pCombinations + sizeof(CC_TERMCAPDESCRIPTORS));
// the first H245_TOTCAPDESC_T follows the array of [H245_TOTCAPDESC_T *]
pTotCaps = (H245_TOTCAPDESC_T *)((BYTE *)pCombinations->pTermCapDescriptorArray +
pAdvertisedSets->wLength*sizeof(H245_TOTCAPDESC_T **));
pTotCaps->CapDescId=(H245_CAPDESCID_T)x;
pTotCaps->CapDesc.Length=0;
if(pTermListAud)
{
uNumAud = min(pTermListAud->wLength, H245_MAX_ALTCAPS);
pTotCaps->CapDesc.SimCapArray[x].Length=(unsigned short)uNumAud;
for(y = 0; y<uNumAud;y++)
{
pTotCaps->CapDesc.SimCapArray[x].AltCaps[y] = pTermListAud->pTermCapArray[y]->CapId;
}
x++;
pTotCaps->CapDesc.Length++;
}
if(pTermListVid && pTermListVid->wLength)
{
uNumVid = min(pTermListVid->wLength, H245_MAX_ALTCAPS);
pTotCaps->CapDesc.SimCapArray[x].Length=(unsigned short)uNumVid;
for(y = 0; y<uNumVid;y++)
{
pTotCaps->CapDesc.SimCapArray[x].AltCaps[y] = pTermListVid->pTermCapArray[y]->CapId;
}
x++;
pTotCaps->CapDesc.Length++;
}
// the T.120 cap
if((m_localT120cap != INVALID_MEDIA_FORMAT) && bT120Publicize)
{
pTotCaps->CapDesc.SimCapArray[x].Length=1;
pTotCaps->CapDesc.SimCapArray[x].AltCaps[0] = (H245_CAPID_T)m_localT120cap;
pTotCaps->CapDesc.Length++;
}
// write into the array of descriptor pointers. pointer[x] = this one
*ppThisDescriptor = pTotCaps;
}
m_pVidTermCaps = pTermListVid;
m_pAudTermCaps = pTermListAud;
*ppCapBuf = pTermCapList;
*ppCombinations = pCombinations;
return hrSuccess;
ERROR_EXIT:
m_pAudTermCaps = NULL;
m_pVidTermCaps = NULL;
if(pTermCapList)
MemFree(pTermCapList);
if(pCombinations)
MemFree(pCombinations);
if(pAudCaps && pTermListAud)
{
hr=pAudCaps->DeleteCapList(pTermListAud);
}
if(pVidCaps && pTermListVid)
{
hr=pVidCaps->DeleteCapList(pTermListVid);
}
return hr;
}
HRESULT CapsCtl::DeleteCapList(PCC_TERMCAPLIST pCapBuf, PCC_TERMCAPDESCRIPTORS pCombinations)
{
MemFree(pCapBuf);
MemFree(pCombinations);
if(m_pAudTermCaps && pAudCaps)
{
pAudCaps->DeleteCapList(m_pAudTermCaps);
}
if(m_pVidTermCaps)
{
pVidCaps->DeleteCapList(m_pVidTermCaps);
}
m_pAudTermCaps = NULL;
m_pVidTermCaps = NULL;
return hrSuccess;
}
HRESULT CapsCtl::GetEncodeParams(LPVOID pBufOut, UINT uBufSize,LPVOID pLocalParams, UINT uLocalSize,DWORD idRemote, DWORD idLocal)
{
LPIH323MediaCap pMediaCap = FindHostForID(idLocal);
if(!pMediaCap)
return CAPS_E_INVALID_PARAM;
// HACK
// Adjust audio packetization depending on call scenario
// unless there is an overriding registry setting
if (pMediaCap == pAudCaps)
{
VIDEO_FORMAT_ID vidLocal=INVALID_MEDIA_FORMAT, vidRemote=INVALID_MEDIA_FORMAT;
VIDEO_CHANNEL_PARAMETERS vidParams;
CC_TERMCAP vidCaps;
UINT audioPacketLength;
// modify the audio packetization parameters based on local bandwidth
// and presence of video
audioPacketLength = AUDIO_PACKET_DURATION_LONG;
// the registry setting overrides, if it is present
if (g_fRegAudioPacketDuration)
audioPacketLength = g_AudioPacketDurationMs;
else if (!m_fNM20) // dont try smaller packets for NM20 because it cant handle them
{
if (pVidCaps && pVidCaps->ResolveEncodeFormat(&vidLocal,&vidRemote) == S_OK
&& (pVidCaps->GetEncodeParams(&vidCaps,sizeof(vidCaps), &vidParams, sizeof(vidParams), vidRemote, vidLocal) == S_OK))
{
// we may potentially send video
if (vidParams.ns_params.maxBitRate*100 > BW_ISDN_BITS)
audioPacketLength = AUDIO_PACKET_DURATION_SHORT;
}
else
{
// no video
// since we dont know the actual connection bandwidth we use
// the local user setting.
// Note: if the remote is on a slow-speed net and the local is on a LAN
// we may end up with an inappropriate setting.
if (dwConSpeed > BW_288KBS_BITS)
audioPacketLength = AUDIO_PACKET_DURATION_SHORT;
else if (dwConSpeed > BW_144KBS_BITS)
audioPacketLength = AUDIO_PACKET_DURATION_MEDIUM;
}
}
// Setting the AudioPacketDurationMs affects the subsequent GetEncodeParams call
pMediaCap->SetAudioPacketDuration(audioPacketLength);
}
return pMediaCap->GetEncodeParams (pBufOut,uBufSize, pLocalParams,
uLocalSize,idRemote,idLocal);
}
HRESULT CapsCtl::GetPublicDecodeParams(LPVOID pBufOut, UINT uBufSize, VIDEO_FORMAT_ID id)
{
LPIH323MediaCap pMediaCap = FindHostForID(id);
if(!pMediaCap)
return CAPS_E_INVALID_PARAM;
return pMediaCap->GetPublicDecodeParams (pBufOut,uBufSize,id);
}
HRESULT CapsCtl::GetDecodeParams(PCC_RX_CHANNEL_REQUEST_CALLBACK_PARAMS pChannelParams,DWORD * pFormatID, LPVOID lpvBuf, UINT uBufSize)
{
LPIH323MediaCap pMediaCap = FindHostForMediaType(pChannelParams->pChannelCapability);
if(!pMediaCap)
return CAPS_E_INVALID_PARAM;
return pMediaCap->GetDecodeParams (pChannelParams,pFormatID,lpvBuf,uBufSize);
}
HRESULT CapsCtl::ResolveToLocalFormat(MEDIA_FORMAT_ID FormatIDLocal,
MEDIA_FORMAT_ID * pFormatIDRemote)
{
LPIH323MediaCap pMediaCap = FindHostForID(FormatIDLocal);
if(!pMediaCap)
return CAPS_E_INVALID_PARAM;
return pMediaCap->ResolveToLocalFormat (FormatIDLocal,pFormatIDRemote);
}
UINT CapsCtl::GetSimCapBufSize (BOOL bRxCaps)
{
UINT uSize;
// get size of cached advertised sets if it exists and more than one media
// type is enabled for publication
if(bAudioPublicize && bVideoPublicize && pAdvertisedSets)
{
// use size of cached buffer
uSize = uAdvertizedSize;
}
else
{
// calculate size of capability descriptors and simultaneous capability structures.
#pragma message ("Figure out the size this needs to be...")
#define NUMBER_TERMCAP_DESCRIPTORS 1
uSize = sizeof(H245_TOTCAPDESC_T) * NUMBER_TERMCAP_DESCRIPTORS+
sizeof (CC_TERMCAPDESCRIPTORS)+NUMBER_TERMCAP_DESCRIPTORS*
sizeof (H245_TOTCAPDESC_T *);
}
return uSize;
}
UINT CapsCtl::GetNumCaps(BOOL bRXCaps)
{
UINT u=0;
if(pAudCaps && bAudioPublicize)
{
u = pAudCaps->GetNumCaps(bRXCaps);
}
if(pVidCaps && bVideoPublicize)
{
u += pVidCaps->GetNumCaps(bRXCaps);
}
if(bT120Publicize)
u++;
return u;
}
UINT CapsCtl::GetLocalSendParamSize(MEDIA_FORMAT_ID dwID)
{
LPIH323MediaCap pMediaCap = FindHostForID(dwID);
if(!pMediaCap)
return 0;
return (pMediaCap->GetLocalSendParamSize(dwID));
}
UINT CapsCtl::GetLocalRecvParamSize(PCC_TERMCAP pCapability)
{
LPIH323MediaCap pMediaCap = FindHostForMediaType(pCapability);
if(!pMediaCap)
return 0;
return (pMediaCap->GetLocalRecvParamSize(pCapability));
}
STDMETHODIMP CapsCtl::GetEncodeFormatDetails(MEDIA_FORMAT_ID FormatID, VOID **ppFormat, UINT *puSize)
{
LPIH323MediaCap pMediaCap = FindHostForID(FormatID);
if(!pMediaCap)
{
*ppFormat = NULL;
*puSize = 0;
return E_INVALIDARG;
}
return pMediaCap->GetEncodeFormatDetails (FormatID, ppFormat, puSize);
}
STDMETHODIMP CapsCtl::GetDecodeFormatDetails(MEDIA_FORMAT_ID FormatID, VOID **ppFormat, UINT *puSize)
{
LPIH323MediaCap pMediaCap = FindHostForID(FormatID);
if(!pMediaCap)
{
*ppFormat = NULL;
*puSize = 0;
return E_INVALIDARG;
}
return pMediaCap->GetDecodeFormatDetails (FormatID, ppFormat, puSize);
}
//
// EnableMediaType controls whether or not capabilities for that media type
// are publicized. In a general implementation (next version?) w/ arbitrary
// number of media types, each of the media capability objects would keep
// track of their own state. This version of Capsctl tracks h323 audio and
// video only
//
HRESULT CapsCtl::EnableMediaType(BOOL bEnable, LPGUID pGuid)
{
if(!pGuid)
return CAPS_E_INVALID_PARAM;
if(*pGuid == MEDIA_TYPE_H323AUDIO)
{
bAudioPublicize = bEnable;
}
else if (*pGuid == MEDIA_TYPE_H323VIDEO)
{
bVideoPublicize = bEnable;
}
else
{
return CAPS_E_INVALID_PARAM;
}
return hrSuccess;
}
//
// Build the PCC_TERMCAPDESCRIPTORS list that we will advertise.
//
// puAudioFormatList/puVideoFormatList MUST BE sorted by preference!
//
//
HRESULT CapsCtl::AddCombinedEntry (MEDIA_FORMAT_ID *puAudioFormatList,UINT uAudNumEntries,MEDIA_FORMAT_ID *puVideoFormatList, UINT uVidNumEntries,DWORD *pIDOut)
{
static USHORT dwLastIDUsed;
DWORD x,y;
BOOL bAllEnabled=TRUE,bRecv,bSend;
unsigned short Length =0;
*pIDOut= (ULONG )CCO_E_SYSTEM_ERROR;
//Validate the Input
if ((!puAudioFormatList && uAudNumEntries > 0 ) || (!puVideoFormatList && uVidNumEntries > 0 ) || (uVidNumEntries == 0 && uAudNumEntries == 0 )) {
//What error code should we return here?
return CCO_E_SYSTEM_ERROR;
}
for (x=0;x<uAudNumEntries;x++)
{
ASSERT(pAudCaps);
pAudCaps->IsFormatEnabled (puAudioFormatList[x],&bRecv,&bSend);
bAllEnabled &= bRecv;
}
for (x=0;x<uVidNumEntries;x++) {
ASSERT(pVidCaps);
pVidCaps->IsFormatEnabled (puAudioFormatList[x],&bRecv,&bSend);
bAllEnabled &= bRecv;
}
if (!bAllEnabled) {
return CCO_E_INVALID_PARAM;
}
if (uAudNumEntries > H245_MAX_ALTCAPS || uVidNumEntries > H245_MAX_ALTCAPS) {
DEBUGMSG (1,("WARNING: Exceeding callcontrol limits!! \r\n"));
return CCO_E_INVALID_PARAM;
}
//If this is the first call, allocate space
if (!pAdvertisedSets){
pAdvertisedSets=(PCC_TERMCAPDESCRIPTORS)MemAlloc (sizeof (CC_TERMCAPDESCRIPTORS));
if (!pAdvertisedSets){
//Error code?
return CCO_E_SYSTEM_ERROR;
}
uAdvertizedSize = sizeof (CC_TERMCAPDESCRIPTORS);
//Allocate space of NUM_SIMCAP_SETS
pAdvertisedSets->pTermCapDescriptorArray=(H245_TOTCAPDESC_T **)
MemAlloc (sizeof (H245_TOTCAPDESC_T *)*NUM_SIMCAP_SETS);
if (!pAdvertisedSets->pTermCapDescriptorArray) {
//Error code?
return CCO_E_SYSTEM_ERROR;
}
//Update the indicies
uAdvertizedSize += sizeof (H245_TOTCAPDESC_T *)*NUM_SIMCAP_SETS;
dwNumInUse=NUM_SIMCAP_SETS;
pAdvertisedSets->wLength=0;
}
//Find an Index to use.
for (x=0;x<pAdvertisedSets->wLength;x++){
if (pAdvertisedSets->pTermCapDescriptorArray[x] == NULL){
break;
}
}
//Did we find space, or do we need a new one?
if (x >= dwNumInUse) {
//Increment the number in use
dwNumInUse++;
PVOID pTempTermCapDescriptorArray = NULL;
pTempTermCapDescriptorArray = MemReAlloc (pAdvertisedSets->pTermCapDescriptorArray,sizeof (H245_TOTCAPDESC_T *)*(dwNumInUse));
if(pTempTermCapDescriptorArray)
{
pAdvertisedSets->pTermCapDescriptorArray = (H245_TOTCAPDESC_T **)pTempTermCapDescriptorArray;
}
else
{
return CCO_E_SYSTEM_ERROR;
}
uAdvertizedSize += (sizeof (H245_TOTCAPDESC_T *)*(dwNumInUse))+sizeof (CC_TERMCAPDESCRIPTORS);
//Index is 0 based, point at the new entry
x=dwNumInUse-1;
}
//x is now the element we are using. Allocate space for a TermCapDescriptorArray
pAdvertisedSets->pTermCapDescriptorArray[x]=(H245_TOTCAPDESC_T *)MemAlloc (sizeof (H245_TOTCAPDESC_T));
if (!pAdvertisedSets->pTermCapDescriptorArray[x]){
return CCO_E_SYSTEM_ERROR;
}
uAdvertizedSize += sizeof (H245_TOTCAPDESC_T);
//Need to update the SetID. (start at 1)...
pAdvertisedSets->pTermCapDescriptorArray[x]->CapDescId=++dwLastIDUsed;
//Set the # of sets
if((m_localT120cap != INVALID_MEDIA_FORMAT) && bT120Publicize)
Length++;
if(uVidNumEntries)
Length++;
if(uAudNumEntries)
Length++;
pAdvertisedSets->pTermCapDescriptorArray[x]->CapDesc.Length= Length;
//Copy the Audio into SimCapArray[0], Video into SimCapArray[1] (if both)
if ((uVidNumEntries > 0 && uAudNumEntries > 0)) {
pAdvertisedSets->pTermCapDescriptorArray[x]->CapDesc.SimCapArray[0].Length=(unsigned short)uAudNumEntries;
pAdvertisedSets->pTermCapDescriptorArray[x]->CapDesc.SimCapArray[1].Length=(unsigned short)uVidNumEntries;
if((m_localT120cap != INVALID_MEDIA_FORMAT) && bT120Publicize)
pAdvertisedSets->pTermCapDescriptorArray[x]->CapDesc.SimCapArray[2].Length=1;
//Copy the format IDs
for (y=0;y<uAudNumEntries;y++) {
pAdvertisedSets->pTermCapDescriptorArray[x]->CapDesc.SimCapArray[0].AltCaps[y]=(USHORT)puAudioFormatList[y];
}
for (y=0;y<uVidNumEntries;y++) {
pAdvertisedSets->pTermCapDescriptorArray[x]->CapDesc.SimCapArray[1].AltCaps[y]=(USHORT)puVideoFormatList[y];
}
if((m_localT120cap != INVALID_MEDIA_FORMAT) && bT120Publicize)
pAdvertisedSets->pTermCapDescriptorArray[x]->CapDesc.SimCapArray[2].AltCaps[0]= (H245_CAPID_T)m_localT120cap;
} else {
if (uAudNumEntries > 0) {
pAdvertisedSets->pTermCapDescriptorArray[x]->CapDesc.SimCapArray[0].Length=(unsigned short)uAudNumEntries;
if((m_localT120cap != INVALID_MEDIA_FORMAT) && bT120Publicize)
pAdvertisedSets->pTermCapDescriptorArray[x]->CapDesc.SimCapArray[1].Length=1;
//Copy Audio only
for (y=0;y<uAudNumEntries;y++) {
pAdvertisedSets->pTermCapDescriptorArray[x]->CapDesc.SimCapArray[0].AltCaps[y]=(USHORT)puAudioFormatList[y];
}
if((m_localT120cap != INVALID_MEDIA_FORMAT) && bT120Publicize)
pAdvertisedSets->pTermCapDescriptorArray[x]->CapDesc.SimCapArray[1].AltCaps[0]= (H245_CAPID_T)m_localT120cap;
} else {
pAdvertisedSets->pTermCapDescriptorArray[x]->CapDesc.SimCapArray[0].Length=(unsigned short)uVidNumEntries;
if((m_localT120cap != INVALID_MEDIA_FORMAT) && bT120Publicize)
pAdvertisedSets->pTermCapDescriptorArray[x]->CapDesc.SimCapArray[1].Length=1;
//copy video entries
for (y=0;y<uVidNumEntries;y++) {
pAdvertisedSets->pTermCapDescriptorArray[x]->CapDesc.SimCapArray[0].AltCaps[y]=(USHORT)puVideoFormatList[y];
}
if((m_localT120cap != INVALID_MEDIA_FORMAT) && bT120Publicize)
pAdvertisedSets->pTermCapDescriptorArray[x]->CapDesc.SimCapArray[1].AltCaps[0]= (H245_CAPID_T)m_localT120cap;
}
}
//Need to update the wLength
pAdvertisedSets->wLength++;
*pIDOut=dwLastIDUsed;
return hrSuccess;
}
HRESULT CapsCtl::RemoveCombinedEntry (DWORD ID)
{
DWORD x;
if (!pAdvertisedSets) {
return CAPS_E_INVALID_PARAM;
}
for (x=0;x<dwNumInUse;x++) {
if (pAdvertisedSets->pTermCapDescriptorArray[x]) {
if (pAdvertisedSets->pTermCapDescriptorArray[x]->CapDescId == ID) {
//Found the one to remove
MemFree ((VOID *)pAdvertisedSets->pTermCapDescriptorArray[x]);
uAdvertizedSize -= sizeof (H245_TOTCAPDESC_T *);
if (x != (dwNumInUse -1)) {
//Not the last one, swap the two pointers
pAdvertisedSets->pTermCapDescriptorArray[x]=pAdvertisedSets->pTermCapDescriptorArray[dwNumInUse-1];
pAdvertisedSets->pTermCapDescriptorArray[dwNumInUse-1]=NULL;
}
//Decrement the number in use, and set the wLengthField
dwNumInUse--;
pAdvertisedSets->wLength--;
return hrSuccess;
}
}
}
//Shouldn't get here, unless it was not found.
return CAPS_E_NOCAPS;
}
// Given a sized list of capability IDs (pointer to array of H245_CAPID_T)
// and a sized list of alternate capabilities (AltCaps) within a single simultaneous
// capability set, (pointer to an array of pointers to H245_SIMCAP_T)
// Determine if the entire list of capability IDs can simultaneously coexist
// with respect to the given set of AltCaps.
BOOL CapsCtl::AreSimCaps(
H245_CAPID_T* pIDArray, UINT uIDArraySize,
H245_SIMCAP_T **ppAltCapArray,UINT uAltCapArraySize)
{
UINT i, u;
SHORT j;
BOOL bSim;
H245_SIMCAP_T *pAltCapEntry, *pFirstAltCapEntry;
// If there are fewer AltCaps than capabilities, doom is obvious. Don't bother searching.
if(uAltCapArraySize < uIDArraySize)
return FALSE;
// find an altcaps entry containing the first ID in the list
for (i=0;i<uAltCapArraySize;i++)
{
pAltCapEntry = *(ppAltCapArray+i);
// scan this altcaps entry for a matching ID
for(j=0;j<pAltCapEntry->Length;j++)
{
if(*pIDArray == pAltCapEntry->AltCaps[j])
{
// found a spot for this capability!
if(uIDArraySize ==1)
return TRUE; // Done! all the capabilities have been found to coexist
// Otherwise, look for the next capability in the *remaining* AltCaps
// *This* AltCaps contains the capability we were looking for
// So, we "used up" this AltCaps and can't select from it anymore
// Pack the array of H245_SIMCAP_T pointers in place so that
// "used" entries are at the beginning and "unused" at the end
// (a la shell sort swap pointers)
if(i != 0) // if not already the same, swap
{
pFirstAltCapEntry = *ppAltCapArray;
*ppAltCapArray = pAltCapEntry;
*(ppAltCapArray+i) = pFirstAltCapEntry;
}
// continue the quest using the remaining capabilities
// and the remaining AltCaps
bSim = AreSimCaps(pIDArray + 1, uIDArraySize - 1,
ppAltCapArray + 1, uAltCapArraySize - 1);
if(bSim)
{
return bSim;// success
}
else // why not? Either a fit does not exist (common), or the altcaps contain
// an odd pattern of multiple instances of some capability IDs, and another
// search order *might* fit. Do not blindly try all permutations of search
// order.
{
// If it failed simply because the recently grabbed slot in the altcaps
// (the one in *(ppAltCapArray+i)) could have been needed by subsequent
// capability IDs, give this one up and look for another instance.
// If not, we know for sure that the n! approach will not yield
// fruit and can be avoided.
for(u=1;(bSim == FALSE)&&(u<uAltCapArraySize);u++)
{
for(j=0;(bSim == FALSE)&&(j<pAltCapEntry->Length);j++)
{ // another capability needed the altcaps we grabbed ?
if(*(pIDArray+u) == pAltCapEntry->AltCaps[j])
{
bSim=TRUE;
break; // look no more here, bail to try again because a fit *might* exist
}
}
}
if(bSim) // going to continue searching - Swap pointers back if they were swapped above
{
if(i != 0) // if not the same, swap back
{
*ppAltCapArray = *(ppAltCapArray+i);
*(ppAltCapArray+i) = pAltCapEntry;
}
break; // next i
}
else // don't waste CPU - a fit does not exist
{
return bSim;
}
}
}
}
}
return FALSE;
}
// Given a sized list of capability IDs (pointer to array of H245_CAPID_T)
// and a list of simultaneous capabilities, try each simultaneous capability
// and determine if the entire list of capability IDs can simultaneously coexist.
BOOL CapsCtl::TestSimultaneousCaps(H245_CAPID_T* pIDArray, UINT uIDArraySize,
PCC_TERMCAPDESCRIPTORS pTermCaps)
{
int iSimSet, iAltSet;
BOOL bResolved = FALSE;
H245_SIMCAP_T * pAltCapArray[H245_MAX_SIMCAPS];
if (!pAdvertisedSets)
return(TRUE);
// try each independent local SimCaps set (each descriptor) until success
for (iSimSet=0; (bResolved == FALSE) && (iSimSet < pTermCaps->wLength);iSimSet++)
{
// EXTRA STEP:
// Build a sortable representation of the AltCaps set. This step will not be necessary if
// and when we change the native representation of a capability descriptor to a variable
// length list of pointers to AltCaps. In the meantime, we know that there are no more
// than H245_MAX_SIMCAPS AltCaps in this SimCaps. This is imposed by the 2 dimensional
// arrays of hardcoded size forced upon us by CALLCONT.DLL.
for (iAltSet=0;iAltSet < pTermCaps->pTermCapDescriptorArray[iSimSet]->CapDesc.Length;iAltSet++)
{
pAltCapArray[iAltSet] = &pTermCaps->pTermCapDescriptorArray[iSimSet]->CapDesc.SimCapArray[iAltSet];
}
// do the work
bResolved = AreSimCaps(pIDArray, uIDArraySize,
(H245_SIMCAP_T **)pAltCapArray,
MAKELONG(pTermCaps->pTermCapDescriptorArray[iSimSet]->CapDesc.Length, 0));
}
return bResolved;
}
// Function: CapsCtl::ResolvePermutations(PRES_CONTEXT pResContext, UINT uNumFixedColumns)
//
// This functions as both a combination generator and a validation mechanism for the
// combinations it generates.
//
// Given a pointer to a resolution context and the number of fixed (i.e. not permutable,
// if "permutable" is even a real word) columns, generate one combination at a time.
// Try each combination until a working combination is found or until all combinations
// have been tried.
//
// The resolution context structure contains a variable number of columns of variable
// length media format ID lists. Each column tracks its current index. When this
// function returns TRUE, the winning combination is indicated by the current column
// indices.
//
// The caller can control which combinations are tried first by arranging the columns
// in descending importance.
//
// Incremental searches can be performed without redundant comparisons by adding 1 format
// at a time to a column, arranging the column order so that the appended column is
// first, and "fixing" that one column at the newly added format. For example,
// some calling function could force evaluations on a round-robin column basis by
// calling this function inside a loop which does the following:
// 1 - adds one format at a time to the rightmost column and sets the current index
// of that column to the new entry
// 2 - rotates the column order so that the rightmost column is now the leftmost
// 3 - fixing the new leftmost column before calling this function again
// The result will be that only the permutations which contain the newly added format
// will be generated.
BOOL CapsCtl::ResolvePermutations(PRES_CONTEXT pResContext, UINT uNumFixedColumns)
{
RES_PAIR *pResolvedPair;
BOOL bResolved = FALSE;
UINT i, uColumns;
UINT uPairIndex;
// converge on one combination in the permutation
if(uNumFixedColumns != pResContext->uColumns)
{
RES_PAIR_LIST *pThisColumn;
// take the first non-fixed column, make that column fixed and
// iterate on it (loop through indices), and try each sub-permutation
// of remaining columns. (until success or all permutations tried)
pThisColumn = *(pResContext->ppPairLists+uNumFixedColumns);
for (i=0; (bResolved == FALSE) && (i<pThisColumn->uSize); i++)
{
pThisColumn->uCurrentIndex = i;
bResolved = ResolvePermutations(pResContext, uNumFixedColumns+1);
}
return bResolved;
}
else
{
// Bottomed out on the final column. Test the viability of this combination
// Build array of local IDs that contians the combination and test the
// combination against local simultaneous capabilities, then against
// remote simultaneous capabilities
// NOTE: be sure to skip empty columns (which represent unresolvable
// or unsupported/nonexistent media types or unsupported additional
// instances of media types)
for(i=0, uColumns=0;i<pResContext->uColumns;i++)
{
if(((*pResContext->ppPairLists)+i)->uSize)
{
// get index (row #) for this column
uPairIndex = ((*pResContext->ppPairLists)+i)->uCurrentIndex;
// get the row
pResolvedPair = ((*pResContext->ppPairLists)+i)->pResolvedPairs+uPairIndex;
// add the ID to the array
*(pResContext->pIDScratch+uColumns) = (H245_CAPID_T)pResolvedPair->idPublicLocal;
uColumns++;
}
// else empty column
}
// Determine if this combination can exist simultaneously
if(TestSimultaneousCaps(pResContext->pIDScratch,
uColumns, pResContext->pTermCapsLocal))
{
// now test remote
// build array of remote IDs and test those against remote
// simultaneous capabilities
for(i=0, uColumns=0;i<pResContext->uColumns;i++)
{
if(((*pResContext->ppPairLists)+i)->uSize)
{
// get index (row #) for this column
uPairIndex = ((*pResContext->ppPairLists)+i)->uCurrentIndex;
// get the row
pResolvedPair = ((*pResContext->ppPairLists)+i)->pResolvedPairs+uPairIndex;
// add the ID to the array
*(pResContext->pIDScratch+uColumns) =(H245_CAPID_T) pResolvedPair->idRemote;
uColumns++;
}
// else empty column
}
bResolved = TestSimultaneousCaps(pResContext->pIDScratch,
uColumns, pResContext->pTermCapsRemote);
}
return bResolved;
// if(bResolved == TRUE)
// The resolved combination of pairs is indicated by the current indices
// of **ppPairList;
}
}
//
// Given a counted list of desired instances of media, produce an output array of
// resolved media format IDs which correspond to the input media type IDs.
// This function returns success if at least one media instance is resolved.
// When an instance of media is unresolveable, the output corresponding to that
// instance contains the value INVALID_MEDIA_FORMAT for local and remote media
// format IDs.
//
// The input is treated as being in preferential order: permutations of the latter
// media type instance are varied first. If all permutations do not yield success,
// then one media type instance at a time is removed from the end.
//
HRESULT CapsCtl::ResolveFormats (LPGUID pMediaGuidArray, UINT uNumMedia,
PRES_PAIR pResOutput)
{
HRESULT hr = hrSuccess;
PRES_PAIR_LIST pResColumnArray = NULL;
PRES_PAIR_LIST *ppPairLists;
RES_PAIR *pResPair;
PRES_CONTEXT pResContext;
LPIH323MediaCap pMediaResolver;
UINT i;
UINT uMaxFormats = 0;
UINT uFixedColumns =0;
UINT uFailedMediaCount = 0;
BOOL bResolved = FALSE;
RES_PAIR UnresolvedPair = {INVALID_MEDIA_FORMAT, INVALID_MEDIA_FORMAT, INVALID_MEDIA_FORMAT};
// create a context structure for the resolution
pResContext = (PRES_CONTEXT)MemAlloc(sizeof(RES_CONTEXT)+ (uNumMedia*sizeof(H245_CAPID_T)));
if(!pResContext)
{
hr = CAPS_E_NOMEM;
goto ERROR_OUT;
}
// initialize resolution context
pResContext->uColumns = 0;
pResContext->pIDScratch = (H245_CAPID_T*)(pResContext+1);
pResContext->pTermCapsLocal = pAdvertisedSets;
pResContext->pTermCapsRemote = pRemAdvSets;
// allocate array of RES_PAIR_LIST (one per column/media type) and
// array of pointers to same
pResColumnArray = (PRES_PAIR_LIST)MemAlloc((sizeof(RES_PAIR_LIST) * uNumMedia)
+ (sizeof(PRES_PAIR_LIST) * uNumMedia));
if(!pResColumnArray)
{
hr = CAPS_E_NOMEM;
goto ERROR_OUT;
}
pResContext->ppPairLists = ppPairLists = (PRES_PAIR_LIST*)(pResColumnArray+uNumMedia);
// build columns of media capabilities
for(i=0;i<uNumMedia;i++)
{
// build array of pointers to RES_PAIR_LIST
*(ppPairLists+i) = pResColumnArray+i;
// initialize RES_PAIR_LIST members
(pResColumnArray+i)->pResolvedPairs = NULL;
(pResColumnArray+i)->uSize =0;
(pResColumnArray+i)->uCurrentIndex = 0;
// Get resolver for this media. Special case: there is no T120 resolver.
// T120 caps are handled right here in this object
if(MEDIA_TYPE_H323_T120 == *(pMediaGuidArray+i))
{
pMediaResolver = NULL;
if((m_localT120cap != INVALID_MEDIA_FORMAT) &&(m_remoteT120cap != INVALID_MEDIA_FORMAT) )
{
(pResColumnArray+i)->uSize =1;
uMaxFormats = 1; // only one T.120 cap
pResPair = (pResColumnArray+i)->pResolvedPairs =
(RES_PAIR *)MemAlloc(uMaxFormats * sizeof(RES_PAIR));
if(!pResPair)
{
hr = CAPS_E_NOMEM;
goto ERROR_OUT;
}
//
pResPair->idLocal = m_localT120cap;
pResPair->idRemote = m_remoteT120cap;
pResPair->idPublicLocal = pResPair->idLocal;
}
}
else
{
pMediaResolver = FindHostForMediaGuid(pMediaGuidArray+i);
}
pResContext->uColumns++;
(pResColumnArray+i)->pMediaResolver = pMediaResolver;
if(pMediaResolver)
{
uMaxFormats = pMediaResolver->GetNumCaps(FALSE); // get transmit format count
if(uMaxFormats)
{
pResPair = (pResColumnArray+i)->pResolvedPairs =
(RES_PAIR *)MemAlloc(uMaxFormats * sizeof(RES_PAIR));
if(!pResPair)
{
hr = CAPS_E_NOMEM;
goto ERROR_OUT;
}
// resolve the best choice for each media type (gotta start somewhere)
pResPair->idLocal = INVALID_MEDIA_FORMAT;
pResPair->idRemote = INVALID_MEDIA_FORMAT;
hr=pMediaResolver->ResolveEncodeFormat (&pResPair->idLocal,&pResPair->idRemote);
if(!HR_SUCCEEDED(hr))
{
if((hr == CAPS_W_NO_MORE_FORMATS)
|| (hr == CAPS_E_NOMATCH)
|| (hr == CAPS_E_NOCAPS))
{
// No resolved format for this media type. Remove this "column"
(pResColumnArray+i)->pResolvedPairs = NULL;
MemFree(pResPair);
(pResColumnArray+i)->uSize =0;
hr = hrSuccess;
}
else
{
goto ERROR_OUT;
}
}
else
{
// this column has one resolved format
pResPair->idPublicLocal = pMediaResolver->GetPublicID(pResPair->idLocal);
(pResColumnArray+i)->uSize =1;
}
}
// else // No formats exist for this media type. this "column" has zero size
}
}
// Special case test simultaneous caps for the most preferred combination:
uFixedColumns = pResContext->uColumns; // << make all columns fixed
bResolved = ResolvePermutations(pResContext, uFixedColumns);
// if the single most preferred combination can't be used, need to handle
// the general case and try permutations until a workable combination is found
while(!bResolved)
{
// make one column at a time permutable, starting with the least-critical media
// type. (e.g. it would be typical for the last column to be video because
// audio+data are more important. Then we try less and less
// preferable video formats before doing anything that would degrade the audio)
if(uFixedColumns > 0) // if not already at the end of the rope...
{
uFixedColumns--; // make another column permutable
}
else
{
// wow - tried all permutations and still no luck ......
// nuke the least important remaining media type (e.g. try it w/o video)
if(pResContext->uColumns <= 1) // already down to one media type?
{
hr = CAPS_E_NOMATCH;
goto ERROR_OUT;
}
// Remove the end column (representing the least important media type)
// and try it with the remaining columns
uFixedColumns = --pResContext->uColumns; // one less column
// set the formats of the nuked column to the unresolved state
(pResColumnArray+uFixedColumns)->uSize =0;
(pResColumnArray+uFixedColumns)->uCurrentIndex =0;
pResPair = (pResColumnArray+uFixedColumns)->pResolvedPairs;
if (NULL != pResPair)
{
pResPair->idLocal = INVALID_MEDIA_FORMAT;
pResPair->idRemote = INVALID_MEDIA_FORMAT;
pResPair->idPublicLocal = INVALID_MEDIA_FORMAT;
}
uFailedMediaCount++; // track the nuking of a column to avoid
// redundantly grabbing all the formats again
// ... would not be here if all permutations
// had not been tried!
// reset the combination indices
for(i=0;i<uFixedColumns;i++)
{
(pResColumnArray+i)->uCurrentIndex = 0;
}
}
// get the rest of the formats for the last known fixed column, make that column
// permutable, etc.
pMediaResolver = (pResColumnArray+uFixedColumns)->pMediaResolver;
if(!pMediaResolver || ((pResColumnArray+uFixedColumns)->uSize ==0))
{
continue; // this media type has no further possibility
}
if(uFailedMediaCount ==0) // If all of the possible resolved pairs
// have not yet been obtained, get them!
{
// get resolved pair IDs for every mutual format of this media type
// first: get pointer to array of pair IDs, then use ResolveEncodeFormat()
// to fill up the array
pResPair = (pResColumnArray+uFixedColumns)->pResolvedPairs;
// Get total # of formats less the one that was already obtained
uMaxFormats = pMediaResolver->GetNumCaps(FALSE) -1;
while(uMaxFormats--) // never exceed the # of remaining local formats...
{
RES_PAIR *pResPairNext;
// recall that ResolveEncodeFormat parameters are I/O - the input
// is the local ID of the last resolved mutual format. (remote id
// is ignored as input). Fixup the input.
pResPairNext = pResPair+1;
// start where the previous resolve stopped
pResPairNext->idLocal = pResPair->idLocal;
// not necessary, ignored ->>> pResPairNext->idRemote = pResPair->idRemote
pResPair = pResPairNext;
hr=pMediaResolver->ResolveEncodeFormat (&pResPair->idLocal,&pResPair->idRemote);
if((hr == CAPS_W_NO_MORE_FORMATS)
|| (hr == CAPS_E_NOMATCH))
// got all of the formats, but not an error
{ // this is likely when less than 100% of local formats have a remote match
hr = hrSuccess;
break;
}
if(!HR_SUCCEEDED(hr))
goto ERROR_OUT;
// get the public ID of the local format (it's *usually* the same, but not always)
pResPair->idPublicLocal = pMediaResolver->GetPublicID(pResPair->idLocal);
// this column has another format - count it!
(pResColumnArray+uFixedColumns)->uSize++;
}
}
// now try the new permutations
bResolved = ResolvePermutations(pResContext, uFixedColumns);
}
if(bResolved)
{
// spew the output
for(i=0;i<uNumMedia;i++)
{
if((pResColumnArray+i)->uSize)
{
pResPair = (pResColumnArray+i)->pResolvedPairs
+ (pResColumnArray+i)->uCurrentIndex;
}
else
{
pResPair = &UnresolvedPair;
}
*(pResOutput+i) = *pResPair;
}
}
else
{
// if there was some error, preserve that error code,
if(HR_SUCCEEDED(hr))
// otherwise the error is....
hr = CAPS_E_NOMATCH;
}
ERROR_OUT: // well, the success case falls out here too
if(pResColumnArray)
{
for(i=0;i<uNumMedia;i++)
{
if((pResColumnArray+i)->pResolvedPairs)
MemFree((pResColumnArray+i)->pResolvedPairs);
}
MemFree(pResColumnArray);
}
if(pResContext)
{
MemFree(pResContext);
}
return hr;
}
HRESULT CapsCtl::ResetCombinedEntries (void)
{
DWORD x;
if (pAdvertisedSets)
{
for (x = 0; x < pAdvertisedSets->wLength; x++)
{
if (pAdvertisedSets->pTermCapDescriptorArray[x])
{
MemFree (pAdvertisedSets->pTermCapDescriptorArray[x]);
}
}
MemFree (pAdvertisedSets->pTermCapDescriptorArray);
pAdvertisedSets->wLength=0;
MemFree (pAdvertisedSets);
pAdvertisedSets = NULL;
}
if (pSetIDs)
{
MemFree(pSetIDs);
pSetIDs = NULL;
}
dwNumInUse=0;
uAdvertizedSize=0;
return hrSuccess;
}