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.
 
 
 
 
 
 

954 lines
30 KiB

/*
ÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛ
(C) Copyright 1998
All rights reserved.
ÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛ
Portions of this software are:
(C) Copyright 1995, 1999 TriplePoint, Inc. -- http://www.TriplePoint.com
License to use this software is granted under the terms outlined in
the TriplePoint Software Services Agreement.
(C) Copyright 1992 Microsoft Corp. -- http://www.Microsoft.com
License to use this software is granted under the terms outlined in
the Microsoft Windows Device Driver Development Kit.
ÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛ
@doc INTERNAL TspiLine TspiLine_c
@module TspiLine.c |
This module implements the Telephony Service Provider Interface for
Line objects (TapiLine).
@head3 Contents |
@index class,mfunc,func,msg,mdata,struct,enum | TspiLine_c
@end
ÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛ
*/
#define __FILEID__ TSPILINE_OBJECT_TYPE
// Unique file ID for error logging
#include "Miniport.h" // Defines all the miniport objects
#include "string.h"
#if defined(NDIS_LCODE)
# pragma NDIS_LCODE // Windows 95 wants this code locked down!
# pragma NDIS_LDATA
#endif
/* @doc INTERNAL TspiLine TspiLine_c TspiOpen
ÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛ
@func
This function opens the line device whose device ID is given, returning
the miniport's handle for the device. The miniport must retain the
Connection Wrapper's handle for the device for use in subsequent calls to
the LINE_EVENT callback procedure.
@parm IN PMINIPORT_ADAPTER_OBJECT | pAdapter |
A pointer to the Miniport's adapter context structure <t MINIPORT_ADAPTER_OBJECT>.
This is the <t MiniportAdapterContext> we passed into <f NdisMSetAttributes>.
@parm IN PNDIS_TAPI_OPEN | Request |
A pointer to the NDIS_TAPI request structure for this call.
@iex
typedef struct _NDIS_TAPI_OPEN
{
IN ULONG ulRequestID;
IN ULONG ulDeviceID;
IN HTAPI_LINE htLine;
OUT HDRV_LINE hdLine;
} NDIS_TAPI_OPEN, *PNDIS_TAPI_OPEN;
@rdesc This routine returns one of the following values:
@flag NDIS_STATUS_SUCCESS |
If this function is successful.
<f Note>: A non-zero return value indicates one of the following error codes:
@iex
NDIS_STATUS_PENDING
NDIS_STATUS_TAPI_ALLOCATED
NDIS_STATUS_TAPI_NODRIVER
*/
NDIS_STATUS TspiOpen(
IN PMINIPORT_ADAPTER_OBJECT pAdapter,
IN PNDIS_TAPI_OPEN Request,
OUT PULONG BytesWritten,
OUT PULONG BytesNeeded
)
{
DBG_FUNC("TspiOpen")
PBCHANNEL_OBJECT pBChannel;
// A Pointer to one of our <t BCHANNEL_OBJECT>'s.
DBG_ENTER(pAdapter);
DBG_PARAMS(pAdapter,
("\n\tulDeviceID=%d\n"
"\thtLine=0x%X\n",
Request->ulDeviceID,
Request->htLine
));
/*
// If there is no DChannel, we can't allow an open line.
*/
if (pAdapter->pDChannel == NULL)
{
DBG_WARNING(pAdapter, ("Returning NDIS_STATUS_TAPI_NODRIVER\n"));
return (NDIS_STATUS_TAPI_NODRIVER);
}
/*
// This request must be associated with a line device.
*/
pBChannel = GET_BCHANNEL_FROM_DEVICEID(pAdapter, Request->ulDeviceID);
if (pBChannel == NULL)
{
DBG_WARNING(pAdapter, ("Returning NDIS_STATUS_TAPI_NODEVICE\n"));
return (NDIS_STATUS_TAPI_NODEVICE);
}
/*
// Make sure the requested line device is not already in use.
*/
if (BChannelOpen(pBChannel, Request->htLine) != NDIS_STATUS_SUCCESS)
{
DBG_WARNING(pAdapter, ("Returning NDIS_STATUS_TAPI_ALLOCATED\n"));
return (NDIS_STATUS_TAPI_ALLOCATED);
}
/*
// Tell the wrapper the line context and set the line/call state.
*/
Request->hdLine = (HDRV_LINE) pBChannel;
/*
// Make sure the line is configured for dialing when we open up.
*/
TspiLineDevStateHandler(pAdapter, pBChannel, LINEDEVSTATE_OPEN);
DBG_RETURN(pAdapter, NDIS_STATUS_SUCCESS);
return (NDIS_STATUS_SUCCESS);
}
/* @doc INTERNAL TspiLine TspiLine_c TspiClose
ÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛ
@func
This request closes the specified open line device after completing or
aborting all outstanding calls and asynchronous requests on the device.
@parm IN PMINIPORT_ADAPTER_OBJECT | pAdapter |
A pointer to the Miniport's adapter context structure <t MINIPORT_ADAPTER_OBJECT>.
This is the <t MiniportAdapterContext> we passed into <f NdisMSetAttributes>.
@parm IN PNDIS_TAPI_CLOSE | Request |
A pointer to the NDIS_TAPI request structure for this call.
@iex
typedef struct _NDIS_TAPI_CLOSE
{
IN ULONG ulRequestID;
IN HDRV_LINE hdLine;
} NDIS_TAPI_CLOSE, *PNDIS_TAPI_CLOSE;
@rdesc This routine returns one of the following values:
@flag NDIS_STATUS_SUCCESS |
If this function is successful.
<f Note>: A non-zero return value indicates one of the following error codes:
@iex
NDIS_STATUS_PENDING
NDIS_STATUS_TAPI_INVALLINEHANDLE
*/
NDIS_STATUS TspiClose(
IN PMINIPORT_ADAPTER_OBJECT pAdapter,
IN PNDIS_TAPI_CLOSE Request,
OUT PULONG BytesRead,
OUT PULONG BytesNeeded
)
{
DBG_FUNC("TspiClose")
PBCHANNEL_OBJECT pBChannel;
// A Pointer to one of our <t BCHANNEL_OBJECT>'s.
NDIS_STATUS Result;
// Holds the result code returned by this function.
DBG_ENTER(pAdapter);
DBG_PARAMS(pAdapter,
("\n\thdLine=0x%X\n",
Request->hdLine
));
/*
// This request must be associated with a line device.
// And it must not be called until all calls are closed or idle.
*/
pBChannel = GET_BCHANNEL_FROM_HDLINE(pAdapter, Request->hdLine);
if (pBChannel == NULL ||
(pBChannel->DevState & LINEDEVSTATE_OPEN) == 0)
{
DBG_WARNING(pAdapter, ("Returning NDIS_STATUS_TAPI_INVALLINEHANDLE\n"));
return (NDIS_STATUS_TAPI_INVALLINEHANDLE);
}
/*
// Close the TAPI line device and release the channel.
*/
BChannelClose(pBChannel);
TspiLineDevStateHandler(pAdapter, pBChannel, LINEDEVSTATE_CLOSE);
Result = NDIS_STATUS_SUCCESS;
DBG_RETURN(pAdapter, Result);
return (Result);
}
/* @doc INTERNAL TspiLine TspiLine_c TspiGetLineDevStatus
ÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛ
@func
This request queries the specified open line device for its current status.
The information returned is global to all addresses on the line.
@parm IN PMINIPORT_ADAPTER_OBJECT | pAdapter |
A pointer to the Miniport's adapter context structure <t MINIPORT_ADAPTER_OBJECT>.
This is the <t MiniportAdapterContext> we passed into <f NdisMSetAttributes>.
@parm IN PNDIS_TAPI_GET_LINE_DEV_STATUS | Request |
A pointer to the NDIS_TAPI request structure for this call.
@iex
typedef struct _NDIS_TAPI_GET_LINE_DEV_STATUS
{
IN ULONG ulRequestID;
IN HDRV_LINE hdLine;
OUT LINE_DEV_STATUS LineDevStatus;
} NDIS_TAPI_GET_LINE_DEV_STATUS, *PNDIS_TAPI_GET_LINE_DEV_STATUS;
typedef struct _LINE_DEV_STATUS
{
ULONG ulTotalSize;
ULONG ulNeededSize;
ULONG ulUsedSize;
ULONG ulNumOpens;
ULONG ulOpenMediaModes;
ULONG ulNumActiveCalls;
ULONG ulNumOnHoldCalls;
ULONG ulNumOnHoldPendCalls;
ULONG ulLineFeatures;
ULONG ulNumCallCompletions;
ULONG ulRingMode;
ULONG ulSignalLevel;
ULONG ulBatteryLevel;
ULONG ulRoamMode;
ULONG ulDevStatusFlags;
ULONG ulTerminalModesSize;
ULONG ulTerminalModesOffset;
ULONG ulDevSpecificSize;
ULONG ulDevSpecificOffset;
} LINE_DEV_STATUS, *PLINE_DEV_STATUS;
@rdesc This routine returns one of the following values:
@flag NDIS_STATUS_SUCCESS |
If this function is successful.
<f Note>: A non-zero return value indicates one of the following error codes:
@iex
NDIS_STATUS_TAPI_INVALLINEHANDLE
*/
NDIS_STATUS TspiGetLineDevStatus(
IN PMINIPORT_ADAPTER_OBJECT pAdapter,
IN PNDIS_TAPI_GET_LINE_DEV_STATUS Request,
OUT PULONG BytesWritten,
OUT PULONG BytesNeeded
)
{
DBG_FUNC("TspiGetLineDevStatus")
PBCHANNEL_OBJECT pBChannel;
// A Pointer to one of our <t BCHANNEL_OBJECT>'s.
DBG_ENTER(pAdapter);
DBG_PARAMS(pAdapter,
("\n\thdLine=0x%X\n",
Request->hdLine
));
/*
// This request must be associated with a line device.
*/
pBChannel = GET_BCHANNEL_FROM_HDLINE(pAdapter, Request->hdLine);
if (pBChannel == NULL)
{
DBG_WARNING(pAdapter, ("Returning NDIS_STATUS_TAPI_INVALLINEHANDLE\n"));
return (NDIS_STATUS_TAPI_INVALLINEHANDLE);
}
Request->LineDevStatus.ulNeededSize =
Request->LineDevStatus.ulUsedSize = sizeof(Request->LineDevStatus);
if (Request->LineDevStatus.ulNeededSize > Request->LineDevStatus.ulTotalSize)
{
DBG_PARAMS(pAdapter,
("STRUCTURETOOSMALL %d<%d\n",
Request->LineDevStatus.ulTotalSize,
Request->LineDevStatus.ulNeededSize));
}
/*
// Return the current line status information.
*/
Request->LineDevStatus.ulNumOpens = 1;
Request->LineDevStatus.ulNumActiveCalls =
pBChannel->CallState <= LINECALLSTATE_IDLE ? 0 : 1;
Request->LineDevStatus.ulLineFeatures =
pBChannel->CallState <= LINECALLSTATE_IDLE ?
LINEFEATURE_MAKECALL : 0;
Request->LineDevStatus.ulRingMode =
pBChannel->CallState == LINECALLSTATE_OFFERING ? 1: 0;
Request->LineDevStatus.ulDevStatusFlags =
(pBChannel->DevState & LINEDEVSTATE_CONNECTED) ?
LINEDEVSTATUSFLAGS_CONNECTED : 0;
Request->LineDevStatus.ulDevStatusFlags |=
(pBChannel->DevState & LINEDEVSTATE_INSERVICE) ?
LINEDEVSTATUSFLAGS_INSERVICE : 0;
DBG_RETURN(pAdapter, NDIS_STATUS_SUCCESS);
return (NDIS_STATUS_SUCCESS);
}
/* @doc INTERNAL TspiLine TspiLine_c TspiSetDefaultMediaDetection
ÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛ
@func
This request informs the miniport of the new set of media modes to detect
for the indicated line (replacing any previous set).
@parm IN PMINIPORT_ADAPTER_OBJECT | pAdapter |
A pointer to the Miniport's adapter context structure <t MINIPORT_ADAPTER_OBJECT>.
This is the <t MiniportAdapterContext> we passed into <f NdisMSetAttributes>.
@parm IN PNDIS_TAPI_SET_DEFAULT_MEDIA_DETECTION | Request |
A pointer to the NDIS_TAPI request structure for this call.
@iex
typedef struct _NDIS_TAPI_SET_DEFAULT_MEDIA_DETECTION
{
IN ULONG ulRequestID;
IN HDRV_LINE hdLine;
IN ULONG ulMediaModes;
} NDIS_TAPI_SET_DEFAULT_MEDIA_DETECTION, *PNDIS_TAPI_SET_DEFAULT_MEDIA_DETECTION;
@rdesc This routine returns one of the following values:
@flag NDIS_STATUS_SUCCESS |
If this function is successful.
<f Note>: A non-zero return value indicates one of the following error codes:
@iex
NDIS_STATUS_TAPI_INVALLINEHANDLE
@comm
<f Note>:
After a miniport NIC driver has received an OPEN request for a line, it
may also receive one or more SET_DEFAULT_MEDIA_DETECTION requests. This
latter request informs the NIC driver of the type(s) of incoming calls,
with respect to media mode, it should indicate to the Connection Wrapper
with the LINE_NEWCALL message. If an incoming call appears with a media
mode type not specified in the last (successfully completed)
SET_DEFAULT_MEDIA_DETECTION request for that line, the miniport should
not indicate the new call to the Connection Wrapper. If a miniport does
not receive a SET_DEFAULT_MEDIA_DETECTION request for a line, it should
not indicate any incoming calls to the Connection Wrapper; that line is
to be used only for outbound calls.
*/
NDIS_STATUS TspiSetDefaultMediaDetection(
IN PMINIPORT_ADAPTER_OBJECT pAdapter,
IN PNDIS_TAPI_SET_DEFAULT_MEDIA_DETECTION Request,
OUT PULONG BytesRead,
OUT PULONG BytesNeeded
)
{
DBG_FUNC("TspiSetDefaultMediaDetection")
PBCHANNEL_OBJECT pBChannel;
// A Pointer to one of our <t BCHANNEL_OBJECT>'s.
DBG_ENTER(pAdapter);
DBG_PARAMS(pAdapter,
("\n\thdLine=0x%X\n"
"\tulMediaModes=0x%X\n",
Request->hdLine,
Request->ulMediaModes
));
/*
// This request must be associated with a line device.
*/
pBChannel = GET_BCHANNEL_FROM_HDLINE(pAdapter, Request->hdLine);
if (pBChannel == NULL)
{
DBG_WARNING(pAdapter, ("Returning NDIS_STATUS_TAPI_INVALLINEHANDLE\n"));
return (NDIS_STATUS_TAPI_INVALLINEHANDLE);
}
/*
// Don't accept the request for media modes we don't support.
*/
if (Request->ulMediaModes & ~pBChannel->MediaModesCaps)
{
DBG_WARNING(pAdapter, ("Returning NDIS_STATUS_TAPI_INVALMEDIAMODE\n"));
return (NDIS_STATUS_TAPI_INVALMEDIAMODE);
}
/*
// Set the media modes mask and make sure the adapter is ready to
// accept incoming calls. If you can detect different medias, you
// will need to notify the approriate interface for the media detected.
*/
pBChannel->MediaModesMask = Request->ulMediaModes & pBChannel->MediaModesCaps;
DBG_RETURN(pAdapter, NDIS_STATUS_SUCCESS);
return (NDIS_STATUS_SUCCESS);
}
/* @doc INTERNAL TspiLine TspiLine_c XXX
@func
This request is invoked by the Connection Wrapper whenever a client
application uses LINEMAPPER as the dwDeviceID in the lineOpen function
to request that lines be scanned to find one that supports the desired
media mode(s) and call parameters. The Connection Wrapper scans based on
the union of the desired media modes and the other media modes currently
being monitored on the line, to give the miniport the opportunity to
indicate if it cannot simultaneously monitor for all of the requested
media modes. If the miniport can monitor for the indicated set of media
modes AND support the capabilities indicated in CallParams, it replies
with a “success” inidication. It leaves the active media monitoring modes
for the line unchanged.
@parm IN PMINIPORT_ADAPTER_OBJECT | pAdapter |
A pointer to the Miniport's adapter context structure <t MINIPORT_ADAPTER_OBJECT>.
This is the <t MiniportAdapterContext> we passed into <f NdisMSetAttributes>.
@parm IN PNDIS_TAPI_CONDITIONAL_MEDIA_DETECTION | Request |
A pointer to the NDIS_TAPI request structure for this call.
@iex
typedef struct _NDIS_TAPI_CONDITIONAL_MEDIA_DETECTION
{
IN ULONG ulRequestID;
IN HDRV_LINE hdLine;
IN ULONG ulMediaModes;
IN LINE_CALL_PARAMS LineCallParams;
} NDIS_TAPI_CONDITIONAL_MEDIA_DETECTION, *PNDIS_TAPI_CONDITIONAL_MEDIA_DETECTION;
typedef struct _LINE_CALL_PARAMS // Defaults:
{
ULONG ulTotalSize; // ---------
ULONG ulBearerMode; // voice
ULONG ulMinRate; // (3.1kHz)
ULONG ulMaxRate; // (3.1kHz)
ULONG ulMediaMode; // interactiveVoice
ULONG ulCallParamFlags; // 0
ULONG ulAddressMode; // addressID
ULONG ulAddressID; // (any available)
LINE_DIAL_PARAMS DialParams; // (0, 0, 0, 0)
ULONG ulOrigAddressSize; // 0
ULONG ulOrigAddressOffset;
ULONG ulDisplayableAddressSize;
ULONG ulDisplayableAddressOffset;
ULONG ulCalledPartySize; // 0
ULONG ulCalledPartyOffset;
ULONG ulCommentSize; // 0
ULONG ulCommentOffset;
ULONG ulUserUserInfoSize; // 0
ULONG ulUserUserInfoOffset;
ULONG ulHighLevelCompSize; // 0
ULONG ulHighLevelCompOffset;
ULONG ulLowLevelCompSize; // 0
ULONG ulLowLevelCompOffset;
ULONG ulDevSpecificSize; // 0
ULONG ulDevSpecificOffset;
} LINE_CALL_PARAMS, *PLINE_CALL_PARAMS;
typedef struct _LINE_DIAL_PARAMS
{
ULONG ulDialPause;
ULONG ulDialSpeed;
ULONG ulDigitDuration;
ULONG ulWaitForDialtone;
} LINE_DIAL_PARAMS, *PLINE_DIAL_PARAMS;
@rdesc This routine returns one of the following values:
@flag NDIS_STATUS_SUCCESS |
If this function is successful.
<f Note>: A non-zero return value indicates one of the following error codes:
@iex
NDIS_STATUS_TAPI_INVALLINEHANDLE
*/
NDIS_STATUS TspiConditionalMediaDetection(
IN PMINIPORT_ADAPTER_OBJECT pAdapter,
IN PNDIS_TAPI_CONDITIONAL_MEDIA_DETECTION Request,
OUT PULONG BytesRead,
OUT PULONG BytesNeeded
)
{
DBG_FUNC("TspiConditionalMediaDetection")
PBCHANNEL_OBJECT pBChannel;
// A Pointer to one of our <t BCHANNEL_OBJECT>'s.
DBG_ENTER(pAdapter);
DBG_PARAMS(pAdapter,
("\n\thdLine=0x%X\n"
"\tulMediaModes=0x%X\n"
"\tLineCallParams=0x%X\n",
Request->hdLine,
Request->ulMediaModes,
&Request->LineCallParams
));
/*
// This request must be associated with a line device.
*/
pBChannel = GET_BCHANNEL_FROM_HDLINE(pAdapter, Request->hdLine);
if (pBChannel == NULL)
{
DBG_WARNING(pAdapter, ("Returning NDIS_STATUS_TAPI_INVALLINEHANDLE\n"));
return (NDIS_STATUS_TAPI_INVALLINEHANDLE);
}
/*
// We don't expect user user info.
*/
ASSERT(Request->LineCallParams.ulUserUserInfoSize == 0);
/*
// Don't accept the request for media modes we don't support.
*/
if (Request->ulMediaModes & ~pBChannel->MediaModesCaps)
{
DBG_WARNING(pAdapter, ("Returning NDIS_STATUS_TAPI_INVALMEDIAMODE\n"));
return (NDIS_STATUS_TAPI_INVALMEDIAMODE);
}
DBG_RETURN(pAdapter, NDIS_STATUS_SUCCESS);
return (NDIS_STATUS_SUCCESS);
}
/* @doc INTERNAL TspiLine TspiLine_c TspiSetStatusMessages
ÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛ
@func
This request enables the Connection Wrapper to specify which notification
messages the miniport should generate for events related to status changes
for the specified line or any of its addresses. By default, address and
line status reporting is initially disabled for a line.
@parm IN PMINIPORT_ADAPTER_OBJECT | pAdapter |
A pointer to the Miniport's adapter context structure <t MINIPORT_ADAPTER_OBJECT>.
This is the <t MiniportAdapterContext> we passed into <f NdisMSetAttributes>.
@parm IN PNDIS_TAPI_SET_STATUS_MESSAGES | Request |
A pointer to the NDIS_TAPI request structure for this call.
@iex
typedef struct _NDIS_TAPI_SET_STATUS_MESSAGES
{
IN ULONG ulRequestID;
IN HDRV_LINE hdLine;
IN ULONG ulLineStates;
IN ULONG ulAddressStates;
} NDIS_TAPI_SET_STATUS_MESSAGES, *PNDIS_TAPI_SET_STATUS_MESSAGES;
@rdesc This routine returns one of the following values:
@flag NDIS_STATUS_SUCCESS |
This function always returns success.
*/
NDIS_STATUS TspiSetStatusMessages(
IN PMINIPORT_ADAPTER_OBJECT pAdapter,
IN PNDIS_TAPI_SET_STATUS_MESSAGES Request,
OUT PULONG BytesRead,
OUT PULONG BytesNeeded
)
{
DBG_FUNC("TspiSetStatusMessages")
NDIS_STATUS Result = NDIS_STATUS_SUCCESS;
// Holds the result code returned by this function.
PBCHANNEL_OBJECT pBChannel;
// A Pointer to one of our <t BCHANNEL_OBJECT>'s.
DBG_ENTER(pAdapter);
DBG_PARAMS(pAdapter,
("\n\thdLine=0x%X\n"
"\tulLineStates=0x%X\n"
"\tulAddressStates=0x%X\n",
Request->hdLine,
Request->ulLineStates,
Request->ulAddressStates
));
/*
// This request must be associated with a line device.
*/
pBChannel = GET_BCHANNEL_FROM_HDLINE(pAdapter, Request->hdLine);
if (pBChannel == NULL)
{
DBG_WARNING(pAdapter, ("Returning NDIS_STATUS_TAPI_INVALLINEHANDLE\n"));
return (NDIS_STATUS_TAPI_INVALLINEHANDLE);
}
/*
// TAPI may pass down more than we are capable of handling.
// We have to accept the request, but can ignore the extras.
*/
if (Request->ulLineStates & ~pBChannel->DevStatesCaps)
{
DBG_WARNING(pAdapter, ("ulLineStates=0x%X !< DevStatesCaps=0x%X\n",
Request->ulLineStates, pBChannel->DevStatesCaps));
Result = NDIS_STATUS_TAPI_INVALPARAM;
}
/*
// TAPI may pass down more than we are capable of handling.
// We have to accept the request, but can ignore the extras.
*/
if (Request->ulAddressStates & ~pBChannel->AddressStatesCaps)
{
DBG_WARNING(pAdapter, ("ulAddressStates=0x%X !< AddressStatesCaps=0x%X\n",
Request->ulAddressStates, pBChannel->AddressStatesCaps));
Result = NDIS_STATUS_TAPI_INVALPARAM;
}
/*
// Save the new event notification masks so we will only indicate the
// appropriate events.
*/
pBChannel->DevStatesMask = Request->ulLineStates & pBChannel->DevStatesCaps;
pBChannel->AddressStatesMask = Request->ulAddressStates & pBChannel->AddressStatesCaps;
DBG_RETURN(pAdapter, Result);
return (Result);
}
/* @doc INTERNAL TspiLine TspiLine_c TspiLineDevStateHandler
ÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛ
@func
<f TspiLineDevStateHandler> will indicate the given LINEDEVSTATE to the
Connection Wrapper if the event has been enabled by the wrapper.
Otherwise the state information is saved, but no indication is made.
*/
VOID TspiLineDevStateHandler(
IN PMINIPORT_ADAPTER_OBJECT pAdapter, // @parm
// A pointer to the <t MINIPORT_ADAPTER_OBJECT> instance.
IN PBCHANNEL_OBJECT pBChannel, // @parm
// A pointer to the <t BCHANNEL_OBJECT> returned by <f BChannelCreate>.
IN ULONG LineDevState // @parm
// The <t LINEDEVSTATE> event to be posted to TAPI/WAN.
)
{
DBG_FUNC("TspiLineDevStateHandler")
NDIS_TAPI_EVENT LineEvent;
NDIS_TAPI_EVENT CallEvent;
// The event structure passed to the Connection Wrapper.
ULONG NewCallState = 0;
ULONG StateParam = 0;
// The line state change may cause a call state change as well.
DBG_ENTER(pAdapter);
DBG_PARAMS(pAdapter,
("#%d OldState=0x%X "
"NewState=0x%X\n",
pBChannel->BChannelIndex,
pBChannel->DevState,
LineDevState
));
LineEvent.ulParam2 = 0;
LineEvent.ulParam3 = 0;
/*
// Handle the line state transition as needed.
*/
switch (LineDevState)
{
case LINEDEVSTATE_RINGING:
/*
// We have an incoming call, see if there's anyone who cares.
*/
if (pBChannel->CallState == 0 &&
pBChannel->MediaModesMask)
{
LineEvent.ulParam2 = 1; // only one RingMode
NewCallState = LINECALLSTATE_OFFERING;
}
else
{
DChannelRejectCall(pAdapter->pDChannel, pBChannel);
}
break;
case LINEDEVSTATE_CONNECTED:
/*
// The line has been connected, but we may already know this.
*/
if ((pBChannel->DevState & LINEDEVSTATE_CONNECTED) == 0)
{
pBChannel->DevState |= LINEDEVSTATE_CONNECTED;
}
else
{
LineDevState = 0;
}
break;
case LINEDEVSTATE_DISCONNECTED:
/*
// The line has been dis-connected, but we may already know this.
// If not, this will effect any calls on the line.
*/
if ((pBChannel->DevState & LINEDEVSTATE_CONNECTED) != 0)
{
pBChannel->DevState &= ~(LINEDEVSTATE_CONNECTED |
LINEDEVSTATE_INSERVICE);
NewCallState = LINECALLSTATE_DISCONNECTED;
StateParam = LINEDISCONNECTMODE_NORMAL;
}
else
{
LineDevState = 0;
}
break;
case LINEDEVSTATE_INSERVICE:
/*
// The line has been placed in service, but we may already know this.
*/
if ((pBChannel->DevState & LINEDEVSTATE_INSERVICE) == 0)
{
pBChannel->DevState |= LINEDEVSTATE_INSERVICE;
}
else
{
LineDevState = 0;
}
break;
case LINEDEVSTATE_OUTOFSERVICE:
/*
// The line has been taken out of service, but we may already know this.
// If not, this will effect any calls on the line.
*/
if ((pBChannel->DevState & LINEDEVSTATE_INSERVICE) != 0)
{
pBChannel->DevState &= ~LINEDEVSTATE_INSERVICE;
NewCallState = LINECALLSTATE_DISCONNECTED;
StateParam = LINEDISCONNECTMODE_UNKNOWN;
}
else
{
LineDevState = 0;
}
break;
case LINEDEVSTATE_OPEN:
pBChannel->DevState |= LINEDEVSTATE_OPEN;
pAdapter->NumLineOpens++;
break;
case LINEDEVSTATE_CLOSE:
pBChannel->DevState &= ~LINEDEVSTATE_OPEN;
pAdapter->NumLineOpens--;
break;
}
/*
// If this is the first indication of an incoming call, we need to
// let TAPI know about it so we can get a htCall handle associated
// with it.
*/
if (pBChannel->DevState & LINEDEVSTATE_OPEN)
{
if (NewCallState == LINECALLSTATE_OFFERING)
{
CallEvent.htLine = pBChannel->htLine;
CallEvent.htCall = (HTAPI_CALL)0;
CallEvent.ulMsg = LINE_NEWCALL;
CallEvent.ulParam1 = (ULONG) (ULONG_PTR) pBChannel;
CallEvent.ulParam2 = 0;
CallEvent.ulParam3 = 0;
NdisMIndicateStatus(
pAdapter->MiniportAdapterHandle,
NDIS_STATUS_TAPI_INDICATION,
&CallEvent,
sizeof(CallEvent)
);
pAdapter->NeedStatusCompleteIndication = TRUE;
pBChannel->htCall = (HTAPI_CALL)CallEvent.ulParam2;
DBG_FILTER(pAdapter, DBG_TAPICALL_ON,
("#%d Call=0x%X CallState=0x%X NEW_CALL\n",
pBChannel->BChannelIndex,
pBChannel->htCall, pBChannel->CallState));
if (pBChannel->htCall == 0)
{
/*
// TAPI won't accept the call, so toss it.
*/
NewCallState = 0;
LineDevState = 0;
}
}
/*
// Only send those line messages TAPI wants to hear about.
*/
if (pBChannel->DevStatesMask & LineDevState)
{
LineEvent.htLine = pBChannel->htLine;
LineEvent.htCall = pBChannel->htCall;
LineEvent.ulMsg = LINE_LINEDEVSTATE;
LineEvent.ulParam1 = LineDevState;
NdisMIndicateStatus(
pAdapter->MiniportAdapterHandle,
NDIS_STATUS_TAPI_INDICATION,
&LineEvent,
sizeof(LineEvent)
);
pAdapter->NeedStatusCompleteIndication = TRUE;
DBG_FILTER(pAdapter, DBG_TAPICALL_ON,
("#%d Line=0x%X LineState=0x%X\n",
pBChannel->BChannelIndex,
pBChannel->htLine, LineDevState));
}
else
{
DBG_NOTICE(pAdapter, ("#%d LINEDEVSTATE=0x%X EVENT NOT ENABLED\n",
pBChannel->BChannelIndex, LineDevState));
}
if (NewCallState != 0)
{
/*
// Check to see if we need to disconnect the call, but only
// if there is one active.
*/
if (NewCallState == LINECALLSTATE_DISCONNECTED)
{
if (pBChannel->CallState != 0 &&
pBChannel->CallState != LINECALLSTATE_IDLE &&
pBChannel->CallState != LINECALLSTATE_DISCONNECTED)
{
TspiCallStateHandler(pAdapter, pBChannel,
NewCallState, StateParam);
#if defined(NDIS40_MINIPORT)
/*
// NDISWAN_BUG
// Under some conditions, NDISWAN does not do a CLOSE_CALL,
// so the line would be left unusable if we don't timeout
// and force a close call condition.
*/
NdisMSetTimer(&pBChannel->CallTimer, CARD_NO_CLOSECALL_TIMEOUT);
#endif // NDIS50_MINIPORT
}
}
else
{
TspiCallStateHandler(pAdapter, pBChannel,
NewCallState, StateParam);
if (NewCallState == LINECALLSTATE_OFFERING)
{
/*
// If an offered call is not accepted within N seconds, we
// need to force the line back to an idle state.
*/
NdisMSetTimer(&pBChannel->CallTimer, pAdapter->NoAcceptTimeOut);
}
}
}
}
DBG_LEAVE(pAdapter);
}