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.
 
 
 
 
 
 

6323 lines
199 KiB

/******************************************************************************
*
* INTEL Corporation Proprietary Information
* Copyright (c) 1994, 1995, 1996 Intel Corporation.
*
* This listing is supplied under the terms of a license agreement
* with INTEL Corporation and may not be used, copied, nor disclosed
* except in accordance with the terms of that agreement.
*
*****************************************************************************/
/******************************************************************************
*
* AUTHOR: cjutzi (Curt Jutzi) Intel Corporation
*
* $Workfile: api_dwn.c $
* $Revision: 1.45 $
* $Modtime: 05 Mar 1997 09:53:36 $
* $Log: S:/STURGEON/SRC/H245/SRC/VCS/api_dwn.c_v $
*
* Rev 1.45 05 Mar 1997 09:56:04 MANDREWS
* Fixed compiler warning in release mode.
*
* Rev 1.44 04 Mar 1997 17:33:26 MANDREWS
* H245CopyCap() and H245CopyCapDescriptor() now return HRESULTs.
*
* Rev 1.43 26 Feb 1997 11:12:06 MANDREWS
* Fixed problem in assigning dynamic term cap IDs; the dynamic IDs were
* overlapping with static IDs.
*
* Rev 1.42 Feb 24 1997 18:30:18 tomitowx
* multiple modedescriptor support
*
* Rev 1.41 07 Feb 1997 15:33:58 EHOWARDX
* Changed H245DelCapDescriptor to match changes to set_cap_descriptor.
*
* Rev 1.40 27 Jan 1997 12:40:16 MANDREWS
*
* Fixed warnings.
*
* Rev 1.39 09 Jan 1997 19:17:04 EHOWARDX
*
* Initialize lError to H245_ERROR_OK to prevent "may be uninitialized"
* warning.
*
* Rev 1.38 19 Dec 1996 17:18:50 EHOWARDX
* Changed to use h245asn1.h definitions instead of _setof3 and _setof8.
*
* Rev 1.37 12 Dec 1996 15:57:22 EHOWARDX
* Master Slave Determination kludge.
*
* Rev 1.36 11 Dec 1996 13:55:44 SBELL1
* Changed H245Init parameters.
*
* Rev 1.35 17 Oct 1996 18:17:36 EHOWARDX
* Changed general string to always be Unicode.
*
* Rev 1.34 14 Oct 1996 14:01:26 EHOWARDX
* Unicode changes.
*
* Rev 1.33 11 Oct 1996 15:19:56 EHOWARDX
* Fixed H245CopyCap() bug.
*
* Rev 1.32 28 Aug 1996 11:37:10 EHOWARDX
* const changes.
*
* Rev 1.31 19 Aug 1996 16:28:36 EHOWARDX
* H245CommunicationModeResponse/H245CommunicationModeCommand bug fixes.
*
* Rev 1.30 15 Aug 1996 15:19:46 EHOWARDX
* First pass at new H245_COMM_MODE_ENTRY_T requested by Mike Andrews.
* Use at your own risk!
*
* Rev 1.29 08 Aug 1996 16:02:58 EHOWARDX
*
* Eliminated api_vers.h.
* Changed H245Init Debug trace to eliminate API_VERSION.
*
* Rev 1.28 19 Jul 1996 12:48:22 EHOWARDX
*
* Multipoint clean-up.
*
* Rev 1.27 01 Jul 1996 22:13:42 EHOWARDX
*
* Added Conference and CommunicationMode structures and functions.
*
* Rev 1.26 18 Jun 1996 14:53:16 EHOWARDX
* Eliminated Channel parameter to MaintenanceLoopRelease.
* Made Multiplex Capability mandatory -- H245SendTermCaps now returns
* H245_ERROR_NO_MUX_CAPS if no Multiplex Capability has been defined.
*
* Rev 1.25 14 Jun 1996 18:57:38 EHOWARDX
* Geneva update.
*
* Rev 1.24 10 Jun 1996 16:59:02 EHOWARDX
* Moved init/shutdown of submodules to CreateInstance/InstanceUnlock.
*
* Rev 1.23 06 Jun 1996 18:50:10 EHOWARDX
* Equivalent of H.324 bugs #808 and 875 fixed.
*
* Rev 1.22 05 Jun 1996 17:16:48 EHOWARDX
* MaintenanceLoop bug fix.
*
* Rev 1.21 04 Jun 1996 13:56:42 EHOWARDX
* Fixed Release build warnings.
*
* Rev 1.20 30 May 1996 23:38:52 EHOWARDX
* Cleanup.
*
* Rev 1.19 29 May 1996 16:08:04 unknown
* Fixed bug in copying nonstandard identifiers.
*
* Rev 1.18 29 May 1996 15:19:48 EHOWARDX
* Change to use HRESULT.
*
* Rev 1.17 28 May 1996 14:25:12 EHOWARDX
* Tel Aviv update.
*
* Rev 1.16 21 May 1996 15:46:54 EHOWARDX
* Fixed bugs in NonStandard messages using object identifier.
*
* Rev 1.15 20 May 1996 22:17:54 EHOWARDX
* Completed NonStandard Message and H.225.0 Maximum Skew indication
* implementation. Added ASN.1 validation to H245SetLocalCap and
* H245SetCapDescriptor. Check-in from Microsoft drop on 17-May-96.
*
* Rev 1.14 20 May 1996 14:35:12 EHOWARDX
* Got rid of asynchronous H245EndConnection/H245ShutDown stuff...
*
* Rev 1.13 17 May 1996 14:53:38 EHOWARDX
* Added calls to StartSystemClose() and EndSystemClose().
*
* Rev 1.12 16 May 1996 19:40:30 EHOWARDX
* Fixed multiplex capability bug.
*
* Rev 1.11 16 May 1996 16:36:10 EHOWARDX
* Fixed typo in H245SetCapDescriptor.
*
* Rev 1.10 16 May 1996 16:14:06 EHOWARDX
* Fixed backwards-compatibility problem in H245SetLocalCap
* (CapId of zero should result in dynamically-allocated cap id being used)
*
* Rev 1.9 16 May 1996 16:03:44 EHOWARDX
* Fixed typo in H245SetCapDescriptor.
*
* Rev 1.8 16 May 1996 15:58:32 EHOWARDX
* Fine-tuning H245SetLocalCap/H245DelLocalCap/H245SetCapDescriptor/
* H245DelCapDescriptor behaviour.
*
* Rev 1.7 15 May 1996 21:49:46 unknown
* Added call to InstanceLock() to increment lock count before call
* to InstanceDelete() in H245EndConnectionPhase2().
*
* Rev 1.6 15 May 1996 19:54:02 unknown
* Fixed H245SetCapDescriptor.
*
* Rev 1.5 14 May 1996 16:56:22 EHOWARDX
* Last minute change from H245_IND_CAPDESC_T to H245_TOTCAPDESC_T.
* H245EnumCaps() callback now uses H245_TOTCAPDESC_T instead
* of separate H245_CAPDESCID_T and H245_CAPDESC_T for consistency.
*
* Rev 1.4 14 May 1996 15:55:44 EHOWARDX
* Added mux cap handling to H245DelLocalCap.
*
* Rev 1.3 14 May 1996 14:06:06 EHOWARDX
* Fixed abort from H245EnumCaps - if Cap Callback returns non-zero,
* Cap Desc Callback is never called.
*
* Rev 1.2 13 May 1996 23:16:42 EHOWARDX
* Fixed remote terminal capability handling.
*
* Rev 1.1 11 May 1996 20:32:52 EHOWARDX
* Checking in for the night...
*
* Rev 1.0 09 May 1996 21:06:06 EHOWARDX
* Initial revision.
*
* Rev 1.25.1.10 09 May 1996 19:31:02 EHOWARDX
* Redesigned thread locking logic.
* Added new API functions.
*
* Rev 1.25.1.9 01 May 1996 19:31:16 EHOWARDX
* Added H245CopyCap(), H245FreeCap(), H245CopyMux(), H245FreeMux().
* Changed H2250_xxx definitions for H.225.0 address types to H245_xxx.
*
* Rev 1.25.1.8 27 Apr 1996 21:09:20 EHOWARDX
* Changed Channel Numbers to words, added H.225.0 support.
*
* Rev 1.25.1.7 25 Apr 1996 20:06:26 EHOWARDX
* Moved setting of EndSessionPdu in EndSessionPhase1 to before call to api_fs
*
* Rev 1.25.1.6 25 Apr 1996 17:57:00 EHOWARDX
* Added dwTxPort argument to H245OpenChannel().
*
* Rev 1.25.1.5 25 Apr 1996 16:51:00 EHOWARDX
* Function changes as per H.245 API Changes spec.
*
* Rev 1.25.1.4 24 Apr 1996 20:54:32 EHOWARDX
* Added new OpenLogicalChannelAck/OpenLogicalChannelReject support.
*
* Rev 1.25.1.3 19 Apr 1996 12:54:40 EHOWARDX
* Updated to 1.30
*
* Rev 1.25.1.2 15 Apr 1996 15:10:48 EHOWARDX
* Updated to match Curt's current version.
*
* Rev 1.25.1.1 03 Apr 1996 17:12:50 EHOWARDX
* Integrated latest H.323 changes.
*
* Rev 1.25.1.0 03 Apr 1996 15:53:42 cjutzi
* Branched for H.323.
*
* Rev 1.20 27 Mar 1996 15:25:34 cjutzi
* - fixed a bug from this morning checkin dynamically allocating
* pdu's.. free_mux_tbl was getting called after pdu was free'd..
* this was a problem since the mux table pointer was in the pdu
*
* Rev 1.19 27 Mar 1996 08:37:08 cjutzi
*
* - removed PDU from stack.. made them dynamically allocated
*
* Rev 1.18 20 Mar 1996 14:47:32 cjutzi
* - added ERROR H245_ERROR_NO_CAPDESC to SendTermCaps.
*
* Rev 1.17 18 Mar 1996 15:23:16 cjutzi
*
*
*
* Rev 1.16 13 Mar 1996 09:15:44 cjutzi
*
* - changed LLPCRITICAL_SECTION to CRITICAL_SECTION *
*
* Rev 1.15 12 Mar 1996 15:51:48 cjutzi
*
* - implemented locking
* - fixed callback bug w/ clenaup on term caps..
* - implemented End Session
* - fixed shutdown
*
* Rev 1.14 08 Mar 1996 14:02:48 cjutzi
*
* - removed H245SetSimultaneous stuff..
* - added H245SetCapDescriptor Stuff..
* - completeed MuxTable Entry Stuff.
* - required H223 -or- some portion of MuxCapbilities to be
* there before you issue SendCaps..
* - NOTE: need to inforce the Simultaneous capabilities in
* the same mannor..
*
* Rev 1.13 05 Mar 1996 17:35:38 cjutzi
*
* - implemented SendMultiplexTable..
* - removed bcopy/bzero and changed free call
* - added master slave indication
*
* Rev 1.12 01 Mar 1996 13:48:24 cjutzi
*
* - added hani's new fsm id's
* - added some support for release on close request.
*
* Rev 1.11 29 Feb 1996 17:27:10 cjutzi
*
* - bi-directional channel working..
*
* Rev 1.10 29 Feb 1996 08:35:52 cjutzi
*
* - added p_ossWorld to initialization
*
* Rev 1.9 27 Feb 1996 13:30:18 cjutzi
*
* - fixed master slave problem with conf_ind and tracker type
* - removed RSP_LCSE in close channel resp
*
* Rev 1.8 26 Feb 1996 17:23:18 cjutzi
*
* - MiscCommand API added
* - Fixed Assert for H245Init.. was not NULL'n out the pointers for the
* context blocks..
*
* Rev 1.7 26 Feb 1996 11:05:16 cjutzi
*
* - added simultaneous caps.. and fixed bugs..
* lot's of changes..
*
* Rev 1.6 16 Feb 1996 13:01:08 cjutzi
*
* - got open / close / request close working in both directions.
*
* Rev 1.5 15 Feb 1996 14:42:54 cjutzi
*
* - fixed trace level bind w/ Instance.. no other change but had to
* add when h245deb.c when in..
*
*
* Rev 1.4 15 Feb 1996 10:50:54 cjutzi
*
* - termcaps working
* - changed API interface for MUX_T
* - changed callback or IND_OPEN
* - changed constants IND_OPEN/IND_OPEN_NEEDRSP etc..
* - cleaned up the open.. (not complete yet.. )
*
* Rev 1.3 09 Feb 1996 16:58:36 cjutzi
*
* - cleanup.. and some fixes..
* - added and or changed headers to reflect the log of changes
*
*****************************************************************************/
/****************************************************************************/
/****************************************************************************/
/****************************************************************************/
/**** *****/
/**** NOTES TO THE READER *****/
/**** *****/
/**** This program has been put together using a a screen which is *****/
/**** wider than 80 characters.. It is best if a similar screen size is *****/
/**** used.. Of course emacs is my preference but 80 col screens will *****/
/**** cause you much frustration.. *****/
/**** *****/
/**** Tabs are set to 8 *****/
/**** *****/
/**** NOTE: *****/
/**** Headers are documented, however they may or may not *****/
/**** coorispond to reality. See the H245Spec.doc from Intel for the *****/
/**** current :-> H245 specification *****/
/**** *****/
/**** DISCLAMER: *****/
/**** *****/
/**** Since this code wasn't developed in Word 7.0, I am fully *****/
/**** responsable for all spelling mistakes in the comments. Please *****/
/**** disregard the spelling mistakes.. or fix them, if you are *****/
/**** currently modifying the code. *****/
/**** *****/
/**** - Thankyou *****/
/**** *****/
/**** Curt Jutzi *****/
/**** Oregon, USA *****/
/**** *****/
/****************************************************************************/
/****************************************************************************/
/****************************************************************************/
#ifndef STRICT
#define STRICT
#endif
#include "precomp.h"
/***********************/
/* H245 INCLUDES */
/***********************/
#define H245DLL_EXPORT
#include "h245api.h"
#include "h245com.h"
#include "h245sys.x"
#include "api_util.x"
#include "pdu.x"
#include "fsmexpor.h"
#include "api_debu.x"
#include "sr_api.h"
#include "h245deb.x"
/*****************************************************************************
*
* TYPE: H245 API
*
*****************************************************************************
*
* PROCEDURE: H245Init
*
* DESCRIPTION
*
* H245_INST_T H245Init (
* H245_CONF_T Configuration,
* DWORD dwH245PhysId,
* DWORD dwLinkLayerPhysId,
* DWORD dwPreserved,
* H245_CONF_IND_CALLBACK_T Callback
* )
* Description:
*
* Called to create an H.245 instance and its related sublayers
* (e.g., SRP). This function must be called before any other
* API calls may be called The current H.245 implementation can
* only have, at most, one client. Therefore H245Init can only be
* called once per physical ID.
* Input
*
* Configuration Indicates the type of configuration the client
* wishes to establish, e.g. H.324, H.323, H.310,
* or DSVD.
* dwH245PhysId Parameter identifying the H245 entry
* pdwLinkLayerPhysId
* Output parameter identifying the linkLayer
* entry.
* dwPreserved Parameter that may be used by H.245 client to
* provide context, passed back to client in all
* confirms and indications.
* Callback Callback routine supplied by the client which
* will be used by the H.245 subsystem to convey
* confirm and indication messages back to the
* client.
* Call Type:
*
* Synchronous
*
* Return Values:
*
* Return value of 0 indicates Failure
* Return value of non 0 is a valid H245_INST_T
*
* Errors:
* N/A
*
* See Also:
* H245EndSession
* H245Shutdown
*
*
*****************************************************************************/
H245DLL H245_INST_T
H245Init (
H245_CONFIG_T Configuration,
unsigned long dwH245PhysId,
unsigned long *pdwLinkLayerPhysId,
DWORD_PTR dwPreserved,
H245_CONF_IND_CALLBACK_T CallBack,
unsigned char byTerminalType
)
{
register struct InstanceStruct *pInstance;
HRESULT lError;
H245TRACE(dwH245PhysId,4,"H245Init(%d, 0x%x, 0x%x, 0x%x, %d) <-",
Configuration, dwH245PhysId, dwPreserved, CallBack, byTerminalType);
switch (Configuration)
{
case H245_CONF_H324:
case H245_CONF_H323:
break;
default:
H245TRACE(dwH245PhysId,1,"H245Init -> Invalid Configuration %d", Configuration);
return H245_INVALID_ID;
} // switch
if (CallBack == NULL)
{
H245TRACE(dwH245PhysId,1,"H245Init -> Null CallBack");
return H245_INVALID_ID;
}
/* see if this physical identifier has been initialized already */
// Send down H245PhysId that was input.
pInstance = InstanceCreate(dwH245PhysId, Configuration);
if (pInstance == NULL)
{
return H245_INVALID_ID;
}
// Get the linkLayer PhysId.
*pdwLinkLayerPhysId = pInstance->SendReceive.hLinkLayerInstance;
// Initialize instance API structure
pInstance->API.dwPreserved = dwPreserved;
pInstance->API.ConfIndCallBack = CallBack;
// Initialize instance FSM structure
pInstance->StateMachine.sv_TT = byTerminalType;
pInstance->StateMachine.sv_STATUS = INDETERMINATE;
H245TRACE(pInstance->dwInst,4,"H245Init -> %d", pInstance->dwInst);
lError = pInstance->dwInst;
InstanceUnlock(pInstance);
return lError;
} // H245Init()
/*****************************************************************************
*
* TYPE: H245 API
*
*****************************************************************************
*
* PROCEDURE: H245EndSession
*
* DESCRIPTION
*
* Yes.. this should be explained.. Since Send Receive needs
* to flush some buffers and send out an End Session.. what we've
* hopefully done is a 2 phase shut down...
*
* call StartSessionClose which initiates the flush..
* when flush is complete EndSessionPhase1 is called..
* The end session pdu is then placed in the send queue..
* When the End Session Pdu is sent.. the EndSession Phase
* 2 is called, and the result is sent up to the client..
*
* Hope that helps..
*
*
* HRESULT H245EndSession ( H245_INST_T dwInst,
* H245_ENDSESSION_T Mode,
* H245_NONSTANDARD_T *pNonStd (*optional*)
* )
*
* Description:
* Called to shutdown the peer to peer session between this H.245
* session and the remote peers H.245 layer.
*
* It will terminate by issuing an EndSession command to the
* remote side and call end session for all the H.245 subsystems.
* All resources are returned; therefore no further action is
* permitted, except H245ShutDown until another H245Init API call
* is made.
*
* input
* dwInst Instance handle returned by H245Init
* Mode Mode which the client wishes to terminat
* the session
* pNonStd If the mode is non standard this is the non
* standard parameter passes to the remote client.
* This parameter is optional, and should be set
* to NULL if not used
*
* Call Type:
* Asynchronous
*
* Return Values:
* See Errors
*
* Errors:
* H245_ERROR_OK
* H245_ERROR_PARAM
* H245_ERROR_INVALID_INST
* H245_ERROR_NOT_CONNECTED
* See Also:
* H245Shutdown
* H245Init
*
*****************************************************************************/
H245DLL HRESULT
H245EndSession (
H245_INST_T dwInst,
H245_ENDSESSION_T Mode,
const H245_NONSTANDARD_PARAMETER_T * pNonStd
)
{
register struct InstanceStruct *pInstance;
register MltmdSystmCntrlMssg *pPdu;
HRESULT lError;
H245TRACE (dwInst,4,"H245EndSession <-");
/* check for valid instance handle */
pInstance = InstanceLock(dwInst);
if (pInstance == NULL)
{
H245TRACE (dwInst,1,"H245EndSession -> %s",map_api_error(H245_ERROR_INVALID_INST));
return H245_ERROR_INVALID_INST;
}
/* system should be in either connecting or connected */
switch (pInstance->API.SystemState)
{
case APIST_Connecting:
case APIST_Connected:
break;
default:
H245TRACE (dwInst,1,"H245EndSession -> %s",map_api_error(H245_ERROR_NOT_CONNECTED));
InstanceUnlock(pInstance);
return H245_ERROR_NOT_CONNECTED;
}
// Allocate the PDU buffer
pPdu = (MltmdSystmCntrlMssg *)MemAlloc(sizeof(*pPdu));
if (pPdu == NULL)
{
H245TRACE (dwInst,1,"H245EndSession -> %s",map_api_error(H245_ERROR_NOMEM));
InstanceUnlock(pInstance);
return H245_ERROR_NOMEM;
}
memset(pPdu, 0, sizeof(MltmdSystmCntrlMssg));
// Build the PDU
lError = pdu_cmd_end_session (pPdu, Mode, pNonStd);
if (lError == H245_ERROR_OK)
{
// Send the PDU
lError = FsmOutgoing(pInstance, pPdu, 0);
}
// Free the PDU buffer
MemFree(pPdu);
if (lError != H245_ERROR_OK)
{
H245TRACE (dwInst,1,"H245EndSession -> %s",map_api_error(lError));
}
else
{
H245TRACE (dwInst,4,"H245EndSession -> OK");
pInstance->API.SystemState = APIST_Disconnected;
}
InstanceUnlock(pInstance);
return lError;
} // H245EndSession()
/*****************************************************************************
*
* TYPE: H245 API
*
*****************************************************************************
*
* PROCEDURE: H245ShutDown
*
* DESCRIPTION
*
* HRESULT H245Shutdown ( H245_INST_T dwInst);
*
* Description:
*
* Called to terminate the specified instance of H.245. If there
* is currently an active session (see H245Init) then the H.245
* subsystem will issue an EndSession to the other side and wait
* for H.245 sublayer termination notifications before it queues
* Callback confirm.
*
* This call will force the client to issue another H245Init
* before it can use any of the H.245 API functions.
*
* Input
* dwInst Instance handle returned by H245Init
*
* Call Type:
* asynchronous
*
* Return Values:
* See Errors
*
* Errors:
* H245_ERROR_OK
* H245_ERROR_INVALID_INST dwInst is not a valid instance handle
*
* See Also:
* H245Init
*
*****************************************************************************/
H245DLL HRESULT
H245ShutDown (H245_INST_T dwInst)
{
register struct InstanceStruct *pInstance;
register HRESULT lError;
H245TRACE (dwInst,4,"H245ShutDown <-");
/* check for valid instance handle */
pInstance = InstanceLock(dwInst);
if (pInstance == NULL)
{
H245TRACE (dwInst,1,"H245ShutDown -> %s",map_api_error(H245_ERROR_INVALID_INST));
return H245_ERROR_INVALID_INST;
}
switch (pInstance->API.SystemState)
{
case APIST_Connecting:
case APIST_Connected:
lError = H245EndSession(dwInst,H245_ENDSESSION_DISCONNECT,NULL);
break;
default:
lError = H245_ERROR_OK;
}
if (lError != H245_ERROR_OK)
H245TRACE (dwInst,1,"H245ShutDown -> %s", map_api_error(lError));
else
H245TRACE (dwInst,4,"H245ShutDown -> OK");
InstanceDelete (pInstance);
return H245_ERROR_OK;
} // H245ShutDown()
/*****************************************************************************
*
* TYPE: H245 API
*
*****************************************************************************
*
* PROCEDURE: H245InitMasterSlave
*
* DESCRIPTION
*
* HRESULT H245InitMasterSlave ( H245_INST_T dwInst,
* DWORD dwTransId )
*
* Description:
* Called to initiate the H.245 master slave negotiation.
* Upon completion of the negotiation the local client will
* receive an H245_CONF_INIT_MSTSLV message indicating the
* result of the negotiation.
* Input
* dwInst Instance handle returned by
* H245GetInstanceId
* dwTransId User supplied object used to identify this
* request in the asynchronous response to
* this call.
*
* Call Type:
* Asynchronous
*
* Return Values:
* See Errors
*
* Callbacks:
* H245_CONF_INIT_MSTSLV
*
* Errors:
* H245_ERROR_OK Master Slave Determination started
* H245_ERROR_INPROCESS Master Slave Determination currently
* in process
* H245_ERROR_NOMEM
* H245_ERROR_INPROCESS In process
* H245_ERROR_INVALID_INST dwInst is not a valid instance handle
*
* See Also:
* H245Init
*
* callbacks
* H245_IND_MSTSLV
*
*
*****************************************************************************/
H245DLL HRESULT
H245InitMasterSlave (
H245_INST_T dwInst,
DWORD_PTR dwTransId
)
{
struct InstanceStruct *pInstance;
Tracker_T *pTracker;
HRESULT lError;
MltmdSystmCntrlMssg *pPdu = NULL;
/* check for valid instance handle */
H245TRACE (dwInst,4,"H245InitMasterSlave <-");
pInstance = InstanceLock(dwInst);
if (pInstance == NULL)
{
H245TRACE (dwInst,1,"H245InitMasterSlave -> %s",map_api_error(H245_ERROR_INVALID_INST));
return H245_ERROR_INVALID_INST;
}
/* if the transaction is in process.. tell client */
if (pInstance->API.MasterSlave == APIMS_InProcess)
{
H245TRACE (dwInst,1,"H245InitMasterSlave -> %s",map_api_error(H245_ERROR_INPROCESS));
InstanceUnlock(pInstance);
return H245_ERROR_INPROCESS;
}
/* if the transaction is already complete */
if (pInstance->API.MasterSlave != APIMS_Undef)
{
if (pInstance->API.ConfIndCallBack)
{
H245_CONF_IND_T confirm_ind_event;
confirm_ind_event.Kind = H245_CONF;
confirm_ind_event.u.Confirm.Confirm = H245_CONF_INIT_MSTSLV;
confirm_ind_event.u.Confirm.dwPreserved = pInstance->API.dwPreserved;
confirm_ind_event.u.Confirm.dwTransId = dwTransId;
confirm_ind_event.u.Confirm.Error = H245_ERROR_OK;
confirm_ind_event.u.Confirm.u.ConfMstSlv =
(pInstance->API.MasterSlave == APIMS_Master) ? H245_MASTER : H245_SLAVE;
(*pInstance->API.ConfIndCallBack)(&confirm_ind_event, NULL);
}
H245TRACE (dwInst,4,"H245InitMasterSlave -> OK");
InstanceUnlock(pInstance);
return H245_ERROR_OK;
}
/* get somthing to keep track of what the heck you're doing.. */
if (!(pTracker = alloc_link_tracker (pInstance,
API_MSTSLV_T,
dwTransId,
API_ST_WAIT_RMTACK,
API_CH_ALLOC_UNDEF,
API_CH_TYPE_UNDEF,
0,
H245_INVALID_CHANNEL,
H245_INVALID_CHANNEL,
0)))
{
H245TRACE(dwInst,1,"H245InitMasterSlave -> %s",map_api_error(H245_ERROR_NOMEM));
InstanceUnlock(pInstance);
return H245_ERROR_NOMEM;
}
if (!(pPdu = (MltmdSystmCntrlMssg *)MemAlloc(sizeof(MltmdSystmCntrlMssg))))
{
H245TRACE (dwInst,1,"H245InitMasterSlave -> %s",map_api_error(H245_ERROR_NOMEM));
InstanceUnlock(pInstance);
return H245_ERROR_NOMEM;
}
/* set master slave in process */
pInstance->API.SystemState = APIST_Connecting;
pInstance->API.MasterSlave = APIMS_InProcess;
memset(pPdu, 0, sizeof(*pPdu));
pPdu->choice = MltmdSystmCntrlMssg_rqst_chosen;
pPdu->u.MltmdSystmCntrlMssg_rqst.choice = masterSlaveDetermination_chosen;
lError = FsmOutgoing(pInstance, pPdu, (DWORD_PTR)pTracker);
MemFree(pPdu);
if (lError != H245_ERROR_OK)
{
unlink_dealloc_tracker (pInstance, pTracker);
H245TRACE (dwInst,1,"H245InitMasterSlave -> %s",map_api_error(lError));
}
else
H245TRACE (dwInst,4,"H245InitMasterSlave -> OK");
InstanceUnlock(pInstance);
return lError;
} // H245InitMasterSlave()
/*****************************************************************************
*
* TYPE: H245 API
*
*****************************************************************************
*
* PROCEDURE: H245SetLocalCap
*
* DESCRIPTION
*
* HRESULT H245SetLocalCap (
* H245_INST_T dwInst,
* H245_TOTCAP_T *pTotCap,
* H245_CAPID_T *pCapId
* )
*
* Description:
* This function allows the client to define a specific
* capability to the H.245 subsystem. When this function is
* called a new capability entry is made in the local capability
* table. The returned value in *pCapId can be used by the client
* to refer to that registered capability. NULL in the *pCapId
* is valid.
*
* This call is used for both client (Audio / Video / Data / Mux)
* capabilities. It is not used for setting capability descriptors.
*
* Note:
* 7 This function does not communicate this update to the
* remote peer until the client calls H245SendTermCaps.
* 7 pTotCap->CapId is of no significance in this call.
*
* pTotCap->CapId is of no significance in this call and should
* be set to 0
*
* if DataType of H245_DATA_MUX is used (i.e. in setting the
* mux table capabilities) No capid is returned, and it can not
* be used in H245SetCapDescritptor api call.
*
* Input
* dwInst Instance handle returned by GetInstanceId
* pTotCap Capability set defining the capability
*
* Note: pTotCap->CapId is of no significance in this call.
*
* output
* pCapId Capability id which client can use to reference
* this capability in the H.245 subsystem. This can
* be NULL, in this case nothing is returned.
*
* Call Type:
* Synchronous
*
* Return Values:
* If pCap is not null, the local cap table id is returned
* to the client in this parameter.
*
* Errors:
* H245_ERROR_OK
* H245_ERROR_PARAM There was an invalid parameter passed
* H245_ERROR_MAXTBL Entry not made because local cap table
* is full
* H245_ERROR_INVALID_INST dwInst is not a valid instance handle
*
* See Also:
* H245DelLocalCap
* H245EnumCaps
* H245SetCapDescriptor
*
*
* ASSUMPTION:
* pTotCap->CapId will be set to H245_INVALID_CAPID
* pTotCap->Dir will be set
*
*****************************************************************************/
H245DLL HRESULT
H245SetLocalCap (
H245_INST_T dwInst,
H245_TOTCAP_T * pTotCap,
H245_CAPID_T * pCapId
)
{
register struct InstanceStruct *pInstance;
struct TerminalCapabilitySet *pTermCapSet;
HRESULT lError;
H245TRACE (dwInst,4,"H245SetLocalCap <-");
/* check for valid instance handle */
pInstance = InstanceLock(dwInst);
if (pInstance == NULL)
{
H245TRACE (dwInst,1,"H245SetLocalCap -> %s",map_api_error(H245_ERROR_INVALID_INST));
return H245_ERROR_INVALID_INST;
}
/* check for valid parameters */
if (pTotCap == NULL ||
pCapId == NULL ||
((*pCapId > H245_MAX_CAPID) && (*pCapId != H245_INVALID_CAPID)) ||
pTotCap->Dir < H245_CAPDIR_LCLRX ||
pTotCap->Dir > H245_CAPDIR_LCLRXTX ||
pTotCap->ClientType < H245_CLIENT_NONSTD ||
pTotCap->ClientType > H245_CLIENT_MUX_H2250)
{
H245TRACE (dwInst,1,"H245SetLocalCap -> %s",map_api_error(H245_ERROR_PARAM));
InstanceUnlock(pInstance);
return H245_ERROR_PARAM;
}
pTermCapSet = &pInstance->API.PDU_LocalTermCap.u.MltmdSystmCntrlMssg_rqst.u.terminalCapabilitySet;
// Don't trust the user filled-in data type!
pTotCap->DataType = DataTypeMap[pTotCap->ClientType];
/* if it's a MUX type handle here */
if (pTotCap->DataType == H245_DATA_MUX)
{
// Add multiplex capability
if (pTermCapSet->bit_mask & multiplexCapability_present)
{
del_mux_cap(pTermCapSet);
}
*pCapId = pTotCap->CapId = 0;
lError = set_mux_cap(pInstance, pTermCapSet, pTotCap);
#if defined(_DEBUG)
if (lError == H245_ERROR_OK)
{
// Validate mux capability
if (check_pdu(pInstance, &pInstance->API.PDU_LocalTermCap))
{
// Bad mux capability - delete it
del_mux_cap(pTermCapSet);
lError = H245_ERROR_ASN1;
}
}
#endif // (DEBUG)
}
else if (*pCapId == 0 || *pCapId == H245_INVALID_CAPID)
{
// Assign the next never-used cap id
if (pInstance->API.LocalCapIdNum == H245_INVALID_CAPID)
{
// All possible capability identifiers have been assigned
H245TRACE (dwInst,1,"H245SetLocalCap -> %s",map_api_error(H245_ERROR_MAXTBL));
InstanceUnlock(pInstance);
return H245_ERROR_MAXTBL;
}
*pCapId = pInstance->API.LocalCapIdNum;
/* insert in the new capability in the local capability set table */
pTotCap->CapId = *pCapId;
lError = set_capability(pInstance, pTermCapSet, pTotCap);
#if defined(_DEBUG)
if (lError == H245_ERROR_OK)
{
// Validate capability
if (check_pdu(pInstance, &pInstance->API.PDU_LocalTermCap))
{
// Bad capability - delete it
H245DelLocalCap(dwInst, *pCapId);
lError = H245_ERROR_ASN1;
}
}
#endif // (DEBUG)
if (lError == H245_ERROR_OK)
pInstance->API.LocalCapIdNum++;
}
else
{
/* insert in the new capability in the local capability set table */
pTotCap->CapId = *pCapId;
lError = set_capability(pInstance, pTermCapSet, pTotCap);
#if defined(_DEBUG)
if (lError == H245_ERROR_OK)
{
// Validate capability
if (check_pdu(pInstance, &pInstance->API.PDU_LocalTermCap))
{
// Bad capability - delete it
H245DelLocalCap(dwInst, *pCapId);
lError = H245_ERROR_ASN1;
}
}
#endif // (DEBUG)
}
if (lError != H245_ERROR_OK)
{
H245TRACE (dwInst,1,"H245SetLocalCap -> %s",map_api_error(lError));
pTotCap->CapId = *pCapId = H245_INVALID_CAPID;
}
else
{
H245TRACE (dwInst,4,"H245SetLocalCap -> OK");
}
InstanceUnlock(pInstance);
return lError;
} // H245SetLocalCap()
/*****************************************************************************
*
* TYPE: H245 API
*
*****************************************************************************
*
* PROCEDURE: H245DelLocalCap
*
* DESCRIPTION Delete Local Cap simply disables the cap.. it
* will not be updated until the client issues
* H245SendTermCaps
*
*
* HRESULT H245DelLocalCap(
* H245_INST_T dwInst,
* H245_CAPID_T CapId
* )
*
* Description:
* This function allows the client to delete a specific
* capability id in the H.245 subsystem.
*
* Note: This function does not communicate this update
* to the remote peer until the client calls H245SendTermCaps.
*
* Input
* dwInst Instance handle returned by H245GetInstanceId
* CapId Cap Id the client wishes to remove from the
* capability table.
*
* If an error occurs no action is taken and the CapId the
* client wished to delete is not changed.
*
* Call Type:
* Synchronous
*
* Return Values:
* See Errors
*
* Errors:
* H245_ERROR_OK Capability deleted
* H245_ERROR_INVALID_INST dwInst is not a valid instance handle
*
* See Also:
* H245SetLocalCap
* H245SendTermCaps
* H245EnumCaps
*
* RETURN:
*
*****************************************************************************/
H245DLL HRESULT
H245DelLocalCap (
H245_INST_T dwInst,
H245_CAPID_T CapId
)
{
register struct InstanceStruct *pInstance;
struct TerminalCapabilitySet *pTermCapSet;
CapabilityTableLink pCapLink;
HRESULT lError = H245_ERROR_OK;
H245TRACE (dwInst,4,"H245DelLocalCap <-");
/* check for valid instance handle */
pInstance = InstanceLock(dwInst);
if (pInstance == NULL)
{
H245TRACE (dwInst,1,"H245DelLocalCap -> %s",map_api_error(H245_ERROR_INVALID_INST));
return H245_ERROR_INVALID_INST;
}
pTermCapSet = &pInstance->API.PDU_LocalTermCap.u.MltmdSystmCntrlMssg_rqst.u.terminalCapabilitySet;
if (CapId == 0)
{
// Delete multiplex capability
del_mux_cap(pTermCapSet);
}
else
{
/* (TBC) if I delete my capability id.. what about simultaneous caps ?? */
/* should I go through the list and deactivate them ?? */
pCapLink = find_capid_by_entrynumber (pTermCapSet, CapId);
if (pCapLink)
{
// Delete terminal capability
disable_cap_link (pCapLink);
}
else
{
lError = H245_ERROR_PARAM;
}
}
if (lError != H245_ERROR_OK)
H245TRACE (dwInst,1,"H245DelLocalCap -> %s",map_api_error(lError));
else
H245TRACE (dwInst,4,"H245DelLocalCap -> OK");
InstanceUnlock(pInstance);
return lError;
} // H245DelLocalCap()
/*****************************************************************************
*
* TYPE: H245 API
*
*****************************************************************************
*
* PROCEDURE: H245SetCapDescriptor
*
* DESCRIPTION
*
* HRESULT H245SetCapDescriptor (
* H245_INST_T dwInst,
* H245_CAPDESC_T *pCapDesc,
* H245_CAPDESCID_T *pCapDescId (* Optional *)
* )
* Description:
* This procedure is called to set local capability descriptors.
* It will return a capability descriptor id in the parameter
* *pCapDescId if it is non null.
*
* Note:
* These capabilities are communicated via the H245SendTermCaps
* API call. Any updates to the CapDescriptor table (either
* additions or deletions ) will not be communicated to the
* remote side until the H245SendTermCaps call is made.
*
* Input
* dwInst Instance handle returned by H245Init
* CapDesc This is the capability Descriptor you wish
* to set
* Output
* pCapDescId optional: Capability id that will be returned.
*
* Call Type:
* Synchronous
*
* Return Values:
* See Errors
*
* Errors:
* H245_ERROR_OK
* H245_ERROR_INVALID_CAPID Capid used in CapDesc was not
* registred
* H245_ERROR_MAXTB Out of table space to store Descriptor
* H245_ERROR_PARAM Descriptor is too long or not valid
* H245_ERROR_NOMEM
* H245_ERROR_INVALID_INST
*
* See Also:
* H245DelCapDescriptor
* H245SendTermCaps
*
* ASSUMES:
* SimCapId is the array entry point in the apabilityDescriptors
* array.. this has a limitation, in that you can never wrap the
* array at 256.. this will be cleaned up when array is turned into
* linked list.
*
*****************************************************************************/
H245DLL HRESULT
H245SetCapDescriptor (
H245_INST_T dwInst,
H245_CAPDESC_T *pCapDesc,
H245_CAPDESCID_T *pCapDescId
)
{
register struct InstanceStruct *pInstance;
HRESULT lError;
H245TRACE (dwInst,4,"H245SetCapDescriptor <-");
/* check for valid instance handle */
pInstance = InstanceLock(dwInst);
if (pInstance == NULL)
{
H245TRACE (dwInst,1,"H245SetCapDescriptor -> %s",map_api_error(H245_ERROR_INVALID_INST));
return H245_ERROR_INVALID_INST;
}
/* must have capdescriptor && */
/* length must be less than 256 */
if (pCapDesc == NULL ||
pCapDesc->Length >= 256 ||
pCapDescId == NULL)
{
H245TRACE (dwInst,1,"H245SetCapDescriptor -> %s",map_api_error(H245_ERROR_PARAM));
InstanceUnlock(pInstance);
return H245_ERROR_PARAM;
}
if (*pCapDescId >= 256)
{
// Assign the next never-used cap id
if (pInstance->API.LocalCapDescIdNum >= 256)
{
// All possible capability identifiers have been assigned
H245TRACE (dwInst,1,"H245CapDescriptor -> %s",map_api_error(H245_ERROR_MAXTBL));
InstanceUnlock(pInstance);
return H245_ERROR_MAXTBL;
}
*pCapDescId = pInstance->API.LocalCapDescIdNum;
/* insert in the new capability descriptor in the local capability descriptor table */
lError = set_cap_descriptor(pInstance, pCapDesc, pCapDescId,
&pInstance->API.PDU_LocalTermCap.u.MltmdSystmCntrlMssg_rqst.u.terminalCapabilitySet);
#if defined(_DEBUG)
if (lError == H245_ERROR_OK)
{
// Validate Capability Descriptor
if (check_pdu(pInstance, &pInstance->API.PDU_LocalTermCap))
{
// Capability Descriptor Invalid - delete it
H245DelCapDescriptor(dwInst, *pCapDescId);
lError = H245_ERROR_ASN1;
}
}
#endif // (DEBUG)
if (lError == H245_ERROR_OK)
pInstance->API.LocalCapDescIdNum++;
}
else
{
/* insert in the new capability in the local capability set table */
lError = set_cap_descriptor(pInstance, pCapDesc, pCapDescId,
&pInstance->API.PDU_LocalTermCap.u.MltmdSystmCntrlMssg_rqst.u.terminalCapabilitySet);
#if defined(_DEBUG)
if (lError == H245_ERROR_OK)
{
// Validate Capability Descriptor
if (check_pdu(pInstance, &pInstance->API.PDU_LocalTermCap))
{
// Capability Descriptor Invalid - delete it
H245DelCapDescriptor(dwInst, *pCapDescId);
lError = H245_ERROR_ASN1;
}
}
#endif // (DEBUG)
}
if (lError != H245_ERROR_OK)
{
H245TRACE (dwInst,1,"H245CapDescriptor -> %s",map_api_error(lError));
*pCapDescId = H245_INVALID_CAPDESCID;
}
else
{
H245TRACE (dwInst,4,"H245CapDescriptor -> OK");
}
InstanceUnlock(pInstance);
return lError;
} // H245SetCapDescriptor()
/*****************************************************************************
*
* TYPE: H245 API
*
*****************************************************************************
*
* PROCEDURE: H245DelCapDescriptor
*
* DESCRIPTION
*
* HRESULT H245DelCapDescriptor (
* H245_INST_T dwInst,
* H245_CAPDESCID_T CapDescId
* )
* Description:
* This procedure is called to delete local capability descriptors.
*
* Note:
* These capabilities are communicated via the
* H245SendTermCaps API call. Any updates to the
* CapDescriptor table (either additions or deletions )
* will not be communicated to the remote side until the
* H245SendTermCaps call is made.
*
* Input
* dwInst Instance handle returned by H245Init
* CapDescId This is the capability Descriptor you wish
* to delete
* Call Type:
* Synchronous
*
* Return Values:
* See Errors
* Errors:
* H245_ERROR_OK
* H245_ERROR_INVALID_INST
*
* See Also:
* H245SetCapDescriptor
* H245SendTermCaps
*
* ASSUMES:
*
* SimCapId is the array entry point in the apabilityDescriptors
* array.. this has a limitation, in that you can never wrap the
* array at 256.. this will be cleaned up when array is turned into
* linked list.
*
*
*****************************************************************************/
H245DLL HRESULT
H245DelCapDescriptor (
H245_INST_T dwInst,
H245_CAPDESCID_T CapDescId
)
{
register struct InstanceStruct *pInstance;
CapabilityDescriptor *p_cap_desc;
struct TerminalCapabilitySet *pTermCapSet;
unsigned int uId;
H245TRACE (dwInst,4,"H245DelCapDescriptor <-");
if (CapDescId >= 256)
{
H245TRACE(dwInst,1,"API:H24DelCapDescriptor -> Invalid cap desc id %d",CapDescId);
return H245_ERROR_INVALID_CAPDESCID;
}
/* check for valid instance handle */
pInstance = InstanceLock(dwInst);
if (pInstance == NULL)
{
H245TRACE (dwInst,1,"API:H24DelCapDescriptor -> %s",map_api_error(H245_ERROR_INVALID_INST));
return H245_ERROR_INVALID_INST;
}
/* get pointer to Capability Descriptor */
p_cap_desc = NULL;
pTermCapSet = &pInstance->API.PDU_LocalTermCap.u.MltmdSystmCntrlMssg_rqst.u.terminalCapabilitySet;
for (uId = 0; uId < pTermCapSet->capabilityDescriptors.count; ++uId)
{
if (pTermCapSet->capabilityDescriptors.value[uId].capabilityDescriptorNumber == CapDescId)
{
p_cap_desc = &pTermCapSet->capabilityDescriptors.value[uId];
break;
}
}
if (p_cap_desc == NULL ||
p_cap_desc->smltnsCpblts == NULL ||
(p_cap_desc->bit_mask & smltnsCpblts_present) == 0)
{
H245TRACE(dwInst,1,"API:H24DelCapDescriptor -> Invalid cap desc id %d",CapDescId);
InstanceUnlock(pInstance);
return H245_ERROR_INVALID_CAPDESCID;
}
/* free up the list */
dealloc_simultaneous_cap (p_cap_desc);
/* (TBC) what if you've removed the last simultaneous cap ? */
/* in this case.. the count does not go down.. it simply */
/* removes the cap descriptor bit from the table.. */
H245TRACE (dwInst,4,"H245DelCapDescriptor -> OK");
InstanceUnlock(pInstance);
return H245_ERROR_OK;
} // H245DelCapDescriptor()
/*****************************************************************************
*
* TYPE: H245 API
*
*****************************************************************************
*
* PROCEDURE: H245SendTermCaps
*
* DESCRIPTION
*
* HRESULT
* H245SendTermCaps (
* H245_INST_T dwInst,
* DWORD dwTransId
* )
*
* Description:
*
* Called to send terminal capabilities to the remote H.245 peer.
* When remote capabilities are receive the client will be
* notified by the H245_IND_CAP indication. When remote side has
* acknowledged the local terminal capabilities and has responded
* with their terminal capabilities the client will receive an
* H245_CONF_ TERMCAP. Between H245Init and H245SendTermCap the
* client may call H245SetLocalCap to register capabilities.
* These capabilities will not be registered to the remote side
* until H245SendTermCap has been called.
*
* Note: As required by the H245 specification, Mutliplex
* capabilities, and Capability descriptors must be
* loaded before the first capability PDU is sent.
*
* Once H245SendTermCap is called, any subsequent calls to
* H245SetLocalTermCap will result in that capability being
* communicated to the remote H.245 peer.
*
* Input
* dwInst Instance handle returned by
* H245GetInstanceId
* dwTransId User supplied object used to identify
* this request in the asynchronous
* response to this call.
* Call Type:
* Asynchronous
*
* Return Values:
* See Errors
*
* Callbacks:
* H245_CONF_TERMCAP
*
* Errors:
* H245_ERROR_OK Function succeeded
* H245_ERROR_NOMEM
* H245_ERROR_INVALID_INST dwInst is not a valid instance handle
* H245_ERROR_NO_MUX_CAPS no Mux capabilities have been set yet
* H245_ERROR_NO_CAPDESC no Capability Descriptors have been set
*
* See Also:
* H245SetLocalCap
* H245Init
*
* callbacks
*
* H245_IND_CAP
* H245_IND_CAPDESC
* H245_IND_CAP_DEL
* H245_IND_CAPDESC_DEL
*
*
*****************************************************************************/
H245DLL HRESULT
H245SendTermCaps (
H245_INST_T dwInst,
DWORD_PTR dwTransId
)
{
struct InstanceStruct *pInstance;
Tracker_T *pTracker;
HRESULT lError;
unsigned char TermCapData = TRUE;
struct TerminalCapabilitySet_capabilityTable TermCap = {0};
H245TRACE(dwInst,4,"H245SendTermCaps <-");
/* check for valid instance handle */
pInstance = InstanceLock(dwInst);
if (pInstance == NULL)
{
H245TRACE(dwInst,1,"H245SendTermCaps -> %s",map_api_error(H245_ERROR_INVALID_INST));
return H245_ERROR_INVALID_INST;
}
/* must have mux parameters set */
if ((pInstance->API.PDU_LocalTermCap.TERMCAPSET.bit_mask & multiplexCapability_present) == 0)
{
H245TRACE(dwInst,1,"H245SendTermCaps -> %s",map_api_error(H245_ERROR_NO_MUX_CAPS));
InstanceUnlock(pInstance);
return H245_ERROR_NO_MUX_CAPS;
}
/* must have capability descriptors set */
if (!(pInstance->API.PDU_LocalTermCap.TERMCAPSET.bit_mask & capabilityDescriptors_present))
{
H245TRACE(dwInst,1,"H245SendTermCaps -> %s",map_api_error(H245_ERROR_NO_CAPDESC));
InstanceUnlock(pInstance);
return H245_ERROR_NO_CAPDESC;
}
if (!(pTracker = alloc_link_tracker (pInstance,
API_TERMCAP_T,
dwTransId,
API_ST_WAIT_RMTACK,
API_CH_ALLOC_UNDEF,
API_CH_TYPE_UNDEF,
0,
H245_INVALID_CHANNEL, H245_INVALID_CHANNEL,
0)))
{
H245TRACE(dwInst,1,"H245SendTermCaps -> %s",map_api_error(H245_ERROR_NOMEM));
InstanceUnlock(pInstance);
return H245_ERROR_NOMEM;
}
pdu_req_termcap_set (&pInstance->API.PDU_LocalTermCap, 0);
TermCap.next = pInstance->API.PDU_LocalTermCap.u.MltmdSystmCntrlMssg_rqst.u.terminalCapabilitySet.capabilityTable;
pInstance->API.PDU_LocalTermCap.u.MltmdSystmCntrlMssg_rqst.u.terminalCapabilitySet.capabilityTable = &TermCap;
TermCap.value.bit_mask = capability_present;
TermCap.value.capabilityTableEntryNumber = pInstance->API.LocalCapIdNum;
TermCap.value.capability.choice = Capability_nonStandard_chosen;
TermCap.value.capability.u.Capability_nonStandard.nonStandardIdentifier.choice = h221NonStandard_chosen;
TermCap.value.capability.u.Capability_nonStandard.nonStandardIdentifier.u.h221NonStandard.t35CountryCode = 0xB5;
TermCap.value.capability.u.Capability_nonStandard.nonStandardIdentifier.u.h221NonStandard.t35Extension = 0x42;
TermCap.value.capability.u.Capability_nonStandard.nonStandardIdentifier.u.h221NonStandard.manufacturerCode = 0x8080;
TermCap.value.capability.u.Capability_nonStandard.data.value = &TermCapData;
TermCap.value.capability.u.Capability_nonStandard.data.length = sizeof(TermCapData);
lError = FsmOutgoing(pInstance, &pInstance->API.PDU_LocalTermCap, (DWORD_PTR)pTracker);
pInstance->API.PDU_LocalTermCap.u.MltmdSystmCntrlMssg_rqst.u.terminalCapabilitySet.capabilityTable = TermCap.next;
if (lError != H245_ERROR_OK)
H245TRACE(dwInst,1,"H245SendTermCaps -> %s",map_api_error(lError));
else
H245TRACE(dwInst,4,"H245SendTermCaps -> OK");
InstanceUnlock(pInstance);
return lError;
} // H245SendTermCaps()
/*****************************************************************************
*
* TYPE: H245 API
*
*****************************************************************************
*
* PROCEDURE: H245EnumCaps
*
* DESCRIPTION
*
* HRESULT H245EnumCaps (
* DWORD dwInst,
* DWORD dwTransId,
* H245_CAPDIR_T Direction,
* H245_DATA_T DataType,
* H245_CLIENT_T ClientType,
* H245_CAP_CALLBACK_T CallBack
* )
*
*
* Callback:
* CallBack (
* DWORD dwTransId,
* H245_TOTCAP_T *pTotCap,
* )
*
* Description:
*
* This function calls the H.245 client back for every
* capability as defined in the API call that complies with the
* request. If the DataType parameter is set to 0 all of the
* caps types are returned (either local or remote based on the
* Direction parameter) no mater what is in the ClientType
* parameter. If the ClientType parameter is 0, it will return
* all of the capabilities of the given DataType.
*
* The user supplied call back is called within the context of
* the call, therefor the call will be considered synchronous.
*
* Input
* dwInst Instance handle returned by H245Init
* Direction Local/Remote Receive, Transmit, or Receive and
* Transmit
* DataType Type of data (Audio, Video, Data, etc.)
* ClientType Client type (H.262, G.711. etc. ).
* dwTransId User supplied object used to identify this
* request in the callback.
*
* CallBack Output
* dwTransId Identical to dwTransId passed in H245EnumCaps
* pTotCap Pointer one of the capabilities.
*
* Note: TotCap parameter must be copied in the callback. This
* data structure is reused for each callback.
*
* Call Type:
* Synchronous Callback - i.e. called back in the context of
* the API call
*
* Errors:
* H245_ERROR_OK
* H245_ERROR_PARAM One or more parameters were invalid
* H245_ERROR_INVALID_INST dwInst is not a valid instance handle.
*
* See Also:
* H245SetLocalCap
* H245ReplLocalCap
*
* callback
*
* H245_IND_CAP
* H245_IND_CAPDESC
*
* RETURN:
*
*****************************************************************************/
H245DLL HRESULT
H245EnumCaps (
H245_INST_T dwInst,
DWORD_PTR dwTransId,
H245_CAPDIR_T Direction,
H245_DATA_T DataType,
H245_CLIENT_T ClientType,
H245_CAP_CALLBACK_T pfCapCallback,
H245_CAPDESC_CALLBACK_T pfCapDescCallback
)
{
register struct InstanceStruct *pInstance;
struct TerminalCapabilitySet *pTermCapSet;
CapabilityTableLink pCapLink;
int lcl_rmt;
H245_TOTCAP_T totcap;
int nResult = 0;
H245TRACE (dwInst,4,"H245EnumCaps <-");
/* check for valid instance handle */
pInstance = InstanceLock(dwInst);
if (pInstance == NULL)
{
H245TRACE (dwInst,1,"H245EnumCaps -> %s",map_api_error(H245_ERROR_INVALID_INST));
return H245_ERROR_INVALID_INST;
}
/* check for callback routine */
if (pfCapCallback == NULL && pfCapDescCallback == NULL)
{
H245TRACE (dwInst,1,"H245EnumCaps -> %s",map_api_error(H245_ERROR_PARAM));
InstanceUnlock(pInstance);
return H245_ERROR_PARAM;
}
/* ok... check the direction.. either remote or local caps.. */
switch (Direction)
{
case H245_CAPDIR_RMTRX:
case H245_CAPDIR_RMTTX:
case H245_CAPDIR_RMTRXTX:
pTermCapSet = &pInstance->API.PDU_RemoteTermCap.u.MltmdSystmCntrlMssg_rqst.u.terminalCapabilitySet;
lcl_rmt = H245_REMOTE;
break;
case H245_CAPDIR_LCLRX:
case H245_CAPDIR_LCLTX:
case H245_CAPDIR_LCLRXTX:
pTermCapSet = &pInstance->API.PDU_LocalTermCap.u.MltmdSystmCntrlMssg_rqst.u.terminalCapabilitySet;
lcl_rmt = H245_LOCAL;
break;
/* must be either local or remote */
// case H245_CAPDIR_DONTCARE:
default:
H245TRACE (dwInst,1,"H245EnumCaps -> %s",map_api_error(H245_ERROR_PARAM));
InstanceUnlock(pInstance);
return H245_ERROR_PARAM;
}
if (pfCapCallback)
{
if (pTermCapSet->bit_mask & multiplexCapability_present &&
build_totcap_from_mux(&totcap, &pTermCapSet->multiplexCapability, Direction) == H245_ERROR_OK)
{
(*pfCapCallback)(dwTransId, &totcap);
}
if (ClientType == H245_CLIENT_DONTCARE)
{
if (DataType == H245_DATA_DONTCARE)
{
for (pCapLink = pTermCapSet->capabilityTable; pCapLink && nResult == 0; pCapLink = pCapLink->next)
{
/* if capability is present */
if (pCapLink->value.bit_mask & capability_present &&
build_totcap_from_captbl(&totcap, pCapLink, lcl_rmt) == H245_ERROR_OK)
{
nResult = (*pfCapCallback)(dwTransId, &totcap);
}
} // for
} // if
else
{
for (pCapLink = pTermCapSet->capabilityTable; pCapLink && nResult == 0; pCapLink = pCapLink->next)
{
/* if capability is present */
if (pCapLink->value.bit_mask & capability_present &&
build_totcap_from_captbl(&totcap, pCapLink, lcl_rmt) == H245_ERROR_OK &&
totcap.DataType == DataType)
{
nResult = (*pfCapCallback)(dwTransId, &totcap);
}
} // for
} // else
} // if
else
{
if (DataType == H245_DATA_DONTCARE)
{
for (pCapLink = pTermCapSet->capabilityTable; pCapLink && nResult == 0; pCapLink = pCapLink->next)
{
/* if capability is present */
if (pCapLink->value.bit_mask & capability_present &&
build_totcap_from_captbl(&totcap, pCapLink, lcl_rmt) == H245_ERROR_OK &&
totcap.ClientType == ClientType)
{
nResult = (*pfCapCallback)(dwTransId, &totcap);
} /* if cap match */
} // for
} // if
else
{
for (pCapLink = pTermCapSet->capabilityTable; pCapLink && nResult == 0; pCapLink = pCapLink->next)
{
/* if capability is present */
if (pCapLink->value.bit_mask & capability_present &&
build_totcap_from_captbl(&totcap,pCapLink,lcl_rmt) == H245_ERROR_OK &&
totcap.ClientType == ClientType &&
totcap.DataType == DataType)
{
nResult = (*pfCapCallback)(dwTransId, &totcap);
}
} // for
} // else
} // else
} // if (pfCapCallback)
if (pfCapDescCallback)
{
// Convert CapabilityDescriptor format to H245_CAPDESC_T format
unsigned int uCapDesc;
register SmltnsCpbltsLink pSimCap;
register unsigned int uAltCap;
H245_TOTCAPDESC_T TotCapDesc;
for (uCapDesc = 0;
uCapDesc < pTermCapSet->capabilityDescriptors.count && nResult == 0;
++uCapDesc)
{
if (pTermCapSet->capabilityDescriptors.value[uCapDesc].bit_mask & smltnsCpblts_present)
{
ASSERT(pTermCapSet->capabilityDescriptors.value[uCapDesc].capabilityDescriptorNumber <= 256);
TotCapDesc.CapDesc.Length = 0;
pSimCap = pTermCapSet->capabilityDescriptors.value[uCapDesc].smltnsCpblts;
ASSERT(pSimCap != NULL);
while (pSimCap)
{
if (TotCapDesc.CapDesc.Length >= H245_MAX_SIMCAPS)
{
H245TRACE (dwInst,1,"H245EnumCaps -> Number of simutaneous capabilities exceeds H245_MAX_SIMCAPS");
InstanceUnlock(pInstance);
return H245_ERROR_MAXTBL;
}
if (pSimCap->value.count > H245_MAX_ALTCAPS)
{
H245TRACE (dwInst,1,"H245EnumCaps -> Number of alternative capabilities exceeds H245_MAX_ALTCAPS");
InstanceUnlock(pInstance);
return H245_ERROR_MAXTBL;
}
TotCapDesc.CapDesc.SimCapArray[TotCapDesc.CapDesc.Length].Length = (WORD) pSimCap->value.count;
for (uAltCap = 0; uAltCap < pSimCap->value.count; ++uAltCap)
{
TotCapDesc.CapDesc.SimCapArray[TotCapDesc.CapDesc.Length].AltCaps[uAltCap] = pSimCap->value.value[uAltCap];
}
TotCapDesc.CapDesc.Length++;
pSimCap = pSimCap->next;
} // while
TotCapDesc.CapDescId = pTermCapSet->capabilityDescriptors.value[uCapDesc].capabilityDescriptorNumber;
nResult = pfCapDescCallback(dwTransId, &TotCapDesc);
} // if
} // for
} // if (pfCapDescCallback)
H245TRACE (dwInst,4,"H245EnumCaps -> OK");
InstanceUnlock(pInstance);
return H245_ERROR_OK;
} // H245EnumCaps()
/*****************************************************************************
*
* TYPE: H245 API
*
* PROCEDURE: H245GetCaps
*
* DESCRIPTION
*
* RETURN:
*
*****************************************************************************/
static H245_TOTCAP_T * * ppTotCapGlobal;
static unsigned long dwTotCapLen;
static unsigned long dwTotCapMax;
static H245_TOTCAPDESC_T * * ppCapDescGlobal;
static unsigned long dwCapDescLen;
static unsigned long dwCapDescMax;
static HRESULT dwGetCapsError;
static int
GetCapsCapCallback(DWORD_PTR dwTransId, H245_TOTCAP_T *pTotCap)
{
H245_TOTCAP_T *pNewCap;
if (dwGetCapsError == H245_ERROR_OK)
{
if (dwTotCapLen >= dwTotCapMax)
{
dwGetCapsError = H245_ERROR_MAXTBL;
}
else
{
dwGetCapsError = H245CopyCap(&pNewCap, pTotCap);
if (dwGetCapsError == H245_ERROR_OK)
{
*ppTotCapGlobal++ = pNewCap;
++dwTotCapLen;
}
}
}
return 0;
} // GetCapsCapCallback()
static int
GetCapsCapDescCallback(DWORD_PTR dwTransId, H245_TOTCAPDESC_T *pCapDesc)
{
H245_TOTCAPDESC_T *pNewCapDesc;
if (dwGetCapsError == H245_ERROR_OK)
{
if (dwCapDescLen >= dwCapDescMax)
{
dwGetCapsError = H245_ERROR_MAXTBL;
}
else
{
dwGetCapsError = H245CopyCapDescriptor(&pNewCapDesc,pCapDesc);
{
*ppCapDescGlobal++ = pNewCapDesc;
++dwCapDescLen;
}
}
}
return 0;
} // GetCapsCapDescCallback()
H245DLL HRESULT
H245GetCaps (
H245_INST_T dwInst,
H245_CAPDIR_T Direction,
H245_DATA_T DataType,
H245_CLIENT_T ClientType,
H245_TOTCAP_T * * ppTotCap,
unsigned long * pdwTotCapLen,
H245_TOTCAPDESC_T * * ppCapDesc,
unsigned long * pdwCapDescLen
)
{
register struct InstanceStruct *pInstance;
H245_CAP_CALLBACK_T CapCallback;
H245_CAPDESC_CALLBACK_T CapDescCallback;
H245TRACE (dwInst,4,"H245GetCaps <-");
/* check for valid instance handle */
pInstance = InstanceLock(dwInst);
if (pInstance == NULL)
{
H245TRACE (dwInst,1,"H245GetCaps -> %s",map_api_error(H245_ERROR_INVALID_INST));
return H245_ERROR_INVALID_INST;
}
dwTotCapLen = 0;
if (ppTotCap == NULL || pdwTotCapLen == NULL || *pdwTotCapLen == 0)
{
CapCallback = NULL;
}
else
{
CapCallback = GetCapsCapCallback;
ppTotCapGlobal = ppTotCap;
dwTotCapMax = *pdwTotCapLen;
}
dwCapDescLen = 0;
if (ppCapDesc == NULL || pdwCapDescLen == NULL || *pdwCapDescLen == 0)
{
CapDescCallback = NULL;
}
else
{
CapDescCallback = GetCapsCapDescCallback;
ppCapDescGlobal = ppCapDesc;
dwCapDescMax = *pdwCapDescLen;
}
/* check parameters */
if (CapCallback == NULL && CapDescCallback == NULL)
{
H245TRACE (dwInst,1,"H245GetCaps -> %s",map_api_error(H245_ERROR_PARAM));
InstanceUnlock(pInstance);
return H245_ERROR_PARAM;
}
dwGetCapsError = H245_ERROR_OK;
H245EnumCaps(dwInst,
0,
Direction,
DataType,
ClientType,
CapCallback,
CapDescCallback);
if (pdwTotCapLen)
*pdwTotCapLen = dwTotCapLen;
if (pdwCapDescLen)
*pdwCapDescLen = dwCapDescLen;
if (dwGetCapsError != H245_ERROR_OK)
H245TRACE (dwInst,1,"H245GetCaps -> %s", map_api_error(dwGetCapsError));
else
H245TRACE (dwInst,4,"H245GetCaps -> OK");
InstanceUnlock(pInstance);
return H245_ERROR_OK;
} // H245GetCaps()
/*****************************************************************************
*
* TYPE: H245 API
*
* PROCEDURE: H245CopyCap
*
* DESCRIPTION
*
* RETURN:
*
*****************************************************************************/
H245DLL HRESULT
H245CopyCap (H245_TOTCAP_T **ppDestTotCap,
const H245_TOTCAP_T *pTotCap)
{
POBJECTID pObjectTo;
POBJECTID pObjectFrom;
HRESULT Status;
if (ppDestTotCap == NULL)
return H245_ERROR_PARAM;
*ppDestTotCap = NULL;
if (pTotCap == NULL)
return H245_ERROR_PARAM;
switch (pTotCap->ClientType)
{
case H245_CLIENT_NONSTD:
case H245_CLIENT_VID_NONSTD:
case H245_CLIENT_AUD_NONSTD:
case H245_CLIENT_MUX_NONSTD:
if (pTotCap->Cap.H245_NonStd.nonStandardIdentifier.choice == object_chosen)
{
*ppDestTotCap = MemAlloc(sizeof(*pTotCap) +
pTotCap->Cap.H245_NonStd.data.length +
ObjectIdLength(&pTotCap->Cap.H245_NonStd.nonStandardIdentifier) * sizeof(OBJECTID));
}
else
{
*ppDestTotCap = MemAlloc(sizeof(*pTotCap) +
pTotCap->Cap.H245_NonStd.data.length);
}
if (*ppDestTotCap == NULL)
return H245_ERROR_NOMEM;
**ppDestTotCap = *pTotCap;
if (pTotCap->Cap.H245_NonStd.data.length != 0)
{
(*ppDestTotCap)->Cap.H245_NonStd.data.value = (unsigned char *)(*ppDestTotCap + 1);
memcpy((*ppDestTotCap)->Cap.H245_NonStd.data.value,
pTotCap->Cap.H245_NonStd.data.value,
pTotCap->Cap.H245_NonStd.data.length);
}
else
{
(*ppDestTotCap)->Cap.H245_NonStd.data.value = NULL;
}
if (pTotCap->Cap.H245_NonStd.nonStandardIdentifier.choice == object_chosen &&
pTotCap->Cap.H245_NonStd.nonStandardIdentifier.u.object != NULL)
{
pObjectTo = (POBJECTID)(((unsigned char *)(*ppDestTotCap + 1)) +
pTotCap->Cap.H245_NonStd.data.length);
(*ppDestTotCap)->Cap.H245_NonStd.nonStandardIdentifier.u.object = pObjectTo;
pObjectFrom = pTotCap->Cap.H245_NonStd.nonStandardIdentifier.u.object;
do
{
pObjectTo->value = pObjectFrom->value;
pObjectTo->next = pObjectTo + 1;
++pObjectTo;
} while ((pObjectFrom = pObjectFrom->next) != NULL);
--pObjectTo;
pObjectTo->next = NULL;
}
break;
case H245_CLIENT_DAT_NONSTD:
if (pTotCap->Cap.H245Dat_NONSTD.application.u.DACy_applctn_nnStndrd.nonStandardIdentifier.choice == object_chosen)
{
*ppDestTotCap = MemAlloc(sizeof(*pTotCap) +
pTotCap->Cap.H245Dat_NONSTD.application.u.DACy_applctn_nnStndrd.data.length +
ObjectIdLength(&pTotCap->Cap.H245Dat_NONSTD.application.u.DACy_applctn_nnStndrd.nonStandardIdentifier) * sizeof(OBJECTID));
}
else
{
*ppDestTotCap = MemAlloc(sizeof(*pTotCap) +
pTotCap->Cap.H245Dat_NONSTD.application.u.DACy_applctn_nnStndrd.data.length);
}
if (*ppDestTotCap == NULL)
return H245_ERROR_NOMEM;
**ppDestTotCap = *pTotCap;
if (pTotCap->Cap.H245_NonStd.data.length != 0)
{
(*ppDestTotCap)->Cap.H245Dat_NONSTD.application.u.DACy_applctn_nnStndrd.data.value =
(unsigned char *)(*ppDestTotCap + 1);
memcpy((*ppDestTotCap)->Cap.H245Dat_NONSTD.application.u.DACy_applctn_nnStndrd.data.value,
pTotCap->Cap.H245Dat_NONSTD.application.u.DACy_applctn_nnStndrd.data.value,
pTotCap->Cap.H245Dat_NONSTD.application.u.DACy_applctn_nnStndrd.data.length);
}
else
{
(*ppDestTotCap)->Cap.H245Dat_NONSTD.application.u.DACy_applctn_nnStndrd.data.value = NULL;
}
if (pTotCap->Cap.H245Dat_NONSTD.application.u.DACy_applctn_nnStndrd.nonStandardIdentifier.choice == object_chosen &&
pTotCap->Cap.H245Dat_NONSTD.application.u.DACy_applctn_nnStndrd.nonStandardIdentifier.u.object != NULL)
{
pObjectTo = (POBJECTID)(((unsigned char *)(*ppDestTotCap + 1)) +
pTotCap->Cap.H245Dat_NONSTD.application.u.DACy_applctn_nnStndrd.data.length);
(*ppDestTotCap)->Cap.H245Dat_NONSTD.application.u.DACy_applctn_nnStndrd.nonStandardIdentifier.u.object = pObjectTo;
pObjectFrom = pTotCap->Cap.H245Dat_NONSTD.application.u.DACy_applctn_nnStndrd.nonStandardIdentifier.u.object;
do
{
pObjectTo->value = pObjectFrom->value;
pObjectTo->next = pObjectTo + 1;
++pObjectTo;
} while ((pObjectFrom = pObjectFrom->next) != NULL);
--pObjectTo;
pObjectTo->next = NULL;
}
break;
case H245_CLIENT_DAT_T120:
case H245_CLIENT_DAT_DSMCC:
case H245_CLIENT_DAT_USERDATA:
case H245_CLIENT_DAT_T434:
case H245_CLIENT_DAT_H224:
case H245_CLIENT_DAT_H222:
if (pTotCap->Cap.H245Dat_T120.application.u.DACy_applctn_t120.choice == DtPrtclCpblty_nnStndrd_chosen)
{
if (pTotCap->Cap.H245Dat_T120.application.u.DACy_applctn_t120.u.DtPrtclCpblty_nnStndrd.nonStandardIdentifier.choice == object_chosen)
{
*ppDestTotCap = MemAlloc(sizeof(*pTotCap) +
pTotCap->Cap.H245Dat_T120.application.u.DACy_applctn_t120.u.DtPrtclCpblty_nnStndrd.data.length +
ObjectIdLength(&pTotCap->Cap.H245Dat_T120.application.u.DACy_applctn_t120.u.DtPrtclCpblty_nnStndrd.nonStandardIdentifier) * sizeof(OBJECTID));
}
else
{
*ppDestTotCap = MemAlloc(sizeof(*pTotCap) +
pTotCap->Cap.H245Dat_T120.application.u.DACy_applctn_t120.u.DtPrtclCpblty_nnStndrd.data.length);
}
if (*ppDestTotCap == NULL)
return H245_ERROR_NOMEM;
**ppDestTotCap = *pTotCap;
if (pTotCap->Cap.H245_NonStd.data.length != 0)
{
(*ppDestTotCap)->Cap.H245Dat_T120.application.u.DACy_applctn_t120.u.DtPrtclCpblty_nnStndrd.data.value =
(unsigned char *)(*ppDestTotCap + 1);
memcpy((*ppDestTotCap)->Cap.H245Dat_T120.application.u.DACy_applctn_t120.u.DtPrtclCpblty_nnStndrd.data.value,
pTotCap->Cap.H245Dat_T120.application.u.DACy_applctn_t120.u.DtPrtclCpblty_nnStndrd.data.value,
pTotCap->Cap.H245Dat_T120.application.u.DACy_applctn_t120.u.DtPrtclCpblty_nnStndrd.data.length);
}
else
{
(*ppDestTotCap)->Cap.H245Dat_T120.application.u.DACy_applctn_t120.u.DtPrtclCpblty_nnStndrd.data.value = NULL;
}
if (pTotCap->Cap.H245Dat_T120.application.u.DACy_applctn_t120.u.DtPrtclCpblty_nnStndrd.nonStandardIdentifier.choice == object_chosen &&
pTotCap->Cap.H245Dat_T120.application.u.DACy_applctn_t120.u.DtPrtclCpblty_nnStndrd.nonStandardIdentifier.u.object != NULL)
{
pObjectTo = (POBJECTID)(((unsigned char *)(*ppDestTotCap + 1)) +
pTotCap->Cap.H245Dat_T120.application.u.DACy_applctn_t120.u.DtPrtclCpblty_nnStndrd.data.length);
(*ppDestTotCap)->Cap.H245Dat_T120.application.u.DACy_applctn_t120.u.DtPrtclCpblty_nnStndrd.nonStandardIdentifier.u.object = pObjectTo;
pObjectFrom = pTotCap->Cap.H245Dat_T120.application.u.DACy_applctn_t120.u.DtPrtclCpblty_nnStndrd.nonStandardIdentifier.u.object;
do
{
pObjectTo->value = pObjectFrom->value;
pObjectTo->next = pObjectTo + 1;
++pObjectTo;
} while ((pObjectFrom = pObjectFrom->next) != NULL);
--pObjectTo;
pObjectTo->next = NULL;
}
}
else
{
*ppDestTotCap = MemAlloc(sizeof(*pTotCap));
if (*ppDestTotCap == NULL)
return H245_ERROR_NOMEM;
**ppDestTotCap = *pTotCap;
}
break;
case H245_CLIENT_CONFERENCE:
{
NonStandardDataLink pList;
NonStandardDataLink pFrom;
NonStandardDataLink pTo;
// Initialize Status here to prevent compiler warning "returning a possibly
// uninitialized value"
Status = H245_ERROR_NOMEM;
*ppDestTotCap = MemAlloc(sizeof(*pTotCap));
if (*ppDestTotCap == NULL)
return H245_ERROR_NOMEM;
**ppDestTotCap = *pTotCap;
pList = NULL;
(*ppDestTotCap)->Cap.H245Conference.nonStandardData = NULL;
pFrom = pTotCap->Cap.H245Conference.nonStandardData;
while (pFrom)
{
pTo = MemAlloc(sizeof(*pTo));
if (pTo == NULL)
Status = H245_ERROR_NOMEM;
if (pTo != NULL)
{
Status = CopyNonStandardParameter(&pTo->value, &pFrom->value);
if (Status != H245_ERROR_OK)
{
MemFree(pTo);
pTo = NULL;
}
}
if (pTo == NULL)
{
while (pList)
{
pTo = pList;
pList = pList->next;
FreeNonStandardParameter(&pTo->value);
MemFree(pTo);
}
MemFree(*ppDestTotCap);
*ppDestTotCap = NULL;
return Status;
}
pTo->next = pList;
pList = pTo;
pFrom = pFrom->next;
} // while
while (pList)
{
pTo = pList;
pList = pList->next;
pTo->next = (*ppDestTotCap)->Cap.H245Conference.nonStandardData;
(*ppDestTotCap)->Cap.H245Conference.nonStandardData = pTo;
} // while
break;
}
case H245_CLIENT_MUX_H222:
{
VCCapabilityLink pList = NULL;
VCCapabilityLink pFrom;
VCCapabilityLink pTo;
*ppDestTotCap = MemAlloc(sizeof(*pTotCap));
if (*ppDestTotCap == NULL)
return H245_ERROR_NOMEM;
**ppDestTotCap = *pTotCap;
(*ppDestTotCap)->Cap.H245Mux_H222.vcCapability = NULL;
pFrom = pTotCap->Cap.H245Mux_H222.vcCapability;
while (pFrom)
{
pTo = MemAlloc(sizeof(*pTo));
if (pTo == NULL)
{
while (pList)
{
pTo = pList;
pList = pList->next;
MemFree(pTo);
}
MemFree(*ppDestTotCap);
*ppDestTotCap = NULL;
return H245_ERROR_NOMEM;
}
pTo->value = pFrom->value;
pTo->next = pList;
pList = pTo;
pFrom = pFrom->next;
} // while
while (pList)
{
pTo = pList;
pList = pList->next;
pTo->next = (*ppDestTotCap)->Cap.H245Mux_H222.vcCapability;
(*ppDestTotCap)->Cap.H245Mux_H222.vcCapability = pList;
} // while
break;
}
case H245_CLIENT_MUX_H2250:
*ppDestTotCap = MemAlloc(sizeof(*pTotCap));
if (*ppDestTotCap == NULL)
return H245_ERROR_NOMEM;
**ppDestTotCap = *pTotCap;
Status = CopyH2250Cap(&(*ppDestTotCap)->Cap.H245Mux_H2250, &pTotCap->Cap.H245Mux_H2250);
if (Status != H245_ERROR_OK)
{
MemFree(*ppDestTotCap);
*ppDestTotCap = NULL;
return Status;
}
break;
default:
*ppDestTotCap = MemAlloc(sizeof(*pTotCap));
if (*ppDestTotCap == NULL)
return H245_ERROR_NOMEM;
**ppDestTotCap = *pTotCap;
} // switch
return H245_ERROR_OK;
} // H245CopyCap()
/*****************************************************************************
*
* TYPE: H245 API
*
* PROCEDURE: H245FreeCap
*
* DESCRIPTION
*
* RETURN:
*
*****************************************************************************/
H245DLL HRESULT
H245FreeCap (H245_TOTCAP_T * pTotCap)
{
if (pTotCap == NULL)
{
return H245_ERROR_PARAM;
}
switch (pTotCap->ClientType)
{
case H245_CLIENT_CONFERENCE:
{
NonStandardDataLink pList;
NonStandardDataLink pTo;
pList = pTotCap->Cap.H245Conference.nonStandardData;
while (pList)
{
pTo = pList;
pList = pList->next;
FreeNonStandardParameter(&pTo->value);
MemFree(pTo);
}
}
break;
case H245_CLIENT_MUX_H222:
{
VCCapabilityLink pList;
VCCapabilityLink pTo;
pList = pTotCap->Cap.H245Mux_H222.vcCapability;
while (pList)
{
pTo = pList;
pList = pList->next;
MemFree(pTo);
}
}
break;
case H245_CLIENT_MUX_H2250:
FreeH2250Cap(&pTotCap->Cap.H245Mux_H2250);
break;
} // switch
MemFree(pTotCap);
return 0;
} // H245FreeCap()
/*****************************************************************************
*
* TYPE: H245 API
*
* PROCEDURE: H245CopyCapDescriptor
*
* DESCRIPTION
*
* RETURN:
*
*****************************************************************************/
H245DLL HRESULT
H245CopyCapDescriptor (H245_TOTCAPDESC_T **ppDestCapDesc,
const H245_TOTCAPDESC_T *pCapDesc)
{
if (ppDestCapDesc == NULL)
return H245_ERROR_PARAM;
*ppDestCapDesc = NULL;
if (pCapDesc == NULL)
return H245_ERROR_PARAM;
*ppDestCapDesc = MemAlloc(sizeof(**ppDestCapDesc));
if (*ppDestCapDesc == NULL)
return H245_ERROR_NOMEM;
**ppDestCapDesc = *pCapDesc;
return H245_ERROR_OK;
} // H245CopyCapDescriptor()
/*****************************************************************************
*
* TYPE: H245 API
*
* PROCEDURE: H245FreeCapDescriptor
*
* DESCRIPTION
*
* RETURN:
*
*****************************************************************************/
H245DLL HRESULT
H245FreeCapDescriptor (H245_TOTCAPDESC_T * pCapDesc)
{
if (pCapDesc == NULL)
{
return H245_ERROR_PARAM;
}
MemFree(pCapDesc);
return 0;
} // H245FreeCapDescriptor()
/*****************************************************************************
*
* TYPE: H245 API
*
* PROCEDURE: H245CopyMux
*
* DESCRIPTION
*
* RETURN:
*
*****************************************************************************/
H245DLL H245_MUX_T *
H245CopyMux (const H245_MUX_T * pMux)
{
register unsigned int uLength;
register H245_MUX_T *pNew;
H2250LCPs_nnStndrdLink pList;
H2250LCPs_nnStndrdLink pFrom;
H2250LCPs_nnStndrdLink pTo;
switch (pMux->Kind)
{
case H245_H222:
uLength = sizeof(*pMux) +
pMux->u.H222.programDescriptors.length +
pMux->u.H222.streamDescriptors.length;
pNew = MemAlloc(uLength);
if (pNew != NULL)
{
*pNew = *pMux;
if (pMux->u.H222.programDescriptors.length != 0)
{
pNew->u.H222.programDescriptors.value = (unsigned char *)(pNew + 1);
memcpy(pNew->u.H222.programDescriptors.value,
pMux->u.H222.programDescriptors.value,
pMux->u.H222.programDescriptors.length);
}
else
{
pNew->u.H222.programDescriptors.value = NULL;
}
if (pMux->u.H222.streamDescriptors.length != 0)
{
pNew->u.H222.streamDescriptors.value = ((unsigned char *)pNew) +
(uLength - pMux->u.H222.streamDescriptors.length);
memcpy(pNew->u.H222.streamDescriptors.value,
pMux->u.H222.streamDescriptors.value,
pMux->u.H222.streamDescriptors.length);
}
else
{
pNew->u.H222.streamDescriptors.value = NULL;
}
}
break;
case H245_H223:
pNew = MemAlloc(sizeof(*pMux) + pMux->u.H223.H223_NONSTD.data.length);
if (pNew != NULL)
{
*pNew = *pMux;
if (pMux->u.H223.H223_NONSTD.data.length != 0)
{
pNew->u.H223.H223_NONSTD.data.value = (unsigned char *)(pNew + 1);
memcpy(pNew->u.H223.H223_NONSTD.data.value,
pMux->u.H223.H223_NONSTD.data.value,
pMux->u.H223.H223_NONSTD.data.length);
}
else
{
pNew->u.H223.H223_NONSTD.data.value = NULL;
}
}
break;
case H245_H2250:
case H245_H2250ACK:
// Caveat: assumes nonstandard list, mediaChannel and mediaControlChannel
// in same place in both structures
if (pMux->u.H2250.mediaChannelPresent &&
(pMux->u.H2250.mediaChannel.type == H245_IPSSR_UNICAST ||
pMux->u.H2250.mediaChannel.type == H245_IPLSR_UNICAST) &&
pMux->u.H2250.mediaChannel.u.ipSourceRoute.route != NULL &&
pMux->u.H2250.mediaChannel.u.ipSourceRoute.dwCount != 0)
{
if (pMux->u.H2250.mediaControlChannelPresent &&
(pMux->u.H2250.mediaControlChannel.type == H245_IPSSR_UNICAST ||
pMux->u.H2250.mediaControlChannel.type == H245_IPLSR_UNICAST) &&
pMux->u.H2250.mediaControlChannel.u.ipSourceRoute.route != NULL &&
pMux->u.H2250.mediaControlChannel.u.ipSourceRoute.dwCount != 0)
{
unsigned int uLength2;
uLength = pMux->u.H2250.mediaChannel.u.ipSourceRoute.dwCount << 2;
uLength2 = pMux->u.H2250.mediaControlChannel.u.ipSourceRoute.dwCount << 2;
pNew = MemAlloc(sizeof(*pMux) + uLength + uLength2);
if (pNew != NULL)
{
*pNew = *pMux;
pNew->u.H2250.mediaChannel.u.ipSourceRoute.route = (unsigned char *) (pNew + 1);
pNew->u.H2250.mediaControlChannel.u.ipSourceRoute.route =
pNew->u.H2250.mediaChannel.u.ipSourceRoute.route + uLength;
memcpy(pNew->u.H2250.mediaChannel.u.ipSourceRoute.route,
pMux->u.H2250.mediaChannel.u.ipSourceRoute.route,
uLength);
memcpy(pNew->u.H2250.mediaControlChannel.u.ipSourceRoute.route,
pMux->u.H2250.mediaControlChannel.u.ipSourceRoute.route,
uLength2);
}
}
else
{
uLength = pMux->u.H2250.mediaChannel.u.ipSourceRoute.dwCount << 2;
pNew = MemAlloc(sizeof(*pMux) + uLength);
if (pNew != NULL)
{
*pNew = *pMux;
pNew->u.H2250.mediaChannel.u.ipSourceRoute.route = (unsigned char *) (pNew + 1);
memcpy(pNew->u.H2250.mediaChannel.u.ipSourceRoute.route,
pMux->u.H2250.mediaChannel.u.ipSourceRoute.route,
uLength);
}
}
}
else if (pMux->u.H2250.mediaControlChannelPresent &&
(pMux->u.H2250.mediaControlChannel.type == H245_IPSSR_UNICAST ||
pMux->u.H2250.mediaControlChannel.type == H245_IPLSR_UNICAST) &&
pMux->u.H2250.mediaControlChannel.u.ipSourceRoute.route != NULL &&
pMux->u.H2250.mediaControlChannel.u.ipSourceRoute.dwCount != 0)
{
uLength = pMux->u.H2250.mediaControlChannel.u.ipSourceRoute.dwCount << 2;
pNew = MemAlloc(sizeof(*pMux) + uLength);
if (pNew != NULL)
{
*pNew = *pMux;
pNew->u.H2250.mediaControlChannel.u.ipSourceRoute.route = (unsigned char *) (pNew + 1);
memcpy(pNew->u.H2250.mediaControlChannel.u.ipSourceRoute.route,
pMux->u.H2250.mediaControlChannel.u.ipSourceRoute.route,
uLength);
}
}
else
{
pNew = MemAlloc(sizeof(*pMux));
if (pNew != NULL)
{
*pNew = *pMux;
}
}
pList = NULL;
pNew->u.H2250.nonStandardList = NULL;
pFrom = pMux->u.H2250.nonStandardList;
while (pFrom)
{
pTo = MemAlloc(sizeof(*pTo));
if (pTo != NULL)
{
if (CopyNonStandardParameter(&pTo->value, &pFrom->value) != H245_ERROR_OK)
{
MemFree(pTo);
pTo = NULL;
}
}
if (pTo == NULL)
{
while (pList)
{
pTo = pList;
pList = pList->next;
FreeNonStandardParameter(&pTo->value);
MemFree(pTo);
}
MemFree(pNew);
return NULL;
}
pTo->next = pList;
pList = pTo;
pFrom = pFrom->next;
} // while
while (pList)
{
pTo = pList;
pList = pList->next;
pTo->next = pNew->u.H2250.nonStandardList;
pNew->u.H2250.nonStandardList = pTo;
} // while
break;
// case H245_VGMUX:
default:
pNew = MemAlloc(sizeof(*pMux));
if (pNew != NULL)
{
*pNew = *pMux;
}
} // switch
return pNew;
} // H245CopyMux()
/*****************************************************************************
*
* TYPE: H245 API
*
* PROCEDURE: H245FreeMux
*
* DESCRIPTION
*
* RETURN:
*
*****************************************************************************/
H245DLL HRESULT
H245FreeMux (H245_MUX_T * pMux)
{
H2250LCPs_nnStndrdLink pLink;
if (pMux == NULL)
{
return H245_ERROR_PARAM;
}
switch (pMux->Kind)
{
case H245_H2250:
case H245_H2250ACK:
// Caveat: assumes nonstandard list is in same place in both structures
while (pMux->u.H2250.nonStandardList)
{
pLink = pMux->u.H2250.nonStandardList;
pMux->u.H2250.nonStandardList = pLink->next;
FreeNonStandardParameter(&pLink->value);
MemFree(pLink);
}
break;
} // switch
MemFree(pMux);
return 0;
} // H245FreeMux()
/*****************************************************************************
*
* TYPE: H245 API
*
*****************************************************************************
*
* PROCEDURE: H245OpenChannel
*
* DESCRIPTION
*
* HRESULT H245OpenChannel (
* H245_INST_T dwInst,
* DWORD dwTransId,
* DWORD dwTxChannel,
* H245_TOTCAP_T *pTxMode,
* H245_MUX_T *pTxMux,
* H245_TOTCAP_T *pRxMode, (* bi-dir only *)
* H245_MUX_T *pRxMux (* bi-dir only *)
* )
*
* Description:
* This function is called to open either a uni-directional,
* or a bi-directional channel. The mode to the remote peer
* will be designated by the *pTxMode.. To open a bi-directional
* channel the client selects a non-null receive mode ( *pRxMode).
* This mode indicates to the remote peer its transmit mode.
* For uni-directional channels the *pRxMode must be NULL.
*
* The dwTxChannel parameter indicates which forward logical
* channel the H.245 will open. If this is a bi-directional
* channel open, the confirm will indicate the logical channel
* specified in the open request by the remote terminal
*
* The pMux parameter will contain a pointer to H.223, H.222,
* VGMUX, or other logical channel parameters depending on the
* system configuration. (see H245_H223_LOGICAL_PARAM). This
* may be NULL for some clients.
*
* Note:
* 7 pTxMode->CapId is of no significance in this call.
* It is not used
* 7 pRxMode->CapId is of no significance in this call.
* It is not used
*
* Input
* dwInst Instance handle returned by H245Init
* dwTransId User supplied object used to identify this
* request in the asynchronous confirm to this
* call.
* dwTxChannel Logical Channel number for forward (Transmit)
* Channel
* pTxMode The capability (mode) used for transmission
* to the remote peer.
* Note: pTxMode->CapId is ignored
* pTxMux The formward logical channel parameters
* for H.223, H.222, VGMUX, etc.
* pRxMode Optional: Transmit mode specified for the
* remote terminal. This is used only for
* Bi-directional Channel opens and must be set
* to NULL if opening a Uni-directional channel.
* Note: pRxMode->CapId is ignored
* pRxMux Optional : The reverse logical channel
* parameters for H.223, H.222, VGMUX, etc. or
* NULL.
*
* Call Type:
* Asynchronous
*
* Return Values:
* See Errors
*
* Callback:
* H245_CONF_OPEN
* H245_CONF_NEEDRSP_OPEN Bi-Directional Channels only
* waiting for confirm.
* Errors:
* H245_ERROR_OK
* H245_ERROR_PARAM One or more parameters were
* invalid
* H245_ERROR_BANDWIDTH_OVERFLOW Open would exceed bandwidth
* limitations
* H245_ERROR_NOMEM
* H245_ERROR_NORESOURCE Out of resources, too many
* open channels or outside scope
* of simultaneous capabilities.
* H245_ERROR_INVALID_INST dwInst is not a valid instance
* handle
* H245_ERROR_INVALID_STATE Not in the proper state to
* issue open
* H245_ERROR_CHANNEL_INUSE Channel is currently open
*
* See Also:
* H245CloseChannel
* H245OpenChannelIndResp
* H245OpenChannelConfResp
*
* callback
*
* H245_CONF_OPEN
* H245_CONF_NEEDRSP_OPEN
* H245_IND_OPEN_CONF
*
*
* RETURN:
*
*****************************************************************************/
H245DLL HRESULT
H245OpenChannel (
H245_INST_T dwInst,
DWORD_PTR dwTransId,
H245_CHANNEL_T wTxChannel,
const H245_TOTCAP_T * pTxMode,
const H245_MUX_T * pTxMux,
H245_PORT_T dwTxPort, // optional
const H245_TOTCAP_T * pRxMode, // bi-dir only
const H245_MUX_T * pRxMux, // bi-dir only
const H245_ACCESS_T * pSeparateStack // optional
)
{
register struct InstanceStruct *pInstance;
HRESULT lError;
Tracker_T *pTracker;
MltmdSystmCntrlMssg *pPdu;
H245TRACE (dwInst,4,"H245OpenChannel <-");
/* check for valid instance handle */
pInstance = InstanceLock(dwInst);
if (pInstance == NULL)
{
H245TRACE (dwInst,1,"H245OpenChannel -> %s",map_api_error(H245_ERROR_INVALID_INST));
return H245_ERROR_INVALID_INST;
}
/* system should be in connected state */
if (pInstance->API.SystemState != APIST_Connected)
{
H245TRACE (dwInst,1,"H245OpenChannel -> %s",map_api_error(H245_ERROR_NOT_CONNECTED));
InstanceUnlock(pInstance);
return H245_ERROR_NOT_CONNECTED;
}
/* must have performed Master Slave Negotiation at this point */
if (pInstance->Configuration == H245_CONF_H324 &&
pInstance->API.MasterSlave != APIMS_Master &&
pInstance->API.MasterSlave != APIMS_Slave)
{
H245TRACE (dwInst,1,"H245OpenChannel -> %s",map_api_error(H245_ERROR_INVALID_STATE));
InstanceUnlock(pInstance);
return H245_ERROR_INVALID_STATE;
}
pTracker = find_tracker_by_txchannel (pInstance, wTxChannel, API_CH_ALLOC_LCL);
/* channel is currently in use */
if (pTracker)
{
H245TRACE (dwInst,1,"H245OpenChannel -> %s",map_api_error(H245_ERROR_CHANNEL_INUSE));
InstanceUnlock(pInstance);
return H245_ERROR_CHANNEL_INUSE;
}
if (!(pPdu = (MltmdSystmCntrlMssg *)MemAlloc(sizeof(MltmdSystmCntrlMssg))))
{
H245TRACE (dwInst,1,"H245OpenChannel -> %s",map_api_error(H245_ERROR_NOMEM));
InstanceUnlock(pInstance);
return H245_ERROR_NOMEM;
}
memset(pPdu, 0, sizeof(*pPdu));
/* lock in the transmit side.. */
/* wait until OpenChannelConfResp to setup RxChannel */
lError = pdu_req_open_channel(pPdu,
wTxChannel,
dwTxPort, /* forward port */
pTxMode,
pTxMux,
pRxMode,
pRxMux,
pSeparateStack);
if (lError != H245_ERROR_OK)
{
H245TRACE (dwInst,1,"H245OpenChannel -> %s",map_api_error(lError));
free_pdu_req_open_channel(pPdu, pTxMode, pRxMode);
InstanceUnlock(pInstance);
return lError;
}
/* if allocation error */
if (!(pTracker = alloc_link_tracker (pInstance,
API_OPEN_CHANNEL_T,
dwTransId,
API_ST_WAIT_RMTACK,
API_CH_ALLOC_LCL,
(pRxMode?API_CH_TYPE_BI:API_CH_TYPE_UNI),
pTxMode->DataType,
wTxChannel, H245_INVALID_CHANNEL,
0)))
{
H245TRACE (dwInst,1,"H245OpenChannel -> %s",map_api_error(H245_ERROR_NOMEM));
free_pdu_req_open_channel(pPdu, pTxMode, pRxMode);
InstanceUnlock(pInstance);
return H245_ERROR_NOMEM;
}
lError = FsmOutgoing(pInstance, pPdu, (DWORD_PTR)pTracker);
free_pdu_req_open_channel(pPdu, pTxMode, pRxMode);
if (lError != H245_ERROR_OK)
{
unlink_dealloc_tracker (pInstance, pTracker);
H245TRACE (dwInst,1,"H245OpenChannel -> %s",map_api_error(lError));
}
else
H245TRACE (dwInst,4,"H245OpenChannel -> OK");
InstanceUnlock(pInstance);
return lError;
} // H245OpenChannel()
/*****************************************************************************
*
* TYPE: H245 API
*
* PROCEDURE: H245OpenChannelAccept
*
* DESCRIPTION
*
* RETURN:
*
*****************************************************************************/
H245DLL HRESULT
H245OpenChannelAccept (
H245_INST_T dwInst,
DWORD_PTR dwTransId,
H245_CHANNEL_T wRxChannel, // RxChannel from IND_OPEN
const H245_MUX_T * pRxMux, // optional H2250LogicalChannelAckParameters
H245_CHANNEL_T wTxChannel, // bi-dir only
const H245_MUX_T * pTxMux, // bi-dir only optional H2250LogicalChannelParameters
H245_PORT_T dwTxPort, // bi-dir only optional
const H245_ACCESS_T * pSeparateStack // optional
)
{
register struct InstanceStruct *pInstance;
Tracker_T * pTracker;
MltmdSystmCntrlMssg * pPdu;
HRESULT lError;
H245TRACE (dwInst,4,"H245OpenChannelAccept <- wRxChannel=%d wTxChannel=%d",
wRxChannel, wTxChannel);
/* check for valid instance handle */
pInstance = InstanceLock(dwInst);
if (pInstance == NULL)
{
H245TRACE (dwInst,1,"H245OpenChannelAccept -> %s",map_api_error(H245_ERROR_INVALID_INST));
return H245_ERROR_INVALID_INST;
}
/* system should be in connected state */
if (pInstance->API.SystemState != APIST_Connected)
{
H245TRACE (dwInst,1,"H245OpenChannelAccept -> %s",map_api_error(H245_ERROR_NOT_CONNECTED));
InstanceUnlock(pInstance);
return H245_ERROR_NOT_CONNECTED;
}
/* if channel not found in tracker list */
/* note that it looks for given channel given who alloced it */
pTracker = find_tracker_by_rxchannel (pInstance, wRxChannel, API_CH_ALLOC_RMT);
/* not locking tracker.. since no indication will come in until I issue the request */
if (!pTracker)
{
H245TRACE (dwInst,1,"H245OpenChannelAccept -> %s",map_api_error(H245_ERROR_INVALID_OP));
InstanceUnlock(pInstance);
return H245_ERROR_INVALID_OP;
}
/* if not open.. invalid op */
if (pTracker->TrackerType != API_OPEN_CHANNEL_T)
{
H245TRACE (dwInst,1,"H245OpenChannelAccept -> %s",map_api_error(H245_ERROR_INVALID_OP));
InstanceUnlock(pInstance);
return H245_ERROR_INVALID_OP;
}
/* if was uni open w/ TxChannel set.. error */
/* -or - */
/* if was bi open w/ !TxChannel set.. error */
/* AND it wasn't a reject */
if (pTracker->u.Channel.ChannelType == API_CH_TYPE_BI && wTxChannel == 0)
{
H245TRACE (dwInst,1,"H245OpenChannelAccept -> %s",map_api_error(H245_ERROR_PARAM));
InstanceUnlock(pInstance);
return H245_ERROR_PARAM;
}
/* for debug */
ASSERT (pTracker->u.Channel.RxChannel == wRxChannel);
ASSERT (pTracker->u.Channel.ChannelAlloc == API_CH_ALLOC_RMT);
/* check state.. must be returning.. */
if (pTracker->State != API_ST_WAIT_LCLACK)
{
H245TRACE (dwInst,1,"H245OpenChannelAccept -> %s",map_api_error(H245_ERROR_INVALID_STATE));
InstanceUnlock(pInstance);
return H245_ERROR_INVALID_STATE;
}
/* setup tracker object for new transaction */
pTracker->TransId = dwTransId;
pPdu = (MltmdSystmCntrlMssg *)MemAlloc(sizeof(*pPdu));
if (pPdu == NULL)
{
H245TRACE (dwInst,1,"H245OpenChannelAccept -> %s",map_api_error(H245_ERROR_NOMEM));
InstanceUnlock(pInstance);
return H245_ERROR_NOMEM;
}
memset(pPdu, 0, sizeof(*pPdu));
switch (pTracker->u.Channel.ChannelType)
{
case API_CH_TYPE_UNI:
pTracker->State = API_ST_IDLE;
pTracker->u.Channel.TxChannel = 0;
lError = pdu_rsp_open_logical_channel_ack(pPdu,
wRxChannel,
pRxMux,
0,
NULL,
dwTxPort,
pSeparateStack);
if (lError != H245_ERROR_OK)
{
// If parameter error, we don't want to deallocate tracker
MemFree (pPdu);
InstanceUnlock(pInstance);
return lError;
}
lError = FsmOutgoing(pInstance, pPdu, (DWORD_PTR)pTracker);
break;
case API_CH_TYPE_BI:
pTracker->State = API_ST_WAIT_CONF; /* waiting for confirmation */
pTracker->u.Channel.TxChannel = wTxChannel;
lError = pdu_rsp_open_logical_channel_ack(pPdu,
wRxChannel,
pRxMux,
wTxChannel,
pTxMux,
dwTxPort,
pSeparateStack);
if (lError != H245_ERROR_OK)
{
// If parameter error, we don't want to deallocate tracker
MemFree (pPdu);
InstanceUnlock(pInstance);
return lError;
}
lError = FsmOutgoing(pInstance, pPdu, (DWORD_PTR)pTracker);
break;
default:
H245TRACE (dwInst,1,"H245OpenChannelAccept: Invalid Tracker Channel Type %d",
pTracker->u.Channel.ChannelType);
lError = H245_ERROR_FATAL;
} // switch
MemFree (pPdu);
switch (lError)
{
case H245_ERROR_OK:
H245TRACE (dwInst,4,"H245OpenChannelAccept -> OK");
break;
default:
// Deallocate tracker object for all errors except parameter error
unlink_dealloc_tracker (pInstance, pTracker);
// Fall-through to next case is intentional
case H245_ERROR_PARAM:
// If parameter error, we don't want to deallocate tracker
H245TRACE (dwInst,1,"H245OpenChannelAccept -> %s",map_api_error(lError));
} // switch
InstanceUnlock(pInstance);
return lError;
} // H245OpenChannelAccept()
/*****************************************************************************
*
* TYPE: H245 API
*
* PROCEDURE: H245OpenChannelReject
*
* DESCRIPTION
*
* RETURN:
*
*****************************************************************************/
H245DLL HRESULT
H245OpenChannelReject (
H245_INST_T dwInst,
H245_CHANNEL_T wRxChannel, // RxChannel from IND_OPEN
unsigned short wCause
)
{
register struct InstanceStruct *pInstance;
Tracker_T * pTracker;
MltmdSystmCntrlMssg * pPdu;
HRESULT lError;
H245TRACE (dwInst,4,"H245OpenChannelReject <- wRxChannel=%d", wRxChannel);
/* check for valid instance handle */
pInstance = InstanceLock(dwInst);
if (pInstance == NULL)
{
H245TRACE (dwInst,1,"H245OpenChannelReject -> %s",map_api_error(H245_ERROR_INVALID_INST));
return H245_ERROR_INVALID_INST;
}
/* system should be in connected state */
if (pInstance->API.SystemState != APIST_Connected)
{
H245TRACE (dwInst,1,"H245OpenChannelReject -> %s",map_api_error(H245_ERROR_NOT_CONNECTED));
InstanceUnlock(pInstance);
return H245_ERROR_NOT_CONNECTED;
}
/* if channel not found in tracker list */
/* note that it looks for given channel given who alloced it */
pTracker = find_tracker_by_rxchannel (pInstance, wRxChannel, API_CH_ALLOC_RMT);
/* not locking tracker.. since no indication will come in until I issue the request */
/* if not open.. invalid op */
if (pTracker == NULL || pTracker->TrackerType != API_OPEN_CHANNEL_T)
{
H245TRACE (dwInst,1,"H245OpenChannelReject -> %s",map_api_error(H245_ERROR_INVALID_OP));
InstanceUnlock(pInstance);
return H245_ERROR_INVALID_OP;
}
/* for debug */
ASSERT (pTracker->u.Channel.RxChannel == wRxChannel);
ASSERT (pTracker->u.Channel.ChannelAlloc == API_CH_ALLOC_RMT);
/* check state.. must be returning.. */
if (pTracker->State != API_ST_WAIT_LCLACK)
{
H245TRACE (dwInst,1,"H245OpenChannelReject -> %s",map_api_error(H245_ERROR_INVALID_STATE));
InstanceUnlock(pInstance);
return H245_ERROR_INVALID_STATE;
}
pPdu = (MltmdSystmCntrlMssg *)MemAlloc(sizeof(*pPdu));
if (pPdu == NULL)
{
H245TRACE (dwInst,1,"H245OpenChannelReject -> %s",map_api_error(H245_ERROR_NOMEM));
InstanceUnlock(pInstance);
return H245_ERROR_NOMEM;
}
memset(pPdu, 0, sizeof(*pPdu));
pdu_rsp_open_logical_channel_rej(pPdu, wRxChannel, wCause);
switch (pTracker->u.Channel.ChannelType)
{
case API_CH_TYPE_UNI:
lError = FsmOutgoing(pInstance, pPdu, (DWORD_PTR)pTracker);
break;
case API_CH_TYPE_BI:
lError = FsmOutgoing(pInstance, pPdu, (DWORD_PTR)pTracker);
break;
default:
H245TRACE (dwInst,1,"H245OpenChannelReject: Invalid Tracker Channel Type %d",
pTracker->u.Channel.ChannelType);
lError = H245_ERROR_FATAL;
} // switch
MemFree (pPdu);
unlink_dealloc_tracker (pInstance, pTracker);
if (lError != H245_ERROR_OK)
H245TRACE (dwInst,1,"H245OpenChannelReject -> %s",map_api_error(lError));
else
H245TRACE (dwInst,4,"H245OpenChannelReject -> OK");
InstanceUnlock(pInstance);
return lError;
} // H245OpenChannelReject()
/*****************************************************************************
*
* TYPE: H245 API
*
*****************************************************************************
*
* PROCEDURE: H245CloseChannel
*
* DESCRIPTION
*
* HRESULT H245CloseChannel (
* H245_INST_T dwInst,
* DWORD dwTransId,
* DWORD wTxChannel,
* )
* Description:
* Called to close a channel upon which the client previously
* issued an OpenChannel request.
*
* Input
* dwInst Instance handle returned by H245Init
* dwTransId User supplied object used to identify this
* request in the asynchronous confirm to this
* call.
* wChannel Logical Channel Number to close
*
* Call Type:
* Asynchronous
*
* Return Values:
* See Errors
*
* Callbacks:
* H245_CONF_CLOSE
* Errors:
* H245_ERROR_OK
* H245_ERROR_PARAM wChannel is not a locally opened
* channel
* H245_ERROR_INVALID_INST dwInst is not a valid intance handle
* H245_ERROR_NOT_CONNECTED
* H245_ERROR_INVALID_CHANNEL
* H245_ERROR_INVALID_OP Can not perform this operation on
* this channel.(See H245CloseChannelReq)
* See Also:
* H245OpenChannel
* H245OpenChannelIndResp
* H245OpenChannelConfResp
* H245CloseChannelReq
*
* callback
* H245_CONF_CLOSE
*
*
* RETURN:
*
*****************************************************************************/
H245DLL HRESULT
H245CloseChannel (
H245_INST_T dwInst,
DWORD_PTR dwTransId,
H245_CHANNEL_T wTxChannel
)
{
register struct InstanceStruct *pInstance;
Tracker_T *pTracker;
DWORD error;
MltmdSystmCntrlMssg *pPdu;
H245TRACE (dwInst,4,"H245CloseChannel <-");
/* check for valid instance handle */
pInstance = InstanceLock(dwInst);
if (pInstance == NULL)
{
H245TRACE (dwInst,1,"H245CloseChannel -> %s",map_api_error(H245_ERROR_INVALID_INST));
return H245_ERROR_INVALID_INST;
}
/* system should be in connected state */
if (pInstance->API.SystemState != APIST_Connected)
{
H245TRACE (dwInst,1,"H245CloseChannel -> %s",map_api_error(H245_ERROR_NOT_CONNECTED));
InstanceUnlock(pInstance);
return H245_ERROR_NOT_CONNECTED;
}
/* if channel not found in tracker list */
/* note that it looks for given channel given who alloced it */
pTracker = find_tracker_by_txchannel (pInstance, wTxChannel, API_CH_ALLOC_LCL);
/* not locking tracker.. since no indication will come in until I issue the request */
if (!pTracker)
{
H245TRACE (dwInst,1,"H245CloseChannel -> %s",map_api_error(H245_ERROR_INVALID_CHANNEL));
InstanceUnlock(pInstance);
return H245_ERROR_INVALID_CHANNEL;
}
/* setup new tracker state */
pTracker->State = API_ST_WAIT_RMTACK;
pTracker->TrackerType = API_CLOSE_CHANNEL_T;
pTracker->TransId = dwTransId;
if (!(pPdu = (MltmdSystmCntrlMssg *)MemAlloc(sizeof(MltmdSystmCntrlMssg))))
{
H245TRACE (dwInst,1,"H245CloseChannel -> %s",map_api_error(H245_ERROR_NOMEM));
InstanceUnlock(pInstance);
return H245_ERROR_NOMEM;
}
memset(pPdu, 0, sizeof(MltmdSystmCntrlMssg));
/* ok.. get pdu */
pdu_req_close_logical_channel(pPdu, wTxChannel, 0/* user */);
error = FsmOutgoing(pInstance, pPdu, (DWORD_PTR)pTracker);
MemFree (pPdu);
/* error.. so deallocate tracker structure */
if (error != H245_ERROR_OK)
{
unlink_dealloc_tracker (pInstance, pTracker);
H245TRACE (dwInst,1,"H245CloseChannel -> %s",map_api_error(error));
}
else
{
H245TRACE (dwInst,4,"H245CloseChannel -> OK");
}
InstanceUnlock(pInstance);
return H245_ERROR_OK;
} // H245CloseChannel()
/*****************************************************************************
*
* TYPE: H245 API
*
*****************************************************************************
*
* PROCEDURE: H245CloseChannelReq
*
* DESCRIPTION
*
* HRESULT H245CloseChannelReq (
* H245_INST_T dwInst,
* DWORD dwTransId,
* DWORD wChannel,
* )
* Description:
* Called to request the remote peer to close a logical channel
* it previously opened
*
* Input
* dwInst Instance handle returned by H245Init
* dwTransId User supplied object used to identify this
* request in the asynchronous confirm to this
* call
* wChannel Logical Channel Number to close
*
* Note: This is only asking permission. Even if the Close
* Request is accepted the channel still has to be closed from
* the remote side. (i.e. this does not close the channel it
* only asked the remote side it issue a close)
*
* Call Type:
* Asynchronous
*
* Return Values:
* See Errors
*
* Callbacks:
* H245_CONF_CLOSE
*
* Errors:
* H245_ERROR_OK
* H245_ERROR_PARAM wChannel is not a channel
* opened by remote peer
* H245_ERROR_INVALID_INST dwInst is not a valid instance
* handle
* H245_ERROR_NOT_CONNECTED
* H245_ERROR_INVALID_CHANNEL
* H245_ERROR_INVALID_OP Can not perform this operation
* on this channel
* (see H245CloseChannel)
* See Also:
* H245OpenChannel
* H245OpenChannelIndResp
* H245OpenChannelConfResp
* H245CloseChannel
*
* callback
*
* H245_CONF_CLOSE
*
* RETURN:
*
*****************************************************************************/
H245DLL HRESULT
H245CloseChannelReq (
H245_INST_T dwInst,
DWORD_PTR dwTransId,
H245_CHANNEL_T wRxChannel
)
{
register struct InstanceStruct *pInstance;
Tracker_T *pTracker;
DWORD error;
MltmdSystmCntrlMssg *pPdu;
H245TRACE (dwInst,4,"H245CloseChannelReq <-");
/* check for valid instance handle */
pInstance = InstanceLock(dwInst);
if (pInstance == NULL)
{
H245TRACE (dwInst,1,"H245CloseChannelReq -> %s",map_api_error(H245_ERROR_INVALID_INST));
return H245_ERROR_INVALID_INST;
}
/* system should be in connected state */
if (pInstance->API.SystemState != APIST_Connected)
{
H245TRACE (dwInst,1,"H245CloseChannelReq -> %s",map_api_error(H245_ERROR_NOT_CONNECTED));
InstanceUnlock(pInstance);
return H245_ERROR_NOT_CONNECTED;
}
/* if channel not found in tracker list */
/* note that it looks for given channel given who alloced it */
pTracker = find_tracker_by_rxchannel (pInstance, wRxChannel, API_CH_ALLOC_RMT);
/* not locking tracker.. since no indication will come in until I issue the request */
if (!pTracker)
{
H245TRACE (dwInst,1,"H245CloseChannelReq -> %s",map_api_error(H245_ERROR_INVALID_OP));
InstanceUnlock(pInstance);
return H245_ERROR_INVALID_OP;
}
/* verify state of tracker */
pTracker->State = API_ST_WAIT_RMTACK;
pTracker->TrackerType = API_CLOSE_CHANNEL_T;
pTracker->TransId = dwTransId;
if (!(pPdu = (MltmdSystmCntrlMssg *)MemAlloc(sizeof(MltmdSystmCntrlMssg))))
{
H245TRACE (dwInst,1,"H245CloseChannelReq -> %s",map_api_error(H245_ERROR_NOMEM));
InstanceUnlock(pInstance);
return H245_ERROR_NOMEM;
}
memset(pPdu, 0, sizeof(MltmdSystmCntrlMssg));
/* ok.. get pdu */
pdu_req_request_close_channel(pPdu, wRxChannel);
error = FsmOutgoing(pInstance, pPdu, (DWORD_PTR)pTracker);
MemFree (pPdu);
if (error != H245_ERROR_OK)
{
unlink_dealloc_tracker (pInstance, pTracker);
H245TRACE (dwInst,1,"H245CloseChannelReq -> %s",map_api_error(error));
}
else
H245TRACE (dwInst,4,"H245CloseChannelReq -> OK");
InstanceUnlock(pInstance);
return error;
} // H245CloseChannelReq()
/*****************************************************************************
*
* TYPE: H245 API
*
*****************************************************************************
*
* PROCEDURE: H245CloseChannelReqResp
*
* DESCRIPTION
*
* HRESULT H245CloseChannelReqResp (
* H245_INST_T dwInst,
* H245_ACC_REJ_T AccRej,
* DWORD wChannel
* )
*
* Description:
* This routine is called to accept or reject a
* RequestChannelClose (H245_IND_REQ_CLOSE indication) from the
* remote peer. The channel must have been locally opened. The
* parameter AccRej is H245_ACC to accept or H245_REJ to reject
* the close. The local client should follow this response with
* a H245CloseChannel call.
*
* If there was a Release CloseChannelRequest event that
* occurred during this transaction there error code returned
* will be H245_ERROR_CANCELED. This indicates to the H.245
* client that no action should be taken.
*
* Input
* dwInst Instance handle returned by H245GetInstanceId
* AccRej this parameter contains either H245_ACC or
* H245_REJ. This indicates to H.245 which
* action to take.
* wChannel Logical Channel Number to close
*
* Call Type:
* Synchronous
*
* Return Values:
* See Errors
* Errors:
* H245_ERROR_OK
* H245_ERROR_PARAM
* H245_ERROR_NOMEM
* H245_ERROR_INVALID_CHANNEL
* H245_ERROR_INVALID_OP Can not perform this operation on this
* channel (see H245CloseChannel)
* H245_ERROR_INVALID_INST dwInst is not a valid instance handle
* H245_ERROR_CANCELED if release was received during the
* processing of this request..
* See Also:
* H245CloseChannel
*
* callback
*
* H245_IND_REQ_CLOSE
* H245_IND_CLOSE
*
*
* RETURN:
*
*****************************************************************************/
H245DLL HRESULT
H245CloseChannelReqResp (
H245_INST_T dwInst,
H245_ACC_REJ_T AccRej,
H245_CHANNEL_T wChannel
)
{
register struct InstanceStruct *pInstance;
Tracker_T *pTracker;
DWORD error;
MltmdSystmCntrlMssg *pPdu;
H245TRACE (dwInst,4,"H245CloseChannelReqResp <-");
/* check for valid instance handle */
pInstance = InstanceLock(dwInst);
if (pInstance == NULL)
{
H245TRACE (dwInst,1,"H245CloseChannelReqResp -> %s",map_api_error(H245_ERROR_INVALID_INST));
return H245_ERROR_INVALID_INST;
}
/* system should be in connected state */
if (pInstance->API.SystemState != APIST_Connected)
{
H245TRACE (dwInst,1,"H245CloseChannelReqResp -> %s",map_api_error(H245_ERROR_NOT_CONNECTED));
InstanceUnlock(pInstance);
return H245_ERROR_NOT_CONNECTED;
}
/* if channel not found in tracker list */
/* note that it looks for given channel given who alloced it */
pTracker = find_tracker_by_txchannel (pInstance, wChannel, API_CH_ALLOC_LCL);
/* not locking tracker.. since no indication will come in until I issue the request */
if (!pTracker)
{
H245TRACE (dwInst,1,"H245CloseChannelReqResp -> %s",map_api_error(H245_ERROR_INVALID_OP));
InstanceUnlock(pInstance);
return H245_ERROR_INVALID_OP;
}
/* if the request was canceled.. tell useer */
if (pTracker->State == API_ST_WAIT_LCLACK_CANCEL)
{
H245TRACE (dwInst,1,"H245CloseChannelReqResp -> %s",map_api_error(H245_ERROR_CANCELED));
InstanceUnlock(pInstance);
return H245_ERROR_CANCELED;
}
/* verify state of tracker */
if ((pTracker->State != API_ST_WAIT_LCLACK) ||
(pTracker->TrackerType != API_CLOSE_CHANNEL_T))
{
H245TRACE (dwInst,1,"H245CloseChannelReqResp -> %s",map_api_error(H245_ERROR_INVALID_OP));
InstanceUnlock(pInstance);
return H245_ERROR_INVALID_OP;
}
/* set the state to idle.. expect this side to close the channel next */
pTracker->State = API_ST_IDLE;
if (!(pPdu = (MltmdSystmCntrlMssg *)MemAlloc(sizeof(MltmdSystmCntrlMssg))))
{
H245TRACE (dwInst,1,"H245CloseChannelReqResp -> %s",map_api_error(H245_ERROR_NOMEM));
InstanceUnlock(pInstance);
return H245_ERROR_NOMEM;
}
memset(pPdu, 0, sizeof(MltmdSystmCntrlMssg));
/* ok.. get pdu */
if (AccRej == H245_ACC)
{
pTracker = NULL;
pdu_rsp_request_channel_close_ack(pPdu, wChannel);
}
else
{
pdu_rsp_request_channel_close_rej(pPdu, wChannel, AccRej);
}
error = FsmOutgoing(pInstance, pPdu, (DWORD_PTR)pTracker);
MemFree (pPdu);
if (error != H245_ERROR_OK)
H245TRACE (dwInst,1,"H245CloseChannelReqResp -> %s",map_api_error(error));
else
H245TRACE (dwInst,4,"H245CloseChannelReqResp ->");
InstanceUnlock(pInstance);
return error;
} // H245CloseChannelReqResp()
/*****************************************************************************
*
* TYPE: H245 API
*
*****************************************************************************
*
* PROCEDURE: H245SendLocalMuxTable
*
* DESCRIPTION
*
* HRESULT H245SendLocalMuxTable (
* H245_INST_T dwInst,
* DWORD dwTransId,
* H245_MUX_TABLE_T *pMuxTbl
* )
* Description:
* This routine is called to send a mux table to the remote
* side. The remote side can either reject or accept each mux
* table entry in a message. The confirm is sent back to the
* calling H.245 client based on the acceptance or non
* acceptance of each Mux table entry with H245_CONF_MUXTBL_SND.
*
* This is a fairly dangerous call, since the mux table
* structure is a linked lise of mux table entries. Invalid
* data structures could cause an access error. Example code is
* supplied in the appendix.
* Input
* dwInst Instance handle returned by H245Init
* dwTransId User supplied object used to identify
* this request in the asynchronous
* confirm to this call.
* pMuxTbl Mux table entry structure
*
* Call Type:
* Asynchronous
*
* Return Values:
* See Errors
*
* Callbacks:
* H245_CONF_MUXTBLSND
*
* Errors:
* H245_ERROR_OK
* H245_ERROR_INVALID_MUXTBLENTRY
* H245_ERROR_PARAM
* H245_ERROR_INVALID_INST
* H245_ERROR_NOT_CONNECTED
* H245_ERROR_NOMEM
*
* See Also:
* APPENDIX Examples
*
* callback
*
* H245_CONF_MUXTBL_SND
*
*
* RETURN:
*
*****************************************************************************/
H245DLL HRESULT
H245SendLocalMuxTable (
H245_INST_T dwInst,
DWORD_PTR dwTransId,
H245_MUX_TABLE_T *pMuxTable
)
{
register struct InstanceStruct *pInstance;
HRESULT lError;
Tracker_T *pTracker;
MltmdSystmCntrlMssg *pPdu;
H245TRACE (dwInst,4,"H245SendLocalMuxTable <-");
/* check for valid instance handle */
pInstance = InstanceLock(dwInst);
if (pInstance == NULL)
{
H245TRACE (dwInst,1,"H245endLocalMuxTable -> %s",map_api_error(H245_ERROR_INVALID_INST));
return H245_ERROR_INVALID_INST;
}
/* system should be in connected state */
if (pInstance->API.SystemState != APIST_Connected)
{
H245TRACE (dwInst,1,"H245endLocalMuxTable -> %s",map_api_error(H245_ERROR_NOT_CONNECTED));
InstanceUnlock(pInstance);
return H245_ERROR_NOT_CONNECTED;
}
/* make sure parameters ar ok.. */
if (!pMuxTable)
{
H245TRACE (dwInst,1,"H245endLocalMuxTable -> %s",map_api_error(H245_ERROR_PARAM));
InstanceUnlock(pInstance);
return H245_ERROR_PARAM;
}
/* allocate tracker for event */
pTracker = alloc_link_tracker (pInstance,
API_SEND_MUX_T,
dwTransId,
API_ST_WAIT_RMTACK,
API_CH_ALLOC_UNDEF,
API_CH_TYPE_UNDEF,
0,
H245_INVALID_CHANNEL, H245_INVALID_CHANNEL,
0);
if (pTracker == NULL)
{
H245TRACE(dwInst,1,"H245SendLocalMuxTable -> %s",map_api_error(H245_ERROR_NOMEM));
InstanceUnlock(pInstance);
return H245_ERROR_NOMEM;
}
// Allocate PDU buffer
pPdu = (MltmdSystmCntrlMssg *)MemAlloc(sizeof(MltmdSystmCntrlMssg));
if (pPdu == NULL)
{
H245TRACE (dwInst,1,"H245SendLocalMuxTable -> %s",H245_ERROR_NOMEM);
InstanceUnlock(pInstance);
return H245_ERROR_NOMEM;
}
memset(pPdu, 0, sizeof(MltmdSystmCntrlMssg));
lError = pdu_req_send_mux_table(pInstance,
pPdu,
pMuxTable,
0,
&pTracker->u.MuxEntryCount);
if (lError == H245_ERROR_OK)
{
lError = FsmOutgoing(pInstance, pPdu, (DWORD_PTR)pTracker);
/* free the list just built */
free_mux_desc_list(pPdu->u.MltmdSystmCntrlMssg_rqst.u.multiplexEntrySend.multiplexEntryDescriptors);
}
/* free the pdu */
MemFree (pPdu);
if (lError != H245_ERROR_OK)
{
unlink_dealloc_tracker (pInstance, pTracker);
H245TRACE (dwInst,1,"H245SendLocalMuxTable -> %s",map_api_error(lError));
}
else
H245TRACE (dwInst,4,"H245SendLocalMuxTable -> OK");
InstanceUnlock(pInstance);
return lError;
} // H245SendLocalMuxTable()
/*****************************************************************************
*
* TYPE: H245 API
*
*****************************************************************************
*
* PROCEDURE: H245MuxTableIndResp
*
* DESCRIPTION
*
* HRESULT H245MuxTableIndResp (
* H45_INST_T dwInst,
* H245_ACC_REJ_MUX_T AccRejMux,
* DWORD Count
* )
* Description:
* This procedure is called to either accept or reject mux
* table entries sent up in the H245_IND_MUX_TBL indication.
*
* Input
* dwInst Instance handle returned by H245Init
* AccRejMux Accept Reject Mux structure
* Count number of entries in the structure
*
* Call Type:
* Asynchronous
*
* Return Values:
* See Errors
* Errors:
* H245_ERROR_OK
* H245_ERROR_INVALID_MUXTBLENTRY
* H245_ERROR_PARAM
* H245_ERROR_INVALID_INST
* H245_ERROR_INVALID_OP
* H245_ERROR_NOT_CONNECTED
* H245_ERROR_NOMEM
* See Also:
* H245SendLocalMuxTable
*
* callback
*
* H245_IND_MUX_TBL
*
*
* RETURN:
*
*****************************************************************************/
H245DLL HRESULT
H245MuxTableIndResp (
H245_INST_T dwInst,
H245_ACC_REJ_MUX_T AccRejMux,
unsigned long dwCount
)
{
register struct InstanceStruct *pInstance;
DWORD ii;
Tracker_T *pTracker;
MltmdSystmCntrlMssg *pPdu;
H245TRACE (dwInst,4,"H245MuxTableIndResp <-");
/* check for valid instance handle */
pInstance = InstanceLock(dwInst);
if (pInstance == NULL)
{
H245TRACE (dwInst,1,"H245MuxTableIndResp -> %s",map_api_error(H245_ERROR_INVALID_INST));
return H245_ERROR_INVALID_INST;
}
/* system should be in connected state */
if (pInstance->API.SystemState != APIST_Connected)
{
H245TRACE (dwInst,1,"H245MuxTableIndResp -> %s",map_api_error(H245_ERROR_NOT_CONNECTED));
InstanceUnlock(pInstance);
return H245_ERROR_NOT_CONNECTED;
}
/* look for tracker.. */
pTracker = NULL;
pTracker = find_tracker_by_type (pInstance, API_RECV_MUX_T, pTracker);
/* if tracker not found.. issue invalid op */
if (!pTracker)
{
H245TRACE (dwInst,1,"H245MuxTableIndResp -> %s",map_api_error(H245_ERROR_INVALID_OP));
InstanceUnlock(pInstance);
return H245_ERROR_INVALID_OP;
}
ASSERT (pTracker->State == API_ST_WAIT_LCLACK);
/* can't ack or reject more than you got */
if ((dwCount > pTracker->u.MuxEntryCount) ||
(dwCount > 15))
{
H245TRACE (dwInst,1,"H245MuxTableIndResp -> %s",map_api_error(H245_ERROR_PARAM));
InstanceUnlock(pInstance);
return H245_ERROR_PARAM;
}
/* verify the mux table entry id's */
for (ii=0;ii<dwCount;ii++)
{
if ((AccRejMux[ii].MuxEntryId > 15) ||
(AccRejMux[ii].MuxEntryId <= 0))
{
H245TRACE (dwInst,1,"H245MuxTableIndResp -> %s",map_api_error(H245_ERROR_PARAM));
InstanceUnlock(pInstance);
return H245_ERROR_PARAM;
}
}
if (!(pPdu = (MltmdSystmCntrlMssg *)MemAlloc(sizeof(MltmdSystmCntrlMssg))))
{
H245TRACE (dwInst,1,"H245MuxTableIndResp -> %s",map_api_error(H245_ERROR_NOMEM));
InstanceUnlock(pInstance);
return H245_ERROR_NOMEM;
}
memset(pPdu, 0, sizeof(MltmdSystmCntrlMssg));
/* if there are any rejects in the list.. send reject */
memset(pPdu, 0, sizeof(MltmdSystmCntrlMssg));
if (pdu_rsp_mux_table_rej (pPdu,0,AccRejMux,dwCount) == H245_ERROR_OK)
FsmOutgoing(pInstance, pPdu, 0);
/* if there are any accepts in the list.. send accept */
memset(pPdu, 0, sizeof(MltmdSystmCntrlMssg));
if (pdu_rsp_mux_table_ack (pPdu,0,AccRejMux,dwCount) == H245_ERROR_OK)
FsmOutgoing(pInstance, pPdu, 0);
/* if we've acked all the entries */
if (!(pTracker->u.MuxEntryCount -= dwCount))
unlink_dealloc_tracker (pInstance, pTracker);
MemFree (pPdu);
H245TRACE (dwInst,4,"H245MuxTableIndResp -> %s",H245_ERROR_OK);
InstanceUnlock(pInstance);
return H245_ERROR_OK;
} // H245MuxTableIndResp()
#if 0
/*****************************************************************************
*
* TYPE: H245 API
*
*****************************************************************************
*
* PROCEDURE: H245MiscCommand
*
* DESCRIPTION
*
* HRESULT H245MiscCommand (
* H245_INST_T dwInst,
* DWORD wChannel,
* H245_MISC_T *pMisc
* )
* Description:
* Send a Misc. command to the remote side (see H245_MISC_T
* data Structure)
*
* Input
* dwInst Instance handle returned by H245Init
* wChannel Logical Channel Number
* pMisc pointer to a misc. command structure
*
* Call Type:
* Synchronous
*
* Return Values:
* See Errors
*
* Errors:
* H245_ERROR_OK
* H245_ERROR_NOT_CONNECTED
* H245_ERROR_INVALID_CHANNEL
* H245_ERROR_NOMEM
* H245_ERROR_PARAM
* H245_ERROR_INVALID_INST
*
* callback
*
* H245_IND_MISC
*
*
* RETURN:
*
*****************************************************************************/
H245DLL HRESULT H245MiscCommand (
H245_INST_T dwInst,
WORD wChannel,
H245_MISC_T *pMisc
)
{
register struct InstanceStruct *pInstance;
HRESULT lError;
MltmdSystmCntrlMssg *pPdu;
H245TRACE (dwInst,4,"H245MiscCommand <-");
/* check for valid instance handle */
pInstance = InstanceLock(dwInst);
if (pInstance == NULL)
{
H245TRACE (dwInst,1,"H245MiscCommand -> %s",map_api_error(H245_ERROR_INVALID_INST));
return H245_ERROR_INVALID_INST;
}
/* system should be in connected state */
if (pInstance->API.SystemState != APIST_Connected)
{
H245TRACE (dwInst,1,"H245MiscCommand -> %s",map_api_error(H245_ERROR_NOT_CONNECTED));
InstanceUnlock(pInstance);
return H245_ERROR_NOT_CONNECTED;
}
/* if the channel can not be found */
if (!find_tracker_by_txchannel(pInstance, wChannel, API_CH_ALLOC_LCL) &&
!find_tracker_by_rxchannel(pInstance, wChannel, API_CH_ALLOC_RMT))
{
H245TRACE (dwInst,1,"H245MiscCommand -> %s",map_api_error(H245_ERROR_INVALID_CHANNEL));
InstanceUnlock(pInstance);
return H245_ERROR_INVALID_CHANNEL;
}
if (!(pPdu = (MltmdSystmCntrlMssg *)MemAlloc(sizeof(MltmdSystmCntrlMssg))))
{
H245TRACE (dwInst,1,"H245MiscCommand -> %s",map_api_error(H245_ERROR_NOMEM));
InstanceUnlock(pInstance);
return H245_ERROR_NOMEM;
}
memset(pPdu, 0, sizeof(MltmdSystmCntrlMssg));
/* budld pdu for misc command */
pdu_cmd_misc (pPdu, pMisc, wChannel);
lError = FsmOutgoing(pInstance, pPdu, 0);
MemFree (pPdu);
if (lError != H245_ERROR_OK)
H245TRACE (dwInst,1,"H245MiscCommand -> %s",map_api_error(lError));
else
H245TRACE (dwInst,4,"H245MiscCommand -> OK");
InstanceUnlock(pInstance);
return lError;
} // H245MiscCommand()
#endif
/*****************************************************************************
*
* TYPE: H245 API
*
* PROCEDURE: H245RequestMultiplexEntry
*
* DESCRIPTION
*
* RETURN:
*
*****************************************************************************/
H245DLL HRESULT
H245RequestMultiplexEntry (
H245_INST_T dwInst,
DWORD_PTR dwTransId,
const unsigned short * pwMultiplexTableEntryNumbers,
unsigned long dwCount
)
{
register struct InstanceStruct *pInstance;
register PDU_T * pPdu;
HRESULT lError;
DWORD dwIndex;
H245TRACE (dwInst,4,"H245RequestMultiplexEntry <-");
if (pwMultiplexTableEntryNumbers == NULL || dwCount < 1 || dwCount > 15)
{
H245TRACE (dwInst,1,"H245RequestMultiplexEntry -> %s",map_api_error(H245_ERROR_PARAM));
return H245_ERROR_PARAM;
}
for (dwIndex = 0; dwIndex < dwCount; ++dwIndex)
{
if (pwMultiplexTableEntryNumbers[dwIndex] < 1 ||
pwMultiplexTableEntryNumbers[dwIndex] > 15)
{
H245TRACE (dwInst,1,"H245RequestMultiplexEntry -> %s",map_api_error(H245_ERROR_PARAM));
return H245_ERROR_PARAM;
}
}
pInstance = InstanceLock(dwInst);
if (pInstance == NULL)
{
H245TRACE (dwInst,1,"H245RequestMultiplexEntry -> %s",map_api_error(H245_ERROR_INVALID_INST));
return H245_ERROR_INVALID_INST;
}
pPdu = (PDU_T *)MemAlloc(sizeof(*pPdu));
if (pPdu == NULL)
{
lError = H245_ERROR_NOMEM;
}
else
{
memset(pPdu, 0, sizeof(*pPdu));
pPdu->choice = MltmdSystmCntrlMssg_rqst_chosen;
pPdu->u.MltmdSystmCntrlMssg_rqst.choice = requestMultiplexEntry_chosen;
pPdu->u.MltmdSystmCntrlMssg_rqst.u.requestMultiplexEntry.entryNumbers.count = (WORD)dwCount;
for (dwIndex = 0; dwIndex < dwCount; ++dwIndex)
{
pPdu->u.MltmdSystmCntrlMssg_rqst.u.requestMultiplexEntry.entryNumbers.value[dwIndex] =
(MultiplexTableEntryNumber) pwMultiplexTableEntryNumbers[dwIndex];
}
lError = FsmOutgoing(pInstance, pPdu, dwTransId);
MemFree(pPdu);
}
if (lError != H245_ERROR_OK)
H245TRACE (dwInst,1,"H245RequestMultiplexEntry -> %s",map_api_error(lError));
else
H245TRACE (dwInst,4,"H245RequestMultiplexEntry -> OK");
InstanceUnlock(pInstance);
return lError;
} // H245RequestMultiplexEntry()
/*****************************************************************************
*
* TYPE: H245 API
*
* PROCEDURE: H245RequestMultiplexEntryAck
*
* DESCRIPTION
*
* RETURN:
*
*****************************************************************************/
H245DLL HRESULT
H245RequestMultiplexEntryAck (
H245_INST_T dwInst,
const unsigned short * pwMultiplexTableEntryNumbers,
unsigned long dwCount
)
{
register struct InstanceStruct *pInstance;
register PDU_T * pPdu;
HRESULT lError;
DWORD dwIndex;
H245TRACE (dwInst,4,"H245RequestMultiplexEntryAck <-");
if (pwMultiplexTableEntryNumbers == NULL || dwCount < 1 || dwCount > 15)
{
H245TRACE (dwInst,1,"H245RequestMultiplexEntryAck -> %s",map_api_error(H245_ERROR_PARAM));
return H245_ERROR_PARAM;
}
pInstance = InstanceLock(dwInst);
if (pInstance == NULL)
{
H245TRACE (dwInst,1,"H245RequestMultiplexEntryAck -> %s",map_api_error(H245_ERROR_INVALID_INST));
return H245_ERROR_INVALID_INST;
}
pPdu = (PDU_T *)MemAlloc(sizeof(*pPdu));
if (pPdu == NULL)
{
lError = H245_ERROR_NOMEM;
}
else
{
memset(pPdu, 0, sizeof(*pPdu));
pPdu->choice = MSCMg_rspns_chosen;
pPdu->u.MSCMg_rspns.choice = requestMultiplexEntryAck_chosen;
pPdu->u.MSCMg_rspns.u.requestMultiplexEntryAck.entryNumbers.count = (WORD)dwCount;
for (dwIndex = 0; dwIndex < dwCount; ++dwIndex)
{
pPdu->u.MSCMg_rspns.u.requestMultiplexEntryAck.entryNumbers.value[dwIndex] =
(MultiplexTableEntryNumber) pwMultiplexTableEntryNumbers[dwIndex];
}
lError = FsmOutgoing(pInstance, pPdu, 0);
MemFree(pPdu);
}
if (lError != H245_ERROR_OK)
H245TRACE (dwInst,1,"H245RequestMultiplexEntryAck -> %s",map_api_error(lError));
else
H245TRACE (dwInst,4,"H245RequestMultiplexEntryAck -> OK");
InstanceUnlock(pInstance);
return lError;
} // H245RequestMultiplexEntryAck()
/*****************************************************************************
*
* TYPE: H245 API
*
* PROCEDURE: H245RequestMultiplexEntryReject
*
* DESCRIPTION
*
* RETURN:
*
*****************************************************************************/
H245DLL HRESULT
H245RequestMultiplexEntryReject (
H245_INST_T dwInst,
const unsigned short * pwMultiplexTableEntryNumbers,
unsigned long dwCount
)
{
register struct InstanceStruct *pInstance;
register PDU_T * pPdu;
HRESULT lError;
DWORD dwIndex;
H245TRACE (dwInst,4,"H245RequestMultiplexEntryReject <-");
if (pwMultiplexTableEntryNumbers == NULL || dwCount < 1 || dwCount > 15)
{
H245TRACE (dwInst,1,"H245RequestMultiplexEntryReject -> %s",map_api_error(H245_ERROR_PARAM));
return H245_ERROR_PARAM;
}
pInstance = InstanceLock(dwInst);
if (pInstance == NULL)
{
H245TRACE (dwInst,1,"H245RequestMultiplexEntryReject -> %s",map_api_error(H245_ERROR_INVALID_INST));
return H245_ERROR_INVALID_INST;
}
pPdu = (PDU_T *)MemAlloc(sizeof(*pPdu));
if (pPdu == NULL)
{
lError = H245_ERROR_NOMEM;
}
else
{
memset(pPdu, 0, sizeof(*pPdu));
pPdu->choice = MSCMg_rspns_chosen;
pPdu->u.MSCMg_rspns.choice = rqstMltplxEntryRjct_chosen;
pPdu->u.MSCMg_rspns.u.rqstMltplxEntryRjct.rejectionDescriptions.count = (WORD)dwCount;
for (dwIndex = 0; dwIndex < dwCount; ++dwIndex)
{
pPdu->u.MSCMg_rspns.u.rqstMltplxEntryRjct.rejectionDescriptions.value[dwIndex].multiplexTableEntryNumber =
(MultiplexTableEntryNumber) pwMultiplexTableEntryNumbers[dwIndex];
pPdu->u.MSCMg_rspns.u.rqstMltplxEntryRjct.rejectionDescriptions.value[dwIndex].cause.choice = RMERDs_cs_unspcfdCs_chosen;
}
lError = FsmOutgoing(pInstance, pPdu, 0);
MemFree(pPdu);
}
if (lError != H245_ERROR_OK)
H245TRACE (dwInst,1,"H245RequestMultiplexEntryReject -> %s",map_api_error(lError));
else
H245TRACE (dwInst,4,"H245RequestMultiplexEntryReject -> OK");
InstanceUnlock(pInstance);
return lError;
} // H245RequestMultiplexEntryReject()
/*****************************************************************************
*
* TYPE: H245 API
*
* PROCEDURE: H245RequestMode
*
* DESCRIPTION
*
* RETURN:
*
*****************************************************************************/
H245DLL HRESULT
H245RequestMode (
H245_INST_T dwInst,
DWORD_PTR dwTransId,
// const ModeElement * pModeElements,
//[email protected]
ModeDescription ModeDescriptions[],
//[email protected]
unsigned long dwCount
)
{
register struct InstanceStruct *pInstance;
register PDU_T * pPdu;
HRESULT lError;
RequestedModesLink pLink;
RequestedModesLink pLink_First;
DWORD dwIndex;
// [email protected]
ULONG ulPDUsize;
// [email protected]
H245TRACE (dwInst,4,"H245RequestMode <-");
//[email protected]
// if (pModeElements == NULL || dwCount == 0 || dwCount > 256)
if (ModeDescriptions == NULL || dwCount == 0 || dwCount > 256)
//[email protected]
{
H245TRACE (dwInst,1,"H245RequestMode -> %s",map_api_error(H245_ERROR_PARAM));
return H245_ERROR_PARAM;
}
pInstance = InstanceLock(dwInst);
if (pInstance == NULL)
{
H245TRACE (dwInst,1,"H245RequestMode -> %s",map_api_error(H245_ERROR_INVALID_INST));
return H245_ERROR_INVALID_INST;
}
//[email protected]
// pPdu = (PDU_T *)MemAlloc(sizeof(*pPdu) + sizeof(*pLink));
pPdu = (PDU_T *)MemAlloc(sizeof(*pPdu) + (sizeof(*pLink)*(dwCount)));
//[email protected]
if (pPdu == NULL)
{
lError = H245_ERROR_NOMEM;
}
else
{
USHORT usModeDescIndex =0;
memset(pPdu, 0, sizeof(*pPdu));
pPdu->choice = MltmdSystmCntrlMssg_rqst_chosen;
pPdu->u.MltmdSystmCntrlMssg_rqst.choice = requestMode_chosen;
//[email protected]
// pLink = (RequestedModesLink)(pPdu + 1);
pLink = (RequestedModesLink)(pPdu + usModeDescIndex+1);
//[email protected]
//[email protected]
// pPdu->u.MltmdSystmCntrlMssg_rqst.u.requestMode.requestedModes = pLink;
//[email protected]
//[email protected]
pLink_First = pLink;
//[email protected]
// --> linked list of mode descriptions ... up to 1..256
//[email protected]
while(usModeDescIndex<=(dwCount-1))
{
//[email protected]
//[email protected]
// pLink->next = NULL;
//[email protected]
// --> number of actual mode-elements associated with this mode description
//[email protected]
// pLink->value.count = (WORD)dwCount;
pLink->value.count = (WORD)ModeDescriptions[usModeDescIndex].count;
//[email protected]
// for (dwIndex = 0; dwIndex < dwCount; ++dwIndex)
for (dwIndex = 0; dwIndex < ModeDescriptions[usModeDescIndex].count; ++dwIndex)
{
//[email protected]
// pLink->value.value[dwIndex] = pModeElements[dwIndex];
pLink->value.value[dwIndex] = ModeDescriptions[usModeDescIndex].value[dwIndex];
//[email protected]
}
//[email protected]
usModeDescIndex++;
if(usModeDescIndex<=(dwCount-1))
{
pLink->next = (RequestedModesLink)(pPdu + usModeDescIndex+1);
pLink = pLink->next;
pLink->next = NULL;
}
//[email protected]
//[email protected]
}
//[email protected]
//[email protected]
pPdu->u.MltmdSystmCntrlMssg_rqst.u.requestMode.requestedModes = pLink_First;
//[email protected]
//--
ulPDUsize = (sizeof(*pPdu) + (sizeof(*pLink)*(dwCount)));
//--
lError = FsmOutgoing(pInstance, pPdu, dwTransId);
MemFree(pPdu);
}
if (lError != H245_ERROR_OK)
H245TRACE (dwInst,1,"H245RequestMode -> %s",map_api_error(lError));
else
H245TRACE (dwInst,4,"H245RequestMode -> OK");
InstanceUnlock(pInstance);
return lError;
} // H245RequestMode()
/*****************************************************************************
*
* TYPE: H245 API
*
* PROCEDURE: H245RequestModeAck
*
* DESCRIPTION
*
* RETURN:
*
*****************************************************************************/
H245DLL HRESULT
H245RequestModeAck (
H245_INST_T dwInst,
unsigned short wResponse
)
{
register struct InstanceStruct *pInstance;
register PDU_T * pPdu;
HRESULT lError;
H245TRACE (dwInst,4,"H245RequestModeAck <-");
pInstance = InstanceLock(dwInst);
if (pInstance == NULL)
{
H245TRACE (dwInst,1,"H245RequestModeAck -> %s",map_api_error(H245_ERROR_INVALID_INST));
return H245_ERROR_INVALID_INST;
}
pPdu = (PDU_T *)MemAlloc(sizeof(*pPdu));
if (pPdu == NULL)
{
lError = H245_ERROR_NOMEM;
}
else
{
memset(pPdu, 0, sizeof(*pPdu));
pPdu->choice = MSCMg_rspns_chosen;
pPdu->u.MSCMg_rspns.choice = requestModeAck_chosen;
pPdu->u.MSCMg_rspns.u.requestModeAck.response.choice = wResponse;
lError = FsmOutgoing(pInstance, pPdu, 0);
MemFree(pPdu);
}
if (lError != H245_ERROR_OK)
H245TRACE (dwInst,1,"H245RequestModeAck -> %s",map_api_error(lError));
else
H245TRACE (dwInst,4,"H245RequestModeAck -> OK");
InstanceUnlock(pInstance);
return lError;
} // H245RequestModeAck()
/*****************************************************************************
*
* TYPE: H245 API
*
* PROCEDURE: H245RequestModeReject
*
* DESCRIPTION
*
* RETURN:
*
*****************************************************************************/
H245DLL HRESULT
H245RequestModeReject (
H245_INST_T dwInst,
unsigned short wCause
)
{
register struct InstanceStruct *pInstance;
register PDU_T * pPdu;
HRESULT lError;
H245TRACE (dwInst,4,"H245RequestModeReject <-");
pInstance = InstanceLock(dwInst);
if (pInstance == NULL)
{
H245TRACE (dwInst,1,"H245RequestModeReject -> %s",map_api_error(H245_ERROR_INVALID_INST));
return H245_ERROR_INVALID_INST;
}
pPdu = (PDU_T *)MemAlloc(sizeof(*pPdu));
if (pPdu == NULL)
{
lError = H245_ERROR_NOMEM;
}
else
{
memset(pPdu, 0, sizeof(*pPdu));
pPdu->choice = MSCMg_rspns_chosen;
pPdu->u.MSCMg_rspns.choice = requestModeReject_chosen;
pPdu->u.MSCMg_rspns.u.requestModeReject.cause.choice = wCause;
lError = FsmOutgoing(pInstance, pPdu, 0);
MemFree(pPdu);
}
if (lError != H245_ERROR_OK)
H245TRACE (dwInst,1,"H245RequestModeReject -> %s",map_api_error(lError));
else
H245TRACE (dwInst,4,"H245RequestModeReject -> OK");
InstanceUnlock(pInstance);
return lError;
} // H245RequestModeReject()
/*****************************************************************************
*
* TYPE: H245 API
*
* PROCEDURE: H245RoundTripDelayRequest
*
* DESCRIPTION
*
* RETURN:
*
*****************************************************************************/
H245DLL HRESULT
H245RoundTripDelayRequest (
H245_INST_T dwInst,
DWORD_PTR dwTransId
)
{
register struct InstanceStruct *pInstance;
register PDU_T * pPdu;
HRESULT lError;
H245TRACE (dwInst,4,"H245RoundTripDelayRequest <-");
pInstance = InstanceLock(dwInst);
if (pInstance == NULL)
{
H245TRACE (dwInst,1,"H245RoundTripDelayRequest -> %s",map_api_error(H245_ERROR_INVALID_INST));
return H245_ERROR_INVALID_INST;
}
pPdu = (PDU_T *)MemAlloc(sizeof(*pPdu));
if (pPdu == NULL)
{
lError = H245_ERROR_NOMEM;
}
else
{
memset(pPdu, 0, sizeof(*pPdu));
pPdu->choice = MltmdSystmCntrlMssg_rqst_chosen;
pPdu->u.MltmdSystmCntrlMssg_rqst.choice = roundTripDelayRequest_chosen;
lError = FsmOutgoing(pInstance, pPdu, dwTransId);
MemFree(pPdu);
}
if (lError != H245_ERROR_OK)
H245TRACE (dwInst,1,"H245RoundTripDelayRequest -> %s",map_api_error(lError));
else
H245TRACE (dwInst,4,"H245RoundTripDelayRequest -> OK");
InstanceUnlock(pInstance);
return lError;
} // H245RoundTripDelayRequest()
/*****************************************************************************
*
* TYPE: H245 API
*
* PROCEDURE: H245MaintenanceLoop
*
* DESCRIPTION
*
* RETURN:
*
*****************************************************************************/
H245DLL HRESULT
H245MaintenanceLoop (
H245_INST_T dwInst,
DWORD_PTR dwTransId,
H245_LOOP_TYPE_T dwLoopType,
H245_CHANNEL_T wChannel
)
{
register struct InstanceStruct *pInstance;
register PDU_T * pPdu;
HRESULT lError;
H245TRACE (dwInst,4,"H245MaintenanceLoop <-");
if (dwLoopType < systemLoop_chosen ||
dwLoopType > logicalChannelLoop_chosen ||
(dwLoopType != systemLoop_chosen && wChannel == 0))
{
H245TRACE (dwInst,1,"H245MaintenanceLoop -> %s",map_api_error(H245_ERROR_PARAM));
return H245_ERROR_PARAM;
}
pInstance = InstanceLock(dwInst);
if (pInstance == NULL)
{
H245TRACE (dwInst,1,"H245MaintenanceLoop -> %s",map_api_error(H245_ERROR_INVALID_INST));
return H245_ERROR_INVALID_INST;
}
pPdu = (PDU_T *)MemAlloc(sizeof(*pPdu));
if (pPdu == NULL)
{
lError = H245_ERROR_NOMEM;
}
else
{
memset(pPdu, 0, sizeof(*pPdu));
pPdu->choice = MltmdSystmCntrlMssg_rqst_chosen;
pPdu->u.MltmdSystmCntrlMssg_rqst.choice = maintenanceLoopRequest_chosen;
pPdu->u.MltmdSystmCntrlMssg_rqst.u.maintenanceLoopRequest.type.choice = (WORD)dwLoopType;
pPdu->u.MltmdSystmCntrlMssg_rqst.u.maintenanceLoopRequest.type.u.mediaLoop = wChannel;
lError = FsmOutgoing(pInstance, pPdu, dwTransId);
MemFree(pPdu);
}
if (lError != H245_ERROR_OK)
H245TRACE (dwInst,1,"H245MaintenanceLoop -> %s",map_api_error(lError));
else
H245TRACE (dwInst,4,"H245MaintenanceLoop -> OK");
InstanceUnlock(pInstance);
return lError;
} // H245MaintenanceLoop()
/*****************************************************************************
*
* TYPE: H245 API
*
* PROCEDURE: H245MaintenanceLoopRelease
*
* DESCRIPTION
*
* RETURN:
*
*****************************************************************************/
H245DLL HRESULT
H245MaintenanceLoopRelease (H245_INST_T dwInst)
{
register struct InstanceStruct *pInstance;
register PDU_T * pPdu;
HRESULT lError;
H245TRACE (dwInst,4,"H245MaintenanceLoopRelease <-");
pInstance = InstanceLock(dwInst);
if (pInstance == NULL)
{
H245TRACE (dwInst,1,"H245MaintenanceLoopRelease -> %s",map_api_error(H245_ERROR_INVALID_INST));
return H245_ERROR_INVALID_INST;
}
pPdu = (PDU_T *)MemAlloc(sizeof(*pPdu));
if (pPdu == NULL)
{
lError = H245_ERROR_NOMEM;
}
else
{
memset(pPdu, 0, sizeof(*pPdu));
pPdu->choice = MSCMg_cmmnd_chosen;
pPdu->u.MSCMg_cmmnd.choice = mntnncLpOffCmmnd_chosen;
lError = FsmOutgoing(pInstance, pPdu, 0);
MemFree(pPdu);
}
if (lError != H245_ERROR_OK)
H245TRACE (dwInst,1,"H245MaintenanceLoopRelease -> %s",map_api_error(lError));
else
H245TRACE (dwInst,4,"H245MaintenanceLoopRelease -> OK");
InstanceUnlock(pInstance);
return lError;
} // H245MaintenanceLoopRelease()
/*****************************************************************************
*
* TYPE: H245 API
*
* PROCEDURE: H245MaintenanceLoopAccept
*
* DESCRIPTION
*
* RETURN:
*
*****************************************************************************/
H245DLL HRESULT
H245MaintenanceLoopAccept (
H245_INST_T dwInst,
H245_CHANNEL_T wChannel
)
{
register struct InstanceStruct *pInstance;
register PDU_T * pPdu;
HRESULT lError;
H245TRACE (dwInst,4,"H245MaintenanceLoopAccept <-");
pInstance = InstanceLock(dwInst);
if (pInstance == NULL)
{
H245TRACE (dwInst,1,"H245MaintenanceLoopAccept -> %s",map_api_error(H245_ERROR_INVALID_INST));
return H245_ERROR_INVALID_INST;
}
pPdu = (PDU_T *)MemAlloc(sizeof(*pPdu));
if (pPdu == NULL)
{
lError = H245_ERROR_NOMEM;
}
else
{
memset(pPdu, 0, sizeof(*pPdu));
pPdu->choice = MSCMg_rspns_chosen;
pPdu->u.MSCMg_rspns.choice = maintenanceLoopAck_chosen;
pPdu->u.MSCMg_rspns.u.maintenanceLoopAck.type.u.mediaLoop = wChannel;
lError = FsmOutgoing(pInstance, pPdu, 0);
MemFree(pPdu);
}
if (lError != H245_ERROR_OK)
H245TRACE (dwInst,1,"H245MaintenanceLoopAccept -> %s",map_api_error(lError));
else
H245TRACE (dwInst,4,"H245MaintenanceLoopAccept -> OK");
InstanceUnlock(pInstance);
return lError;
} // H245MaintenanceLoopAccept()
/*****************************************************************************
*
* TYPE: H245 API
*
* PROCEDURE: H245MaintenanceLoopReject
*
* DESCRIPTION
*
* RETURN:
*
*****************************************************************************/
H245DLL HRESULT
H245MaintenanceLoopReject (
H245_INST_T dwInst,
H245_CHANNEL_T wChannel,
unsigned short wCause
)
{
register struct InstanceStruct *pInstance;
register PDU_T * pPdu;
HRESULT lError;
H245TRACE (dwInst,4,"H245MaintenanceLoopReject <-");
pInstance = InstanceLock(dwInst);
if (pInstance == NULL)
{
H245TRACE (dwInst,1,"H245MaintenanceLoopReject -> %s",map_api_error(H245_ERROR_INVALID_INST));
return H245_ERROR_INVALID_INST;
}
pPdu = (PDU_T *)MemAlloc(sizeof(*pPdu));
if (pPdu == NULL)
{
lError = H245_ERROR_NOMEM;
}
else
{
memset(pPdu, 0, sizeof(*pPdu));
pPdu->choice = MSCMg_rspns_chosen;
pPdu->u.MSCMg_rspns.choice = maintenanceLoopReject_chosen;
pPdu->u.MSCMg_rspns.u.maintenanceLoopReject.type.u.mediaLoop = wChannel;
pPdu->u.MSCMg_rspns.u.maintenanceLoopReject.cause.choice = wCause;
lError = FsmOutgoing(pInstance, pPdu, 0);
MemFree(pPdu);
}
if (lError != H245_ERROR_OK)
H245TRACE (dwInst,1,"H245MaintenanceLoopReject -> %s",map_api_error(lError));
else
H245TRACE (dwInst,4,"H245MaintenanceLoopReject -> OK");
InstanceUnlock(pInstance);
return lError;
} // H245MaintenanceLoopReject()
/*****************************************************************************
*
* TYPE: H245 API
*
* PROCEDURE: H245NonStandardObject
*
* DESCRIPTION
*
* RETURN:
*
*****************************************************************************/
H245DLL HRESULT
H245NonStandardObject (
H245_INST_T dwInst,
H245_MESSAGE_TYPE_T MessageType,
const unsigned char * pData,
unsigned long dwDataLength,
const unsigned short * pwObjectId,
unsigned long dwObjectIdLength
)
{
register struct InstanceStruct *pInstance;
register PDU_T * pPdu;
HRESULT lError;
POBJECTID pObject;
H245TRACE (dwInst,4,"H245NonStandardObject <-");
if (pData == NULL || dwDataLength == 0 || pwObjectId == NULL || dwObjectIdLength == 0)
{
H245TRACE (dwInst,1,"H245NonStandardObject -> %s",map_api_error(H245_ERROR_PARAM));
return H245_ERROR_PARAM;
}
pInstance = InstanceLock(dwInst);
if (pInstance == NULL)
{
H245TRACE (dwInst,1,"H245NonStandardObject -> %s",map_api_error(H245_ERROR_INVALID_INST));
return H245_ERROR_INVALID_INST;
}
pPdu = (PDU_T *)MemAlloc(sizeof(*pPdu) + dwObjectIdLength * sizeof(*pObject));
if (pPdu == NULL)
{
lError = H245_ERROR_NOMEM;
}
else
{
memset(pPdu, 0, sizeof(*pPdu));
switch (MessageType)
{
case H245_MESSAGE_REQUEST:
pPdu->choice = MltmdSystmCntrlMssg_rqst_chosen;
pPdu->u.MltmdSystmCntrlMssg_rqst.choice = RqstMssg_nonStandard_chosen;
break;
case H245_MESSAGE_RESPONSE:
pPdu->choice = MSCMg_rspns_chosen;
pPdu->u.MSCMg_rspns.choice = RspnsMssg_nonStandard_chosen;
break;
case H245_MESSAGE_COMMAND:
pPdu->choice = MSCMg_cmmnd_chosen;
pPdu->u.MSCMg_cmmnd.choice = CmmndMssg_nonStandard_chosen;
break;
case H245_MESSAGE_INDICATION:
pPdu->choice = indication_chosen;
pPdu->u.indication.choice = IndctnMssg_nonStandard_chosen;
break;
default:
H245TRACE (dwInst,1,"H245NonStandardObject -> %s",map_api_error(H245_ERROR_PARAM));
InstanceUnlock(pInstance);
return H245_ERROR_PARAM;
} // switch
pPdu->u.indication.u.IndctnMssg_nonStandard.nonStandardData.data.length = (WORD)dwDataLength;
pPdu->u.indication.u.IndctnMssg_nonStandard.nonStandardData.data.value = (unsigned char *)pData;
pPdu->u.indication.u.IndctnMssg_nonStandard.nonStandardData.nonStandardIdentifier.choice = object_chosen;
// Copy the object identifier
pObject = (POBJECTID) (pPdu + 1);
pPdu->u.indication.u.IndctnMssg_nonStandard.nonStandardData.nonStandardIdentifier.u.object = pObject;
do
{
pObject->next = pObject + 1;
pObject->value = *pwObjectId++;
++pObject;
} while (--dwObjectIdLength);
// Null terminate the linked list
--pObject;
pObject->next = NULL;
lError = FsmOutgoing(pInstance, pPdu, 0);
MemFree(pPdu);
}
if (lError != H245_ERROR_OK)
H245TRACE (dwInst,1,"H245NonStandardObject -> %s",map_api_error(lError));
else
H245TRACE (dwInst,4,"H245NonStandardObject -> OK");
InstanceUnlock(pInstance);
return lError;
} // H245NonStandardObject()
/*****************************************************************************
*
* TYPE: H245 API
*
* PROCEDURE: H245NonStandardH221
*
* DESCRIPTION
*
* RETURN:
*
*****************************************************************************/
H245DLL HRESULT
H245NonStandardH221 (
H245_INST_T dwInst,
H245_MESSAGE_TYPE_T MessageType,
const unsigned char * pData,
unsigned long dwDataLength,
unsigned char byCountryCode,
unsigned char byExtension,
unsigned short wManufacturerCode
)
{
register struct InstanceStruct *pInstance;
register PDU_T * pPdu;
HRESULT lError;
H245TRACE (dwInst,4,"H245NonStandard221 <-");
pInstance = InstanceLock(dwInst);
if (pInstance == NULL)
{
H245TRACE (dwInst,1,"H245NonStandardH221 -> %s",map_api_error(H245_ERROR_INVALID_INST));
return H245_ERROR_INVALID_INST;
}
pPdu = (PDU_T *)MemAlloc(sizeof(*pPdu));
if (pPdu == NULL)
{
lError = H245_ERROR_NOMEM;
}
else
{
memset(pPdu, 0, sizeof(*pPdu));
switch (MessageType)
{
case H245_MESSAGE_REQUEST:
pPdu->choice = MltmdSystmCntrlMssg_rqst_chosen;
pPdu->u.MltmdSystmCntrlMssg_rqst.choice = RqstMssg_nonStandard_chosen;
break;
case H245_MESSAGE_RESPONSE:
pPdu->choice = MSCMg_rspns_chosen;
pPdu->u.MSCMg_rspns.choice = RspnsMssg_nonStandard_chosen;
break;
case H245_MESSAGE_COMMAND:
pPdu->choice = MSCMg_cmmnd_chosen;
pPdu->u.MSCMg_cmmnd.choice = CmmndMssg_nonStandard_chosen;
break;
case H245_MESSAGE_INDICATION:
pPdu->choice = indication_chosen;
pPdu->u.indication.choice = IndctnMssg_nonStandard_chosen;
break;
default:
H245TRACE (dwInst,1,"H245NonStandardH221 -> %s",map_api_error(H245_ERROR_PARAM));
InstanceUnlock(pInstance);
return H245_ERROR_PARAM;
} // switch
pPdu->u.indication.u.IndctnMssg_nonStandard.nonStandardData.data.length = (WORD)dwDataLength;
pPdu->u.indication.u.IndctnMssg_nonStandard.nonStandardData.data.value = (unsigned char *)pData;
pPdu->u.indication.u.IndctnMssg_nonStandard.nonStandardData.nonStandardIdentifier.choice = h221NonStandard_chosen;
// Fill in the H.221 identifier
pPdu->u.indication.u.IndctnMssg_nonStandard.nonStandardData.nonStandardIdentifier.u.h221NonStandard.t35CountryCode = byCountryCode;
pPdu->u.indication.u.IndctnMssg_nonStandard.nonStandardData.nonStandardIdentifier.u.h221NonStandard.t35Extension = byExtension;
pPdu->u.indication.u.IndctnMssg_nonStandard.nonStandardData.nonStandardIdentifier.u.h221NonStandard.manufacturerCode = wManufacturerCode;
lError = FsmOutgoing(pInstance, pPdu, 0);
MemFree(pPdu);
}
if (lError != H245_ERROR_OK)
H245TRACE (dwInst,1,"H245NonStandardH221 -> %s",map_api_error(lError));
else
H245TRACE (dwInst,4,"H245NonStandardH221 -> OK");
InstanceUnlock(pInstance);
return lError;
} // H245NonStandardH221
/*****************************************************************************
*
* TYPE: H245 API
*
* PROCEDURE: H245CommunicationModeRequest
*
* DESCRIPTION
*
* RETURN:
*
*****************************************************************************/
H245DLL HRESULT
H245CommunicationModeRequest(H245_INST_T dwInst)
{
register struct InstanceStruct *pInstance;
register PDU_T * pPdu;
HRESULT lError;
H245TRACE (dwInst,4,"H245CommunicationModeRequest <-");
pInstance = InstanceLock(dwInst);
if (pInstance == NULL)
{
H245TRACE (dwInst,1,"H245CommunicationModeRequest -> %s",map_api_error(H245_ERROR_INVALID_INST));
return H245_ERROR_INVALID_INST;
}
pPdu = (PDU_T *)MemAlloc(sizeof(*pPdu));
if (pPdu == NULL)
{
lError = H245_ERROR_NOMEM;
}
else
{
memset(pPdu, 0, sizeof(*pPdu));
pPdu->choice = MltmdSystmCntrlMssg_rqst_chosen;
pPdu->u.MltmdSystmCntrlMssg_rqst.choice = communicationModeRequest_chosen;
lError = FsmOutgoing(pInstance, pPdu, 0);
MemFree(pPdu);
}
if (lError != H245_ERROR_OK)
H245TRACE (dwInst,1,"H245CommunicationModeRequest -> %s",map_api_error(lError));
else
H245TRACE (dwInst,4,"H245CommunicationModeRequest -> OK");
InstanceUnlock(pInstance);
return lError;
} // H245CommunicationModeRequest
/*****************************************************************************
*
* TYPE: H245 API
*
* PROCEDURE: H245CommunicationModeResponse
*
* DESCRIPTION
*
* RETURN:
*
*****************************************************************************/
H245DLL HRESULT
H245CommunicationModeResponse(
H245_INST_T dwInst,
const H245_COMM_MODE_ENTRY_T *pTable,
unsigned char byTableCount
)
{
register struct InstanceStruct *pInstance;
register PDU_T * pPdu;
HRESULT lError = H245_ERROR_OK;
CommunicationModeTableLink pLink;
unsigned int uIndex;
H245TRACE (dwInst,4,"H245CommunicationModeResponse <-");
pInstance = InstanceLock(dwInst);
if (pInstance == NULL)
{
H245TRACE (dwInst,1,"H245CommunicationModeResponse -> %s",map_api_error(H245_ERROR_INVALID_INST));
return H245_ERROR_INVALID_INST;
}
pPdu = (PDU_T *)MemAlloc(sizeof(*pPdu) + byTableCount * sizeof(*pLink));
if (pPdu == NULL)
{
lError = H245_ERROR_NOMEM;
}
else
{
memset(pPdu, 0, sizeof(*pPdu));
pPdu->choice = MSCMg_rspns_chosen;
pPdu->u.MSCMg_rspns.choice = cmmnctnMdRspns_chosen;
pPdu->u.MSCMg_rspns.u.cmmnctnMdRspns.choice = communicationModeTable_chosen;
pLink = (CommunicationModeTableLink)(pPdu + 1);
pPdu->u.MSCMg_rspns.u.cmmnctnMdRspns.u.communicationModeTable = pLink;
for (uIndex = 0; uIndex < byTableCount; ++uIndex)
{
pLink[uIndex].next = &pLink[uIndex + 1];
lError = SetupCommModeEntry(&pLink[uIndex].value, &pTable[uIndex]);
if (lError != H245_ERROR_OK)
break;
}
pLink[byTableCount - 1].next = NULL;
if (lError == H245_ERROR_OK)
{
lError = FsmOutgoing(pInstance, pPdu, 0);
}
MemFree(pPdu);
}
if (lError != H245_ERROR_OK)
H245TRACE (dwInst,1,"H245CommunicationModeResponse -> %s",map_api_error(lError));
else
H245TRACE (dwInst,4,"H245CommunicationModeResponse -> OK");
InstanceUnlock(pInstance);
return lError;
} // H245CommunicationModeResponse()
/*****************************************************************************
*
* TYPE: H245 API
*
* PROCEDURE: H245CommunicationModeCommand
*
* DESCRIPTION
*
* RETURN:
*
*****************************************************************************/
H245DLL HRESULT
H245CommunicationModeCommand(
H245_INST_T dwInst,
const H245_COMM_MODE_ENTRY_T *pTable,
unsigned char byTableCount
)
{
register struct InstanceStruct *pInstance;
register PDU_T * pPdu;
HRESULT lError = H245_ERROR_OK;
CommunicationModeCommandLink pLink;
unsigned int uIndex;
H245TRACE (dwInst,4,"H245CommunicationModeCommand <-");
pInstance = InstanceLock(dwInst);
if (pInstance == NULL)
{
H245TRACE (dwInst,1,"H245CommunicationModeCommand -> %s",map_api_error(H245_ERROR_INVALID_INST));
return H245_ERROR_INVALID_INST;
}
pPdu = (PDU_T *)MemAlloc(sizeof(*pPdu) + byTableCount * sizeof(*pLink));
if (pPdu == NULL)
{
lError = H245_ERROR_NOMEM;
}
else
{
memset(pPdu, 0, sizeof(*pPdu));
pPdu->choice = MSCMg_cmmnd_chosen;
pPdu->u.MSCMg_cmmnd.choice = communicationModeCommand_chosen;
pLink = (CommunicationModeCommandLink)(pPdu + 1);
pPdu->u.MSCMg_cmmnd.u.communicationModeCommand.communicationModeTable = pLink;
for (uIndex = 0; uIndex < byTableCount; ++uIndex)
{
pLink[uIndex].next = &pLink[uIndex + 1];
lError = SetupCommModeEntry(&pLink[uIndex].value, &pTable[uIndex]);
if (lError != H245_ERROR_OK)
break;
}
pLink[byTableCount - 1].next = NULL;
if (lError == H245_ERROR_OK)
{
lError = FsmOutgoing(pInstance, pPdu, 0);
}
MemFree(pPdu);
}
if (lError != H245_ERROR_OK)
H245TRACE (dwInst,1,"H245CommunicationModeCommand -> %s",map_api_error(lError));
else
H245TRACE (dwInst,4,"H245CommunicationModeCommand -> OK");
InstanceUnlock(pInstance);
return lError;
} // H245CommunicationModeCommand()
/*****************************************************************************
*
* TYPE: H245 API
*
* PROCEDURE: H245ConferenceRequest
*
* DESCRIPTION
*
* RETURN:
*
*****************************************************************************/
H245DLL HRESULT
H245ConferenceRequest (
H245_INST_T dwInst,
H245_CONFER_REQ_ENUM_T RequestType,
unsigned char byMcuNumber,
unsigned char byTerminalNumber
)
{
register struct InstanceStruct *pInstance;
register PDU_T * pPdu;
HRESULT lError;
H245TRACE (dwInst,4,"H245ConferenceRequest <-");
pInstance = InstanceLock(dwInst);
if (pInstance == NULL)
{
H245TRACE (dwInst,1,"H245ConferenceRequest -> %s",map_api_error(H245_ERROR_INVALID_INST));
return H245_ERROR_INVALID_INST;
}
pPdu = (PDU_T *)MemAlloc(sizeof(*pPdu));
if (pPdu == NULL)
{
lError = H245_ERROR_NOMEM;
}
else
{
memset(pPdu, 0, sizeof(*pPdu));
pPdu->choice = MltmdSystmCntrlMssg_rqst_chosen;
pPdu->u.MltmdSystmCntrlMssg_rqst.choice = conferenceRequest_chosen;
pPdu->u.MltmdSystmCntrlMssg_rqst.u.conferenceRequest.choice = (WORD)RequestType;
switch (RequestType)
{
case dropTerminal_chosen:
case requestTerminalID_chosen:
pPdu->u.MltmdSystmCntrlMssg_rqst.u.conferenceRequest.u.dropTerminal.mcuNumber = byMcuNumber;
pPdu->u.MltmdSystmCntrlMssg_rqst.u.conferenceRequest.u.dropTerminal.terminalNumber = byTerminalNumber;
// Fall-through to next case is intentional
case terminalListRequest_chosen:
case makeMeChair_chosen:
case cancelMakeMeChair_chosen:
case enterH243Password_chosen:
case enterH243TerminalID_chosen:
case enterH243ConferenceID_chosen:
lError = FsmOutgoing(pInstance, pPdu, 0);
break;
default:
lError = H245_ERROR_PARAM;
} // switch
MemFree(pPdu);
}
if (lError != H245_ERROR_OK)
H245TRACE (dwInst,1,"H245ConferenceRequest -> %s",map_api_error(lError));
else
H245TRACE (dwInst,4,"H245ConferenceRequest -> OK");
InstanceUnlock(pInstance);
return lError;
} // H245ConferenceRequest()
/*****************************************************************************
*
* TYPE: H245 API
*
* PROCEDURE: H245ConferenceResponse
*
* DESCRIPTION
*
* RETURN:
*
*****************************************************************************/
H245DLL HRESULT
H245ConferenceResponse (
H245_INST_T dwInst,
H245_CONFER_RSP_ENUM_T ResponseType,
unsigned char byMcuNumber,
unsigned char byTerminalNumber,
const unsigned char *pOctetString,
unsigned char byOctetStringLength,
const TerminalLabel *pTerminalList,
unsigned short wTerminalListCount
)
{
register struct InstanceStruct *pInstance;
register PDU_T * pPdu;
HRESULT lError;
unsigned uIndex;
H245TRACE (dwInst,4,"H245ConferenceResponse <-");
pInstance = InstanceLock(dwInst);
if (pInstance == NULL)
{
H245TRACE (dwInst,1,"H245ConferenceResponse -> %s",map_api_error(H245_ERROR_INVALID_INST));
return H245_ERROR_INVALID_INST;
}
pPdu = (PDU_T *)MemAlloc(sizeof(*pPdu));
if (pPdu == NULL)
{
lError = H245_ERROR_NOMEM;
}
else
{
memset(pPdu, 0, sizeof(*pPdu));
pPdu->choice = MSCMg_rspns_chosen;
pPdu->u.MSCMg_rspns.choice = conferenceResponse_chosen;
pPdu->u.MSCMg_rspns.u.conferenceResponse.choice = (WORD)ResponseType;
switch (ResponseType)
{
case mCTerminalIDResponse_chosen:
case terminalIDResponse_chosen:
case conferenceIDResponse_chosen:
case passwordResponse_chosen:
if (pOctetString == NULL ||
byOctetStringLength == 0 ||
byOctetStringLength > 128)
{
H245TRACE (dwInst,1,"H245ConferenceResponse -> %s",map_api_error(H245_ERROR_PARAM));
return H245_ERROR_PARAM;
}
pPdu->u.MSCMg_rspns.u.conferenceResponse.u.mCTerminalIDResponse.terminalLabel.mcuNumber = byMcuNumber;
pPdu->u.MSCMg_rspns.u.conferenceResponse.u.mCTerminalIDResponse.terminalLabel.terminalNumber = byTerminalNumber;
pPdu->u.MSCMg_rspns.u.conferenceResponse.u.mCTerminalIDResponse.terminalID.length = byOctetStringLength;
memcpy(pPdu->u.MSCMg_rspns.u.conferenceResponse.u.mCTerminalIDResponse.terminalID.value,
pOctetString,
byOctetStringLength);
// Fall-through to next case is intentional
case videoCommandReject_chosen:
case terminalDropReject_chosen:
lError = FsmOutgoing(pInstance, pPdu, 0);
break;
case terminalListResponse_chosen:
if (pTerminalList == NULL ||
wTerminalListCount == 0 ||
wTerminalListCount > 256)
{
H245TRACE (dwInst,1,"H245ConferenceResponse -> %s",map_api_error(H245_ERROR_PARAM));
return H245_ERROR_PARAM;
}
pPdu->u.MSCMg_rspns.u.conferenceResponse.u.terminalListResponse.count = wTerminalListCount;
for (uIndex = 0; uIndex < wTerminalListCount; ++uIndex)
{
pPdu->u.MSCMg_rspns.u.conferenceResponse.u.terminalListResponse.value[uIndex] =
pTerminalList[uIndex];
}
lError = FsmOutgoing(pInstance, pPdu, 0);
break;
case H245_RSP_DENIED_CHAIR_TOKEN:
pPdu->u.MSCMg_rspns.u.conferenceResponse.choice = makeMeChairResponse_chosen;
pPdu->u.MSCMg_rspns.u.conferenceResponse.u.makeMeChairResponse.choice = deniedChairToken_chosen;
lError = FsmOutgoing(pInstance, pPdu, 0);
break;
case H245_RSP_GRANTED_CHAIR_TOKEN:
pPdu->u.MSCMg_rspns.u.conferenceResponse.choice = makeMeChairResponse_chosen;
pPdu->u.MSCMg_rspns.u.conferenceResponse.u.makeMeChairResponse.choice = grantedChairToken_chosen;
lError = FsmOutgoing(pInstance, pPdu, 0);
break;
default:
lError = H245_ERROR_PARAM;
} // switch
MemFree(pPdu);
}
if (lError != H245_ERROR_OK)
H245TRACE (dwInst,1,"H245ConferenceResponse -> %s",map_api_error(lError));
else
H245TRACE (dwInst,4,"H245ConferenceResponse -> OK");
InstanceUnlock(pInstance);
return lError;
} // H245ConferenceResponse()
/*****************************************************************************
*
* TYPE: H245 API
*
* PROCEDURE: H245ConferenceCommand
*
* DESCRIPTION
*
* RETURN:
*
*****************************************************************************/
H245DLL HRESULT
H245ConferenceCommand (
H245_INST_T dwInst,
H245_CONFER_CMD_ENUM_T CommandType,
H245_CHANNEL_T Channel,
unsigned char byMcuNumber,
unsigned char byTerminalNumber
)
{
register struct InstanceStruct *pInstance;
register PDU_T * pPdu;
HRESULT lError;
H245TRACE (dwInst,4,"H245ConferenceCommand <-");
pInstance = InstanceLock(dwInst);
if (pInstance == NULL)
{
H245TRACE (dwInst,1,"H245ConferenceCommand -> %s",map_api_error(H245_ERROR_INVALID_INST));
return H245_ERROR_INVALID_INST;
}
pPdu = (PDU_T *)MemAlloc(sizeof(*pPdu));
if (pPdu == NULL)
{
lError = H245_ERROR_NOMEM;
}
else
{
memset(pPdu, 0, sizeof(*pPdu));
pPdu->choice = MSCMg_cmmnd_chosen;
pPdu->u.MSCMg_cmmnd.choice = conferenceCommand_chosen;
pPdu->u.MSCMg_cmmnd.u.conferenceCommand.choice = (WORD)CommandType;
switch (CommandType)
{
case brdcstMyLgclChnnl_chosen:
case cnclBrdcstMyLgclChnnl_chosen:
pPdu->u.MSCMg_cmmnd.u.conferenceCommand.u.brdcstMyLgclChnnl = Channel;
lError = FsmOutgoing(pInstance, pPdu, 0);
break;
case ConferenceCommand_makeTerminalBroadcaster_chosen:
case ConferenceCommand_sendThisSource_chosen:
pPdu->u.MSCMg_cmmnd.u.conferenceCommand.u.makeTerminalBroadcaster.mcuNumber = byMcuNumber;
pPdu->u.MSCMg_cmmnd.u.conferenceCommand.u.makeTerminalBroadcaster.terminalNumber = byTerminalNumber;
// Fall-through to next case is intentional
case cnclMkTrmnlBrdcstr_chosen:
case cancelSendThisSource_chosen:
case dropConference_chosen:
lError = FsmOutgoing(pInstance, pPdu, 0);
break;
default:
lError = H245_ERROR_PARAM;
} // switch
MemFree(pPdu);
}
if (lError != H245_ERROR_OK)
H245TRACE (dwInst,1,"H245ConferenceCommand -> %s",map_api_error(lError));
else
H245TRACE (dwInst,4,"H245ConferenceCommand -> OK");
InstanceUnlock(pInstance);
return lError;
} // H245ConferenceCommand()
/*****************************************************************************
*
* TYPE: H245 API
*
* PROCEDURE: H245ConferenceIndication
*
* DESCRIPTION
*
* RETURN:
*
*****************************************************************************/
H245DLL HRESULT
H245ConferenceIndication(
H245_INST_T dwInst,
H245_CONFER_IND_ENUM_T IndicationType,
unsigned char bySbeNumber,
unsigned char byMcuNumber,
unsigned char byTerminalNumber
)
{
register struct InstanceStruct *pInstance;
register PDU_T * pPdu;
HRESULT lError;
H245TRACE (dwInst,4,"H245ConferenceIndication <-");
pInstance = InstanceLock(dwInst);
if (pInstance == NULL)
{
H245TRACE (dwInst,1,"H245ConferenceIndication -> %s",map_api_error(H245_ERROR_INVALID_INST));
return H245_ERROR_INVALID_INST;
}
pPdu = (PDU_T *)MemAlloc(sizeof(*pPdu));
if (pPdu == NULL)
{
lError = H245_ERROR_NOMEM;
}
else
{
memset(pPdu, 0, sizeof(*pPdu));
pPdu->choice = indication_chosen;
pPdu->u.indication.choice = conferenceIndication_chosen;
pPdu->u.indication.u.conferenceIndication.choice = (WORD)IndicationType;
switch (IndicationType)
{
case sbeNumber_chosen:
pPdu->u.indication.u.conferenceIndication.u.sbeNumber = bySbeNumber;
lError = FsmOutgoing(pInstance, pPdu, 0);
break;
case terminalNumberAssign_chosen:
case terminalJoinedConference_chosen:
case terminalLeftConference_chosen:
case terminalYouAreSeeing_chosen:
pPdu->u.indication.u.conferenceIndication.u.terminalNumberAssign.mcuNumber = byMcuNumber;
pPdu->u.indication.u.conferenceIndication.u.terminalNumberAssign.terminalNumber = byTerminalNumber;
// Fall-through to next case is intentional
case seenByAtLeastOneOther_chosen:
case cnclSnByAtLstOnOthr_chosen:
case seenByAll_chosen:
case cancelSeenByAll_chosen:
case requestForFloor_chosen:
lError = FsmOutgoing(pInstance, pPdu, 0);
break;
default:
lError = H245_ERROR_PARAM;
} // switch
MemFree(pPdu);
}
if (lError != H245_ERROR_OK)
H245TRACE (dwInst,1,"H245ConferenceIndication -> %s",map_api_error(lError));
else
H245TRACE (dwInst,4,"H245ConferenceIndication -> OK");
InstanceUnlock(pInstance);
return lError;
} // H245ConferenceIndication()
/*****************************************************************************
*
* TYPE: H245 API
*
*****************************************************************************
*
* PROCEDURE: H245UserInput
*
* DESCRIPTION
*
* HRESULT H245UserInput (
* H245_INST_T dwInst,
* char *pGenString,
* H245_NONSTANDARD_PARAMETER_T *pNonStd
* )
* Description:
*
* Send a User Input indiation to the remote side. One of the
* two parameters must be set (pGenString, pNonStd). The client
* can either send a string or a NonStandard parameter set to the
* remote client. Only one of the two parameters can contain a
* value. The other is required to be NULL.
*
* Input
* dwInst Instance handle returned by H245Init
* pGenString choice: String to be sent to remote
* side in accordance with T.51 specification.
* pNonStd choice: NonStandard Parameter
*
* Call Type:
* Synchronous
*
* Return Values:
* See Errors
*
* Errors:
* H245_ERROR_OK
* H245_ERROR_NOT_CONNECTED
* H245_ERROR_NOMEM
* H245_ERROR_PARAM
*
* callback
* H245_IND_USERINPUT
*
*
* RETURN:
*
*****************************************************************************/
H245DLL HRESULT
H245UserInput (
H245_INST_T dwInst,
const WCHAR * pGenString,
const H245_NONSTANDARD_PARAMETER_T * pNonStd
)
{
register struct InstanceStruct *pInstance;
HRESULT lError;
MltmdSystmCntrlMssg *pPdu;
#if 1
int nLength;
char * pszGeneral = NULL;
#endif
H245TRACE (dwInst,4,"H245UserInput <-");
/* check for valid instance handle */
pInstance = InstanceLock(dwInst);
if (pInstance == NULL)
{
H245TRACE (dwInst,1,"H245UserInput -> %s",map_api_error(H245_ERROR_INVALID_INST));
return H245_ERROR_INVALID_INST;
}
/* system should be in connected state */
if (pInstance->API.SystemState != APIST_Connected)
{
H245TRACE (dwInst,1,"H245UserInput -> %s",map_api_error(H245_ERROR_NOT_CONNECTED));
InstanceUnlock(pInstance);
return H245_ERROR_NOT_CONNECTED;
}
pPdu = (MltmdSystmCntrlMssg *)MemAlloc(sizeof(MltmdSystmCntrlMssg));
if (pPdu == NULL)
{
H245TRACE (dwInst,1,"H245UserInput -> %s",map_api_error(H245_ERROR_NOMEM));
InstanceUnlock(pInstance);
return H245_ERROR_NOMEM;
}
memset(pPdu, 0, sizeof(MltmdSystmCntrlMssg));
/* build PDU */
#if 1
if (pGenString)
{
nLength = WideCharToMultiByte(CP_ACP, // code page
0, // dwFlags
pGenString, // Unicode string
-1, // Unicode string length (bytes)
NULL, // ASCII string
0, // max ASCII string length
NULL, // default character
NULL); // default character used
pszGeneral = MemAlloc(nLength);
if (pszGeneral == NULL)
{
H245TRACE (dwInst,1,"H245UserInput -> %s",map_api_error(H245_ERROR_NOMEM));
InstanceUnlock(pInstance);
return H245_ERROR_NOMEM;
}
nLength = WideCharToMultiByte(CP_ACP, // code page
0, // dwFlags
pGenString, // Unicode string
-1, // Unicode string length (bytes)
pszGeneral, // ASCII string
nLength, // max ASCII string length
NULL, // default character
NULL); // default character used
lError = pdu_ind_usrinpt (pPdu, NULL, pszGeneral);
}
else
{
lError = pdu_ind_usrinpt (pPdu, pNonStd, NULL);
}
#else
lError = pdu_ind_usrinpt (pPdu, pNonStd, pGenString);
#endif
if (lError == H245_ERROR_OK)
lError = FsmOutgoing(pInstance, pPdu, 0);
#if 1
if (pszGeneral)
MemFree(pszGeneral);
#endif
MemFree (pPdu);
if (lError != H245_ERROR_OK)
H245TRACE (dwInst,1,"H245UserInput -> %s",map_api_error(lError));
else
H245TRACE (dwInst,4,"H245UserInput -> OK");
InstanceUnlock(pInstance);
return lError;
} // H245UserInput()
/*****************************************************************************
*
* TYPE: H245 API
*
* PROCEDURE: H245FlowControl
*
* DESCRIPTION
*
* RETURN:
*
*****************************************************************************/
H245DLL HRESULT
H245FlowControl (
H245_INST_T dwInst,
H245_SCOPE_T Scope,
H245_CHANNEL_T Channel, // only used if Scope is H245_SCOPE_CHANNEL_NUMBER
unsigned short wResourceID, // only used if Scope is H245_SCOPE_RESOURCE_ID
unsigned long dwRestriction // H245_NO_RESTRICTION if no restriction
)
{
register struct InstanceStruct *pInstance;
register PDU_T * pPdu;
HRESULT lError;
H245TRACE (dwInst,4,"H245FlowControl <-");
pInstance = InstanceLock(dwInst);
if (pInstance == NULL)
{
H245TRACE (dwInst,1,"H245FlowControl -> %s",map_api_error(H245_ERROR_INVALID_INST));
return H245_ERROR_INVALID_INST;
}
pPdu = (PDU_T *)MemAlloc(sizeof(*pPdu));
if (pPdu == NULL)
{
lError = H245_ERROR_NOMEM;
}
else
{
memset(pPdu, 0, sizeof(*pPdu));
pPdu->choice = MSCMg_cmmnd_chosen;
pPdu->u.MSCMg_cmmnd.choice = flowControlCommand_chosen;
pPdu->u.MSCMg_cmmnd.u.flowControlCommand.scope.choice = (WORD)Scope;
if (dwRestriction == H245_NO_RESTRICTION)
{
pPdu->u.MSCMg_cmmnd.u.flowControlCommand.restriction.choice = noRestriction_chosen;
}
else
{
pPdu->u.MSCMg_cmmnd.u.flowControlCommand.restriction.choice = maximumBitRate_chosen;
pPdu->u.MSCMg_cmmnd.u.flowControlCommand.restriction.u.maximumBitRate = dwRestriction;
}
switch (Scope)
{
case FCCd_scp_lgclChnnlNmbr_chosen:
pPdu->u.MSCMg_cmmnd.u.flowControlCommand.scope.u.FCCd_scp_lgclChnnlNmbr = Channel;
lError = FsmOutgoing(pInstance, pPdu, 0);
break;
case FlwCntrlCmmnd_scp_rsrcID_chosen:
pPdu->u.MSCMg_cmmnd.u.flowControlCommand.scope.u.FlwCntrlCmmnd_scp_rsrcID = wResourceID;
// Fall-through to next case
case FCCd_scp_whlMltplx_chosen:
lError = FsmOutgoing(pInstance, pPdu, 0);
break;
default:
lError = H245_ERROR_PARAM;
} // switch
MemFree(pPdu);
}
if (lError != H245_ERROR_OK)
H245TRACE (dwInst,1,"H245FlowControl -> %s",map_api_error(lError));
else
H245TRACE (dwInst,4,"H245FlowControl -> OK");
InstanceUnlock(pInstance);
return lError;
} // H245FlowControl()
/*****************************************************************************
*
* TYPE: H245 API
*
* PROCEDURE: H245H223SkewIndication
*
* DESCRIPTION
*
* RETURN:
*
*****************************************************************************/
H245DLL HRESULT
H245H223SkewIndication (
H245_INST_T dwInst,
H245_CHANNEL_T wLogicalChannelNumber1,
H245_CHANNEL_T wLogicalChannelNumber2,
unsigned short wSkew
)
{
register struct InstanceStruct *pInstance;
register PDU_T * pPdu;
HRESULT lError;
H245TRACE (dwInst,4,"H245H223SkewIndication <-");
pInstance = InstanceLock(dwInst);
if (pInstance == NULL)
{
H245TRACE (dwInst,1,"H245H223SkewIndication -> %s",map_api_error(H245_ERROR_INVALID_INST));
return H245_ERROR_INVALID_INST;
}
pPdu = (PDU_T *)MemAlloc(sizeof(*pPdu));
if (pPdu == NULL)
{
lError = H245_ERROR_NOMEM;
}
else
{
memset(pPdu, 0, sizeof(*pPdu));
pPdu->choice = indication_chosen;
pPdu->u.indication.choice = h223SkewIndication_chosen;
pPdu->u.indication.u.h223SkewIndication.logicalChannelNumber1 = wLogicalChannelNumber1;
pPdu->u.indication.u.h223SkewIndication.logicalChannelNumber2 = wLogicalChannelNumber2;
pPdu->u.indication.u.h223SkewIndication.skew = wSkew;
lError = FsmOutgoing(pInstance, pPdu, 0);
MemFree(pPdu);
}
if (lError != H245_ERROR_OK)
H245TRACE (dwInst,1,"H245H223SkewIndication -> %s",map_api_error(lError));
else
H245TRACE (dwInst,4,"H245H223SkewIndication -> OK");
InstanceUnlock(pInstance);
return lError;
} // H245H223SkewIndication()
/*****************************************************************************
*
* TYPE: H245 API
*
* PROCEDURE: H245H2250MaximumSkewIndication
*
* DESCRIPTION
*
* RETURN:
*
*****************************************************************************/
H245DLL HRESULT
H245H2250MaximumSkewIndication(
H245_INST_T dwInst,
H245_CHANNEL_T wLogicalChannelNumber1,
H245_CHANNEL_T wLogicalChannelNumber2,
unsigned short wMaximumSkew
)
{
register struct InstanceStruct *pInstance;
register PDU_T * pPdu;
HRESULT lError;
H245TRACE (dwInst,4,"H245H2250MaximumSkewIndication <-");
pInstance = InstanceLock(dwInst);
if (pInstance == NULL)
{
H245TRACE (dwInst,1,"H245H2250MaximumSkewIndication -> %s",map_api_error(H245_ERROR_INVALID_INST));
return H245_ERROR_INVALID_INST;
}
pPdu = (PDU_T *)MemAlloc(sizeof(*pPdu));
if (pPdu == NULL)
{
lError = H245_ERROR_NOMEM;
}
else
{
memset(pPdu, 0, sizeof(*pPdu));
pPdu->choice = indication_chosen;
pPdu->u.indication.choice = h2250MxmmSkwIndctn_chosen;
pPdu->u.indication.u.h2250MxmmSkwIndctn.logicalChannelNumber1 = wLogicalChannelNumber1;
pPdu->u.indication.u.h2250MxmmSkwIndctn.logicalChannelNumber2 = wLogicalChannelNumber2;
pPdu->u.indication.u.h2250MxmmSkwIndctn.maximumSkew = wMaximumSkew;
lError = FsmOutgoing(pInstance, pPdu, 0);
MemFree(pPdu);
}
if (lError != H245_ERROR_OK)
H245TRACE (dwInst,1,"H245H2250MaximumSkewIndication -> %s",map_api_error(lError));
else
H245TRACE (dwInst,4,"H245H2250MaximumSkewIndication -> OK");
InstanceUnlock(pInstance);
return lError;
} // H245H2250MaximumSkewIndication()
/*****************************************************************************
*
* TYPE: H245 API
*
* PROCEDURE: H245MCLocationIndication
*
* DESCRIPTION
*
* RETURN:
*
*****************************************************************************/
H245DLL HRESULT
H245MCLocationIndication(
H245_INST_T dwInst,
const H245_TRANSPORT_ADDRESS_T * pSignalAddress
)
{
register struct InstanceStruct *pInstance;
register PDU_T * pPdu;
HRESULT lError;
H245TRACE (dwInst,4,"H245MCLocationIndication <-");
pInstance = InstanceLock(dwInst);
if (pInstance == NULL)
{
H245TRACE (dwInst,1,"H245MCLocationIndication -> %s",map_api_error(H245_ERROR_INVALID_INST));
return H245_ERROR_INVALID_INST;
}
pPdu = (PDU_T *)MemAlloc(sizeof(*pPdu));
if (pPdu == NULL)
{
lError = H245_ERROR_NOMEM;
}
else
{
memset(pPdu, 0, sizeof(*pPdu));
pPdu->choice = indication_chosen;
pPdu->u.indication.choice = mcLocationIndication_chosen;
lError = SetupTransportAddress(&pPdu->u.indication.u.mcLocationIndication.signalAddress,
pSignalAddress);
if (lError == H245_ERROR_OK)
{
lError = FsmOutgoing(pInstance, pPdu, 0);
}
MemFree(pPdu);
}
if (lError != H245_ERROR_OK)
H245TRACE (dwInst,1,"H245MCLocationIndication -> %s",map_api_error(lError));
else
H245TRACE (dwInst,4,"H245MCLocationIndication -> OK");
InstanceUnlock(pInstance);
return lError;
} // H245MCLocationIndication()
/*****************************************************************************
*
* TYPE: H245 API
*
* PROCEDURE: H245VendorIdentification
*
* DESCRIPTION
*
* RETURN:
*
*****************************************************************************/
H245DLL HRESULT
H245VendorIdentification(
H245_INST_T dwInst,
const H245_NONSTANDID_T *pIdentifier,
const unsigned char *pProductNumber,
unsigned char byProductNumberLength,
const unsigned char *pVersionNumber,
unsigned char byVersionNumberLength
)
{
register struct InstanceStruct *pInstance;
register PDU_T * pPdu;
HRESULT lError;
H245TRACE (dwInst,4,"H245VendorIdentification <-");
pInstance = InstanceLock(dwInst);
if (pInstance == NULL)
{
H245TRACE (dwInst,1,"H245VendorIdentification -> %s",map_api_error(H245_ERROR_INVALID_INST));
return H245_ERROR_INVALID_INST;
}
pPdu = (PDU_T *)MemAlloc(sizeof(*pPdu));
if (pPdu == NULL)
{
lError = H245_ERROR_NOMEM;
}
else
{
memset(pPdu, 0, sizeof(*pPdu));
pPdu->choice = indication_chosen;
pPdu->u.indication.choice = vendorIdentification_chosen;
pPdu->u.indication.u.vendorIdentification.bit_mask = 0;
pPdu->u.indication.u.vendorIdentification.vendor = *pIdentifier;
if (pProductNumber != NULL && byProductNumberLength != 0)
{
pPdu->u.indication.u.vendorIdentification.bit_mask |= productNumber_present;
pPdu->u.indication.u.vendorIdentification.productNumber.length = byProductNumberLength;
memcpy(pPdu->u.indication.u.vendorIdentification.productNumber.value,
pProductNumber,
byProductNumberLength);
}
if (pVersionNumber != NULL && byVersionNumberLength != 0)
{
pPdu->u.indication.u.vendorIdentification.bit_mask |= versionNumber_present;
pPdu->u.indication.u.vendorIdentification.versionNumber.length = byVersionNumberLength;
memcpy(pPdu->u.indication.u.vendorIdentification.versionNumber.value,
pVersionNumber,
byVersionNumberLength);
}
lError = FsmOutgoing(pInstance, pPdu, 0);
MemFree(pPdu);
}
if (lError != H245_ERROR_OK)
H245TRACE (dwInst,1,"H245VendorIdentification -> %s",map_api_error(lError));
else
H245TRACE (dwInst,4,"H245VendorIdentification -> OK");
InstanceUnlock(pInstance);
return lError;
} // H245VendorIdentification()
/*****************************************************************************
*
* TYPE: H245 API
*
* PROCEDURE: H245SendPDU
*
* DESCRIPTION
*
* RETURN:
*
*****************************************************************************/
H245DLL HRESULT
H245SendPDU (
H245_INST_T dwInst,
PDU_T * pPdu
)
{
register struct InstanceStruct *pInstance;
HRESULT lError;
H245TRACE (dwInst,4,"H245SendPDU <-");
// Check for valid instance handle
pInstance = InstanceLock(dwInst);
if (pInstance == NULL)
{
H245TRACE (dwInst,1,"H245SendPDU -> %s",map_api_error(H245_ERROR_INVALID_INST));
return H245_ERROR_INVALID_INST;
}
lError = FsmOutgoing(pInstance, pPdu, 0);
if (lError != H245_ERROR_OK)
H245TRACE (dwInst,1,"H245SendPDU -> %s",map_api_error(lError));
else
H245TRACE (dwInst,4,"H245SendPDU -> OK");
InstanceUnlock(pInstance);
return lError;
} // H245SendPDU()
/*****************************************************************************
*
* TYPE: H245 API
*
*****************************************************************************
*
* PROCEDURE: H245SystemControl
*
* DESCRIPTION
*
* HRESULT H245SystemControl
* ( H245_INST_T dwInst,
* DWORD Request ,
* VOID *pData
* )
*
* Description:
* This function should not be used by clients who
* normally interface to the H.245 subsystem. It is
* defined here to help during development and debug
* of the H.245 subsystem.
*
* This is a roll your own.. and can do what
* ever the user needs.. It's a hook to allow
* IOCTL (unix) calls that can either be
* passed to lower stack elements (AT&T Streams IOCTL
* would be an example - :) or simply to get or put
* information to the H245 SubSytem.
*
* Input
* dwInst Instance handle returned by H245Init
* Request Requested system control
* pData In the case of sending information
* down to H.245 this is an input
* parameter, and it's data format
* is determined by the Request.
* output
* pData In the case of retrieving information
* from H.245 this can be an output
* parameter, and it's data format is
* determined by the Request. It may not
* have valid data if the request is a
* synchronous request. (See Request Options).
* Call Type:
*
* Synchronous
*
* Request Options:
*
* H245_SYSCON_GET_STATS Retrieves Statistics
* from H.245 subsystem
* parameter pData = &H245_SYSCON_STAT_T
* H245_ SYSCON_RESET_STATS Resets the statistics
* pData = NULL
* H245_SYS_TRACE Set Trace Level
* pData = &DWORD (Trace Level)
*
* Return Values:
* See Request Options
*
* Errors:
* H245_ERROR_OK
*
* RETURN:
*
*****************************************************************************/
H245DLL HRESULT
H245SystemControl (
H245_INST_T dwInst,
unsigned long dwRequest,
void * pData
)
{
HRESULT lError;
DWORD dwTemp;
H245TRACE(dwInst,4,"H245SystemControl <-");
if (dwRequest == H245_SYSCON_DUMP_TRACKER)
{
register struct InstanceStruct *pInstance = InstanceLock(dwInst);
if (pInstance == NULL)
{
lError = H245_ERROR_INVALID_INST;
}
else
{
dump_tracker(pInstance);
InstanceUnlock(pInstance);
lError = H245_ERROR_OK;
}
}
else if (pData == NULL)
{
lError = H245_ERROR_PARAM;
}
else
{
lError = H245_ERROR_OK;
switch (dwRequest)
{
case H245_SYSCON_GET_FSM_N100:
*((DWORD *)pData) = (DWORD) uN100;
H245TRACE(dwInst,20,"H245SystemControl: Get N100 = %d",*((DWORD *)pData));
break;
case H245_SYSCON_GET_FSM_T101:
*((DWORD *)pData) = (DWORD) uT101;
H245TRACE(dwInst,20,"H245SystemControl: Get T101 = %d",*((DWORD *)pData));
break;
case H245_SYSCON_GET_FSM_T102:
*((DWORD *)pData) = (DWORD) uT102;
H245TRACE(dwInst,20,"H245SystemControl: Get T102 = %d",*((DWORD *)pData));
break;
case H245_SYSCON_GET_FSM_T103:
*((DWORD *)pData) = (DWORD) uT103;
H245TRACE(dwInst,20,"H245SystemControl: Get T103 = %d",*((DWORD *)pData));
break;
case H245_SYSCON_GET_FSM_T104:
*((DWORD *)pData) = (DWORD) uT104;
H245TRACE(dwInst,20,"H245SystemControl: Get T104 = %d",*((DWORD *)pData));
break;
case H245_SYSCON_GET_FSM_T105:
*((DWORD *)pData) = (DWORD) uT105;
H245TRACE(dwInst,20,"H245SystemControl: Get T105 = %d",*((DWORD *)pData));
break;
case H245_SYSCON_GET_FSM_T106:
*((DWORD *)pData) = (DWORD) uT106;
H245TRACE(dwInst,20,"H245SystemControl: Get T106 = %d",*((DWORD *)pData));
break;
case H245_SYSCON_GET_FSM_T107:
*((DWORD *)pData) = (DWORD) uT107;
H245TRACE(dwInst,20,"H245SystemControl: Get T107 = %d",*((DWORD *)pData));
break;
case H245_SYSCON_GET_FSM_T108:
*((DWORD *)pData) = (DWORD) uT108;
H245TRACE(dwInst,20,"H245SystemControl: Get T108 = %d",*((DWORD *)pData));
break;
case H245_SYSCON_GET_FSM_T109:
*((DWORD *)pData) = (DWORD) uT109;
H245TRACE(dwInst,20,"H245SystemControl: Get T109 = %d",*((DWORD *)pData));
break;
case H245_SYSCON_SET_FSM_N100:
dwTemp = (DWORD) uN100;
uN100 = (unsigned int) *((DWORD *)pData);
H245TRACE(dwInst,20,"H245SystemControl: Set N100 = %d",uN100);
*((DWORD *)pData) = dwTemp;
break;
case H245_SYSCON_SET_FSM_T101:
dwTemp = (DWORD) uT101;
uT101 = (unsigned int) *((DWORD *)pData);
H245TRACE(dwInst,20,"H245SystemControl: Set T101 = %d",uT101);
*((DWORD *)pData) = dwTemp;
break;
case H245_SYSCON_SET_FSM_T102:
dwTemp = (DWORD) uT102;
uT102 = (unsigned int) *((DWORD *)pData);
H245TRACE(dwInst,20,"H245SystemControl: Set T102 = %d",uT102);
*((DWORD *)pData) = dwTemp;
break;
case H245_SYSCON_SET_FSM_T103:
dwTemp = (DWORD) uT103;
uT103 = (unsigned int) *((DWORD *)pData);
H245TRACE(dwInst,20,"H245SystemControl: Set T103 = %d",uT103);
*((DWORD *)pData) = dwTemp;
break;
case H245_SYSCON_SET_FSM_T104:
dwTemp = (DWORD) uT104;
uT104 = (unsigned int) *((DWORD *)pData);
H245TRACE(dwInst,20,"H245SystemControl: Set T104 = %d",uT104);
*((DWORD *)pData) = dwTemp;
break;
case H245_SYSCON_SET_FSM_T105:
dwTemp = (DWORD) uT105;
uT105 = (unsigned int) *((DWORD *)pData);
H245TRACE(dwInst,20,"H245SystemControl: Set T105 = %d",uT105);
*((DWORD *)pData) = dwTemp;
break;
case H245_SYSCON_SET_FSM_T106:
dwTemp = (DWORD) uT106;
uT106 = (unsigned int) *((DWORD *)pData);
H245TRACE(dwInst,20,"H245SystemControl: Set T106 = %d",uT106);
*((DWORD *)pData) = dwTemp;
break;
case H245_SYSCON_SET_FSM_T107:
dwTemp = (DWORD) uT107;
uT107 = (unsigned int) *((DWORD *)pData);
H245TRACE(dwInst,20,"H245SystemControl: Set T107 = %d",uT107);
*((DWORD *)pData) = dwTemp;
break;
case H245_SYSCON_SET_FSM_T108:
dwTemp = (DWORD) uT108;
uT108 = (unsigned int) *((DWORD *)pData);
H245TRACE(dwInst,20,"H245SystemControl: Set T108 = %d",uT108);
*((DWORD *)pData) = dwTemp;
break;
case H245_SYSCON_SET_FSM_T109:
dwTemp = (DWORD) uT109;
uT109 = (unsigned int) *((DWORD *)pData);
H245TRACE(dwInst,20,"H245SystemControl: Set T109 = %d",uT109);
*((DWORD *)pData) = dwTemp;
break;
case H245_SYSCON_TRACE_LVL:
dwTemp = TraceLevel;
TraceLevel = *(DWORD *)pData;
H245TRACE(dwInst,20,"H245SystemControl: Set TraceLevel = %d",TraceLevel);
*((DWORD *)pData) = dwTemp;
break;
default:
lError = H245_ERROR_NOTIMP;
} // switch
} // else
if (lError != H245_ERROR_OK)
H245TRACE(dwInst,1,"H245SystemControl -> %s",map_api_error(lError));
else
H245TRACE(dwInst,4,"H245SystemControl -> OK");
return lError;
} // H245SystemControl()