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.
2507 lines
74 KiB
2507 lines
74 KiB
/************************************************************************
|
|
* *
|
|
* INTEL CORPORATION PROPRIETARY INFORMATION *
|
|
* *
|
|
* This software is supplied under the terms of a license *
|
|
* agreement or non-disclosure agreement with Intel Corporation *
|
|
* and may not be copied or disclosed except in accordance *
|
|
* with the terms of that agreement. *
|
|
* *
|
|
* Copyright (C) 1997 Intel Corp. All Rights Reserved *
|
|
* *
|
|
* $Archive: S:\sturgeon\src\gki\vcs\gkreg.cpv $
|
|
* *
|
|
* $Revision: 1.6 $
|
|
* $Date: 26 Feb 1997 15:33:34 $
|
|
* *
|
|
* $Author: CHULME $
|
|
* *
|
|
* $Log: S:\sturgeon\src\gki\vcs\gkreg.cpv $
|
|
//
|
|
// Rev 1.6 26 Feb 1997 15:33:34 CHULME
|
|
// Call Coder.Free in case of error - potential memory leak plugged
|
|
//
|
|
// Rev 1.5 14 Feb 1997 16:43:06 CHULME
|
|
// Updated comments and removed inaccurate comments
|
|
//
|
|
// Rev 1.4 12 Feb 1997 01:12:52 CHULME
|
|
// Redid thread synchronization to use Gatekeeper.Lock
|
|
//
|
|
// Rev 1.3 08 Feb 1997 12:15:48 CHULME
|
|
// Terminate retry thread in destructor via semaphore
|
|
//
|
|
// Rev 1.2 21 Jan 1997 17:24:06 CHULME
|
|
// Removed gatekeeper identifier from gatekeeper request
|
|
//
|
|
// Rev 1.1 17 Jan 1997 09:02:22 CHULME
|
|
// Changed reg.h to gkreg.h to avoid name conflict with inc directory
|
|
//
|
|
// Rev 1.0 17 Jan 1997 08:48:08 CHULME
|
|
// Initial revision.
|
|
//
|
|
// Rev 1.7 10 Jan 1997 16:15:58 CHULME
|
|
// Removed MFC dependency
|
|
//
|
|
// Rev 1.6 20 Dec 1996 16:39:14 CHULME
|
|
// Removed extraneous debug statements
|
|
//
|
|
// Rev 1.5 20 Dec 1996 01:27:24 CHULME
|
|
// Fixed memory leak with gatekeeper identifier
|
|
//
|
|
// Rev 1.4 10 Dec 1996 11:26:36 CHULME
|
|
// Fixed handling of IRQ to not require response address in PDU
|
|
//
|
|
// Rev 1.3 02 Dec 1996 23:49:58 CHULME
|
|
// Added premptive synchronization code
|
|
//
|
|
// Rev 1.2 22 Nov 1996 15:22:16 CHULME
|
|
// Added VCS log to the header
|
|
*************************************************************************/
|
|
|
|
// registration.cpp : Provides the implementation for the CRegistration class
|
|
//
|
|
#include "precomp.h"
|
|
|
|
#include <process.h>
|
|
#include <stdlib.h>
|
|
#include <time.h>
|
|
#include "GKICOM.H"
|
|
#include "dspider.h"
|
|
#include "dgkilit.h"
|
|
#include "DGKIPROT.H"
|
|
#include "gksocket.h"
|
|
#include "GKREG.H"
|
|
#include "GATEKPR.H"
|
|
#include "h225asn.h"
|
|
#include "coder.hpp"
|
|
#include "dgkiext.h"
|
|
#include "ccerror.h"
|
|
|
|
#ifdef _DEBUG
|
|
#undef THIS_FILE
|
|
static char THIS_FILE[] = __FILE__;
|
|
#endif
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
// CLinkedList Implementation
|
|
|
|
template <class T> CLinkedList<T>::CLinkedList()
|
|
{
|
|
pTail = NULL;
|
|
iCount = 0;
|
|
}
|
|
|
|
template <class T> CLinkedList<T>::~CLinkedList()
|
|
{
|
|
RemoveAll();
|
|
}
|
|
|
|
template <class T> void CLinkedList<T>::AddTail (const T& NewItem)
|
|
{
|
|
AddTailPriv(new TItem<T>(NewItem));
|
|
}
|
|
|
|
template <class T> void CLinkedList<T>::AddTailPriv(TItem<T> *pNewItem)
|
|
{
|
|
if (pTail) // if the list is non-empty - add to tail
|
|
{
|
|
pNewItem->pPrev = pTail;
|
|
pNewItem->pNext = pTail->pNext; // preserve the pointer to the head.
|
|
pTail->pNext->pPrev = pNewItem;
|
|
pTail->pNext = pNewItem; // insert the new element
|
|
}
|
|
else // insert first element
|
|
{ // new element is the tail
|
|
pNewItem->pPrev = pNewItem->pNext = pNewItem;
|
|
}
|
|
|
|
pTail = pNewItem; // move tail to the new item
|
|
iCount++;
|
|
}
|
|
|
|
template <class T> BOOL CLinkedList<T>::IsEmpty(void)
|
|
{
|
|
if (pTail == NULL)
|
|
{ return TRUE; }
|
|
else
|
|
{ return FALSE; }
|
|
}
|
|
|
|
template <class T> POS CLinkedList<T>::GetFirstPos (void)
|
|
{
|
|
if (pTail)
|
|
{ return (POS) pTail->pNext;
|
|
}
|
|
else
|
|
{ return NULL;
|
|
}
|
|
}
|
|
|
|
template <class T> T CLinkedList<T>::GetNext (POS &Position)
|
|
{
|
|
TItem<T> *pCurItem = (TItem<T> *)Position;
|
|
T RetValue;
|
|
|
|
if (Position)
|
|
{
|
|
RetValue = pCurItem->Value;
|
|
if (pCurItem == pTail) // we are at the end of the list
|
|
{ Position = NULL;
|
|
}
|
|
else // move to the next position
|
|
{ Position = (POS)(pCurItem->pNext);
|
|
}
|
|
}
|
|
return RetValue;
|
|
}
|
|
|
|
template <class T> POS CLinkedList<T>::Find (const T& Item)
|
|
{
|
|
TItem<T> *pCurItem;
|
|
|
|
if (!pTail)
|
|
{ return NULL;
|
|
}
|
|
else
|
|
{
|
|
pCurItem = pTail;
|
|
do
|
|
{
|
|
pCurItem = pCurItem->pNext; // starting with the head
|
|
if (pCurItem->Value == Item)
|
|
{ return ((POS) pCurItem); }
|
|
}
|
|
while (pCurItem != pTail);
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
// It moves Position to the next item after removint the current one.
|
|
template <class T> BOOL CLinkedList<T>::RemoveAt (POS &Position)
|
|
{
|
|
TItem<T> *pCurItem = (TItem<T> *)Position;
|
|
|
|
if (!pCurItem)
|
|
{ return FALSE; }
|
|
else if (pCurItem == pCurItem->pNext) // The only element
|
|
{
|
|
Position = NULL;
|
|
pTail = NULL;
|
|
delete pCurItem;
|
|
}
|
|
else
|
|
{
|
|
Position = (POS) pCurItem->pNext;
|
|
pCurItem->pPrev->pNext = pCurItem->pNext;
|
|
pCurItem->pNext->pPrev = pCurItem->pPrev;
|
|
if (pCurItem == pTail)
|
|
{
|
|
pTail = pCurItem->pPrev;
|
|
}
|
|
delete pCurItem;
|
|
}
|
|
iCount--;
|
|
return TRUE;
|
|
}
|
|
|
|
template <class T> T CLinkedList<T>::GetAt(const POS Position)
|
|
{
|
|
TItem<T> *pCurItem = (TItem<T> *)Position;
|
|
T RetValue;
|
|
|
|
if (Position)
|
|
{ RetValue = pCurItem->Value;
|
|
}
|
|
|
|
return RetValue;
|
|
}
|
|
|
|
template <class T> void CLinkedList<T>::RemoveAll(void)
|
|
{
|
|
TItem<T> *pCurItem;
|
|
TItem<T> *pNextItem;
|
|
|
|
if (pTail)
|
|
{
|
|
pCurItem = pTail->pNext; // Start with the head.
|
|
pTail->pNext = NULL;
|
|
|
|
while (pCurItem != NULL)
|
|
{
|
|
pNextItem = pCurItem->pNext;
|
|
delete pCurItem;
|
|
pCurItem = pNextItem;
|
|
}
|
|
}
|
|
pTail = NULL;
|
|
iCount = 0;
|
|
}
|
|
|
|
template <class T> int CLinkedList<T>::GetCount(void)
|
|
{
|
|
return iCount;
|
|
}
|
|
|
|
|
|
VOID CALLBACK RetryTimerProc(HWND hwnd, UINT uMsg, UINT_PTR idEvent, DWORD dwTime)
|
|
{
|
|
|
|
#ifdef _DEBUG
|
|
char szGKDebug[80];
|
|
#endif
|
|
HRESULT hResult = GKI_OK;
|
|
|
|
ASSERT(g_pGatekeeper);
|
|
if(g_pGatekeeper == NULL)
|
|
return;
|
|
|
|
g_pGatekeeper->Lock();
|
|
if(g_pReg)
|
|
{
|
|
hResult = g_pReg->Retry();
|
|
}
|
|
|
|
if(hResult != GKI_OK)
|
|
{
|
|
SPIDER_TRACE(SP_NEWDEL, "del g_pReg = %X\n", g_pReg);
|
|
delete g_pReg;
|
|
g_pReg = 0;
|
|
}
|
|
g_pGatekeeper->Unlock();
|
|
}
|
|
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
// CRegistration construction
|
|
|
|
CRegistration::CRegistration()
|
|
{
|
|
// ABSTRACT: The constructor for the CRegistration class will initialize
|
|
// the member variables. Notably missing is the construction
|
|
// of the pointed to socket object. This must be done after
|
|
// constructing this object to allow for checking the error code.
|
|
// AUTHOR: Colin Hulme
|
|
|
|
#ifdef _DEBUG
|
|
char szGKDebug[80];
|
|
#endif
|
|
|
|
SPIDER_TRACE(SP_CONDES, "CRegistration::CRegistration()\n", 0);
|
|
m_pVendorInfo = NULL;
|
|
m_pCallSignalAddress = 0;
|
|
memset(&m_terminalType, 0, sizeof(EndpointType));
|
|
m_pRgstrtnRqst_trmnlAls = 0;
|
|
m_hWnd = 0;
|
|
m_wBaseMessage = 0;
|
|
m_usRegistrationTransport = 0;
|
|
|
|
m_pLocationInfo = 0;
|
|
memset(&m_Location[0], 0, sizeof(TransportAddress) * 2);
|
|
|
|
memset(&m_RCm_gtkprIdntfr, 0, sizeof(GatekeeperIdentifier));
|
|
memset(&m_endpointID, 0, sizeof(EndpointIdentifier));
|
|
|
|
m_requestSeqNum = 0;
|
|
m_pRASAddress = 0;
|
|
|
|
m_State = GK_REG_PENDING;
|
|
SPIDER_TRACE(SP_STATE, "m_State = GK_REG_PENDING (%X)\n", this);
|
|
|
|
m_pRasMessage = 0;
|
|
m_pSocket = 0;
|
|
m_hRcvThread = 0;
|
|
// m_hRetryThread = 0;
|
|
|
|
m_uTimer = 0;
|
|
m_uRetryResetCount = GKR_RETRY_INTERVAL_SECONDS * (1000/GKR_RETRY_TICK_MS);
|
|
m_uRetryCountdown = GKR_RETRY_INTERVAL_SECONDS;
|
|
m_uMaxRetryCount = GKR_RETRY_MAX;
|
|
m_usRetryCount = 0;
|
|
InitializeCriticalSection(&m_SocketCRS);
|
|
|
|
#ifdef BROADCAST_DISCOVERY
|
|
m_hDiscThread = 0;
|
|
#endif
|
|
// m_dwLockingThread = 0;
|
|
// m_hRetrySemaphore = NULL;
|
|
|
|
// InitializeCriticalSection(&m_CriticalSection);
|
|
|
|
// Initialize the base call reference value to a random number
|
|
srand( (unsigned)time( NULL ) );
|
|
m_usCallReferenceValue = (unsigned short)rand();
|
|
}
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
// CRegistration destruction
|
|
|
|
CRegistration::~CRegistration()
|
|
{
|
|
// ABSTRACT: The destructor for the CRegistration class must free the
|
|
// memory allocated for the Transport addresses and Alias
|
|
// addresses. It does this by deleting the structures and
|
|
// walking the link list.
|
|
// AUTHOR: Colin Hulme
|
|
|
|
SeqTransportAddr *pTA1, *pTA2;
|
|
SeqAliasAddr *pAA1, *pAA2;
|
|
DWORD dwErrorCode;
|
|
#ifdef _DEBUG
|
|
char szGKDebug[80];
|
|
#endif
|
|
|
|
SPIDER_TRACE(SP_CONDES, "CRegistration::~CRegistration()\n", 0);
|
|
|
|
#if(0)
|
|
// Terminate the Retry Thread
|
|
if(m_hRetryThread)
|
|
{
|
|
if(m_hRetrySemaphore)
|
|
{
|
|
SPIDER_TRACE(SP_THREAD, "Release retry thread %X\n", m_hRetryThread);
|
|
// Signal the thread to shutdown
|
|
ReleaseSemaphore(m_hRetrySemaphore,1,NULL);
|
|
|
|
// Wait for the thread to terminate
|
|
dwErrorCode = WaitForSingleObject(m_hRetryThread, TIMEOUT_THREAD);
|
|
m_hRetryThread = NULL;
|
|
}
|
|
}
|
|
#else
|
|
|
|
// Stop retry timer
|
|
if(m_uTimer)
|
|
{
|
|
KillTimer(m_hWnd, m_uTimer);
|
|
}
|
|
#endif
|
|
if(m_pVendorInfo)
|
|
FreeVendorInfo(m_pVendorInfo);
|
|
|
|
// Delete allocated memory for sequence of call signal addresses
|
|
pTA1 = m_pCallSignalAddress;
|
|
while (pTA1 != 0)
|
|
{
|
|
pTA2 = pTA1->next;
|
|
SPIDER_TRACE(SP_NEWDEL, "del pTA1 = %X\n", pTA1);
|
|
delete pTA1;
|
|
pTA1 = pTA2;
|
|
}
|
|
|
|
// Delete allocated memory for sequence of alias addresses
|
|
pAA1 = m_pRgstrtnRqst_trmnlAls;
|
|
while (pAA1 != 0)
|
|
{
|
|
pAA2 = pAA1->next;
|
|
if (pAA1->value.choice == h323_ID_chosen)
|
|
{
|
|
SPIDER_TRACE(SP_NEWDEL, "del pAA1->value.u.h323_ID.value = %X\n", pAA1->value.u.h323_ID.value);
|
|
delete pAA1->value.u.h323_ID.value;
|
|
}
|
|
SPIDER_TRACE(SP_NEWDEL, "del pAA1 = %X\n", pAA1);
|
|
delete pAA1;
|
|
pAA1 = pAA2;
|
|
}
|
|
|
|
// Delete allocated memory for sequence of location alias addresses
|
|
pAA1 = m_pLocationInfo;
|
|
while (pAA1 != 0)
|
|
{
|
|
pAA2 = pAA1->next;
|
|
if (pAA1->value.choice == h323_ID_chosen)
|
|
{
|
|
SPIDER_TRACE(SP_NEWDEL, "del pAA1->value.u.h323_ID.value = %X\n", pAA1->value.u.h323_ID.value);
|
|
delete pAA1->value.u.h323_ID.value;
|
|
}
|
|
SPIDER_TRACE(SP_NEWDEL, "del pAA1 = %X\n", pAA1);
|
|
delete pAA1;
|
|
pAA1 = pAA2;
|
|
}
|
|
|
|
// Delete allocated memory for identifiers
|
|
if (m_RCm_gtkprIdntfr.length)
|
|
{
|
|
SPIDER_TRACE(SP_NEWDEL, "del m_RCm_gtkprIdntfr.value = %X\n", m_RCm_gtkprIdntfr.value);
|
|
delete m_RCm_gtkprIdntfr.value;
|
|
}
|
|
if (m_endpointID.length)
|
|
{
|
|
SPIDER_TRACE(SP_NEWDEL, "del m_endpointID.value = %X\n", m_endpointID.value);
|
|
delete m_endpointID.value;
|
|
}
|
|
|
|
// Delete allocated memory for sequence of RAS addresses
|
|
pTA1 = m_pRASAddress;
|
|
while (pTA1 != 0)
|
|
{
|
|
pTA2 = pTA1->next;
|
|
SPIDER_TRACE(SP_NEWDEL, "del pTA1 = %X\n", pTA1);
|
|
delete pTA1;
|
|
pTA1 = pTA2;
|
|
}
|
|
|
|
if (!m_Calls.IsEmpty())
|
|
{
|
|
// Free up any call objects
|
|
// on this registration object
|
|
POS pos;
|
|
for( pos = m_Calls.GetFirstPos(); pos != NULL; )
|
|
{
|
|
// Delete the call object
|
|
CCall *pCall = m_Calls.GetNext(pos);
|
|
SPIDER_TRACE(SP_NEWDEL, "del pCall = %X\n", pCall);
|
|
delete pCall;
|
|
}
|
|
// Now remove all pointers from the list
|
|
m_Calls.RemoveAll();
|
|
}
|
|
|
|
// Delete memory for last RAS message if still allocated
|
|
if (m_pRasMessage)
|
|
{
|
|
SPIDER_TRACE(SP_NEWDEL, "del m_pRasMessage = %X\n", m_pRasMessage);
|
|
delete m_pRasMessage;
|
|
}
|
|
|
|
// if (m_dwLockingThread)
|
|
// Unlock();
|
|
// DeleteCriticalSection(&m_CriticalSection);
|
|
#if(0)
|
|
if(m_hRetrySemaphore)
|
|
{
|
|
CloseHandle(m_hRetrySemaphore);
|
|
m_hRetrySemaphore = NULL;
|
|
}
|
|
#endif
|
|
|
|
|
|
m_pSocket->Close();
|
|
|
|
LockSocket();
|
|
// Close the socket and delete the socket object
|
|
SPIDER_TRACE(SP_NEWDEL, "del m_pSocket = %X\n", m_pSocket);
|
|
delete m_pSocket;
|
|
UnlockSocket();
|
|
|
|
DeleteCriticalSection(&m_SocketCRS);
|
|
}
|
|
|
|
UINT_PTR CRegistration::StartRetryTimer(void)
|
|
{
|
|
if(m_uTimer)
|
|
{
|
|
KillTimer(m_hWnd, m_uTimer);
|
|
m_uTimer = 0;
|
|
}
|
|
m_uRetryResetCount = GKR_RETRY_INTERVAL_SECONDS * (1000/GKR_RETRY_TICK_MS);
|
|
m_uRetryCountdown = GKR_RETRY_INTERVAL_SECONDS;
|
|
m_uMaxRetryCount = GKR_RETRY_MAX;
|
|
m_usRetryCount = 0;
|
|
|
|
m_uTimer = SetTimer(NULL, NULL, GKR_RETRY_TICK_MS, RetryTimerProc);
|
|
//m_uTimer = SetTimer(hWnd, GKREG_TIMER_ID, GKR_RETRY_TICK_MS, RetryTimerProc);
|
|
return m_uTimer;
|
|
}
|
|
|
|
HRESULT
|
|
CRegistration::AddVendorInfo(PCC_VENDORINFO pVendorInfo)
|
|
{
|
|
HRESULT hr = GKI_OK;
|
|
if(m_pVendorInfo)
|
|
{
|
|
FreeVendorInfo(m_pVendorInfo);
|
|
m_pVendorInfo = NULL;
|
|
}
|
|
if(pVendorInfo)
|
|
{
|
|
hr = CopyVendorInfo(&m_pVendorInfo, pVendorInfo);
|
|
if(hr != CC_OK)
|
|
{
|
|
m_pVendorInfo = NULL;
|
|
}
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
HRESULT
|
|
CRegistration::AddCallSignalAddr(TransportAddress& rvalue)
|
|
{
|
|
// ABSTRACT: This procedure is called to add a call signal address
|
|
// to the link list of call signal addresses. This will
|
|
// be called for each transport on receiving a GKI_RegistrationRequest.
|
|
// A local copy is made to avoid reliance on the client
|
|
// keeping the memory valid. This procedure returns 0 if
|
|
// successful and non-zero for a failure.
|
|
// AUTHOR: Colin Hulme
|
|
|
|
SeqTransportAddr *pCSA;
|
|
#ifdef _DEBUG
|
|
char szGKDebug[80];
|
|
#endif
|
|
|
|
SPIDER_TRACE(SP_FUNC, "CRegistration::AddCallSignalAddr(%X)\n", rvalue.choice);
|
|
|
|
if (m_pCallSignalAddress == 0) // First one in the list
|
|
{
|
|
m_pCallSignalAddress = new SeqTransportAddr;
|
|
SPIDER_TRACE(SP_NEWDEL, "new m_pCallSignalAddress = %X\n", m_pCallSignalAddress);
|
|
if (m_pCallSignalAddress == 0)
|
|
return (GKI_NO_MEMORY);
|
|
memset(m_pCallSignalAddress, 0, sizeof(SeqTransportAddr));
|
|
pCSA = m_pCallSignalAddress;
|
|
}
|
|
else
|
|
{
|
|
for (pCSA = m_pCallSignalAddress; pCSA->next != 0; pCSA = pCSA->next)
|
|
; // walk the list til last entry
|
|
pCSA->next = new SeqTransportAddr;
|
|
SPIDER_TRACE(SP_NEWDEL, "new pCSA->next = %X\n", pCSA->next);
|
|
if (pCSA->next == 0)
|
|
return (GKI_NO_MEMORY);
|
|
memset(pCSA->next, 0, sizeof(SeqTransportAddr));
|
|
pCSA = pCSA->next;
|
|
}
|
|
pCSA->next = 0; // initialize new structure fields
|
|
pCSA->value = rvalue;
|
|
return (GKI_OK);
|
|
}
|
|
|
|
HRESULT
|
|
CRegistration::AddRASAddr(TransportAddress& rvalue, unsigned short usPort)
|
|
{
|
|
// ABSTRACT: This procedure is called to add a RAS address
|
|
// to the link list of RAS addresses. This will
|
|
// be called only for the transport used for the registration
|
|
// request. This procedure returns 0 if successful and non-zero
|
|
// for a failure.
|
|
// AUTHOR: Colin Hulme
|
|
|
|
#ifdef _DEBUG
|
|
char szGKDebug[80];
|
|
#endif
|
|
|
|
SPIDER_TRACE(SP_FUNC, "CRegistration::AddRASAddr(%X, usPort)\n", rvalue.choice);
|
|
|
|
m_pRASAddress = new SeqTransportAddr;
|
|
SPIDER_TRACE(SP_NEWDEL, "new m_pRASAddress = %X\n", m_pRASAddress);
|
|
if (m_pRASAddress == 0)
|
|
return (GKI_NO_MEMORY);
|
|
memset(m_pRASAddress, 0, sizeof(SeqTransportAddr));
|
|
m_pRASAddress->next = 0; // initialize new structure fields
|
|
m_pRASAddress->value = rvalue;
|
|
|
|
// Add actual RAS port to RAS address
|
|
switch (m_pRASAddress->value.choice)
|
|
{
|
|
case ipAddress_chosen:
|
|
m_pRASAddress->value.u.ipAddress.port = usPort;
|
|
break;
|
|
case ipxAddress_chosen:
|
|
m_pRASAddress->value.u.ipxAddress.port.value[0] = HIBYTE(usPort);
|
|
m_pRASAddress->value.u.ipxAddress.port.value[1] = LOBYTE(usPort);
|
|
break;
|
|
}
|
|
return (GKI_OK);
|
|
}
|
|
|
|
HRESULT
|
|
CRegistration::AddAliasAddr(AliasAddress& rvalue)
|
|
{
|
|
// ABSTRACT: This procedure is called to add an alias address
|
|
// to the link list of alias addresses. This will
|
|
// be called for each alias on receiving a GKI_RegistrationRequest.
|
|
// A local copy is made to avoid reliance on the client
|
|
// keeping the memory valid.
|
|
// In the eventuality that the gatekeeper assigns alias
|
|
// addresses, this procedure will be called for each alias
|
|
// contained in the registrationConfirm message.
|
|
// This procedure returns 0 if successful and non-zero
|
|
// for a failure.
|
|
// AUTHOR: Colin Hulme
|
|
|
|
SeqAliasAddr *p1;
|
|
unsigned short uIdx;
|
|
unsigned short *pus;
|
|
#ifdef _DEBUG
|
|
char szGKDebug[80];
|
|
#endif
|
|
|
|
SPIDER_TRACE(SP_FUNC, "CRegistration::AddAliasAddr(%X)\n", rvalue.choice);
|
|
|
|
if (m_pRgstrtnRqst_trmnlAls == 0) // First one in the list
|
|
{
|
|
m_pRgstrtnRqst_trmnlAls = new SeqAliasAddr;
|
|
SPIDER_TRACE(SP_NEWDEL, "new m_pRgstrtnRqst_trmnlAls = %X\n", m_pRgstrtnRqst_trmnlAls);
|
|
if (m_pRgstrtnRqst_trmnlAls == 0)
|
|
return (GKI_NO_MEMORY);
|
|
memset(m_pRgstrtnRqst_trmnlAls, 0, sizeof(SeqAliasAddr));
|
|
p1 = m_pRgstrtnRqst_trmnlAls;
|
|
}
|
|
else
|
|
{
|
|
for (p1 = m_pRgstrtnRqst_trmnlAls; p1->next != 0; p1 = p1->next)
|
|
; // walk the list til last entry
|
|
p1->next = new SeqAliasAddr;
|
|
SPIDER_TRACE(SP_NEWDEL, "new p1->next = %X\n", p1->next);
|
|
if (p1->next == 0)
|
|
return (GKI_NO_MEMORY);
|
|
memset(p1->next, 0, sizeof(SeqAliasAddr));
|
|
p1 = p1->next;
|
|
}
|
|
p1->next = 0; // initialize new structure fields
|
|
p1->value = rvalue;
|
|
if (p1->value.choice == h323_ID_chosen)
|
|
{
|
|
pus = new unsigned short[p1->value.u.h323_ID.length];
|
|
SPIDER_TRACE(SP_NEWDEL, "new pus = %X\n", pus);
|
|
if (pus == 0)
|
|
return (GKI_NO_MEMORY);
|
|
memset(pus, 0, sizeof(unsigned short) * p1->value.u.h323_ID.length);
|
|
for (uIdx = 0; uIdx < p1->value.u.h323_ID.length; uIdx++)
|
|
*(pus + uIdx) = *(p1->value.u.h323_ID.value + uIdx);
|
|
p1->value.u.h323_ID.value = pus;
|
|
}
|
|
return (GKI_OK);
|
|
}
|
|
|
|
HRESULT
|
|
CRegistration::AddLocationInfo(AliasAddress& rvalue)
|
|
{
|
|
// ABSTRACT: This procedure is called to add an alias address
|
|
// to the link list of alias addresses. This will
|
|
// be called for each alias on receiving a GKI_RegistrationRequest.
|
|
// A local copy is made to avoid reliance on the client
|
|
// keeping the memory valid.
|
|
// In the eventuality that the gatekeeper assigns alias
|
|
// addresses, this procedure will be called for each alias
|
|
// contained in the registrationConfirm message.
|
|
// This procedure returns 0 if successful and non-zero
|
|
// for a failure.
|
|
// AUTHOR: Colin Hulme
|
|
|
|
SeqAliasAddr *p1;
|
|
unsigned short uIdx;
|
|
unsigned short *pus;
|
|
#ifdef _DEBUG
|
|
char szGKDebug[80];
|
|
#endif
|
|
|
|
SPIDER_TRACE(SP_FUNC, "CRegistration::AddLocationInfo(%X)\n", rvalue.choice);
|
|
|
|
if (m_pLocationInfo == 0) // First one in the list
|
|
{
|
|
m_pLocationInfo = new SeqAliasAddr;
|
|
SPIDER_TRACE(SP_NEWDEL, "new m_pLocationInfo = %X\n", m_pLocationInfo);
|
|
if (m_pLocationInfo == 0)
|
|
return (GKI_NO_MEMORY);
|
|
memset(m_pLocationInfo, 0, sizeof(SeqAliasAddr));
|
|
p1 = m_pLocationInfo;
|
|
}
|
|
else
|
|
{
|
|
for (p1 = m_pLocationInfo; p1->next != 0; p1 = p1->next)
|
|
; // walk the list til last entry
|
|
p1->next = new SeqAliasAddr;
|
|
SPIDER_TRACE(SP_NEWDEL, "new p1->next = %X\n", p1->next);
|
|
if (p1->next == 0)
|
|
return (GKI_NO_MEMORY);
|
|
memset(p1->next, 0, sizeof(SeqAliasAddr));
|
|
p1 = p1->next;
|
|
}
|
|
p1->next = 0; // initialize new structure fields
|
|
p1->value = rvalue;
|
|
if (p1->value.choice == h323_ID_chosen)
|
|
{
|
|
pus = new unsigned short[p1->value.u.h323_ID.length];
|
|
SPIDER_TRACE(SP_NEWDEL, "new pus = %X\n", pus);
|
|
if (pus == 0)
|
|
return (GKI_NO_MEMORY);
|
|
memset(pus, 0, sizeof(unsigned short) * p1->value.u.h323_ID.length);
|
|
for (uIdx = 0; uIdx < p1->value.u.h323_ID.length; uIdx++)
|
|
*(pus + uIdx) = *(p1->value.u.h323_ID.value + uIdx);
|
|
p1->value.u.h323_ID.value = pus;
|
|
}
|
|
return (GKI_OK);
|
|
}
|
|
|
|
TransportAddress *
|
|
CRegistration::GetTransportAddress(unsigned short usCallTransport)
|
|
{
|
|
SeqTransportAddr *pCSA;
|
|
|
|
for (pCSA = m_pCallSignalAddress; pCSA != 0; pCSA = pCSA->next)
|
|
{
|
|
if (pCSA->value.choice == usCallTransport)
|
|
return (&pCSA->value);
|
|
}
|
|
return (NULL); // Didn't find it
|
|
}
|
|
|
|
HRESULT
|
|
CRegistration::RegistrationRequest(BOOL fDiscovery)
|
|
{
|
|
// ABSTRACT: This procedure will create a RegistrationRequest structure
|
|
// call the encoder and send the PDU. If it is successful, it
|
|
// will return 0, else it will return an error code. Note: The
|
|
// memory allocated for the RAS Message is not freed until either
|
|
// a response from the gatekeeper or it times out. This allows
|
|
// for retransmission without having to rebuild this message.
|
|
// AUTHOR: Colin Hulme
|
|
|
|
ASN1_BUF Asn1Buf;
|
|
DWORD dwErrorCode;
|
|
#ifdef _DEBUG
|
|
char szGKDebug[80];
|
|
#endif
|
|
|
|
SPIDER_TRACE(SP_FUNC, "CRegistration::RegistrationRequest()\n", 0);
|
|
ASSERT(g_pCoder);
|
|
if (g_pCoder == NULL)
|
|
return (GKI_NOT_INITIALIZED);
|
|
|
|
// Allocate a RasMessage structure and initialized to 0
|
|
m_usRetryCount = 0;
|
|
m_uRetryCountdown = m_uRetryResetCount;
|
|
|
|
m_pRasMessage = new RasMessage;
|
|
SPIDER_TRACE(SP_NEWDEL, "new m_pRasMessage = %X\n", m_pRasMessage);
|
|
if (m_pRasMessage == 0)
|
|
return (GKI_NO_MEMORY);
|
|
memset(m_pRasMessage, 0, sizeof(RasMessage));
|
|
|
|
// Setup structure fields for RegistrationRequest
|
|
m_pRasMessage->choice = registrationRequest_chosen;
|
|
if (m_pRgstrtnRqst_trmnlAls != 0)
|
|
m_pRasMessage->u.registrationRequest.bit_mask |= RgstrtnRqst_trmnlAls_present;
|
|
if (m_RCm_gtkprIdntfr.length != 0)
|
|
m_pRasMessage->u.registrationRequest.bit_mask |= RgstrtnRqst_gtkprIdntfr_present;
|
|
|
|
m_pRasMessage->u.registrationRequest.requestSeqNum = ++m_requestSeqNum;
|
|
|
|
// discoveryComplete is a ASN1_BOOL (char) and fDiscovery is a BOOL (int) so the
|
|
// cast was added to remove a compiler warning. Since the value of fDiscovery
|
|
// is always 0 or 1, no loss occurs in the cast. -- DLD
|
|
m_pRasMessage->u.registrationRequest.discoveryComplete = (ASN1_BOOL)fDiscovery;
|
|
m_pRasMessage->u.registrationRequest.callSignalAddress = (PRegistrationRequest_callSignalAddress)m_pCallSignalAddress;
|
|
m_pRasMessage->u.registrationRequest.rasAddress = (PRegistrationRequest_rasAddress)m_pRASAddress;
|
|
m_pRasMessage->u.registrationRequest.terminalType = m_terminalType;
|
|
m_pRasMessage->u.registrationRequest.RgstrtnRqst_trmnlAls = (PRegistrationRequest_terminalAlias)m_pRgstrtnRqst_trmnlAls;
|
|
m_pRasMessage->u.registrationRequest.RgstrtnRqst_gtkprIdntfr = m_RCm_gtkprIdntfr;
|
|
m_pRasMessage->u.registrationRequest.endpointVendor.bit_mask = 0;
|
|
|
|
if(m_pVendorInfo)
|
|
{
|
|
m_pRasMessage->u.registrationRequest.endpointVendor.vendor.t35CountryCode
|
|
= m_pVendorInfo->bCountryCode;
|
|
|
|
m_pRasMessage->u.registrationRequest.endpointVendor.vendor.t35Extension
|
|
= m_pVendorInfo->bExtension;
|
|
m_pRasMessage->u.registrationRequest.endpointVendor.vendor.manufacturerCode
|
|
= m_pVendorInfo->wManufacturerCode;
|
|
|
|
if(m_pVendorInfo->pProductNumber
|
|
&& m_pVendorInfo->pProductNumber->pOctetString
|
|
&& m_pVendorInfo->pProductNumber->wOctetStringLength)
|
|
{
|
|
UINT uSize = min(m_pVendorInfo->pProductNumber->wOctetStringLength,
|
|
sizeof(m_pRasMessage->u.registrationRequest.endpointVendor.productId.value));
|
|
|
|
m_pRasMessage->u.registrationRequest.endpointVendor.bit_mask |= productId_present;
|
|
// truncate to fit size of registrationRequest.endpointVendor.productId.value
|
|
m_pRasMessage->u.registrationRequest.endpointVendor.productId.length = uSize;
|
|
memcpy(&m_pRasMessage->u.registrationRequest.endpointVendor.productId.value,
|
|
m_pVendorInfo->pProductNumber->pOctetString, uSize);
|
|
}
|
|
|
|
if(m_pVendorInfo->pVersionNumber
|
|
&& m_pVendorInfo->pVersionNumber->pOctetString
|
|
&& m_pVendorInfo->pVersionNumber->wOctetStringLength)
|
|
{
|
|
UINT uSize = min(m_pVendorInfo->pVersionNumber->wOctetStringLength,
|
|
sizeof(m_pRasMessage->u.registrationRequest.endpointVendor.versionId.value));
|
|
m_pRasMessage->u.registrationRequest.endpointVendor.bit_mask |= versionId_present;
|
|
// truncate to fit size of registrationRequest.endpointVendor.versionId.value
|
|
m_pRasMessage->u.registrationRequest.endpointVendor.versionId.length = uSize;
|
|
memcpy(&m_pRasMessage->u.registrationRequest.endpointVendor.versionId.value,
|
|
m_pVendorInfo->pVersionNumber->pOctetString, uSize);
|
|
}
|
|
}
|
|
#ifdef _DEBUG
|
|
if (dwGKIDLLFlags & SP_DUMPMEM)
|
|
DumpMem(m_pRasMessage, sizeof(RasMessage));
|
|
#endif
|
|
|
|
// Assign ProtocolIdentifier
|
|
g_pCoder->SetProtocolIdentifier(*m_pRasMessage);
|
|
|
|
// Encode the PDU & send it
|
|
dwErrorCode = g_pCoder->Encode(m_pRasMessage, &Asn1Buf);
|
|
if (dwErrorCode)
|
|
return (GKI_ENCODER_ERROR);
|
|
|
|
// Create a backup copy of the encoded PDU if using debug echo support
|
|
if (fGKIEcho)
|
|
{
|
|
pEchoBuff = new char[Asn1Buf.length];
|
|
SPIDER_TRACE(SP_NEWDEL, "new pEchoBuff = %X\n", pEchoBuff);
|
|
if (pEchoBuff == 0)
|
|
return (GKI_NO_MEMORY);
|
|
memcpy(pEchoBuff, (char *)Asn1Buf.value, Asn1Buf.length);
|
|
nEchoLen = Asn1Buf.length;
|
|
}
|
|
|
|
SPIDER_TRACE(SP_PDU, "Send RRQ; g_pReg = %X\n", this);
|
|
if (fGKIDontSend == FALSE)
|
|
if (m_pSocket->Send((char *)Asn1Buf.value, Asn1Buf.length) == SOCKET_ERROR)
|
|
return (GKI_WINSOCK2_ERROR(SOCKET_ERROR));
|
|
|
|
// Free the encoder memory
|
|
g_pCoder->Free(Asn1Buf);
|
|
|
|
return (GKI_OK);
|
|
}
|
|
|
|
HRESULT
|
|
CRegistration::UnregistrationRequest(void)
|
|
{
|
|
// ABSTRACT: This procedure will create an UnregistrationRequest structure
|
|
// call the encoder and send the PDU. If it is successful, it
|
|
// will return 0, else it will return an error code.
|
|
// AUTHOR: Colin Hulme
|
|
|
|
ASN1_BUF Asn1Buf;
|
|
DWORD dwErrorCode;
|
|
#ifdef _DEBUG
|
|
char szGKDebug[80];
|
|
#endif
|
|
|
|
SPIDER_TRACE(SP_FUNC, "CRegistration::UnregistrationRequest()\n", 0);
|
|
ASSERT(g_pCoder);
|
|
if (g_pCoder == NULL)
|
|
return (GKI_NOT_INITIALIZED);
|
|
|
|
// Allocate a RasMessage structure and initialized to 0
|
|
m_usRetryCount = 0;
|
|
m_uRetryCountdown = m_uRetryResetCount;
|
|
|
|
m_pRasMessage = new RasMessage;
|
|
SPIDER_TRACE(SP_NEWDEL, "new m_pRasMessage = %X\n", m_pRasMessage);
|
|
if (m_pRasMessage == 0)
|
|
return (GKI_NO_MEMORY);
|
|
memset(m_pRasMessage, 0, sizeof(RasMessage));
|
|
|
|
// Setup structure fields for UnregistrationRequest
|
|
m_pRasMessage->choice = unregistrationRequest_chosen;
|
|
if (m_pRgstrtnRqst_trmnlAls != 0)
|
|
m_pRasMessage->u.unregistrationRequest.bit_mask |= UnrgstrtnRqst_endpntAls_present;
|
|
if (m_endpointID.length != 0)
|
|
m_pRasMessage->u.unregistrationRequest.bit_mask |= URt_endpntIdntfr_present;
|
|
|
|
m_pRasMessage->u.unregistrationRequest.requestSeqNum = ++m_requestSeqNum;
|
|
m_pRasMessage->u.unregistrationRequest.callSignalAddress = (PUnregistrationRequest_callSignalAddress)m_pCallSignalAddress;
|
|
m_pRasMessage->u.unregistrationRequest.UnrgstrtnRqst_endpntAls = (PUnregistrationRequest_endpointAlias)m_pRgstrtnRqst_trmnlAls;
|
|
m_pRasMessage->u.unregistrationRequest.URt_endpntIdntfr = m_endpointID;
|
|
#ifdef _DEBUG
|
|
if (dwGKIDLLFlags & SP_DUMPMEM)
|
|
DumpMem(m_pRasMessage, sizeof(RasMessage));
|
|
#endif
|
|
|
|
// Encode the PDU & send it
|
|
dwErrorCode = g_pCoder->Encode(m_pRasMessage, &Asn1Buf);
|
|
if (dwErrorCode)
|
|
return (GKI_ENCODER_ERROR);
|
|
|
|
// Create a backup copy of the encoded PDU if using debug echo support
|
|
if (fGKIEcho)
|
|
{
|
|
pEchoBuff = new char[Asn1Buf.length];
|
|
SPIDER_TRACE(SP_NEWDEL, "new pEchoBuff = %X\n", pEchoBuff);
|
|
if (pEchoBuff == 0)
|
|
return (GKI_NO_MEMORY);
|
|
memcpy(pEchoBuff, (char *)Asn1Buf.value, Asn1Buf.length);
|
|
nEchoLen = Asn1Buf.length;
|
|
}
|
|
|
|
m_State = GK_UNREG_PENDING;
|
|
SPIDER_TRACE(SP_STATE, "m_State = GK_UNREG_PENDING (%X)\n", this);
|
|
|
|
SPIDER_TRACE(SP_PDU, "Send URQ; g_pReg = %X\n", this);
|
|
if (fGKIDontSend == FALSE)
|
|
if (m_pSocket->Send((char *)Asn1Buf.value, Asn1Buf.length) == SOCKET_ERROR)
|
|
return (GKI_WINSOCK2_ERROR(SOCKET_ERROR));
|
|
|
|
// Free the encoder memory
|
|
g_pCoder->Free(Asn1Buf);
|
|
|
|
return (GKI_OK);
|
|
}
|
|
|
|
HRESULT
|
|
CRegistration::LocationRequest(void)
|
|
{
|
|
// ABSTRACT: This procedure will create a LocationRequest structure
|
|
// call the encoder and send the PDU. If it is successful, it
|
|
// will return 0, else it will return an error code. Note: The
|
|
// memory allocated for the RAS Message is not freed until either
|
|
// a response from the gatekeeper or it times out. This allows
|
|
// for retransmission without having to rebuild this message.
|
|
// AUTHOR: Colin Hulme
|
|
|
|
ASN1_BUF Asn1Buf;
|
|
DWORD dwErrorCode;
|
|
#ifdef _DEBUG
|
|
char szGKDebug[80];
|
|
#endif
|
|
|
|
SPIDER_TRACE(SP_FUNC, "CRegistration::LocationRequest()\n", 0);
|
|
ASSERT(g_pCoder);
|
|
if (g_pCoder == NULL)
|
|
return (GKI_NOT_INITIALIZED);
|
|
|
|
// Allocate a RasMessage structure and initialized to 0
|
|
m_usRetryCount = 0;
|
|
m_uRetryCountdown = m_uRetryResetCount;
|
|
|
|
m_pRasMessage = new RasMessage;
|
|
SPIDER_TRACE(SP_NEWDEL, "new m_pRasMessage = %X\n", m_pRasMessage);
|
|
if (m_pRasMessage == 0)
|
|
return (GKI_NO_MEMORY);
|
|
memset(m_pRasMessage, 0, sizeof(RasMessage));
|
|
|
|
// Setup structure fields for LocationRequest
|
|
m_pRasMessage->choice = locationRequest_chosen;
|
|
if (m_endpointID.length != 0)
|
|
m_pRasMessage->u.locationRequest.bit_mask |= LctnRqst_endpntIdntfr_present;
|
|
|
|
m_pRasMessage->u.locationRequest.requestSeqNum = ++m_requestSeqNum;
|
|
|
|
m_pRasMessage->u.locationRequest.LctnRqst_endpntIdntfr = m_endpointID;
|
|
m_pRasMessage->u.locationRequest.destinationInfo =
|
|
(PLocationRequest_destinationInfo)m_pLocationInfo;
|
|
m_pRasMessage->u.locationRequest.replyAddress = m_pRASAddress->value;
|
|
|
|
#ifdef _DEBUG
|
|
if (dwGKIDLLFlags & SP_DUMPMEM)
|
|
DumpMem(m_pRasMessage, sizeof(RasMessage));
|
|
#endif
|
|
|
|
// Encode the PDU & send it
|
|
dwErrorCode = g_pCoder->Encode(m_pRasMessage, &Asn1Buf);
|
|
if (dwErrorCode)
|
|
return (GKI_ENCODER_ERROR);
|
|
|
|
// Create a backup copy of the encoded PDU if using debug echo support
|
|
if (fGKIEcho)
|
|
{
|
|
pEchoBuff = new char[Asn1Buf.length];
|
|
SPIDER_TRACE(SP_NEWDEL, "new pEchoBuff = %X\n", pEchoBuff);
|
|
if (pEchoBuff == 0)
|
|
return (GKI_NO_MEMORY);
|
|
memcpy(pEchoBuff, (char *)Asn1Buf.value, Asn1Buf.length);
|
|
nEchoLen = Asn1Buf.length;
|
|
}
|
|
|
|
m_State = GK_LOC_PENDING;
|
|
SPIDER_TRACE(SP_STATE, "m_State = GK_LOC_PENDING (%X)\n", this);
|
|
|
|
SPIDER_TRACE(SP_PDU, "Send LRQ; g_pReg = %X\n", this);
|
|
if (fGKIDontSend == FALSE)
|
|
if (m_pSocket->Send((char *)Asn1Buf.value, Asn1Buf.length) == SOCKET_ERROR)
|
|
return (GKI_WINSOCK2_ERROR(SOCKET_ERROR));
|
|
|
|
// Free the encoder memory
|
|
g_pCoder->Free(Asn1Buf);
|
|
|
|
return (GKI_OK);
|
|
}
|
|
|
|
HRESULT
|
|
CRegistration::GatekeeperRequest(void)
|
|
{
|
|
// ABSTRACT: This procedure will create a GatekeeperRequest structure
|
|
// call the encoder and send the PDU. If it is successful, it
|
|
// will return 0, else it will return an error code.
|
|
// AUTHOR: Colin Hulme
|
|
|
|
ASN1_BUF Asn1Buf;
|
|
DWORD dwErrorCode;
|
|
#ifdef _DEBUG
|
|
char szGKDebug[80];
|
|
#endif
|
|
|
|
SPIDER_TRACE(SP_FUNC, "CRegistration::GatekeeperRequest()\n", 0);
|
|
ASSERT(g_pCoder);
|
|
if (g_pCoder == NULL)
|
|
return (GKI_NOT_INITIALIZED);
|
|
|
|
// Allocate a RasMessage structure and initialized to 0
|
|
m_usRetryCount = 0;
|
|
m_uRetryCountdown = m_uRetryResetCount;
|
|
|
|
m_pRasMessage = new RasMessage;
|
|
SPIDER_TRACE(SP_NEWDEL, "new m_pRasMessage = %X\n", m_pRasMessage);
|
|
if (m_pRasMessage == 0)
|
|
return (GKI_NO_MEMORY);
|
|
memset(m_pRasMessage, 0, sizeof(RasMessage));
|
|
|
|
// Setup structure fields for GatekeeperRequest
|
|
m_pRasMessage->choice = gatekeeperRequest_chosen;
|
|
if (m_pRgstrtnRqst_trmnlAls != 0)
|
|
m_pRasMessage->u.gatekeeperRequest.bit_mask |= GtkprRqst_endpointAlias_present;
|
|
|
|
m_pRasMessage->u.gatekeeperRequest.requestSeqNum = ++m_requestSeqNum;
|
|
|
|
m_pRasMessage->u.gatekeeperRequest.rasAddress = m_pRASAddress->value;
|
|
m_pRasMessage->u.gatekeeperRequest.endpointType = m_terminalType;
|
|
m_pRasMessage->u.gatekeeperRequest.GtkprRqst_endpointAlias = (PGatekeeperRequest_endpointAlias)m_pRgstrtnRqst_trmnlAls;
|
|
|
|
// Assign ProtocolIdentifier
|
|
g_pCoder->SetProtocolIdentifier(*m_pRasMessage);
|
|
|
|
// Encode the PDU & send it
|
|
dwErrorCode = g_pCoder->Encode(m_pRasMessage, &Asn1Buf);
|
|
if (dwErrorCode)
|
|
return (GKI_ENCODER_ERROR);
|
|
|
|
// Create a backup copy of the encoded PDU if using debug echo support
|
|
if (fGKIEcho)
|
|
{
|
|
pEchoBuff = new char[Asn1Buf.length];
|
|
SPIDER_TRACE(SP_NEWDEL, "new pEchoBuff = %X\n", pEchoBuff);
|
|
if (pEchoBuff == 0)
|
|
return (GKI_NO_MEMORY);
|
|
memcpy(pEchoBuff, (char *)Asn1Buf.value, Asn1Buf.length);
|
|
nEchoLen = Asn1Buf.length;
|
|
}
|
|
|
|
SPIDER_TRACE(SP_PDU, "Send GRQ; g_pReg = %X\n", this);
|
|
if (fGKIDontSend == FALSE)
|
|
{
|
|
if (m_pSocket->SendBroadcast((char *)Asn1Buf.value, Asn1Buf.length) == SOCKET_ERROR)
|
|
{
|
|
g_pCoder->Free(Asn1Buf);
|
|
return (GKI_WINSOCK2_ERROR(SOCKET_ERROR));
|
|
}
|
|
}
|
|
|
|
// Free the encoder memory
|
|
g_pCoder->Free(Asn1Buf);
|
|
|
|
return (GKI_OK);
|
|
}
|
|
|
|
HRESULT
|
|
CRegistration::PDUHandler(RasMessage *pRasMessage)
|
|
{
|
|
// ABSTRACT: This procedure will interpret the received PDU and dispatch
|
|
// to the appropriate handler.
|
|
// AUTHOR: Colin Hulme
|
|
|
|
#ifdef _DEBUG
|
|
char szGKDebug[80];
|
|
#endif
|
|
HRESULT hResult = GKI_OK;
|
|
|
|
SPIDER_TRACE(SP_FUNC, "CRegistration::PDUHandler(%X)\n", pRasMessage);
|
|
ASSERT(g_pCoder);
|
|
if (g_pCoder == NULL)
|
|
return (GKI_NOT_INITIALIZED);
|
|
|
|
switch (pRasMessage->choice)
|
|
{
|
|
// Incoming response PDUs
|
|
case gatekeeperConfirm_chosen:
|
|
SPIDER_TRACE(SP_PDU, "Rcv GCF; g_pReg = %X\n", this);
|
|
break;
|
|
case gatekeeperReject_chosen:
|
|
SPIDER_TRACE(SP_PDU, "Rcv GRJ; g_pReg = %X\n", this);
|
|
break;
|
|
case registrationConfirm_chosen:
|
|
SPIDER_TRACE(SP_PDU, "Rcv RCF; g_pReg = %X\n", this);
|
|
if ((m_State == GK_REG_PENDING) &&
|
|
(pRasMessage->u.registrationConfirm.requestSeqNum ==
|
|
m_pRasMessage->u.registrationRequest.requestSeqNum))
|
|
hResult = RegistrationConfirm(pRasMessage);
|
|
else
|
|
hResult = UnknownMessage(pRasMessage);
|
|
break;
|
|
case registrationReject_chosen:
|
|
SPIDER_TRACE(SP_PDU, "Rcv RRJ; g_pReg = %X\n", this);
|
|
if ((m_State == GK_REG_PENDING) &&
|
|
(pRasMessage->u.registrationReject.requestSeqNum ==
|
|
m_pRasMessage->u.registrationRequest.requestSeqNum))
|
|
hResult = RegistrationReject(pRasMessage);
|
|
else
|
|
hResult = UnknownMessage(pRasMessage);
|
|
break;
|
|
case unregistrationConfirm_chosen:
|
|
SPIDER_TRACE(SP_PDU, "Rcv UCF; g_pReg = %X\n", this);
|
|
if ((m_State == GK_UNREG_PENDING) &&
|
|
(pRasMessage->u.unregistrationConfirm.requestSeqNum ==
|
|
m_pRasMessage->u.unregistrationRequest.requestSeqNum))
|
|
hResult = UnregistrationConfirm(pRasMessage);
|
|
else
|
|
hResult = UnknownMessage(pRasMessage);
|
|
break;
|
|
case unregistrationReject_chosen:
|
|
SPIDER_TRACE(SP_PDU, "Rcv URJ; g_pReg = %X\n", this);
|
|
if ((m_State == GK_UNREG_PENDING) &&
|
|
(pRasMessage->u.unregistrationReject.requestSeqNum ==
|
|
m_pRasMessage->u.unregistrationRequest.requestSeqNum))
|
|
hResult = UnregistrationReject(pRasMessage);
|
|
else
|
|
hResult = UnknownMessage(pRasMessage);
|
|
break;
|
|
case admissionConfirm_chosen:
|
|
{
|
|
// The sequence number of this RAS message seems to be the
|
|
// only thing we can link back to the ARQ so we use it
|
|
// to look up the call that this ACF is associated with
|
|
RequestSeqNum seqNum = pRasMessage->u.admissionConfirm.requestSeqNum;
|
|
CCall *pCall = FindCallBySeqNum(seqNum);
|
|
if ((m_State == GK_REGISTERED) && (pCall))
|
|
{
|
|
SPIDER_TRACE(SP_PDU, "Rcv ACF; pCall = %X\n", pCall);
|
|
hResult = pCall->AdmissionConfirm(pRasMessage);
|
|
}
|
|
else
|
|
{
|
|
SPIDER_TRACE(SP_PDU, "Rcv ACF; g_pReg = %X\n", this);
|
|
hResult = UnknownMessage(pRasMessage);
|
|
}
|
|
}
|
|
break;
|
|
case admissionReject_chosen:
|
|
{
|
|
// The sequence number of this RAS message seems to be the
|
|
// only thing we can link back to the ARQ so we use it
|
|
// to look up the call that this ARJ is associated with
|
|
RequestSeqNum seqNum = pRasMessage->u.admissionReject.requestSeqNum;
|
|
CCall *pCall = FindCallBySeqNum(seqNum);
|
|
if ((m_State == GK_REGISTERED) && (pCall))
|
|
{
|
|
SPIDER_TRACE(SP_PDU, "Rcv ARJ; pCall = %X\n", pCall);
|
|
hResult = pCall->AdmissionReject(pRasMessage);
|
|
if (hResult == GKI_DELETE_CALL)
|
|
{
|
|
DeleteCall(pCall);
|
|
hResult = GKI_OK; // Don't want to exit PostReceive loop
|
|
}
|
|
}
|
|
else
|
|
{
|
|
SPIDER_TRACE(SP_PDU, "Rcv ARJ; g_pReg = %X\n", this);
|
|
hResult = UnknownMessage(pRasMessage);
|
|
}
|
|
}
|
|
break;
|
|
case bandwidthConfirm_chosen:
|
|
{
|
|
// The sequence number of this RAS message seems to be the
|
|
// only thing we can link back to the BRQ so we use it
|
|
// to look up the call that this BCF is associated with
|
|
RequestSeqNum seqNum = pRasMessage->u.bandwidthConfirm.requestSeqNum;
|
|
CCall *pCall = FindCallBySeqNum(seqNum);
|
|
if ((m_State == GK_REGISTERED) && (pCall))
|
|
{
|
|
SPIDER_TRACE(SP_PDU, "Rcv BCF; pCall = %X\n", pCall);
|
|
hResult = pCall->BandwidthConfirm(pRasMessage);
|
|
}
|
|
else
|
|
{
|
|
SPIDER_TRACE(SP_PDU, "Rcv BCF; g_pReg = %X\n", this);
|
|
hResult = UnknownMessage(pRasMessage);
|
|
}
|
|
}
|
|
break;
|
|
case bandwidthReject_chosen:
|
|
{
|
|
// The sequence number of this RAS message seems to be the
|
|
// only thing we can link back to the BRQ so we use it
|
|
// to look up the call that this BCF is associated with
|
|
RequestSeqNum seqNum = pRasMessage->u.bandwidthReject.requestSeqNum;
|
|
CCall *pCall = FindCallBySeqNum(seqNum);
|
|
if ((m_State == GK_REGISTERED) && (pCall))
|
|
{
|
|
SPIDER_TRACE(SP_PDU, "Rcv BRJ; pCall = %X\n", pCall);
|
|
hResult = pCall->BandwidthReject(pRasMessage);
|
|
}
|
|
else
|
|
{
|
|
SPIDER_TRACE(SP_PDU, "Rcv BRJ; g_pReg = %X\n", this);
|
|
hResult = UnknownMessage(pRasMessage);
|
|
}
|
|
}
|
|
break;
|
|
case disengageConfirm_chosen:
|
|
{
|
|
// The sequence number of this RAS message seems to be the
|
|
// only thing we can link back to the DRQ so we use it
|
|
// to look up the call that this DCF is associated with
|
|
RequestSeqNum seqNum = pRasMessage->u.disengageConfirm.requestSeqNum;
|
|
CCall *pCall = FindCallBySeqNum(seqNum);
|
|
if ((m_State == GK_REGISTERED) && (pCall))
|
|
{
|
|
SPIDER_TRACE(SP_PDU, "Rcv DCF; pCall = %X\n", pCall);
|
|
hResult = pCall->DisengageConfirm(pRasMessage);
|
|
if (hResult == GKI_DELETE_CALL)
|
|
{
|
|
DeleteCall(pCall);
|
|
hResult = GKI_OK; // Don't want to exit PostReceive loop
|
|
}
|
|
}
|
|
else
|
|
{
|
|
SPIDER_TRACE(SP_PDU, "Rcv DCF; g_pReg = %X\n", this);
|
|
hResult = UnknownMessage(pRasMessage);
|
|
}
|
|
}
|
|
break;
|
|
case disengageReject_chosen:
|
|
{
|
|
// The sequence number of this RAS message seems to be the
|
|
// only thing we can link back to the DRQ so we use it
|
|
// to look up the call that this DRJ is associated with
|
|
RequestSeqNum seqNum = pRasMessage->u.disengageReject.requestSeqNum;
|
|
CCall *pCall = FindCallBySeqNum(seqNum);
|
|
if ((m_State == GK_REGISTERED) && (pCall))
|
|
{
|
|
SPIDER_TRACE(SP_PDU, "Rcv DRJ; pCall = %X\n", pCall);
|
|
hResult = pCall->DisengageReject(pRasMessage);
|
|
if (hResult == GKI_DELETE_CALL)
|
|
{
|
|
DeleteCall(pCall);
|
|
hResult = GKI_OK; // Don't want to exit PostReceive loop
|
|
}
|
|
}
|
|
else
|
|
{
|
|
SPIDER_TRACE(SP_PDU, "Rcv DRJ; g_pReg = %X\n", this);
|
|
hResult = UnknownMessage(pRasMessage);
|
|
}
|
|
}
|
|
break;
|
|
case locationConfirm_chosen:
|
|
SPIDER_TRACE(SP_PDU, "Rcv LCF; g_pReg = %X\n", this);
|
|
if ((m_State == GK_LOC_PENDING) &&
|
|
(pRasMessage->u.locationConfirm.requestSeqNum ==
|
|
m_pRasMessage->u.locationRequest.requestSeqNum))
|
|
hResult = LocationConfirm(pRasMessage);
|
|
else
|
|
hResult = UnknownMessage(pRasMessage);
|
|
break;
|
|
case locationReject_chosen:
|
|
SPIDER_TRACE(SP_PDU, "Rcv LRJ; g_pReg = %X\n", this);
|
|
if ((m_State == GK_LOC_PENDING) &&
|
|
(pRasMessage->u.locationReject.requestSeqNum ==
|
|
m_pRasMessage->u.locationRequest.requestSeqNum))
|
|
hResult = LocationReject(pRasMessage);
|
|
else
|
|
hResult = UnknownMessage(pRasMessage);
|
|
break;
|
|
|
|
case nonStandardMessage_chosen:
|
|
SPIDER_TRACE(SP_PDU, "Rcv NSM; g_pReg = %X\n", this);
|
|
hResult = UnknownMessage(pRasMessage);
|
|
break;
|
|
case unknownMessageResponse_chosen:
|
|
SPIDER_TRACE(SP_PDU, "Rcv XRS; g_pReg = %X\n", this);
|
|
break;
|
|
|
|
// Incoming Request PDUs
|
|
case unregistrationRequest_chosen:
|
|
SPIDER_TRACE(SP_PDU, "Rcv URQ; g_pReg = %X\n", this);
|
|
if (m_State == GK_REGISTERED)
|
|
{
|
|
WORD wReason;
|
|
// Notify user of received unregistration request and reason
|
|
if(pRasMessage->u.unregistrationRequest.bit_mask
|
|
& UnregistrationRequest_reason_present)
|
|
{
|
|
wReason = pRasMessage->u.unregistrationRequest.reason.choice;
|
|
}
|
|
else
|
|
wReason = 0;
|
|
SPIDER_TRACE(SP_GKI, "PostMessage(m_hWnd, m_wBaseMessage + GKI_UNREG_REQUEST, 0, 0)\n", 0);
|
|
PostMessage(m_hWnd, m_wBaseMessage + GKI_UNREG_REQUEST, wReason, 0);
|
|
|
|
hResult = SendUnregistrationConfirm(pRasMessage);
|
|
if (hResult == GKI_OK)
|
|
hResult = GKI_EXIT_THREAD;
|
|
}
|
|
else
|
|
hResult = UnknownMessage(pRasMessage);
|
|
break;
|
|
case bandwidthRequest_chosen:
|
|
{
|
|
// Check the CRV in this BRQ and see if we have a call
|
|
// that corresponds to it.
|
|
CallReferenceValue crv = pRasMessage->u.bandwidthRequest.callReferenceValue;
|
|
CCall *pCall = FindCallByCRV(crv);
|
|
if ((m_State == GK_REGISTERED) && (pCall))
|
|
{
|
|
SPIDER_TRACE(SP_PDU, "Rcv BRQ; pCall = %X\n", pCall);
|
|
hResult = pCall->SendBandwidthConfirm(pRasMessage);
|
|
}
|
|
else
|
|
{
|
|
SPIDER_TRACE(SP_PDU, "Rcv BRQ; g_pReg = %X\n", this);
|
|
hResult = UnknownMessage(pRasMessage);
|
|
}
|
|
}
|
|
break;
|
|
case disengageRequest_chosen:
|
|
{
|
|
// Check the CRV in this DRQ and see if we have a call
|
|
// that corresponds to it.
|
|
CallReferenceValue crv = pRasMessage->u.disengageRequest.callReferenceValue;
|
|
CCall *pCall = FindCallByCRV(crv);
|
|
if ((m_State == GK_REGISTERED) && (pCall))
|
|
{
|
|
SPIDER_TRACE(SP_PDU, "Rcv DRQ; pCall = %X\n", pCall);
|
|
hResult = pCall->SendDisengageConfirm(pRasMessage);
|
|
if (hResult == GKI_DELETE_CALL)
|
|
{
|
|
DeleteCall(pCall);
|
|
hResult = GKI_OK; // Don't want to exit PostReceive loop
|
|
}
|
|
}
|
|
else
|
|
{
|
|
SPIDER_TRACE(SP_PDU, "Rcv DRQ; g_pReg = %X\n", this);
|
|
hResult = UnknownMessage(pRasMessage);
|
|
}
|
|
}
|
|
break;
|
|
case infoRequest_chosen:
|
|
SPIDER_TRACE(SP_PDU, "Rcv IRQ; g_pReg = %X\n", this);
|
|
if ((m_State != GK_UNREGISTERED) && (m_State != GK_REG_PENDING))
|
|
{
|
|
// Check the CRV in this DRQ and see if we have a call
|
|
// that corresponds to it.
|
|
CallReferenceValue crv = pRasMessage->u.infoRequest.callReferenceValue;
|
|
CCall *pCall = NULL;
|
|
// A zero in the CRV means provide info for all calls, so we start
|
|
// the chain with the first one.
|
|
if (crv == 0)
|
|
{
|
|
if (m_Calls.IsEmpty())
|
|
hResult = SendInfoRequestResponse(0, pRasMessage);
|
|
else
|
|
{
|
|
POS pos = m_Calls.GetFirstPos();
|
|
pCall = m_Calls.GetAt(pos);
|
|
hResult = pCall->SendInfoRequestResponse(0, pRasMessage, FALSE);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// This is a call specific request so if we don't find
|
|
// a matching call, we'll send an XRS
|
|
pCall = FindCallByCRV(crv);
|
|
if (pCall)
|
|
hResult = pCall->SendInfoRequestResponse(0, pRasMessage, TRUE);
|
|
else
|
|
hResult = UnknownMessage(pRasMessage);
|
|
}
|
|
}
|
|
break;
|
|
|
|
// Should never see these PDUs
|
|
case gatekeeperRequest_chosen:
|
|
case registrationRequest_chosen:
|
|
case admissionRequest_chosen:
|
|
case locationRequest_chosen:
|
|
case infoRequestResponse_chosen:
|
|
SPIDER_TRACE(SP_PDU, "Rcv unexpected PDU; g_pReg = %X\n", this);
|
|
SPIDER_TRACE(SP_PDU, "pRasMessage->choice = %X\n", pRasMessage->choice);
|
|
hResult = UnknownMessage(pRasMessage);
|
|
break;
|
|
|
|
// Everything else - probably a bad PDU
|
|
default:
|
|
SPIDER_TRACE(SP_PDU, "Rcv unrecognized PDU; g_pReg = %X\n", this);
|
|
SPIDER_TRACE(SP_PDU, "pRasMessage->choice = %X\n", pRasMessage->choice);
|
|
hResult = UnknownMessage(pRasMessage);
|
|
break;
|
|
}
|
|
|
|
return (hResult);
|
|
}
|
|
|
|
HRESULT
|
|
CRegistration::RegistrationConfirm(RasMessage *pRasMessage)
|
|
{
|
|
// ABSTRACT: This function is called if a registrationConfirm is
|
|
// received and matches an outstanding registrationRequest.
|
|
// It will delete the memory used for the registrationRequest
|
|
// change the state and notify the user by posting a message.
|
|
// Additional information contained in the registrationConfirm
|
|
// is stored in the CRegistration class.
|
|
// AUTHOR: Colin Hulme
|
|
|
|
SeqAliasAddr *pAA;
|
|
#ifdef _DEBUG
|
|
char szGKDebug[80];
|
|
#endif
|
|
|
|
SPIDER_TRACE(SP_FUNC, "CRegistration::RegistrationConfirm(%X)\n", pRasMessage);
|
|
ASSERT(g_pCoder);
|
|
if (g_pCoder == NULL)
|
|
return (GKI_NOT_INITIALIZED);
|
|
|
|
// Delete allocated RasMessage storage
|
|
SPIDER_TRACE(SP_NEWDEL, "del m_pRasMessage = %X\n", m_pRasMessage);
|
|
delete m_pRasMessage;
|
|
m_pRasMessage = 0;
|
|
|
|
// Update member variables
|
|
m_State = GK_REGISTERED;
|
|
SPIDER_TRACE(SP_STATE, "m_State = GK_REGISTERED (%X)\n", this);
|
|
|
|
if (pRasMessage->u.registrationConfirm.bit_mask & RgstrtnCnfrm_trmnlAls_present)
|
|
{
|
|
// Copy alias addresses
|
|
for (pAA = (SeqAliasAddr *)pRasMessage->u.registrationConfirm.RgstrtnCnfrm_trmnlAls;
|
|
pAA != 0; pAA = pAA->next)
|
|
AddAliasAddr(pAA->value);
|
|
}
|
|
if ((pRasMessage->u.registrationConfirm.bit_mask & RCm_gtkprIdntfr_present) &&
|
|
(m_RCm_gtkprIdntfr.value == 0))
|
|
{
|
|
// Copy gatekeeper identifier
|
|
m_RCm_gtkprIdntfr.length = pRasMessage->u.registrationConfirm.RCm_gtkprIdntfr.length;
|
|
m_RCm_gtkprIdntfr.value = new unsigned short[m_RCm_gtkprIdntfr.length];
|
|
SPIDER_TRACE(SP_NEWDEL, "new m_RCm_gtkprIdntfr.value = %X\n", m_RCm_gtkprIdntfr.value);
|
|
if (m_RCm_gtkprIdntfr.value == 0)
|
|
return (GKI_NO_MEMORY);
|
|
memcpy(m_RCm_gtkprIdntfr.value,
|
|
pRasMessage->u.registrationConfirm.RCm_gtkprIdntfr.value,
|
|
m_RCm_gtkprIdntfr.length * sizeof(unsigned short));
|
|
|
|
}
|
|
// Copy endpoint identifier
|
|
m_endpointID.length = pRasMessage->u.registrationConfirm.endpointIdentifier.length;
|
|
m_endpointID.value = new unsigned short[m_endpointID.length];
|
|
SPIDER_TRACE(SP_NEWDEL, "new m_endpointID.value = %X\n", m_endpointID.value);
|
|
if (m_endpointID.value == 0)
|
|
return (GKI_NO_MEMORY);
|
|
memcpy(m_endpointID.value,
|
|
pRasMessage->u.registrationConfirm.endpointIdentifier.value,
|
|
m_endpointID.length * sizeof(unsigned short));
|
|
|
|
|
|
SPIDER_TRACE(SP_GKI, "PostMessage(m_hWnd, m_wBaseMessage + GKI_REG_CONFIRM, 0, 0)\n", 0);
|
|
PostMessage(m_hWnd, m_wBaseMessage + GKI_REG_CONFIRM, 0, 0);
|
|
|
|
return (GKI_OK);
|
|
}
|
|
|
|
HRESULT
|
|
CRegistration::RegistrationReject(RasMessage *pRasMessage)
|
|
{
|
|
// ABSTRACT: This function is called if a registrationReject is
|
|
// received and matches an outstanding registrationRequest.
|
|
// It will delete the memory used for the registrationRequest
|
|
// change the state and notify the user by posting a message
|
|
// Returning a non-zero value, indicates that the PostReceive
|
|
// loop should terminate, delete the registration object
|
|
// and exit the thread. If the rejectReason is discovery
|
|
// required, this function execs the discovery thread and
|
|
// notifies PostReceive to exit the thread without deleting
|
|
// the registration object and socket.
|
|
// AUTHOR: Colin Hulme
|
|
|
|
HANDLE hThread;
|
|
SeqTransportAddr *pTA1, *pTA2;
|
|
int nRet;
|
|
HRESULT hResult;
|
|
#ifdef _DEBUG
|
|
char szGKDebug[80];
|
|
#endif
|
|
|
|
SPIDER_TRACE(SP_FUNC, "CRegistration::RegistrationReject(%X)\n", pRasMessage);
|
|
ASSERT(g_pCoder);
|
|
if (g_pCoder == NULL)
|
|
return (GKI_NOT_INITIALIZED);
|
|
|
|
#ifdef BROADCAST_DISCOVERY
|
|
if (pRasMessage->u.registrationReject.rejectReason.choice == discoveryRequired_chosen)
|
|
{
|
|
SPIDER_TRACE(SP_NEWDEL, "del m_pRasMessage = %X\n", m_pRasMessage);
|
|
delete m_pRasMessage;
|
|
m_pRasMessage = 0;
|
|
|
|
// Close socket and reopen in non-connected state to allow sendto
|
|
if ((nRet = m_pSocket->Close()) != 0)
|
|
return (GKI_WINSOCK2_ERROR(nRet));
|
|
if ((nRet = m_pSocket->Create(m_pSocket->GetAddrFam(), 0)) != 0)
|
|
return (GKI_WINSOCK2_ERROR(nRet));
|
|
|
|
// Delete allocated memory for sequence of RAS addresses
|
|
pTA1 = m_pRASAddress;
|
|
while (pTA1 != 0)
|
|
{
|
|
pTA2 = pTA1->next;
|
|
SPIDER_TRACE(SP_NEWDEL, "del pTA1 = %X\n", pTA1);
|
|
delete pTA1;
|
|
pTA1 = pTA2;
|
|
}
|
|
|
|
// Update RAS Address in CRegistration
|
|
for (pTA1 = m_pCallSignalAddress; pTA1 != 0; pTA1 = pTA1->next)
|
|
{
|
|
if (pTA1->value.choice == m_usRegistrationTransport)
|
|
if ((hResult = AddRASAddr(pTA1->value, m_pSocket->GetPort())) != GKI_OK)
|
|
return (hResult);
|
|
}
|
|
|
|
hThread = (HANDLE)_beginthread(GKDiscovery, 0, 0);
|
|
SPIDER_TRACE(SP_THREAD, "_beginthread(GKDiscovery, 0, 0); <%X>\n", hThread);
|
|
if (hThread == (HANDLE)-1)
|
|
return (GKI_NO_THREAD);
|
|
SetDiscThread(hThread);
|
|
return (GKI_REDISCOVER);
|
|
}
|
|
#endif // BROADCAST_DISCOVERY
|
|
|
|
m_State = GK_UNREGISTERED;
|
|
SPIDER_TRACE(SP_STATE, "m_State = GK_UNREGISTERED (%X)\n", this);
|
|
|
|
SPIDER_TRACE(SP_GKI, "PostMessage(m_hWnd, m_wBaseMessage + GKI_REG_REJECT, %X, 0)\n",
|
|
pRasMessage->u.registrationReject.rejectReason.choice);
|
|
PostMessage(m_hWnd, m_wBaseMessage + GKI_REG_REJECT,
|
|
(WORD)pRasMessage->u.registrationReject.rejectReason.choice, 0L);
|
|
|
|
return (GKI_EXIT_THREAD);
|
|
}
|
|
|
|
HRESULT
|
|
CRegistration::UnregistrationConfirm(RasMessage *pRasMessage)
|
|
{
|
|
// ABSTRACT: This function is called if an unregistrationConfirm is
|
|
// received and matches an outstanding unregistrationRequest.
|
|
// It will delete the memory used for the unregistrationRequest
|
|
// change the state and notify the user by posting a message.
|
|
// Returning a non-zero value, indicates that the PostReceive
|
|
// loop should terminate, delete the registration object
|
|
// and exit the thread.
|
|
// AUTHOR: Colin Hulme
|
|
|
|
#ifdef _DEBUG
|
|
char szGKDebug[80];
|
|
#endif
|
|
|
|
SPIDER_TRACE(SP_FUNC, "CRegistration::UnregistrationConfirm(%X)\n", pRasMessage);
|
|
|
|
// We deliberately don't free the RasMessage memory. Let the registration
|
|
// destructor do it - this provides protection from other requests.
|
|
|
|
// Update member variables
|
|
m_State = GK_UNREGISTERED;
|
|
SPIDER_TRACE(SP_STATE, "m_State = GK_UNREGISTERED (%X)\n", this);
|
|
|
|
// Notify user application
|
|
SPIDER_TRACE(SP_GKI, "PostMessage(m_hWnd, m_wBaseMessage + GKI_UNREG_CONFIRM, 0, 0)\n", 0);
|
|
PostMessage(m_hWnd, m_wBaseMessage + GKI_UNREG_CONFIRM, 0, 0L);
|
|
|
|
return (GKI_EXIT_THREAD);
|
|
}
|
|
|
|
HRESULT
|
|
CRegistration::UnregistrationReject(RasMessage *pRasMessage)
|
|
{
|
|
// ABSTRACT: This function is called if an unregistrationReject is
|
|
// received and matches an outstanding unregistrationRequest.
|
|
// It will delete the memory used for the unregistrationRequest
|
|
// change the state and notify the user by posting a message
|
|
// Returning a non-zero value, indicates that the PostReceive
|
|
// loop should terminate, delete the registration object
|
|
// and exit the thread.
|
|
// AUTHOR: Colin Hulme
|
|
|
|
#ifdef _DEBUG
|
|
char szGKDebug[80];
|
|
#endif
|
|
HRESULT hResult = GKI_OK;
|
|
|
|
SPIDER_TRACE(SP_FUNC, "CRegistration::UnregistrationReject(%X)\n", pRasMessage);
|
|
|
|
|
|
// Update member variables
|
|
switch (pRasMessage->u.unregistrationReject.rejectReason.choice)
|
|
{
|
|
case callInProgress_chosen: // return to registered state
|
|
// Delete allocate RasMessage storage
|
|
SPIDER_TRACE(SP_NEWDEL, "del m_pRasMessage = %X\n", m_pRasMessage);
|
|
delete m_pRasMessage;
|
|
m_pRasMessage = 0;
|
|
|
|
m_State = GK_REGISTERED;
|
|
SPIDER_TRACE(SP_STATE, "m_State = GK_REGISTERED (%X)\n", this);
|
|
break;
|
|
case notCurrentlyRegistered_chosen:
|
|
default:
|
|
m_State = GK_UNREGISTERED;
|
|
SPIDER_TRACE(SP_STATE, "m_State = GK_UNREGISTERED (%X)\n", this);
|
|
hResult = GKI_EXIT_THREAD; // kill registration and PostReceive thread
|
|
break;
|
|
}
|
|
|
|
// Notify user application
|
|
SPIDER_TRACE(SP_GKI, "PostMessage(m_hWnd, m_wBaseMessage + GKI_UNREG_REJECT, %X, 0)\n",
|
|
pRasMessage->u.unregistrationReject.rejectReason.choice);
|
|
PostMessage(m_hWnd, m_wBaseMessage + GKI_UNREG_REJECT,
|
|
(WORD)pRasMessage->u.unregistrationReject.rejectReason.choice, 0L);
|
|
|
|
return (hResult);
|
|
}
|
|
|
|
HRESULT
|
|
CRegistration::LocationConfirm(RasMessage *pRasMessage)
|
|
{
|
|
// ABSTRACT: This function is called if a locationConfirm is
|
|
// received and matches an outstanding locationRequest.
|
|
// It will delete the memory used for the locationRequest
|
|
// change the state and notify the user by posting a message.
|
|
// AUTHOR: Colin Hulme
|
|
|
|
SeqAliasAddr *pAA1, *pAA2;
|
|
#ifdef _DEBUG
|
|
char szGKDebug[80];
|
|
#endif
|
|
|
|
SPIDER_TRACE(SP_FUNC, "CRegistration::LocationConfirm(%X)\n", pRasMessage);
|
|
|
|
// Delete allocated RasMessage storage
|
|
SPIDER_TRACE(SP_NEWDEL, "del m_pRasMessage = %X\n", m_pRasMessage);
|
|
delete m_pRasMessage;
|
|
m_pRasMessage = 0;
|
|
|
|
// Delete allocated memory for sequence of location alias addresses
|
|
pAA1 = m_pLocationInfo;
|
|
while (pAA1 != 0)
|
|
{
|
|
pAA2 = pAA1->next;
|
|
if (pAA1->value.choice == h323_ID_chosen)
|
|
{
|
|
SPIDER_TRACE(SP_NEWDEL, "del pAA1->value.u.h323_ID.value = %X\n", pAA1->value.u.h323_ID.value);
|
|
delete pAA1->value.u.h323_ID.value;
|
|
}
|
|
SPIDER_TRACE(SP_NEWDEL, "del pAA1 = %X\n", pAA1);
|
|
delete pAA1;
|
|
pAA1 = pAA2;
|
|
}
|
|
m_pLocationInfo = 0;
|
|
|
|
// Update member variables
|
|
m_State = GK_REGISTERED;
|
|
SPIDER_TRACE(SP_STATE, "m_State = GK_REGISTERED (%X)\n", this);
|
|
m_Location[0] = pRasMessage->u.locationConfirm.callSignalAddress;
|
|
m_Location[1] = pRasMessage->u.locationConfirm.rasAddress;
|
|
|
|
// Notify user application
|
|
SPIDER_TRACE(SP_GKI, "PostMessage(m_hWnd, m_wBaseMessage + GKI_LOCATION_CONFIRM, 0, &m_Location[0])\n", 0);
|
|
PostMessage(m_hWnd, m_wBaseMessage + GKI_LOCATION_CONFIRM,
|
|
0, (LPARAM)&m_Location[0]);
|
|
|
|
return (GKI_OK);
|
|
}
|
|
|
|
HRESULT
|
|
CRegistration::LocationReject(RasMessage *pRasMessage)
|
|
{
|
|
// ABSTRACT: This function is called if a locationReject is
|
|
// received and matches an outstanding locationRequest.
|
|
// It will delete the memory used for the locationRequest
|
|
// change the state and notify the user by posting a message
|
|
// AUTHOR: Colin Hulme
|
|
|
|
SeqAliasAddr *pAA1, *pAA2;
|
|
#ifdef _DEBUG
|
|
char szGKDebug[80];
|
|
#endif
|
|
|
|
SPIDER_TRACE(SP_FUNC, "CRegistration::LocationReject(%X)\n", pRasMessage);
|
|
|
|
// Delete allocate RasMessage storage
|
|
SPIDER_TRACE(SP_NEWDEL, "del m_pRasMessage = %X\n", m_pRasMessage);
|
|
delete m_pRasMessage;
|
|
m_pRasMessage = 0;
|
|
|
|
// Delete allocated memory for sequence of location alias addresses
|
|
pAA1 = m_pLocationInfo;
|
|
while (pAA1 != 0)
|
|
{
|
|
pAA2 = pAA1->next;
|
|
if (pAA1->value.choice == h323_ID_chosen)
|
|
{
|
|
SPIDER_TRACE(SP_NEWDEL, "del pAA1->value.u.h323_ID.value = %X\n", pAA1->value.u.h323_ID.value);
|
|
delete pAA1->value.u.h323_ID.value;
|
|
}
|
|
SPIDER_TRACE(SP_NEWDEL, "del pAA1 = %X\n", pAA1);
|
|
delete pAA1;
|
|
pAA1 = pAA2;
|
|
}
|
|
m_pLocationInfo = 0;
|
|
|
|
// Update member variables
|
|
m_State = GK_REGISTERED;
|
|
SPIDER_TRACE(SP_STATE, "m_State = GK_REGISTERED (%X)\n", this);
|
|
|
|
// Notify user application
|
|
SPIDER_TRACE(SP_GKI, "PostMessage(m_hWnd, m_wBaseMessage + GKI_LOCATION_REJECT, %X, 0)\n",
|
|
pRasMessage->u.locationReject.rejectReason.choice);
|
|
PostMessage(m_hWnd, m_wBaseMessage + GKI_LOCATION_REJECT,
|
|
(WORD)pRasMessage->u.locationReject.rejectReason.choice, 0L);
|
|
|
|
return (GKI_OK);
|
|
}
|
|
|
|
HRESULT
|
|
CRegistration::UnknownMessage(RasMessage *pRasMessage)
|
|
{
|
|
// ABSTRACT: This member function is called to respond to the gatekeeper
|
|
// with an XRS PDU indicated that the received PDU is an unknown
|
|
// message
|
|
// AUTHOR: Colin Hulme
|
|
|
|
ASN1_BUF Asn1Buf;
|
|
DWORD dwErrorCode;
|
|
RasMessage *pOutRasMessage;
|
|
#ifdef _DEBUG
|
|
char szGKDebug[80];
|
|
#endif
|
|
|
|
SPIDER_TRACE(SP_FUNC, "CRegistration::UnknownMessage(%X)\n", pRasMessage);
|
|
ASSERT(g_pCoder);
|
|
if (g_pCoder == NULL)
|
|
return (GKI_NOT_INITIALIZED);
|
|
|
|
// Allocate a RasMessage structure and initialized to 0
|
|
pOutRasMessage = new RasMessage;
|
|
SPIDER_TRACE(SP_NEWDEL, "new pOutRasMessage = %X\n", pOutRasMessage);
|
|
if (pOutRasMessage == 0)
|
|
return (GKI_NO_MEMORY);
|
|
memset(pOutRasMessage, 0, sizeof(RasMessage));
|
|
|
|
// Setup structure fields for UnregistrationRequest
|
|
pOutRasMessage->choice = unknownMessageResponse_chosen;
|
|
pOutRasMessage->u.unknownMessageResponse.requestSeqNum =
|
|
pRasMessage->u.registrationRequest.requestSeqNum; // can use from
|
|
// from any RAS Message, since SeqNum
|
|
// is always in same position.
|
|
|
|
// Encode the PDU & send it
|
|
dwErrorCode = g_pCoder->Encode(pOutRasMessage, &Asn1Buf);
|
|
if (dwErrorCode)
|
|
return (GKI_ENCODER_ERROR);
|
|
|
|
SPIDER_TRACE(SP_PDU, "Send XRS; g_pReg = %X\n", this);
|
|
|
|
if (fGKIDontSend == FALSE)
|
|
if (m_pSocket->Send((char *)Asn1Buf.value, Asn1Buf.length) == SOCKET_ERROR)
|
|
return (GKI_WINSOCK2_ERROR(SOCKET_ERROR));
|
|
|
|
// Free the encoder memory
|
|
g_pCoder->Free(Asn1Buf);
|
|
|
|
SPIDER_TRACE(SP_NEWDEL, "del pOutRasMessage = %X\n", pOutRasMessage);
|
|
delete pOutRasMessage;
|
|
pOutRasMessage = 0;
|
|
|
|
return (GKI_OK);
|
|
}
|
|
|
|
HRESULT
|
|
CRegistration::GatekeeperConfirm(RasMessage *pRasMessage)
|
|
{
|
|
// ABSTRACT: This function is called if a gatekeeperConfirm is
|
|
// received. Note this member function must first ascertain
|
|
// that the supplied confirmation sequence number matches
|
|
// the outstanding request sequence number, if not - it
|
|
// will send an XRS response.
|
|
// AUTHOR: Colin Hulme
|
|
|
|
char szBuffer[80];
|
|
HRESULT hResult;
|
|
#ifdef _DEBUG
|
|
char szGKDebug[80];
|
|
#endif
|
|
|
|
SPIDER_TRACE(SP_FUNC, "CRegistration::GatekeeperConfirm(%X)\n", pRasMessage);
|
|
ASSERT(g_pGatekeeper);
|
|
if(g_pGatekeeper == NULL)
|
|
return (GKI_NOT_INITIALIZED);
|
|
|
|
if (m_pRasMessage == 0)
|
|
return (0);
|
|
|
|
if (pRasMessage->u.gatekeeperConfirm.requestSeqNum !=
|
|
m_pRasMessage->u.gatekeeperRequest.requestSeqNum)
|
|
{
|
|
hResult = g_pReg->UnknownMessage(pRasMessage);
|
|
return (hResult);
|
|
}
|
|
|
|
// Delete allocated RasMessage storage
|
|
SPIDER_TRACE(SP_NEWDEL, "del m_pRasMessage = %X\n", m_pRasMessage);
|
|
delete m_pRasMessage;
|
|
m_pRasMessage = 0;
|
|
|
|
// Update member variables
|
|
if ((pRasMessage->u.gatekeeperConfirm.bit_mask & GtkprCnfrm_gtkprIdntfr_present) &&
|
|
(m_RCm_gtkprIdntfr.value == 0))
|
|
{
|
|
// Copy gatekeeper identifier
|
|
m_RCm_gtkprIdntfr.length = pRasMessage->u.gatekeeperConfirm.GtkprCnfrm_gtkprIdntfr.length;
|
|
m_RCm_gtkprIdntfr.value = new unsigned short[m_RCm_gtkprIdntfr.length];
|
|
SPIDER_TRACE(SP_NEWDEL, "new m_RCm_gtkprIdntfr.value = %X\n", m_RCm_gtkprIdntfr.value);
|
|
if (m_RCm_gtkprIdntfr.value == 0)
|
|
return (GKI_NO_MEMORY);
|
|
memcpy(m_RCm_gtkprIdntfr.value,
|
|
pRasMessage->u.gatekeeperConfirm.GtkprCnfrm_gtkprIdntfr.value,
|
|
m_RCm_gtkprIdntfr.length * sizeof(unsigned short));
|
|
}
|
|
|
|
// Copy gatekeeper RAS Address
|
|
ASSERT((pRasMessage->u.gatekeeperConfirm.rasAddress.choice == ipAddress_chosen) ||
|
|
(pRasMessage->u.gatekeeperConfirm.rasAddress.choice == ipxAddress_chosen));
|
|
|
|
switch (pRasMessage->u.gatekeeperConfirm.rasAddress.choice)
|
|
{
|
|
case ipAddress_chosen:
|
|
wsprintf(szBuffer, "%d.%d.%d.%d",
|
|
pRasMessage->u.gatekeeperConfirm.rasAddress.u.ipAddress.ip.value[0],
|
|
pRasMessage->u.gatekeeperConfirm.rasAddress.u.ipAddress.ip.value[1],
|
|
pRasMessage->u.gatekeeperConfirm.rasAddress.u.ipAddress.ip.value[2],
|
|
pRasMessage->u.gatekeeperConfirm.rasAddress.u.ipAddress.ip.value[3]);
|
|
g_pGatekeeper->SetIPAddress(szBuffer);
|
|
break;
|
|
#if(0)
|
|
case ipxAddress_chosen:
|
|
wsprintf(szBuffer, "%02X%02X%02X%02X:%02X%02X%02X%02X%02X%02X",
|
|
pRasMessage->u.gatekeeperConfirm.rasAddress.u.ipxAddress.netnum.value[0],
|
|
pRasMessage->u.gatekeeperConfirm.rasAddress.u.ipxAddress.netnum.value[1],
|
|
pRasMessage->u.gatekeeperConfirm.rasAddress.u.ipxAddress.netnum.value[2],
|
|
pRasMessage->u.gatekeeperConfirm.rasAddress.u.ipxAddress.netnum.value[3],
|
|
pRasMessage->u.gatekeeperConfirm.rasAddress.u.ipxAddress.node.value[0],
|
|
pRasMessage->u.gatekeeperConfirm.rasAddress.u.ipxAddress.node.value[1],
|
|
pRasMessage->u.gatekeeperConfirm.rasAddress.u.ipxAddress.node.value[2],
|
|
pRasMessage->u.gatekeeperConfirm.rasAddress.u.ipxAddress.node.value[3],
|
|
pRasMessage->u.gatekeeperConfirm.rasAddress.u.ipxAddress.node.value[4],
|
|
pRasMessage->u.gatekeeperConfirm.rasAddress.u.ipxAddress.node.value[5]);
|
|
g_pGatekeeper->SetIPXAddress(szBuffer);
|
|
break;
|
|
#endif // if(0)
|
|
default:
|
|
break;
|
|
}
|
|
|
|
g_pGatekeeper->Write();
|
|
|
|
return (GKI_OK);
|
|
}
|
|
|
|
HRESULT
|
|
CRegistration::GatekeeperReject(RasMessage *pRasMessage)
|
|
{
|
|
// ABSTRACT: This function is called if a gatekeeperReject is
|
|
// received. Note this member function must first ascertain
|
|
// that the supplied rejection sequence number matches
|
|
// the outstanding request sequence number, if not - it
|
|
// will send an XRS response.
|
|
// AUTHOR: Colin Hulme
|
|
|
|
HRESULT hResult;
|
|
#ifdef _DEBUG
|
|
char szGKDebug[80];
|
|
#endif
|
|
|
|
SPIDER_TRACE(SP_FUNC, "CRegistration::GatekeeperReject(%X)\n", pRasMessage);
|
|
ASSERT(g_pGatekeeper);
|
|
if(g_pGatekeeper == NULL)
|
|
return (GKI_NOT_INITIALIZED);
|
|
|
|
if (m_pRasMessage == 0)
|
|
return (GKI_OK);
|
|
|
|
if (pRasMessage->u.gatekeeperReject.requestSeqNum !=
|
|
m_pRasMessage->u.gatekeeperRequest.requestSeqNum)
|
|
{
|
|
hResult = g_pReg->UnknownMessage(pRasMessage);
|
|
return (hResult);
|
|
}
|
|
|
|
g_pGatekeeper->SetRejectFlag(TRUE); // Indicate that atleast one GRJ was received
|
|
return (GKI_OK);
|
|
}
|
|
|
|
HRESULT
|
|
CRegistration::SendUnregistrationConfirm(RasMessage *pRasMessage)
|
|
{
|
|
// ABSTRACT: This function is called when an unregistrationRequest is
|
|
// received from the gatekeeper. It will create the
|
|
// unregistrationConfirm structure, encode it and send
|
|
// it on the net. It posts a message to the user
|
|
// notifying them.
|
|
// AUTHOR: Colin Hulme
|
|
|
|
ASN1_BUF Asn1Buf;
|
|
DWORD dwErrorCode;
|
|
RasMessage *pRespRasMessage;
|
|
#ifdef _DEBUG
|
|
char szGKDebug[80];
|
|
#endif
|
|
|
|
SPIDER_TRACE(SP_FUNC, "CRegistration::UnregistrationConfirm(%X)\n", pRasMessage);
|
|
ASSERT(g_pCoder);
|
|
if (g_pCoder == NULL)
|
|
return (GKI_NOT_INITIALIZED);
|
|
|
|
// Allocate a RasMessage structure and initialized to 0
|
|
pRespRasMessage = new RasMessage;
|
|
SPIDER_TRACE(SP_NEWDEL, "new pRespRasMessage = %X\n", pRespRasMessage);
|
|
if (pRespRasMessage == 0)
|
|
return (GKI_NO_MEMORY);
|
|
memset(pRespRasMessage, 0, sizeof(RasMessage));
|
|
|
|
// Setup structure fields for UnregistrationConfirm
|
|
pRespRasMessage->choice = unregistrationConfirm_chosen;
|
|
pRespRasMessage->u.unregistrationConfirm.requestSeqNum =
|
|
pRasMessage->u.unregistrationRequest.requestSeqNum;
|
|
|
|
#ifdef _DEBUG
|
|
if (dwGKIDLLFlags & SP_DUMPMEM)
|
|
DumpMem(pRespRasMessage, sizeof(RasMessage));
|
|
#endif
|
|
|
|
// Encode the PDU & send it
|
|
dwErrorCode = g_pCoder->Encode(pRespRasMessage, &Asn1Buf);
|
|
if (dwErrorCode)
|
|
return (GKI_ENCODER_ERROR);
|
|
|
|
m_State = GK_UNREGISTERED;
|
|
SPIDER_TRACE(SP_STATE, "m_State = GK_UNREGISTERED (%X)\n", this);
|
|
|
|
SPIDER_TRACE(SP_PDU, "Send UCF; g_pReg = %X\n", this);
|
|
if (fGKIDontSend == FALSE)
|
|
if (m_pSocket->Send((char *)Asn1Buf.value, Asn1Buf.length) == SOCKET_ERROR)
|
|
return (GKI_WINSOCK2_ERROR(SOCKET_ERROR));
|
|
|
|
// Free the encoder memory
|
|
g_pCoder->Free(Asn1Buf);
|
|
|
|
// Delete allocated RasMessage storage
|
|
SPIDER_TRACE(SP_NEWDEL, "del pRespRasMessage = %X\n", pRespRasMessage);
|
|
delete pRespRasMessage;
|
|
|
|
// fake "received unregistration confirm" because the upper
|
|
// state machine code depends on it
|
|
SPIDER_TRACE(SP_GKI, "PostMessage(m_hWnd, m_wBaseMessage + GKI_UNREG_CONFIRM, 0, 0)\n", 0);
|
|
PostMessage(m_hWnd, m_wBaseMessage + GKI_UNREG_CONFIRM, 0, 0);
|
|
|
|
return (GKI_OK);
|
|
}
|
|
|
|
HRESULT
|
|
CRegistration::Retry(void)
|
|
{
|
|
// ABSTRACT: This function is called by the background Retry thread
|
|
// at the configured time interval. It will check if there
|
|
// are any outstanding PDUs for the Registration object
|
|
// If so, they will be retransmitted. If the maximum number of
|
|
// retries has expired, the memory will be cleaned up.
|
|
// This function will return 0 to the background thread unless
|
|
// it wants the thread to terminate.
|
|
// AUTHOR: Colin Hulme
|
|
|
|
ASN1_BUF Asn1Buf;
|
|
DWORD dwErrorCode;
|
|
HANDLE hThread;
|
|
SeqTransportAddr *pTA1, *pTA2;
|
|
SeqAliasAddr *pAA1, *pAA2;
|
|
int nRet;
|
|
#ifdef _DEBUG
|
|
char szGKDebug[80];
|
|
#endif
|
|
HRESULT hResult = GKI_OK;
|
|
|
|
// SPIDER_TRACE(SP_FUNC, "CRegistration::Retry() %X\n", m_pCall);
|
|
ASSERT(g_pCoder);
|
|
if ((g_pCoder == NULL) && (g_pGatekeeper == NULL))
|
|
return (GKI_NOT_INITIALIZED);
|
|
|
|
// Allow calls to do retry processing
|
|
if (!m_Calls.IsEmpty())
|
|
{
|
|
// Loop through and let each call do it's retry processing
|
|
// It should be safe to call DeleteCall() from within the
|
|
// iteration since pos1 should still be valid after the
|
|
// removal.
|
|
POS pos1;
|
|
for( pos1 = m_Calls.GetFirstPos(); pos1 != NULL; )
|
|
{
|
|
// Call Retry() for this call
|
|
CCall *pCall = m_Calls.GetNext(pos1);
|
|
ASSERT (pCall);
|
|
hResult = pCall->Retry();
|
|
if (hResult == GKI_DELETE_CALL)
|
|
{
|
|
DeleteCall(pCall);
|
|
hResult = GKI_OK;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Check if any outstanding registration PDUs
|
|
if (m_pRasMessage && (--m_uRetryCountdown == 0))
|
|
{
|
|
// going to retry, reset countdown
|
|
m_uRetryCountdown = m_uRetryResetCount;
|
|
|
|
if (m_usRetryCount <= m_uMaxRetryCount)
|
|
{
|
|
// Encode the PDU & resend it
|
|
dwErrorCode = g_pCoder->Encode(m_pRasMessage, &Asn1Buf);
|
|
if (dwErrorCode)
|
|
return (GKI_ENCODER_ERROR);
|
|
|
|
SPIDER_TRACE(SP_PDU, "RESend PDU; g_pReg = %X\n", this);
|
|
if (fGKIDontSend == FALSE)
|
|
{
|
|
if (m_pRasMessage->choice == gatekeeperRequest_chosen)
|
|
{
|
|
if (m_pSocket->SendBroadcast((char *)Asn1Buf.value, Asn1Buf.length) == SOCKET_ERROR)
|
|
return (GKI_WINSOCK2_ERROR(SOCKET_ERROR));
|
|
}
|
|
else
|
|
{
|
|
if (m_pSocket->Send((char *)Asn1Buf.value, Asn1Buf.length) == SOCKET_ERROR)
|
|
return (GKI_WINSOCK2_ERROR(SOCKET_ERROR));
|
|
}
|
|
}
|
|
|
|
// Free the encoder memory
|
|
g_pCoder->Free(Asn1Buf);
|
|
m_usRetryCount++;
|
|
}
|
|
else // Retries expired - clean up
|
|
{
|
|
switch (m_pRasMessage->choice)
|
|
{
|
|
case gatekeeperRequest_chosen:
|
|
|
|
#ifdef BROADCAST_DISCOVERY
|
|
|
|
m_State = GK_UNREGISTERED;
|
|
SPIDER_TRACE(SP_STATE, "m_State = GK_UNREGISTERED (%X)\n", this);
|
|
|
|
// We deliberately don't free the RasMessage memory. Let the
|
|
// registration destructor do it - this provides protection
|
|
// from other requests.
|
|
|
|
// Close socket - this will terminate the Discovery thread
|
|
if ((nRet = m_pSocket->Close()) != 0)
|
|
return (GKI_WINSOCK2_ERROR(nRet));
|
|
|
|
// Delete cached address from the registry
|
|
g_pGatekeeper->DeleteCachedAddresses();
|
|
|
|
if (g_pGatekeeper->GetRejectFlag() == FALSE)
|
|
{
|
|
SPIDER_TRACE(SP_GKI, "PostMessage(m_hWnd, m_wBaseMessage + GKI_REG_BYPASS, 0, 0)\n", 0);
|
|
PostMessage(m_hWnd, m_wBaseMessage + GKI_REG_BYPASS,
|
|
0, 0);
|
|
return (GKI_EXIT_THREAD);
|
|
}
|
|
else
|
|
hResult = GKI_EXIT_THREAD;
|
|
#else
|
|
ASSERT(0);
|
|
hResult = GKI_EXIT_THREAD;
|
|
#endif //BROADCAST_DISCOVERY
|
|
break;
|
|
case registrationRequest_chosen:
|
|
SPIDER_TRACE(SP_NEWDEL, "del m_pRasMessage = %X\n", m_pRasMessage);
|
|
delete m_pRasMessage;
|
|
m_pRasMessage = 0;
|
|
|
|
#ifdef BROADCAST_DISCOVERY
|
|
// Need to attempt gatekeeper discovery
|
|
|
|
// Close socket and reopen in non-connected state to allow sendto
|
|
// This will also terminate the PostRecv thread
|
|
if ((nRet = m_pSocket->Close()) != 0)
|
|
return (GKI_WINSOCK2_ERROR(nRet));
|
|
if ((nRet = m_pSocket->Create(m_pSocket->GetAddrFam(), 0)) != 0)
|
|
return (GKI_WINSOCK2_ERROR(nRet));
|
|
|
|
// Delete allocated memory for sequence of RAS addresses
|
|
pTA1 = m_pRASAddress;
|
|
while (pTA1 != 0)
|
|
{
|
|
pTA2 = pTA1->next;
|
|
SPIDER_TRACE(SP_NEWDEL, "del pTA1 = %X\n", pTA1);
|
|
delete pTA1;
|
|
pTA1 = pTA2;
|
|
}
|
|
|
|
// Update RAS Address in CRegistration
|
|
for (pTA1 = m_pCallSignalAddress; pTA1 != 0; pTA1 = pTA1->next)
|
|
{
|
|
if (pTA1->value.choice == m_usRegistrationTransport)
|
|
if ((hResult = AddRASAddr(pTA1->value, m_pSocket->GetPort())) != GKI_OK)
|
|
return (hResult);
|
|
}
|
|
|
|
// Start the discovery thread
|
|
hThread = (HANDLE)_beginthread(GKDiscovery, 0, 0);
|
|
SPIDER_TRACE(SP_THREAD, "_beginthread(GKDiscovery, 0, 0); <%X>\n", hThread);
|
|
if (hThread == (HANDLE)-1)
|
|
return (GKI_NO_THREAD);
|
|
SetDiscThread(hThread);
|
|
|
|
hResult = GKI_REDISCOVER;
|
|
break;
|
|
#else // not BROADCAST_DISCOVERY
|
|
hResult = GKI_EXIT_THREAD;
|
|
#endif // BROADCAST_DISCOVERY
|
|
|
|
case unregistrationRequest_chosen:
|
|
m_State = GK_UNREGISTERED;
|
|
SPIDER_TRACE(SP_STATE, "m_State = GK_UNREGISTERED (%X)\n", this);
|
|
|
|
SPIDER_TRACE(SP_NEWDEL, "del m_pRasMessage = %X\n", m_pRasMessage);
|
|
delete m_pRasMessage;
|
|
m_pRasMessage = 0;
|
|
|
|
// Close socket - this will terminate the Receive thread
|
|
if ((nRet = m_pSocket->Close()) != 0)
|
|
return (GKI_WINSOCK2_ERROR(nRet));
|
|
|
|
hResult = GKI_EXIT_THREAD;
|
|
break;
|
|
|
|
case locationRequest_chosen:
|
|
m_State = GK_REGISTERED;
|
|
SPIDER_TRACE(SP_STATE, "m_State = GK_REGISTERED (%X)\n", this);
|
|
|
|
SPIDER_TRACE(SP_NEWDEL, "del m_pRasMessage = %X\n", m_pRasMessage);
|
|
delete m_pRasMessage;
|
|
m_pRasMessage = 0;
|
|
|
|
// Delete allocated memory for sequence of location alias addresses
|
|
pAA1 = m_pLocationInfo;
|
|
while (pAA1 != 0)
|
|
{
|
|
pAA2 = pAA1->next;
|
|
if (pAA1->value.choice == h323_ID_chosen)
|
|
{
|
|
SPIDER_TRACE(SP_NEWDEL, "del pAA1->value.u.h323_ID.value = %X\n", pAA1->value.u.h323_ID.value);
|
|
delete pAA1->value.u.h323_ID.value;
|
|
}
|
|
SPIDER_TRACE(SP_NEWDEL, "del pAA1 = %X\n", pAA1);
|
|
delete pAA1;
|
|
pAA1 = pAA2;
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
// Notify user that gatekeeper didn't respond
|
|
if (hResult != GKI_REDISCOVER)
|
|
{
|
|
SPIDER_TRACE(SP_GKI, "PostMessage(m_hWnd, m_wBaseMessage + GKI_ERROR, 0, GKI_NO_RESPONSE)\n", 0);
|
|
PostMessage(m_hWnd, m_wBaseMessage + GKI_ERROR,
|
|
0, GKI_NO_RESPONSE);
|
|
}
|
|
else
|
|
hResult = GKI_OK; // Don't exit retry thread
|
|
}
|
|
}
|
|
|
|
return (hResult);
|
|
}
|
|
|
|
HRESULT
|
|
CRegistration::SendInfoRequestResponse(CallInfoStruct *pCallInfo, RasMessage *pRasMessage)
|
|
{
|
|
// ABSTRACT: This function is called from one or more call object
|
|
// to create an IRR RasMessage, encapsulate the supplied
|
|
// call information and send the message to the
|
|
// gatekeeper.
|
|
// AUTHOR: Colin Hulme
|
|
|
|
ASN1_BUF Asn1Buf;
|
|
DWORD dwErrorCode;
|
|
RasMessage *pRespRasMessage;
|
|
struct sockaddr_in sAddrIn;
|
|
#ifdef _DEBUG
|
|
char szGKDebug[80];
|
|
#endif
|
|
|
|
|
|
SPIDER_TRACE(SP_FUNC, "CRegistration::SendInfoRequestResponse()\n", 0);
|
|
ASSERT(g_pCoder);
|
|
if (g_pCoder == NULL)
|
|
return (GKI_NOT_INITIALIZED);
|
|
|
|
// Allocate a RasMessage structure and initialized to 0
|
|
pRespRasMessage = new RasMessage;
|
|
SPIDER_TRACE(SP_NEWDEL, "new pRespRasMessage = %X\n", pRespRasMessage);
|
|
if (pRespRasMessage == 0)
|
|
return (GKI_NO_MEMORY);
|
|
memset(pRespRasMessage, 0, sizeof(RasMessage));
|
|
|
|
// Setup structure fields for InfoRequestResponse
|
|
pRespRasMessage->choice = infoRequestResponse_chosen;
|
|
if (m_pRgstrtnRqst_trmnlAls != 0)
|
|
pRespRasMessage->u.infoRequestResponse.bit_mask |= InfRqstRspns_endpntAls_present;
|
|
if (pCallInfo != 0)
|
|
pRespRasMessage->u.infoRequestResponse.bit_mask |= perCallInfo_present;
|
|
|
|
if (pRasMessage)
|
|
{
|
|
pRespRasMessage->u.infoRequestResponse.requestSeqNum =
|
|
pRasMessage->u.infoRequest.requestSeqNum;
|
|
if (pRasMessage->u.infoRequest.bit_mask & replyAddress_present)
|
|
{
|
|
switch (pRasMessage->u.infoRequest.replyAddress.choice)
|
|
{
|
|
case ipAddress_chosen:
|
|
sAddrIn.sin_family = AF_INET;
|
|
sAddrIn.sin_port = htons(pRasMessage->u.infoRequest.replyAddress.u.ipAddress.port);
|
|
break;
|
|
case ipxAddress_chosen:
|
|
sAddrIn.sin_family = AF_IPX;
|
|
sAddrIn.sin_port = htons(GKIPX_RAS_PORT); //Need to use reply port
|
|
break;
|
|
}
|
|
memcpy(&sAddrIn.sin_addr,
|
|
&pRasMessage->u.infoRequest.replyAddress.u.ipAddress.ip.value[0], 4);
|
|
}
|
|
}
|
|
else
|
|
// unsolicited IRRs must have a sequence number of 1!!!! (H.225 says so)
|
|
pRespRasMessage->u.infoRequestResponse.requestSeqNum = 1;
|
|
|
|
|
|
pRespRasMessage->u.infoRequestResponse.endpointType = m_terminalType;
|
|
pRespRasMessage->u.infoRequestResponse.endpointIdentifier = m_endpointID;
|
|
pRespRasMessage->u.infoRequestResponse.rasAddress = m_pRASAddress->value;
|
|
pRespRasMessage->u.infoRequestResponse.callSignalAddress =
|
|
(PInfoRequestResponse_callSignalAddress)m_pCallSignalAddress;
|
|
pRespRasMessage->u.infoRequestResponse.InfRqstRspns_endpntAls =
|
|
(PInfoRequestResponse_endpointAlias)m_pRgstrtnRqst_trmnlAls;
|
|
|
|
pRespRasMessage->u.infoRequestResponse.perCallInfo =
|
|
(PInfoRequestResponse_perCallInfo)pCallInfo;
|
|
|
|
#ifdef _DEBUG
|
|
if (dwGKIDLLFlags & SP_DUMPMEM)
|
|
DumpMem(pRespRasMessage, sizeof(RasMessage));
|
|
#endif
|
|
|
|
// Encode the PDU & send it
|
|
dwErrorCode = g_pCoder->Encode(pRespRasMessage, &Asn1Buf);
|
|
if (dwErrorCode)
|
|
return (GKI_ENCODER_ERROR);
|
|
|
|
SPIDER_TRACE(SP_PDU, "Send IRR; g_pReg = %X\n", this);
|
|
if (fGKIDontSend == FALSE)
|
|
{
|
|
if (pRasMessage && (pRasMessage->u.infoRequest.bit_mask & replyAddress_present))
|
|
{
|
|
if (g_pReg->m_pSocket->SendTo((char *)Asn1Buf.value, Asn1Buf.length,
|
|
(LPSOCKADDR)&sAddrIn, sizeof(sAddrIn)) == SOCKET_ERROR)
|
|
return (GKI_WINSOCK2_ERROR(SOCKET_ERROR));
|
|
}
|
|
else
|
|
{
|
|
if (g_pReg->m_pSocket->Send((char *)Asn1Buf.value, Asn1Buf.length) == SOCKET_ERROR)
|
|
return (GKI_WINSOCK2_ERROR(SOCKET_ERROR));
|
|
}
|
|
}
|
|
|
|
// Free the encoder memory
|
|
g_pCoder->Free(Asn1Buf);
|
|
|
|
// Delete allocated RasMessage storage
|
|
SPIDER_TRACE(SP_NEWDEL, "del pRespRasMessage = %X\n", pRespRasMessage);
|
|
delete pRespRasMessage;
|
|
|
|
return (GKI_OK);
|
|
}
|
|
|
|
|
|
//
|
|
// FindCallBySeqNum()
|
|
//
|
|
// ABSTRACT:
|
|
// This function attempts to locate a call within the list of calls
|
|
// in the registration object that has an outstanding RAS request that has this
|
|
// sequence number.
|
|
//
|
|
// RETURNS:
|
|
// Pointer to call associated with the sequence number or
|
|
// NULL if no call is found.
|
|
//
|
|
// NOTES:
|
|
// This function is usually called by the CRegistration::PDUHandler()
|
|
// when it receives a reply message that needs to be associated with a
|
|
// particular call.
|
|
//
|
|
// ASSUMPTIONS:
|
|
// Each call object holds on to the sequence numbers for RAS
|
|
// requests that it has not received replies for.
|
|
//
|
|
// AUTHOR: Dan Dexter
|
|
CCall *
|
|
CRegistration::FindCallBySeqNum(RequestSeqNum seqNum)
|
|
{
|
|
// If there are no calls, we can just return now
|
|
if (m_Calls.IsEmpty())
|
|
return(NULL);
|
|
|
|
// Initialize return value to "call not found"
|
|
CCall *RetVal = NULL;
|
|
|
|
// Otherwise, iterate through the calls and ask them
|
|
// if this sequence number belongs to them
|
|
POS pos;
|
|
for( pos = m_Calls.GetFirstPos(); pos != NULL; )
|
|
{
|
|
// Ask call if sequence number is theirs
|
|
CCall *pCall = m_Calls.GetNext(pos);
|
|
if (pCall->MatchSeqNum(seqNum))
|
|
{
|
|
RetVal = pCall;
|
|
break;
|
|
}
|
|
}
|
|
return(RetVal);
|
|
}
|
|
|
|
//
|
|
// FindCallByCRV()
|
|
//
|
|
// ABSTRACT:
|
|
// This function attempts to locate a call within the list of calls
|
|
// in the registration object that is associated with the passed in
|
|
// CallReferenceValue.
|
|
//
|
|
// RETURNS:
|
|
// Pointer to call associated with the CRV or
|
|
// NULL if no call is found.
|
|
//
|
|
// NOTES:
|
|
// This function is usually called by the CRegistration::PDUHandler()
|
|
// when it receives a reply message that needs to be associated with a
|
|
// particular call.
|
|
//
|
|
// AUTHOR: Dan Dexter
|
|
CCall *
|
|
CRegistration::FindCallByCRV(CallReferenceValue crv)
|
|
{
|
|
// If there are no calls, we can just return now
|
|
if (m_Calls.IsEmpty())
|
|
return(NULL);
|
|
|
|
// Initialize return value to "call not found"
|
|
CCall *RetVal = NULL;
|
|
|
|
// Otherwise, iterate through the calls and ask them
|
|
// if this CRV number belongs to them
|
|
POS pos;
|
|
for( pos = m_Calls.GetFirstPos(); pos != NULL; )
|
|
{
|
|
// Ask call if sequence number is theirs
|
|
CCall *pCall = m_Calls.GetNext(pos);
|
|
if (pCall->MatchCRV(crv))
|
|
{
|
|
RetVal = pCall;
|
|
break;
|
|
}
|
|
}
|
|
return(RetVal);
|
|
}
|
|
|
|
void
|
|
CRegistration::DeleteCall(CCall *pCall)
|
|
{
|
|
#ifdef _DEBUG
|
|
char szGKDebug[80];
|
|
#endif
|
|
|
|
POS pos = m_Calls.Find(pCall);
|
|
// We don't expect to be asked to delete
|
|
// calls that aren't in the list, so ASSERT
|
|
ASSERT(pos);
|
|
|
|
if (pos)
|
|
{
|
|
CCall *pCallObject = m_Calls.GetAt(pos);
|
|
m_Calls.RemoveAt(pos);
|
|
SPIDER_TRACE(SP_NEWDEL, "del pCallObject = %X\n", pCallObject);
|
|
delete pCallObject;
|
|
}
|
|
}
|
|
|
|
void
|
|
CRegistration::AddCall(CCall *pCall)
|
|
{
|
|
m_Calls.AddTail(pCall);
|
|
}
|
|
|
|
CCall *
|
|
CRegistration::GetNextCall(CCall *pCall)
|
|
{
|
|
CCall *RetVal = NULL;
|
|
|
|
if (pCall)
|
|
{
|
|
// The call list should never be empty
|
|
// if we're called with a non-NULL call pointer
|
|
ASSERT(!m_Calls.IsEmpty());
|
|
|
|
POS pos = m_Calls.Find(pCall);
|
|
// The call passed in better have been found
|
|
ASSERT(pos);
|
|
|
|
if (pos)
|
|
{
|
|
// This actually gets the existing call, but sets
|
|
// pos to point to the next call.
|
|
CCall *pNextCall = m_Calls.GetNext(pos);
|
|
if (pos)
|
|
{
|
|
// This call sets up the return value
|
|
RetVal = m_Calls.GetAt(pos);
|
|
}
|
|
}
|
|
}
|
|
return(RetVal);
|
|
}
|
|
|
|
#if 0
|
|
void
|
|
CRegistration::Lock(void)
|
|
{
|
|
EnterCriticalSection(&m_CriticalSection);
|
|
m_dwLockingThread = GetCurrentThreadId();
|
|
}
|
|
|
|
void
|
|
CRegistration::Unlock(void)
|
|
{
|
|
// Assert that the unlock is done by the
|
|
// thread that holds the lock
|
|
ASSERT(m_dwLockingThread == GetCurrentThreadId());
|
|
|
|
m_dwLockingThread = 0;
|
|
LeaveCriticalSection(&m_CriticalSection);
|
|
}
|
|
#endif
|
|
|