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.
342 lines
10 KiB
342 lines
10 KiB
#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;
|
|
}
|