Source code of Windows XP (NT5)
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.
 
 
 
 
 
 

3725 lines
108 KiB

/*++
Copyright (c) 1999 Microsoft Corporation
Module Name:
confcall.cpp
Abstract:
TAPI Service Provider functions related to conference calls.
TSPI_lineAddToConference
TSPI_lineCompleteTransfer
TSPI_linePrepareAddToConference
TSPI_lineRemoveFromConference
TSPI_lineSetupConference
TSPI_lineSetupTransfer
TSPI_lineDial
TSPI_lineCompleteTransfer
TSPI_lineForward
TSPI_lineSetStatusMessage
TSPI_lineRedirect
TSPI_lineHold
TSPI_lineUnhold
Author:
Nikhil Bobde (NikhilB)
Revision History:
--*/
//
// Include files
//
#include "globals.h"
#include "line.h"
#include "q931obj.h"
#include "ras.h"
#include "call.h"
#define CALL_DIVERTNA_NUM_RINGS 8000
extern DWORD g_dwTSPIVersion;
//Queues a supplementary service work item to the thread pool
BOOL
QueueSuppServiceWorkItem(
IN DWORD EventID,
IN HDRVCALL hdCall,
IN ULONG_PTR dwParam1
)
{
SUPP_REQUEST_DATA * pCallRequestData = new SUPP_REQUEST_DATA;
BOOL fResult = TRUE;
if( pCallRequestData != NULL )
{
pCallRequestData -> EventID = EventID;
pCallRequestData -> hdCall = hdCall;
pCallRequestData -> dwParam1 = dwParam1;
if( !QueueUserWorkItem( ProcessSuppServiceWorkItem,
pCallRequestData, WT_EXECUTEDEFAULT) )
{
delete pCallRequestData;
fResult = FALSE;
}
}
else
{
fResult = FALSE;
}
return fResult;
}
#if DBG
DWORD
ProcessSuppServiceWorkItem(
IN PVOID ContextParameter
)
{
__try
{
return ProcessSuppServiceWorkItemFre( ContextParameter );
}
__except( 1 )
{
SUPP_REQUEST_DATA* pRequestData = (SUPP_REQUEST_DATA*)ContextParameter;
H323DBG(( DEBUG_LEVEL_TRACE, "TSPI %s event threw exception: %p, %p.",
EventIDToString(pRequestData -> EventID),
pRequestData -> hdCall,
pRequestData -> dwParam1 ));
_ASSERTE( FALSE );
return 0;
}
}
#endif
DWORD
ProcessSuppServiceWorkItemFre(
IN PVOID ContextParameter
)
{
_ASSERTE( ContextParameter );
PH323_CALL pCall = NULL;
SUPP_REQUEST_DATA* pRequestData = (SUPP_REQUEST_DATA*)ContextParameter;
H323DBG(( DEBUG_LEVEL_TRACE, "TSPI %s event recvd.",
EventIDToString(pRequestData -> EventID) ));
switch( pRequestData -> EventID )
{
case H450_PLACE_DIVERTEDCALL:
g_pH323Line -> PlaceDivertedCall( pRequestData->hdCall,
pRequestData->pAliasNames );
break;
case TSPI_DIAL_TRNASFEREDCALL:
g_pH323Line -> PlaceTransferedCall( pRequestData->hdCall,
pRequestData->pAliasNames );
break;
case SWAP_REPLACEMENT_CALL:
g_pH323Line -> SwapReplacementCall( pRequestData->hdCall,
pRequestData->hdReplacementCall, TRUE );
break;
case DROP_PRIMARY_CALL:
g_pH323Line -> SwapReplacementCall( pRequestData->hdCall,
pRequestData->hdReplacementCall, FALSE );
break;
case STOP_CTIDENTIFYRR_TIMER:
pCall=g_pH323Line -> FindH323CallAndLock( pRequestData->hdCall );
if( pCall != NULL )
{
pCall -> StopCTIdentifyRRTimer( pRequestData->hdReplacementCall );
pCall -> Unlock();
}
else
{
//set the m_hdRelatedCall of dwParam1 to NULL
}
break;
case SEND_CTINITIATE_MESSAGE:
pCall=g_pH323Line -> FindH323CallAndLock(pRequestData->hdCall);
if( pCall != NULL )
{
pCall -> SendCTInitiateMessagee( pRequestData->pCTIdentifyRes );
pCall -> Unlock();
}
break;
}
delete ContextParameter;
return EXIT_SUCCESS;
}
//!!must be always called in a lock
void
CH323Call::TransferInfoToDivertedCall(
IN PH323_CALL pDivertedCall
)
{
H323DBG(( DEBUG_LEVEL_TRACE, "TransferInfoToDivertedCall entered." ));
pDivertedCall -> SetDivertedCallInfo(
m_hdCall,
m_pCallReroutingInfo,
m_dwCallDiversionState,
m_hdMSPLine,
m_pCallerAliasNames,
m_htMSPLine,
m_pFastStart,
m_hdRelatedCall,
m_dwCallType,
m_dwAppSpecific,
&m_CallData );
//reset the reference to this struct
m_pCallReroutingInfo = NULL;
m_pCallerAliasNames = NULL;
m_dwCallDiversionState = H4503_CALLSTATE_IDLE;
m_dwCallType = CALLTYPE_NORMAL;
m_dwCallState = LINECALLSTATE_IDLE;
m_pFastStart = NULL;
ZeroMemory( (PVOID)&m_CallData, sizeof(H323_OCTETSTRING) );
H323DBG(( DEBUG_LEVEL_TRACE, "TransferInfoToDivertedCall exited." ));
}
//!!always called in a lock
BOOL
CH323Call::TransferInfoToTransferedCall(
IN PH323_CALL pTransferedCall
)
{
H323DBG(( DEBUG_LEVEL_TRACE, "TransferInfoToTransferedCall entered." ));
if( !pTransferedCall -> SetTransferedCallInfo(
m_hdCall, m_pCallerAliasNames, m_pCTCallIdentity ) )
{
pTransferedCall -> Unlock();
return FALSE;
}
m_hdRelatedCall = pTransferedCall -> GetCallHandle();
H323DBG(( DEBUG_LEVEL_TRACE, "TransferInfoToTransferedCall exited." ));
return TRUE;
}
void
CH323Call::SetDivertedCallInfo(
HDRVCALL hdCall,
CALLREROUTINGINFO* pCallReroutingInfo,
SUPP_CALLSTATE dwCallDiversionState,
HDRVMSPLINE hdMSPLine,
PH323_ALIASNAMES pCallerAliasNames,
HTAPIMSPLINE htMSPLine,
PH323_FASTSTART pFastStart,
HDRVCALL hdRelatedCall,
DWORD dwCallType,
DWORD dwAppSpecific,
PH323_OCTETSTRING pCallData
)
{
H323DBG(( DEBUG_LEVEL_TRACE, "SetDivertedCallInfo entered:%p.", this ));
Lock();
m_hdCall = hdCall;
m_pCallReroutingInfo = pCallReroutingInfo;
m_dwCallDiversionState = dwCallDiversionState;
m_hdMSPLine = hdMSPLine;
m_htMSPLine = htMSPLine;
FreeAliasNames( m_pCallerAliasNames );
m_pCallerAliasNames = pCallerAliasNames;
H323DBG(( DEBUG_LEVEL_ERROR, "Caller alias count:%d : %p", m_pCallerAliasNames->wCount, this ));
m_dwCallState = LINECALLSTATE_DIALING;
m_pFastStart = pFastStart;
m_CallData = *pCallData;
m_dwAppSpecific = dwAppSpecific;
//If the original call had any call type apart from being a diverted call
//copy it.
if( (dwCallType & CALLTYPE_TRANSFEREDSRC) ||
(dwCallType & CALLTYPE_DIVERTEDTRANSFERED) )
{
m_dwCallType |= CALLTYPE_DIVERTEDTRANSFERED;
m_hdRelatedCall = hdRelatedCall;
}
Unlock();
H323DBG(( DEBUG_LEVEL_TRACE, "SetDivertedCallInfo exited:%p.", this ));
}
//!!must be always called in a lock
BOOL
CH323Call::SetTransferedCallInfo(
HDRVCALL hdCall,
PH323_ALIASNAMES pCallerAliasNames,
BYTE * pCTCallIdentity
)
{
H323DBG(( DEBUG_LEVEL_TRACE, "SetTransferedCallInfo entered:%p.", this ));
m_hdRelatedCall = hdCall;
m_pCallerAliasNames = NULL;
if( (pCallerAliasNames != NULL) && (pCallerAliasNames->wCount != 0) )
{
m_pCallerAliasNames = new H323_ALIASNAMES;
if( m_pCallerAliasNames == NULL )
{
return FALSE;
}
ZeroMemory( (PVOID)m_pCallerAliasNames, sizeof(H323_ALIASNAMES) );
H323DBG(( DEBUG_LEVEL_ERROR, "Caller alias count:%d : %p", m_pCallerAliasNames->wCount, this ));
if( !AddAliasItem( m_pCallerAliasNames,
pCallerAliasNames->pItems[0].pData,
pCallerAliasNames->pItems[0].wType ) )
{
return FALSE;
}
H323DBG(( DEBUG_LEVEL_ERROR, "Caller alias count:%d : %p", m_pCallerAliasNames->wCount, this ));
}
CopyMemory( (PVOID)m_pCTCallIdentity, pCTCallIdentity,
sizeof(m_pCTCallIdentity) );
m_dwCallState = LINECALLSTATE_DIALING;
H323DBG(( DEBUG_LEVEL_TRACE, "SetTransferedCallInfo exited:%p.", this ));
return TRUE;
}
void
CH323Line::PlaceDivertedCall(
IN HDRVCALL hdCall,
IN PH323_ALIASNAMES divertedToAlias
)
{
H323DBG(( DEBUG_LEVEL_TRACE, "PlaceDivertedCall entered:%p.", this ));
int iIndex = MakeCallIndex( hdCall );
int iDivertedCallIndex;
PH323_CALL pCall;
BOOL fDelete = FALSE;
PH323_CALL pDivertedCall = NULL;
Lock();
LockCallTable();
//lock the call so that nobody else would be able to delete the call
if( (pCall=m_H323CallTable[iIndex]) != NULL )
{
pCall -> Lock();
if( pCall->GetCallHandle() == hdCall )
{
pDivertedCall = pCall-> CreateNewDivertedCall( divertedToAlias );
if( pDivertedCall == NULL )
{
pCall -> Unlock();
UnlockCallTable();
pCall->CloseCall( 0 );
Unlock();
return;
}
//remove the diverted call from the table
iDivertedCallIndex = pDivertedCall -> GetCallIndex();
m_H323CallTable[iDivertedCallIndex] = NULL;
//transfer the required information to the diverted call.
//put the original call in IDLE mode
pCall -> TransferInfoToDivertedCall( pDivertedCall );
//This DropCall is supposed to send only the DRQ if required
pCall->DropCall( 0 );
//close the original call
pCall -> Shutdown( &fDelete );
H323DBG(( DEBUG_LEVEL_VERBOSE, "call 0x%08lx closed.", pCall ));
pCall -> Unlock();
//release the original call object
if( fDelete == TRUE )
{
H323DBG(( DEBUG_LEVEL_VERBOSE, "call delete:0x%08lx.", pCall ));
delete pCall;
}
//place the diverted call in the place of the original call
m_H323CallTable[iIndex] = pDivertedCall;
}
else
{
pCall -> Unlock();
}
}
UnlockCallTable();
//dial the diverted call
if( pDivertedCall )
{
pDivertedCall -> DialCall();
}
Unlock();
H323DBG(( DEBUG_LEVEL_TRACE, "PlaceDivertedCall exited:%p.", this ));
}
//!!always called in a lock
PH323_CALL
CH323Call::CreateNewDivertedCall(
IN PH323_ALIASNAMES pwszCalleeAlias
)
{
PH323_CONFERENCE pConf = NULL;
BOOL fDelete = FALSE;
PH323_CALL pCall = new CH323Call();
H323DBG(( DEBUG_LEVEL_TRACE, "CreateNewDivertedCall entered:%p.", this ));
if( pCall == NULL )
{
H323DBG(( DEBUG_LEVEL_ERROR,
"could not allocate diverted call." ));
return NULL;
}
// save tapi handle and specify outgoing call direction
if( !pCall -> Initialize( m_htCall, LINECALLORIGIN_OUTBOUND,
CALLTYPE_DIVERTEDSRC ) )
{
H323DBG(( DEBUG_LEVEL_ERROR,
"could not allocate outgoing call." ));
goto cleanup;
}
// bind outgoing call
pConf = pCall -> CreateConference(NULL);
if( pConf == NULL )
{
H323DBG(( DEBUG_LEVEL_ERROR,
"could not create conference." ));
goto cleanup;
}
if( !g_pH323Line -> GetH323ConfTable() -> Add(pConf) )
{
H323DBG(( DEBUG_LEVEL_ERROR, "could not add conf to conf table." ));
goto cleanup;
}
if( pwszCalleeAlias->pItems[0].wType == e164_chosen )
{
pCall->SetAddressType( e164_chosen );
}
if (!RasIsRegistered())
{
if( !pCall->ResolveAddress( pwszCalleeAlias->pItems[0].pData ) )
{
goto cleanup;
}
}
if( !pCall->SetCalleeAlias( pwszCalleeAlias->pItems[0].pData,
pwszCalleeAlias->pItems[0].wType ) )
{
goto cleanup;
}
//send the informarion to the user that call has been diverted
PostLineEvent(
LINE_CALLINFO,
LINECALLINFOSTATE_REDIRECTIONID,
0, 0 );
H323DBG(( DEBUG_LEVEL_TRACE, "diverted call created:%p.", pCall ));
H323DBG(( DEBUG_LEVEL_TRACE, "CreateNewDivertedCall exited.\n:%p", this ));
return pCall;
cleanup:
if( pCall != NULL )
{
pCall -> Shutdown( &fDelete );
delete pCall;
H323DBG((DEBUG_LEVEL_TRACE, "call delete:%p.", pCall ));
}
return NULL;
}
//!!always called in a lock
void
CH323Call::Hold()
{
//1.Send MSP call hold message
//2.Send hold H450 APDU
if( m_dwCallState == LINECALLSTATE_ONHOLD )
{
return;
}
if( !SendQ931Message( NO_INVOKEID, 0, 0, FACILITYMESSAGETYPE,
HOLDNOTIFIC_OPCODE | H450_INVOKE ) )
{
CloseCall( 0 );
return;
}
SendMSPMessage( SP_MSG_Hold, 0, 1, NULL );
//Call put on hold by local endpoint
m_dwFlags |= TSPI_CALL_LOCAL_HOLD;
ChangeCallState( LINECALLSTATE_ONHOLD, 0 );
return;
}
//!!always called in a lock
void
CH323Call::UnHold()
{
//1.Send MSP call unhold message
//2.Send unhold H450 APDU
if( (m_dwCallState == LINECALLSTATE_ONHOLD) &&
(m_dwFlags & TSPI_CALL_LOCAL_HOLD) )
{
if( !SendQ931Message( NO_INVOKEID, 0, 0, FACILITYMESSAGETYPE,
RETRIEVENOTIFIC_OPCODE | H450_INVOKE ) )
{
CloseCall( 0 );
return;
}
SendMSPMessage( SP_MSG_Hold, 0, 0, NULL );
m_dwFlags &= (~TSPI_CALL_LOCAL_HOLD);
ChangeCallState( LINECALLSTATE_CONNECTED, 0 );
}
return;
}
//!!always called in a lock
void
CH323Call::OnCallReroutingReceive(
IN DWORD dwInvokeID
)
{
H323DBG(( DEBUG_LEVEL_TRACE, "OnCallReroutingReceive entered:%p.", this ));
if( (m_dwOrigin != LINECALLORIGIN_OUTBOUND) ||
( (m_dwStateMachine != Q931_SETUP_SENT) &&
(m_dwStateMachine != Q931_PROCEED_RECVD) &&
(m_dwStateMachine != Q931_ALERT_RECVD)
)
)
{
goto error;
}
//If setupsent timer is still alive stop it.
if( m_hSetupSentTimer != NULL )
{
DeleteTimerQueueTimer( H323TimerQueue, m_hSetupSentTimer, NULL );
m_hSetupSentTimer = NULL;
}
if( dwInvokeID != NO_INVOKEID )
{
if(!SendQ931Message( dwInvokeID,
0,
0,
FACILITYMESSAGETYPE,
CALLREROUTING_OPCODE | H450_RETURNRESULT ))
{
//goto error;
}
}
m_fCallInTrnasition = TRUE;
if( !SendQ931Message( NO_INVOKEID,
0,
0,
RELEASECOMPLMESSAGETYPE,
NO_H450_APDU) )
{
//goto error;
}
if( !QueueSuppServiceWorkItem( H450_PLACE_DIVERTEDCALL, m_hdCall,
(ULONG_PTR)m_pCallReroutingInfo->divertedToNrAlias ) )
{
H323DBG(( DEBUG_LEVEL_ERROR, "could not post place diverted event." ));
}
H323DBG(( DEBUG_LEVEL_TRACE, "OnCallReroutingReceive exited:%p.", this ));
return;
error:
CloseCall( 0 );
}
//!!always called in a lock
BOOL
CH323Call::IsValidInvokeID(
IN DWORD dwInvokeId
)
{
if( m_dwCallType != CALLTYPE_NORMAL )
{
if( m_dwInvokeID == dwInvokeId )
{
return TRUE;
}
H323DBG(( DEBUG_LEVEL_ERROR, "invoke id not matched:%d:%d.",
m_dwInvokeID, dwInvokeId ));
return FALSE;
}
H323DBG(( DEBUG_LEVEL_ERROR, "IsValidinvokeID called on wrong call." ));
return FALSE;
}
BOOL
CH323Call::StartTimerForCallDiversionOnNA(
IN PH323_ALIASITEM pwszDivertedToAlias
)
{
if( m_pCallReroutingInfo == NULL )
{
m_pCallReroutingInfo = new CALLREROUTINGINFO;
if( m_pCallReroutingInfo == NULL )
{
goto cleanup;
}
ZeroMemory( m_pCallReroutingInfo, sizeof(CALLREROUTINGINFO) );
}
m_pCallReroutingInfo->divertedToNrAlias = new H323_ALIASNAMES;
if( m_pCallReroutingInfo->divertedToNrAlias == NULL )
{
goto cleanup;
}
ZeroMemory( m_pCallReroutingInfo->divertedToNrAlias,
sizeof(H323_ALIASNAMES) );
if( !AddAliasItem( m_pCallReroutingInfo->divertedToNrAlias,
(BYTE*)pwszDivertedToAlias->pData,
sizeof(WCHAR) * (wcslen(pwszDivertedToAlias->pData) +1),
pwszDivertedToAlias->wType ) )
{
goto cleanup;
}
if( !CreateTimerQueueTimer(
&m_hCallDivertOnNATimer,
H323TimerQueue,
CH323Call::CallDivertOnNACallback,
(PVOID)m_hdCall,
(g_pH323Line->m_dwNumRingsNoAnswer * 1000), 0,
WT_EXECUTEINIOTHREAD | WT_EXECUTEONLYONCE) )
{
goto cleanup;
}
return TRUE;
cleanup:
FreeCallReroutingInfo();
return FALSE;
}
LONG
CH323Call::SetDivertedToAlias(
WCHAR* pwszDivertedToAddr,
WORD wAliasType
)
{
if( m_pCallReroutingInfo == NULL )
{
m_pCallReroutingInfo = new CALLREROUTINGINFO;
if( m_pCallReroutingInfo == NULL )
{
return LINEERR_NOMEM;
}
ZeroMemory( m_pCallReroutingInfo, sizeof(CALLREROUTINGINFO) );
}
m_pCallReroutingInfo->divertedToNrAlias = new H323_ALIASNAMES;
if( m_pCallReroutingInfo->divertedToNrAlias == NULL )
{
delete m_pCallReroutingInfo;
m_pCallReroutingInfo = NULL;
return LINEERR_NOMEM;
}
ZeroMemory( m_pCallReroutingInfo->divertedToNrAlias, sizeof(H323_ALIASNAMES) );
if( !AddAliasItem( m_pCallReroutingInfo->divertedToNrAlias,
(BYTE*)pwszDivertedToAddr,
sizeof(WCHAR) * (wcslen(pwszDivertedToAddr) +1),
wAliasType ) )
{
delete m_pCallReroutingInfo->divertedToNrAlias;
delete m_pCallReroutingInfo;
m_pCallReroutingInfo = NULL;
return LINEERR_NOMEM;
}
return NOERROR;
}
//!!always called in a lock
//this function is called to replace a TRANSFERD_PRIMARY call with a connected
//TRANSFEREDSRC call or to preplace a TRANSFERED2_PRIMARY call with a connected
//TRANSFEREDDEST call
BOOL
CH323Call::InitiateCallReplacement(
PH323_FASTSTART pFastStart,
PH323_ADDR pH245Addr
)
{
H323DBG(( DEBUG_LEVEL_ERROR, "InitiateCallReplacement entered:%p.",this ));
SendMSPStartH245( pH245Addr, pFastStart );
SendMSPMessage( SP_MSG_ConnectComplete, 0, 0, m_hdRelatedCall );
m_fCallInTrnasition = TRUE;
if( (m_dwRASCallState == RASCALL_STATE_REGISTERED ) ||
(m_dwRASCallState == RASCALL_STATE_ARQSENT ) )
{
//disengage from the GK
SendDRQ( forcedDrop_chosen, NOT_RESEND_SEQ_NUM, FALSE );
}
if( !SendQ931Message( m_dwInvokeID,
0,
0,
RELEASECOMPLMESSAGETYPE,
CTINITIATE_OPCODE | H450_RETURNRESULT) )
{
goto cleanup;
}
H323DBG(( DEBUG_LEVEL_ERROR, "InitiateCallReplacement exited:%p.",this ));
return TRUE;
cleanup:
CloseCall( 0 );
return FALSE;
}
//this function is called to replace a TRANSFERD_PRIMARY call with TRANSFEREDSRC
//call or to replace a TRANSFERD2_CONSULT call with a TRANSFEREDDEST call
void
CH323Line::SwapReplacementCall(
HDRVCALL hdReplacementCall,
HDRVCALL hdPrimaryCall,
BOOL fChangeCallSate
)
{
H323DBG(( DEBUG_LEVEL_TRACE, "SwapReplacementCall entered:%p.", this ));
int iReplacementCallIndex = (int)LOWORD(hdReplacementCall);
int iPrimaryCallIndex = (int)LOWORD(hdPrimaryCall);
PH323_CALL pReplacementCall;
BOOL fDelete = FALSE;
PH323_CALL pPrimaryCall;
Lock();
LockCallTable();
//lock the replacement call before primary call to avoid deadlock
if( (pReplacementCall=m_H323CallTable[iReplacementCallIndex]) != NULL )
{
pReplacementCall -> Lock();
if( pReplacementCall -> GetCallHandle() == hdReplacementCall )
{
if( (pPrimaryCall=m_H323CallTable[iPrimaryCallIndex]) != NULL )
{
pPrimaryCall -> Lock();
if( pPrimaryCall -> GetCallHandle() == hdPrimaryCall )
{
pPrimaryCall -> InitiateCallReplacement(
pReplacementCall->GetPeerFastStart(),
pReplacementCall->GetPeerH245Addr() );
//remove the replacement call from the table
m_H323CallTable[iReplacementCallIndex] = NULL;
//Transfer the required information to the replacement call.
//put the primary call in IDLE mode
pPrimaryCall -> TransferInfoToReplacementCall( pReplacementCall );
//close the original call
pPrimaryCall -> Shutdown( &fDelete );
H323DBG(( DEBUG_LEVEL_VERBOSE, "call 0x%08lx closed.",
pPrimaryCall ));
pPrimaryCall -> Unlock();
//release the primary call object
if( fDelete == TRUE )
{
H323DBG(( DEBUG_LEVEL_VERBOSE, "call delete:0x%08lx.",
pPrimaryCall ));
delete pPrimaryCall;
}
//Place the replacement call in the place of primary call.
m_H323CallTable[iPrimaryCallIndex] = pReplacementCall;
}
else
{
pPrimaryCall -> Unlock();
}
}
else
{
pReplacementCall-> CloseCall( 0 );
}
//inform TAPI that the transfered call is in connected state
if( fChangeCallSate == TRUE )
{
pReplacementCall->ChangeCallState( LINECALLSTATE_CONNECTED, 0 );
}
}
pReplacementCall -> Unlock();
}
UnlockCallTable();
Unlock();
H323DBG(( DEBUG_LEVEL_TRACE, "SwapReplacementCall exited:%p.", this ));
}
//!!both the calls are locked when this function is called
void
CH323Call::TransferInfoToReplacementCall(
PH323_CALL pReplacementCall
)
{
H323DBG(( DEBUG_LEVEL_TRACE,
"TransferInfoToReplacementCall entered:%p.", this ));
m_dwCallDiversionState = H4503_CALLSTATE_IDLE;
m_dwCallType = CALLTYPE_NORMAL;
m_dwCallState = LINECALLSTATE_IDLE;
pReplacementCall->SetReplacementCallInfo(
m_hdCall,
m_hdMSPLine,
m_htCall,
m_htMSPLine,
m_dwAppSpecific,
&m_CallData );
//don't release this octet string while releasing tis call.
ZeroMemory( (PVOID)&m_CallData, sizeof(H323_OCTETSTRING) );
H323DBG(( DEBUG_LEVEL_TRACE,
"TransferInfoToReplacementCall exited:%p.", this ));
}
//!!always called in a lock
void
CH323Call::SetReplacementCallInfo(
HDRVCALL hdCall,
HDRVMSPLINE hdMSPLine,
HTAPICALL htCall,
HTAPIMSPLINE htMSPLine,
DWORD dwAppSpecific,
PH323_OCTETSTRING pCallData
)
{
H323DBG(( DEBUG_LEVEL_TRACE,
"SetReplacementCallInfo entered:%p.", this ));
m_hdCall = hdCall;
m_hdMSPLine = hdMSPLine;
m_htMSPLine = htMSPLine;
m_htCall = htCall;
m_hdRelatedCall = NULL;
m_dwCallType = CALLTYPE_NORMAL;
m_dwAppSpecific = dwAppSpecific;
m_CallData = *pCallData;
H323DBG(( DEBUG_LEVEL_TRACE,
"SetReplacementCallInfo exited:%p.", this ));
}
//!!always called in a lock
void
CH323Call::CompleteTransfer(
PH323_CALL pCall
)
{
BOOL retVal;
//set the call type of both the calls
pCall -> SetCallType( CALLTYPE_TRANSFERING_PRIMARY );
m_dwCallType |= CALLTYPE_TRANSFERING_CONSULT;
m_hdRelatedCall = pCall -> GetCallHandle();
//send CallTransferIdentify message to the transferredTo endpoint over the
//consultation call
retVal = SendQ931Message( NO_INVOKEID, 0, 0/*undefinedReason*/,
FACILITYMESSAGETYPE, CTIDENTIFY_OPCODE | H450_INVOKE );
m_dwCallDiversionState = H4502_CTIDENTIFY_SENT;
//start the timer for CTIdenity message
if( retVal )
{
retVal = CreateTimerQueueTimer(
&m_hCTIdentifyTimer,
H323TimerQueue,
CH323Call::CTIdentifyExpiredCallback,
(PVOID)m_hdCall,
CTIDENTIFY_SENT_TIMEOUT, 0,
WT_EXECUTEINIOTHREAD | WT_EXECUTEONLYONCE );
}
if( retVal == FALSE )
{
CloseCall( 0 );
}
}
void
CH323Line::PlaceTransferedCall(
IN HDRVCALL hdCall,
IN PH323_ALIASNAMES pTransferedToAlias
)
{
H323DBG(( DEBUG_LEVEL_TRACE, "PlaceTransferedCall entered:%p.", this ));
PH323_CALL pCall = NULL;
PH323_CALL pTransferedCall = NULL;
BOOL fDelete = FALSE;
Lock();
LockCallTable();
pCall=g_pH323Line -> FindH323CallAndLock(hdCall);
if( pCall == NULL )
{
goto cleanup;
}
pTransferedCall = CreateNewTransferedCall( pTransferedToAlias );
if( pTransferedCall == NULL )
{
goto cleanup;
}
//transfer the required information to the transfered call
if( !pCall -> TransferInfoToTransferedCall( pTransferedCall ) )
{
goto cleanup;
}
pCall -> SetCallState( LINECALLSTATE_ONHOLD );
pCall -> SendMSPMessage( SP_MSG_Hold, 0, 1, NULL );
pCall -> Unlock();
//dial the transfered call
pTransferedCall -> DialCall();
UnlockCallTable();
Unlock();
H323DBG(( DEBUG_LEVEL_TRACE, "PlaceTransferedCall exited:%p.", this ));
return;
cleanup:
if( pCall != NULL )
{
//close the primary call
QueueTAPILineRequest(
TSPI_CLOSE_CALL,
hdCall,
NULL,
LINEDISCONNECTMODE_NORMAL,
NULL);
pCall -> Unlock();
}
if( pTransferedCall )
{
pTransferedCall -> Shutdown( &fDelete );
delete pTransferedCall;
}
UnlockCallTable();
Unlock();
}
//!!always called in a lock
PH323_CALL
CH323Line::CreateNewTransferedCall(
IN PH323_ALIASNAMES pwszCalleeAlias
)
{
PH323_CONFERENCE pConf = NULL;
BOOL fDelete = FALSE;
PH323_CALL pCall = new CH323Call();
H323DBG(( DEBUG_LEVEL_TRACE, "CreateNewTransferedCall entered:%p.", this ));
if( pCall == NULL )
{
H323DBG(( DEBUG_LEVEL_ERROR,
"could not allocate Transfered call." ));
return NULL;
}
// no tapi handle for this call
if( !pCall -> Initialize( NULL, LINECALLORIGIN_OUTBOUND,
CALLTYPE_TRANSFEREDSRC ) )
{
H323DBG(( DEBUG_LEVEL_ERROR,
"could not allocate outgoing call." ));
goto cleanup;
}
// bind outgoing call
pConf = pCall -> CreateConference(NULL);
if( pConf == NULL )
{
H323DBG(( DEBUG_LEVEL_ERROR,
"could not create conference." ));
goto cleanup;
}
if( !g_pH323Line -> GetH323ConfTable() -> Add(pConf) )
{
H323DBG(( DEBUG_LEVEL_ERROR, "could not add conf to conf table." ));
goto cleanup;
}
if( pwszCalleeAlias->pItems[0].wType == e164_chosen )
{
pCall->SetAddressType( e164_chosen );
}
if (!RasIsRegistered())
{
if( !pCall->ResolveAddress( pwszCalleeAlias->pItems[0].pData ) )
{
goto cleanup;
}
}
if( !pCall->SetCalleeAlias( pwszCalleeAlias->pItems[0].pData,
pwszCalleeAlias->pItems[0].wType ) )
{
goto cleanup;
}
H323DBG(( DEBUG_LEVEL_TRACE, "Transfered call created:%p.", pCall ));
H323DBG(( DEBUG_LEVEL_TRACE, "CreateNewTransferedCall exited:%p.", this ));
return pCall;
cleanup:
if( pCall != NULL )
{
pCall -> Shutdown( &fDelete );
delete pCall;
H323DBG((DEBUG_LEVEL_TRACE, "call delete:%p.", pCall ));
}
return NULL;
}
//
// TSPI procedures
//
LONG
TSPIAPI
TSPI_lineAddToConference(
DRV_REQUESTID dwRequestID,
HDRVCALL hdConfCall,
HDRVCALL hdConsultCall
)
/*++
Routine Description:
This function adds the call specified by hdConsultCall to the conference
call specified by hdConfCall.
Note that the call handle of the added party remains valid after adding
the call to a conference; its state will typically change to conferenced
while the state of the conference call will typically become connected.
The handle to an individual participating call can be used later to remove
that party from the conference call using TSPI_lineRemoveFromConference.
The call states of the calls participating in a conference are not
independent. For example, when dropping a conference call, all
participating calls may automatically become idle. The TAPI DLL may consult
the line's device capabilities to determine what form of conference removal
is available. The TAPI DLL or its client applications should track the
LINE_CALLSTATE messages to determine what really happened to the calls
involved.
The conference call is established either via TSPI_lineSetupConference or
TSPI_lineCompleteTransfer. The call added to a conference will typically be
established using TSPI_lineSetupConference or
TSPI_linePrepareAddToConference. Some switches may allow adding of an
arbitrary calls to conference, and such a call may have been set up using
TSPI_lineMakeCall and be on (hard) hold.
Arguments:
dwRequestID - Specifies the identifier of the asynchronous request.
The Service Provider returns this value if the function completes
asynchronously.
hdConfCall - Specifies the Service Provider's opaque handle to the
conference call. Valid call states: onHoldPendingConference, onHold.
hdAddCall - Specifies the Service Provider's opaque handle to the call to
be added to the conference call. Valid call states: connected, onHold.
Return Values:
Returns zero if the function is successful, the (positive) dwRequestID
value if the function will be completed asynchronously, or a negative error
number if an error has occurred. Possible error returns are:
LINEERR_INVALCONFCALLHANDLE - The specified call handle for the
conference call is invalid or is not a handle for a conference
call.
LINEERR_INVALCALLHANDLE - The specified call handle for the added
call is invalid.
LINEERR_INVALCALLSTATE - One or both of the specified calls are not
in a valid state for the requested operation.
LINEERR_CONFERENCEFULL - The maximum number of parties for a
conference has been reached.
LINEERR_OPERATIONUNAVAIL - The specified operation is not available.
LINEERR_OPERATIONFAILED - The specified operation failed for
unspecified reason.
--*/
{
return LINEERR_OPERATIONUNAVAIL; // CODEWORK...
}
LONG
TSPIAPI
TSPI_lineBlindTransfer(
DRV_REQUESTID dwRequestID,
HDRVCALL hdCall,
LPCWSTR lpszDestAddress,
DWORD dwCountryCode
)
{
PH323_CALL pCall = NULL;
LONG retVal = ERROR_SUCCESS;
H323DBG(( DEBUG_LEVEL_TRACE, "TSPI_lineBlindTransfer - Entered." ));
if( lpszDestAddress == NULL )
{
return LINEERR_INVALPARAM;
}
// retrieve call pointer from handle
pCall=g_pH323Line -> FindH323CallAndLock(hdCall);
if( pCall == NULL )
{
return LINEERR_INVALCALLHANDLE;
}
retVal = pCall -> InitiateBlindTransfer( lpszDestAddress );
pCall -> Unlock();
if( retVal == ERROR_SUCCESS )
{
// complete the async accept operation now
H323CompleteRequest (dwRequestID, ERROR_SUCCESS);
retVal = dwRequestID;
}
H323DBG(( DEBUG_LEVEL_TRACE, "TSPI_lineBlindTransfer - Exited." ));
return retVal;
}
//!!always called in a lock
LONG
CH323Call::InitiateBlindTransfer(
IN LPCWSTR lpszDestAddress
)
{
WORD wAliasType = h323_ID_chosen;
DWORD dwMaxAddrSize = MAX_H323_ADDR_LEN;
DWORD dwAddrLen;
H323DBG(( DEBUG_LEVEL_TRACE, "InitiateBlindTransfer - Entered." ));
if( m_dwCallState != LINECALLSTATE_CONNECTED )
{
return LINEERR_INVALCALLSTATE;
}
if( (*lpszDestAddress==L'T') &&
IsValidE164String((WCHAR*)lpszDestAddress+1) )
{
wAliasType = e164_chosen;
//strip off the leading 'T'
lpszDestAddress++;
dwMaxAddrSize = MAX_E164_ADDR_LEN;
}
else if( IsValidE164String( (WCHAR*)lpszDestAddress) )
{
wAliasType = e164_chosen;
dwMaxAddrSize = MAX_E164_ADDR_LEN;
}
dwAddrLen = wcslen( lpszDestAddress );
if( (dwAddrLen > dwMaxAddrSize) || (dwAddrLen == 0) )
{
return LINEERR_INVALPARAM;
}
if( m_pTransferedToAlias )
{
FreeAliasNames( m_pTransferedToAlias );
m_pTransferedToAlias = NULL;
}
m_pTransferedToAlias = new H323_ALIASNAMES;
if( m_pTransferedToAlias == NULL )
{
return LINEERR_OPERATIONFAILED;
}
ZeroMemory( (PVOID)m_pTransferedToAlias, sizeof(H323_ALIASNAMES) );
if( !AddAliasItem( m_pTransferedToAlias,
(BYTE*)lpszDestAddress,
sizeof(WCHAR) * (wcslen(lpszDestAddress) +1),
wAliasType ) )
{
return LINEERR_OPERATIONFAILED;
}
if( !SendQ931Message( NO_INVOKEID, 0, 0, FACILITYMESSAGETYPE,
CTINITIATE_OPCODE | H450_INVOKE ) )
{
return LINEERR_OPERATIONFAILED;
}
m_dwCallDiversionState = H4502_CTINITIATE_SENT;
m_dwCallType |= CALLTYPE_TRANSFERING_PRIMARY;
H323DBG(( DEBUG_LEVEL_TRACE, "InitiateBlindTransfer - Exited." ));
return ERROR_SUCCESS;
}
/*++
Routine Description:
This function completes the transfer of the specified call to the party
connected in the consultation call.
This operation completes the transfer of the original call, hdCall, to
the party currently connected via hdConsultCall. The consultation call
will typically have been dialed on the consultation call allocated as
part of TSPI_lineSetupTransfer, but it may be any call to which the
switch is capable of transferring hdCall.
The transfer request can be resolved either as a transfer or as a
three-way conference call. When resolved as a transfer, the parties
connected via hdCall and hdConsultCall will be connected to each other,
and both hdCall and hdConsultCall will typically be removed from the line
they were on and both will transition to the idle state. Note that the
Service Provider's opaque handles for these calls must remain valid after
the transfer has completed. The TAPI DLL causes these handles to be
invalidated when it is no longer interested in them using
TSPI_lineCloseCall.
When resolved as a conference, all three parties will enter in a
conference call. Both existing call handles remain valid, but will
transition to the conferenced state. A conference call handle will created
and returned, and it will transition to the connected state.
It may also be possible to perform a blind transfer of a call using
TSPI_lineBlindTransfer.
Arguments:
dwRequestID - Specifies the identifier of the asynchronous request.
The Service Provider returns this value if the function completes
asynchronously.
hdCall - Specifies the Service Provider's opaque handle to the call to be
transferred. Valid call states: onHoldPendingTransfer.
hdConsultCall - Specifies a handle to the call that represents a connection
with the destination of the transfer. Valid call states: connected,
ringback, busy.
htConfCall - Specifies the TAPI DLL's opaque handle to the new call. If
dwTransferMode is specified as LINETRANSFERMODE_CONFERENCE then the
Service Provider must save this and use it in all subsequent calls to
the LINEEVENT procedure reporting events on the call. Otherwise this
parameter is ignored.
lphdConfCall - Specifies a far pointer to an opaque HDRVCALL representing
the Service Provider's identifier for the call. If dwTransferMode is
specified as LINETRANSFERMODE_CONFERENCE then the Service Provider must
fill this location with its opaque handle for the new conference call
before this procedure returns, whether it decides to execute the
request sychronously or asynchronously. This handle is invalid if the
function results in an error (either synchronously or asynchronously).
If dwTransferMode is some other value this parameter is ignored.
dwTransferMode - Specifies how the initiated transfer request is to be
resolved, of type LINETRANSFERMODE. Values are:
LINETRANSFERMODE_TRANSFER - Resolve the initiated transfer by
transferring the initial call to the consultation call.
LINETRANSFERMODE_CONFERENCE - Resolve the initiated transfer by
conferencing all three parties into a three-way conference call.
A conference call is created and returned to the TAPI DLL.
Return Values:
Returns zero if the function is successful, the (positive) dwRequestID
value if the function will be completed asynchronously, or a negative error
number if an error has occurred. Possible error returns are:
LINEERR_INVALCALLHANDLE - The specified call handle is invalid.
LINEERR_INVALCONSULTCALLHANDLE - The specified consultation call
handle is invalid.
LINEERR_INVALCALLSTATE - One or both calls are not in a valid state
for the requested operation.
LINEERR_INVALTRANSFERMODE - The specified transfer mode parameter is
invalid.
LINEERR_INVALPOINTER - The specified pointer parameter is invalid.
LINEERR_OPERATIONUNAVAIL - The specified operation is not available.
LINEERR_OPERATIONFAILED - The specified operation failed for
unspecified reason.
--*/
LONG
TSPIAPI
TSPI_lineCompleteTransfer(
DRV_REQUESTID dwRequestID,
HDRVCALL hdCall,
HDRVCALL hdConsultCall,
HTAPICALL htConfCall,
LPHDRVCALL lphdConfCall,
DWORD dwTransferMode
)
{
LONG retVal = (DWORD)dwRequestID;
PH323_CALL pCall = NULL;
PH323_CALL pConsultCall = NULL;
H323DBG(( DEBUG_LEVEL_TRACE, "TSPI_lineCompleteTransfer - Entered." ));
if( dwTransferMode != LINETRANSFERMODE_TRANSFER )
{
return LINEERR_INVALTRANSFERMODE; // CODEWORK...
}
// retrieve call pointer from handle
pCall=g_pH323Line -> Find2H323CallsAndLock(
hdCall, hdConsultCall, &pConsultCall);
if( pCall == NULL )
{
return LINEERR_INVALCALLHANDLE;
}
if( (pCall -> GetCallState() != LINECALLSTATE_CONNECTED) &&
!pCall -> IsCallOnHold() )
{
H323DBG(( DEBUG_LEVEL_ERROR, "call 0x%08lx not connected.", pCall ));
retVal = LINEERR_INVALCALLSTATE;
goto cleanup;
}
if( (pConsultCall -> GetCallState() != LINECALLSTATE_CONNECTED) &&
!pConsultCall -> IsCallOnHold() )
{
H323DBG(( DEBUG_LEVEL_ERROR, "call 0x%08lx not connected.", pCall ));
retVal = LINEERR_INVALCALLSTATE;
goto cleanup;
}
if( !QueueTAPILineRequest(
TSPI_COMPLETE_TRANSFER,
hdCall,
hdConsultCall,
0,
NULL ))
{
H323DBG(( DEBUG_LEVEL_ERROR, "could not post transfer complete event." ));
retVal = LINEERR_OPERATIONFAILED;
goto cleanup;
}
// complete the async accept operation now
H323CompleteRequest (dwRequestID, ERROR_SUCCESS);
H323DBG(( DEBUG_LEVEL_TRACE, "TSPI_lineCompleteTransfer - Exited." ));
cleanup:
if( pCall != NULL )
{
pCall -> Unlock();
}
if( pConsultCall )
{
pConsultCall -> Unlock();
}
return retVal;
}
/*++
Parameters:
dwRequestID - The identifier of the asynchronous request.
hdCall - The handle to the call to be transferred. The call state of hdCall
can be connected.
htConsultCall - The TAPI handle to the new, temporary consultation call.
The service provider must save this and use it in all subsequent calls
to the LINEEVENT procedure reporting events on the new consultation call.
lphdConsultCall - A pointer to an HDRVCALL representing the service
provider's identifier for the new consultation call. The service
provider must fill this location with its handle for the new
consultation call before this procedure returns. This handle is ignored
by TAPI if the function results in an error. The call state of
hdConsultCall is not applicable.
When setting a call up for transfer, another call (a consultation call)
is automatically allocated to enable the application (through TAPI) to
dial the address (using TSPI_lineDial) of the party to where the call
is to be transferred. The originating party can carry on a conversation
over this consultation call prior to completing the transfer.
This transfer procedure may not be valid for some line devices. Instead
of calling this procedure, TAPI may need to unhold an existing held
call (using TSPI_lineUnhold) to identify the destination of the transfer.
On switches that support cross-address call transfer, the consultation
call can exist on a different address than the call to be transferred.
It may also be necessary to set up the consultation call as an entirely
new call using TSPI_lineMakeCall, to the destination of the transfer.
The transferHeld and transferMake flags in the LINEADDRESSCAPS data
structure report what model the service provider uses.
lpCallParams - A pointer to call parameters to be used when establishing
the consultation call. This parameter can be set to NULL if no special
call setup parameters are desired (the service provider uses defaults).
Return Values:
Returns dwRequestID, or an error number if an error occurs. The lResult
actual parameter of the corresponding ASYNC_COMPLETION is zero if the
function succeeds, or an error number if an error occurs. Possible return
values are as follows:
LINEERR_INVALCALLHANDLE,
LINEERR_INVALBEARERMODE,
LINEERR_INVALCALLSTATE,
LINEERR_INVALRATE,
LINEERR_CALLUNAVAIL,
LINEERR_INVALCALLPARAMS,
LINEERR_NOMEM,
LINEERR_INVALLINESTATE,
LINEERR_OPERATIONUNAVAIL,
LINEERR_INVALMEDIAMODE,
LINEERR_OPERATIONFAILED,
LINEERR_INUSE,
LINEERR_RESOURCEUNAVAIL,
LINEERR_NOMEM,
LINEERR_BEARERMODEUNAVAIL,
LINEERR_RATEUNAVAIL,
LINEERR_INVALADDRESSMODE,
LINEERR_USERUSERINFOTOOBIG.
--*/
LONG
TSPIAPI
TSPI_lineSetupTransfer(
DRV_REQUESTID dwRequestID,
HDRVCALL hdCall,
HTAPICALL htConsultCall,
LPHDRVCALL phdConsultCall,
LPLINECALLPARAMS const lpCallParams
)
{
LONG retVal = (DWORD)dwRequestID;
PH323_CALL pCall = NULL;
PH323_CALL pConsultCall = NULL;
H323_CONFERENCE * pConf = NULL;
BOOL fDelete = FALSE;
DWORD dwCallState;
H323DBG(( DEBUG_LEVEL_TRACE, "TSPI_lineSetupTransfer - Entered." ));
// Acquire the call table lock.
g_pH323Line -> LockCallTable();
pCall = g_pH323Line -> FindH323CallAndLock( hdCall );
if( pCall == NULL )
{
retVal = LINEERR_INVALCALLHANDLE;
goto cleanup;
}
dwCallState = pCall -> GetCallState();
if( (dwCallState != LINECALLSTATE_CONNECTED) &&
!pCall -> IsCallOnLocalHold() )
{
H323DBG(( DEBUG_LEVEL_ERROR, "call 0x%08lx not connected.", pCall ));
retVal = LINEERR_INVALCALLSTATE;
goto cleanup;
}
dwCallState = pCall -> GetCallDiversionState();
if( dwCallState == H4502_CONSULTCALL_INITIATED )
{
H323DBG(( DEBUG_LEVEL_ERROR, "call 0x%08lx already is consulting.",
pCall ));
retVal = LINEERR_INVALCALLSTATE;
goto cleanup;
}
// allocate outgoing call
pConsultCall = new CH323Call();
if( pConsultCall == NULL )
{
H323DBG(( DEBUG_LEVEL_ERROR, "could not allocate outgoing call." ));
// no memory available
retVal = LINEERR_NOMEM;
goto cleanup;
}
// save tapi handle and specify outgoing call direction
if( !pConsultCall -> Initialize( htConsultCall, LINECALLORIGIN_OUTBOUND,
CALLTYPE_TRANSFERING_CONSULT ) )
{
H323DBG(( DEBUG_LEVEL_ERROR, "could not allocate outgoing call." ));
// no memory available
retVal = LINEERR_NOMEM;
goto cleanup;
}
// transfer handle
*phdConsultCall = pConsultCall -> GetCallHandle();
// bind outgoing call
pConf = pConsultCall -> CreateConference(NULL);
if( pConf == NULL )
{
H323DBG(( DEBUG_LEVEL_ERROR,
"could not create conference." ));
// no memory available
retVal = LINEERR_NOMEM;
// failure
goto cleanup;
}
if( !g_pH323Line -> GetH323ConfTable() -> Add(pConf) )
{
H323DBG(( DEBUG_LEVEL_ERROR, "could not add conf to conf table." ));
// no memory available
retVal = LINEERR_NOMEM;
// failure
goto cleanup;
}
// complete the async accept operation now
H323CompleteRequest (dwRequestID, ERROR_SUCCESS);
pConsultCall -> ChangeCallState( LINECALLSTATE_DIALTONE, 0 );
// Put the primary call on hold
pCall-> Hold();
pCall -> SetCallDiversionState( H4502_CONSULTCALL_INITIATED );
pCall -> Unlock();
// Release the call table lock
g_pH323Line -> UnlockCallTable();
// Create a new call. put it in DIALTONE mode and return its handle
H323DBG(( DEBUG_LEVEL_TRACE, "TSPI_lineSetupTransfer - Exited." ));
return retVal;
cleanup:
if( pCall != NULL )
{
pCall -> Unlock();
}
if( pConf != NULL )
{
g_pH323Line -> GetH323ConfTable() -> Remove( pConf );
delete pConf;
pConf = NULL;
}
if( pConsultCall != NULL )
{
pConsultCall -> Shutdown( &fDelete );
H323DBG((DEBUG_LEVEL_TRACE, "call delete:%p.", pCall ));
delete pConsultCall;
pCall = NULL;
}
// Release the call table lock
g_pH323Line -> UnlockCallTable();
return retVal;
}
/*
Parameters:
dwRequestID - The identifier of the asynchronous request.
hdCall - The service provider's handle to the call to be dialed. The call
state of hdCall can be any state except idle and disconnected.
lpszDestAddress - The destination to be dialed using the standard dialable
number format.
dwCountryCode - The country code of the destination. This is used by the
implementation to select the call progress protocols for the destination
address. If a value of 0 is specified, a default call-progress protocol
defined by the service provider is used. This parameter is not validated
by TAPI when this function is called.
Return Values -
Returns dwRequestID or an error number if an error occurs. The lResult
actual parameter of the corresponding ASYNC_COMPLETION is zero if the
function succeeds or an error number if an error occurs. Possible return
values are as follows:
LINEERR_INVALCALLHANDLE,
LINEERR_OPERATIONFAILED,
LINEERR_INVALADDRESS,
LINEERR_RESOURCEUNAVAIL,
LINEERR_INVALCOUNTRYCODE,
LINEERR_DIALBILLING,
LINEERR_INVALCALLSTATE,
LINEERR_DIALQUIET,
LINEERR_ADDRESSBLOCKED,
LINEERR_DIALDIALTONE,
LINEERR_NOMEM,
LINEERR_DIALPROMPT,
LINEERR_OPERATIONUNAVAIL.
Remarks -
The service provider returns LINEERR_INVALCALLSTATE if the current state
of the call does not allow dialing.
The service provider carries out no dialing if it returns
LINEERR_INVALADDRESS.
If the service provider returns LINEERR_DIALBILLING, LINEERR_DIALQUIET,
LINEERR_DIALDIALTONE, or LINEERR_DIALPROMPT, it should perform none of
the actions otherwise performed by TSPI_lineDial (for example, no
partial dialing, and no going offhook). This is because the service
provider should pre-scan the number for unsupported characters first.
TSPI_lineDial is used for dialing on an existing call appearance; for
example, call handles returned from TSPI_lineMakeCall with NULL as the
lpszDestAddress or ending in ';', call handles returned from
TSPI_lineSetupTransfer or TSPI_lineSetupConference. TSPI_lineDial can
be invoked multiple times in the course of dialing in the case of
multistage dialing, if the line's device capabilities permit it.
If the string pointed to by the lpszDestAddress parameter in the
previous call to the TSPI_lineMakeCall or TSPI_lineDial function is
terminated with a semicolon, an empty string in the current call to
TSPI_lineDial indicates that dialing is complete.
Multiple addresses can be provided in a single dial string separated by
CRLF. Service providers that provide inverse multiplexing can establish
individual physical calls with each of the addresses, and return a
single call handle to the aggregate of all calls to the application.
All addresses would use the same country code.
Dialing is considered complete after the address has been accepted by
the service provider, not after the call is finally connected. Service
providers that provide inverse multiplexing may allow multiple addresses
to be provided at once. The service provider must send LINE_CALLSTATE
messages to TAPI to inform it about the progress of the call.
*/
LONG
TSPIAPI
TSPI_lineDial(
DRV_REQUESTID dwRequestID,
HDRVCALL hdCall,
LPCWSTR lpszDestAddress,
DWORD dwCountryCode
)
{
LONG retVal = (DWORD)dwRequestID;
PH323_CALL pCall = NULL;
WORD wAliasType = h323_ID_chosen;
DWORD dwCallState;
WCHAR* wszMachineName;
H323DBG(( DEBUG_LEVEL_TRACE, "TSPI_lineDial - Entered." ));
if( lpszDestAddress == NULL )
{
return LINEERR_INVALPARAM;
}
pCall=g_pH323Line -> FindH323CallAndLock(hdCall);
if( pCall == NULL )
{
return LINEERR_INVALCALLHANDLE;
}
dwCallState = pCall -> GetCallState();
if( dwCallState == LINECALLSTATE_CONNECTED )
{
//no need to dial the call. Inform TAPI3 about it once again.
pCall -> ChangeCallState( LINECALLSTATE_CONNECTED, 0 );
goto func_exit;
}
if( dwCallState != LINECALLSTATE_DIALTONE )
{
H323DBG(( DEBUG_LEVEL_ERROR, "call 0x%08lx no dialtone.", pCall ));
pCall -> Unlock();
return LINEERR_INVALCALLSTATE;
}
if( (*lpszDestAddress==L'T') &&
IsValidE164String((WCHAR*)lpszDestAddress+1) )
{
wAliasType = e164_chosen;
//strip off the leading 'T'
lpszDestAddress++;
}
else if( IsValidE164String( (WCHAR*)lpszDestAddress) )
{
wAliasType = e164_chosen;
}
pCall->SetAddressType( wAliasType );
if( !pCall->SetCalleeAlias( (WCHAR*)lpszDestAddress, wAliasType ) )
{
H323DBG(( DEBUG_LEVEL_ERROR, "could not set callee alias." ));
pCall -> Unlock();
// invalid destination addr
return LINEERR_NOMEM;
}
//set the caller alias name
if( RasIsRegistered() )
{
//ARQ message must have a caller alias
PH323_ALIASNAMES pAliasList = RASGetRegisteredAliasList();
wszMachineName = pAliasList -> pItems[0].pData;
wAliasType = pAliasList -> pItems[0].wType;
}
else
{
wszMachineName = g_pH323Line->GetMachineName();
wAliasType = h323_ID_chosen;
}
//set the value for m_pCallerAliasNames
if( !pCall->SetCallerAlias( wszMachineName, wAliasType ) )
{
H323DBG(( DEBUG_LEVEL_ERROR, "could not set caller alias." ));
pCall -> Unlock();
// invalid destination addr
return LINEERR_NOMEM;
}
if( !pCall -> QueueTAPICallRequest( TSPI_MAKE_CALL, NULL ))
{
H323DBG(( DEBUG_LEVEL_ERROR, "could not post transfer complete event." ));
pCall -> Unlock();
return LINEERR_OPERATIONFAILED;
}
func_exit:
// complete the async accept operation now
H323CompleteRequest (dwRequestID, ERROR_SUCCESS);
//create a new call. put it in DIALTONE mode and return its handle
H323DBG(( DEBUG_LEVEL_TRACE, "TSPI_lineDial - Exited." ));
pCall -> Unlock();
return retVal;
}
LONG
TSPIAPI
TSPI_linePrepareAddToConference(
DRV_REQUESTID dwRequestID,
HDRVCALL hdConfCall,
HTAPICALL htConsultCall,
LPHDRVCALL lphdConsultCall,
LPLINECALLPARAMS const lpCallParams
)
/*++
Routine Description:
This function prepares an existing conference call for the addition of
another party. It creates a new, temporary consultation call. The new
consulatation call can be subsequently added to the conference call.
A conference call handle can be obtained via TSPI_lineSetupConference or
via TSPI_lineCompleteTransfer that is resolved as a three-way conference
call. The function TSPI_linePrepareAddToConference typically places the
existing conference call in the onHoldPendingConference state and creates
a consultation call that can be added later to the existing conference
call via TSPI_lineAddToConference.
The consultation call can be canceled using TSPI_lineDrop. It may also
be possible for the TAPI DLL to swap between the consultation call and
the held conference call via TSPI_lineSwapHold.
The Service Provider initially does media monitoring on the new call for
at least the set of media modes that were monitored for on the line.
Arguments:
dwRequestID - Specifies the identifier of the asynchronous request.
The Service Provider returns this value if the function completes
asynchronously.
hdConfCall - Specifies the Service Provider's opaque handle to a
conference call. Valid call states: connected.
htAddCall - Specifies the TAPI DLL's opaque handle to the new, temporary
consultation call. The Service Provider must save this and use it in
all subsequent calls to the LINEEVENT procedure reporting events on
the new call.
lphdAddCall - Specifies a far pointer to an opaque HDRVCALL representing
the Service Provider's identifier for the new, temporary consultation
call. The Service Provider must fill this location with its opaque
handle for the new call before this procedure returns, whether it
decides to execute the request sychronously or asynchronously. This
handle is invalid if the function results in an error (either
synchronously or asynchronously).
lpCallParams - Specifies a far pointer to call parameters to be used when
establishing the consultation call. This parameter may be set to NULL
if no special call setup parameters are desired.
Return Values:
Returns zero if the function is successful, the (positive) dwRequestID
value if the function will be completed asynchronously, or a negative error
number if an error has occurred. Possible error returns are:
LINEERR_INVALCONFCALLHANDLE - The specified call handle for the
conference call is invalid.
LINEERR_INVALPOINTER - One or more of the specified pointer
parameters are invalid.
LINEERR_INVALCALLSTATE - The conference call is not in a valid state
for the requested operation.
LINEERR_CALLUNAVAIL - All call appearances on the specified address
are currently allocated.
LINEERR_CONFERENCEFULL - The maximum number of parties for a
conference has been reached.
LINEERR_INVALCALLPARAMS - The specified call parameters are invalid.
LINEERR_OPERATIONUNAVAIL - The specified operation is not available.
LINEERR_OPERATIONFAILED - The specified operation failed for
unspecified reason.
--*/
{
return LINEERR_OPERATIONUNAVAIL; // CODEWORK...
}
LONG
TSPIAPI
TSPI_lineSetupConference(
DRV_REQUESTID dwRequestID,
HDRVCALL hdCall,
HDRVLINE hdLine,
HTAPICALL htConfCall,
LPHDRVCALL lphdConfCall,
HTAPICALL htConsultCall,
LPHDRVCALL lphdConsultCall,
DWORD dwNumParties,
LPLINECALLPARAMS const lpCallParams
)
/*++
Routine Description:
This function sets up a conference call for the addition of the third
party.
TSPI_lineSetupConference provides two ways for establishing a new
conference call, depending on whether a normal two-party call is required
to pre-exist or not. When setting up a conference call from an existing
two-party call, the hdCall parameter is a valid call handle that is
initially added to the conference call by the TSPI_lineSetupConference
request and hdLine is ignored. On switches where conference call setup
does not start with an existing call, hdCall must be NULL and hdLine
must be specified to identify the line device on which to initiate the
conference call. In either case, a consultation call is allocated for
connecting to the party that is to be added to the call. The TAPI DLL
can use TSPI_lineDial to dial the address of the other party.
The conference call will typically transition into the
onHoldPendingConference state, the consultation call dialtone state and
the initial call (if one) into the conferenced state.
A conference call can also be set up via a TSPI_lineCompleteTransfer that
is resolved into a three-way conference.
The TAPI DLL may be able to toggle between the consultation call and the
conference call by using TSPI_lineSwapHold.
Arguments:
dwRequestID - Specifies the identifier of the asynchronous request.
The Service Provider returns this value if the function completes
asynchronously.
hdCall - Specifies the Service Provider's opaque handle to the initial
call that identifies the first party of a conference call. In some
environments, a call must exist in order to start a conference call.
In other telephony environments, no call initially exists and hdCall
is left NULL. Valid call states: connected.
hdLine - Specifies the Service Provider's opaque handle to the line device
on which to originate the conference call if hdCall is NULL. The
hdLine parameter is ignored if hdCall is non-NULL. The Service
Provider reports which model it supports through the setupConfNull
flag of the LINEADDRESSCAPS data structure.
htConfCall - Specifies the TAPI DLL's opaque handle to the new conference
call. The Service Provider must save this and use it in all subsequent
calls to the LINEEVENT procedure reporting events on the new call.
lphdConfCall - Specifies a far pointer to an opaque HDRVCALL representing
the Service Provider's identifier for the newly created conference
call. The Service Provider must fill this location with its opaque
handle for the new call before this procedure returns, whether it
decides to execute the request sychronously or asynchronously. This
handle is invalid if the function results in an error (either
synchronously or asynchronously).
htAddCall - Specifies the TAPI DLL's opaque handle to a new call. When
setting up a call for the addition of a new party, a new temporary call
(consultation call) is automatically allocated. The Service Provider
must save the htAddCall and use it in all subsequent calls to the
LINEEVENT procedure reporting events on the new consultation call.
lphdAddCall - Specifies a far pointer to an opaque HDRVCALL representing
the Service Provider's identifier for a call. When setting up a call
for the addition of a new party, a new temporary call (consultation
call) is automatically allocated. The Service Provider must fill this
location with its opaque handle for the new consultation call before
this procedure returns, whether it decides to execute the request
sychronously or asynchronously. This handle is invalid if the
function results in an error (either synchronously or asynchronously).
dwNumParties - Specifies the expected number of parties in the conference
call. The service provider is free to do with this number as it
pleases; ignore it, use it a hint to allocate the right size
conference bridge inside the switch, etc.
lpCallParams - Specifies a far pointer to call parameters to be used when
establishing the consultation call. This parameter may be set to NULL
if no special call setup parameters are desired.
Return Values:
Returns zero if the function is successful, the (positive) dwRequestID
value if the function will be completed asynchronously, or a negative error
number if an error has occurred. Possible error returns are:
LINEERR_INVALCALLHANDLE - The specified call handle for the conference
call is invalid. This error may also indicate that the telephony
environment requires an initial call to set up a conference but a
NULL call handle was supplied.
LINEERR_INVALLINEHANDLE - The specified line handle for the line
containing the conference call is invalid. This error may also
indicate that the telephony environment requires an initial line
to set up a conference but a non-NULL call handle was supplied
instead.
LINEERR_INVALCALLSTATE - The call is not in a valid state for the
requested operation.
LINEERR_CALLUNAVAIL - All call appearances on the specified address
are currently allocated.
LINEERR_CONFERENCEFULL - The requested number of parties cannot be
satisfied.
LINEERR_INVALPOINTER - One or more of the specified pointer
parameters are invalid.
LINEERR_INVALCALLPARAMS - The specified call parameters are invalid.
LINEERR_OPERATIONUNAVAIL - The specified operation is not available.
LINEERR_OPERATIONFAILED - The specified operation failed for
unspecified reason.
--*/
{
return LINEERR_OPERATIONUNAVAIL; // CODEWORK...
}
LONG
TSPIAPI
TSPI_lineRemoveFromConference(
DRV_REQUESTID dwRequestID,
HDRVCALL hdCall
)
/*++
Routine Description:
This function removes the specified call from the conference call to
which it currently belongs. The remaining calls in the conference
call are unaffected.
This operation removes a party that currently belongs to a conference
call. After the call has been successfully removed, it may be possible
to further manipulate it using its handle. The availability of this
operation and its result are likely to be limited in many
implementations. For example, in many implementations, only the most
recently added party may be removed from a conference, and the removed
call may be automatically dropped (becomes idle). Consult the line's
device capabilities to determine the available effects of removing a
call from a conference.
Arguments:
dwRequestID - Specifies the identifier of the asynchronous request.
The Service Provider returns this value if the function completes
asynchronously.
hdCall - Specifies the Service Provider's opaque handle to the call
to be removed from the conference. Valid call states: conferenced.
Return Values:
Returns zero if the function is successful, the (positive) dwRequestID
value if the function will be completed asynchronously, or a negative error
number if an error has occurred. Possible error returns are:
LINEERR_INVALCALLHANDLE - The specified call handle is invalid.
LINEERR_INVALCALLSTATE - The call is not in a valid state for the
requested operation.
LINEERR_OPERATIONUNAVAIL - The specified operation is not available.
LINEERR_OPERATIONFAILED - The specified operation failed for
unspecified reasons.
--*/
{
return LINEERR_OPERATIONUNAVAIL; // CODEWORK...
}
/*
Parameters:
dwRequestID - The identifier of the asynchronous request.
hdLine - The service provider's handle to the line to be forwarded.
bAllAddresses - Specifies whether all originating addresses on the line or
just the one specified is to be forwarded. If TRUE, all addresses on
the line are forwarded and dwAddressID is ignored; if FALSE, only the
address specified as dwAddressID is forwarded. This parameter is not
validated by TAPI when this function is called.
dwAddressID - The address on the specified line whose incoming calls are to
be forwarded. This parameter is ignored if bAllAddresses is TRUE. This
parameter is not validated by TAPI when this function is called.
lpForwardList - A pointer to a variably sized data structure of type
LINEFORWARDLIST that describes the specific forwarding instructions.
dwNumRingsNoAnswer - Specifies the number of rings before an incoming call
is considered a "no answer." If dwNumRingsNoAnswer is out of range, the
actual value is set to the nearest value in the allowable range. This
parameter is not validated by TAPI when this function is called.
htConsultCall - The TAPI handle to a new call, if such a call must be created
by the service provider. In some telephony environments, forwarding a
call has the side effect of creating a consultation call used to consult
the party that is being forwarded to. In such an environment, the service
provider creates the new consutation call and must save this value and
use it in all subsequent calls to the LINEEVENT procedure reporting
events on the call. If no consultation call is created, this value can
be ignored by the service provider.
lphdConsultCall - A pointer to an HDRVCALL representing the service
provider's identifier for the call. In telephony environments where
forwarding a call has the side effect of creating a consultation call
used to consult the party that is being forwarded to, the service
provider must fill this location with its handle for the call before
this procedure returns. The service provider is permitted to do callbacks
regarding the new call before it returns from this procedure. If no
consultation call is created, the HDRVCALL must be left NULL.
lpCallParams - A pointer to a structure of type LINECALLPARAMS. This pointer
is ignored by the service provider unless lineForward requires the
establishment of a call to the forwarding destination (and
lphdConsultCall is returned, in which case lpCallParams is optional).
If NULL, default call parameters are used. Otherwise, the specified call
parameters are used for establishing htConsultCall.
Return Values:
Returns dwRequestID or an error number if an error occurs. The lResult
actual parameter of the corresponding ASYNC_COMPLETION is zero if the
function succeeds or an error number if an error occurs. Possible return
values are as follows:
LINEERR_INVALLINEHANDLE,
LINEERR_NOMEM,
LINEERR_INVALADDRESS,
LINEERR_OPERATIONUNAVAIL,
LINEERR_INVALADDRESSID,
LINEERR_OPERATIONFAILED,
LINEERR_INVALCOUNTRYCODE,
LINEERR_RESOURCEUNAVAIL,
LINEERR_INVALPARAM,
LINEERR_STRUCTURETOOSMALL.
Remarks
*/
LONG
TSPIAPI TSPI_lineForward(
DRV_REQUESTID dwRequestID,
HDRVLINE hdLine,
DWORD bAllAddresses,
DWORD dwAddressID,
LPLINEFORWARDLIST const lpForwardList,
DWORD dwNumRingsNoAnswer,
HTAPICALL htConsultCall,
LPHDRVCALL lphdConsultCall,
LPLINECALLPARAMS const lpCallParams
)
{
DWORD dwStatus = dwRequestID;
PH323_CALL pCall = NULL;
H323_CONFERENCE * pConf = NULL;
BOOL fDelete = FALSE;
DWORD dwState;
PVOID pForwardParams = NULL;
DWORD event = NULL;
H323DBG(( DEBUG_LEVEL_TRACE, "TSPI_lineForward - Entered." ));
//lock the line device
g_pH323Line -> Lock();
if( hdLine != g_pH323Line -> GetHDLine() )
{
g_pH323Line ->Unlock();
return LINEERR_RESOURCEUNAVAIL;
}
// validate line state
dwState = g_pH323Line -> GetState();
if( ( dwState != H323_LINESTATE_OPENED) &&
( dwState != H323_LINESTATE_LISTENING) )
{
H323DBG(( DEBUG_LEVEL_ERROR, "H323 line is not currently opened:%d.",
dwState ));
// release line device
g_pH323Line ->Unlock();
// line needs to be opened
return LINEERR_INVALLINESTATE;
}
if( lpForwardList == NULL )
{
//forwarding is disabled
g_pH323Line -> DisableCallForwarding();
g_pH323Line ->Unlock();
*lphdConsultCall = NULL;
//inform the user about change in line forward state
(*g_pfnLineEventProc)(
g_pH323Line->m_htLine,
(HTAPICALL)NULL,
(DWORD)LINE_ADDRESSSTATE,
(DWORD)LINEADDRESSSTATE_FORWARD,
(DWORD)LINEADDRESSSTATE_FORWARD,
(DWORD)0
);
// complete the async accept operation now
H323CompleteRequest (dwRequestID, ERROR_SUCCESS);
return dwRequestID;
}
// allocate outgoing call
pCall = new CH323Call();
if( pCall == NULL )
{
H323DBG(( DEBUG_LEVEL_ERROR, "could not allocate outgoing call." ));
// no memory available
dwStatus = LINEERR_NOMEM;
goto cleanup;
}
// save tapi handle and specify outgoing call direction
if( !pCall -> Initialize( htConsultCall, LINECALLORIGIN_OUTBOUND,
CALLTYPE_FORWARDCONSULT ) )
{
H323DBG(( DEBUG_LEVEL_ERROR, "could not allocate outgoing call." ));
// no memory available
dwStatus = LINEERR_NOMEM;
goto cleanup;
}
dwStatus = pCall -> ValidateForwardParams(
lpForwardList, &pForwardParams, &event );
if( dwStatus != ERROR_SUCCESS )
{
// failure
goto cleanup;
}
_ASSERTE( event );
// transfer handle
*lphdConsultCall = (HDRVCALL)NULL /*pCall -> GetCallHandle()*/;
// bind outgoing call
pConf = pCall -> CreateConference(NULL);
if( pConf == NULL )
{
H323DBG(( DEBUG_LEVEL_ERROR,
"could not create conference." ));
// no memory available
dwStatus = LINEERR_NOMEM;
// failure
goto cleanup;
}
if( !g_pH323Line -> GetH323ConfTable() -> Add(pConf) )
{
H323DBG(( DEBUG_LEVEL_ERROR, "could not add conf to conf table." ));
// no memory available
dwStatus = LINEERR_NOMEM;
// failure
goto cleanup;
}
pCall -> Lock();
// post line forward request to callback thread
if( !pCall->QueueTAPICallRequest( event, (PVOID)pForwardParams ) )
{
H323DBG(( DEBUG_LEVEL_ERROR, "could not post forward message." ));
// could not complete operation
dwStatus = LINEERR_OPERATIONFAILED;
pCall-> Unlock();
// failure
goto cleanup;
}
g_pH323Line -> m_fForwardConsultInProgress = TRUE;
if( (dwNumRingsNoAnswer >= H323_NUMRINGS_LO) &&
(dwNumRingsNoAnswer <= H323_NUMRINGS_HI) )
{
g_pH323Line -> m_dwNumRingsNoAnswer = dwNumRingsNoAnswer;
}
// complete the async accept operation now
H323CompleteRequest (dwRequestID, ERROR_SUCCESS);
//unlock the call object.
pCall-> Unlock();
// release line device
g_pH323Line -> Unlock();
H323DBG(( DEBUG_LEVEL_TRACE, "TSPI_lineForward - Exited." ));
// success
return dwRequestID;
cleanup:
if( pCall != NULL )
{
pCall -> Shutdown( &fDelete );
H323DBG((DEBUG_LEVEL_TRACE, "call delete:%p.", pCall ));
delete pCall;
pCall = NULL;
}
*lphdConsultCall = NULL;
if( pForwardParams != NULL )
{
delete pForwardParams;
pForwardParams = NULL;
}
// release line device
g_pH323Line -> Unlock();
// failure
return dwStatus;
}
LONG
TSPIAPI TSPI_lineRedirect(
DRV_REQUESTID dwRequestID,
HDRVCALL hdCall,
LPCWSTR lpszDestAddress,
DWORD dwCountryCode
)
{
LONG retVal = (DWORD)dwRequestID;
PH323_CALL pCall = NULL;
CALLREROUTINGINFO* pCallReroutingInfo = NULL;
WORD wAliasType = h323_ID_chosen;
DWORD dwMaxAddrSize = MAX_H323_ADDR_LEN;
DWORD dwAddrLen;
H323DBG(( DEBUG_LEVEL_TRACE, "TSPI_lineRedirect - Entered." ));
if( lpszDestAddress == NULL )
return LINEERR_INVALPARAM;
pCall=g_pH323Line -> FindH323CallAndLock(hdCall);
if( pCall == NULL )
{
return LINEERR_INVALCALLHANDLE;
}
if( pCall -> GetCallState() != LINECALLSTATE_OFFERING )
{
H323DBG(( DEBUG_LEVEL_ERROR, "call 0x%08lx not ringback.", pCall ));
pCall -> Unlock();
return LINEERR_INVALCALLSTATE;
}
if( (*lpszDestAddress==L'T') &&
IsValidE164String((WCHAR*)lpszDestAddress+1) )
{
wAliasType = e164_chosen;
//strip off the leading 'T'
lpszDestAddress++;
dwMaxAddrSize = MAX_E164_ADDR_LEN;
}
else if( IsValidE164String( (WCHAR*)lpszDestAddress) )
{
wAliasType = e164_chosen;
dwMaxAddrSize = MAX_E164_ADDR_LEN;
}
dwAddrLen = wcslen(lpszDestAddress);
if( (dwAddrLen > dwMaxAddrSize) || (dwAddrLen == 0) )
{
return LINEERR_INVALPARAM;
}
retVal = pCall -> SetDivertedToAlias( (WCHAR*)lpszDestAddress, wAliasType );
if( retVal != NOERROR )
{
pCall->Unlock();
return retVal;
}
if( !pCall -> QueueTAPICallRequest( TSPI_CALL_DIVERT, NULL ))
{
H323DBG(( DEBUG_LEVEL_ERROR, "could not post call divert event." ));
pCall -> Unlock();
return LINEERR_OPERATIONFAILED;
}
// complete the async accept operation now
H323CompleteRequest (dwRequestID, ERROR_SUCCESS);
pCall -> Unlock();
H323DBG(( DEBUG_LEVEL_TRACE, "TSPI_lineRedirect - Exited." ));
return dwRequestID;
}
LONG TSPIAPI TSPI_lineUnhold (
DRV_REQUESTID dwRequestID,
HDRVCALL hdCall
)
{
LONG retVal = (DWORD)dwRequestID;
PH323_CALL pCall = NULL;
H323DBG(( DEBUG_LEVEL_TRACE, "TSPI_lineUnHold - Entered." ));
pCall=g_pH323Line -> FindH323CallAndLock(hdCall);
if( pCall == NULL )
{
return LINEERR_INVALCALLHANDLE;
}
if( pCall -> IsCallOnHold() == FALSE )
{
H323DBG(( DEBUG_LEVEL_ERROR, "call 0x%08lx not ringback.", pCall ));
pCall -> Unlock();
return LINEERR_INVALCALLSTATE;
}
if( !pCall -> QueueTAPICallRequest( TSPI_CALL_UNHOLD, NULL ))
{
H323DBG(( DEBUG_LEVEL_ERROR, "could not post transfer complete event." ));
pCall -> Unlock();
return LINEERR_OPERATIONFAILED;
}
// complete the async accept operation now
H323CompleteRequest (dwRequestID, ERROR_SUCCESS);
pCall -> Unlock();
H323DBG(( DEBUG_LEVEL_TRACE, "TSPI_lineUnHold - Exited." ));
return dwRequestID;
}
LONG TSPIAPI TSPI_lineHold(
DRV_REQUESTID dwRequestID,
HDRVCALL hdCall
)
{
LONG retVal = (DWORD)dwRequestID;
PH323_CALL pCall = NULL;
H323DBG(( DEBUG_LEVEL_TRACE, "TSPI_lineHold - Entered." ));
pCall=g_pH323Line -> FindH323CallAndLock(hdCall);
if( pCall == NULL )
{
return LINEERR_INVALCALLHANDLE;
}
if( (pCall -> GetCallState() != LINECALLSTATE_CONNECTED) ||
(pCall -> IsCallOnHold()) )
{
H323DBG(( DEBUG_LEVEL_ERROR, "call 0x%08lx not ringback.", pCall ));
pCall -> Unlock();
return LINEERR_INVALCALLSTATE;
}
if( !pCall -> QueueTAPICallRequest( TSPI_CALL_HOLD, NULL ))
{
H323DBG(( DEBUG_LEVEL_ERROR, "could not post transfer complete event." ));
pCall -> Unlock();
return LINEERR_OPERATIONFAILED;
}
// complete the async accept operation now
H323CompleteRequest (dwRequestID, ERROR_SUCCESS);
pCall -> Unlock();
H323DBG(( DEBUG_LEVEL_TRACE, "TSPI_lineHold - Exited." ));
return retVal;
}
BOOL
CH323Call::ResolveToIPAddress(
IN WCHAR* pwszAddr,
IN SOCKADDR_IN* psaAddr
)
{
CHAR szDelimiters[] = "@ \t\n";
CHAR szAddr[H323_MAXDESTNAMELEN+1];
LPSTR pszUser = NULL;
LPSTR pszDomain = NULL;
DWORD dwIPAddr;
struct hostent* pHost;
H323DBG(( DEBUG_LEVEL_ERROR, "ResolveToIPAddress entered:%p.",this ));
ZeroMemory( psaAddr, sizeof(SOCKADDR) );
// validate pointerr
if (pwszAddr == NULL)
{
H323DBG(( DEBUG_LEVEL_ERROR, "null destination address." ));
// failure
return FALSE;
}
// convert address from unicode
if (WideCharToMultiByte(
CP_ACP,
0,
pwszAddr,
-1,
szAddr,
sizeof(szAddr),
NULL,
NULL
) == 0)
{
H323DBG(( DEBUG_LEVEL_ERROR,
"could not convert address from unicode." ));
// failure
return FALSE;
}
// check whether phone number has been specified
if( IsPhoneNumber( szAddr ) )
{
// need to direct call to pstn gateway
if ((g_RegistrySettings.fIsGatewayEnabled == FALSE) ||
(g_RegistrySettings.gatewayAddr.nAddrType == 0))
{
H323DBG(( DEBUG_LEVEL_ERROR, "pstn gateway not specified." ));
// failure
return FALSE;
}
psaAddr->sin_family = AF_INET;
psaAddr->sin_addr.S_un.S_addr =
htonl( g_RegistrySettings.gatewayAddr.Addr.IP_Binary.dwAddr );
psaAddr->sin_port = g_RegistrySettings.gatewayAddr.Addr.IP_Binary.wPort;
return TRUE;
}
// parse user name
pszUser = strtok(szAddr, szDelimiters);
// parse domain name
pszDomain = strtok(NULL, szDelimiters);
// validate pointer
if (pszUser == NULL)
{
H323DBG(( DEBUG_LEVEL_ERROR,
"could not parse destination address." ));
// failure
return FALSE;
}
// validate pointer
if (pszDomain == NULL)
{
// switch pointers
pszDomain = pszUser;
// re-initialize
pszUser = NULL;
}
H323DBG(( DEBUG_LEVEL_VERBOSE,
"resolving user %s domain %s.",
pszUser, pszDomain ));
// attempt to convert ip address
dwIPAddr = inet_addr(szAddr);
// see if address converted
if( dwIPAddr == INADDR_NONE )
{
// attempt to lookup hostname
pHost = gethostbyname(szAddr);
// validate pointer
if( pHost != NULL )
{
// retrieve host address from structure
dwIPAddr = *(unsigned long *)pHost->h_addr;
}
}
// see if address converted
if( dwIPAddr == INADDR_NONE )
{
H323DBG(( DEBUG_LEVEL_ERROR,
"error 0x%08lx resolving IP address.",
WSAGetLastError() ));
// make sure proxy has been specified
if ((g_RegistrySettings.fIsProxyEnabled == FALSE) ||
(g_RegistrySettings.proxyAddr.nAddrType == 0))
{
H323DBG(( DEBUG_LEVEL_ERROR, "proxy not specified." ));
// failure
return FALSE;
}
psaAddr->sin_family = AF_INET;
psaAddr->sin_addr.S_un.S_addr =
htonl( g_RegistrySettings.proxyAddr.Addr.IP_Binary.dwAddr );
psaAddr->sin_port = g_RegistrySettings.proxyAddr.Addr.IP_Binary.wPort;
return TRUE;
}
// save converted address
psaAddr->sin_family = AF_INET;
psaAddr->sin_addr.S_un.S_addr = dwIPAddr;
H323DBG(( DEBUG_LEVEL_TRACE,
"callee address resolved to %s:%d.",
H323AddrToString(dwIPAddr),
m_CalleeAddr.Addr.IP_Binary.wPort ));
return TRUE;
}
//!!always called in a lock
LONG
CH323Line::CopyAddressForwardInfo(
IN LPLINEADDRESSSTATUS lpAddressStatus
)
{
LINEFORWARD * lineForwardStructArray;
LPFORWARDADDRESS pForwardedAddress;
if( m_pCallForwardParams-> fForwardForAllOrigins == TRUE )
{
lpAddressStatus->dwForwardNumEntries = 1;
lpAddressStatus->dwForwardSize = sizeof(LINEFORWARD) +
sizeof(WCHAR) * (m_pCallForwardParams->divertedToAlias.wDataLength+1);
}
else
{
lpAddressStatus->dwForwardNumEntries = 0;
pForwardedAddress = m_pCallForwardParams->pForwardedAddresses;
while( pForwardedAddress )
{
lpAddressStatus->dwForwardNumEntries++;
lpAddressStatus->dwForwardSize += sizeof(LINEFORWARD) +
sizeof(WCHAR) * (pForwardedAddress->callerAlias.wDataLength+1) +
sizeof(WCHAR) * (pForwardedAddress->divertedToAlias.wDataLength+1);
pForwardedAddress = pForwardedAddress->next;
}
}
lpAddressStatus->dwNeededSize += lpAddressStatus->dwForwardSize;
if( lpAddressStatus->dwTotalSize < lpAddressStatus->dwNeededSize )
{
return LINEERR_STRUCTURETOOSMALL;
}
lineForwardStructArray = (LINEFORWARD*)
((BYTE*)lpAddressStatus + lpAddressStatus->dwUsedSize);
lpAddressStatus->dwUsedSize += lpAddressStatus->dwForwardNumEntries *
sizeof(LINEFORWARD);
if( m_pCallForwardParams-> fForwardForAllOrigins == TRUE )
{
//copy the first structure
lineForwardStructArray[0].dwForwardMode =
m_pCallForwardParams->dwForwardTypeForAllOrigins;
//copy dest address alias
lineForwardStructArray[0].dwDestAddressOffset =
lpAddressStatus->dwUsedSize;
lineForwardStructArray[0].dwDestAddressSize =
sizeof(WCHAR)* (m_pCallForwardParams->divertedToAlias.wDataLength + 1);
CopyMemory(
(PVOID)((BYTE*)lpAddressStatus+lpAddressStatus->dwUsedSize),
m_pCallForwardParams->divertedToAlias.pData,
lineForwardStructArray[0].dwDestAddressSize );
lpAddressStatus->dwUsedSize +=
lineForwardStructArray[0].dwDestAddressSize;
lineForwardStructArray[0].dwDestCountryCode = 0;
}
else
{
pForwardedAddress = m_pCallForwardParams->pForwardedAddresses;
for( DWORD indexI = 0; indexI < lpAddressStatus->dwForwardNumEntries; indexI++ )
{
_ASSERTE( pForwardedAddress );
lineForwardStructArray[indexI].dwForwardMode =
pForwardedAddress->dwForwardType;
//copy caller address alias
lineForwardStructArray[indexI].dwCallerAddressOffset =
lpAddressStatus->dwUsedSize;
lineForwardStructArray[indexI].dwCallerAddressSize =
sizeof(WCHAR)* (pForwardedAddress->callerAlias.wDataLength + 1);
CopyMemory(
(PVOID)((BYTE*)lpAddressStatus+lpAddressStatus->dwUsedSize),
(PVOID)pForwardedAddress->callerAlias.pData,
lineForwardStructArray[indexI].dwCallerAddressSize );
lpAddressStatus->dwUsedSize +=
lineForwardStructArray[indexI].dwCallerAddressSize;
//copy dest address alias
lineForwardStructArray[indexI].dwDestAddressOffset =
lpAddressStatus->dwUsedSize;
lineForwardStructArray[indexI].dwDestAddressSize =
sizeof(WCHAR)* (pForwardedAddress->divertedToAlias.wDataLength + 1);
CopyMemory(
(PVOID)((BYTE*)lpAddressStatus+lpAddressStatus->dwUsedSize),
pForwardedAddress->divertedToAlias.pData,
lineForwardStructArray[indexI].dwDestAddressSize);
lpAddressStatus->dwUsedSize +=
lineForwardStructArray[indexI].dwDestAddressSize;
lineForwardStructArray[indexI].dwDestCountryCode = 0;
pForwardedAddress = pForwardedAddress->next;
}
}
_ASSERTE( lpAddressStatus->dwUsedSize == lpAddressStatus->dwNeededSize);
return NOERROR;
}
//!!always called in a lock
LONG
CH323Call::ValidateForwardParams(
IN LPLINEFORWARDLIST lpLineForwardList,
OUT PVOID* ppForwardParams,
OUT DWORD* pEvent
)
{
LPLINEFORWARD pLineForwardStruct;
LPWSTR pwszDestAddr = NULL;
LPWSTR pAllocAddrBuffer = NULL;
DWORD dwStatus = ERROR_SUCCESS;
CALLFORWARDPARAMS * pCallForwardParams = NULL;
FORWARDADDRESS * pForwardAddress = NULL;
WORD wAliasType = h323_ID_chosen;
DWORD dwMaxAddrSize = MAX_H323_ADDR_LEN;
DWORD dwAddrLen;
*pEvent = 0;
if( (lpLineForwardList->dwNumEntries == 0) ||
(lpLineForwardList->dwTotalSize == 0) )
{
return LINEERR_INVALPARAM;
}
pLineForwardStruct = &(lpLineForwardList->ForwardList[0]);
if( pLineForwardStruct->dwDestAddressSize == 0 )
{
return LINEERR_INVALPARAM;
}
//resolve the diverted-to address
pAllocAddrBuffer = pwszDestAddr =
(WCHAR*)new BYTE[pLineForwardStruct->dwDestAddressSize];
if( pwszDestAddr == NULL )
{
return LINEERR_NOMEM;
}
CopyMemory( pwszDestAddr,
(BYTE*)lpLineForwardList + pLineForwardStruct->dwDestAddressOffset,
pLineForwardStruct->dwDestAddressSize );
//If negotiated version is 3.1 get the alias type
if( g_dwTSPIVersion >= 0x00030001 )
{
switch( pLineForwardStruct->dwDestAddressType )
{
case LINEADDRESSTYPE_PHONENUMBER:
wAliasType = e164_chosen;
if( *pwszDestAddr == L'T' )
{
//strip off the leading 'T'
pwszDestAddr++;
}
if( IsValidE164String( (WCHAR*)pwszDestAddr) == FALSE )
{
delete pAllocAddrBuffer;
return LINEERR_INVALPARAM;
}
dwMaxAddrSize = MAX_E164_ADDR_LEN;
break;
case LINEADDRESSTYPE_DOMAINNAME:
case LINEADDRESSTYPE_IPADDRESS:
wAliasType = h323_ID_chosen;
break;
default:
H323DBG(( DEBUG_LEVEL_VERBOSE, "Wrong address type:.",
pLineForwardStruct->dwDestAddressType ));
delete pAllocAddrBuffer;
return LINEERR_INVALPARAM;
}
}
else
{
if( (*pwszDestAddr==L'T') &&
IsValidE164String((WCHAR*)pwszDestAddr+1) )
{
wAliasType = e164_chosen;
//strip off the leading 'T'
pwszDestAddr++;
dwMaxAddrSize = MAX_E164_ADDR_LEN;
}
else if( IsValidE164String( (WCHAR*)pwszDestAddr) )
{
wAliasType = e164_chosen;
dwMaxAddrSize = MAX_E164_ADDR_LEN;
}
}
dwAddrLen = wcslen( pwszDestAddr );
if( (dwAddrLen > dwMaxAddrSize) || (dwAddrLen == 0) )
{
delete pAllocAddrBuffer;
return LINEERR_INVALPARAM;
}
m_dwAddressType = wAliasType;
//don't resolve the address if GK enabled
if( !ResolveAddress( pwszDestAddr ) )
{
if( !RasIsRegistered())
{
delete pAllocAddrBuffer;
return LINEERR_INVALPARAM;
}
}
else if(m_CallerAddr.Addr.IP_Binary.dwAddr == HOST_LOCAL_IP_ADDR_INTERFACE )
{
delete pAllocAddrBuffer;
return LINEERR_INVALPARAM;
}
if( !SetCalleeAlias( (WCHAR*)pwszDestAddr, wAliasType ) )
{
H323DBG(( DEBUG_LEVEL_ERROR, "could not set callee alias." ));
delete pAllocAddrBuffer;
return LINEERR_NOMEM;
}
switch( pLineForwardStruct->dwForwardMode )
{
case LINEFORWARDMODE_UNCOND:
case LINEFORWARDMODE_BUSY:
case LINEFORWARDMODE_NOANSW:
case LINEFORWARDMODE_BUSYNA:
pCallForwardParams = new CALLFORWARDPARAMS;
if( pCallForwardParams == NULL )
{
delete pAllocAddrBuffer;
return LINEERR_NOMEM;
}
ZeroMemory( pCallForwardParams, sizeof(CALLFORWARDPARAMS) );
//Forward all calls unconditionally, irrespective of their origin.
pCallForwardParams->fForwardForAllOrigins = TRUE;
pCallForwardParams->dwForwardTypeForAllOrigins =
pLineForwardStruct->dwForwardMode;
//set the diverted-to alias
pCallForwardParams->divertedToAlias.wType = wAliasType;
pCallForwardParams->divertedToAlias.wPrefixLength = 0;
pCallForwardParams->divertedToAlias.pPrefix = NULL;
pCallForwardParams->divertedToAlias.wDataLength =
(WORD)wcslen(pwszDestAddr);// UNICODE character count
pCallForwardParams->divertedToAlias.pData = pwszDestAddr;
//enable forwarding
pCallForwardParams->fForwardingEnabled = TRUE;
*ppForwardParams = (PVOID)pCallForwardParams;
*pEvent = TSPI_LINEFORWARD_NOSPECIFIC;
break;
case LINEFORWARDMODE_BUSYNASPECIFIC:
case LINEFORWARDMODE_UNCONDSPECIFIC:
case LINEFORWARDMODE_BUSYSPECIFIC:
case LINEFORWARDMODE_NOANSWSPECIFIC:
if( pLineForwardStruct-> dwCallerAddressSize == 0 )
{
delete pAllocAddrBuffer;
return LINEERR_INVALPARAM;
}
/*if( g_pH323Line->ForwardEnabledForAllOrigins() )
{
//specific forward can't be enabled when non-specific forward for
//all caller addresses is enabled
delete pAllocAddrBuffer;
return LINEERR_INVALPARAM;
}*/
pForwardAddress = new FORWARDADDRESS;
if( pForwardAddress == NULL )
{
delete pAllocAddrBuffer;
return LINEERR_NOMEM;
}
ZeroMemory( pForwardAddress, sizeof(FORWARDADDRESS) );
pForwardAddress->dwForwardType = pLineForwardStruct->dwForwardMode;
//set the caller alias(alias to be forward).
pForwardAddress->callerAlias.wType = h323_ID_chosen;
pForwardAddress->callerAlias.pData =
(WCHAR*)new BYTE[pLineForwardStruct-> dwCallerAddressSize];
if( pForwardAddress->callerAlias.pData == NULL )
{
delete pForwardAddress;
delete pAllocAddrBuffer;
return LINEERR_NOMEM;
}
CopyMemory( pForwardAddress->callerAlias.pData,
(BYTE*)lpLineForwardList + pLineForwardStruct->dwCallerAddressOffset,
pLineForwardStruct-> dwCallerAddressSize );
pForwardAddress->callerAlias.wDataLength =
(WORD)wcslen( pForwardAddress->callerAlias.pData );
//set the diverted-to alias.
pForwardAddress->divertedToAlias.wType = wAliasType;
pForwardAddress->divertedToAlias.wDataLength =
(WORD)wcslen( pwszDestAddr );
pForwardAddress->divertedToAlias.pData = pwszDestAddr;
//set the diverted-to address
/*pForwardAddress->saDivertedToAddr.sin_family = AF_INET;
pForwardAddress->saDivertedToAddr.sin_addr.S_un.S_addr
= htonl( m_CalleeAddr.Addr.IP_Binary.dwAddr );
pForwardAddress->saDivertedToAddr.sin_port =
htons( m_CalleeAddr.Addr.IP_Binary.wPort );*/
//here we could have a for loop and process multiple pLineForwardStructs
//post the event for line forward
*ppForwardParams = (PVOID)pForwardAddress;
*pEvent = TSPI_LINEFORWARD_SPECIFIC;
break;
case LINEFORWARDMODE_UNCONDINTERNAL:
case LINEFORWARDMODE_UNCONDEXTERNAL:
case LINEFORWARDMODE_BUSYINTERNAL:
case LINEFORWARDMODE_BUSYEXTERNAL:
case LINEFORWARDMODE_NOANSWINTERNAL:
case LINEFORWARDMODE_NOANSWEXTERNAL:
case LINEFORWARDMODE_BUSYNAINTERNAL:
case LINEFORWARDMODE_BUSYNAEXTERNAL:
delete pAllocAddrBuffer;
return LINEERR_INVALPARAM;
default:
delete pAllocAddrBuffer;
return LINEERR_INVALPARAM;
}
return ERROR_SUCCESS;
}
void
CH323Call::Forward(
DWORD event,
PVOID pForwardInfo
)
{
WCHAR * pwszDialableAddr;
WCHAR * wszMachineName;
WORD wAliasType = h323_ID_chosen;
if( event == TSPI_LINEFORWARD_SPECIFIC )
{
m_pForwardAddress = (LPFORWARDADDRESS)pForwardInfo;
pwszDialableAddr = m_pForwardAddress->divertedToAlias.pData;
wAliasType = m_pForwardAddress->divertedToAlias.wType;
}
else if( event == TSPI_LINEFORWARD_NOSPECIFIC )
{
m_pCallForwardParams = (CALLFORWARDPARAMS*)pForwardInfo;
pwszDialableAddr = m_pCallForwardParams->divertedToAlias.pData;
wAliasType = m_pCallForwardParams->divertedToAlias.wType;
}
else
{
// Wrong event, shouldn't get
// here at all...
return;
}
_ASSERTE( pwszDialableAddr );
//set the values for m_pCalleeAliasNames
if( !AddAliasItem( m_pCalleeAliasNames,
(BYTE*)(pwszDialableAddr),
sizeof(WCHAR) * (wcslen(pwszDialableAddr) + 1 ),
wAliasType ) )
{
H323DBG(( DEBUG_LEVEL_ERROR, "could not allocate callee name." ));
DropCall( 0 );
}
if( RasIsRegistered() )
{
PH323_ALIASNAMES pAliasList = RASGetRegisteredAliasList();
H323DBG(( DEBUG_LEVEL_ERROR, "Caller alias count:%d : %p", m_pCallerAliasNames->wCount, this ));
if( !AddAliasItem( m_pCallerAliasNames,
pAliasList->pItems[0].pData,
pAliasList->pItems[0].wType ) )
{
H323DBG(( DEBUG_LEVEL_ERROR, "could not allocate caller name." ));
DropCall( 0 );
}
//set the value of m_pCalleeAddr
else if( !SendARQ( NOT_RESEND_SEQ_NUM ) )
{
// drop call using disconnect mode
DropCall( 0 );
}
H323DBG(( DEBUG_LEVEL_ERROR, "Caller alias count:%d : %p", m_pCallerAliasNames->wCount, this ));
}
else
{
wszMachineName = g_pH323Line->GetMachineName();
H323DBG(( DEBUG_LEVEL_ERROR, "Caller alias count:%d : %p", m_pCallerAliasNames->wCount, this ));
//set the value for m_pCallerAliasNames
if( !AddAliasItem( m_pCallerAliasNames,
(BYTE*)(wszMachineName),
sizeof(WCHAR) * (wcslen(wszMachineName) + 1 ),
h323_ID_chosen ) )
{
H323DBG(( DEBUG_LEVEL_ERROR, "could not allocate caller name." ));
DropCall( 0 );
}
else if( !PlaceCall() )
{
// drop call using disconnect mode
DropCall( LINEDISCONNECTMODE_UNREACHABLE );
}
H323DBG(( DEBUG_LEVEL_ERROR, "Caller alias count:%d : %p", m_pCallerAliasNames->wCount, this ));
}
}
//returns the alias of the divertedTo endpoint
PH323_ALIASITEM
CH323Line::CallToBeDiverted(
IN WCHAR* pwszCallerName,
IN DWORD dwCallerNameSize,
IN DWORD dwForwardMode
)
{
LPFORWARDADDRESS pForwardAddress;
DWORD dwForwardCallerLength;
H323DBG(( DEBUG_LEVEL_TRACE, "CallToBeDiverted entered:%p.", this ));
if( (m_pCallForwardParams == NULL) ||
(!m_pCallForwardParams->fForwardingEnabled) )
{
return NULL;
}
if( m_pCallForwardParams->fForwardForAllOrigins == TRUE )
{
if( m_pCallForwardParams->dwForwardTypeForAllOrigins & dwForwardMode )
{
return &(m_pCallForwardParams->divertedToAlias);
}
if( m_pCallForwardParams->dwForwardTypeForAllOrigins ==
LINEFORWARDMODE_BUSYNA )
{
if( (dwForwardMode == LINEFORWARDMODE_BUSY) ||
(dwForwardMode == LINEFORWARDMODE_NOANSW) )
{
return &(m_pCallForwardParams->divertedToAlias);
}
}
}
if( pwszCallerName == NULL )
{
return NULL;
}
pForwardAddress = m_pCallForwardParams->pForwardedAddresses;
while( pForwardAddress )
{
dwForwardCallerLength = (pForwardAddress->callerAlias.wDataLength +1)*sizeof(WCHAR);
if( (dwForwardCallerLength == dwCallerNameSize) &&
(memcmp( pwszCallerName, (PVOID)(pForwardAddress->callerAlias.pData),
dwCallerNameSize) == 0 ) )
{
if( pForwardAddress->dwForwardType & dwForwardMode )
return &(pForwardAddress->divertedToAlias);
if( pForwardAddress->dwForwardType == LINEFORWARDMODE_BUSYNA )
{
if( (dwForwardMode == LINEFORWARDMODE_BUSY) ||
(dwForwardMode == LINEFORWARDMODE_NOANSW) )
{
return &(pForwardAddress->divertedToAlias);
}
}
return NULL;
}
pForwardAddress = pForwardAddress->next;
}
H323DBG(( DEBUG_LEVEL_TRACE, "CallToBeDiverted exited:%p.", this ));
return NULL;
}
LONG
TSPIAPI
TSPI_lineSetStatusMessages(
HDRVLINE hdLine,
DWORD dwLineStates,
DWORD dwAddressStates
)
{
return NOERROR;
}
#if DBG
DWORD
SendMSPMessageOnRelatedCall(
IN PVOID ContextParameter
)
{
__try
{
return SendMSPMessageOnRelatedCallFre( ContextParameter );
}
__except( 1 )
{
MSPMessageData* pMSPMessageData = (MSPMessageData*)ContextParameter;
H323DBG(( DEBUG_LEVEL_TRACE,
"TSPI event threw exception: %p, %d, %p, %d, %p.",
pMSPMessageData -> hdCall,
pMSPMessageData -> messageType,
pMSPMessageData -> pbEncodedBuf,
pMSPMessageData -> wLength,
pMSPMessageData -> hReplacementCall ));
_ASSERTE( FALSE );
return 0;
}
}
#endif
DWORD
SendMSPMessageOnRelatedCallFre(
IN PVOID ContextParameter
)
{
H323DBG(( DEBUG_LEVEL_TRACE, "SendMSPMessageOnRelatedCall entered." ));
_ASSERTE( ContextParameter );
MSPMessageData* pMSPMessageData = (MSPMessageData*)ContextParameter;
PH323_CALL pCall = NULL;
pCall = g_pH323Line->FindH323CallAndLock(pMSPMessageData -> hdCall);
if( pCall != NULL )
{
pCall -> SendMSPMessage( pMSPMessageData->messageType,
pMSPMessageData->pbEncodedBuf, pMSPMessageData->wLength,
pMSPMessageData->hReplacementCall );
pCall -> Unlock();
}
if( pMSPMessageData->pbEncodedBuf != NULL )
{
delete pMSPMessageData->pbEncodedBuf;
}
delete pMSPMessageData;
H323DBG(( DEBUG_LEVEL_TRACE, "SendMSPMessageOnRelatedCall exited." ));
return EXIT_SUCCESS;
}
// static
void
NTAPI CH323Call::CTIdentifyExpiredCallback(
IN PVOID DriverCallHandle, // HDRVCALL
IN BOOLEAN bTimer
)
{
PH323_CALL pCall = NULL;
H323DBG(( DEBUG_LEVEL_TRACE, "CTIdentifyExpiredCallback entered." ));
//if the timer expired
_ASSERTE( bTimer );
H323DBG(( DEBUG_LEVEL_TRACE, "CTIdentity expired event recvd." ));
pCall=g_pH323Line -> FindH323CallAndLock((HDRVCALL) DriverCallHandle);
if( pCall != NULL )
{
pCall -> CTIdentifyExpired();
pCall -> Unlock();
}
H323DBG(( DEBUG_LEVEL_TRACE, "CTIdentifyExpiredCallback exited." ));
}
// static
void
NTAPI CH323Call::CTIdentifyRRExpiredCallback(
IN PVOID DriverCallHandle, // HDRVCALL
IN BOOLEAN bTimer
)
{
PH323_CALL pCall = NULL;
H323DBG(( DEBUG_LEVEL_TRACE, "CTIdentifyRRExpiredCallback entered." ));
//if the timer expired
_ASSERTE( bTimer );
H323DBG(( DEBUG_LEVEL_TRACE, "CTIdentity expired event recvd." ));
pCall=g_pH323Line -> FindH323CallAndLock((HDRVCALL) DriverCallHandle);
if( pCall != NULL )
{
pCall -> CTIdentifyRRExpired();
pCall -> Unlock();
}
H323DBG(( DEBUG_LEVEL_TRACE, "CTIdentifyRRExpiredCallback exited." ));
}
// static
void
NTAPI CH323Call::CTInitiateExpiredCallback(
IN PVOID DriverCallHandle, // HDRVCALL
IN BOOLEAN bTimer
)
{
PH323_CALL pCall = NULL;
H323DBG(( DEBUG_LEVEL_TRACE, "CTInitiateExpiredCallback entered." ));
//if the timer expired
_ASSERTE( bTimer );
H323DBG(( DEBUG_LEVEL_TRACE, "CTInitiate expired event recvd." ));
if( !QueueTAPILineRequest(
TSPI_CLOSE_CALL,
(HDRVCALL)DriverCallHandle,
NULL,
LINEDISCONNECTMODE_NOANSWER,
NULL) )
{
H323DBG(( DEBUG_LEVEL_TRACE, "could not post close call event." ));
}
H323DBG(( DEBUG_LEVEL_TRACE, "CTInitiateExpiredCallback exited." ));
}
void
CH323Call::CTIdentifyExpired()
{
if( m_hCTIdentifyTimer != NULL )
{
DeleteTimerQueueTimer( H323TimerQueue, m_hCTIdentifyTimer, NULL );
m_hCTIdentifyTimer = NULL;
}
if( m_dwCallDiversionState != H4502_CIIDENTIFY_RRSUCC )
{
CloseCall( 0 );
}
}
void
CH323Call::CTIdentifyRRExpired()
{
if( m_hCTIdentifyRRTimer != NULL )
{
DeleteTimerQueueTimer( H323TimerQueue, m_hCTIdentifyRRTimer, NULL );
m_hCTIdentifyRRTimer = NULL;
}
if( m_dwCallDiversionState != H4502_CTSETUP_RECV )
{
CloseCall( 0 );
}
}
void
NTAPI CH323Call::CallDivertOnNACallback(
IN PVOID Parameter1,
IN BOOLEAN bTimer
)
{
PH323_CALL pCall = NULL;
H323DBG(( DEBUG_LEVEL_TRACE, "CallDivertOnNACallback entered." ));
//if the timer expired
_ASSERTE( bTimer );
pCall=g_pH323Line -> FindH323CallAndLock( (HDRVCALL)Parameter1 );
if( pCall != NULL )
{
pCall -> CallDivertOnNoAnswer();
}
pCall -> Unlock();
H323DBG(( DEBUG_LEVEL_TRACE, "CallDivertOnNACallback exited." ));
}
//!!always called in a lock
void
CH323Call::CallDivertOnNoAnswer()
{
//stop the timer that caused this event
if( m_hCallDivertOnNATimer != NULL )
{
DeleteTimerQueueTimer(H323TimerQueue, m_hCallDivertOnNATimer, NULL);
m_hCallDivertOnNATimer = NULL;
}
//Make sure that the call is still alerting, it has not ben accepted and
//it has not been transfered in the mean time
if( (m_dwCallState != LINECALLSTATE_OFFERING) ||
(m_fCallAccepted == TRUE) ||
(m_dwCallType & CALLTYPE_TRANSFERED2_CONSULT)
)
{
return;
}
//divert the call. divertedToAlias is already set, so pass NULL
if( !InitiateCallDiversion( NULL, DiversionReason_cfnr) )
{
//shutdown the Q931 call
CloseCall( 0 );
}
}
BOOL
CH323Call::SendCTInitiateMessagee(
IN CTIdentifyRes * pCTIdentifyRes
)
{
H323DBG(( DEBUG_LEVEL_TRACE, "SendCTInitiateMessagee entered:%p.", this ));
//get the cookie for transferred call
CopyMemory( (PVOID)m_pCTCallIdentity, (PVOID)pCTIdentifyRes->callIdentity,
sizeof(pCTIdentifyRes->callIdentity) );
if( m_pTransferedToAlias != NULL )
{
FreeAliasNames( m_pTransferedToAlias );
m_pTransferedToAlias = NULL;
}
//argument.reroutingNr
if( !AliasAddrToAliasNames( &m_pTransferedToAlias,
(PSetup_UUIE_sourceAddress)
(pCTIdentifyRes->reroutingNumber.destinationAddress) ) )
{
H323DBG(( DEBUG_LEVEL_ERROR,
"couldn't allocate for T-2 alias:%p.", this ));
return FALSE;
}
ASN1_FreeDecoded( m_H450ASNCoderInfo.pDecInfo, pCTIdentifyRes,
CTIdentifyRes_PDU );
if( !SendQ931Message( NO_INVOKEID, 0, 0, FACILITYMESSAGETYPE,
CTINITIATE_OPCODE | H450_INVOKE ) )
{
return FALSE;
}
m_dwCallDiversionState = H4502_CTINITIATE_SENT;
//start the timer for CTIdenity message
if( !CreateTimerQueueTimer(
&m_hCTInitiateTimer,
H323TimerQueue,
CH323Call::CTInitiateExpiredCallback,
(PVOID)m_hdCall,
CTINITIATE_SENT_TIMEOUT, 0,
WT_EXECUTEINIOTHREAD | WT_EXECUTEONLYONCE) )
{
CloseCall( 0 );
}
H323DBG(( DEBUG_LEVEL_TRACE, "SendCTInitiateMessagee exited:%p.", this ));
return TRUE;
}
BOOL
CH323Call::InitiateCallDiversion(
IN PH323_ALIASITEM pwszDivertedToAlias,
IN DiversionReason eDiversionMode
)
{
H323DBG(( DEBUG_LEVEL_TRACE, "InitiateCallDiversion entered:%p.", this ));
m_dwCallType |= CALLTYPE_DIVERTED_SERVED;
if( m_pCallReroutingInfo == NULL )
{
m_pCallReroutingInfo = new CALLREROUTINGINFO;
if( m_pCallReroutingInfo == NULL )
{
return FALSE;
}
ZeroMemory( (PVOID)m_pCallReroutingInfo, sizeof(CALLREROUTINGINFO) );
}
m_pCallReroutingInfo ->diversionReason = eDiversionMode;
m_pCallReroutingInfo->divertingNrAlias = new H323_ALIASNAMES;
if( m_pCallReroutingInfo->divertingNrAlias == NULL )
{
goto cleanup;
}
ZeroMemory( (PVOID)m_pCallReroutingInfo->divertingNrAlias,
sizeof(H323_ALIASNAMES) );
if( !AddAliasItem( m_pCallReroutingInfo->divertingNrAlias,
(BYTE*)m_pCalleeAliasNames->pItems[0].pData,
sizeof(WCHAR) * (m_pCalleeAliasNames->pItems[0].wDataLength+1),
m_pCalleeAliasNames->pItems[0].wType ) )
{
goto cleanup;
}
if( m_pCallReroutingInfo->originalCalledNr == NULL )
{
m_pCallReroutingInfo->originalCalledNr = new H323_ALIASNAMES;
if( m_pCallReroutingInfo->originalCalledNr == NULL )
{
goto cleanup;
}
ZeroMemory( (PVOID)m_pCallReroutingInfo->originalCalledNr,
sizeof(H323_ALIASNAMES) );
if( !AddAliasItem(
m_pCallReroutingInfo->originalCalledNr,
(BYTE*)m_pCalleeAliasNames->pItems[0].pData,
sizeof(WCHAR) * (m_pCalleeAliasNames->pItems[0].wDataLength+1),
m_pCalleeAliasNames->pItems[0].wType )
)
{
goto cleanup;
}
}
if( m_pCallReroutingInfo->divertedToNrAlias == NULL )
{
_ASSERTE( pwszDivertedToAlias );
m_pCallReroutingInfo->divertedToNrAlias = new H323_ALIASNAMES;
if( m_pCallReroutingInfo->divertedToNrAlias == NULL )
{
goto cleanup;
}
ZeroMemory( (PVOID)m_pCallReroutingInfo->divertedToNrAlias,
sizeof(H323_ALIASNAMES) );
if( !AddAliasItem( m_pCallReroutingInfo->divertedToNrAlias,
(BYTE*)pwszDivertedToAlias->pData,
sizeof(WCHAR) * (wcslen(pwszDivertedToAlias->pData) +1),
pwszDivertedToAlias->wType
) )
{
goto cleanup;
}
}
if( !SendQ931Message( NO_INVOKEID, 0, 0, FACILITYMESSAGETYPE,
CALLREROUTING_OPCODE | H450_INVOKE ) )
{
goto cleanup;
}
m_dwCallDiversionState = H4503_CALLREROUTING_SENT;
if( !CreateTimerQueueTimer(
&m_hCallReroutingTimer,
H323TimerQueue,
CH323Call::CallReroutingTimerCallback,
(PVOID)m_hdCall,
CALLREROUTING_EXPIRE_TIME, 0,
WT_EXECUTEINIOTHREAD | WT_EXECUTEONLYONCE) )
{
goto cleanup;
}
H323DBG(( DEBUG_LEVEL_TRACE, "InitiateCallDiversion exited:%p.", this ));
return TRUE;
cleanup:
FreeCallReroutingInfo();
return FALSE;
}