windows-nt/Source/XPSP1/NT/net/tcpip/driver/ipsec/sys/driver.c
2020-09-26 16:20:57 +08:00

3211 lines
82 KiB
C

/*++
Copyright (c) 1997-2001 Microsoft Corporation
Module Name:
driver.c
Abstract:
This module contains the DriverEntry and other initialization
code for the IPSEC module of the Tcpip transport.
Author:
Sanjay Anand (SanjayAn) 2-January-1997
ChunYe
Environment:
Kernel mode
Revision History:
--*/
#include "precomp.h"
#pragma hdrstop
#ifdef ALLOC_PRAGMA
#pragma alloc_text(INIT, DriverEntry)
#pragma alloc_text(INIT, IPSecGeneralInit)
#pragma alloc_text(PAGE, IPSecDispatch)
#pragma alloc_text(PAGE, IPSecBindToIP)
#pragma alloc_text(PAGE, IPSecUnbindFromIP)
#pragma alloc_text(PAGE, IPSecFreeConfig)
#pragma alloc_text(PAGE, IPSecInitMdlPool)
#pragma alloc_text(PAGE, AllocateCacheStructures)
#pragma alloc_text(PAGE, FreeExistingCache)
#pragma alloc_text(PAGE, FreePatternDbase)
#pragma alloc_text(PAGE, OpenRegKey)
#pragma alloc_text(PAGE, GetRegDWORDValue)
#pragma alloc_text(PAGE, IPSecCryptoInitialize)
#pragma alloc_text(PAGE, IPSecCryptoDeinitialize)
#endif
NTSTATUS
DriverEntry(
IN PDRIVER_OBJECT DriverObject,
IN PUNICODE_STRING RegistryPath
)
/*++
Routine Description:
This routine performs initialization of the IPSEC module.
It creates the device object for the transport
provider and performs other driver initialization.
Arguments:
DriverObject - Pointer to driver object created by the system.
RegistryPath - The name of IPSEC's node in the registry.
Return Value:
The function value is the final status from the initialization operation.
--*/
{
PDEVICE_OBJECT deviceObject = NULL;
WCHAR deviceNameBuffer[] = DD_IPSEC_DEVICE_NAME;
WCHAR symbolicLinkBuffer[] = DD_IPSEC_SYM_NAME;
UNICODE_STRING symbolicLinkName;
UNICODE_STRING deviceNameUnicodeString;
NTSTATUS status;
NTSTATUS status1;
//DbgBreakPoint();
IPSEC_DEBUG(LOAD, ("Entering DriverEntry\n"));
//
// Init g_ipsec structure and read reg keys.
//
IPSecZeroMemory(&g_ipsec, sizeof(g_ipsec));
IPSecReadRegistry();
//
// Create the device - do we need a device at all?
//
// Setup the handlers.
//
//
// Initialize the driver object with this driver's entry points.
//
g_ipsec.IPSecDriverObject = DriverObject;
DriverObject->MajorFunction [IRP_MJ_CREATE] =
DriverObject->MajorFunction [IRP_MJ_CLOSE] =
DriverObject->MajorFunction [IRP_MJ_CLEANUP] =
DriverObject->MajorFunction [IRP_MJ_INTERNAL_DEVICE_CONTROL] =
DriverObject->MajorFunction [IRP_MJ_DEVICE_CONTROL] = IPSecDispatch;
DriverObject->DriverUnload = IPSecUnload;
RtlInitUnicodeString (&deviceNameUnicodeString, deviceNameBuffer);
status = IoCreateDevice(
DriverObject,
0, // DeviceExtensionSize
&deviceNameUnicodeString, // DeviceName
FILE_DEVICE_NETWORK, // DeviceType
FILE_DEVICE_SECURE_OPEN, // DeviceCharacteristics
FALSE, // Exclusive
&deviceObject); // *DeviceObject
if (!NT_SUCCESS (status)) {
IPSEC_DEBUG(LOAD, ("Failed to create device: %lx\n", status));
LOG_EVENT(
DriverObject,
EVENT_IPSEC_CREATE_DEVICE_FAILED,
1,
1,
&deviceNameUnicodeString.Buffer,
0,
NULL);
goto err;
}
deviceObject->Flags |= DO_BUFFERED_IO;
IPSecDevice = deviceObject;
RtlInitUnicodeString (&symbolicLinkName, symbolicLinkBuffer);
status = IoCreateSymbolicLink(&symbolicLinkName, &deviceNameUnicodeString);
if (!NT_SUCCESS (status)) {
IPSEC_DEBUG(LOAD, ("Failed to create symbolic link: %lx\n", status));
LOG_EVENT(
DriverObject,
EVENT_IPSEC_CREATE_DEVICE_FAILED,
2,
1,
&deviceNameUnicodeString.Buffer,
0,
NULL);
IoDeleteDevice(DriverObject->DeviceObject);
goto err;
}
//
// General structs init here.
// Allocates the SA Table etc.
//
status = IPSecGeneralInit();
if (!NT_SUCCESS (status)) {
IPSEC_DEBUG(LOAD, ("Failed to init general structs: %lx\n", status));
//
// Free the general structs and SA Table etc.
//
status1 = IPSecGeneralFree();
if (!NT_SUCCESS (status1)) {
IPSEC_DEBUG(LOAD, ("Failed to free config: %lx\n", status1));
}
LOG_EVENT(
DriverObject,
EVENT_IPSEC_NO_RESOURCES_FOR_INIT,
1,
0,
NULL,
0,
NULL);
IoDeleteSymbolicLink(&symbolicLinkName);
IoDeleteDevice(DriverObject->DeviceObject);
goto err;
}
//
// Wait for TCP/IP to load and call IOCTL_IPSEC_SET_TCPIP_STATUS where we
// would finish the initialization.
//
status = STATUS_SUCCESS;
IPSEC_DEBUG(LOAD, ("Exiting DriverEntry; SUCCESS\n"));
err:
return status;
}
VOID
IPSecUnload(
IN PDRIVER_OBJECT DriverObject
)
/*++
Routine Description:
Called when the driver is unloaded.
Arguments:
DriverObject
Return Value:
None
--*/
{
UNICODE_STRING IPSecLinkName;
KIRQL OldIrq;
KIRQL kIrql;
NTSTATUS status;
INT class;
IPSEC_DEBUG(LOAD, ("Entering IPSecUnload\n"));
//
// Set IPSEC_DRIVER_UNLOADING bit.
//
IPSEC_DRIVER_UNLOADING() = TRUE;
//
// Stop the reaper timer.
//
IPSecStopTimer(&g_ipsec.ReaperTimer);
//
// Stop the EventLog timer.
//
IPSecStopTimer(&g_ipsec.EventLogTimer);
//
// Complete the Acquire Irp with error status
//
if (g_ipsec.AcquireInfo.Irp) {
IPSEC_DEBUG(ACQUIRE, ("Unload: Completing Irp..\n"));
if (g_ipsec.AcquireInfo.InMe) {
IoAcquireCancelSpinLock(&g_ipsec.AcquireInfo.Irp->CancelIrql);
IPSecAcquireIrpCancel(NULL, g_ipsec.AcquireInfo.Irp);
}
}
//
// Stop timers for all SAs (of all states)
//
IPSecStopSATimers();
//
// Wait for all timers to clear before going further
//
while (IPSEC_GET_VALUE(g_ipsec.NumTimers) != 0) {
IPSEC_DELAY_EXECUTION();
}
//
// Cleanup any larval SAs
//
AcquireWriteLock(&g_ipsec.SADBLock, &kIrql);
ACQUIRE_LOCK(&g_ipsec.AcquireInfo.Lock, &OldIrq);
IPSecFlushLarvalSAList();
IPSecFlushSAExpirations();
RELEASE_LOCK(&g_ipsec.AcquireInfo.Lock, OldIrq);
ReleaseWriteLock(&g_ipsec.SADBLock, kIrql);
//
// Free the SA Table.
//
status = IPSecFreeConfig();
if (!NT_SUCCESS (status)) {
IPSEC_DEBUG(LOAD, ("Failed to free config: %lx\n", status));
}
//
// Free the MDL pools and run down all buffered packets.
//
status = IPSecQuiesce();
if (!NT_SUCCESS (status)) {
IPSEC_DEBUG(LOAD, ("Failed to reach quiescent state: %lx\n", status));
}
//
// Destroy timer structures
//
ACQUIRE_LOCK(&g_ipsec.TimerLock, &kIrql);
for (class = 0; class < IPSEC_CLASS_MAX; class++) {
ASSERT(g_ipsec.TimerList[class].TimerCount == 0);
IPSecFreeMemory(g_ipsec.TimerList[class].pTimers);
}
RELEASE_LOCK(&g_ipsec.TimerLock, kIrql);
IPSecCryptoDeinitialize();
#if GPC
IPSecGpcDeinitialize();
#endif
RtlInitUnicodeString(&IPSecLinkName, DD_IPSEC_SYM_NAME);
IoDeleteSymbolicLink(&IPSecLinkName);
IoDeleteDevice(DriverObject->DeviceObject);
IPSEC_DEBUG(LOAD, ("Exiting IPSecUnload\n"));
}
NTSTATUS
IPSecDispatch(
IN PDEVICE_OBJECT DeviceObject,
IN PIRP Irp
)
/*++
Routine Description:
Dispatch Routine for the driver. Gets the current irp stack location, validates
the parameters and routes the calls
Arguments:
DeviceObject
Irp
Return Value:
Status as returned by the worker functions
--*/
{
PIO_STACK_LOCATION irpStack;
PVOID pvIoBuffer;
LONG inputBufferLength;
LONG outputBufferLength;
ULONG ioControlCode;
NTSTATUS status = STATUS_SUCCESS;
LONG dwSize = 0;
PAGED_CODE();
IPSEC_DEBUG(IOCTL, ("Entering IPSecDispath\n"));
Irp->IoStatus.Status = status;
Irp->IoStatus.Information = 0;
//
// Get a pointer to the current location in the Irp. This is where
// the function codes and parameters are located.
//
irpStack = IoGetCurrentIrpStackLocation(Irp);
//
// Get the pointer to the input/output buffer and its length.
//
pvIoBuffer = Irp->AssociatedIrp.SystemBuffer;
inputBufferLength = irpStack->Parameters.DeviceIoControl.InputBufferLength;
outputBufferLength = irpStack->Parameters.DeviceIoControl.OutputBufferLength;
switch (irpStack->MajorFunction) {
case IRP_MJ_CREATE: {
IPSEC_DEBUG(IOCTL, ("IRP_MJ_CREATE\n"));
break;
}
case IRP_MJ_CLOSE: {
IPSEC_DEBUG(IOCTL, ("IRP_MJ_CLOSE\n"));
break;
}
case IRP_MJ_DEVICE_CONTROL: {
ioControlCode = irpStack->Parameters.DeviceIoControl.IoControlCode;
IPSEC_DEBUG(IOCTL, ("IRP_MJ_DEVICE_CONTROL: %lx\n", ioControlCode));
if (ioControlCode != IOCTL_IPSEC_SET_TCPIP_STATUS) {
if (IPSEC_DRIVER_IS_INACTIVE()) {
status = STATUS_INVALID_DEVICE_STATE;
break;
}
if (!IPSecCryptoInitialize()) {
status = STATUS_CRYPTO_SYSTEM_INVALID;
break;
}
}
IPSEC_INCREMENT(g_ipsec.NumIoctls);
switch (ioControlCode) {
case IOCTL_IPSEC_ADD_FILTER: {
PIPSEC_ADD_FILTER pAddFilter = (PIPSEC_ADD_FILTER)pvIoBuffer;
IPSEC_DEBUG(IOCTL, ("IOCTL_IPSEC_ADD_FILTER\n"));
dwSize = sizeof(IPSEC_ADD_FILTER);
if (inputBufferLength < dwSize) {
status = STATUS_BUFFER_TOO_SMALL;
break;
}
//
// Check the size of the entry
//
if (pAddFilter->NumEntries == 0) {
status = STATUS_SUCCESS;
} else {
dwSize = FIELD_OFFSET(IPSEC_ADD_FILTER, pInfo[0]) +
pAddFilter->NumEntries * sizeof(IPSEC_FILTER_INFO);
if (dwSize < FIELD_OFFSET(IPSEC_ADD_FILTER, pInfo[0]) ||
inputBufferLength < dwSize) {
status = STATUS_BUFFER_TOO_SMALL;
IPSEC_DEBUG(IOCTL, ("returning: %lx\n", status));
break;
}
status = IPSecAddFilter(pAddFilter);
}
break;
}
case IOCTL_IPSEC_DELETE_FILTER: {
PIPSEC_DELETE_FILTER pDelFilter = (PIPSEC_DELETE_FILTER)pvIoBuffer;
IPSEC_DEBUG(IOCTL, ("IOCTL_IPSEC_DELETE_FILTER\n"));
dwSize = sizeof(IPSEC_DELETE_FILTER);
if (inputBufferLength < dwSize) {
status = STATUS_BUFFER_TOO_SMALL;
break;
}
//
// Check the size of the entry
//
if (pDelFilter->NumEntries == 0) {
status = STATUS_SUCCESS;
} else {
dwSize = FIELD_OFFSET(IPSEC_DELETE_FILTER, pInfo[0]) +
pDelFilter->NumEntries * sizeof(IPSEC_FILTER_INFO);
if (dwSize < FIELD_OFFSET(IPSEC_DELETE_FILTER, pInfo[0]) ||
inputBufferLength < dwSize) {
status = STATUS_BUFFER_TOO_SMALL;
break;
}
status = IPSecDeleteFilter(pDelFilter);
}
break;
}
case IOCTL_IPSEC_ENUM_SAS: {
IPSEC_DEBUG(IOCTL, ("IOCTL_IPSEC_ENUM_SAS\n"));
dwSize = sizeof(IPSEC_ENUM_SAS);
//
// Output/Input in the same buffer at MdlAddress
//
status = IPSecEnumSAs(Irp, &dwSize);
break;
}
case IOCTL_IPSEC_ENUM_FILTERS: {
IPSEC_DEBUG(IOCTL, ("IOCTL_IPSEC_ENUM_FILTERS\n"));
dwSize = sizeof(IPSEC_ENUM_FILTERS);
//
// Output/Input in the same buffer at MdlAddress
//
status = IPSecEnumFilters(Irp, &dwSize);
break;
}
case IOCTL_IPSEC_QUERY_STATS: {
//
// The minimum size is without any Keys
//
IPSEC_DEBUG(IOCTL, ("IOCTL_IPSEC_QUERY_STATS\n"));
dwSize = sizeof(IPSEC_QUERY_STATS);
if (outputBufferLength < dwSize) {
status = STATUS_BUFFER_TOO_SMALL;
break;
}
*((PIPSEC_QUERY_STATS)pvIoBuffer) = g_ipsec.Statistics;
status = STATUS_SUCCESS;
break;
}
case IOCTL_IPSEC_ADD_SA: {
//
// Adds the SA to the relevant database.
// Typically used to add outbound SAs to the DB.
//
IPSEC_DEBUG(IOCTL, ("IOCTL_IPSEC_ADD_SA\n"));
//
// The minimum size is without any Keys
//
dwSize = IPSEC_ADD_SA_NO_KEY_SIZE;
if (inputBufferLength < dwSize) {
status = STATUS_BUFFER_TOO_SMALL;
break;
}
status = IPSecAddSA((PIPSEC_ADD_SA)pvIoBuffer, inputBufferLength);
ASSERT(status != STATUS_PENDING);
if (outputBufferLength >= sizeof(NTSTATUS)) {
(*(NTSTATUS *)pvIoBuffer) = status;
dwSize = sizeof(NTSTATUS);
} else {
dwSize = 0;
}
status = STATUS_SUCCESS;
break;
}
case IOCTL_IPSEC_UPDATE_SA: {
//
// This completes the negotiation kicked off via the Acquire.
//
// Adds the SA to the relevant database.
// Typically used to complete inbound SA acquisitions.
//
IPSEC_DEBUG(IOCTL, ("IOCTL_IPSEC_UPDATE_SA\n"));
//
// The minimum size is without any Keys
//
dwSize = IPSEC_UPDATE_SA_NO_KEY_SIZE;
if (inputBufferLength < dwSize) {
status = STATUS_BUFFER_TOO_SMALL;
break;
}
status = IPSecUpdateSA((PIPSEC_UPDATE_SA)pvIoBuffer, inputBufferLength);
ASSERT(status != STATUS_PENDING);
if (outputBufferLength >= sizeof(NTSTATUS)) {
(*(NTSTATUS *)pvIoBuffer)=status;
dwSize = sizeof(NTSTATUS);
} else {
dwSize = 0;
}
status = STATUS_SUCCESS;
break;
}
case IOCTL_IPSEC_EXPIRE_SA: {
//
// Deref the particular SA - delete when ref cnt drops to 0.
//
IPSEC_DEBUG(IOCTL, ("IOCTL_IPSEC_EXPIRE_SA\n"));
dwSize = sizeof(IPSEC_EXPIRE_SA);
if (inputBufferLength < dwSize) {
status = STATUS_BUFFER_TOO_SMALL;
break;
}
status = IPSecExpireSA((PIPSEC_EXPIRE_SA)pvIoBuffer);
break;
}
case IOCTL_IPSEC_GET_SPI: {
//
// returns the SPI for an inbound SA
//
IPSEC_DEBUG(IOCTL, ("IOCTL_IPSEC_GET_SPI\n"));
dwSize = sizeof(IPSEC_GET_SPI);
if (outputBufferLength < dwSize) {
status = STATUS_BUFFER_TOO_SMALL;
break;
}
status = IPSecGetSPI((PIPSEC_GET_SPI)pvIoBuffer);
break;
}
case IOCTL_IPSEC_POST_FOR_ACQUIRE_SA: {
//
// The SAAPI client posts a request that we complete when
// an SA needs to be initialized or updated (due to
// re-key).
// We keep the Irp around until we need an SA to be
// negotiated.
//
IPSEC_DEBUG(IOCTL, ("IPSEC_POST_FOR_ACQUIRE_SA\n"));
dwSize = sizeof(IPSEC_POST_FOR_ACQUIRE_SA);
if (outputBufferLength < dwSize) {
IPSEC_DEBUG(IOCTL, ("IPSEC_POST_FOR_ACQUIRE_SA: bad size: dwSize: %lx, input: %lx\n",
dwSize, inputBufferLength));
status = STATUS_BUFFER_TOO_SMALL;
break;
}
Irp->IoStatus.Status = STATUS_PENDING;
status = IPSecHandleAcquireRequest( Irp,
(PIPSEC_POST_FOR_ACQUIRE_SA)pvIoBuffer);
if (status == STATUS_PENDING) {
IPSEC_DECREMENT(g_ipsec.NumIoctls);
return status;
}
break;
}
case IOCTL_IPSEC_QUERY_EXPORT: {
//
// Queries whether the driver is built for export. Used by the IPSEC components
// to decide what key lengths to use for encryption.
//
IPSEC_DEBUG(IOCTL, ("IPSEC_QUERY_EXPORT\n"));
dwSize = sizeof(IPSEC_QUERY_EXPORT);
if (outputBufferLength < dwSize) {
IPSEC_DEBUG(IOCTL, ("IPSEC_QUERY_EXPORT: bad size: dwSize: %lx, input: %lx\n",
dwSize, inputBufferLength));
status = STATUS_BUFFER_TOO_SMALL;
break;
}
((PIPSEC_QUERY_EXPORT)pvIoBuffer)->Export = FALSE;
status = STATUS_SUCCESS;
break;
}
case IOCTL_IPSEC_QUERY_SPI: {
IPSEC_DEBUG(IOCTL, ("Entered Query SPI\n"));
dwSize = sizeof(IPSEC_QUERY_SPI);
if (inputBufferLength < dwSize) {
IPSEC_DEBUG(IOCTL, ("IPSEC_QUERY_SPI: bad size: dwSize: %lx, input: %lx\n",
dwSize, inputBufferLength));
status = STATUS_BUFFER_TOO_SMALL;
break;
}
status = IPSecQuerySpi((PIPSEC_QUERY_SPI)pvIoBuffer);
break;
}
case IOCTL_IPSEC_DELETE_SA: {
IPSEC_DEBUG(IOCTL, ("Entered Delete SA\n"));
dwSize = sizeof(IPSEC_DELETE_SA);
if (inputBufferLength < dwSize) {
IPSEC_DEBUG(IOCTL, ("IPSEC_DELETE_SA: bad size: dwSize: %lx, input: %lx\n",
dwSize, inputBufferLength));
status = STATUS_BUFFER_TOO_SMALL;
break;
}
status = IPSecDeleteSA((PIPSEC_DELETE_SA)pvIoBuffer);
break;
}
case IOCTL_IPSEC_SET_OPERATION_MODE: {
IPSEC_DEBUG(IOCTL, ("Entered Set Operation Mode\n"));
dwSize = sizeof(IPSEC_SET_OPERATION_MODE);
if (inputBufferLength < dwSize) {
IPSEC_DEBUG(IOCTL, ("IPSEC_SET_OPERATION_MODE: bad size: dwSize: %lx, input: %lx\n",
dwSize, inputBufferLength));
status = STATUS_BUFFER_TOO_SMALL;
break;
}
status = IPSecSetOperationMode((PIPSEC_SET_OPERATION_MODE)pvIoBuffer);
break;
}
case IOCTL_IPSEC_SET_TCPIP_STATUS: {
IPSEC_DEBUG(IOCTL, ("Entered Set Tcpip Status\n"));
if (Irp->RequestorMode != KernelMode) {
status = STATUS_ACCESS_DENIED;
break;
}
dwSize = sizeof(IPSEC_SET_TCPIP_STATUS);
if (inputBufferLength < dwSize) {
IPSEC_DEBUG(IOCTL, ("IPSEC_SET_TCPIP_STATUS: bad size: dwSize: %lx, input: %lx\n",
dwSize, inputBufferLength));
status = STATUS_BUFFER_TOO_SMALL;
break;
}
status = IPSecSetTcpipStatus((PIPSEC_SET_TCPIP_STATUS)pvIoBuffer);
break;
}
case IOCTL_IPSEC_REGISTER_PROTOCOL: {
dwSize = sizeof(IPSEC_REGISTER_PROTOCOL);
if (inputBufferLength < dwSize) {
status = STATUS_BUFFER_TOO_SMALL;
break;
}
status = IPSecRegisterProtocols(
(PIPSEC_REGISTER_PROTOCOL) pvIoBuffer
);
break;
}
default: {
status = STATUS_INVALID_PARAMETER;
break;
}
}
IPSEC_DECREMENT(g_ipsec.NumIoctls);
break ;
}
default: {
IPSEC_DEBUG(IOCTL, ("IPSEC: unknown IRP_MJ_XXX: %lx\n", irpStack->MajorFunction));
status = STATUS_INVALID_PARAMETER;
break;
}
}
ASSERT(status != STATUS_PENDING);
ASSERT(KeGetCurrentIrql() <= APC_LEVEL);
Irp->IoStatus.Status = status;
Irp->IoStatus.Information = MIN(dwSize, outputBufferLength);
IoCompleteRequest(Irp, IO_NO_INCREMENT);
IPSEC_DEBUG(IOCTL, ("Exiting IPSecDispath\n"));
return status;
}
NTSTATUS
IPSecBindToIP()
/*++
Routine Description:
This bind exchanges a number of entrypoints with IP so that
- packets relevant to IPSEC can be handed over from the IP driver.
- buffered packets can be flushed.
- SA Table indices can be plumbed.
- ....
Arguments:
NONE
Return Value:
The function value is the final status from the bind operation.
--*/
{
NTSTATUS status;
IPSEC_FUNCTIONS ipsecFns;
PAGED_CODE();
IPSEC_DEBUG(LOAD, ("Entering IPSecBindToIP\n"));
ipsecFns.Version = IP_IPSEC_BIND_VERSION;
ipsecFns.IPSecHandler = IPSecHandlePacket;
ipsecFns.IPSecQStatus = IPSecQueryStatus;
ipsecFns.IPSecSendCmplt = IPSecSendComplete;
ipsecFns.IPSecNdisStatus = IPSecNdisStatus;
ipsecFns.IPSecRcvFWPacket = IPSecRcvFWPacket;
status = TCPIP_SET_IPSEC(&ipsecFns);
if (status != IP_SUCCESS) {
IPSEC_DEBUG(LOAD, ("Failed to bind to IP: %lx\n", status));
} else {
IPSEC_DRIVER_BOUND() = TRUE;
IPSEC_DRIVER_SEND_BOUND() = TRUE;
}
IPSEC_DEBUG(LOAD, ("Exiting IPSecBindToIP\n"));
return status;
}
NTSTATUS
IPSecUnbindFromIP()
/*++
Routine Description:
This unbinds from the Filter Driver
Arguments:
NONE
Return Value:
The function value is the final status from the bind operation.
--*/
{
NTSTATUS status;
IPSEC_FUNCTIONS ipsecFns={0};
PAGED_CODE();
IPSEC_DEBUG(LOAD, ("Entering IPSecUnbindFromIP\n"));
ipsecFns.Version = IP_IPSEC_BIND_VERSION;
status = TCPIP_UNSET_IPSEC(&ipsecFns);
if (status != IP_SUCCESS) {
IPSEC_DEBUG(LOAD, ("Failed to bind to IP: %lx\n", status));
} else {
IPSEC_DRIVER_BOUND() = FALSE;
}
IPSEC_DEBUG(LOAD, ("Exiting IPSecUnbindFromIP\n"));
return status;
}
NTSTATUS
IPSecUnbindSendFromIP()
/*++
Routine Description:
Unbinds just the send handler from IP
Arguments:
NONE
Return Value:
The function value is the final status from the bind operation.
--*/
{
NTSTATUS status;
IPSEC_FUNCTIONS ipsecFns={0};
PAGED_CODE();
IPSEC_DEBUG(LOAD, ("Entering IPSecUnbindSendFromIP\n"));
ipsecFns.Version = IP_IPSEC_BIND_VERSION;
status = TCPIP_UNSET_IPSEC_SEND(&ipsecFns);
if (status != IP_SUCCESS) {
IPSEC_DEBUG(LOAD, ("Failed to bind to IP: %lx\n", status));
} else {
IPSEC_DRIVER_SEND_BOUND() = FALSE;
}
IPSEC_DEBUG(LOAD, ("Exiting IPSecUnbindSendFromIP\n"));
return status;
}
NTSTATUS
OpenRegKey(
PHANDLE HandlePtr,
PWCHAR KeyName
)
/*++
Routine Description:
Opens a Registry key and returns a handle to it.
Arguments:
HandlePtr - The varible into which to write the opened handle.
KeyName - The name of the Registry key to open.
Return Value:
STATUS_SUCCESS or an appropriate failure code.
--*/
{
NTSTATUS Status;
OBJECT_ATTRIBUTES ObjectAttributes;
UNICODE_STRING UKeyName;
PAGED_CODE();
RtlInitUnicodeString(&UKeyName, KeyName);
memset(&ObjectAttributes, 0, sizeof(OBJECT_ATTRIBUTES));
InitializeObjectAttributes(&ObjectAttributes,
&UKeyName,
OBJ_CASE_INSENSITIVE,
NULL,
NULL);
Status = ZwOpenKey(HandlePtr,
KEY_READ,
&ObjectAttributes);
return Status;
}
NTSTATUS
GetRegDWORDValue(
HANDLE KeyHandle,
PWCHAR ValueName,
PULONG ValueData
)
/*++
Routine Description:
Reads a REG_DWORD value from the registry into the supplied variable.
Arguments:
KeyHandle - Open handle to the parent key of the value to read.
ValueName - The name of the value to read.
ValueData - The variable into which to read the data.
Return Value:
STATUS_SUCCESS or an appropriate failure code.
--*/
{
NTSTATUS status;
ULONG resultLength;
PKEY_VALUE_FULL_INFORMATION keyValueFullInformation;
UCHAR keybuf[WORK_BUFFER_SIZE];
UNICODE_STRING UValueName;
PAGED_CODE();
RtlInitUnicodeString(&UValueName, ValueName);
keyValueFullInformation = (PKEY_VALUE_FULL_INFORMATION)keybuf;
RtlZeroMemory(keyValueFullInformation, sizeof(keyValueFullInformation));
status = ZwQueryValueKey(KeyHandle,
&UValueName,
KeyValueFullInformation,
keyValueFullInformation,
WORK_BUFFER_SIZE,
&resultLength);
if (NT_SUCCESS(status)) {
if (keyValueFullInformation->Type != REG_DWORD) {
status = STATUS_INVALID_PARAMETER_MIX;
} else {
*ValueData = *((ULONG UNALIGNED *)((PCHAR)keyValueFullInformation +
keyValueFullInformation->DataOffset));
}
}
return status;
}
NTSTATUS
GetRegStringValue(
HANDLE KeyHandle,
PWCHAR ValueName,
PKEY_VALUE_PARTIAL_INFORMATION *ValueData,
PUSHORT ValueSize
)
/*++
Routine Description:
Reads a REG_*_SZ string value from the Registry into the supplied
key value buffer. If the buffer string buffer is not large enough,
it is reallocated.
Arguments:
KeyHandle - Open handle to the parent key of the value to read.
ValueName - The name of the value to read.
ValueData - Destination for the read data.
ValueSize - Size of the ValueData buffer. Updated on output.
Return Value:
STATUS_SUCCESS or an appropriate failure code.
--*/
{
NTSTATUS status;
ULONG resultLength;
UNICODE_STRING UValueName;
PAGED_CODE();
RtlInitUnicodeString(&UValueName, ValueName);
status = ZwQueryValueKey(
KeyHandle,
&UValueName,
KeyValuePartialInformation,
*ValueData,
(ULONG) *ValueSize,
&resultLength
);
if ( (status == STATUS_BUFFER_OVERFLOW) ||
(status == STATUS_BUFFER_TOO_SMALL)
)
{
PVOID temp;
//
// Free the old buffer and allocate a new one of the
// appropriate size.
//
ASSERT(resultLength > (ULONG) *ValueSize);
if (resultLength <= 0xFFFF) {
temp = IPSecAllocateMemory(resultLength, IPSEC_TAG_IOCTL);
if (temp != NULL) {
if (*ValueData != NULL) {
IPSecFreeMemory(*ValueData);
}
*ValueData = temp;
*ValueSize = (USHORT) resultLength;
status = ZwQueryValueKey(KeyHandle,
&UValueName,
KeyValuePartialInformation,
*ValueData,
*ValueSize,
&resultLength
);
ASSERT( (status != STATUS_BUFFER_OVERFLOW) &&
(status != STATUS_BUFFER_TOO_SMALL)
);
}
else {
status = STATUS_INSUFFICIENT_RESOURCES;
}
}
else {
status = STATUS_BUFFER_TOO_SMALL;
}
}
return status;
}
NTSTATUS
GetRegMultiSZValue(
HANDLE KeyHandle,
PWCHAR ValueName,
PUNICODE_STRING ValueData
)
/*++
Routine Description:
Reads a REG_MULTI_SZ string value from the Registry into the supplied
Unicode string. If the Unicode string buffer is not large enough,
it is reallocated.
Arguments:
KeyHandle - Open handle to the parent key of the value to read.
ValueName - The name of the value to read.
ValueData - Destination Unicode string for the value data.
Return Value:
STATUS_SUCCESS or an appropriate failure code.
--*/
{
NTSTATUS status;
ULONG resultLength;
PKEY_VALUE_PARTIAL_INFORMATION keyValuePartialInformation;
UNICODE_STRING UValueName;
PAGED_CODE();
ValueData->Length = 0;
status = GetRegStringValue(
KeyHandle,
ValueName,
(PKEY_VALUE_PARTIAL_INFORMATION *) &(ValueData->Buffer),
&(ValueData->MaximumLength)
);
if (NT_SUCCESS(status)) {
keyValuePartialInformation =
(PKEY_VALUE_PARTIAL_INFORMATION) ValueData->Buffer;
if (keyValuePartialInformation->Type == REG_MULTI_SZ) {
ValueData->Length = (USHORT)
keyValuePartialInformation->DataLength;
RtlCopyMemory(
ValueData->Buffer,
&(keyValuePartialInformation->Data),
ValueData->Length
);
}
else {
status = STATUS_INVALID_PARAMETER_MIX;
}
}
return status;
} // GetRegMultiSZValue
VOID
IPSecReadRegistry()
/*++
Routine Description:
Reads config info from registry into g_ipsec
Arguments:
Return Value:
status of the read.
--*/
{
NTSTATUS status;
HANDLE hRegKey;
WCHAR IPSecParametersRegistryKey[] = IPSEC_REG_KEY;
BOOLEAN isAs = MmIsThisAnNtAsSystem();
g_ipsec.EnableOffload = IPSEC_DEFAULT_ENABLE_OFFLOAD;
g_ipsec.DefaultSAIdleTime = IPSEC_DEFAULT_SA_IDLE_TIME;
g_ipsec.LogInterval = IPSEC_DEFAULT_LOG_INTERVAL;
g_ipsec.EventQueueSize = IPSEC_DEFAULT_EVENT_QUEUE_SIZE;
g_ipsec.RekeyTime = IPSEC_DEFAULT_REKEY;
g_ipsec.NoDefaultExempt = IPSEC_DEFAULT_NO_DEFAULT_EXEMPT;
g_ipsec.NoDefaultExempt = DEFAULT_IPSEC_OPERATION_MODE;
g_ipsec.DiagnosticMode = IPSEC_DEFAULT_ENABLE_DIAGNOSTICS;
if (isAs) {
g_ipsec.CacheSize = IPSEC_DEFAULT_AS_CACHE_SIZE;
g_ipsec.SAHashSize = IPSEC_DEFAULT_AS_SA_HASH_SIZE;
} else {
g_ipsec.CacheSize = IPSEC_DEFAULT_CACHE_SIZE;
g_ipsec.SAHashSize = IPSEC_DEFAULT_SA_HASH_SIZE;
}
status = OpenRegKey(&hRegKey,
IPSecParametersRegistryKey);
if (NT_SUCCESS(status)) {
//
// Expected configuration values. We use reasonable defaults if they
// aren't available for some reason.
//
IPSEC_REG_READ_DWORD( hRegKey,
IPSEC_REG_PARAM_ENABLE_OFFLOAD,
&g_ipsec.EnableOffload,
IPSEC_DEFAULT_ENABLE_OFFLOAD,
IPSEC_MAX_ENABLE_OFFLOAD,
IPSEC_MIN_ENABLE_OFFLOAD);
IPSEC_REG_READ_DWORD( hRegKey,
IPSEC_REG_PARAM_SA_IDLE_TIME,
&g_ipsec.DefaultSAIdleTime,
IPSEC_DEFAULT_SA_IDLE_TIME,
IPSEC_MAX_SA_IDLE_TIME,
IPSEC_MIN_SA_IDLE_TIME);
IPSEC_REG_READ_DWORD( hRegKey,
IPSEC_REG_PARAM_EVENT_QUEUE_SIZE,
&g_ipsec.EventQueueSize,
IPSEC_DEFAULT_EVENT_QUEUE_SIZE,
IPSEC_MAX_EVENT_QUEUE_SIZE,
IPSEC_MIN_EVENT_QUEUE_SIZE);
IPSEC_REG_READ_DWORD( hRegKey,
IPSEC_REG_PARAM_LOG_INTERVAL,
&g_ipsec.LogInterval,
IPSEC_DEFAULT_LOG_INTERVAL,
IPSEC_MAX_LOG_INTERVAL,
IPSEC_MIN_LOG_INTERVAL);
IPSEC_REG_READ_DWORD( hRegKey,
IPSEC_REG_PARAM_REKEY_TIME,
&g_ipsec.RekeyTime,
IPSEC_DEFAULT_REKEY,
IPSEC_MAX_REKEY,
IPSEC_MIN_REKEY);
IPSEC_REG_READ_DWORD( hRegKey,
IPSEC_REG_PARAM_CACHE_SIZE,
&g_ipsec.CacheSize,
isAs? IPSEC_DEFAULT_AS_CACHE_SIZE:
IPSEC_DEFAULT_CACHE_SIZE,
IPSEC_MAX_CACHE_SIZE,
IPSEC_MIN_CACHE_SIZE);
IPSEC_REG_READ_DWORD( hRegKey,
IPSEC_REG_PARAM_SA_HASH_SIZE,
&g_ipsec.SAHashSize,
isAs? IPSEC_DEFAULT_AS_SA_HASH_SIZE:
IPSEC_DEFAULT_SA_HASH_SIZE,
IPSEC_MAX_SA_HASH_SIZE,
IPSEC_MIN_SA_HASH_SIZE);
IPSEC_REG_READ_DWORD( hRegKey,
IPSEC_REG_PARAM_NO_DEFAULT_EXEMPT,
&g_ipsec.NoDefaultExempt,
IPSEC_DEFAULT_NO_DEFAULT_EXEMPT,
IPSEC_MAX_NO_DEFAULT_EXEMPT,
IPSEC_MIN_NO_DEFAULT_EXEMPT);
IPSEC_REG_READ_DWORD( hRegKey,
IPSEC_REG_PARAM_ENABLE_DIAGNOSTICS,
&g_ipsec.DiagnosticMode,
IPSEC_DEFAULT_ENABLE_DIAGNOSTICS,
IPSEC_MAX_ENABLE_DIAGNOSTICS,
IPSEC_MIN_ENABLE_DIAGNOSTICS);
IPSEC_REG_READ_DWORD( hRegKey,
IPSEC_REG_PARAM_OPERATION_MODE,
&(ULONG)g_ipsec.OperationMode,
DEFAULT_IPSEC_OPERATION_MODE,
IPSEC_OPERATION_MODE_MAX-1,
0);
ZwClose(hRegKey);
}
g_ipsec.CacheHalfSize = g_ipsec.CacheSize / 2;
//
// Init SAIdleTime for low memory reaper
//
IPSEC_CONVERT_SECS_TO_100NS(g_ipsec.SAIdleTime, g_ipsec.DefaultSAIdleTime);
}
NTSTATUS
IPSecGeneralInit()
/*++
Routine Description:
General structures are initialized here.
Arguments:
None
Return Value:
--*/
{
PSA_TABLE_ENTRY pSA;
LONG i;
NTSTATUS status = STATUS_SUCCESS;;
PAGED_CODE();
IPSEC_DEBUG(LOAD, ("Entering IPSecGeneralInit\n"));
//
// init the acquireinfo struct
//
InitializeListHead(&g_ipsec.AcquireInfo.PendingAcquires);
InitializeListHead(&g_ipsec.AcquireInfo.PendingNotifies);
InitializeListHead(&g_ipsec.LarvalSAList);
INIT_LOCK(&g_ipsec.LarvalListLock);
INIT_LOCK(&g_ipsec.AcquireInfo.Lock);
//
// Set up the hashes/tables
//
InitializeMRSWLock(&g_ipsec.SADBLock);
InitializeMRSWLock(&g_ipsec.SPIListLock);
g_ipsec.IPProtInfo.pi_xmitdone = IPSecProtocolSendComplete;
g_ipsec.IPProtInfo.pi_protocol = PROTOCOL_ESP;
//
// init filter linked lists
//
for (i = MIN_FILTER; i <= MAX_FILTER; i++) {
InitializeListHead(&g_ipsec.FilterList[i]);
}
//
// SAs in a hash table, hashed by <SPI, Dest addr>
//
g_ipsec.pSADb = IPSecAllocateMemory(g_ipsec.SAHashSize * sizeof(SA_HASH), IPSEC_TAG_INIT);
if (!g_ipsec.pSADb) {
IPSEC_DEBUG(LOAD, ("Failed to alloc SADb hash\n"));
return STATUS_INSUFFICIENT_RESOURCES;
}
IPSecInitFlag |= INIT_SA_DATABASE;
IPSecZeroMemory(g_ipsec.pSADb, g_ipsec.SAHashSize * sizeof(SA_HASH));
for (i = 0; i < g_ipsec.SAHashSize; i++) {
PSA_HASH Entry = &g_ipsec.pSADb[i];
InitializeListHead(&Entry->SAList);
}
//
// Initialize the MDL pools.
//
status = IPSecInitMdlPool();
if (!NT_SUCCESS (status)) {
IPSEC_DEBUG(LOAD, ("Failed to alloc MDL pools\n"));
return STATUS_INSUFFICIENT_RESOURCES;
}
IPSecInitFlag |= INIT_MDL_POOLS;
//
// Initialize the cache structures.
//
if (!AllocateCacheStructures()) {
IPSEC_DEBUG(LOAD, ("Failed to alloc cache structs\n"));
return STATUS_INSUFFICIENT_RESOURCES;
}
IPSecInitFlag |= INIT_CACHE_STRUCT;
//
// Allocate EventQueue memory.
//
g_ipsec.IPSecLogMemory = IPSecAllocateMemory( g_ipsec.EventQueueSize * sizeof(IPSEC_EVENT_CTX),
IPSEC_TAG_EVT_QUEUE);
if (!g_ipsec.IPSecLogMemory) {
return STATUS_INSUFFICIENT_RESOURCES;
}
IPSecInitFlag |= INIT_DEBUG_MEMORY;
g_ipsec.IPSecLogMemoryLoc = &g_ipsec.IPSecLogMemory[0];
g_ipsec.IPSecLogMemoryEnd = &g_ipsec.IPSecLogMemory[g_ipsec.EventQueueSize * sizeof(IPSEC_EVENT_CTX)];
//
// Init the timer stuff.
//
if (!IPSecInitTimer()) {
IPSEC_DEBUG(LOAD, ("Failed to init timer\n"));
return STATUS_INSUFFICIENT_RESOURCES;
}
IPSecInitFlag |= INIT_TIMERS;
#if GPC
status = IPSecGpcInitialize();
if (status != STATUS_SUCCESS) {
IPSEC_DEBUG(LOAD, ("Failed to register GPC clients\n"));
}
#endif
//
// Arm the reaper timer
//
IPSEC_DEBUG(LOAD, ("Starting ReaperTimer\n"));
IPSecStartTimer(&g_ipsec.ReaperTimer,
IPSecReaper,
IPSEC_REAPER_TIME,
(PVOID)NULL);
//
// Start EventLog timer
//
IPSEC_DEBUG(LOAD, ("Starting EventLogTimer\n"));
IPSecStartTimer(&g_ipsec.EventLogTimer,
IPSecFlushEventLog,
g_ipsec.LogInterval,
(PVOID)NULL);
IPSEC_DEBUG(LOAD, ("Exiting IPSecGeneralInit\n"));
return STATUS_SUCCESS;
}
NTSTATUS
IPSecGeneralFree()
/*++
Routine Description:
Free general structures if IPSecGeneralInit fails.
Arguments:
None
Return Value:
--*/
{
INT index;
//
// Free SA database.
//
if (IPSecInitFlag & INIT_SA_DATABASE) {
if (g_ipsec.pSADb) {
IPSecFreeMemory(g_ipsec.pSADb);
}
}
//
// Free MDL pool.
//
if (IPSecInitFlag & INIT_MDL_POOLS) {
IPSecDeinitMdlPool();
}
//
// Free cache struct.
//
if (IPSecInitFlag & INIT_CACHE_STRUCT) {
FreeExistingCache();
}
//
// Free EventQueue memory.
//
if (IPSecInitFlag & INIT_DEBUG_MEMORY) {
if (g_ipsec.IPSecLogMemory) {
IPSecFreeMemory(g_ipsec.IPSecLogMemory);
}
}
//
// Free timers allocated.
//
if (IPSecInitFlag & INIT_TIMERS) {
for (index = 0; index < IPSEC_CLASS_MAX; index++) {
IPSecFreeMemory(g_ipsec.TimerList[index].pTimers);
}
}
return STATUS_SUCCESS;
}
NTSTATUS
IPSecFreeConfig()
/*++
Routine Description:
Free the SA table etc.
Arguments:
None
Return Value:
--*/
{
IPSEC_DEBUG(LOAD, ("Entering IPSecFreeConfig\n"));
PAGED_CODE();
FreeExistingCache();
FreePatternDbase();
if (g_ipsec.IPSecLogMemory) {
IPSecFreeMemory(g_ipsec.IPSecLogMemory);
}
IPSEC_DEBUG(LOAD, ("Exiting IPSecFreeConfig\n"));
return STATUS_SUCCESS;
}
NTSTATUS
IPSecInitMdlPool()
/*++
Routine Description:
Create the MDL pool for AH and ESP headers.
Arguments:
None
Return Value:
--*/
{
PAGED_CODE();
IPSEC_DEBUG(LOAD, ("Entering IPSecInitMdlPool\n"));
g_ipsec.IPSecSmallBufferSize = IPSEC_SMALL_BUFFER_SIZE;
g_ipsec.IPSecLargeBufferSize = IPSEC_LARGE_BUFFER_SIZE;
g_ipsec.IPSecSendCompleteCtxSize = sizeof(IPSEC_SEND_COMPLETE_CONTEXT);
g_ipsec.IPSecSmallBufferListDepth = IPSEC_LIST_DEPTH;
g_ipsec.IPSecLargeBufferListDepth = IPSEC_LIST_DEPTH;
g_ipsec.IPSecSendCompleteCtxDepth = IPSEC_LIST_DEPTH;
g_ipsec.IPSecCacheLineSize = IPSEC_CACHE_LINE_SIZE;
//
// Initialize the lookaside lists.
//
g_ipsec.IPSecLookasideLists = IPSecAllocateMemory(
sizeof(*g_ipsec.IPSecLookasideLists),
IPSEC_TAG_LOOKASIDE_LISTS);
if (g_ipsec.IPSecLookasideLists == NULL) {
return STATUS_INSUFFICIENT_RESOURCES;
}
//
// Initialize the IPSEC buffer lookaside lists.
//
ExInitializeNPagedLookasideList(&g_ipsec.IPSecLookasideLists->LargeBufferList,
IPSecAllocateBufferPool,
NULL,
0,
g_ipsec.IPSecLargeBufferSize,
IPSEC_TAG_BUFFER_POOL,
(USHORT)g_ipsec.IPSecLargeBufferListDepth);
ExInitializeNPagedLookasideList(&g_ipsec.IPSecLookasideLists->SmallBufferList,
IPSecAllocateBufferPool,
NULL,
0,
g_ipsec.IPSecSmallBufferSize,
IPSEC_TAG_BUFFER_POOL,
(USHORT)g_ipsec.IPSecSmallBufferListDepth);
ExInitializeNPagedLookasideList(&g_ipsec.IPSecLookasideLists->SendCompleteCtxList,
NULL,
NULL,
0,
g_ipsec.IPSecSendCompleteCtxSize,
IPSEC_TAG_SEND_COMPLETE,
(USHORT)g_ipsec.IPSecSendCompleteCtxDepth);
IPSEC_DEBUG(LOAD, ("Exiting IPSecInitMdlPool\n"));
return STATUS_SUCCESS;
}
VOID
IPSecDeinitMdlPool()
/*++
Routine Description:
Free the MDL pool for AH and ESP headers.
Arguments:
None
Return Value:
--*/
{
PAGED_CODE();
IPSEC_DEBUG(LOAD, ("Entering IPSecDeinitMdlPool\n"));
//
// Destroy the lookaside lists.
//
if (g_ipsec.IPSecLookasideLists != NULL) {
ExDeleteNPagedLookasideList(&g_ipsec.IPSecLookasideLists->LargeBufferList);
ExDeleteNPagedLookasideList(&g_ipsec.IPSecLookasideLists->SmallBufferList);
ExDeleteNPagedLookasideList(&g_ipsec.IPSecLookasideLists->SendCompleteCtxList);
IPSecFreeMemory(g_ipsec.IPSecLookasideLists);
}
IPSEC_DEBUG(LOAD, ("Exiting IPSecDeinitMdlPool\n"));
}
NTSTATUS
IPSecQuiesce()
/*++
Routine Description:
Destroy MDL pools and run down all driver activity
Arguments:
None
Return Value:
--*/
{
IPSEC_DEBUG(LOAD, ("Entering IPSecQuiesce\n"));
IPSecDeinitMdlPool();
IPSEC_DEBUG(LOAD, ("Exiting IPSecQuiesce\n"));
return STATUS_SUCCESS;
}
BOOLEAN
AllocateCacheStructures()
/*++
Routine Description:
Allocates the necessary memory for cache (which is an array of pointers to
cache entries)
Allocates necessary number of cache entries (but doesnt initialize them)
Allocates a small number of entries and puts them on the free list (doesnt
initialize these either)
Arguments:
None
Return Value:
True if the function completely succeeds, else FALSE. If FALSE, it is upto
the CALLER to do a rollback and clear any allocated memory
--*/
{
ULONG i;
PAGED_CODE();
g_ipsec.ppCache = IPSecAllocateMemory(g_ipsec.CacheSize * sizeof(PFILTER_CACHE), IPSEC_TAG_INIT);
if (!g_ipsec.ppCache) {
IPSEC_DEBUG(LOAD, ("Couldnt allocate memory for Input Cache\n"));
return FALSE;
}
IPSecZeroMemory(g_ipsec.ppCache, g_ipsec.CacheSize * sizeof(PFILTER_CACHE));
for (i = 0; i < g_ipsec.CacheSize; i++) {
PFILTER_CACHE pTemp1;
pTemp1 = IPSecAllocateMemory(sizeof(FILTER_CACHE), IPSEC_TAG_INIT);
if (!pTemp1) {
FreeExistingCache();
return FALSE;
}
IPSecZeroMemory(pTemp1, sizeof(FILTER_CACHE));
g_ipsec.ppCache[i] = pTemp1;
}
return TRUE;
}
VOID
FreeExistingCache()
/*++
Routine Description
Frees all the cache entries, free entries and cache pointer array
Arguments
None
Return Value
None
--*/
{
ULONG i;
PAGED_CODE();
IPSEC_DEBUG(LOAD, ("Freeing existing cache...\n"));
IPSecResetCacheTable();
if (g_ipsec.ppCache) {
for (i = 0; i < g_ipsec.CacheSize; i++) {
if (g_ipsec.ppCache[i]) {
ExFreePool(g_ipsec.ppCache[i]);
}
}
ExFreePool(g_ipsec.ppCache);
g_ipsec.ppCache = NULL;
}
}
VOID
FreePatternDbase()
/*++
Routine Description
Frees all filters and SAs.
Arguments
None
Return Value
None
--*/
{
PLIST_ENTRY pEntry;
PFILTER pFilter;
PSA_TABLE_ENTRY pSA;
LONG i, j;
PAGED_CODE();
//
// Free all masked filters and associated (outbound) SAs
//
for (i = MIN_FILTER; i <= MAX_FILTER; i++) {
while (!IsListEmpty(&g_ipsec.FilterList[i])) {
pEntry = RemoveHeadList(&g_ipsec.FilterList[i]);
pFilter = CONTAINING_RECORD(pEntry,
FILTER,
MaskedLinkage);
IPSEC_DEBUG(LOAD, ("Freeing filter: %lx\n", pFilter));
//
// Free each SA under it.
//
for (j = 0; j < pFilter->SAChainSize; j++) {
while (!IsListEmpty(&pFilter->SAChain[j])) {
pEntry = RemoveHeadList(&pFilter->SAChain[j]);
pSA = CONTAINING_RECORD(pEntry,
SA_TABLE_ENTRY,
sa_FilterLinkage);
IPSEC_DEBUG(LOAD, ("Freeing SA: %lx\n", pSA));
//
// Remove SA from miniport if plumbed
//
if (pSA->sa_Flags & FLAGS_SA_HW_PLUMBED) {
IPSecDelHWSA(pSA);
}
//
// Also remove the inbound SAs from their SPI list
// so we dont double free them below.
//
IPSecRemoveSPIEntry(pSA);
//
// Stop the timer if armed and deref SA.
//
IPSecStopTimerDerefSA(pSA);
}
}
#if GPC
IPSecUninstallGpcFilter(pFilter);
#endif
IPSecFreeFilter(pFilter);
}
}
//
// Free all SAs under the SPI hashes.
//
for (i = 0; i < g_ipsec.SAHashSize; i++) {
PSA_HASH pHash = &g_ipsec.pSADb[i];
while (!IsListEmpty(&pHash->SAList)) {
pEntry = RemoveHeadList(&pHash->SAList);
pSA = CONTAINING_RECORD(pEntry,
SA_TABLE_ENTRY,
sa_SPILinkage);
IPSEC_DEBUG(LOAD, ("Freeing SA: %lx\n", pSA));
if (pSA->sa_Flags & FLAGS_SA_HW_PLUMBED) {
IPSecDelHWSA(pSA);
}
IPSecStopTimerDerefSA(pSA);
}
}
IPSecFreeMemory(g_ipsec.pSADb);
IPSEC_DEBUG(LOAD, ("Freed filters/SAs\n"));
}
SIZE_T
IPSecCalculateBufferSize(
IN SIZE_T BufferDataSize
)
/*++
Routine Description:
Determines the size of an AFD buffer structure given the amount of
data that the buffer contains.
Arguments:
BufferDataSize - data length of the buffer.
AddressSize - length of address structure for the buffer.
Return Value:
Number of bytes needed for an IPSEC_LA_BUFFER structure for data of
this size.
--*/
{
SIZE_T mdlSize;
SIZE_T bufferSize;
ASSERT(BufferDataSize != 0);
ASSERT(g_ipsec.IPSecCacheLineSize < 100);
//
// Determine the sizes of the various components of an IPSEC_LA_BUFFER
// structure. Note that these are all worst-case calculations--
// actual sizes of the MDL and the buffer may be smaller.
//
bufferSize = BufferDataSize + g_ipsec.IPSecCacheLineSize;
mdlSize = MmSizeOfMdl( (PVOID)(PAGE_SIZE-1), bufferSize );
return ((sizeof(IPSEC_LA_BUFFER) + mdlSize + bufferSize + 3) & ~3);
}
VOID
IPSecInitializeBuffer(
IN PIPSEC_LA_BUFFER IPSecBuffer,
IN SIZE_T BufferDataSize
)
/*++
Routine Description:
Initializes an IPSec buffer. Sets up fields in the actual IPSEC_LA_BUFFER
structure and initializes the MDL associated with the buffer. This routine
assumes that the caller has properly allocated sufficient space for all this.
Arguments:
IPSecBuffer - points to the IPSEC_LA_BUFFER structure to initialize.
BufferDataSize - the size of the data buffer that goes along with the
buffer structure.
Return Value:
None
--*/
{
SIZE_T mdlSize;
//
// Set up the MDL pointer but don't build it yet. We have to wait
// until after the data buffer is built to build the MDL.
//
mdlSize = MmSizeOfMdl( (PVOID)(PAGE_SIZE-1), BufferDataSize );
IPSecBuffer->Mdl = (PMDL)&IPSecBuffer->Data[0];
IPSEC_DEBUG(POOL, ("IPSecBuffer: %lx, MDL: %lx\n", IPSecBuffer, IPSecBuffer->Mdl));
//
// Set up the data buffer pointer and length. Note that the buffer
// MUST begin on a cache line boundary so that we can use the fast
// copy routines like RtlCopyMemory on the buffer.
//
IPSecBuffer->Buffer = (PVOID)
(((ULONG_PTR)((PCHAR)IPSecBuffer->Mdl + mdlSize) +
g_ipsec.IPSecCacheLineSize - 1 ) & ~((ULONG_PTR)(g_ipsec.IPSecCacheLineSize - 1)));
IPSecBuffer->BufferLength = (ULONG)BufferDataSize; // Sundown - FIX
//
// Now build the MDL and set up a pointer to the MDL in the IRP.
//
MmInitializeMdl( IPSecBuffer->Mdl, IPSecBuffer->Buffer, BufferDataSize );
MmBuildMdlForNonPagedPool( IPSecBuffer->Mdl );
}
PVOID
IPSecAllocateBufferPool(
IN POOL_TYPE PoolType,
IN SIZE_T NumberOfBytes,
IN ULONG Tag
)
/*++
Routine Description:
Used by the lookaside list allocation function to allocate a new
IPSec buffer structure. The returned structure will be fully
initialized.
Arguments:
PoolType - passed to ExAllocatePoolWithTag.
NumberOfBytes - the number of bytes required for the data buffer
portion of the IPSec buffer.
Tag - passed to ExAllocatePoolWithTag.
Return Value:
PVOID - a fully initialized PIPSEC_LA_BUFFER, or NULL if the allocation
attempt fails.
--*/
{
PIPSEC_LA_BUFFER IPSecBuffer;
SIZE_T bytesRequired;
//
// The requested length must be the same as one of the standard
// IPSec buffer sizes.
//
ASSERT( NumberOfBytes == g_ipsec.IPSecSmallBufferSize ||
NumberOfBytes == g_ipsec.IPSecLargeBufferSize );
//
// Determine how much data we'll actually need for the buffer.
//
bytesRequired = IPSecCalculateBufferSize(NumberOfBytes);
//
// Get nonpaged pool for the buffer.
//
IPSecBuffer = IPSecAllocateMemory( bytesRequired, Tag );
if ( IPSecBuffer == NULL ) {
return NULL;
}
//
// Initialize the buffer and return a pointer to it.
//
IPSecInitializeBuffer( IPSecBuffer, NumberOfBytes );
return IPSecBuffer;
}
PIPSEC_LA_BUFFER
IPSecGetBuffer(
IN CLONG BufferDataSize,
IN ULONG Tag
)
/*++
Routine Description:
Obtains a buffer of the appropriate size for the caller. Uses
the preallocated buffers if possible, or else allocates a new buffer
structure if required.
Arguments:
BufferDataSize - the size of the data buffer that goes along with the
buffer structure.
Return Value:
PIPSEC_LA_BUFFER - a pointer to an IPSEC_LA_BUFFER structure, or NULL if one
was not available or could not be allocated.
--*/
{
PIPSEC_LA_BUFFER IPSecBuffer;
SIZE_T bufferSize;
PLIST_ENTRY listEntry;
PNPAGED_LOOKASIDE_LIST lookasideList;
//
// If possible, allocate the buffer from one of the lookaside lists.
//
if (BufferDataSize <= g_ipsec.IPSecLargeBufferSize) {
if ( BufferDataSize <= g_ipsec.IPSecSmallBufferSize ) {
lookasideList = &g_ipsec.IPSecLookasideLists->SmallBufferList;
BufferDataSize = g_ipsec.IPSecSmallBufferSize;
} else {
lookasideList = &g_ipsec.IPSecLookasideLists->LargeBufferList;
BufferDataSize = g_ipsec.IPSecLargeBufferSize;
}
IPSecBuffer = ExAllocateFromNPagedLookasideList( lookasideList );
if (!IPSecBuffer) {
return NULL;
}
IPSecBuffer->Tag = Tag;
return IPSecBuffer;
}
//
// Couldn't find an appropriate buffer that was preallocated.
// Allocate one manually. If the buffer size requested was
// zero bytes, give them four bytes. This is because some of
// the routines like MmSizeOfMdl() cannot handle getting passed
// in a length of zero.
//
// !!! It would be good to ROUND_TO_PAGES for this allocation
// if appropriate, then use entire buffer size.
//
if ( BufferDataSize == 0 ) {
BufferDataSize = sizeof(ULONG);
}
bufferSize = IPSecCalculateBufferSize(BufferDataSize);
IPSecBuffer = IPSecAllocateMemory(bufferSize, IPSEC_TAG_BUFFER_POOL);
if ( IPSecBuffer == NULL ) {
return NULL;
}
//
// Initialize the IPSec buffer structure and return it.
//
IPSecInitializeBuffer(IPSecBuffer, BufferDataSize);
IPSecBuffer->Tag = Tag;
return IPSecBuffer;
}
VOID
IPSecReturnBuffer (
IN PIPSEC_LA_BUFFER IPSecBuffer
)
/*++
Routine Description:
Returns an IPSec buffer to the appropriate global list, or frees
it if necessary.
Arguments:
IPSecBufferHeader - points to the IPSec_BUFFER_HEADER structure to return or free.
Return Value:
None
--*/
{
PNPAGED_LOOKASIDE_LIST lookasideList;
//
// If appropriate, return the buffer to one of the IPSec buffer
// lookaside lists.
//
if (IPSecBuffer->BufferLength <= g_ipsec.IPSecLargeBufferSize) {
if (IPSecBuffer->BufferLength==g_ipsec.IPSecSmallBufferSize) {
lookasideList = &g_ipsec.IPSecLookasideLists->SmallBufferList;
} else {
ASSERT (IPSecBuffer->BufferLength==g_ipsec.IPSecLargeBufferSize);
lookasideList = &g_ipsec.IPSecLookasideLists->LargeBufferList;
}
ExFreeToNPagedLookasideList( lookasideList, IPSecBuffer );
return;
}
IPSecFreeMemory(IPSecBuffer);
}
NTSTATUS
IPSecWriteEvent(
PDRIVER_OBJECT IPSecDriverObject,
IN ULONG EventCode,
IN NTSTATUS NtStatusCode,
IN ULONG OffloadStatus,
IN ULONG ExtraStatus1,
IN ULONG ExtraStatus2,
IN PVOID RawDataBuffer,
IN USHORT RawDataLength,
IN USHORT NumberOfInsertionStrings,
...
)
#define LAST_NAMED_ARGUMENT NumberOfInsertionStrings
/*++
Routine Description:
This function allocates an I/O error log record, fills it in and writes it
to the I/O error log.
Arguments:
Return Value:
None.
--*/
{
PIO_ERROR_LOG_PACKET ErrorLogEntry;
va_list ParmPtr; // Pointer to stack parms.
PCHAR DumpData;
LONG Length;
ULONG i, SizeOfRawData, RemainingSpace, TotalErrorLogEntryLength;
ULONG SizeOfStringData = 0;
PWSTR StringOffset, InsertionString;
if (NumberOfInsertionStrings != 0)
{
va_start (ParmPtr, LAST_NAMED_ARGUMENT);
for (i = 0; i < NumberOfInsertionStrings; i += 1)
{
InsertionString = va_arg (ParmPtr, PWSTR);
Length = wcslen (InsertionString);
while ((Length > 0) && (InsertionString[Length-1] == L' '))
{
Length--;
}
SizeOfStringData += (Length + 1) * sizeof(WCHAR);
}
}
//
// Ideally we want the packet to hold the servername and ExtraInformation.
// Usually the ExtraInformation gets truncated.
//
TotalErrorLogEntryLength = min (RawDataLength + sizeof(IO_ERROR_LOG_PACKET) + 1 + SizeOfStringData,
ERROR_LOG_MAXIMUM_SIZE);
RemainingSpace = TotalErrorLogEntryLength - FIELD_OFFSET(IO_ERROR_LOG_PACKET, DumpData);
if (RemainingSpace > SizeOfStringData)
{
SizeOfRawData = RemainingSpace - SizeOfStringData;
}
else
{
SizeOfStringData = RemainingSpace;
SizeOfRawData = 0;
}
ErrorLogEntry = IoAllocateErrorLogEntry (IPSecDriverObject, (UCHAR) TotalErrorLogEntryLength);
if (ErrorLogEntry == NULL)
{
return(STATUS_INSUFFICIENT_RESOURCES);
}
//
// Fill in the error log entry
//
ErrorLogEntry->ErrorCode = EventCode;
ErrorLogEntry->UniqueErrorValue = OffloadStatus;
ErrorLogEntry->FinalStatus = NtStatusCode;
ErrorLogEntry->MajorFunctionCode = 0;
ErrorLogEntry->RetryCount = 0;
ErrorLogEntry->IoControlCode = 0;
ErrorLogEntry->DeviceOffset.LowPart = ExtraStatus1;
ErrorLogEntry->DeviceOffset.HighPart = ExtraStatus2;
ErrorLogEntry->DumpDataSize = 0;
ErrorLogEntry->NumberOfStrings = 0;
ErrorLogEntry->SequenceNumber = 0;
ErrorLogEntry->StringOffset = (USHORT) (ROUND_UP_COUNT (FIELD_OFFSET(IO_ERROR_LOG_PACKET, DumpData)
+ SizeOfRawData, ALIGN_WORD));
//
// Append the dump data. This information is typically an SMB header.
//
if ((RawDataBuffer) && (SizeOfRawData))
{
DumpData = (PCHAR) ErrorLogEntry->DumpData;
Length = min (RawDataLength, (USHORT)SizeOfRawData);
RtlCopyMemory (DumpData, RawDataBuffer, Length);
ErrorLogEntry->DumpDataSize = (USHORT)Length;
}
//
// Add the debug informatuion strings
//
if (NumberOfInsertionStrings)
{
StringOffset = (PWSTR) ((PCHAR)ErrorLogEntry + ErrorLogEntry->StringOffset);
//
// Set up ParmPtr to point to first of the caller's parameters.
//
va_start(ParmPtr, LAST_NAMED_ARGUMENT);
for (i = 0 ; i < NumberOfInsertionStrings ; i+= 1)
{
InsertionString = va_arg(ParmPtr, PWSTR);
Length = wcslen(InsertionString);
while ( (Length > 0) && (InsertionString[Length-1] == L' '))
{
Length--;
}
if (((Length + 1) * sizeof(WCHAR)) > SizeOfStringData)
{
Length = (SizeOfStringData/sizeof(WCHAR)) - 1;
}
if (Length > 0)
{
RtlCopyMemory (StringOffset, InsertionString, Length*sizeof(WCHAR));
StringOffset += Length;
*StringOffset++ = L'\0';
SizeOfStringData -= (Length + 1) * sizeof(WCHAR);
ErrorLogEntry->NumberOfStrings += 1;
}
}
}
IoWriteErrorLogEntry(ErrorLogEntry);
return(STATUS_SUCCESS);
}
VOID
IPSecLogEvents(
IN PVOID Context
)
/*++
Routine Description:
Dumps events from the circular buffer to the eventlog when the
circular buffer overflows.
Arguments:
Context - unused.
Return Value:
None
--*/
{
PIPSEC_LOG_EVENT pLogEvent;
LONG LogSize;
PUCHAR pLog;
pLogEvent = (PIPSEC_LOG_EVENT)Context;
LogSize = 0;
pLog = (PUCHAR)pLogEvent + FIELD_OFFSET(IPSEC_LOG_EVENT, pLog[0]);
while (LogSize < pLogEvent->LogSize) {
PIPSEC_EVENT_CTX ctx = (PIPSEC_EVENT_CTX)pLog;
if (ctx->EventCode == EVENT_IPSEC_DROP_PACKET_INBOUND ||
ctx->EventCode == EVENT_IPSEC_DROP_PACKET_OUTBOUND) {
WCHAR IPAddrBufferS[(sizeof(IPAddr) * 4) + 1];
WCHAR IPAddrBufferD[(sizeof(IPAddr) * 4) + 1];
WCHAR IPProtocolBuffer[(sizeof(IPAddr) * 4) + 1];
WCHAR IPSPortBuffer[(sizeof(IPAddr) * 4) + 1];
WCHAR IPDPortBuffer[(sizeof(IPAddr) * 4) + 1];
PWCHAR stringlist[5];
IPHeader UNALIGNED *pIPH;
USHORT SrcPort=0;
USHORT DestPort=0;
ULONG HeaderLen;
pIPH = (IPHeader UNALIGNED *)ctx->pPacket;
HeaderLen=(pIPH->iph_verlen & (UCHAR)~IP_VER_FLAG) << 2;
IPSecIPAddrToUnicodeString( pIPH->iph_src,
IPAddrBufferS);
IPSecIPAddrToUnicodeString( pIPH->iph_dest,
IPAddrBufferD);
IPSecCountToUnicodeString ( pIPH->iph_protocol,
IPProtocolBuffer);
if (pIPH->iph_protocol == PROTOCOL_TCP ||
pIPH->iph_protocol == PROTOCOL_UDP) {
RtlCopyMemory(&SrcPort,&ctx->pPacket[HeaderLen],sizeof(USHORT));
RtlCopyMemory(&DestPort,&ctx->pPacket[HeaderLen+sizeof(USHORT)],sizeof(USHORT));
}
IPSecCountToUnicodeString ( NET_SHORT(SrcPort),
IPSPortBuffer);
IPSecCountToUnicodeString ( NET_SHORT(DestPort),
IPDPortBuffer);
IPSecWriteEvent(
g_ipsec.IPSecDriverObject,
ctx->EventCode,
ctx->DropStatus.IPSecStatus,
ctx->DropStatus.OffloadStatus,
ctx->DropStatus.Flags,
0,
ctx->pPacket,
(USHORT)ctx->PacketSize,
5,
IPAddrBufferS,
IPAddrBufferD,
IPProtocolBuffer,
IPSPortBuffer,
IPDPortBuffer);
} else if (ctx->Addr && ctx->EventCount > 0) {
WCHAR IPAddrBuffer[(sizeof(IPAddr) * 4) + 1];
WCHAR CountBuffer[MAX_COUNT_STRING_LEN + 1];
PWCHAR stringList[2];
IPSecIPAddrToUnicodeString( ctx->Addr,
IPAddrBuffer);
IPSecCountToUnicodeString( ctx->EventCount,
CountBuffer);
stringList[0] = CountBuffer;
stringList[1] = IPAddrBuffer;
LOG_EVENT(
g_ipsec.IPSecDriverObject,
ctx->EventCode,
ctx->UniqueEventValue,
2,
stringList,
0,
NULL);
} else if (ctx->Addr) {
WCHAR IPAddrBuffer[(sizeof(IPAddr) * 4) + 1];
PWCHAR stringList[1];
IPSecIPAddrToUnicodeString( ctx->Addr,
IPAddrBuffer);
stringList[0] = IPAddrBuffer;
LOG_EVENT(
g_ipsec.IPSecDriverObject,
ctx->EventCode,
ctx->UniqueEventValue,
1,
stringList,
0,
NULL);
} else {
LOG_EVENT(
g_ipsec.IPSecDriverObject,
ctx->EventCode,
ctx->UniqueEventValue,
0,
NULL,
0,
NULL);
}
if (ctx->pPacket) {
IPSecFreeLogBuffer(ctx->pPacket);
ctx->pPacket=NULL;
}
pLog += sizeof(IPSEC_EVENT_CTX);
LogSize += sizeof(IPSEC_EVENT_CTX);
}
IPSecFreeMemory(pLogEvent);
IPSEC_DECREMENT(g_ipsec.NumWorkers);
}
VOID
IPSecBufferEvent(
IN IPAddr Addr,
IN ULONG EventCode,
IN ULONG UniqueEventValue,
IN BOOLEAN fBufferEvent
)
/*++
Routine Description:
Buffers events in a circular buffer; dumps them to the eventlog when the
circular buffer overflows.
Arguments:
Addr - [OPTIONAL] the source IP addr of the offending peer.
EventCode - Identifies the error message.
UniqueEventValue - Identifies this instance of a given error message.
Return Value:
None
--*/
{
KIRQL kIrql;
if (!(g_ipsec.DiagnosticMode & IPSEC_DIAGNOSTIC_ENABLE_LOG)) {
return;
}
ACQUIRE_LOCK(&g_ipsec.EventLogLock, &kIrql);
if (fBufferEvent) {
PIPSEC_EVENT_CTX ctx;
g_ipsec.IPSecBufferedEvents++;
ctx = (PIPSEC_EVENT_CTX)g_ipsec.IPSecLogMemoryLoc;
ctx--;
while (ctx >= (PIPSEC_EVENT_CTX)g_ipsec.IPSecLogMemory) {
if (ctx->Addr == Addr &&
ctx->EventCode == EventCode &&
ctx->UniqueEventValue == UniqueEventValue) {
//
// Found a duplicate; update count and exit.
//
ctx->EventCount++;
if (g_ipsec.IPSecBufferedEvents >= g_ipsec.EventQueueSize) {
goto logit;
}
RELEASE_LOCK(&g_ipsec.EventLogLock, kIrql);
return;
}
ctx--;
}
}
((PIPSEC_EVENT_CTX)g_ipsec.IPSecLogMemoryLoc)->Addr = Addr;
((PIPSEC_EVENT_CTX)g_ipsec.IPSecLogMemoryLoc)->EventCode = EventCode;
((PIPSEC_EVENT_CTX)g_ipsec.IPSecLogMemoryLoc)->UniqueEventValue = UniqueEventValue;
((PIPSEC_EVENT_CTX)g_ipsec.IPSecLogMemoryLoc)->pPacket=NULL;
((PIPSEC_EVENT_CTX)g_ipsec.IPSecLogMemoryLoc)->PacketSize=0;
if (fBufferEvent) {
((PIPSEC_EVENT_CTX)g_ipsec.IPSecLogMemoryLoc)->EventCount = 1;
} else {
((PIPSEC_EVENT_CTX)g_ipsec.IPSecLogMemoryLoc)->EventCount = 0;
}
g_ipsec.IPSecLogMemoryLoc += sizeof(IPSEC_EVENT_CTX);
logit:
if (!fBufferEvent ||
g_ipsec.IPSecLogMemoryLoc >= g_ipsec.IPSecLogMemoryEnd ||
g_ipsec.IPSecBufferedEvents >= g_ipsec.EventQueueSize) {
//
// Flush the logs.
//
IPSecQueueLogEvent();
}
RELEASE_LOCK(&g_ipsec.EventLogLock, kIrql);
}
NTSTATUS
CopyOutboundPacketToBuffer(
IN PUCHAR pIPHeader,
IN PVOID pData,
OUT PUCHAR * pPacket,
OUT ULONG * PacketSize
)
{
PNDIS_BUFFER pTemp;
ULONG Length;
ULONG dataLength=0;
IPHeader UNALIGNED *pIPH;
ULONG HeaderLen=0;
PUCHAR pBuffer;
ULONG CopyPos=0;
PUCHAR pPacketData;
pIPH = (IPHeader UNALIGNED *)pIPHeader;
pTemp = (PNDIS_BUFFER)pData;
while (pTemp) {
pBuffer = NULL;
Length = 0;
NdisQueryBufferSafe(pTemp,
&pBuffer,
&Length,
NormalPagePriority);
if (!pBuffer) {
return STATUS_UNSUCCESSFUL;
}
dataLength += Length;
pTemp = NDIS_BUFFER_LINKAGE(pTemp);
}
HeaderLen=(pIPH->iph_verlen & (UCHAR)~IP_VER_FLAG) << 2;
dataLength += HeaderLen;
if (dataLength > IPSEC_LOG_PACKET_SIZE) {
dataLength = IPSEC_LOG_PACKET_SIZE;
}
if (dataLength < sizeof(IPHeader)) {
// doesn't even have a full ip header
return STATUS_UNSUCCESSFUL;
}
if ((pIPH->iph_protocol == PROTOCOL_TCP) ||
(pIPH->iph_protocol == PROTOCOL_UDP)) {
if (dataLength - HeaderLen < 8) {
// not enough room for ports
return STATUS_UNSUCCESSFUL;
}
}
*pPacket = IPSecAllocateLogBuffer(dataLength);
if (! (*pPacket)) {
return STATUS_UNSUCCESSFUL;
}
*PacketSize=dataLength;
pTemp = (PNDIS_BUFFER)pData;
CopyPos=0;
while (pTemp && CopyPos < dataLength) {
IPSecQueryNdisBuf(pTemp,&pPacketData,&Length);
if (CopyPos + Length > dataLength) {
Length = (dataLength - CopyPos);
}
RtlCopyMemory(*pPacket+CopyPos,pPacketData,Length);
CopyPos += Length;
pTemp = NDIS_BUFFER_LINKAGE(pTemp);
}
return STATUS_SUCCESS;
}
//
// pData is data after IPHeader, IPRcvBuf.
//
NTSTATUS
CopyInboundPacketToBuffer(
IN PUCHAR pIPHeader,
IN PVOID pData,
OUT PUCHAR * pPacket,
OUT ULONG * PacketSize
)
{
IPRcvBuf *pTemp;
ULONG Length;
ULONG dataLength=0;
IPHeader UNALIGNED *pIPH;
ULONG HeaderLen=0;
PUCHAR pBuffer;
ULONG CopyPos=0;
PUCHAR pPacketData;
pIPH = (IPHeader UNALIGNED *)pIPHeader;
pTemp = (IPRcvBuf*)pData;
while (pTemp) {
pBuffer = NULL;
Length = 0;
IPSecQueryRcvBuf(pTemp,
&pBuffer,
&Length);
if (!pBuffer) {
return STATUS_UNSUCCESSFUL;
}
dataLength += Length;
pTemp = IPSEC_BUFFER_LINKAGE(pTemp);
}
HeaderLen=(pIPH->iph_verlen & (UCHAR)~IP_VER_FLAG) << 2;
dataLength += HeaderLen;
if (dataLength > IPSEC_LOG_PACKET_SIZE) {
dataLength = IPSEC_LOG_PACKET_SIZE;
}
// Sanity check length
if (dataLength < sizeof(IPHeader)) {
// doesn't even have a full ip header
return STATUS_UNSUCCESSFUL;
}
if ((pIPH->iph_protocol == PROTOCOL_TCP) ||
(pIPH->iph_protocol == PROTOCOL_UDP)) {
if (dataLength - HeaderLen < 8) {
// not enough room for ports
return STATUS_UNSUCCESSFUL;
}
}
*pPacket = IPSecAllocateLogBuffer(dataLength);
if (! (*pPacket)) {
return STATUS_UNSUCCESSFUL;
}
*PacketSize=dataLength;
pTemp = (IPRcvBuf*)pData;
RtlCopyMemory(*pPacket,pIPH,HeaderLen);
CopyPos=HeaderLen;
while (pTemp && CopyPos < dataLength) {
IPSecQueryRcvBuf(pTemp,&pPacketData,&Length);
if (CopyPos + Length > dataLength) {
Length = (dataLength - CopyPos);
}
RtlCopyMemory(*pPacket+CopyPos,pPacketData,Length);
CopyPos += Length;
pTemp = IPSEC_BUFFER_LINKAGE(pTemp);
}
return STATUS_SUCCESS;
}
VOID
IPSecBufferPacketDrop(
IN PUCHAR pIPHeader,
IN PVOID pData,
IN OUT PULONG pIpsecFlags,
IN PIPSEC_DROP_STATUS pDropStatus
)
/*++
Routine Description:
Buffers events in a circular buffer; dumps them to the eventlog when the
circular buffer overflows.
Arguments:
EventCode - Identifies the error message.
Return Value:
None
--*/
{
KIRQL kIrql;
PIPSEC_EVENT_CTX ctx;
IPHeader UNALIGNED *pIPH;
PNDIS_BUFFER pTemp;
PUCHAR pPacket=NULL;
ULONG PacketSize=0;
ULONG Status;
BOOL bLockHeld=FALSE;
pIPH = (IPHeader UNALIGNED *)pIPHeader;
if (*pIpsecFlags & IPSEC_FLAG_INCOMING) {
if (!(g_ipsec.DiagnosticMode & IPSEC_DIAGNOSTIC_INBOUND)) {
// Don't log
goto out;
}
Status=CopyInboundPacketToBuffer(pIPHeader,
pData,
&pPacket,
&PacketSize);
} else {
if (!(g_ipsec.DiagnosticMode & IPSEC_DIAGNOSTIC_OUTBOUND)) {
//Don't log
goto out;
}
Status=CopyOutboundPacketToBuffer(pIPHeader,
pData,
&pPacket,
&PacketSize);
}
if (Status != STATUS_SUCCESS) {
goto out;
}
ACQUIRE_LOCK(&g_ipsec.EventLogLock, &kIrql);
bLockHeld=TRUE;
g_ipsec.IPSecBufferedEvents++;
ctx = (PIPSEC_EVENT_CTX)g_ipsec.IPSecLogMemoryLoc;
ctx--;
((PIPSEC_EVENT_CTX)g_ipsec.IPSecLogMemoryLoc)->Addr=pIPH->iph_src;
if (*pIpsecFlags & IPSEC_FLAG_INCOMING) {
((PIPSEC_EVENT_CTX)g_ipsec.IPSecLogMemoryLoc)->EventCode = EVENT_IPSEC_DROP_PACKET_INBOUND;
} else {
((PIPSEC_EVENT_CTX)g_ipsec.IPSecLogMemoryLoc)->EventCode = EVENT_IPSEC_DROP_PACKET_OUTBOUND;;
}
((PIPSEC_EVENT_CTX)g_ipsec.IPSecLogMemoryLoc)->EventCount = 1;
((PIPSEC_EVENT_CTX)g_ipsec.IPSecLogMemoryLoc)->pPacket = pPacket;
((PIPSEC_EVENT_CTX)g_ipsec.IPSecLogMemoryLoc)->PacketSize = PacketSize;
if (pDropStatus) {
RtlCopyMemory(&(((PIPSEC_EVENT_CTX)g_ipsec.IPSecLogMemoryLoc)->DropStatus),
pDropStatus,sizeof(IPSEC_DROP_STATUS));
} else {
RtlZeroMemory(&(((PIPSEC_EVENT_CTX)g_ipsec.IPSecLogMemoryLoc)->DropStatus),
sizeof(IPSEC_DROP_STATUS));
}
g_ipsec.IPSecLogMemoryLoc += sizeof(IPSEC_EVENT_CTX);
if (g_ipsec.IPSecLogMemoryLoc >= g_ipsec.IPSecLogMemoryEnd ||
g_ipsec.IPSecBufferedEvents >= g_ipsec.EventQueueSize) {
//
// Flush the logs.
//
IPSecQueueLogEvent();
}
out:
if (bLockHeld) {
RELEASE_LOCK(&g_ipsec.EventLogLock, kIrql);
}
}
VOID
IPSecQueueLogEvent(
VOID
)
/*++
Routine Description:
Copies the LogMemory to a temporary buffer and schedule an event to
flush logs.
Arguments:
None
Return Value:
None
Notes:
Called with EventLogLock held.
--*/
{
PIPSEC_LOG_EVENT pLogEvent;
LONG LogSize;
PUCHAR pLog;
LogSize = (LONG)(g_ipsec.IPSecLogMemoryLoc - g_ipsec.IPSecLogMemory);
//
// Reset the log memory so we can record again.
//
g_ipsec.IPSecLogMemoryLoc = g_ipsec.IPSecLogMemory;
g_ipsec.IPSecBufferedEvents = 0;
if (LogSize <= 0) {
ASSERT(FALSE);
return;
}
pLogEvent = IPSecAllocateMemory(LogSize + FIELD_OFFSET(IPSEC_LOG_EVENT, pLog[0]),
IPSEC_TAG_EVT_QUEUE);
if (!pLogEvent) {
return;
}
pLogEvent->LogSize = LogSize;
pLog = (PUCHAR)pLogEvent + FIELD_OFFSET(IPSEC_LOG_EVENT, pLog[0]);
RtlCopyMemory(pLog, g_ipsec.IPSecLogMemory, LogSize);
//
// Queue work item to dump these into the eventlog.
//
ExInitializeWorkItem(&pLogEvent->LogQueueItem, IPSecLogEvents, pLogEvent);
ExQueueWorkItem(&pLogEvent->LogQueueItem, DelayedWorkQueue);
IPSEC_INCREMENT(g_ipsec.NumWorkers);
}
#if FIPS
BOOLEAN
IPSecFipsInitialize(
VOID
)
/*++
Routine Description:
Initialize the FIPS library table.
Arguments:
Called at PASSIVE level.
Return Value:
TRUE/FALSE.
--*/
{
UNICODE_STRING DeviceName;
PDEVICE_OBJECT pFipsDeviceObject = NULL;
PIRP pIrp;
IO_STATUS_BLOCK StatusBlock;
KEVENT Event;
NTSTATUS status;
PAGED_CODE();
//
// Return success if FIPS already initialized.
//
if (IPSEC_DRIVER_INIT_FIPS()) {
return TRUE;
}
RtlInitUnicodeString(&DeviceName, FIPS_DEVICE_NAME);
//
// Get the file and device objects for FIPS.
//
status = IoGetDeviceObjectPointer( &DeviceName,
FILE_ALL_ACCESS,
&g_ipsec.FipsFileObject,
&pFipsDeviceObject);
if (!NT_SUCCESS(status)) {
g_ipsec.FipsFileObject = NULL;
return FALSE;
}
//
// Build the request to send to FIPS to get library table.
//
KeInitializeEvent(&Event, SynchronizationEvent, FALSE);
pIrp = IoBuildDeviceIoControlRequest( IOCTL_FIPS_GET_FUNCTION_TABLE,
pFipsDeviceObject,
NULL,
0,
&g_ipsec.FipsFunctionTable,
sizeof(FIPS_FUNCTION_TABLE),
FALSE,
&Event,
&StatusBlock);
if (pIrp == NULL) {
IPSEC_DEBUG(LOAD, ("IoBuildDeviceIoControlRequest IOCTL_FIPS_GET_FUNCTION_TABLE failed.\n"));
ObDereferenceObject(g_ipsec.FipsFileObject);
g_ipsec.FipsFileObject = NULL;
return FALSE;
}
status = IoCallDriver(pFipsDeviceObject, pIrp);
if (status == STATUS_PENDING) {
status = KeWaitForSingleObject( &Event,
Executive,
KernelMode,
FALSE,
NULL);
if (status == STATUS_SUCCESS) {
status = StatusBlock.Status;
}
}
if (status != STATUS_SUCCESS) {
IPSEC_DEBUG(LOAD, ("IoCallDriver: IOCTL_FIPS_GET_FUNCTION_TABLE failed %#x\n", status));
ObDereferenceObject(g_ipsec.FipsFileObject);
g_ipsec.FipsFileObject = NULL;
return FALSE;
}
IPSEC_DRIVER_INIT_FIPS() = TRUE;
return TRUE;
}
#endif
BOOLEAN
IPSecCryptoInitialize(
VOID
)
/*++
Routine Description:
Initialize RNG and FIPS library table.
Arguments:
None
Return Value:
TRUE/FALSE
--*/
{
PAGED_CODE();
if (IPSEC_DRIVER_INIT_CRYPTO()) {
return TRUE;
}
#if FIPS
//
// Init the FIPS crypto library.
//
if (!IPSecFipsInitialize()) {
return FALSE;
}
#endif
//
// Init the RC4 key for RNG.
//
if (!IPSEC_DRIVER_INIT_RNG()) {
InitializeRNG(NULL);
if (!IPSecInitRandom()) {
ShutdownRNG(NULL);
return FALSE;
}
IPSEC_DRIVER_INIT_RNG() = TRUE;
}
IPSEC_DRIVER_INIT_CRYPTO() = TRUE;
return TRUE;
}
BOOLEAN
IPSecCryptoDeinitialize(
VOID
)
/*++
Routine Description:
Deinitialize RNG and dereference FipsFileObject.
Arguments:
None
Return Value:
TRUE/FALSE
--*/
{
PAGED_CODE();
//
// Don't forget to shutdown RNG or we will leak memory.
//
if (IPSEC_DRIVER_INIT_RNG()) {
ShutdownRNG(NULL);
}
#if FIPS
//
// Dereference FipsFileObject.
//
if (g_ipsec.FipsFileObject) {
ObDereferenceObject(g_ipsec.FipsFileObject);
}
#endif
return TRUE;
}
NTSTATUS
IPSecRegisterProtocols(
PIPSEC_REGISTER_PROTOCOL pIpsecRegisterProtocol
)
{
if (pIpsecRegisterProtocol->RegisterProtocol == IPSEC_REGISTER_PROTOCOLS) {
if (!IPSEC_GET_VALUE(gdwInitEsp)) {
if (TCPIP_REGISTER_PROTOCOL(
PROTOCOL_ESP,
NULL,
NULL,
IPSecESPStatus,
NULL,
NULL,
NULL
)) {
IPSEC_SET_VALUE(gdwInitEsp, 1);
}
else {
ASSERT(FALSE);
return (STATUS_INSUFFICIENT_RESOURCES);
}
}
if (!IPSEC_GET_VALUE(gdwInitAh)) {
if (TCPIP_REGISTER_PROTOCOL(
PROTOCOL_AH,
NULL,
NULL,
IPSecAHStatus,
NULL,
NULL,
NULL
)) {
IPSEC_SET_VALUE(gdwInitAh, 1);
}
else {
ASSERT(FALSE);
TCPIP_DEREGISTER_PROTOCOL(PROTOCOL_ESP);
IPSEC_SET_VALUE(gdwInitEsp, 0);
return (STATUS_INSUFFICIENT_RESOURCES);
}
}
}
else if (pIpsecRegisterProtocol->RegisterProtocol == IPSEC_DEREGISTER_PROTOCOLS) {
if (IPSEC_GET_VALUE(gdwInitEsp)) {
TCPIP_DEREGISTER_PROTOCOL(PROTOCOL_ESP);
IPSEC_SET_VALUE(gdwInitEsp, 0);
}
if (IPSEC_GET_VALUE(gdwInitAh)) {
TCPIP_DEREGISTER_PROTOCOL(PROTOCOL_AH);
IPSEC_SET_VALUE(gdwInitAh, 0);
}
}
else {
return (STATUS_INVALID_PARAMETER);
}
return (STATUS_SUCCESS);
}