1473 lines
42 KiB
C
1473 lines
42 KiB
C
|
/*++
|
||
|
|
||
|
Copyright (c) 1996 Microsoft Corporation
|
||
|
|
||
|
Module Name:
|
||
|
|
||
|
rasirda.c
|
||
|
|
||
|
Abstract:
|
||
|
|
||
|
Miniport peice of the irda NDIS 5 miniport (WAN driver)
|
||
|
|
||
|
Author:
|
||
|
|
||
|
mbert 9-97
|
||
|
|
||
|
--*/
|
||
|
|
||
|
#include "rasirdap.h"
|
||
|
|
||
|
// Globals
|
||
|
#if DBG
|
||
|
|
||
|
#include <irioctl.h>
|
||
|
#if 0
|
||
|
int DbgSettings =
|
||
|
DBG_INIT |
|
||
|
DBG_CONFIG |
|
||
|
DBG_CONNECT |
|
||
|
// DBG_SEND |
|
||
|
// DBG_RECV |
|
||
|
DBG_LIB_OBJ |
|
||
|
DBG_LIB_CONNECT |
|
||
|
DBG_ERROR |
|
||
|
1;
|
||
|
#else
|
||
|
int DbgSettings = 0;
|
||
|
#endif
|
||
|
int DbgOutput =
|
||
|
DBG_OUTPUT_BUFFER |
|
||
|
DBG_OUTPUT_DEBUGGER |
|
||
|
0;
|
||
|
|
||
|
PDRIVER_OBJECT pRasIrDriverObject;
|
||
|
PDEVICE_OBJECT pRasIrDeviceObject;
|
||
|
|
||
|
NTSTATUS
|
||
|
DbgDispatch(
|
||
|
PDEVICE_OBJECT pDeviceObject,
|
||
|
PIRP pIrp);
|
||
|
#endif
|
||
|
|
||
|
NDIS_SPIN_LOCK RasIrSpinLock;
|
||
|
LIST_ENTRY RasIrAdapterList;
|
||
|
|
||
|
const USHORT FcsTable[256] = {
|
||
|
0x0000,0x1189,0x2312,0x329B,0x4624,0x57AD,0x6536,0x74BF,
|
||
|
0x8C48,0x9DC1,0xAF5A,0xBED3,0xCA6C,0xDBE5,0xE97E,0xF8F7,
|
||
|
0x1081,0x0108,0x3393,0x221A,0x56A5,0x472C,0x75B7,0x643E,
|
||
|
0x9CC9,0x8D40,0xBFDB,0xAE52,0xDAED,0xCB64,0xF9FF,0xE876,
|
||
|
0x2102,0x308B,0x0210,0x1399,0x6726,0x76AF,0x4434,0x55BD,
|
||
|
0xAD4A,0xBCC3,0x8E58,0x9FD1,0xEB6E,0xFAE7,0xC87C,0xD9F5,
|
||
|
0x3183,0x200A,0x1291,0x0318,0x77A7,0x662E,0x54B5,0x453C,
|
||
|
0xBDCB,0xAC42,0x9ED9,0x8F50,0xFBEF,0xEA66,0xD8FD,0xC974,
|
||
|
0x4204,0x538D,0x6116,0x709F,0x0420,0x15A9,0x2732,0x36BB,
|
||
|
0xCE4C,0xDFC5,0xED5E,0xFCD7,0x8868,0x99E1,0xAB7A,0xBAF3,
|
||
|
0x5285,0x430C,0x7197,0x601E,0x14A1,0x0528,0x37B3,0x263A,
|
||
|
0xDECD,0xCF44,0xFDDF,0xEC56,0x98E9,0x8960,0xBBFB,0xAA72,
|
||
|
0x6306,0x728F,0x4014,0x519D,0x2522,0x34AB,0x0630,0x17B9,
|
||
|
0xEF4E,0xFEC7,0xCC5C,0xDDD5,0xA96A,0xB8E3,0x8A78,0x9BF1,
|
||
|
0x7387,0x620E,0x5095,0x411C,0x35A3,0x242A,0x16B1,0x0738,
|
||
|
0xFFCF,0xEE46,0xDCDD,0xCD54,0xB9EB,0xA862,0x9AF9,0x8B70,
|
||
|
0x8408,0x9581,0xA71A,0xB693,0xC22C,0xD3A5,0xE13E,0xF0B7,
|
||
|
0x0840,0x19C9,0x2B52,0x3ADB,0x4E64,0x5FED,0x6D76,0x7CFF,
|
||
|
0x9489,0x8500,0xB79B,0xA612,0xD2AD,0xC324,0xF1BF,0xE036,
|
||
|
0x18C1,0x0948,0x3BD3,0x2A5A,0x5EE5,0x4F6C,0x7DF7,0x6C7E,
|
||
|
0xA50A,0xB483,0x8618,0x9791,0xE32E,0xF2A7,0xC03C,0xD1B5,
|
||
|
0x2942,0x38CB,0x0A50,0x1BD9,0x6F66,0x7EEF,0x4C74,0x5DFD,
|
||
|
0xB58B,0xA402,0x9699,0x8710,0xF3AF,0xE226,0xD0BD,0xC134,
|
||
|
0x39C3,0x284A,0x1AD1,0x0B58,0x7FE7,0x6E6E,0x5CF5,0x4D7C,
|
||
|
0xC60C,0xD785,0xE51E,0xF497,0x8028,0x91A1,0xA33A,0xB2B3,
|
||
|
0x4A44,0x5BCD,0x6956,0x78DF,0x0C60,0x1DE9,0x2F72,0x3EFB,
|
||
|
0xD68D,0xC704,0xF59F,0xE416,0x90A9,0x8120,0xB3BB,0xA232,
|
||
|
0x5AC5,0x4B4C,0x79D7,0x685E,0x1CE1,0x0D68,0x3FF3,0x2E7A,
|
||
|
0xE70E,0xF687,0xC41C,0xD595,0xA12A,0xB0A3,0x8238,0x93B1,
|
||
|
0x6B46,0x7ACF,0x4854,0x59DD,0x2D62,0x3CEB,0x0E70,0x1FF9,
|
||
|
0xF78F,0xE606,0xD49D,0xC514,0xB1AB,0xA022,0x92B9,0x8330,
|
||
|
0x7BC7,0x6A4E,0x58D5,0x495C,0x3DE3,0x2C6A,0x1EF1,0x0F78
|
||
|
};
|
||
|
|
||
|
NDIS_STATUS
|
||
|
DriverEntry(
|
||
|
IN PDRIVER_OBJECT DriverObject,
|
||
|
IN PUNICODE_STRING RegistryPath);
|
||
|
|
||
|
NDIS_STATUS
|
||
|
QueryInformation(
|
||
|
IN PRASIR_ADAPTER pAdapter,
|
||
|
IN PRASIR_VC pVc,
|
||
|
IN NDIS_OID Oid,
|
||
|
IN PVOID InformationBuffer,
|
||
|
IN ULONG InformationBufferLength,
|
||
|
OUT PULONG BytesWritten,
|
||
|
OUT PULONG BytesNeeded);
|
||
|
|
||
|
NDIS_STATUS
|
||
|
SetInformation(
|
||
|
IN PRASIR_ADAPTER pAdapter,
|
||
|
IN PRASIR_VC pVc,
|
||
|
IN NDIS_OID Oid,
|
||
|
IN PVOID InformationBuffer,
|
||
|
IN ULONG InformationBufferLength,
|
||
|
OUT PULONG BytesRead,
|
||
|
OUT PULONG BytesNeeded);
|
||
|
|
||
|
|
||
|
#pragma NDIS_INIT_FUNCTION(DriverEntry)
|
||
|
|
||
|
NDIS_STATUS
|
||
|
DriverEntry(
|
||
|
IN PDRIVER_OBJECT DriverObject,
|
||
|
IN PUNICODE_STRING RegistryPath )
|
||
|
{
|
||
|
NDIS_STATUS Status;
|
||
|
NDIS_MINIPORT_CHARACTERISTICS MiniportChr;
|
||
|
NDIS_HANDLE NdisWrapperHandle;
|
||
|
#if DBG
|
||
|
UNICODE_STRING DeviceName;
|
||
|
int i;
|
||
|
#endif
|
||
|
|
||
|
DEBUGMSG(DBG_INIT, ("RASIR: DriverEntry\n"));
|
||
|
|
||
|
IrdaClientInitialize();
|
||
|
|
||
|
NdisAllocateSpinLock(&RasIrSpinLock);
|
||
|
|
||
|
InitializeListHead(&RasIrAdapterList);
|
||
|
|
||
|
NdisMInitializeWrapper(&NdisWrapperHandle, DriverObject,
|
||
|
RegistryPath, NULL );
|
||
|
NdisZeroMemory( &MiniportChr, sizeof(MiniportChr) );
|
||
|
|
||
|
MiniportChr.MajorNdisVersion = NDIS_MajorVersion;
|
||
|
MiniportChr.MinorNdisVersion = NDIS_MinorVersion;
|
||
|
MiniportChr.Reserved = NDIS_USE_WAN_WRAPPER;
|
||
|
MiniportChr.HaltHandler = RasIrHalt;
|
||
|
MiniportChr.InitializeHandler = RasIrInitialize;
|
||
|
MiniportChr.ResetHandler = RasIrReset;
|
||
|
MiniportChr.ReturnPacketHandler = RasIrReturnPacket;
|
||
|
MiniportChr.CoActivateVcHandler = RasIrCoActivateVc;
|
||
|
MiniportChr.CoDeactivateVcHandler = RasIrCoDeactivateVc;
|
||
|
MiniportChr.CoSendPacketsHandler = RasIrCoSendPackets;
|
||
|
MiniportChr.CoRequestHandler = RasIrCoRequest;
|
||
|
|
||
|
// no CheckForHangHandler
|
||
|
// no DisableInterruptHandler
|
||
|
// no EnableInterruptHandler
|
||
|
// no HandleInterruptHandler
|
||
|
// no ISRHandler
|
||
|
// no QueryInformationHandler (see CoRequestHandler)
|
||
|
// no SendHandler (see CoSendPacketsHandler)
|
||
|
// no WanSendHandler (see CoSendPacketsHandler)
|
||
|
// no SetInformationHandler (see CoRequestHandler)
|
||
|
// no TransferDataHandler
|
||
|
// no WanTransferDataHandler
|
||
|
// no SendPacketsHandler (see CoSendPacketsHandler)
|
||
|
// no AllocateCompleteHandler
|
||
|
|
||
|
|
||
|
Status = NdisMRegisterMiniport(NdisWrapperHandle, &MiniportChr, sizeof(MiniportChr));
|
||
|
|
||
|
if (Status != NDIS_STATUS_SUCCESS)
|
||
|
{
|
||
|
DEBUGMSG(DBG_ERROR, ("RASIR: ->NdisMRegisterMiniport failed %x\n", Status));
|
||
|
NdisTerminateWrapper(NdisWrapperHandle, NULL);
|
||
|
}
|
||
|
|
||
|
#if 0
|
||
|
|
||
|
DbgMsgInit();
|
||
|
|
||
|
RtlInitUnicodeString(&DeviceName, L"\\Device\\RASIR");
|
||
|
|
||
|
Status = IoCreateDevice(
|
||
|
DriverObject, // DriverObject
|
||
|
0, // DeviceExtensionSize
|
||
|
&DeviceName, // DeviceName
|
||
|
FILE_DEVICE_NETWORK, // DeviceType
|
||
|
0, // DeviceCharacteristics
|
||
|
FALSE, // Exclusive?
|
||
|
&pRasIrDeviceObject); // DeviceObject pointer returned
|
||
|
|
||
|
if (Status != STATUS_SUCCESS)
|
||
|
DbgPrint("Failed to create device\n");
|
||
|
|
||
|
else
|
||
|
{
|
||
|
DriverObject->DriverUnload = NULL;
|
||
|
DriverObject->FastIoDispatch = NULL;
|
||
|
|
||
|
for (i=0; i <= IRP_MJ_MAXIMUM_FUNCTION; i++)
|
||
|
{
|
||
|
DriverObject->MajorFunction[i] = DbgDispatch;
|
||
|
}
|
||
|
|
||
|
pRasIrDeviceObject->Flags |= DO_DIRECT_IO;
|
||
|
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
return Status;
|
||
|
}
|
||
|
|
||
|
VOID
|
||
|
FreeAdapter(PRASIR_ADAPTER pAdapter)
|
||
|
{
|
||
|
|
||
|
ASSERT(IsListEmpty(&pAdapter->VcList));
|
||
|
|
||
|
#if DBG
|
||
|
pAdapter->Sig = 0xBAD;
|
||
|
#endif
|
||
|
|
||
|
NdisDeleteNPagedLookasideList(&pAdapter->WorkItemsLList);
|
||
|
NdisDeleteNPagedLookasideList(&pAdapter->AsyncBufLList);
|
||
|
|
||
|
NdisFreeMemory(pAdapter, sizeof(PRASIR_ADAPTER), 0);
|
||
|
}
|
||
|
|
||
|
NDIS_STATUS
|
||
|
RasIrInitialize(
|
||
|
OUT PNDIS_STATUS OpenErrorStatus,
|
||
|
OUT PUINT SelectedMediumIndex,
|
||
|
IN PNDIS_MEDIUM MediumArray,
|
||
|
IN UINT MediumArraySize,
|
||
|
IN NDIS_HANDLE MiniportAdapterHandle,
|
||
|
IN NDIS_HANDLE WrapperConfigurationContext)
|
||
|
{
|
||
|
UINT i;
|
||
|
PRASIR_ADAPTER pAdapter;
|
||
|
NDIS_CALL_MANAGER_CHARACTERISTICS CallMgrChr;
|
||
|
CO_ADDRESS_FAMILY Family;
|
||
|
NDIS_STATUS Status;
|
||
|
NDIS_HANDLE hConfig;
|
||
|
PNDIS_CONFIGURATION_PARAMETER Value;
|
||
|
NDIS_STRING TapiLineNameString = NDIS_STRING_CONST("TapiLineName");
|
||
|
NDIS_STRING ModemPortString = NDIS_STRING_CONST("ModemPort");
|
||
|
ULONG ModemPort;
|
||
|
|
||
|
DEBUGMSG(DBG_ERROR, ("RASIR: Miniport initialize\n"));
|
||
|
|
||
|
for (i = 0; i < MediumArraySize; ++i)
|
||
|
{
|
||
|
if (MediumArray[i] == NdisMediumCoWan)
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
if (i == MediumArraySize)
|
||
|
{
|
||
|
DEBUGMSG(DBG_ERROR, ("RASIR: No matching media\n"));
|
||
|
return NDIS_STATUS_FAILURE;
|
||
|
}
|
||
|
|
||
|
*SelectedMediumIndex = i;
|
||
|
|
||
|
NdisAllocateMemoryWithTag((PVOID *)&pAdapter, sizeof(RASIR_ADAPTER),
|
||
|
MT_RASIR_ADAPTER);
|
||
|
|
||
|
if (pAdapter == NULL)
|
||
|
return NDIS_STATUS_RESOURCES;
|
||
|
|
||
|
NdisZeroMemory(pAdapter, sizeof(RASIR_ADAPTER));
|
||
|
|
||
|
InitializeListHead(&pAdapter->VcList);
|
||
|
|
||
|
InitializeListHead(&pAdapter->EndpList);
|
||
|
|
||
|
NdisInitializeNPagedLookasideList(
|
||
|
&pAdapter->WorkItemsLList,
|
||
|
NULL, NULL, 0,
|
||
|
sizeof(NDIS_WORK_ITEM),
|
||
|
MT_RASIR_WORKITEM, // RasIr work item
|
||
|
10);
|
||
|
|
||
|
NdisInitializeNPagedLookasideList(
|
||
|
&pAdapter->AsyncBufLList,
|
||
|
NULL, NULL, 0,
|
||
|
sizeof(ASYNC_BUFFER),
|
||
|
MT_RASIR_ASYNCBUF, // RasIr work item
|
||
|
0);
|
||
|
|
||
|
NdisAllocateSpinLock(&pAdapter->SpinLock);
|
||
|
|
||
|
NdisOpenConfiguration(&Status, &hConfig, WrapperConfigurationContext);
|
||
|
|
||
|
if (Status == NDIS_STATUS_SUCCESS)
|
||
|
{
|
||
|
pAdapter->TapiLineNameBuf[0] = 0;
|
||
|
pAdapter->TapiLineName.Buffer = pAdapter->TapiLineNameBuf;
|
||
|
pAdapter->TapiLineName.Length = 0;
|
||
|
pAdapter->TapiLineName.MaximumLength = sizeof(pAdapter->TapiLineNameBuf);
|
||
|
|
||
|
NdisReadConfiguration(&Status,
|
||
|
&Value,
|
||
|
hConfig,
|
||
|
&TapiLineNameString,
|
||
|
NdisParameterString);
|
||
|
|
||
|
if (Status == NDIS_STATUS_SUCCESS)
|
||
|
{
|
||
|
RtlCopyUnicodeString(&pAdapter->TapiLineName, &Value->ParameterData.StringData);
|
||
|
*(PWCHAR)(pAdapter->TapiLineName.Buffer + (pAdapter->TapiLineName.MaximumLength)/sizeof(WCHAR)) = UNICODE_NULL;
|
||
|
|
||
|
} else {
|
||
|
//
|
||
|
// could not get the line name
|
||
|
//
|
||
|
NdisCloseConfiguration(hConfig);
|
||
|
goto Done;
|
||
|
}
|
||
|
|
||
|
NdisReadConfiguration(&Status,
|
||
|
&Value,
|
||
|
hConfig,
|
||
|
&ModemPortString,
|
||
|
NdisParameterInteger);
|
||
|
|
||
|
if (Status == NDIS_STATUS_SUCCESS)
|
||
|
{
|
||
|
pAdapter->ModemPort = (BOOLEAN) Value->ParameterData.IntegerData;
|
||
|
}
|
||
|
|
||
|
NdisCloseConfiguration(hConfig);
|
||
|
|
||
|
} else {
|
||
|
//
|
||
|
// could open the config info
|
||
|
//
|
||
|
goto Done;
|
||
|
}
|
||
|
|
||
|
DEBUGMSG(DBG_CONFIG, ("RASIR: ModemPort = %d\n", pAdapter->ModemPort));
|
||
|
|
||
|
pAdapter->MiniportAdapterHandle = MiniportAdapterHandle;
|
||
|
|
||
|
pAdapter->Info.MaxFrameSize = IRDA_MAX_DATA_SIZE;
|
||
|
pAdapter->Info.MaxSendWindow = 4;
|
||
|
pAdapter->Info.FramingBits = PPP_FRAMING |
|
||
|
PPP_COMPRESS_ADDRESS_CONTROL |
|
||
|
PPP_COMPRESS_PROTOCOL_FIELD;
|
||
|
pAdapter->Info.DesiredACCM = 0;
|
||
|
|
||
|
#if DBG
|
||
|
pAdapter->Sig = (ULONG)ADAPTER_SIG;
|
||
|
#endif
|
||
|
|
||
|
// Inform NDIS of the attributes of our adapter. Set the
|
||
|
// 'MiniportAdapterContext' returned to us by NDIS when it calls our
|
||
|
// handlers to the address of our adapter control block. Turn off
|
||
|
// hardware oriented timeouts.
|
||
|
//
|
||
|
|
||
|
NdisMSetAttributesEx(
|
||
|
MiniportAdapterHandle,
|
||
|
(NDIS_HANDLE)pAdapter,
|
||
|
(UINT)-1,
|
||
|
NDIS_ATTRIBUTE_IGNORE_PACKET_TIMEOUT
|
||
|
| NDIS_ATTRIBUTE_IGNORE_REQUEST_TIMEOUT,
|
||
|
NdisInterfaceInternal);
|
||
|
|
||
|
|
||
|
NdisZeroMemory(&Family, sizeof(Family));
|
||
|
Family.MajorVersion = NDIS_MajorVersion;
|
||
|
Family.MinorVersion = NDIS_MinorVersion;
|
||
|
Family.AddressFamily = CO_ADDRESS_FAMILY_TAPI_PROXY;
|
||
|
|
||
|
NdisZeroMemory(&CallMgrChr, sizeof(CallMgrChr));
|
||
|
CallMgrChr.MajorVersion = NDIS_MajorVersion;
|
||
|
CallMgrChr.MinorVersion = NDIS_MinorVersion;
|
||
|
CallMgrChr.CmCreateVcHandler = RasIrCmCreateVc;
|
||
|
CallMgrChr.CmDeleteVcHandler = RasIrCmDeleteVc;
|
||
|
CallMgrChr.CmOpenAfHandler = RasIrCmOpenAf;
|
||
|
CallMgrChr.CmCloseAfHandler = RasIrCmCloseAf;
|
||
|
CallMgrChr.CmRegisterSapHandler = RasIrCmRegisterSap;
|
||
|
CallMgrChr.CmDeregisterSapHandler = RasIrCmDeregisterSap;
|
||
|
CallMgrChr.CmMakeCallHandler = RasIrCmMakeCall;
|
||
|
CallMgrChr.CmCloseCallHandler = RasIrCmCloseCall;
|
||
|
CallMgrChr.CmIncomingCallCompleteHandler= RasIrCmIncomingCallComplete;
|
||
|
CallMgrChr.CmActivateVcCompleteHandler = RasIrCmActivateVcComplete;
|
||
|
CallMgrChr.CmDeactivateVcCompleteHandler= RasIrCmDeactivateVcComplete;
|
||
|
CallMgrChr.CmModifyCallQoSHandler = RasIrCmModifyCallQoS;
|
||
|
CallMgrChr.CmRequestHandler = RasIrCmRequest;
|
||
|
// no CmAddPartyHandler
|
||
|
// no CmDropPartyHandler
|
||
|
|
||
|
Status = NdisMCmRegisterAddressFamily(
|
||
|
MiniportAdapterHandle, &Family, &CallMgrChr,
|
||
|
sizeof(CallMgrChr));
|
||
|
|
||
|
Done:
|
||
|
|
||
|
if (Status != NDIS_STATUS_SUCCESS)
|
||
|
{
|
||
|
FreeAdapter(pAdapter);
|
||
|
DEBUGMSG(DBG_ERROR, ("RASIR: NdisMCmRegisterFamily failed %X\n",
|
||
|
Status));
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
NdisInterlockedInsertTailList(&RasIrAdapterList,
|
||
|
&pAdapter->Linkage,
|
||
|
&RasIrSpinLock);
|
||
|
}
|
||
|
|
||
|
return Status;
|
||
|
}
|
||
|
|
||
|
VOID
|
||
|
RasIrHalt(
|
||
|
IN NDIS_HANDLE MiniportAdapterContext)
|
||
|
{
|
||
|
PRASIR_ADAPTER pAdapter = MiniportAdapterContext;
|
||
|
|
||
|
GOODADAPTER(pAdapter);
|
||
|
|
||
|
NdisAcquireSpinLock(&RasIrSpinLock);
|
||
|
|
||
|
RemoveEntryList(&pAdapter->Linkage);
|
||
|
|
||
|
NdisReleaseSpinLock(&RasIrSpinLock);
|
||
|
|
||
|
DEBUGMSG(DBG_ERROR, ("RASIR: Halt!\n"));
|
||
|
|
||
|
FreeAdapter(pAdapter);
|
||
|
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
NDIS_STATUS
|
||
|
RasIrReset(
|
||
|
OUT PBOOLEAN AddressingReset,
|
||
|
IN NDIS_HANDLE MiniportAdapterContext)
|
||
|
{
|
||
|
return NDIS_STATUS_NOT_RESETTABLE;
|
||
|
}
|
||
|
|
||
|
NDIS_STATUS
|
||
|
RasIrCoActivateVc(
|
||
|
IN NDIS_HANDLE MiniportVcContext,
|
||
|
IN OUT PCO_CALL_PARAMETERS CallParameters)
|
||
|
{
|
||
|
return STATUS_SUCCESS;
|
||
|
}
|
||
|
|
||
|
|
||
|
NDIS_STATUS
|
||
|
RasIrCoDeactivateVc(
|
||
|
IN NDIS_HANDLE MiniportVcContext)
|
||
|
{
|
||
|
return STATUS_SUCCESS;
|
||
|
}
|
||
|
|
||
|
UINT
|
||
|
AddAsyncFraming(
|
||
|
IN PUCHAR pOutBuf,
|
||
|
IN PNDIS_BUFFER pNdisBuf,
|
||
|
IN ULONG Accm)
|
||
|
{
|
||
|
PUCHAR pData;
|
||
|
int DataLen;
|
||
|
USHORT Fcs = 0xFFFF;
|
||
|
PUCHAR pBuf = pOutBuf;
|
||
|
PUCHAR pEndBuf = pOutBuf + ASYNC_BUF_SIZE;
|
||
|
|
||
|
NdisQueryBufferSafe(pNdisBuf, &pData, &DataLen, NormalPagePriority);
|
||
|
|
||
|
if (pData == NULL)
|
||
|
{
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
ASSERT(DataLen);
|
||
|
|
||
|
*pBuf++ = AF_FLAG_CHAR;
|
||
|
|
||
|
while (DataLen != 0)
|
||
|
{
|
||
|
int i;
|
||
|
|
||
|
for (i = 0; i < DataLen; i++)
|
||
|
{
|
||
|
Fcs = CALC_FCS(Fcs, pData[i]);
|
||
|
STUFF_BYTE(pBuf, pData[i], Accm);
|
||
|
}
|
||
|
|
||
|
NdisGetNextBuffer(pNdisBuf, &pNdisBuf);
|
||
|
|
||
|
if (pNdisBuf)
|
||
|
{
|
||
|
NdisQueryBufferSafe(pNdisBuf, &pData, &DataLen, NormalPagePriority);
|
||
|
|
||
|
if (pData == NULL)
|
||
|
{
|
||
|
return 0;
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
if (pBuf > pEndBuf)
|
||
|
{
|
||
|
return 0;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
Fcs = ~(Fcs);
|
||
|
|
||
|
STUFF_BYTE(pBuf, ((CHAR) (Fcs & 0xFF)), Accm);
|
||
|
STUFF_BYTE(pBuf, ((CHAR) (Fcs >> 8)), Accm);
|
||
|
|
||
|
*pBuf++ = AF_FLAG_CHAR;
|
||
|
|
||
|
return (UINT)((UINT_PTR)pBuf - (UINT_PTR) pOutBuf);
|
||
|
}
|
||
|
|
||
|
VOID
|
||
|
RasIrCoSendPackets(
|
||
|
IN NDIS_HANDLE MiniportVcContext,
|
||
|
IN PPNDIS_PACKET PacketArray,
|
||
|
IN UINT NumberOfPackets )
|
||
|
{
|
||
|
PRASIR_VC pVc = MiniportVcContext;
|
||
|
UINT i;
|
||
|
PNDIS_BUFFER pNdisBuf;
|
||
|
UINT PacketLen;
|
||
|
NTSTATUS Status = NDIS_STATUS_SUCCESS;
|
||
|
BOOLEAN VcDown = FALSE;
|
||
|
PASYNC_BUFFER pAsyncBuf;
|
||
|
PMDL pSendNdisBuf;
|
||
|
|
||
|
GOODVC(pVc);
|
||
|
|
||
|
for (i = 0; i < NumberOfPackets; i++)
|
||
|
{
|
||
|
Status = NDIS_STATUS_SUCCESS;
|
||
|
|
||
|
NdisQueryPacket(PacketArray[i], NULL, NULL, &pNdisBuf, &PacketLen);
|
||
|
|
||
|
ASSERT(PacketLen <= IRDA_MAX_DATA_SIZE);
|
||
|
|
||
|
*((PVOID *) &PacketArray[i]->MiniportReserved[0]) = NULL;
|
||
|
|
||
|
NdisAcquireSpinLock(&pVc->pAdapter->SpinLock);
|
||
|
|
||
|
REFADD(&pVc->RefCnt, 'DNES');
|
||
|
|
||
|
pVc->OutstandingSends += 1;
|
||
|
|
||
|
if (!(pVc->Flags & VCF_IRDA_OPEN))
|
||
|
{
|
||
|
DEBUGMSG(DBG_ERROR, ("RASIR: Not sending on Vc %X, Irda is closed\n", pVc));
|
||
|
Status = NDIS_STATUS_CLOSED;
|
||
|
}
|
||
|
|
||
|
NdisReleaseSpinLock(&pVc->pAdapter->SpinLock);
|
||
|
|
||
|
if (Status == NDIS_STATUS_SUCCESS)
|
||
|
{
|
||
|
if (!pVc->AsyncFraming)
|
||
|
{
|
||
|
pSendNdisBuf = pNdisBuf;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
pAsyncBuf = NdisAllocateFromNPagedLookasideList(
|
||
|
&pVc->pAdapter->AsyncBufLList);
|
||
|
|
||
|
*((PVOID *) &PacketArray[i]->MiniportReserved[0]) = pAsyncBuf;
|
||
|
|
||
|
if (!pAsyncBuf)
|
||
|
{
|
||
|
Status = NDIS_STATUS_RESOURCES;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
UINT FrameLen;
|
||
|
|
||
|
pAsyncBuf->pNdisBuf = NULL;
|
||
|
|
||
|
FrameLen = AddAsyncFraming(pAsyncBuf->Buf, pNdisBuf,
|
||
|
pVc->LinkInfo.SendACCM);
|
||
|
if (!FrameLen)
|
||
|
{
|
||
|
DEBUGMSG(DBG_ERROR, ("RASIR: not sending, AsyncFraming failed\n"));
|
||
|
IrdaSendComplete(pVc, PacketArray[i], Status);
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
NdisAllocateBuffer(&Status, &pAsyncBuf->pNdisBuf,
|
||
|
pVc->TxBufferPool, pAsyncBuf->Buf, FrameLen);
|
||
|
|
||
|
if (Status == NDIS_STATUS_SUCCESS)
|
||
|
{
|
||
|
pSendNdisBuf = pAsyncBuf->pNdisBuf;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
DEBUGMSG(DBG_ERROR, ("RASIR: NdisAllocateBuffer failed %X\n", Status));
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (Status == NDIS_STATUS_SUCCESS)
|
||
|
{
|
||
|
IrdaSend(pVc->IrdaConnContext, pSendNdisBuf, PacketArray[i]);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
IrdaSendComplete(pVc, PacketArray[i], Status);
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
DEBUGMSG(DBG_SEND, ("RASIR: IrdaSend %X len %d pending\n",
|
||
|
PacketArray[i], PacketLen));
|
||
|
}
|
||
|
}
|
||
|
|
||
|
VOID
|
||
|
RasIrReceiveComplete(
|
||
|
PRASIR_VC pVc,
|
||
|
PVOID pRxBuf)
|
||
|
{
|
||
|
if (pVc->AsyncFraming)
|
||
|
{
|
||
|
// Async frames where already completed in
|
||
|
// ProcessAsyncRxPacket() so just return the
|
||
|
// buffer owned by rasir to the BufList
|
||
|
|
||
|
NdisFreeToNPagedLookasideList(
|
||
|
&pVc->pAdapter->AsyncBufLList,
|
||
|
pRxBuf);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
IrdaReceiveComplete(pVc->IrdaConnContext,
|
||
|
(PIRDA_RECVBUF) pRxBuf);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
VOID
|
||
|
RasIrReturnPacket(
|
||
|
IN NDIS_HANDLE MiniportAdapterContext,
|
||
|
IN PNDIS_PACKET pNdisPacket)
|
||
|
{
|
||
|
PVOID pRxBuf;
|
||
|
PRASIR_VC pVc;
|
||
|
PNDIS_BUFFER pNdisBuf;
|
||
|
|
||
|
pVc = *((PRASIR_VC *) &pNdisPacket->MiniportReserved[0]);
|
||
|
pRxBuf = *((PVOID *) &pNdisPacket->MiniportReserved[sizeof(PUCHAR)]);
|
||
|
|
||
|
|
||
|
GOODVC(pVc);
|
||
|
|
||
|
DEBUGMSG(DBG_RECV, ("RASIR: ReturnPacket %X\n", pRxBuf));
|
||
|
|
||
|
RasIrReceiveComplete(pVc, pRxBuf);
|
||
|
|
||
|
NdisUnchainBufferAtFront(pNdisPacket, &pNdisBuf);
|
||
|
|
||
|
NdisFreeBuffer(pNdisBuf);
|
||
|
|
||
|
NdisFreePacket(pNdisPacket);
|
||
|
|
||
|
REFDEL(&pVc->RefCnt, 'VCER');
|
||
|
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
NDIS_STATUS
|
||
|
RasIrCoRequest(
|
||
|
IN NDIS_HANDLE MiniportAdapterContext,
|
||
|
IN NDIS_HANDLE MiniportVcContext,
|
||
|
IN OUT PNDIS_REQUEST NdisRequest)
|
||
|
{
|
||
|
NDIS_STATUS Status;
|
||
|
PRASIR_ADAPTER pAdapter = MiniportAdapterContext;
|
||
|
PRASIR_VC pVc = MiniportVcContext;
|
||
|
|
||
|
DEBUGMSG(DBG_CONFIG, ("RASIR: RasIrCoRequest() type %X\n",
|
||
|
NdisRequest->RequestType));
|
||
|
|
||
|
switch (NdisRequest->RequestType)
|
||
|
{
|
||
|
case NdisRequestQueryInformation:
|
||
|
{
|
||
|
Status = QueryInformation(
|
||
|
pAdapter,
|
||
|
pVc,
|
||
|
NdisRequest->DATA.QUERY_INFORMATION.Oid,
|
||
|
NdisRequest->DATA.QUERY_INFORMATION.InformationBuffer,
|
||
|
NdisRequest->DATA.QUERY_INFORMATION.InformationBufferLength,
|
||
|
&NdisRequest->DATA.QUERY_INFORMATION.BytesWritten,
|
||
|
&NdisRequest->DATA.QUERY_INFORMATION.BytesNeeded );
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
case NdisRequestSetInformation:
|
||
|
{
|
||
|
Status = SetInformation(
|
||
|
pAdapter,
|
||
|
pVc,
|
||
|
NdisRequest->DATA.SET_INFORMATION.Oid,
|
||
|
NdisRequest->DATA.SET_INFORMATION.InformationBuffer,
|
||
|
NdisRequest->DATA.SET_INFORMATION.InformationBufferLength,
|
||
|
&NdisRequest->DATA.SET_INFORMATION.BytesRead,
|
||
|
&NdisRequest->DATA.SET_INFORMATION.BytesNeeded );
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
default:
|
||
|
{
|
||
|
Status = NDIS_STATUS_NOT_SUPPORTED;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return Status;
|
||
|
}
|
||
|
|
||
|
NDIS_STATUS
|
||
|
QueryInformation(
|
||
|
IN PRASIR_ADAPTER pAdapter,
|
||
|
IN PRASIR_VC pVc,
|
||
|
IN NDIS_OID Oid,
|
||
|
IN PVOID InformationBuffer,
|
||
|
IN ULONG InformationBufferLength,
|
||
|
OUT PULONG BytesWritten,
|
||
|
OUT PULONG BytesNeeded)
|
||
|
{
|
||
|
NDIS_STATUS status;
|
||
|
ULONG ulInfo;
|
||
|
VOID* pInfo;
|
||
|
ULONG ulInfoLen;
|
||
|
|
||
|
status = NDIS_STATUS_SUCCESS;
|
||
|
|
||
|
// The cases in this switch statement find or create a buffer containing
|
||
|
// the requested information and point 'pInfo' at it, noting it's length
|
||
|
// in 'ulInfoLen'. Since many of the OIDs return a ULONG, a 'ulInfo'
|
||
|
// buffer is set up as the default.
|
||
|
//
|
||
|
ulInfo = 0;
|
||
|
pInfo = &ulInfo;
|
||
|
ulInfoLen = sizeof(ulInfo);
|
||
|
|
||
|
DEBUGMSG(DBG_CONFIG, ("RASIR: QueryInformation oid:%X\n", Oid));
|
||
|
|
||
|
switch (Oid)
|
||
|
{
|
||
|
case OID_GEN_MAXIMUM_LOOKAHEAD:
|
||
|
{
|
||
|
DEBUGMSG(DBG_CONFIG, (" OID_GET_MAXIMUM_LOOKAHEAD = %d\n",
|
||
|
IRDA_MAX_DATA_SIZE));
|
||
|
ulInfo = IRDA_MAX_DATA_SIZE;
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
case OID_GEN_MAC_OPTIONS:
|
||
|
{
|
||
|
// Report a bitmask defining optional properties of the driver.
|
||
|
//
|
||
|
// NDIS_MAC_OPTION_COPY_LOOKAHEAD_DATA promises that our receive
|
||
|
// buffer is not on a device-specific card.
|
||
|
//
|
||
|
// NDIS_MAC_OPTION_TRANSFERS_NOT_PEND promises we won't return
|
||
|
// NDIS_STATUS_PENDING from our TransferData handler which is true
|
||
|
// since we don't have one.
|
||
|
//
|
||
|
ulInfo = NDIS_MAC_OPTION_COPY_LOOKAHEAD_DATA
|
||
|
| NDIS_MAC_OPTION_TRANSFERS_NOT_PEND;
|
||
|
DEBUGMSG(DBG_CONFIG, (" OID_GEN_MAC_OPTIONS = %d\n",ulInfo));
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
case OID_WAN_MEDIUM_SUBTYPE:
|
||
|
{
|
||
|
// Report the media subtype we support. NDISWAN may use this in
|
||
|
// the future (doesn't now) to provide framing differences for
|
||
|
// different media.
|
||
|
//
|
||
|
DEBUGMSG(DBG_CONFIG, (" OID_WAN_MEDIUM_SUBTYPE\n"));
|
||
|
if (pAdapter->ModemPort)
|
||
|
{
|
||
|
ulInfo = NdisWanMediumSerial;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
ulInfo = NdisWanMediumIrda;
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
case OID_WAN_CO_GET_INFO:
|
||
|
{
|
||
|
// Report the capabilities of the adapter.
|
||
|
//
|
||
|
DEBUGMSG(DBG_CONFIG, (" OID_WAN_CO_GET_INFO\n"));
|
||
|
pInfo = &pAdapter->Info;
|
||
|
ulInfoLen = sizeof(NDIS_WAN_CO_INFO);
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
case OID_WAN_CO_GET_LINK_INFO:
|
||
|
{
|
||
|
// Report the current state of the link.
|
||
|
//
|
||
|
DEBUGMSG(DBG_CONFIG, (" OID_WAN_CO_GET_LINK_INFO\n"));
|
||
|
|
||
|
if (!pVc)
|
||
|
return NDIS_STATUS_INVALID_DATA;
|
||
|
|
||
|
pInfo = &pVc->LinkInfo;
|
||
|
|
||
|
ulInfoLen = sizeof(NDIS_WAN_CO_GET_LINK_INFO);
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
case OID_WAN_CO_GET_COMP_INFO:
|
||
|
{
|
||
|
// Report the type of compression we provide, which is none.
|
||
|
//
|
||
|
DEBUGMSG(DBG_CONFIG, (" OID_WAN_CO_GET_COMP_INFO\n"));
|
||
|
status = NDIS_STATUS_NOT_SUPPORTED;
|
||
|
ulInfoLen = 0;
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
case OID_WAN_CO_GET_STATS_INFO:
|
||
|
{
|
||
|
// Because L2TP doesn't do compression, NDISWAN will use it's own
|
||
|
// statistics and not query ours.
|
||
|
//
|
||
|
ASSERT( !"OID_WAN_CO_GET_STATS_INFO?" );
|
||
|
status = NDIS_STATUS_NOT_SUPPORTED;
|
||
|
ulInfoLen = 0;
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
case OID_PNP_QUERY_POWER:
|
||
|
DEBUGMSG(DBG_CONFIG, (" OID_PNP_QUERY_POWER\n"));
|
||
|
status = NDIS_STATUS_NOT_SUPPORTED;
|
||
|
ulInfoLen = 0;
|
||
|
break;
|
||
|
|
||
|
case OID_PNP_CAPABILITIES:
|
||
|
DEBUGMSG(DBG_CONFIG, (" OID_PNP_CAP\n"));
|
||
|
status = NDIS_STATUS_NOT_SUPPORTED;
|
||
|
ulInfoLen = 0;
|
||
|
break;
|
||
|
|
||
|
case OID_PNP_SET_POWER:
|
||
|
DEBUGMSG(DBG_CONFIG, (" OID_PNP_SET_POWER\n"));
|
||
|
status = NDIS_STATUS_NOT_SUPPORTED;
|
||
|
ulInfoLen = 0;
|
||
|
break;
|
||
|
|
||
|
default:
|
||
|
{
|
||
|
DEBUGMSG(DBG_CONFIG, ( " Oid $%08x?\n", Oid));
|
||
|
status = NDIS_STATUS_INVALID_OID;
|
||
|
ulInfoLen = 0;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (ulInfoLen > InformationBufferLength)
|
||
|
{
|
||
|
// Caller's buffer is too small. Tell him what he needs.
|
||
|
//
|
||
|
*BytesNeeded = ulInfoLen;
|
||
|
status = NDIS_STATUS_INVALID_LENGTH;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
// Copy the found result to caller's buffer.
|
||
|
//
|
||
|
if (ulInfoLen > 0)
|
||
|
{
|
||
|
NdisMoveMemory( InformationBuffer, pInfo, ulInfoLen );
|
||
|
}
|
||
|
|
||
|
*BytesNeeded = *BytesWritten = ulInfoLen;
|
||
|
}
|
||
|
|
||
|
return status;
|
||
|
}
|
||
|
|
||
|
NDIS_STATUS
|
||
|
SetInformation(
|
||
|
IN PRASIR_ADAPTER pAdapter,
|
||
|
IN PRASIR_VC pVc,
|
||
|
IN NDIS_OID Oid,
|
||
|
IN PVOID InformationBuffer,
|
||
|
IN ULONG InformationBufferLength,
|
||
|
OUT PULONG BytesRead,
|
||
|
OUT PULONG BytesNeeded)
|
||
|
{
|
||
|
NDIS_STATUS Status;
|
||
|
|
||
|
DEBUGMSG(DBG_CONFIG, ("RASIR: SetInformation oid:%X\n", Oid));
|
||
|
|
||
|
Status = NDIS_STATUS_SUCCESS;
|
||
|
|
||
|
switch (Oid)
|
||
|
{
|
||
|
case OID_WAN_CO_SET_LINK_INFO:
|
||
|
{
|
||
|
// Read new link state settings.
|
||
|
//
|
||
|
DEBUGMSG(DBG_CONFIG, (" OID_WAN_CO_SET_LINK_INFO\n"));
|
||
|
|
||
|
if (InformationBufferLength < sizeof(NDIS_WAN_CO_SET_LINK_INFO))
|
||
|
{
|
||
|
Status = NDIS_STATUS_INVALID_LENGTH;
|
||
|
*BytesRead = 0;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
if (!pVc)
|
||
|
return NDIS_STATUS_INVALID_DATA;
|
||
|
|
||
|
ASSERT( sizeof(pVc->LinkInfo)
|
||
|
== sizeof(NDIS_WAN_CO_SET_LINK_INFO) );
|
||
|
|
||
|
NdisMoveMemory(&pVc->LinkInfo, InformationBuffer,
|
||
|
sizeof(pVc->LinkInfo) );
|
||
|
|
||
|
DEBUGMSG(DBG_CONFIG, (" MaxSend %d MaxRecv %d SendAccm %X RecvAccm %X\n",
|
||
|
pVc->LinkInfo.MaxSendFrameSize,
|
||
|
pVc->LinkInfo.MaxRecvFrameSize,
|
||
|
pVc->LinkInfo.SendACCM,
|
||
|
pVc->LinkInfo.RecvACCM));
|
||
|
|
||
|
*BytesRead = sizeof(NDIS_WAN_CO_SET_LINK_INFO);
|
||
|
}
|
||
|
*BytesNeeded = sizeof(NDIS_WAN_CO_SET_LINK_INFO);
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case OID_WAN_CO_SET_COMP_INFO:
|
||
|
{
|
||
|
// L2TP doesn't provide compression. Neither does IrDA
|
||
|
//
|
||
|
DEBUGMSG(DBG_CONFIG, (" OID_WAN_CO_SET_COMP_INFO\n"));
|
||
|
Status = NDIS_STATUS_NOT_SUPPORTED;
|
||
|
*BytesRead = *BytesNeeded = 0;
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
case OID_PNP_QUERY_POWER:
|
||
|
DEBUGMSG(DBG_CONFIG, (" OID_PNP_QUERY_POWER\n"));
|
||
|
Status = NDIS_STATUS_NOT_SUPPORTED;
|
||
|
*BytesRead = *BytesNeeded = 0;
|
||
|
break;
|
||
|
|
||
|
case OID_PNP_CAPABILITIES:
|
||
|
DEBUGMSG(DBG_CONFIG, (" OID_PNP_CAP\n"));
|
||
|
Status = NDIS_STATUS_NOT_SUPPORTED;
|
||
|
*BytesRead = *BytesNeeded = 0;
|
||
|
break;
|
||
|
|
||
|
case OID_PNP_SET_POWER:
|
||
|
DEBUGMSG(DBG_CONFIG, (" OID_PNP_SET_POWER\n"));
|
||
|
Status = NDIS_STATUS_NOT_SUPPORTED;
|
||
|
*BytesRead = *BytesNeeded = 0;
|
||
|
break;
|
||
|
|
||
|
#if 0
|
||
|
// These OIDs are mandatory according to current doc, but since
|
||
|
// NDISWAN never requests them they are omitted.
|
||
|
//
|
||
|
case OID_GEN_CURRENT_PACKET_FILTER:
|
||
|
case OID_GEN_CURRENT_LOOKAHEAD:
|
||
|
case OID_GEN_PROTOCOL_OPTIONS:
|
||
|
case OID_WAN_PROTOCOL_TYPE:
|
||
|
case OID_WAN_HEADER_FORMAT:
|
||
|
#endif
|
||
|
default:
|
||
|
{
|
||
|
DEBUGMSG(DBG_CONFIG, ("Oid $%08x?\n", Oid ));
|
||
|
Status = NDIS_STATUS_INVALID_OID;
|
||
|
*BytesRead = *BytesNeeded = 0;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return Status;
|
||
|
}
|
||
|
|
||
|
VOID
|
||
|
ProcessAsyncRxPacket(
|
||
|
PRASIR_VC pVc,
|
||
|
PIRDA_RECVBUF pIrdaRxBuf)
|
||
|
{
|
||
|
PASYNC_BUFFER pReturnRxBuf = NULL;
|
||
|
BOOLEAN EndOfFrame = FALSE;
|
||
|
PASYNC_BUFFER pCurrAsyncBuf;
|
||
|
UINT i;
|
||
|
CHAR Byte;
|
||
|
|
||
|
//DbgMsgBuf(pIrdaRxBuf->Buf, pIrdaRxBuf->BufLen);
|
||
|
|
||
|
for (i = 0; i < pIrdaRxBuf->BufLen; i++)
|
||
|
{
|
||
|
//
|
||
|
// If we are not currently reasembling a frame
|
||
|
// in pCurrAsyncBuf then get a new one
|
||
|
//
|
||
|
|
||
|
if (pVc->pCurrAsyncBuf == NULL)
|
||
|
{
|
||
|
pVc->AsyncFramingState = RX_READY;
|
||
|
|
||
|
if (!(pCurrAsyncBuf = NdisAllocateFromNPagedLookasideList(
|
||
|
&pVc->pAdapter->AsyncBufLList)))
|
||
|
{
|
||
|
DEBUGMSG(DBG_ERROR, ("RASIR: Failed to allocate AsyncBuf\n"));
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
pCurrAsyncBuf->BufLen = 0;
|
||
|
pVc->pCurrAsyncBuf = pCurrAsyncBuf;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
pCurrAsyncBuf = pVc->pCurrAsyncBuf;
|
||
|
}
|
||
|
|
||
|
Byte = pIrdaRxBuf->Buf[i];
|
||
|
|
||
|
switch (pVc->AsyncFramingState)
|
||
|
{
|
||
|
case RX_READY:
|
||
|
|
||
|
if (Byte == AF_FLAG_CHAR)
|
||
|
{
|
||
|
pVc->AsyncFramingState = RX_RX;
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case RX_RX:
|
||
|
switch (Byte)
|
||
|
{
|
||
|
case AF_FLAG_CHAR:
|
||
|
|
||
|
// We have a complete frame, put it on the completed
|
||
|
// list
|
||
|
|
||
|
pVc->AsyncFramingState = RX_READY;
|
||
|
|
||
|
if (pCurrAsyncBuf->BufLen > 2)
|
||
|
{
|
||
|
// remove FCS, IrDA is reliable so I don't look at it
|
||
|
|
||
|
pCurrAsyncBuf->BufLen -= 2;
|
||
|
|
||
|
NdisInterlockedInsertTailList(
|
||
|
&pVc->CompletedAsyncBufList,
|
||
|
&pCurrAsyncBuf->Linkage,
|
||
|
&pVc->pAdapter->SpinLock);
|
||
|
|
||
|
DEBUGMSG(DBG_RECV, ("RASIR: Complete async frame %d, %d of %d\n",
|
||
|
pVc->pCurrAsyncBuf->BufLen, i, pIrdaRxBuf->BufLen));
|
||
|
|
||
|
pVc->pCurrAsyncBuf = NULL;
|
||
|
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
// frame too small, we'll just ignore it
|
||
|
pCurrAsyncBuf->BufLen = 0;
|
||
|
|
||
|
DEBUGMSG(DBG_ERROR, ("RASIR: Async failure 1\n"));
|
||
|
}
|
||
|
|
||
|
break;
|
||
|
|
||
|
case AF_ESC_CHAR:
|
||
|
pVc->AsyncFramingState = RX_ESC;
|
||
|
break;
|
||
|
|
||
|
default:
|
||
|
if (pCurrAsyncBuf->BufLen >= ASYNC_BUF_SIZE)
|
||
|
{
|
||
|
pCurrAsyncBuf->BufLen = 0;
|
||
|
pVc->AsyncFramingState = RX_READY;
|
||
|
DEBUGMSG(DBG_ERROR, ("RASIR: Async buffer overflow\n"));
|
||
|
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
pCurrAsyncBuf->Buf[pCurrAsyncBuf->BufLen] = Byte;
|
||
|
pCurrAsyncBuf->BufLen += 1;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
break;
|
||
|
|
||
|
case RX_ESC:
|
||
|
if (pCurrAsyncBuf->BufLen >= ASYNC_BUF_SIZE)
|
||
|
{
|
||
|
pCurrAsyncBuf->BufLen = 0;
|
||
|
pVc->AsyncFramingState = RX_READY;
|
||
|
DEBUGMSG(DBG_ERROR, ("RASIR: Async failure 2\n"));
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
pCurrAsyncBuf->Buf[pCurrAsyncBuf->BufLen] = (Byte ^= AF_COMP_BIT);
|
||
|
pCurrAsyncBuf->BufLen += 1;
|
||
|
pVc->AsyncFramingState = RX_RX;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
VOID
|
||
|
IndicateReceive(
|
||
|
PRASIR_VC pVc,
|
||
|
PIRDA_RECVBUF pRecvBuf,
|
||
|
PASYNC_BUFFER pAsyncBuf,
|
||
|
BOOLEAN LastBuf)
|
||
|
{
|
||
|
PNDIS_PACKET pNdisPacket;
|
||
|
PNDIS_BUFFER pNdisBuf;
|
||
|
NDIS_STATUS Status;
|
||
|
KIRQL OldIrql;
|
||
|
PVOID pRecvOrAsyncBuf = NULL;
|
||
|
int Len;
|
||
|
|
||
|
REFADD(&pVc->RefCnt, 'VCER');
|
||
|
|
||
|
if (pVc->AsyncFraming)
|
||
|
{
|
||
|
ASSERT(pAsyncBuf && !pRecvBuf);
|
||
|
pRecvOrAsyncBuf = pAsyncBuf;
|
||
|
Len = pAsyncBuf->BufLen;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
ASSERT(pRecvBuf && !pAsyncBuf);
|
||
|
pRecvOrAsyncBuf = pRecvBuf;
|
||
|
Len = pRecvBuf->BufLen;
|
||
|
}
|
||
|
|
||
|
if (!(pVc->Flags & VCF_OPEN))
|
||
|
{
|
||
|
DEBUGMSG(DBG_ERROR, ("RASIR: VC not open yet, dropping packet\n"));
|
||
|
goto error1;
|
||
|
}
|
||
|
NdisAllocatePacket(&Status, &pNdisPacket, pVc->RxPacketPool);
|
||
|
|
||
|
if (Status != NDIS_STATUS_SUCCESS)
|
||
|
{
|
||
|
DEBUGMSG(DBG_ERROR, ("RASIR: NdisAllocatePacket failed, %X\n",
|
||
|
Status));
|
||
|
goto error1;
|
||
|
}
|
||
|
|
||
|
if (pVc->AsyncFraming)
|
||
|
{
|
||
|
NdisAllocateBuffer(&Status, &pNdisBuf, pVc->RxBufferPool,
|
||
|
pAsyncBuf->Buf, pAsyncBuf->BufLen);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
NdisAllocateBuffer(&Status, &pNdisBuf, pVc->RxBufferPool,
|
||
|
pRecvBuf->Buf, pRecvBuf->BufLen);
|
||
|
}
|
||
|
|
||
|
if (Status != NDIS_STATUS_SUCCESS)
|
||
|
{
|
||
|
DEBUGMSG(DBG_ERROR, ("RASIR: NdisAllocateBuffer failed, %X\n",
|
||
|
Status));
|
||
|
goto error2;
|
||
|
}
|
||
|
|
||
|
NdisChainBufferAtFront(pNdisPacket, pNdisBuf);
|
||
|
|
||
|
*((PRASIR_VC *) &pNdisPacket->MiniportReserved[0]) = pVc;
|
||
|
*((PVOID *) &pNdisPacket->MiniportReserved[sizeof(PUCHAR)]) = pRecvOrAsyncBuf;
|
||
|
|
||
|
KeRaiseIrql(DISPATCH_LEVEL, &OldIrql);
|
||
|
|
||
|
if (LastBuf)
|
||
|
{
|
||
|
NDIS_SET_PACKET_STATUS(pNdisPacket, NDIS_STATUS_RESOURCES);
|
||
|
|
||
|
DEBUGMSG(DBG_RECV, ("RASIR: Irtdi out of buffers\n"));
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
NDIS_SET_PACKET_STATUS(pNdisPacket, NDIS_STATUS_SUCCESS);
|
||
|
}
|
||
|
|
||
|
Status = NDIS_GET_PACKET_STATUS(pNdisPacket);
|
||
|
|
||
|
DEBUGMSG(DBG_RECV, ("RASIR: ->NdisMCoIndicateReceive %d\n", Len));
|
||
|
|
||
|
NdisMCoIndicateReceivePacket(pVc->NdisVcHandle, &pNdisPacket, 1);
|
||
|
|
||
|
NdisMCoReceiveComplete(pVc->pAdapter->MiniportAdapterHandle);
|
||
|
|
||
|
KeLowerIrql(OldIrql);
|
||
|
|
||
|
if (Status == NDIS_STATUS_SUCCESS) {
|
||
|
//
|
||
|
// the packets will be returned later
|
||
|
//
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
|
||
|
NdisUnchainBufferAtFront(pNdisPacket, &pNdisBuf);
|
||
|
|
||
|
NdisFreeBuffer(pNdisBuf);
|
||
|
|
||
|
error2:
|
||
|
|
||
|
NdisFreePacket(pNdisPacket);
|
||
|
|
||
|
error1:
|
||
|
|
||
|
DEBUGMSG(DBG_RECV, ("RASIR: RecvIndication completed synchronously for %X\n",
|
||
|
pRecvOrAsyncBuf));
|
||
|
|
||
|
RasIrReceiveComplete(pVc, pRecvOrAsyncBuf);
|
||
|
|
||
|
REFDEL(&pVc->RefCnt, 'VCER');
|
||
|
|
||
|
}
|
||
|
|
||
|
VOID
|
||
|
IrdaReceiveIndication(
|
||
|
PVOID ConnectionContext,
|
||
|
PIRDA_RECVBUF pRecvBuf,
|
||
|
BOOLEAN LastBuf)
|
||
|
{
|
||
|
PRASIR_VC pVc = ConnectionContext;
|
||
|
PASYNC_BUFFER pAsyncRxBuf;
|
||
|
|
||
|
|
||
|
DEBUGMSG(DBG_RECV, ("RASIR: IrdaReceiveIndication %X, %d bytes\n",
|
||
|
pRecvBuf, pRecvBuf->BufLen));
|
||
|
|
||
|
if (!pVc)
|
||
|
{
|
||
|
DEBUGMSG(DBG_ERROR, ("RASIR: IrdaReceiveIndication NULL vc\n"));
|
||
|
ASSERT(0);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
GOODVC(pVc);
|
||
|
|
||
|
GOODADAPTER(pVc->pAdapter);
|
||
|
|
||
|
if (pVc->IrModemCall && pVc->ModemState < MS_ONLINE)
|
||
|
{
|
||
|
ProcessOfflineRxBuf(pVc, pRecvBuf);
|
||
|
|
||
|
IrdaReceiveComplete(pVc->IrdaConnContext, pRecvBuf);
|
||
|
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (pVc->AsyncFraming)
|
||
|
{
|
||
|
ProcessAsyncRxPacket(pVc, pRecvBuf);
|
||
|
|
||
|
// We are done with pRecvBuf. The async framing has
|
||
|
// been removed and the data is now contained in a buffer
|
||
|
// on the CompletedAsyncBufList or if not a full frame
|
||
|
// then in pCurrAsyncBuf
|
||
|
|
||
|
IrdaReceiveComplete(pVc->IrdaConnContext, pRecvBuf);
|
||
|
|
||
|
while ((pAsyncRxBuf = (PASYNC_BUFFER) NdisInterlockedRemoveHeadList(
|
||
|
&pVc->CompletedAsyncBufList,
|
||
|
&pVc->pAdapter->SpinLock)) != NULL)
|
||
|
{
|
||
|
IndicateReceive(pVc, NULL, pAsyncRxBuf, LastBuf);
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
IndicateReceive(pVc, pRecvBuf, NULL, LastBuf);
|
||
|
}
|
||
|
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
VOID
|
||
|
IrdaSendComplete(
|
||
|
PVOID ConnectContext,
|
||
|
PVOID SendContext,
|
||
|
NTSTATUS Status)
|
||
|
{
|
||
|
PRASIR_VC pVc = ConnectContext;
|
||
|
PNDIS_PACKET pNdisPacket = (PNDIS_PACKET) SendContext;
|
||
|
|
||
|
DEBUGMSG(DBG_SEND, ("RASIR: IrdaSendComplete vc:%X, Packet %X\n", pVc, SendContext));
|
||
|
|
||
|
GOODVC(pVc);
|
||
|
|
||
|
if (SendContext == RASIR_INTERNAL_SEND)
|
||
|
{
|
||
|
ASSERT(pVc->pOfflineNdisBuf);
|
||
|
|
||
|
NdisFreeBuffer(pVc->pOfflineNdisBuf);
|
||
|
|
||
|
pVc->pOfflineNdisBuf = NULL;
|
||
|
|
||
|
goto EXIT;
|
||
|
}
|
||
|
|
||
|
if (pVc->AsyncFraming)
|
||
|
{
|
||
|
PASYNC_BUFFER pAsyncBuf;
|
||
|
|
||
|
pAsyncBuf = *((PASYNC_BUFFER *) &pNdisPacket->MiniportReserved[0]);
|
||
|
|
||
|
if (pAsyncBuf)
|
||
|
{
|
||
|
if (pAsyncBuf->pNdisBuf)
|
||
|
{
|
||
|
NdisFreeBuffer(pAsyncBuf->pNdisBuf);
|
||
|
}
|
||
|
|
||
|
NdisFreeToNPagedLookasideList(&pVc->pAdapter->AsyncBufLList,
|
||
|
pAsyncBuf);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
NdisMCoSendComplete(Status, pVc->NdisVcHandle, pNdisPacket);
|
||
|
|
||
|
NdisAcquireSpinLock(&pVc->pAdapter->SpinLock);
|
||
|
|
||
|
pVc->OutstandingSends -= 1; // do interlockedDec dork
|
||
|
|
||
|
NdisReleaseSpinLock(&pVc->pAdapter->SpinLock);
|
||
|
|
||
|
if ((pVc->Flags & VCF_CLOSE_PEND) && (pVc->OutstandingSends == 0))
|
||
|
{
|
||
|
DEBUGMSG(DBG_CONNECT, ("RASIR: Outstanding send to 0, CompleteClose\n"));
|
||
|
CompleteClose(pVc);
|
||
|
}
|
||
|
|
||
|
EXIT:
|
||
|
|
||
|
REFDEL(&pVc->RefCnt, 'DNES');
|
||
|
}
|
||
|
|
||
|
NDIS_STATUS
|
||
|
ScheduleWork(
|
||
|
IN PRASIR_ADAPTER pAdapter,
|
||
|
IN NDIS_PROC pProc,
|
||
|
IN PVOID pContext )
|
||
|
|
||
|
// Schedules a PASSIVE IRQL callback to routine 'pProc' which will be
|
||
|
// passed 'pContext'. 'PAdapter' is the adapter control block from which
|
||
|
// the work item is allocated.
|
||
|
//
|
||
|
// Returns NDIS_STATUS_SUCCESS or an error code.
|
||
|
//
|
||
|
{
|
||
|
NDIS_STATUS status;
|
||
|
NDIS_WORK_ITEM* pWork;
|
||
|
|
||
|
pWork = NdisAllocateFromNPagedLookasideList(&pAdapter->WorkItemsLList);
|
||
|
|
||
|
if (!pWork)
|
||
|
{
|
||
|
ASSERT( !"Alloc work?" );
|
||
|
return NDIS_STATUS_RESOURCES;
|
||
|
}
|
||
|
|
||
|
NdisInitializeWorkItem( pWork, pProc, pContext );
|
||
|
|
||
|
status = NdisScheduleWorkItem( pWork );
|
||
|
if (status != NDIS_STATUS_SUCCESS)
|
||
|
{
|
||
|
ASSERT( !"SchedWork?" );
|
||
|
NdisFreeToNPagedLookasideList(&pAdapter->WorkItemsLList, pWork);
|
||
|
}
|
||
|
|
||
|
return status;
|
||
|
}
|
||
|
|
||
|
|
||
|
#if DBG
|
||
|
|
||
|
NTSTATUS
|
||
|
DbgDispatch(
|
||
|
PDEVICE_OBJECT pDeviceObject,
|
||
|
PIRP pIrp)
|
||
|
{
|
||
|
PIO_STACK_LOCATION pIrpSp;
|
||
|
|
||
|
pIrpSp = IoGetCurrentIrpStackLocation(pIrp);
|
||
|
|
||
|
if (pIrpSp->MajorFunction != IRP_MJ_DEVICE_CONTROL)
|
||
|
{
|
||
|
pIrp->IoStatus.Status = STATUS_SUCCESS;
|
||
|
pIrp->IoStatus.Information = 0;
|
||
|
IoCompleteRequest(pIrp, IO_NETWORK_INCREMENT);
|
||
|
return STATUS_SUCCESS;
|
||
|
}
|
||
|
|
||
|
ASSERT(pIrpSp->MajorFunction != IRP_MJ_INTERNAL_DEVICE_CONTROL);
|
||
|
|
||
|
switch(pIrpSp->Parameters.DeviceIoControl.IoControlCode)
|
||
|
{
|
||
|
case IOCTL_IRDA_GET_DBG_MSGS:
|
||
|
{
|
||
|
NTSTATUS Status;
|
||
|
|
||
|
if ((Status = DbgMsgIrp(pIrp, pIrpSp)) == STATUS_PENDING)
|
||
|
{
|
||
|
return STATUS_PENDING;
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
case IOCTL_IRDA_GET_DBG_SETTINGS:
|
||
|
{
|
||
|
UINT *Settings = pIrp->AssociatedIrp.SystemBuffer;
|
||
|
|
||
|
Settings[0] = DbgSettings;
|
||
|
Settings[1] = DbgOutput;
|
||
|
|
||
|
pIrp->IoStatus.Information = sizeof(UINT)*2;
|
||
|
pIrp->IoStatus.Status = STATUS_SUCCESS;
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
case IOCTL_IRDA_SET_DBG_SETTINGS:
|
||
|
{
|
||
|
UINT *Settings = pIrp->AssociatedIrp.SystemBuffer;
|
||
|
|
||
|
DbgSettings = Settings[0];
|
||
|
DbgOutput = Settings[1];
|
||
|
|
||
|
pIrp->IoStatus.Information = 0;
|
||
|
pIrp->IoStatus.Status = STATUS_SUCCESS;
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
default:
|
||
|
pIrp->IoStatus.Information = 0;
|
||
|
pIrp->IoStatus.Status = STATUS_SUCCESS;
|
||
|
}
|
||
|
|
||
|
IoCompleteRequest(pIrp, IO_NETWORK_INCREMENT);
|
||
|
|
||
|
return STATUS_SUCCESS;
|
||
|
}
|
||
|
|
||
|
#endif
|