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.
 
 
 
 
 
 

873 lines
31 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 Request Request_c
@module Request.c |
This module implements the NDIS request routines for the Miniport.
@head3 Contents |
@index class,mfunc,func,msg,mdata,struct,enum | Request_c
@end
ÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛ
*/
/* @doc EXTERNAL INTERNAL
ÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛ
@topic 3.2 Query/Set Request Processing |
For querying and setting network interface card binding information, the
NDIS library calls <f MiniportQueryInformation> or <f MiniportSetInformation>.
The upper layers place an object identifier (OID) in the structure for
an object in the miniport NIC driver MIB that it wants to query or set.
The <f MiniportQueryInformation> function fills in results and returns an
appropriate status code to the NDIS library. See Part I of the Network
Driver Reference for more information on OIDs.
These two functions are potentially asynchronous. If they behave
synchronously, they return immediately with a status code other than
NDIS_STATUS_PENDING. If asynchronous, the function returns
NDIS_STATUS_PENDING; and the miniport NIC driver later completes the
request operation by a call to NdisMQueryInformationComplete for the
query function or NdisMSetInformationComplete for the set function.
The NDIS library guarantees that the miniport NIC driver will have only
one outstanding request at a time so there is no need for the miniport
NIC driver to queue requests.
@end
*/
#define __FILEID__ REQUEST_OBJECT_TYPE
// Unique file ID for error logging
#include "Miniport.h" // Defines all the miniport objects
#if defined(NDIS_LCODE)
# pragma NDIS_LCODE // Windows 95 wants this code locked down!
# pragma NDIS_LDATA
#endif
/*
// The following is a list of all the possible NDIS QuereyInformation requests
// that might be directed to the miniport.
// Comment out any that are not supported by this driver.
*/
static const NDIS_OID g_SupportedOidArray[] =
{
OID_GEN_SUPPORTED_LIST,
OID_GEN_HARDWARE_STATUS,
OID_GEN_MEDIA_SUPPORTED,
OID_GEN_MEDIA_IN_USE,
OID_GEN_MAXIMUM_LOOKAHEAD,
OID_GEN_MAC_OPTIONS,
OID_GEN_VENDOR_ID,
OID_GEN_VENDOR_DESCRIPTION,
OID_GEN_DRIVER_VERSION,
OID_GEN_CURRENT_LOOKAHEAD,
OID_WAN_PERMANENT_ADDRESS,
OID_WAN_CURRENT_ADDRESS,
OID_WAN_MEDIUM_SUBTYPE,
OID_WAN_GET_INFO,
OID_WAN_SET_LINK_INFO,
OID_WAN_GET_LINK_INFO,
#if defined(NDIS50_MINIPORT)
OID_WAN_LINE_COUNT,
OID_PNP_CAPABILITIES,
OID_PNP_SET_POWER,
OID_PNP_QUERY_POWER,
#endif // NDIS50_MINIPORT
0
};
#if DBG
/*
// Make sure the following list is in the same order as the list above!
*/
static char *g_SupportedOidNames[] =
{
"OID_GEN_SUPPORTED_LIST",
"OID_GEN_HARDWARE_STATUS",
"OID_GEN_MEDIA_SUPPORTED",
"OID_GEN_MEDIA_IN_USE",
"OID_GEN_MAXIMUM_LOOKAHEAD",
"OID_GEN_MAC_OPTIONS",
"OID_GEN_VENDOR_ID",
"OID_GEN_VENDOR_DESCRIPTION",
"OID_GEN_DRIVER_VERSION",
"OID_GEN_CURRENT_LOOKAHEAD",
"OID_WAN_PERMANENT_ADDRESS",
"OID_WAN_CURRENT_ADDRESS",
"OID_WAN_MEDIUM_SUBTYPE",
"OID_WAN_GET_INFO",
"OID_WAN_SET_LINK_INFO",
"OID_WAN_GET_LINK_INFO",
#if defined(NDIS50_MINIPORT)
"OID_WAN_LINE_COUNT",
"OID_PNP_CAPABILITIES",
"OID_PNP_SET_POWER",
"OID_PNP_QUERY_POWER",
#endif // NDIS50_MINIPORT
"OID_UNKNOWN"
};
#define NUM_OID_ENTRIES (sizeof(g_SupportedOidArray) / sizeof(g_SupportedOidArray[0]))
/*
// This debug routine will lookup the printable name for the selected OID.
*/
static char * DbgGetOidString(NDIS_OID Oid)
{
UINT i;
for (i = 0; i < NUM_OID_ENTRIES-1; i++)
{
if (g_SupportedOidArray[i] == Oid)
{
break;
}
}
return(g_SupportedOidNames[i]);
}
#endif // DBG
/*
// Returned from an OID_WAN_PERMANENT_ADDRESS MiniportQueryInformation request.
// The WAN wrapper wants the miniport to return a unique address for this
// adapter. This is used as an ethernet address presented to the protocols.
// The least significant bit of the first byte must not be a 1, or it could
// be interpreted as an ethernet multicast address. If the vendor has an
// assigned ethernet vendor code (the first 3 bytes), they should be used
// to assure that the address does not conflict with another vendor's address.
// The last digit is replaced during the call with the adapter instance number.
*/
static UCHAR g_PermanentWanAddress[6] = VER_VENDOR_ID;
/*
// Returned from an OID_GEN_VENDOR_ID MiniportQueryInformation request.
// Again, the vendor's assigned ethernet vendor code should be used if possible.
*/
static UCHAR g_Vendor3ByteID[4] = VER_VENDOR_ID;
/*
// Returned from an OID_GEN_VENDOR_DESCRIPTION MiniportQueryInformation request.
// This is an arbitrary string which may be used by upper layers to present
// a user friendly description of the adapter.
*/
static NDIS_STRING g_VendorDescriptionString = INIT_STRING_CONST(VER_PRODUCT_NAME_STR);
/* @doc INTERNAL Request Request_c MiniportQueryInformation
ÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛ
@func
<f MiniportQueryInformation> is a required function that
returns information about the capabilities and status of
the driver and/or its NIC.
@comm
NDIS calls the <f MiniportQueryInformation> function either
on its own behalf, such as to determine which options the
driver supports or to manage binding-specific information
for the miniport, or when a bound protocol driver calls
<f NdisRequest>.
NDIS makes one or more calls to <f MiniportQueryInformation>
just after a driver's <f MiniportInitialize> function returns
NDIS_STATUS_SUCCESS. NDIS supplies the following OIDs in
its initialization-time calls to the driver's
<f MiniportQueryInformation> function:
<f OID_GEN_MAXIMUM_LOOKAHEAD><nl>
<f MiniportQueryInformation> must return how many bytes of lookahead
data the NIC can provide, that is, the initial transfer capacity
of the NIC.<nl>
Even if a driver supports multipacket receives and, therefore,
will indicate an array of pointers to fully set up packets,
MiniportQueryInformation must supply this information. Such a
driver should return the maximum packet size it can indicate.
<f OID_GEN_MAC_OPTIONS><nl>
<f MiniportQueryInformation> must return a bitmask set with the
appropriate NDIS_MAC_OPTION_XXX flags indicating which options
it (or its NIC) supports, or it can return zero at InformationBuffer
if the driver supports none of the options designated by these flags.
For example, a NIC driver always sets the
NDIS_MAC_OPTION_NO_LOOPBACK flag if its NIC has no
internal hardware support for loopback. This tells
NDIS to manage loopback for the driver, which cannot
provide software loopback code as efficient as the NDIS
library's because NDIS manages all binding-specific
information for miniports. Any miniport that tries to provide
software loopback must check the destination address of every
send packet against the currently set filter addresses to
determine whether to loop back each packet. WAN NIC drivers
must set this flag.
If the NIC driver sets the NDIS_MAC_OPTION_FULL_DUPLEX flag,
the NDIS library serializes calls to the MiniportSendPackets
or <f MiniportWanSend> function separately from its serialized
calls to other MiniportXxx functions in SMP machines. However,
NDIS returns incoming send packets to protocols while such a
driver's <f MiniportReset> function is executing: that is, NDIS
never calls a full-duplex miniport to transmit a packet until
its reset operation is completed. The designer of any full-duplex
driver can expect that driver to achieve significantly higher
performance in SMP machines, but the driver must synchronize
its accesses to shared resources carefully to prevent race
conditions or deadlocks from occurring. NDIS assumes that
all intermediate drivers are full-duplex drivers.
Depending on the NdisMediumXxx that <f MiniportInitialize> selected,
NDIS submits additional intialization-time requests to
<f MiniportQueryInformation>, such as the following:
<f OID_XXX_CURRENT_ADDRESS><nl>
If the driver's <f MiniportInitialize> function selected an NdisMediumXxx
for which the system supplies a filter, NDIS calls
<f MiniportQueryInformation> to return the NIC's current
address in medium-specific format. For FDDI drivers, NDIS
requests both long and short current addresses.
<f OID_802_3_MAXIMUM_LIST_SIZE><nl>
For Ethernet drivers, NDIS requests the multicast list size.
<f OID_FDDI_LONG>/<f SHORT_MAX_LIST_SIZE><nl>
For FDDI drivers, NDIS requests the multicast list sizes.
If possible, <f MiniportQueryInformation> should not return
<f NDIS_STATUS_PENDING> for initialization-time requests.
Until NDIS has sufficient information to set up bindings
to the miniport, such requests should be handled synchronously.
If the Miniport does not complete the call immediately (by returning
<f NDIS_STATUS_PENDING>), it must call NdisMQueryInformationComplete to
complete the call. The Miniport controls the buffers pointed to by
InformationBuffer, BytesWritten, and BytesNeeded until the request
completes.
No other requests will be submitted to the Miniport until
this request has been completed.
<f Note>: that the wrapper will intercept all queries of the following OIDs:
OID_GEN_CURRENT_PACKET_FILTER,
OID_GEN_PROTOCOL_OPTIONS,
OID_802_5_CURRENT_FUNCTIONAL,
OID_802_3_MULTICAST_LIST,
OID_FDDI_LONG_MULTICAST_LIST,
OID_FDDI_SHORT_MULTICAST_LIST.
<f Note>: Interrupts will be in any state when called.
@rdesc
<f MiniportQueryInformation> can return one of the following:
@flag NDIS_STATUS_SUCCESS |
<f MiniportQueryInformation> returned the requested information at
InformationBuffer and set the variable at BytesWritten to the amount
of information it returned.
@flag NDIS_STATUS_PENDING |
The driver will complete the request asynchronously with a call to
NdisMQueryInformationComplete when it has gathered the requested
information.
@flag NDIS_STATUS_INVALID_OID |
<f MiniportQueryInformation> does not recognize the Oid.
@flag NDIS_STATUS_INVALID_LENGTH |
The InformationBufferLength does not match the length required
by the given Oid. <f MiniportQueryInformation> returned how many
bytes the buffer should be at BytesNeeded.
@flag NDIS_STATUS_NOT_ACCEPTED |
<f MiniportQueryInformation> attempted to gather the requested
information from the NIC but was unsuccessful.
@flag NDIS_STATUS_NOT_SUPPORTED |
<f MiniportQueryInformation> does not support the Oid, which
is optional.
@flag NDIS_STATUS_RESOURCES |
<f MiniportQueryInformation> could not allocate sufficient
resources to return the requested information. This return
value does not necessarily mean that the same request,
submitted at a later time, will be failed for the same
reason.
@xref
<f MiniportInitialize>
<f MiniportSetInformation>
*/
NDIS_STATUS MiniportQueryInformation(
IN PMINIPORT_ADAPTER_OBJECT pAdapter, // @parm
// A pointer to the <t MINIPORT_ADAPTER_OBJECT> instance.
IN NDIS_OID Oid, // @parm
// The OID. (See section 7.4 of the NDIS 3.0 specification for a complete
// description of OIDs.)
IN PVOID InformationBuffer, // @parm
// The buffer that will receive the information. (See section 7.4 of the
// NDIS 3.0 specification for a description of the length required for
// each OID.)
IN ULONG InformationBufferLength, // @parm
// The length in bytes of InformationBuffer.
OUT PULONG BytesWritten, // @parm
// Returns the number of bytes written into InformationBuffer.
OUT PULONG BytesNeeded // @parm
// Returns the number of additional bytes needed to satisfy the OID.
)
{
DBG_FUNC("MiniportQueryInformation")
NDIS_STATUS Status = NDIS_STATUS_SUCCESS;
// Holds the status result returned by this function.
PVOID SourceBuffer;
// Pointer to driver data to be copied back to caller's InformationBuffer
ULONG SourceBufferLength;
// Number of bytes to be copied from driver.
ULONG GenericULong = 0;
// Most return values are long integers, so this is used to hold the
// return value of a constant or computed result.
UCHAR VendorId[4];
// Used to store vendor ID string.
#if defined(NDIS50_MINIPORT)
NDIS_PNP_CAPABILITIES PnpCapabilities;
// Used to return our PNP capabilities.
#endif // NDIS50_MINIPORT
/*
// If this is a TAPI OID, pass it on over.
*/
if ((Oid & 0xFFFFFF00L) == (OID_TAPI_ACCEPT & 0xFFFFFF00L))
{
Status = TspiRequestHandler(pAdapter,
Oid,
InformationBuffer,
InformationBufferLength,
BytesWritten,
BytesNeeded
);
return (Status);
}
DBG_ENTER(pAdapter);
DBG_REQUEST(pAdapter,
("(OID=0x%08X %s)\n\t\tInfoLength=%d InfoBuffer=0x%X\n",
Oid, DbgGetOidString(Oid),
InformationBufferLength,
InformationBuffer
));
/*
// Initialize these once, since this is the majority of cases.
*/
SourceBuffer = &GenericULong;
SourceBufferLength = sizeof(ULONG);
/*
// Determine which OID is being requested and do the right thing.
// Refer to section 7.4 of the NDIS 3.0 specification for a complete
// description of OIDs and their return values.
*/
switch (Oid)
{
case OID_GEN_SUPPORTED_LIST:
/*
// NDIS wants to know which OID's to pass down to us.
// So we report back these new IOCTL's in addition to any NDIS OID's.
*/
SourceBuffer = (PVOID)g_SupportedOidArray;
SourceBufferLength = sizeof(g_SupportedOidArray);
break;
case OID_GEN_HARDWARE_STATUS:
GenericULong = NdisHardwareStatusReady;
break;
case OID_GEN_MEDIA_SUPPORTED:
GenericULong = NdisMediumWan;
break;
case OID_GEN_MEDIA_IN_USE:
GenericULong = NdisMediumWan;
break;
case OID_GEN_VENDOR_ID:
NdisMoveMemory((PVOID)VendorId, (PVOID)g_PermanentWanAddress, 3);
VendorId[3] = 0x0;
SourceBuffer = &g_PermanentWanAddress[0];
SourceBufferLength = sizeof(VendorId);
break;
case OID_GEN_VENDOR_DESCRIPTION:
SourceBuffer = (PUCHAR) g_VendorDescriptionString.Buffer;
SourceBufferLength = g_VendorDescriptionString.MaximumLength;
break;
case OID_GEN_MAXIMUM_LOOKAHEAD:
GenericULong = CARD_MAX_LOOKAHEAD;
break;
case OID_GEN_CURRENT_LOOKAHEAD:
GenericULong = CARD_MAX_LOOKAHEAD;
break;
case OID_GEN_MAC_OPTIONS:
GenericULong = NDIS_MAC_OPTION_RECEIVE_SERIALIZED |
NDIS_MAC_OPTION_NO_LOOPBACK |
NDIS_MAC_OPTION_TRANSFERS_NOT_PEND;
break;
case OID_WAN_PERMANENT_ADDRESS:
case OID_WAN_CURRENT_ADDRESS:
g_PermanentWanAddress[5] = (UCHAR) ((pAdapter->ObjectID & 0xFF) + '0');
SourceBuffer = g_PermanentWanAddress;
SourceBufferLength = sizeof(g_PermanentWanAddress);
break;
case OID_WAN_MEDIUM_SUBTYPE:
GenericULong = NdisWanMediumIsdn;
break;
case OID_WAN_GET_INFO:
SourceBuffer = &pAdapter->WanInfo;
SourceBufferLength = sizeof(NDIS_WAN_INFO);
break;
case OID_WAN_GET_LINK_INFO:
{
PNDIS_WAN_GET_LINK_INFO pGetWanLinkInfo;
PBCHANNEL_OBJECT pBChannel;
// A Pointer to one of our <t BCHANNEL_OBJECT>'s.
/*
// The InformationBuffer really points to a NDIS_WAN_GET_LINK_INFO
// which contains a pointer to one of our BCHANNEL_OBJECT's in the
// NdisLinkHandle field.
*/
pGetWanLinkInfo = (PNDIS_WAN_GET_LINK_INFO)InformationBuffer;
pBChannel = (PBCHANNEL_OBJECT) pGetWanLinkInfo->NdisLinkHandle;
/*
// Make sure what I just said is true.
*/
if (!IS_VALID_BCHANNEL(pAdapter, pBChannel))
{
SourceBufferLength = 0;
Status = NDIS_STATUS_INVALID_DATA;
break;
}
DBG_PARAMS(pAdapter,
("Returning:\n"
"NdisLinkHandle = %08lX\n"
"MaxSendFrameSize = %08lX\n"
"MaxRecvFrameSize = %08lX\n"
"SendFramingBits = %08lX\n"
"RecvFramingBits = %08lX\n"
"SendACCM = %08lX\n"
"RecvACCM = %08lX\n",
pBChannel->WanLinkInfo.NdisLinkHandle,
pBChannel->WanLinkInfo.MaxSendFrameSize ,
pBChannel->WanLinkInfo.MaxRecvFrameSize ,
pBChannel->WanLinkInfo.SendFramingBits ,
pBChannel->WanLinkInfo.RecvFramingBits ,
pBChannel->WanLinkInfo.SendACCM ,
pBChannel->WanLinkInfo.RecvACCM ));
SourceBuffer = &(pBChannel->WanLinkInfo);
SourceBufferLength = sizeof(NDIS_WAN_GET_LINK_INFO);
}
break;
#if defined(NDIS50_MINIPORT)
case OID_WAN_LINE_COUNT:
GenericULong = pAdapter->NumBChannels;
break;
case OID_PNP_CAPABILITIES:
// The sample just returns success for all PM events even though we
// don't really do anything with them.
PnpCapabilities.WakeUpCapabilities.MinMagicPacketWakeUp =
NdisDeviceStateUnspecified;
PnpCapabilities.WakeUpCapabilities.MinPatternWakeUp =
NdisDeviceStateUnspecified;
PnpCapabilities.WakeUpCapabilities.MinLinkChangeWakeUp =
NdisDeviceStateUnspecified;
SourceBuffer = &PnpCapabilities;
SourceBufferLength = sizeof(PnpCapabilities);
break;
case OID_PNP_QUERY_POWER:
// The sample just returns success for all PM events even though we
// don't really do anything with them.
break;
#endif // NDIS50_MINIPORT
default:
/*
// Unknown OID
*/
Status = NDIS_STATUS_INVALID_OID;
SourceBufferLength = 0;
DBG_WARNING(pAdapter,("UNSUPPORTED Oid=0x%08x\n", Oid));
break;
}
/*
// Now we copy the data into the caller's buffer if there's enough room,
// otherwise, we report the error and tell em how much we need.
*/
if (SourceBufferLength > InformationBufferLength)
{
*BytesNeeded = SourceBufferLength;
*BytesWritten = 0;
Status = NDIS_STATUS_INVALID_LENGTH;
}
else if (SourceBufferLength)
{
NdisMoveMemory(InformationBuffer,
SourceBuffer,
SourceBufferLength
);
*BytesNeeded = *BytesWritten = SourceBufferLength;
}
else
{
*BytesNeeded = *BytesWritten = 0;
}
DBG_REQUEST(pAdapter,
("RETURN: Status=0x%X Needed=%d Written=%d\n",
Status, *BytesNeeded, *BytesWritten));
DBG_RETURN(pAdapter, Status);
return (Status);
}
/* @doc INTERNAL Request Request_c MiniportSetInformation
ÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛÛ
@func
<f MiniportSetInformation> is a required function that allows
bound protocol drivers (or NDIS) to request changes in the
state information that the miniport maintains for
particular OIDs, such as changes in multicast addresses.
@comm
NDIS calls <f MiniportSetInformation> either on its own
behalf, such as to manage bindings to the miniport, or
when a bound protocol driver calls <f NdisRequest>.
If <f MiniportSetInformation> returns NDIS_STATUS_PENDING, the
driver must complete the request later by calling
NdisMSetInformationComplete. Until it completes any request,
the miniport can safely access the memory at InformationBuffer,
BytesRead, and BytesNeeded. After the miniport completes any set
request, ownership of these variables and the buffer reverts to
NDIS or the caller of <f NdisRequest>, whichever allocated the memory.
No other requests will be submitted to the WAN driver until the
current set request is complete. If the WAN driver does not complete
the call immediately (by returning NDIS_STATUS_PENDING), it must call
NdisMSetInformationComplete to complete the call.
Any of the settable NDIS global OIDs can be used, although a WAN
miniport cannot set the <f NDIS_MAC_OPTION_FULL_DUPLEX> flag in
response to an <f OID_GEN_MAC_OPTIONS> request. The following
WAN-specific OID is passed to MiniportSetInformation.
<f OID_WAN_SET_LINK_INFO><nl>
This OID is used to set the link characteristics.
The parameters in the structure passed for this OID
are described previously for OID_WAN_GET_LINK_INFO.
For more information about the system-defined OIDs, see Part 2 of the
Network Drivers Network Reference document.
<f MiniportSetInformation> can be pre-empted by an interrupt.
By default, <f MiniportSetInformation> runs at IRQL DISPATCH_LEVEL.
Calls to MiniportSetInformation changes information maintained by
the miniport. This function definition and operation is the same
as in a LAN miniport NIC driver except that certain WAN-specific
OIDs must be recognized.
@rdesc
<f MiniportSetInformation> can return one of the following:
@flag NDIS_STATUS_SUCCESS |
MiniportSetInformation used the data at InformationBuffer to
set itself or its NIC to the state required by the given Oid,
and it set the variable at BytesRead to the amount of supplied
data it used.
@flag NDIS_STATUS_PENDING |
The driver will complete the request asynchronously with a call
to NdisMSetInformationComplete when it has set itself or its NIC
to the state requested.
@flag NDIS_STATUS_INVALID_OID |
MiniportSetInformation did not recognize the Oid.
@flag NDIS_STATUS_INVALID_LENGTH |
The InformationBufferLength does not match the length required
by the given Oid. MiniportSetInformation returned how many bytes
the buffer should be at BytesNeeded.
@flag NDIS_STATUS_INVALID_DATA |
The data supplied at InformationBuffer was invalid for the given Oid.
@flag NDIS_STATUS_NOT_ACCEPTED |
MiniportSetInformation attempted the requested set operation on
the NIC but was unsuccessful.
@flag NDIS_STATUS_NOT_SUPPORTED |
MiniportSetInformation does not support the Oid, which is optional.
@flag NDIS_STATUS_RESOURCES |
MiniportSetInformation could not carry out the requested operation
due to resource constraints. This return value does not necessarily
mean that the same request, submitted at a later time, will be
failed for the same reason.
@xref
<f MiniportInitialize>
<f MiniportQueryInformation>
*/
NDIS_STATUS MiniportSetInformation(
IN PMINIPORT_ADAPTER_OBJECT pAdapter, // @parm
// A pointer to the <t MINIPORT_ADAPTER_OBJECT> instance.
IN NDIS_OID Oid, // @parm
// The OID. (See section 7.4 of the NDIS 3.0 specification for a complete
// description of OIDs.)
IN PVOID InformationBuffer, // @parm
// The buffer that will receive the information. (See section 7.4 of the
// NDIS 3.0 specification for a description of the length required for
// each OID.)
IN ULONG InformationBufferLength, // @parm
// The length in bytes of InformationBuffer.
OUT PULONG BytesRead, // @parm
// Returns the number of bytes read from InformationBuffer.
OUT PULONG BytesNeeded // @parm
// Returns the number of additional bytes needed to satisfy the OID.
)
{
DBG_FUNC("MiniportSetInformation")
NDIS_STATUS Status;
// Holds the status result returned by this function.
/*
// If this is a TAPI OID, pass it on over.
*/
if ((Oid & 0xFFFFFF00L) == (OID_TAPI_ACCEPT & 0xFFFFFF00L))
{
Status = TspiRequestHandler(pAdapter,
Oid,
InformationBuffer,
InformationBufferLength,
BytesRead,
BytesNeeded
);
return (Status);
}
DBG_ENTER(pAdapter);
DBG_REQUEST(pAdapter,
("(OID=0x%08X %s)\n\t\tInfoLength=%d InfoBuffer=0x%X\n",
Oid, DbgGetOidString(Oid),
InformationBufferLength,
InformationBuffer
));
/*
// Assume no extra bytes are needed.
*/
ASSERT(BytesRead && BytesNeeded);
*BytesRead = 0;
*BytesNeeded = 0;
/*
// Determine which OID is being requested and do the right thing.
*/
switch (Oid)
{
case OID_GEN_CURRENT_LOOKAHEAD:
/*
// WAN drivers always indicate the entire packet regardless of the
// lookahead size. So this request should be politely ignored.
*/
DBG_NOTICE(pAdapter,("OID_GEN_CURRENT_LOOKAHEAD: set=%d expected=%d\n",
*(PULONG) InformationBuffer, CARD_MAX_LOOKAHEAD));
ASSERT(InformationBufferLength == sizeof(ULONG));
*BytesNeeded = *BytesRead = sizeof(ULONG);
Status = NDIS_STATUS_SUCCESS;
break;
case OID_WAN_SET_LINK_INFO:
if (InformationBufferLength == sizeof(NDIS_WAN_SET_LINK_INFO))
{
PNDIS_WAN_SET_LINK_INFO pSetWanLinkInfo;
PBCHANNEL_OBJECT pBChannel;
// A Pointer to one of our <t BCHANNEL_OBJECT>'s.
/*
// The InformationBuffer really points to a NDIS_WAN_SET_LINK_INFO
// which contains a pointer to one of our BCHANNEL_OBJECT's in the
// NdisLinkHandle field.
*/
pSetWanLinkInfo = (PNDIS_WAN_SET_LINK_INFO)InformationBuffer;
pBChannel = (PBCHANNEL_OBJECT) pSetWanLinkInfo->NdisLinkHandle;
/*
// Make sure what I just said is true.
*/
if (!IS_VALID_BCHANNEL(pAdapter, pBChannel))
{
Status = NDIS_STATUS_INVALID_DATA;
break;
}
ASSERT(pBChannel->WanLinkInfo.NdisLinkHandle == pBChannel);
ASSERT(!(pBChannel->WanLinkInfo.SendFramingBits & ~pAdapter->WanInfo.FramingBits));
ASSERT(!(pBChannel->WanLinkInfo.RecvFramingBits & ~pAdapter->WanInfo.FramingBits));
/*
// Copy the data into our WanLinkInfo sturcture.
*/
NdisMoveMemory(&(pBChannel->WanLinkInfo),
InformationBuffer,
InformationBufferLength
);
*BytesRead = sizeof(NDIS_WAN_SET_LINK_INFO);
Status = NDIS_STATUS_SUCCESS;
if (pBChannel->WanLinkInfo.MaxSendFrameSize != pAdapter->WanInfo.MaxFrameSize ||
pBChannel->WanLinkInfo.MaxRecvFrameSize != pAdapter->WanInfo.MaxFrameSize)
{
DBG_NOTICE(pAdapter,("Line=%d - "
"NdisLinkHandle=%08lX - "
"SendFrameSize=%08lX - "
"RecvFrameSize=%08lX\n",
pBChannel->BChannelIndex,
pBChannel->WanLinkInfo.NdisLinkHandle,
pBChannel->WanLinkInfo.MaxSendFrameSize,
pBChannel->WanLinkInfo.MaxRecvFrameSize));
}
DBG_PARAMS(pAdapter,
("\n setting expected\n"
"NdisLinkHandle = %08lX=?=%08lX\n"
"MaxSendFrameSize = %08lX=?=%08lX\n"
"MaxRecvFrameSize = %08lX=?=%08lX\n"
"SendFramingBits = %08lX=?=%08lX\n"
"RecvFramingBits = %08lX=?=%08lX\n"
"SendACCM = %08lX=?=%08lX\n"
"RecvACCM = %08lX=?=%08lX\n",
pBChannel->WanLinkInfo.NdisLinkHandle , pBChannel,
pBChannel->WanLinkInfo.MaxSendFrameSize , pAdapter->WanInfo.MaxFrameSize,
pBChannel->WanLinkInfo.MaxRecvFrameSize , pAdapter->WanInfo.MaxFrameSize,
pBChannel->WanLinkInfo.SendFramingBits , pAdapter->WanInfo.FramingBits,
pBChannel->WanLinkInfo.RecvFramingBits , pAdapter->WanInfo.FramingBits,
pBChannel->WanLinkInfo.SendACCM , pAdapter->WanInfo.DesiredACCM,
pBChannel->WanLinkInfo.RecvACCM , pAdapter->WanInfo.DesiredACCM));
}
else
{
DBG_WARNING(pAdapter, ("OID_WAN_SET_LINK_INFO: Invalid size:%d expected:%d\n",
InformationBufferLength, sizeof(NDIS_WAN_SET_LINK_INFO)));
Status = NDIS_STATUS_INVALID_LENGTH;
}
*BytesNeeded = sizeof(NDIS_WAN_SET_LINK_INFO);
break;
#if defined(NDIS50_MINIPORT)
case OID_PNP_SET_POWER:
// The sample just returns success for all PM events even though we
// don't really do anything with them.
break;
#endif // NDIS50_MINIPORT
default:
/*
// Unknown OID
*/
Status = NDIS_STATUS_INVALID_OID;
DBG_WARNING(pAdapter,("UNSUPPORTED Oid=0x%08x\n", Oid));
break;
}
DBG_REQUEST(pAdapter,
("RETURN: Status=0x%X Needed=%d Read=%d\n",
Status, *BytesNeeded, *BytesRead));
DBG_RETURN(pAdapter, Status);
return (Status);
}