|
|
#include "stdafx.h"
#include "cbridge.h"
// SOURCE_Q931_INFO methods
/* virtual */ SOURCE_Q931_INFO::~SOURCE_Q931_INFO( ) { }
// this should never get called, but needs to be supported
// as the base class implementation is pure virtual
// virtual
HRESULT SOURCE_Q931_INFO::AcceptCallback ( IN DWORD Status, IN SOCKET Socket, IN SOCKADDR_IN * LocalAddress, IN SOCKADDR_IN * RemoteAddress) { // we should never receive an accept call back for the
// Q931 source instance
_ASSERTE(FALSE);
return E_UNEXPECTED; }
// This function is called by the event manager.
// The caller will free the PDU. This function may modify
// some of the fields of the PDU.
// this is called when an async receive operation completes
// virtual
HRESULT SOURCE_Q931_INFO::ReceiveCallback ( IN Q931_MESSAGE *pQ931Message, IN H323_UserInformation *pH323UserInfo ) { HRESULT HResult; // we must have valid decoded PDUs
_ASSERTE(NULL != pQ931Message);
// The ASN.1 part is not present in the case of some PDUs
//_ASSERTE(NULL != pH323UserInfo);
// if RELEASE COMPLETE PDU
if (pH323UserInfo != NULL && releaseComplete_chosen == pH323UserInfo->h323_uu_pdu.h323_message_body.choice) { DebugF (_T("Q931: 0x%x caller sent 'Release Complete'.\n"), &GetCallBridge ()); HResult = HandleReleaseCompletePDU( pQ931Message, pH323UserInfo );
return HResult; }
// check current state and handle the incoming PDU
switch(m_Q931SourceState) { case Q931_SOURCE_STATE_CON_ESTD: { // processes PDUs when in Q931_SOURCE_STATE_CON_ESTD state
HResult = HandleStateSrcConEstd( pQ931Message, pH323UserInfo ); } break;
case Q931_SOURCE_STATE_SETUP_RCVD: { // Pass on the PDU to the Q931 destination instance which
// passes it on after due modifications
HResult = GetDestQ931Info().ProcessSourcePDU( pQ931Message, pH323UserInfo ); } break;
case Q931_SOURCE_STATE_INIT: case Q931_SOURCE_STATE_REL_COMP_RCVD: default: { // we can't be in Q931_SOURCE_STATE_INIT as we wouldn't have
// queued an async receive by then
// we can't be in Q931_SOURCE_STATE_REL_COMP_RCVD as we not have
// queued this receive
// I.K. 0819999 _ASSERTE(FALSE);
HResult = E_UNEXPECTED; } break; };
// if there is an error
if (FAILED(HResult)) { goto shutdown; }
// we must queue an async receive irrespective of whether
// the PDU was dropped (IPTEL_E_INVALID_PDU == HResult)
// queue an async receive
HResult = QueueReceive(); if (FAILED(HResult)) { goto shutdown; }
return HResult;
shutdown:
// initiate shutdown
GetCallBridge().Terminate ();
return HResult; }
// handles RELEASE_COMPLETE PDUs
HRESULT SOURCE_Q931_INFO::HandleReleaseCompletePDU( IN Q931_MESSAGE *pQ931Message, IN H323_UserInformation *pH323UserInfo ) { // it must be a release complete PDU
_ASSERTE(releaseComplete_chosen == \ pH323UserInfo->h323_uu_pdu.h323_message_body.choice);
// we can handle a RELEASE COMPLETE PDU in any state except the following
_ASSERTE(Q931_SOURCE_STATE_INIT != m_Q931SourceState); _ASSERTE(Q931_SOURCE_STATE_REL_COMP_RCVD != m_Q931SourceState);
// pass on the pdu to the Q931 source instance
// ignore return error code, if any
GetDestQ931Info().ProcessSourcePDU( pQ931Message, pH323UserInfo ); //
// NetMeeting client doesn't like the fact that we send RST right after
// a ReleaseComplete.
// Yes, it is ugly and bad practice but this is a QFE
// for details look up bug# WinSE 31054, 691666 (read both 35928 and 33546).
//
Sleep( 500 );
// state transition to Q931_SOURCE_STATE_REL_COMP_RCVD
m_Q931SourceState = Q931_SOURCE_STATE_REL_COMP_RCVD;
// initiate shutdown - this cancels the timers, but doesn't close
// the sockets. the sockets are closed when the send callback is made
GetCallBridge().TerminateCallOnReleaseComplete();
GetSocketInfo ().Clear (TRUE);
return S_OK; }
// processes PDUs when in Q931_SOURCE_STATE_CON_EST state
HRESULT SOURCE_Q931_INFO::HandleStateSrcConEstd( IN Q931_MESSAGE *pQ931Message, IN H323_UserInformation *pH323UserInfo ) { if (!pH323UserInfo) { DebugF(_T("SOURCE_Q931_INFO::HandleStateSrcConEstd: no UUIE data! ignoring message.\n")); return E_INVALIDARG; }
// we can only handle a setup PDU in this state
// all other PDUs are THROWN AWAY (as we don't know
// whom to pass it to)
if (setup_chosen != pH323UserInfo->h323_uu_pdu.h323_message_body.choice) { DebugF( _T("SOURCE_Q931_INFO::HandleStateSrcConEstd: received a pdu other than Setup before receiving a Setup, pdu cannot be processed\n")); return E_INVALIDARG; }
// save the caller's call reference value now as we may reuse the
// PDU structure in ProcessSourcePDU
// The Setup PDU is sent by the originator and so the call reference flag
// should not be set.
// -XXX- this should not be an assert!!!! FIX THIS! -- arlied
_ASSERTE(!(pQ931Message->CallReferenceValue & CALL_REF_FLAG)); m_CallRefVal = pQ931Message->CallReferenceValue | CALL_REF_FLAG;
// pass on the setup pdu to the Q931 destination instance
HRESULT HResult = GetDestQ931Info().ProcessSourcePDU( pQ931Message, pH323UserInfo );
if (FAILED (HResult)) { return HResult; } // state transition to Q931_SOURCE_STATE_SETUP_RCVD
m_Q931SourceState = Q931_SOURCE_STATE_SETUP_RCVD;
// try to create a CALL PROCEEDING PDU
// if we fail, don't try to recover
// Q.931 requires that gateways in the call path must identify
// themselves to the callee
Q931_MESSAGE CallProcQ931Message; H323_UserInformation CallProcH323UserInfo; HResult = Q931EncodeCallProceedingMessage( m_CallRefVal, &CallProcQ931Message, &CallProcH323UserInfo );
// try to send a CALL PROCEEDING PDU to the caller
// if we fail, don't try to recover
HResult = QueueSend( &CallProcQ931Message, &CallProcH323UserInfo);
return HResult; }
// TimerValue contains the timer value in seconds, for a timer event
// to be created when a queued send completes
HRESULT SOURCE_Q931_INFO::ProcessDestPDU( IN Q931_MESSAGE *pQ931Message, IN H323_UserInformation *pH323UserInfo ) { HRESULT HResult = E_FAIL;
// handle PDU from the source Q931 instance
switch(m_Q931SourceState) { case Q931_SOURCE_STATE_SETUP_RCVD: { if (connect_chosen == pH323UserInfo->h323_uu_pdu.h323_message_body.choice) { DebugF (_T("Q931: 0x%x forwarding 'Connect' to caller.\n"), &GetCallBridge ()); HResult = ProcessConnectPDU( pQ931Message, pH323UserInfo ); if (FAILED(HResult)) { DebugF(_T("SOURCE_Q931_INFO::ProcessDestPDU: ProcessConnectPDU failed, returning %x\n"), HResult); return HResult; } } } break;
case Q931_SOURCE_STATE_INIT: case Q931_SOURCE_STATE_CON_ESTD: case Q931_SOURCE_STATE_REL_COMP_RCVD: default: { DebugF( _T("SOURCE_Q931_INFO::ProcessDestPDU: bogus state, returning E_UNEXPECTED\n")); return E_UNEXPECTED; } break; };
// Q931 Header - CallReferenceValue
// pQ931Message->CallReferenceValue = GetCallRefVal();
// queue async send for the PDU
HResult = QueueSend(pQ931Message, pH323UserInfo); if (HResult != S_OK) { DebugF( _T("SOURCE_Q931_INFO::ProcessDestPDU: failed to queue sendreturning %x\n"), HResult); return HResult; } return HResult; }
// NOTE: CRV modification is handled in ProcessDestPDU
HRESULT SOURCE_Q931_INFO::ProcessConnectPDU( IN Q931_MESSAGE *pQ931Message, IN H323_UserInformation *pH323UserInfo ) { Connect_UUIE * Connect; HRESULT Result; SOCKADDR_IN H245ListenAddress;
// it must be a CONNECT PDU
_ASSERTE(connect_chosen == pH323UserInfo->h323_uu_pdu.h323_message_body.choice);
Connect = &pH323UserInfo -> h323_uu_pdu.h323_message_body.u.connect;
// we must have already checked to see if an h245 transport
// address was specified by the callee in the dest instance
_ASSERTE(Connect_UUIE_h245Address_present & Connect -> bit_mask); _ASSERTE(ipAddress_chosen & Connect -> h245Address.choice);
// queue an overlapped accept, get ready to accept an incoming
// connection on the local address/port
H245ListenAddress.sin_addr.s_addr = htonl (GetCallBridge (). GetSourceInterfaceAddress ()); H245ListenAddress.sin_port = htons (0);
Result = GetSourceH245Info().ListenForCaller (&H245ListenAddress); if (FAILED (Result)) { DebugF (_T("H245: 0x%x failed to listen for caller.\n"), &GetCallBridge ());
return Result; } //_ASSERTE(S_FALSE != HResult);
// replace the h245 address/port in the connect PDU
FillTransportAddress (H245ListenAddress, Connect -> h245Address);
DebugF (_T("H245: 0x%x listens for H.245 connection from caller on %08X:%04X.\n"), &GetCallBridge (), SOCKADDR_IN_PRINTF (&H245ListenAddress)); return S_OK; }
|