windows-nt/Source/XPSP1/NT/inetsrv/iis/iisrearc/ul/drv/ultci.cxx
2020-09-26 16:20:57 +08:00

4522 lines
116 KiB
C++

/*++
Copyright (c) 2000-2001 Microsoft Corporation
Module Name:
ultci.cxx - UL TrafficControl Interface
Abstract:
This module implements a wrapper for QoS TC (Traffic Control)
Interface since the Kernel level API don't exist at this time.
Any HTTP module can use this interface to invoke QoS calls.
Author:
Ali Ediz Turkoglu (aliTu) 28-Jul-2000 Created a draft
version
Revision History:
Ali Ediz Turkoglu (aliTu) 03-11-2000 Modified to handle
Flow & Filter (re)config
as well as various other
major changes. In other
word i've put it into a
shape to be a component
--*/
#include "precomp.h"
//
// A nonpaged resource - TciIfcResource - guards the interface list
// and its flows.
//
LIST_ENTRY g_TciIfcListHead = {NULL,NULL};
//LIST_ENTRY g_TcCGroupListHead = {NULL,NULL};
BOOLEAN g_InitTciCalled = FALSE;
//
// GPC handles to talk to
//
HANDLE g_GpcFileHandle; // result of CreateFile on GPC device
GPC_HANDLE g_GpcClientHandle; // result of GPC client registration
//
// For querying the interface info like index & mtu size
//
HANDLE g_TcpDeviceHandle = NULL;
//
// Shows if PSCHED is installed or not
//
LONG g_PSchedInstalled = 0;
//
// Shows if Global Bandwidth Throttling is enabled or not
//
LONG g_GlobalThrottling = 0;
//
// For interface notifications
//
PVOID g_TcInterfaceUpNotificationObject = NULL;
PVOID g_TcInterfaceDownNotificationObject = NULL;
PVOID g_TcInterfaceChangeNotificationObject = NULL;
#ifdef ALLOC_PRAGMA
#pragma alloc_text(INIT, UlTcInitialize)
#pragma alloc_text(PAGE, UlTcTerminate)
#pragma alloc_text(PAGE, UlpTcInitializeGpc)
#pragma alloc_text(PAGE, UlpTcRegisterGpcClient)
#pragma alloc_text(PAGE, UlpTcDeRegisterGpcClient)
#pragma alloc_text(PAGE, UlpTcGetFriendlyNames)
#pragma alloc_text(PAGE, UlpTcReleaseAll)
#pragma alloc_text(PAGE, UlpTcCloseInterface)
#pragma alloc_text(PAGE, UlpTcCloseAllInterfaces)
#pragma alloc_text(PAGE, UlpTcDeleteFlow)
#endif // ALLOC_PRAGMA
#if 0
NOT PAGEABLE -- UlpRemoveFilterEntry
NOT PAGEABLE -- UlpInsertFilterEntry
#endif
//
// Init & Terminate stuff comes here.
//
/***************************************************************************++
Routine Description:
UlTcInitialize :
Will also initiate the Gpc client registration and make few WMI calls
down to psched.
Return Value:
NTSTATUS - Completion status.
--***************************************************************************/
NTSTATUS
UlTcInitialize (
VOID
)
{
NTSTATUS Status;
PAGED_CODE();
Status = STATUS_SUCCESS;
ASSERT(!g_InitTciCalled);
g_GlobalThrottling = 0;
g_PSchedInstalled = 0;
if (!g_InitTciCalled)
{
InitializeListHead(&g_TciIfcListHead);
//InitializeListHead(&g_TcCGroupListHead);
Status = UlInitializeResource(
&g_pUlNonpagedData->TciIfcResource,
"TciIfcResource",
0,
UL_TCI_RESOURCE_TAG
);
ASSERT(NT_SUCCESS(Status));
Status = UlpTcInitializeGpc();
if (!NT_SUCCESS(Status))
goto cleanup;
UlTrace( TC, ("Ul!UlTcInitialize: InitializeGpc Status %08lx \n", Status ));
Status = UlpTcInitializeTcpDevice();
if (!NT_SUCCESS(Status))
goto cleanup;
UlTrace( TC, ("Ul!UlTcInitialize: InitializeTcp Status %08lx \n", Status ));
Status = UlpTcGetFriendlyNames();
if (!NT_SUCCESS(Status))
goto cleanup;
UlTrace( TC, ("Ul!UlTcInitialize: GetFriendlyNames Status %08lx \n", Status ));
Status = UlpTcRegisterForCallbacks();
if (!NT_SUCCESS(Status))
goto cleanup;
UlTrace( TC, ("Ul!UlTcInitialize: UlpTcRegisterForCallbacks Status %08lx \n", Status ));
//
// Success !
//
g_InitTciCalled = TRUE;
}
cleanup:
if (!NT_SUCCESS(Status))
{
NTSTATUS TempStatus;
UlTrace( TC, ("Ul!UlTcInitialize: FAILURE %08lx \n", Status ));
TempStatus = UlDeleteResource( &g_pUlNonpagedData->TciIfcResource );
ASSERT(NT_SUCCESS(TempStatus));
//
// Do not forget to DeRegister Gpc Client & Close Device Handle
//
if (g_GpcClientHandle != NULL)
{
UlpTcDeRegisterGpcClient();
ASSERT(g_GpcFileHandle);
ZwClose(g_GpcFileHandle);
UlTrace( TC, ("Ul!UlTcInitialize: Gpc Device Handle Closed.\n" ));
}
if (g_TcpDeviceHandle != NULL)
{
ZwClose(g_TcpDeviceHandle);
g_TcpDeviceHandle=NULL;
UlTrace( TC, ("Ul!UlTcInitialize: Tcp Device Handle Closed.\n" ));
}
}
return Status;
}
/***************************************************************************++
Routine Description:
UlTcTerminate :
Terminates the TCI module by releasing our TCI resource and
cleaning up all the qos stuff.
Return Value:
NTSTATUS - Completion status.
--***************************************************************************/
VOID
UlTcTerminate(
VOID
)
{
NTSTATUS Status;
PAGED_CODE();
Status = STATUS_SUCCESS;
UlTrace( TC, ("Ul!UlTcTerminate: ... \n" ));
if (g_InitTciCalled)
{
//
// No more Wmi callbacks for interface changes
//
if (g_TcInterfaceUpNotificationObject!=NULL)
{
ObDereferenceObject(g_TcInterfaceUpNotificationObject);
g_TcInterfaceUpNotificationObject=NULL;
}
if(g_TcInterfaceDownNotificationObject!=NULL)
{
ObDereferenceObject(g_TcInterfaceDownNotificationObject);
g_TcInterfaceDownNotificationObject = NULL;
}
if(g_TcInterfaceChangeNotificationObject!=NULL)
{
ObDereferenceObject(g_TcInterfaceChangeNotificationObject);
g_TcInterfaceChangeNotificationObject = NULL;
}
//
// Make sure terminate all the QoS stuff
//
Status = UlpTcReleaseAll();
ASSERT(NT_SUCCESS(Status));
if (g_TcpDeviceHandle != NULL)
{
ZwClose(g_TcpDeviceHandle);
g_TcpDeviceHandle = NULL;
}
Status = UlDeleteResource(
&g_pUlNonpagedData->TciIfcResource
);
ASSERT(NT_SUCCESS(Status));
g_InitTciCalled = FALSE;
}
UlTrace( TC, ("Ul!UlTcTerminate: Completed.\n" ));
}
/***************************************************************************++
Routine Description:
UlpTcInitializeGpc :
It will open the Gpc file handle and attempt to register as Gpc
client.
Return Value:
NTSTATUS - Completion status.
--***************************************************************************/
NTSTATUS
UlpTcInitializeGpc(
VOID
)
{
NTSTATUS Status;
IO_STATUS_BLOCK IoStatusBlock;
UNICODE_STRING GpcNameString;
OBJECT_ATTRIBUTES GpcObjAttribs;
Status = STATUS_SUCCESS;
//
// Open Gpc Device Handle
//
RtlInitUnicodeString(&GpcNameString, DD_GPC_DEVICE_NAME);
InitializeObjectAttributes(&GpcObjAttribs,
&GpcNameString,
OBJ_CASE_INSENSITIVE | UL_KERNEL_HANDLE,
NULL,
NULL
);
Status = ZwCreateFile(&g_GpcFileHandle,
SYNCHRONIZE | FILE_READ_DATA | FILE_WRITE_DATA,
&GpcObjAttribs,
&IoStatusBlock,
NULL,
FILE_ATTRIBUTE_NORMAL,
FILE_SHARE_READ | FILE_SHARE_WRITE,
FILE_OPEN_IF,
0,
NULL,
0
);
if (!NT_SUCCESS(Status))
{
goto end;
}
ASSERT( g_GpcFileHandle != NULL );
UlTrace( TC, ("Ul!UlpTcInitializeGpc: Gpc Device Opened. %p\n",
g_GpcFileHandle ));
//
// Register as GPC_CF_QOS Gpc Client
//
Status = UlpTcRegisterGpcClient(GPC_CF_QOS);
end:
return Status;
}
/***************************************************************************++
Routine Description:
UlpTcRegisterGpcClient :
Will build up the necessary structures and make a register call down
to Gpc
Arguments:
CfInfoType - Should be GPC_CF_QOS for our purposes.
Return Value:
NTSTATUS - Completion status.
--***************************************************************************/
NTSTATUS
UlpTcRegisterGpcClient(
IN ULONG CfInfoType
)
{
NTSTATUS Status;
GPC_REGISTER_CLIENT_REQ GpcReq;
GPC_REGISTER_CLIENT_RES GpcRes;
ULONG InBuffSize;
ULONG OutBuffSize;
IO_STATUS_BLOCK IoStatBlock;
Status = STATUS_SUCCESS;
if ( g_GpcFileHandle == NULL )
{
return STATUS_INVALID_PARAMETER;
}
InBuffSize = sizeof(GPC_REGISTER_CLIENT_REQ);
OutBuffSize = sizeof(GPC_REGISTER_CLIENT_RES);
//
// In HTTP we should only register for GPC_CF_QOS.
//
ASSERT(CfInfoType == GPC_CF_QOS);
GpcReq.CfId = CfInfoType;
GpcReq.Flags = GPC_FLAGS_FRAGMENT;
GpcReq.MaxPriorities = 1;
GpcReq.ClientContext = (GPC_CLIENT_HANDLE) 0; // ???????? Possible BUGBUG ...
//GpcReq.ClientContext = (GPC_CLIENT_HANDLE)GetCurrentProcessId(); // process id
Status = UlpTcDeviceControl(g_GpcFileHandle,
NULL,
NULL,
NULL,
&IoStatBlock,
IOCTL_GPC_REGISTER_CLIENT,
&GpcReq,
InBuffSize,
&GpcRes,
OutBuffSize
);
if (!NT_SUCCESS(Status))
{
UlTrace( TC, ("Ul!UlpTcRegisterGpcClient: FAILURE 1 %08lx \n", Status ));
goto end;
}
Status = GpcRes.Status;
if ( NT_SUCCESS(Status) )
{
g_GpcClientHandle = GpcRes.ClientHandle;
UlTrace( TC, ("Ul!UlpTcRegisterGpcClient: Gpc Client %p Registered.\n",
g_GpcClientHandle
));
}
else
{
g_GpcClientHandle = NULL;
UlTrace( TC, ("Ul!UlpTcRegisterGpcClient: FAILURE 2 %08lx \n", Status ));
}
end:
return Status;
}
/***************************************************************************++
Routine Description:
UlpTcDeRegisterGpcClient :
Self explainatory.
Return Value:
NTSTATUS - Completion status.
--***************************************************************************/
NTSTATUS
UlpTcDeRegisterGpcClient(
VOID
)
{
NTSTATUS Status;
GPC_DEREGISTER_CLIENT_REQ GpcReq;
GPC_DEREGISTER_CLIENT_RES GpcRes;
ULONG InBuffSize;
ULONG OutBuffSize;
IO_STATUS_BLOCK IoStatBlock;
Status = STATUS_SUCCESS;
if (g_GpcFileHandle == NULL && g_GpcClientHandle == NULL)
{
return STATUS_INVALID_PARAMETER;
}
InBuffSize = sizeof(GPC_REGISTER_CLIENT_REQ);
OutBuffSize = sizeof(GPC_REGISTER_CLIENT_RES);
GpcReq.ClientHandle = g_GpcClientHandle;
Status = UlpTcDeviceControl(g_GpcFileHandle,
NULL,
NULL,
NULL,
&IoStatBlock,
IOCTL_GPC_DEREGISTER_CLIENT,
&GpcReq,
InBuffSize,
&GpcRes,
OutBuffSize
);
if (!NT_SUCCESS(Status))
{
goto end;
}
Status = GpcRes.Status;
if ( NT_SUCCESS(Status) )
{
g_GpcClientHandle = NULL;
UlTrace( TC, ("Ul!UlpTcDeRegisterGpcClient: Client DeRegistered.\n" ));
}
else
{
UlTrace( TC, ("Ul!UlpTcDeRegisterGpcClient: FAILURE %08lx \n", Status ));
}
end:
return Status;
}
/***************************************************************************++
Routine Description:
UlpTcGetIpAddr :
Return Value:
NTSTATUS - Completion status.
--***************************************************************************/
BOOLEAN
UlpTcGetIpAddr(
IN PADDRESS_LIST_DESCRIPTOR pAddressListDesc,
OUT PULONG pIn_addr,
OUT PULONG pSpecificLinkCtx
)
{
NETWORK_ADDRESS UNALIGNED64 *pAddr;
NETWORK_ADDRESS_IP UNALIGNED64 *pIpNetAddr = NULL;
NETWORK_ADDRESS_IP UNALIGNED64 *p2ndIpNetAddr = NULL;
ULONG cAddr;
ULONG index;
cAddr = pAddressListDesc->AddressList.AddressCount;
if (cAddr == 0)
{
return FALSE;
}
pAddr = (UNALIGNED64 NETWORK_ADDRESS *) &pAddressListDesc->AddressList.Address[0];
for (index = 0; index < cAddr; index++)
{
if (pAddr->AddressType == NDIS_PROTOCOL_ID_TCP_IP)
{
pIpNetAddr = (UNALIGNED64 NETWORK_ADDRESS_IP *)&pAddr->Address[0];
break;
}
pAddr = (UNALIGNED64 NETWORK_ADDRESS *)(((PUCHAR)pAddr)
+ pAddr->AddressLength
+ FIELD_OFFSET(NETWORK_ADDRESS, Address));
}
// Findout the SpecificLinkCtx (Remote IP address) for WAN links
if( pAddressListDesc->MediaType == NdisMediumWan &&
index+1 < cAddr )
{
//
// There is another address that contains
// the remote client address
// this should be used as the link ID
//
pAddr = (UNALIGNED64 NETWORK_ADDRESS *)(((PUCHAR)pAddr)
+ pAddr->AddressLength
+ FIELD_OFFSET(NETWORK_ADDRESS, Address));
if (pAddr->AddressType == NDIS_PROTOCOL_ID_TCP_IP)
{
//
// Parse the second IP address,
// this would be the remote IP address for dialin WAN
//
p2ndIpNetAddr = (UNALIGNED64 NETWORK_ADDRESS_IP *)&pAddr->Address[0];
*pSpecificLinkCtx = p2ndIpNetAddr->in_addr;
}
}
if ( pIpNetAddr )
{
(*pIn_addr) = pIpNetAddr->in_addr;
return TRUE;
}
else
{
return FALSE;
}
}
/***************************************************************************++
Routine Description:
UlpTcInitializeTcpDevice :
Arguments:
Return Value:
NTSTATUS - Completion status.
--***************************************************************************/
NTSTATUS
UlpTcInitializeTcpDevice(
VOID
)
{
NTSTATUS Status;
IO_STATUS_BLOCK IoStatusBlock;
UNICODE_STRING TcpNameString;
OBJECT_ATTRIBUTES TcpObjAttribs;
Status = STATUS_SUCCESS;
//
// Open Gpc Device
//
RtlInitUnicodeString(&TcpNameString, DD_TCP_DEVICE_NAME);
InitializeObjectAttributes(&TcpObjAttribs,
&TcpNameString,
OBJ_CASE_INSENSITIVE | UL_KERNEL_HANDLE,
NULL,
NULL);
Status = ZwCreateFile( &g_TcpDeviceHandle,
GENERIC_EXECUTE,
&TcpObjAttribs,
&IoStatusBlock,
NULL,
FILE_ATTRIBUTE_NORMAL,
FILE_SHARE_READ | FILE_SHARE_WRITE,
FILE_OPEN_IF,
0,
NULL,
0);
if ( !NT_SUCCESS(Status) )
{
goto end;
}
ASSERT( g_TcpDeviceHandle != NULL );
end:
return Status;
}
/***************************************************************************++
Routine Description:
UlpTcUpdateInterfaceMTU :
Helper function to get the interface MTU sizes by querrying the TCP.
Acquire the TciIfcResource before callling this function.
Make sure that the if_indexes are correct in the individual tc_flow
structures.
WORKITEM:
Current approach is not really scalable in terms of interface count.
But assuming the #of interfaces will be few simplify the code a lot.
--***************************************************************************/
NTSTATUS
UlpTcUpdateInterfaceMTU(
VOID
)
{
NTSTATUS Status;
IO_STATUS_BLOCK IoStatBlock;
TDIObjectID *ID;
TCP_REQUEST_QUERY_INFORMATION_EX trqiInBuf;
TDIEntityID *EntityTable;
TDIEntityID *pEntity;
IFEntry *pIFEntry;
ULONG InBufLen;
ULONG OutBufLen;
ULONG NumEntities;
ULONG NumInterfacesUpdated;
ULONG index;
PLIST_ENTRY pInterfaceEntry;
PUL_TCI_INTERFACE pInterface;
//
// Initialize & Sanity check first
//
Status = STATUS_SUCCESS;
ASSERT( g_TcpDeviceHandle != NULL );
pIFEntry = NULL;
NumInterfacesUpdated = 0;
UlTrace(TC,("Ul!UlpTcUpdateInterfaceMTU ...\n" ));
//
// Enumerate the interfaces by querying the TCP. Get TDI entity count
// search through entities to find out the IF_ENTITYs. The make yet
// another query to get the full interface info including the MTU size
//
InBufLen = sizeof(TCP_REQUEST_QUERY_INFORMATION_EX);
OutBufLen = sizeof(TDIEntityID) * MAX_TDI_ENTITIES;
EntityTable = (TDIEntityID *) UL_ALLOCATE_ARRAY(
PagedPool,
UCHAR,
OutBufLen,
UL_TCI_GENERIC_POOL_TAG
);
if (EntityTable == NULL)
{
Status = STATUS_NO_MEMORY;
goto end;
}
RtlZeroMemory(EntityTable,OutBufLen);
RtlZeroMemory(&trqiInBuf,sizeof(TCP_REQUEST_QUERY_INFORMATION_EX));
ID = &(trqiInBuf.ID);
ID->toi_entity.tei_entity = GENERIC_ENTITY;
ID->toi_entity.tei_instance = 0;
ID->toi_class = INFO_CLASS_GENERIC;
ID->toi_type = INFO_TYPE_PROVIDER;
ID->toi_id = ENTITY_LIST_ID;
Status = UlpTcDeviceControl(
g_TcpDeviceHandle,
NULL,
NULL,
NULL,
&IoStatBlock,
IOCTL_TCP_QUERY_INFORMATION_EX,
&trqiInBuf,
InBufLen,
EntityTable,
OutBufLen
);
if (!NT_SUCCESS(Status))
{
UlTrace(TC,("Ul!UlpTcUpdateInterfaceMTU: Get TDIEntityID failed\n"));
goto end;
}
// Now we have all the TDI entities
NumEntities = ((ULONG)(IoStatBlock.Information)) / sizeof(TDIEntityID);
UlTrace(TC,("Ul!UlpTcUpdateInterfaceMTU: #Of TDI Entities %d\n", NumEntities));
// Search through the interface entries
for (index=0,pEntity=EntityTable; index < NumEntities; index++,pEntity++)
{
if (pEntity->tei_entity == IF_ENTITY)
{
// Allocate a buffer for the querry
if (pIFEntry == NULL)
{
OutBufLen = sizeof(IFEntry) + MAX_IFDESCR_LEN;
pIFEntry = (IFEntry *) UL_ALLOCATE_ARRAY(
PagedPool,
UCHAR,
OutBufLen,
UL_TCI_GENERIC_POOL_TAG
);
if (pIFEntry == NULL)
{
Status = STATUS_NO_MEMORY;
goto end;
}
RtlZeroMemory(pIFEntry,OutBufLen);
}
// Get the full IFEntry. It's a pitty that we only look at the
// Mtu size after getting such a big structure.
InBufLen = sizeof(TCP_REQUEST_QUERY_INFORMATION_EX);
RtlZeroMemory(&trqiInBuf,sizeof(TCP_REQUEST_QUERY_INFORMATION_EX));
ID = &(trqiInBuf.ID);
ID->toi_entity.tei_entity = IF_ENTITY;
ID->toi_entity.tei_instance = pEntity->tei_instance;
ID->toi_class = INFO_CLASS_PROTOCOL;
ID->toi_type = INFO_TYPE_PROVIDER;
ID->toi_id = IF_MIB_STATS_ID;
Status = UlpTcDeviceControl(
g_TcpDeviceHandle,
NULL,
NULL,
NULL,
&IoStatBlock,
IOCTL_TCP_QUERY_INFORMATION_EX,
&trqiInBuf,
InBufLen,
pIFEntry,
OutBufLen
);
if (!NT_SUCCESS(Status))
{
UlTrace(TC,("Ul!UlpTcUpdateInterfaceMTU: Get_IF_MIB_STATS_ID failed\n" ));
goto end;
}
// Now we have the interface info including the mtu size for this entity
// Find the corresponding UL_TCI_INTERFACE by looking at the index
pInterfaceEntry = g_TciIfcListHead.Flink;
while ( pInterfaceEntry != &g_TciIfcListHead )
{
pInterface = CONTAINING_RECORD(
pInterfaceEntry,
UL_TCI_INTERFACE,
Linkage
);
if (pIFEntry->if_index == pInterface->IfIndex)
{
pInterface->MTUSize = pIFEntry->if_mtu;
UlTrace(TC,
("Ul!UlpTcUpdateInterfaceMTU: if_index %d if_mtu %d if_speed %d\n",
pIFEntry->if_index, pIFEntry->if_mtu, pIFEntry->if_speed ));
UL_DUMP_TC_INTERFACE(pInterface);
NumInterfacesUpdated++;
}
// search through next interface
pInterfaceEntry = pInterfaceEntry->Flink;
}
}
}
UlTrace(TC,("Ul!UlpTcUpdateInterfaceMTU: %d interfaces updated.\n",
NumInterfacesUpdated ));
end:
// Whine about the problems
if (!NT_SUCCESS(Status))
{
UlTrace( TC,("Ul!UlpTcUpdateInterfaceMTU: FAILED Status %08lx\n",
Status ));
}
// Release the private buffers
if ( pIFEntry != NULL )
{
UL_FREE_POOL( pIFEntry, UL_TCI_GENERIC_POOL_TAG );
}
if ( EntityTable != NULL )
{
UL_FREE_POOL( EntityTable, UL_TCI_GENERIC_POOL_TAG );
}
return Status;
}
/***************************************************************************++
Routine Description:
UlpTcGetInterfaceIndex :
Helper function to get the interface index from TCP for our internal
interface structure.
Arguments:
PUL_TCI_INTERFACE pIntfc - The interface we will find the index for.
--***************************************************************************/
NTSTATUS
UlpTcGetInterfaceIndex(
IN PUL_TCI_INTERFACE pIntfc
)
{
NTSTATUS Status;
IPAddrEntry *pIpAddrTbl;
ULONG IpAddrTblSize;
ULONG n,k;
IO_STATUS_BLOCK IoStatBlock;
TDIObjectID *ID;
TCP_REQUEST_QUERY_INFORMATION_EX trqiInBuf;
ULONG InBuffLen;
ULONG NumEntries;
//
// Initialize & Sanity check first
//
Status = STATUS_SUCCESS;
NumEntries = 0;
pIpAddrTbl = NULL;
//
// BUGBUG should get ip address size and allocate enough buffer.
// Or handle the STATUS_BUFFER_TOO_SMALL
//
IpAddrTblSize = sizeof(IPAddrEntry) * 1024;
UlTrace(TC,("Ul!UlpTcGetInterfaceIndex: ....\n" ));
ASSERT( g_TcpDeviceHandle != NULL );
if (pIntfc->IpAddr)
{
// Allocate a private buffer to retrieve Ip Address table from TCP
pIpAddrTbl = (IPAddrEntry *) UL_ALLOCATE_ARRAY(
PagedPool,
UCHAR,
IpAddrTblSize,
UL_TCI_GENERIC_POOL_TAG
);
if (pIpAddrTbl == NULL)
{
Status = STATUS_NO_MEMORY;
goto end;
}
// Send down an Ioctl to Tcp to get the ip address table
RtlZeroMemory(pIpAddrTbl,IpAddrTblSize);
RtlZeroMemory(&trqiInBuf,sizeof(TCP_REQUEST_QUERY_INFORMATION_EX));
InBuffLen = sizeof(TCP_REQUEST_QUERY_INFORMATION_EX);
ID = &(trqiInBuf.ID);
ID->toi_entity.tei_entity = CL_NL_ENTITY;
ID->toi_entity.tei_instance = 0;
ID->toi_class = INFO_CLASS_PROTOCOL;
ID->toi_type = INFO_TYPE_PROVIDER;
ID->toi_id = IP_MIB_ADDRTABLE_ENTRY_ID;
Status = UlpTcDeviceControl(
g_TcpDeviceHandle,
NULL,
NULL,
NULL,
&IoStatBlock,
IOCTL_TCP_QUERY_INFORMATION_EX,
&trqiInBuf,
InBuffLen,
pIpAddrTbl,
IpAddrTblSize
);
if (!NT_SUCCESS(Status))
{
goto end;
}
// Look at how many entries were written to the output buffer (pIpAddrTbl)
NumEntries = (((ULONG)IoStatBlock.Information)/sizeof(IPAddrEntry));
UlTrace(TC,("Ul!UlpTcGetInterfaceIndex: NumEntries %d\n", NumEntries ));
//
// Search for the matching IP address to IpAddr
// in the table we got back from the stack
//
for (k=0; k<NumEntries; k++)
{
if (pIpAddrTbl[k].iae_addr == pIntfc->IpAddr)
{
// Found it found it! Get the index baby.
pIntfc->IfIndex = pIpAddrTbl[k].iae_index;
UlTrace(TC,("Ul!UlpTcGetInterfaceIndex: got for index %d\n",
pIntfc->IfIndex ));
break;
}
}
}
end:
if (!NT_SUCCESS(Status))
{
UlTrace(TC,("Ul!UlpTcGetInterfaceIndex: FAILED Status %08lx\n",
Status));
}
if ( pIpAddrTbl != NULL )
{
UL_FREE_POOL( pIpAddrTbl, UL_TCI_GENERIC_POOL_TAG );
}
return Status;
}
/***************************************************************************++
Routine Description:
UlpTcAllocateInterface :
... fairly straightforward ...
Argument:
Return Value:
PUL_TCI_INTERFACE - Newly allocated interface structure
--***************************************************************************/
PUL_TCI_INTERFACE
UlpTcAllocateInterface(
IN ULONG DescSize,
IN PADDRESS_LIST_DESCRIPTOR Desc,
IN ULONG NameLength,
IN PUCHAR Name,
IN ULONG InstanceIDLength,
IN PUCHAR InstanceID
)
{
PUL_TCI_INTERFACE pTcIfc;
//
// Sanity Checks
//
ASSERT(NameLength <= MAX_STRING_LENGTH);
ASSERT(InstanceIDLength <= MAX_STRING_LENGTH);
//
// Allocate a new interface structure & initialize it
//
pTcIfc = UL_ALLOCATE_STRUCT(
PagedPool,
UL_TCI_INTERFACE,
UL_TCI_INTERFACE_POOL_TAG
);
if ( pTcIfc == NULL )
{
return NULL;
}
RtlZeroMemory( pTcIfc, sizeof(UL_TCI_INTERFACE) );
pTcIfc->Signature = UL_TCI_INTERFACE_POOL_TAG;
InitializeListHead( &pTcIfc->FlowList );
// Variable size addresslist
pTcIfc->pAddressListDesc = (PADDRESS_LIST_DESCRIPTOR)
UL_ALLOCATE_ARRAY(
PagedPool,
UCHAR,
DescSize,
UL_TCI_INTERFACE_POOL_TAG
);
if ( pTcIfc->pAddressListDesc == NULL )
{
UL_FREE_POOL_WITH_SIG(pTcIfc, UL_TCI_INTERFACE_POOL_TAG);
return NULL;
}
pTcIfc->AddrListBytesCount = DescSize;
// Copy the instance name string data
RtlCopyMemory(pTcIfc->Name,Name,NameLength);
pTcIfc->NameLength = (USHORT)NameLength;
pTcIfc->Name[NameLength/sizeof(WCHAR)] = UNICODE_NULL;
// Copy the instance ID string data
RtlCopyMemory(pTcIfc->InstanceID,InstanceID,InstanceIDLength);
pTcIfc->InstanceIDLength = (USHORT)InstanceIDLength;
pTcIfc->InstanceID[InstanceIDLength/sizeof(WCHAR)] = UNICODE_NULL;
// Copy the Description data and extract the corresponding ip address
RtlCopyMemory(pTcIfc->pAddressListDesc, Desc, DescSize);
// IP Address of the interface is hidden in this desc data
// we will find out and save it for faster lookup.
pTcIfc->IsQoSEnabled =
UlpTcGetIpAddr( pTcIfc->pAddressListDesc,
&pTcIfc->IpAddr,
&pTcIfc->SpecificLinkCtx
);
return pTcIfc;
}
VOID
UlpTcFreeInterface(
IN OUT PUL_TCI_INTERFACE pTcIfc
)
{
// Clean up the interface & addreslist pointer
if (pTcIfc)
{
if (pTcIfc->pAddressListDesc)
{
UL_FREE_POOL(pTcIfc->pAddressListDesc,
UL_TCI_INTERFACE_POOL_TAG
);
}
UL_FREE_POOL_WITH_SIG(pTcIfc, UL_TCI_INTERFACE_POOL_TAG);
}
}
/***************************************************************************++
Routine Description:
UlpTcGetFriendlyNames :
Make a Wmi Querry to get the firendly names of all interfaces.
Its basically replica of the tcdll enumerate interfaces call.
This function also allocates the global interface list. If it's not
successfull it doesn't though.
Return Value:
NTSTATUS - Completion status.
--***************************************************************************/
NTSTATUS
UlpTcGetFriendlyNames(
VOID
)
{
NTSTATUS Status;
PVOID WmiObject;
ULONG MyBufferSize;
PWNODE_ALL_DATA pWnode;
PWNODE_ALL_DATA pWnodeBuffer;
PUL_TCI_INTERFACE pTcIfc;
GUID QoSGuid;
PLIST_ENTRY pEntry;
PUL_TCI_INTERFACE pInterface;
//
// Initialize defaults
//
Status = STATUS_SUCCESS;
WmiObject = NULL;
pWnodeBuffer = NULL;
pTcIfc = NULL;
MyBufferSize = UL_DEFAULT_WMI_QUERY_BUFFER_SIZE;
QoSGuid = GUID_QOS_TC_SUPPORTED;
//
// Get a WMI block handle to the GUID_QOS_SUPPORTED
//
Status = IoWMIOpenBlock( (GUID *) &QoSGuid, 0, &WmiObject );
if (!NT_SUCCESS(Status))
{
if (Status == STATUS_WMI_GUID_NOT_FOUND)
{
// This means there is no TC data provider (which's Psched)
UlTrace(TC,("Ul!UlpTcGetFriendlyNames: PSCHED hasn't been installed !\n"));
}
else
{
UlTrace(TC,("Ul!UlpTcGetFriendlyNames:IoWMIOpenBlock FAILED Status %08lx\n",
Status));
}
return Status;
}
//
// Mark that PSched is installed
//
g_PSchedInstalled = 1;
do
{
//
// Allocate a private buffer to retrieve all wnodes
//
pWnodeBuffer = (PWNODE_ALL_DATA) UL_ALLOCATE_ARRAY(
NonPagedPool,
UCHAR,
MyBufferSize,
UL_TCI_WMI_POOL_TAG
);
if (pWnodeBuffer == NULL)
{
ObDereferenceObject(WmiObject);
return STATUS_NO_MEMORY;
}
__try
{
Status = IoWMIQueryAllData(WmiObject, &MyBufferSize, pWnodeBuffer);
UlTrace( TC,
("Ul!UlpTcGetFriendlyNames: IoWMIQueryAllData Status %08lx\n",
Status
));
}
__except ( UL_EXCEPTION_FILTER() )
{
Status = GetExceptionCode();
}
if (Status == STATUS_BUFFER_TOO_SMALL)
{
//
// Failed since the buffer was too small.
// Release the buffer and double the size.
//
MyBufferSize *= 2;
UL_FREE_POOL( pWnodeBuffer, UL_TCI_WMI_POOL_TAG );
pWnodeBuffer = NULL;
}
} while (Status == STATUS_BUFFER_TOO_SMALL);
if (NT_SUCCESS(Status))
{
ULONG dwInstanceNum;
ULONG InstanceSize;
PULONG lpdwNameOffsets;
BOOLEAN bFixedSize = FALSE;
USHORT usNameLength;
ULONG DescSize;
PTC_SUPPORTED_INFO_BUFFER pTcInfoBuffer;
pWnode = pWnodeBuffer;
ASSERT(pWnode->WnodeHeader.Flags & WNODE_FLAG_ALL_DATA);
do
{
//
// Check for fixed instance size
//
if (pWnode->WnodeHeader.Flags & WNODE_FLAG_FIXED_INSTANCE_SIZE)
{
InstanceSize = pWnode->FixedInstanceSize;
bFixedSize = TRUE;
pTcInfoBuffer =
(PTC_SUPPORTED_INFO_BUFFER)OffsetToPtr(pWnode,
pWnode->DataBlockOffset);
}
//
// Get a pointer to the array of offsets to the instance names
//
lpdwNameOffsets = (PULONG) OffsetToPtr(pWnode,
pWnode->OffsetInstanceNameOffsets);
for ( dwInstanceNum = 0;
dwInstanceNum < pWnode->InstanceCount;
dwInstanceNum++ )
{
usNameLength = *(PUSHORT)OffsetToPtr(pWnode,lpdwNameOffsets[dwInstanceNum]);
//
// Length and offset for variable data
//
if ( !bFixedSize )
{
InstanceSize =
pWnode->OffsetInstanceDataAndLength[dwInstanceNum].LengthInstanceData;
pTcInfoBuffer =
(PTC_SUPPORTED_INFO_BUFFER)OffsetToPtr(
(PBYTE)pWnode,
pWnode->OffsetInstanceDataAndLength[dwInstanceNum].OffsetInstanceData);
}
//
// We have all that is needed.
//
ASSERT(usNameLength < MAX_STRING_LENGTH);
DescSize = InstanceSize - FIELD_OFFSET(TC_SUPPORTED_INFO_BUFFER, AddrListDesc);
//
// Allocate a new interface structure & initialize it with
// the wmi data we have acquired.
//
pTcIfc = UlpTcAllocateInterface(
DescSize,
&pTcInfoBuffer->AddrListDesc,
usNameLength,
(PUCHAR) OffsetToPtr(pWnode,lpdwNameOffsets[dwInstanceNum] + sizeof(USHORT)),
pTcInfoBuffer->InstanceIDLength,
(PUCHAR) &pTcInfoBuffer->InstanceID[0]
);
if ( pTcIfc == NULL )
{
Status = STATUS_NO_MEMORY;
goto end;
}
//
// Get the interface index from TCP
//
Status = UlpTcGetInterfaceIndex( pTcIfc );
ASSERT(NT_SUCCESS(Status));
//
// Add this interface to the global interface list
//
UlAcquireResourceExclusive(&g_pUlNonpagedData->TciIfcResource, TRUE);
InsertTailList(&g_TciIfcListHead, &pTcIfc->Linkage );
UlReleaseResource(&g_pUlNonpagedData->TciIfcResource);
//
// Set to Null so we don't try to cleanup after we insert it
// to the global list.
//
pTcIfc = NULL;
}
//
// Update Wnode to point to next node
//
if ( pWnode->WnodeHeader.Linkage != 0)
{
pWnode = (PWNODE_ALL_DATA) OffsetToPtr( pWnode,
pWnode->WnodeHeader.Linkage);
}
else
{
pWnode = NULL;
}
}
while ( pWnode != NULL && NT_SUCCESS(Status) );
//
// Update the mtu sizes for all interfaces now.
//
UlAcquireResourceExclusive(&g_pUlNonpagedData->TciIfcResource, TRUE);
UlpTcUpdateInterfaceMTU();
UlReleaseResource(&g_pUlNonpagedData->TciIfcResource);
UlTrace(TC,("Ul!UlpTcGetFriendlyNames: got all the names.\n"));
}
end:
if (!NT_SUCCESS(Status))
{
UlTrace(TC,("Ul!UlpTcGetFriendlyNames: FAILED Status %08lx\n",
Status
));
if (pTcIfc)
{
UlpTcFreeInterface( pTcIfc );
}
//
// Cleanup the partially done interface list if not empty
//
while ( !IsListEmpty( &g_TciIfcListHead ) )
{
pEntry = g_TciIfcListHead.Flink;
pInterface = CONTAINING_RECORD( pEntry,
UL_TCI_INTERFACE,
Linkage
);
RemoveEntryList( pEntry );
UlpTcFreeInterface( pInterface );
}
}
//
// Release resources and close WMI handle
//
if (WmiObject != NULL)
{
ObDereferenceObject(WmiObject);
}
if (pWnodeBuffer)
{
UL_FREE_POOL(pWnodeBuffer, UL_TCI_WMI_POOL_TAG);
}
return Status;
}
/***************************************************************************++
Routine Description:
UlpTcReleaseAll :
Close all interfaces, all flows and all filters.
Also deregister GPC clients and release all TC ineterfaces.
Return Value:
NTSTATUS - Completion status.
--***************************************************************************/
NTSTATUS
UlpTcReleaseAll(
VOID
)
{
NTSTATUS Status;
//
// Close all interfaces their flows & filters
//
UlpTcCloseAllInterfaces();
//
// DeRegister the QoS GpcClient
//
Status = UlpTcDeRegisterGpcClient();
if (!NT_SUCCESS(Status))
{
UlTrace( TC, ("Ul!UlpTcReleaseAll: FAILURE %08lx \n", Status ));
}
//
// Finally close our gpc file handle
//
ZwClose(g_GpcFileHandle);
return Status;
}
/***************************************************************************++
Routine Description:
UlpTcCloseAllInterfaces :
Return Value:
NTSTATUS - Completion status.
--***************************************************************************/
NTSTATUS
UlpTcCloseAllInterfaces(
VOID
)
{
NTSTATUS Status;
PLIST_ENTRY pEntry;
PUL_TCI_INTERFACE pInterface;
Status = STATUS_SUCCESS;
UlAcquireResourceExclusive(&g_pUlNonpagedData->TciIfcResource, TRUE);
//
// Close all interfaces in our global list
//
while ( !IsListEmpty( &g_TciIfcListHead ) )
{
pEntry = g_TciIfcListHead.Flink;
pInterface = CONTAINING_RECORD( pEntry,
UL_TCI_INTERFACE,
Linkage
);
UlpTcCloseInterface( pInterface );
RemoveEntryList( pEntry );
UlpTcFreeInterface( pInterface );
}
UlReleaseResource(&g_pUlNonpagedData->TciIfcResource);
return Status;
}
/***************************************************************************++
Routine Description:
Cleans up all the flows on the interface.
Arguments:
pInterface - to be closed
--***************************************************************************/
NTSTATUS
UlpTcCloseInterface(
PUL_TCI_INTERFACE pInterface
)
{
NTSTATUS Status;
PLIST_ENTRY pEntry;
PUL_TCI_FLOW pFlow;
ASSERT(IS_VALID_TCI_INTERFACE(pInterface));
//
// Go clean up all flows for the interface and remove itself as well
//
Status = STATUS_SUCCESS;
while (!IsListEmpty(&pInterface->FlowList))
{
pEntry= pInterface->FlowList.Flink;
pFlow = CONTAINING_RECORD(
pEntry,
UL_TCI_FLOW,
Linkage
);
ASSERT(IS_VALID_TCI_FLOW(pFlow));
//
// Remove flow from the corresponding cg's flowlist
// as well if it's not a global flow. We understand
// that by looking at its config group pointer.
//
if (pFlow->pConfigGroup)
{
ASSERT(IS_VALID_CONFIG_GROUP(pFlow->pConfigGroup));
RemoveEntryList(&pFlow->Siblings);
pFlow->Siblings.Flink = pFlow->Siblings.Blink = NULL;
pFlow->pConfigGroup = NULL;
}
//
// Now remove from the interface.
//
Status = UlpTcDeleteFlow(pFlow);
ASSERT(NT_SUCCESS(Status));
}
UlTrace(TC,("Ul!UlpTcCloseInterface: All flows deleted on Ifc @ %p\n",
pInterface ));
return Status;
}
/***************************************************************************++
Routine Description:
UlpTcWalkWnode :
Arguments:
... the WMI provided data buffer ...
--***************************************************************************/
NTSTATUS
UlpTcWalkWnode(
IN PWNODE_HEADER pWnodeHdr,
IN PUL_TC_NOTIF_HANDLER pNotifHandler
)
{
NTSTATUS Status;
PWCHAR NamePtr;
USHORT NameSize;
PUCHAR DataBuffer;
ULONG DataSize;
ULONG Flags;
PULONG NameOffset;
//
// Try to capture the data frm WMI Buffer
//
ASSERT(pNotifHandler);
Status = STATUS_SUCCESS;
Flags = pWnodeHdr->Flags;
if (Flags & WNODE_FLAG_ALL_DATA)
{
//
// WNODE_ALL_DATA structure has multiple interfaces
//
PWNODE_ALL_DATA pWnode = (PWNODE_ALL_DATA)pWnodeHdr;
ULONG Instance;
UlTrace(TC,("Ul!UlpTcWalkWnode: ALL_DATA ... \n" ));
NameOffset = (PULONG) OffsetToPtr(pWnode,
pWnode->OffsetInstanceNameOffsets );
DataBuffer = (PUCHAR) OffsetToPtr(pWnode,
pWnode->DataBlockOffset);
for (Instance = 0;
Instance < pWnode->InstanceCount;
Instance++)
{
// Instance Name
NamePtr = (PWCHAR) OffsetToPtr(pWnode,NameOffset[Instance] + sizeof(USHORT));
NameSize = * (PUSHORT) OffsetToPtr(pWnode,NameOffset[Instance]);
// Instance Data
if ( Flags & WNODE_FLAG_FIXED_INSTANCE_SIZE )
{
DataSize = pWnode->FixedInstanceSize;
}
else
{
DataSize =
pWnode->OffsetInstanceDataAndLength[Instance].LengthInstanceData;
DataBuffer =
(PUCHAR)OffsetToPtr(pWnode,
pWnode->OffsetInstanceDataAndLength[Instance].OffsetInstanceData);
}
// Call the handler
pNotifHandler( NamePtr, NameSize, (PTC_INDICATION_BUFFER) DataBuffer, DataSize );
}
}
else if (Flags & WNODE_FLAG_SINGLE_INSTANCE)
{
//
// WNODE_SINGLE_INSTANCE structure has only one instance
//
PWNODE_SINGLE_INSTANCE pWnode = (PWNODE_SINGLE_INSTANCE)pWnodeHdr;
if (Flags & WNODE_FLAG_STATIC_INSTANCE_NAMES)
{
return STATUS_SUCCESS;
}
UlTrace(TC,("Ul!UlpTcWalkWnode: SINGLE_INSTANCE ... \n" ));
NamePtr = (PWCHAR)OffsetToPtr(pWnode,pWnode->OffsetInstanceName + sizeof(USHORT));
NameSize = * (USHORT *) OffsetToPtr(pWnode,pWnode->OffsetInstanceName);
// Instance Data
DataSize = pWnode->SizeDataBlock;
DataBuffer = (PUCHAR)OffsetToPtr (pWnode, pWnode->DataBlockOffset);
// Call the handler
pNotifHandler( NamePtr, NameSize, (PTC_INDICATION_BUFFER) DataBuffer, DataSize );
}
else if (Flags & WNODE_FLAG_SINGLE_ITEM)
{
//
// WNODE_SINGLE_ITEM is almost identical to single_instance
//
PWNODE_SINGLE_ITEM pWnode = (PWNODE_SINGLE_ITEM)pWnodeHdr;
if (Flags & WNODE_FLAG_STATIC_INSTANCE_NAMES)
{
return STATUS_SUCCESS;
}
UlTrace(TC,("Ul!UlpTcWalkWnode: SINGLE_ITEM ... \n" ));
NamePtr = (PWCHAR)OffsetToPtr(pWnode,pWnode->OffsetInstanceName + sizeof(USHORT));
NameSize = * (USHORT *) OffsetToPtr(pWnode, pWnode->OffsetInstanceName);
// Instance Data
DataSize = pWnode->SizeDataItem;
DataBuffer = (PUCHAR)OffsetToPtr (pWnode, pWnode->DataBlockOffset);
// Call the handler
pNotifHandler( NamePtr, NameSize, (PTC_INDICATION_BUFFER) DataBuffer, DataSize );
}
return Status;
}
/***************************************************************************++
Routine Description:
UlpTcHandleIfcUp :
This functions handles the interface change notifications.
We register for the corresponding notifications during init.
Arguments:
PVOID Wnode - PSched data provided with WMI way
--***************************************************************************/
VOID
UlpTcHandleIfcUp(
IN PWSTR Name,
IN ULONG NameSize,
IN PTC_INDICATION_BUFFER pTcBuffer,
IN ULONG BufferSize
)
{
NTSTATUS Status;
ULONG AddrListDescSize;
PTC_SUPPORTED_INFO_BUFFER pTcInfoBuffer;
PUL_TCI_INTERFACE pTcIfc;
PUL_TCI_INTERFACE pTcIfcTemp;
PLIST_ENTRY pEntry;
Status = STATUS_SUCCESS;
UlTrace(TC,("Ul!UlpTcHandleIfcUp: Adding %ws %d\n", Name, BufferSize ));
UlAcquireResourceExclusive(&g_pUlNonpagedData->TciIfcResource, TRUE);
//
// Allocate a new interface structure for the newcoming interface
//
AddrListDescSize = BufferSize
- FIELD_OFFSET(TC_INDICATION_BUFFER,InfoBuffer)
- FIELD_OFFSET(TC_SUPPORTED_INFO_BUFFER, AddrListDesc);
UlTrace(TC,("Ul!UlpTcHandleIfcUp: AddrListDescSize %d\n", AddrListDescSize ));
pTcInfoBuffer = & pTcBuffer->InfoBuffer;
pTcIfc = UlpTcAllocateInterface(
AddrListDescSize,
&pTcInfoBuffer->AddrListDesc,
NameSize,
(PUCHAR) Name,
pTcInfoBuffer->InstanceIDLength,
(PUCHAR) &pTcInfoBuffer->InstanceID[0]
);
if ( pTcIfc == NULL )
{
Status = STATUS_NO_MEMORY;
goto end;
}
UL_DUMP_TC_INTERFACE( pTcIfc );
//
// If we are receiving a notification for an interface already exist then
// drop this call. Prevent global interface list corruption if we receive
// inconsistent notifications. But there may be multiple interfaces with
// same zero IPs.
//
pEntry = g_TciIfcListHead.Flink;
while ( pEntry != &g_TciIfcListHead )
{
pTcIfcTemp = CONTAINING_RECORD( pEntry, UL_TCI_INTERFACE, Linkage );
if ((pTcIfc->IpAddr != 0 && pTcIfcTemp->IpAddr == pTcIfc->IpAddr) ||
(wcsncmp(pTcIfcTemp->Name, pTcIfc->Name, NameSize/sizeof(WCHAR))==0))
{
ASSERT(!"Conflict in the global interface list !");
Status = STATUS_CONFLICTING_ADDRESSES;
goto end;
}
pEntry = pEntry->Flink;
}
//
// Get the interface index from TCP.
//
Status = UlpTcGetInterfaceIndex( pTcIfc );
if (!NT_SUCCESS(Status))
goto end;
//
// Insert to the global interface list
//
InsertTailList( &g_TciIfcListHead, &pTcIfc->Linkage );
//
// Update the MTU Size
//
UlpTcUpdateInterfaceMTU();
end:
if (!NT_SUCCESS(Status))
{
UlTrace(TC,("Ul!UlpTcHandleIfcUp: FAILURE %08lx \n", Status ));
if (pTcIfc != NULL)
{
UlpTcFreeInterface(pTcIfc);
}
}
UlReleaseResource(&g_pUlNonpagedData->TciIfcResource);
return;
}
/***************************************************************************++
Routine Description:
UlpTcHandleIfcDown :
This functions handles the interface change notifications.
We register for the corresponding notifications during init.
Arguments:
PVOID Wnode - PSched data provided with WMI way
--***************************************************************************/
VOID
UlpTcHandleIfcDown(
IN PWSTR Name,
IN ULONG NameSize,
IN PTC_INDICATION_BUFFER pTcBuffer,
IN ULONG BufferSize
)
{
NTSTATUS Status;
ULONG AddrListDescSize;
PTC_SUPPORTED_INFO_BUFFER pTcInfoBuffer;
PUL_TCI_INTERFACE pTcIfc;
PUL_TCI_INTERFACE pTcIfcTemp;
PLIST_ENTRY pEntry;
Status = STATUS_SUCCESS;
UlTrace(TC,("Ul!UlpTcHandleIfcDown: Removing %ws\n", Name ));
UlAcquireResourceExclusive(&g_pUlNonpagedData->TciIfcResource, TRUE);
//
// Find the corresponding ifc structure we keep.
//
pTcIfc = NULL;
pEntry = g_TciIfcListHead.Flink;
while ( pEntry != &g_TciIfcListHead )
{
pTcIfcTemp = CONTAINING_RECORD( pEntry, UL_TCI_INTERFACE, Linkage );
if ( wcsncmp(pTcIfcTemp->Name, Name, NameSize) == 0 )
{
pTcIfc = pTcIfcTemp;
break;
}
pEntry = pEntry->Flink;
}
if (pTcIfc == NULL)
{
ASSERT(FALSE);
Status = STATUS_NOT_FOUND;
goto end;
}
//
// Remove this interface and its flows etc ...
//
UlpTcCloseInterface( pTcIfc );
RemoveEntryList( &pTcIfc->Linkage );
UlpTcFreeInterface( pTcIfc );
end:
if (!NT_SUCCESS(Status))
{
UlTrace(TC,("Ul!UlpTcHandleIfcDown: FAILURE %08lx \n", Status ));
}
UlReleaseResource(&g_pUlNonpagedData->TciIfcResource);
return;
}
/***************************************************************************++
Routine Description:
UlpTcHandleIfcChange :
This functions handles the interface change notifications.
We register for the corresponding notifications during init.
Arguments:
PVOID Wnode - PSched data provided with WMI way
--***************************************************************************/
VOID
UlpTcHandleIfcChange(
IN PWSTR Name,
IN ULONG NameSize,
IN PTC_INDICATION_BUFFER pTcBuffer,
IN ULONG BufferSize
)
{
NTSTATUS Status;
ULONG AddrListDescSize;
PTC_SUPPORTED_INFO_BUFFER pTcInfoBuffer;
PUL_TCI_INTERFACE pTcIfc;
PUL_TCI_INTERFACE pTcIfcTemp;
PLIST_ENTRY pEntry;
PADDRESS_LIST_DESCRIPTOR pAddressListDesc;
Status = STATUS_SUCCESS;
UlTrace(TC,("Ul!UlpTcHandleIfcChange: Updating %ws\n", Name ));
UlAcquireResourceExclusive(&g_pUlNonpagedData->TciIfcResource, TRUE);
AddrListDescSize = BufferSize
- FIELD_OFFSET(TC_INDICATION_BUFFER,InfoBuffer)
- FIELD_OFFSET(TC_SUPPORTED_INFO_BUFFER, AddrListDesc);
pTcInfoBuffer = & pTcBuffer->InfoBuffer;
// Find the corresponding ifc structure we keep.
pTcIfc = NULL;
pEntry = g_TciIfcListHead.Flink;
while ( pEntry != &g_TciIfcListHead )
{
pTcIfcTemp = CONTAINING_RECORD( pEntry, UL_TCI_INTERFACE, Linkage );
if ( wcsncmp(pTcIfcTemp->Name, Name, NameSize) == 0 )
{
pTcIfc = pTcIfcTemp;
break;
}
pEntry = pEntry->Flink;
}
if (pTcIfc == NULL)
{
ASSERT(FALSE);
Status = STATUS_NOT_FOUND;
goto end;
}
// Instance id
RtlCopyMemory(pTcIfc->InstanceID,
pTcInfoBuffer->InstanceID,
pTcInfoBuffer->InstanceIDLength
);
pTcIfc->InstanceIDLength = pTcInfoBuffer->InstanceIDLength;
pTcIfc->InstanceID[pTcIfc->InstanceIDLength/sizeof(WCHAR)] = UNICODE_NULL;
// The Description data and extract the corresponding ip address
// ReWrite the fresh data. Size of the description data might be changed
// so wee need to dynamically allocate it everytime changes
pAddressListDesc =
(PADDRESS_LIST_DESCRIPTOR) UL_ALLOCATE_ARRAY(
PagedPool,
UCHAR,
AddrListDescSize,
UL_TCI_INTERFACE_POOL_TAG
);
if ( pAddressListDesc == NULL )
{
Status = STATUS_NO_MEMORY;
goto end;
}
if (pTcIfc->pAddressListDesc)
{
UL_FREE_POOL(pTcIfc->pAddressListDesc,UL_TCI_INTERFACE_POOL_TAG);
}
pTcIfc->pAddressListDesc = pAddressListDesc;
pTcIfc->AddrListBytesCount = AddrListDescSize;
RtlCopyMemory( pTcIfc->pAddressListDesc,
&pTcInfoBuffer->AddrListDesc,
AddrListDescSize
);
// IP Address of the interface is hidden in this desc data
pTcIfc->IsQoSEnabled =
UlpTcGetIpAddr( pTcIfc->pAddressListDesc,
&pTcIfc->IpAddr,
&pTcIfc->SpecificLinkCtx
);
// ReFresh the interface index from TCP.
Status = UlpTcGetInterfaceIndex( pTcIfc );
if (!NT_SUCCESS(Status))
goto end;
// Update the MTU Size
UlpTcUpdateInterfaceMTU();
end:
if (!NT_SUCCESS(Status))
{
UlTrace(TC,("Ul!UlpTcHandleIfcChange: FAILURE %08lx \n", Status ));
}
UlReleaseResource(&g_pUlNonpagedData->TciIfcResource);
return;
}
/***************************************************************************++
Routine Description:
UlTcNotifyCallback :
This callback functions handles the interface change notifications.
We register for the corresponding notifications during init.
Arguments:
PVOID Wnode - PSched data provided with WMI way
--***************************************************************************/
VOID
UlTcNotifyCallback(
IN PVOID pWnode,
IN PVOID Context
)
{
GUID *pGuid;
PWNODE_HEADER pWnodeHeader;
UlTrace( TC, ("Ul!UlTcNotifyCallback: ... \n" ));
pWnodeHeader = (PWNODE_HEADER) pWnode;
pGuid = &pWnodeHeader->Guid;
if (UL_COMPARE_QOS_NOTIFICATION(pGuid,&GUID_QOS_TC_INTERFACE_UP_INDICATION))
{
UlpTcWalkWnode( pWnodeHeader, UlpTcHandleIfcUp );
}
else if
(UL_COMPARE_QOS_NOTIFICATION(pGuid, &GUID_QOS_TC_INTERFACE_DOWN_INDICATION))
{
UlpTcWalkWnode( pWnodeHeader, UlpTcHandleIfcDown );
}
else if
(UL_COMPARE_QOS_NOTIFICATION(pGuid, &GUID_QOS_TC_INTERFACE_CHANGE_INDICATION))
{
UlpTcWalkWnode( pWnodeHeader, UlpTcHandleIfcChange );
}
UlTrace( TC, ("Ul!UlTcNotifyCallback: Handled.\n" ));
}
/***************************************************************************++
Routine Description:
UlpTcRegisterForCallbacks :
We will open Block object until termination for each type of
notification. And we will deref each object upon termination
Return Value:
NTSTATUS - Completion status.
--***************************************************************************/
NTSTATUS
UlpTcRegisterForCallbacks(
VOID
)
{
NTSTATUS Status = STATUS_SUCCESS;
GUID Guid;
//
// Get a WMI block handle register all the callback functions.
//
Guid = GUID_QOS_TC_INTERFACE_UP_INDICATION;
Status = IoWMIOpenBlock(&Guid,
WMIGUID_NOTIFICATION,
&g_TcInterfaceUpNotificationObject
);
if (NT_SUCCESS(Status))
{
Status = IoWMISetNotificationCallback(
g_TcInterfaceUpNotificationObject,
(WMI_NOTIFICATION_CALLBACK) UlTcNotifyCallback,
NULL
);
if (!NT_SUCCESS(Status))
goto end;
}
Guid = GUID_QOS_TC_INTERFACE_DOWN_INDICATION;
Status = IoWMIOpenBlock(&Guid,
WMIGUID_NOTIFICATION,
&g_TcInterfaceDownNotificationObject
);
if (NT_SUCCESS(Status))
{
Status = IoWMISetNotificationCallback(
g_TcInterfaceDownNotificationObject,
(WMI_NOTIFICATION_CALLBACK) UlTcNotifyCallback,
NULL
);
if (!NT_SUCCESS(Status))
goto end;
}
Guid = GUID_QOS_TC_INTERFACE_CHANGE_INDICATION;
Status = IoWMIOpenBlock(&Guid,
WMIGUID_NOTIFICATION,
&g_TcInterfaceChangeNotificationObject
);
if (NT_SUCCESS(Status))
{
Status = IoWMISetNotificationCallback(
g_TcInterfaceChangeNotificationObject,
(WMI_NOTIFICATION_CALLBACK) UlTcNotifyCallback,
NULL
);
if (!NT_SUCCESS(Status))
goto end;
}
end:
// Cleanup if necessary
if (!NT_SUCCESS(Status))
{
UlTrace(TC,("Ul!UlpTcRegisterForCallbacks: FAILED %08lx\n",Status));
if(g_TcInterfaceUpNotificationObject!=NULL)
{
ObDereferenceObject(g_TcInterfaceUpNotificationObject);
g_TcInterfaceUpNotificationObject = NULL;
}
if(g_TcInterfaceDownNotificationObject!=NULL)
{
ObDereferenceObject(g_TcInterfaceDownNotificationObject);
g_TcInterfaceDownNotificationObject = NULL;
}
if(g_TcInterfaceChangeNotificationObject!=NULL)
{
ObDereferenceObject(g_TcInterfaceChangeNotificationObject);
g_TcInterfaceChangeNotificationObject = NULL;
}
}
return Status;
}
//
// Following functions provide public/private interfaces for flow & filter
// creation/removal/modification for site & global flows.
//
/***************************************************************************++
Routine Description:
UlpTcDeleteFlow :
you should own the TciIfcResource exclusively before calling
this function
Arguments:
Return Value:
NTSTATUS - Completion status.
--***************************************************************************/
NTSTATUS
UlpTcDeleteFlow(
IN PUL_TCI_FLOW pFlow
)
{
NTSTATUS Status;
PLIST_ENTRY pEntry;
PUL_TCI_FILTER pFilter;
HANDLE FlowHandle;
PUL_TCI_INTERFACE pInterface;
//
// Initialize
//
Status = STATUS_SUCCESS;
ASSERT(g_InitTciCalled);
ASSERT(IS_VALID_TCI_FLOW(pFlow));
//
// First remove all the filters belong to us
//
while (!IsListEmpty(&pFlow->FilterList))
{
pEntry = pFlow->FilterList.Flink;
pFilter = CONTAINING_RECORD(
pEntry,
UL_TCI_FILTER,
Linkage
);
Status = UlpTcDeleteFilter( pFlow, pFilter );
ASSERT(NT_SUCCESS(Status));
}
//
// Now remove the flow itself from our flowlist on the interface
//
pInterface = pFlow->pInterface;
ASSERT( pInterface != NULL );
RemoveEntryList( &pFlow->Linkage );
ASSERT(pInterface->FlowListSize > 0);
pInterface->FlowListSize -= 1;
pFlow->Linkage.Flink = pFlow->Linkage.Blink = NULL;
FlowHandle = pFlow->FlowHandle;
UlTrace( TC, ("Ul!UlpTcDeleteFlow: Flow deleted. %p\n", pFlow ));
UL_FREE_POOL_WITH_SIG( pFlow, UL_TCI_FLOW_POOL_TAG );
//
// Finally talk to TC
//
Status = UlpTcDeleteGpcFlow( FlowHandle );
if (!NT_SUCCESS(Status))
{
UlTrace( TC, ("Ul!UlpTcDeleteFlow: FAILURE %08lx \n", Status ));
}
else
{
UlTrace( TC, ("Ul!UlpTcDeleteFlow: FlowHandle %d deleted in TC as well.\n",
FlowHandle
));
}
return Status;
}
/***************************************************************************++
Routine Description:
UlpTcDeleteFlow :
remove a flow from existing QoS Enabled interface
Arguments:
Return Value:
NTSTATUS - Completion status.
--***************************************************************************/
NTSTATUS
UlpTcDeleteGpcFlow(
IN HANDLE FlowHandle
)
{
NTSTATUS Status;
ULONG InBuffSize;
ULONG OutBuffSize;
GPC_REMOVE_CF_INFO_REQ GpcReq;
GPC_REMOVE_CF_INFO_RES GpcRes;
IO_STATUS_BLOCK IoStatusBlock;
//
// Remove the flow frm psched
//
InBuffSize = sizeof(GPC_REMOVE_CF_INFO_REQ);
OutBuffSize = sizeof(GPC_REMOVE_CF_INFO_RES);
GpcReq.ClientHandle = g_GpcClientHandle;
GpcReq.GpcCfInfoHandle = FlowHandle;
Status = UlpTcDeviceControl( g_GpcFileHandle,
NULL,
NULL,
NULL,
&IoStatusBlock,
IOCTL_GPC_REMOVE_CF_INFO,
&GpcReq,
InBuffSize,
&GpcRes,
OutBuffSize
);
if (!NT_SUCCESS(Status))
{
UlTrace( TC, ("Ul!UlpTcDeleteGpcFlow: FAILURE %08lx \n", Status ));
}
return Status;
}
/***************************************************************************++
Routine Description:
UlpTcAllocateFlow :
Allocates a flow and setup the FlowSpec according the passed BWT
parameter
Arguments:
HTTP_BANDWIDTH_LIMIT - FlowSpec will be created using this BWT limit
in B/s
Return Value
PUL_TCI_FLOW - The newly allocated flow
NULL - If memory allocation failed
--***************************************************************************/
PUL_TCI_FLOW
UlpTcAllocateFlow(
IN HTTP_BANDWIDTH_LIMIT MaxBandwidth,
IN ULONG MtuSize
)
{
PUL_TCI_FLOW pFlow;
TC_GEN_FLOW TcGenFlow;
//
// Setup the FlowSpec frm MaxBandwidth passed by the config handler
//
RtlZeroMemory(&TcGenFlow,sizeof(TcGenFlow));
UL_SET_FLOWSPEC(TcGenFlow,MaxBandwidth,MtuSize);
//
// Since we hold a spinlock inside the flow structure allocating from
// NonPagedPool. We will have this allocation only for bt enabled sites.
//
pFlow = UL_ALLOCATE_STRUCT(
NonPagedPool,
UL_TCI_FLOW,
UL_TCI_FLOW_POOL_TAG
);
if( pFlow == NULL )
{
return NULL;
}
// Initialize the rest
RtlZeroMemory( pFlow, sizeof(UL_TCI_FLOW) );
pFlow->Signature = UL_TCI_FLOW_POOL_TAG;
pFlow->GenFlow = TcGenFlow;
UlInitializeSpinLock( &pFlow->FilterListSpinLock, "FilterListSpinLock" );
InitializeListHead( &pFlow->FilterList );
pFlow->pConfigGroup = NULL;
return pFlow;
}
/***************************************************************************++
Routine Description:
UlpModifyFlow :
Modify an existing flow by sending an IOCTL down to GPC. Basically
what this function does is to provide an updated TC_GEN_FLOW field
to GPC for an existing flow.
Arguments:
PUL_TCI_INTERFACE - Required to get the interfaces friendly name.
PUL_TCI_FLOW - To get the GPC flow handle as well as to be able to
update the new flow parameters.
--***************************************************************************/
NTSTATUS
UlpModifyFlow(
IN PUL_TCI_INTERFACE pInterface,
IN PUL_TCI_FLOW pFlow
)
{
PCF_INFO_QOS Kflow;
PGPC_MODIFY_CF_INFO_REQ pGpcReq;
GPC_MODIFY_CF_INFO_RES GpcRes;
ULONG InBuffSize;
ULONG OutBuffSize;
IO_STATUS_BLOCK IoStatusBlock;
NTSTATUS Status;
//
// Sanity check
//
ASSERT(g_GpcClientHandle);
ASSERT(IS_VALID_TCI_INTERFACE(pInterface));
ASSERT(IS_VALID_TCI_FLOW(pFlow));
InBuffSize = sizeof(GPC_MODIFY_CF_INFO_REQ) + sizeof(CF_INFO_QOS);
OutBuffSize = sizeof(GPC_MODIFY_CF_INFO_RES);
pGpcReq = UL_ALLOCATE_STRUCT_WITH_SPACE(
PagedPool,
GPC_MODIFY_CF_INFO_REQ,
sizeof(CF_INFO_QOS),
UL_TCI_GENERIC_POOL_TAG
);
if (pGpcReq == NULL)
{
return STATUS_NO_MEMORY;
}
RtlZeroMemory(pGpcReq, InBuffSize);
RtlZeroMemory(&GpcRes, OutBuffSize);
pGpcReq->ClientHandle = g_GpcClientHandle;
pGpcReq->GpcCfInfoHandle = pFlow->FlowHandle;
pGpcReq->CfInfoSize = sizeof(CF_INFO_QOS);
Kflow = (PCF_INFO_QOS)&pGpcReq->CfInfo;
Kflow->InstanceNameLength = (USHORT) pInterface->NameLength;
RtlCopyMemory(Kflow->InstanceName,
pInterface->Name,
pInterface->NameLength* sizeof(WCHAR));
RtlCopyMemory(&Kflow->GenFlow,
&pFlow->GenFlow,
sizeof(TC_GEN_FLOW));
Status = UlpTcDeviceControl( g_GpcFileHandle,
NULL,
NULL,
NULL,
&IoStatusBlock,
IOCTL_GPC_MODIFY_CF_INFO,
pGpcReq,
InBuffSize,
&GpcRes,
OutBuffSize
);
if ( NT_SUCCESS(Status) )
{
UlTrace( TC, ("Ul!UlpModifyFlow: flow %p modified on interface %p \n",
pFlow,
pInterface
));
}
else
{
UlTrace( TC, ("Ul!UlpModifyFlow: FAILURE %08lx for GpcClient %u\n",
Status,
g_GpcClientHandle
));
}
UL_FREE_POOL( pGpcReq, UL_TCI_GENERIC_POOL_TAG );
return Status;
}
/***************************************************************************++
Routine Description:
UlpAddFlow :
Add a flow on existing QoS Enabled interface
Arguments:
Return Value:
NTSTATUS - Completion status.
--***************************************************************************/
NTSTATUS
UlpAddFlow(
IN PUL_TCI_INTERFACE pInterface,
IN PUL_TCI_FLOW pGenericFlow,
OUT PHANDLE pHandle
)
{
NTSTATUS Status;
PCF_INFO_QOS Kflow;
PGPC_ADD_CF_INFO_REQ pGpcReq;
GPC_ADD_CF_INFO_RES GpcRes;
ULONG InBuffSize;
ULONG OutBuffSize;
IO_STATUS_BLOCK IoStatusBlock;
//
// Find the interface from handle
//
ASSERT(g_GpcClientHandle);
InBuffSize = sizeof(GPC_ADD_CF_INFO_REQ) + sizeof(CF_INFO_QOS);
OutBuffSize = sizeof(GPC_ADD_CF_INFO_RES);
pGpcReq = UL_ALLOCATE_STRUCT_WITH_SPACE(
PagedPool,
GPC_ADD_CF_INFO_REQ,
sizeof(CF_INFO_QOS),
UL_TCI_GENERIC_POOL_TAG
);
if (pGpcReq == NULL)
{
return STATUS_NO_MEMORY;
}
RtlZeroMemory( pGpcReq, InBuffSize);
RtlZeroMemory( &GpcRes, OutBuffSize);
pGpcReq->ClientHandle = g_GpcClientHandle;
//pGpcReq->ClientCfInfoContext= GPC_CF_QOS; // ?? Not sure about this
pGpcReq->CfInfoSize = sizeof( CF_INFO_QOS);
Kflow = (PCF_INFO_QOS)&pGpcReq->CfInfo;
Kflow->InstanceNameLength = (USHORT) pInterface->NameLength;
RtlCopyMemory( Kflow->InstanceName,
pInterface->Name,
pInterface->NameLength* sizeof(WCHAR)
);
RtlCopyMemory( &Kflow->GenFlow,
&pGenericFlow->GenFlow,
sizeof(TC_GEN_FLOW)
);
Status = UlpTcDeviceControl( g_GpcFileHandle,
NULL,
NULL,
NULL,
&IoStatusBlock,
IOCTL_GPC_ADD_CF_INFO,
pGpcReq,
InBuffSize,
&GpcRes,
OutBuffSize
);
if (NT_SUCCESS(Status))
{
(*pHandle) = (HANDLE) GpcRes.GpcCfInfoHandle;
UlTrace( TC, ("Ul!UlpAddFlow: a new flow added %p on interface %p \n",
pGenericFlow,
pInterface
));
}
else
{
UlTrace( TC, ("Ul!UlpAddFlow: FAILURE %08lx for GpcClient %u\n",
Status,
g_GpcClientHandle
));
}
UL_FREE_POOL( pGpcReq, UL_TCI_GENERIC_POOL_TAG );
return Status;
}
/***************************************************************************++
Routine Description:
UlTcAddFlowsForSite :
Add a flow on existing QoS Enabled interface
Arguments:
pConfigGroup - The config group of the site
MaxBandwidth - The Max bandwidth we are going to enforce by a FlowSpec
in B/s
Return Value:
NTSTATUS - Completion status.
--***************************************************************************/
NTSTATUS
UlTcAddFlowsForSite(
IN PUL_CONFIG_GROUP_OBJECT pConfigGroup,
IN HTTP_BANDWIDTH_LIMIT MaxBandwidth
)
{
NTSTATUS Status;
PLIST_ENTRY pInterfaceEntry;
PUL_TCI_INTERFACE pInterface;
PUL_TCI_FLOW pFlow;
//
// Sanity check first
//
Status = STATUS_SUCCESS;
//
// If we have been called w/o being initialized
//
ASSERT(g_InitTciCalled);
ASSERT(IS_VALID_CONFIG_GROUP(pConfigGroup));
UlTrace(TC,("Ul!UlTcAddFlowsForSite: For cgroup %p BWT %d KB/s\n",
pConfigGroup,
MaxBandwidth
));
//
// Visit each interface and add a flow for this site. Following lock also
// protects the FlowListHead of the ConfigGroup object. We only change it
// when we acquire this lock here. Or when removing the site's flows.
//
UlAcquireResourceExclusive(&g_pUlNonpagedData->TciIfcResource, TRUE);
// TODO: Remember this cgroup incase interface goes up/down later on,we can still
// TODO: recover and reinstall the flows of the cgroup properly.
// InsertTailList(&g_TcCGroupListHead, &pConfigGroup->Linkage );
// Proceed and add the flows to the inetrfaces
pInterfaceEntry = g_TciIfcListHead.Flink;
while ( pInterfaceEntry != &g_TciIfcListHead )
{
pInterface = CONTAINING_RECORD(
pInterfaceEntry,
UL_TCI_INTERFACE,
Linkage
);
//
// Allocate a flow sturcture
//
pFlow = UlpTcAllocateFlow( MaxBandwidth, pInterface->MTUSize );
if ( pFlow == NULL )
{
UlReleaseResource(&g_pUlNonpagedData->TciIfcResource);
return STATUS_NO_MEMORY;
}
//
// Add the flow by making a TC call down to gpc
//
Status = UlpAddFlow( pInterface,
pFlow,
&pFlow->FlowHandle
);
if (!NT_SUCCESS(Status))
{
UlTrace( TC, ("Ul!UlTcAddFlowsForSite: FAILURE %08lx \n", Status ));
UlReleaseResource(&g_pUlNonpagedData->TciIfcResource);
UL_FREE_POOL_WITH_SIG( pFlow, UL_TCI_FLOW_POOL_TAG );
return Status;
}
//
// Proceed with further initialization as we have successfully installed
// the flow. First link the flow back to its owner interface.
//
pFlow->pInterface = pInterface;
//
// Add this to the interface's flowlist as well
//
InsertHeadList( &pInterface->FlowList, &pFlow->Linkage );
pInterface->FlowListSize += 1;
//
// Also add this to the cgroup's flowlist. Set the cgroup pointer.
// Do not bump up the cgroup refcount. Otherwise cgroup cannot be
// cleaned up until Tc terminates. And flows cannot be removed un
// till termination.
//
InsertHeadList( &pConfigGroup->FlowListHead, &pFlow->Siblings );
pFlow->pConfigGroup = pConfigGroup;
UlTrace( TC,
("Ul!UlTcAddFlowsForSite: Added the pFlow %p on pInterface %p\n",
pFlow,
pInterface
));
UL_DUMP_TC_FLOW(pFlow);
//
// Proceed to the next interface
//
pInterfaceEntry = pInterfaceEntry->Flink;
}
UlReleaseResource(&g_pUlNonpagedData->TciIfcResource);
UlTrace( TC, ("Ul!UlTcAddFlowsForSite: SUCCESS \n" ));
return Status;
}
/***************************************************************************++
Routine Description:
UlTcModifyFlowsForSite :
This function will be called when there's a cgroup change happening for
the site's bandwidth throttling settings.
Its caller responsiblity to remember the new settings in the cgroup.
We will update the FlowSpec on the existing flows
Arguments:
PUL_CONFIG_GROUP_OBJECT - Pointer to the cgroup of the site.
HTTP_BANDWIDTH_LIMIT - The new bandwidth throttling setting
in B/s
--***************************************************************************/
NTSTATUS
UlTcModifyFlowsForSite(
IN PUL_CONFIG_GROUP_OBJECT pConfigGroup,
IN HTTP_BANDWIDTH_LIMIT NewBandwidth
)
{
NTSTATUS Status;
PLIST_ENTRY pFlowEntry;
PUL_TCI_FLOW pFlow;
HTTP_BANDWIDTH_LIMIT OldBandwidth;
//
// Sanity check
//
Status = STATUS_SUCCESS;
ASSERT(IS_VALID_CONFIG_GROUP(pConfigGroup));
//
// If we have been called w/o being initialized
//
ASSERT(g_InitTciCalled);
//
// We do not handle the remove case. it should be handled by cgroup
//
ASSERT(NewBandwidth != HTTP_LIMIT_INFINITE);
UlTrace(TC,("Ul!UlTcModifyFlowsForSite: For cgroup %p.\n",pConfigGroup));
//
// Modify the flow list in the cgroup as it shows the flows of this site
//
UlAcquireResourceExclusive(&g_pUlNonpagedData->TciIfcResource, TRUE);
pFlowEntry = pConfigGroup->FlowListHead.Flink;
while ( pFlowEntry != &pConfigGroup->FlowListHead )
{
pFlow = CONTAINING_RECORD(
pFlowEntry,
UL_TCI_FLOW,
Siblings
);
// Yet another sanity check
ASSERT(IS_VALID_CONFIG_GROUP(pFlow->pConfigGroup));
ASSERT(pConfigGroup == pFlow->pConfigGroup);
// Overwrite the new bandwidth but remember the old
OldBandwidth = UL_GET_BW_FRM_FLOWSPEC(pFlow->GenFlow);
UL_SET_FLOWSPEC(pFlow->GenFlow,NewBandwidth,pFlow->pInterface->MTUSize);
Status = UlpModifyFlow(pFlow->pInterface, pFlow);
if (!NT_SUCCESS(Status))
{
// Whine about it, but still continue
UlTrace( TC, ("Ul!UlTcModifyFlowsForSite: FAILURE %08lx \n", Status ));
// Restore the original flowspec back
UL_SET_FLOWSPEC(pFlow->GenFlow,OldBandwidth,pFlow->pInterface->MTUSize);
}
UL_DUMP_TC_FLOW(pFlow);
// Proceed to the next flow
pFlowEntry = pFlowEntry->Flink;
}
UlReleaseResource(&g_pUlNonpagedData->TciIfcResource);
UlTrace( TC, ("Ul!UlTcModifyFlowsForSite: Frm %d KB/s To %d KB/s done.\n",
OldBandwidth,
NewBandwidth
));
return Status;
}
/***************************************************************************++
Routine Description:
UlTcRemoveFlowsForSite :
Add a flow on existing QoS Enabled interface
Arguments:
Return Value:
NTSTATUS - Completion status.
--***************************************************************************/
NTSTATUS
UlTcRemoveFlowsForSite(
IN PUL_CONFIG_GROUP_OBJECT pConfigGroup
)
{
NTSTATUS Status;
PLIST_ENTRY pFlowEntry;
PUL_TCI_FLOW pFlow;
//
// Sanity check first
//
Status = STATUS_SUCCESS;
ASSERT(IS_VALID_CONFIG_GROUP(pConfigGroup));
//
// If we have been called w/o being initialized
//
ASSERT(g_InitTciCalled);
UlTrace(TC,("Ul!UlTcRemoveFlowsForSite: For cgroup %p\n", pConfigGroup));
//
// Remove frm the cgroup list and remove frm the interface list
//
UlAcquireResourceExclusive(&g_pUlNonpagedData->TciIfcResource, TRUE);
while (!IsListEmpty(&pConfigGroup->FlowListHead))
{
pFlowEntry = pConfigGroup->FlowListHead.Flink;
pFlow = CONTAINING_RECORD(
pFlowEntry,
UL_TCI_FLOW,
Siblings
);
// Yet another sanity check
ASSERT(pConfigGroup == pFlow->pConfigGroup);
ASSERT(IS_VALID_CONFIG_GROUP(pFlow->pConfigGroup));
// Remove frm cgroup's flowlist and release our reference
RemoveEntryList(&pFlow->Siblings);
pFlow->Siblings.Flink = pFlow->Siblings.Blink = NULL;
pFlow->pConfigGroup = NULL;
// Now frm interface list. This will also make the TC call
Status = UlpTcDeleteFlow(pFlow);
ASSERT(NT_SUCCESS(Status));
}
UlReleaseResource(&g_pUlNonpagedData->TciIfcResource);
UlTrace(TC,("Ul!UlTcRemoveFlowsForSite: The cgroup %p 's flows are removed.\n",
pConfigGroup
));
return Status;
}
/***************************************************************************++
Routine Description:
To see whether packet scheduler is installed or not. We also bail
out if we weren't able to initialize properly.
TODO we might want to handle the change (PSched installed later)
on-the-fly w/o requiring driver to be restarted.
--***************************************************************************/
BOOLEAN
UlTcPSchedInstalled(
VOID
)
{
return (g_InitTciCalled && g_PSchedInstalled);
}
/***************************************************************************++
Routine Description:
UlTcGlobalThrottlingEnabled :
Will return TRUE if global bandwidth throttling is enabled in TC.
Its enabled when global flows are installed and TC is initialized
otherwise disabled.
Make sure that if !UlTcPSchedInstalled() then Global throttling is
always disabled.
--***************************************************************************/
__inline BOOLEAN
UlTcGlobalThrottlingEnabled(
VOID
)
{
return (UL_IS_GLOBAL_THROTTLING_ENABLED());
}
/***************************************************************************++
Routine Description:
UlTcAddGlobalFlows :
Visits and creates the global flow on each interface
Arguments:
HTTP_BANDWIDTH_LIMIT - The bandwidth throttling limit in KB/s
--***************************************************************************/
NTSTATUS
UlTcAddGlobalFlows(
IN HTTP_BANDWIDTH_LIMIT MaxBandwidth
)
{
NTSTATUS Status;
PLIST_ENTRY pInterfaceEntry;
PUL_TCI_INTERFACE pInterface;
PUL_TCI_FLOW pFlow;
TC_GEN_FLOW TcGenFlow;
//
// Sanity Check
//
Status = STATUS_SUCCESS;
//
// If we have been called w/o being initialized
//
ASSERT(g_InitTciCalled);
UlTrace(TC,("Ul!UlTcAddGlobalFlows: Installing for %d KB/s\n", MaxBandwidth));
//
// To ensure the new filters can get attached to the global flows
//
UL_ENABLE_GLOBAL_THROTTLING();
//
// Visit each interface and add a global flow for this site
// Acquire Exclusive because we will add a flow to the list
//
UlAcquireResourceExclusive(&g_pUlNonpagedData->TciIfcResource, TRUE);
pInterfaceEntry = g_TciIfcListHead.Flink;
while ( pInterfaceEntry != &g_TciIfcListHead )
{
pInterface = CONTAINING_RECORD(
pInterfaceEntry,
UL_TCI_INTERFACE,
Linkage
);
//
// Nobody should try to add a global flow when there's already one
//
ASSERT(pInterface->pGlobalFlow == NULL);
//
// Allocate a flow structure
//
pFlow = UlpTcAllocateFlow( MaxBandwidth, pInterface->MTUSize );
if ( pFlow == NULL )
{
UlReleaseResource(&g_pUlNonpagedData->TciIfcResource);
return STATUS_NO_MEMORY;
}
//
// Add the flow by making a TC call down to gpc
//
Status = UlpAddFlow( pInterface,
pFlow,
&pFlow->FlowHandle
);
if (!NT_SUCCESS(Status))
{
UlTrace( TC, ("Ul!UlTcAddGlobalFlows: FAILURE %08lx \n", Status ));
UlReleaseResource(&g_pUlNonpagedData->TciIfcResource);
UL_FREE_POOL_WITH_SIG( pFlow, UL_TCI_FLOW_POOL_TAG );
return Status;
}
//
// Proceed with further initialization as we have successfully installed
// the flow. First link the flow back to its owner interface and remeber
// this was a global flow for the interface. Make sure that the config
// group pointer is null for the global flows.
//
pFlow->pInterface = pInterface;
pFlow->pConfigGroup = NULL;
pInterface->pGlobalFlow = pFlow;
//
// Add this to the interface's flowlist as well
//
InsertHeadList( &pInterface->FlowList, &pFlow->Linkage );
pInterface->FlowListSize += 1;
UlTrace( TC,
("Ul!UlTcAddGlobalFlows: Added the pGlobalFlow %p on pInterface %p\n",
pFlow,
pInterface
));
UL_DUMP_TC_FLOW(pFlow);
//
// search through next interface
//
pInterfaceEntry = pInterfaceEntry->Flink;
}
UlReleaseResource(&g_pUlNonpagedData->TciIfcResource);
UlTrace( TC, ("Ul!UlTcAddGlobalFlows: Installed.\n" ));
return Status;
}
/***************************************************************************++
Routine Description:
UlTcModifyGlobalFlows :
This function will be called when there's a config channel change
happening for the global bandwidth throttling setting.
Its caller responsiblity to remember the new settings in the control
channel.
We will simply update the FlowSpec on the existing global flows.
Arguments:
HTTP_BANDWIDTH_LIMIT - The new bandwidth throttling setting
in B/s
--***************************************************************************/
NTSTATUS
UlTcModifyGlobalFlows(
IN HTTP_BANDWIDTH_LIMIT NewBandwidth
)
{
NTSTATUS Status;
PLIST_ENTRY pInterfaceEntry;
PUL_TCI_INTERFACE pInterface;
PUL_TCI_FLOW pFlow;
HTTP_BANDWIDTH_LIMIT OldBandwidth;
//
// Sanity check
//
Status = STATUS_SUCCESS;
//
// If we have been called w/o being initialized
//
ASSERT(g_InitTciCalled);
//
// We do not handle the remove case.It should be handled by control channel
//
ASSERT(NewBandwidth != HTTP_LIMIT_INFINITE);
UlTrace(TC,("Ul!UlTcModifyGlobalFlows: to %d KB/s \n",NewBandwidth));
//
// Modify the global flows of all the interfaces
//
UlAcquireResourceExclusive(&g_pUlNonpagedData->TciIfcResource, TRUE);
pInterfaceEntry = g_TciIfcListHead.Flink;
while ( pInterfaceEntry != &g_TciIfcListHead )
{
pInterface = CONTAINING_RECORD(
pInterfaceEntry,
UL_TCI_INTERFACE,
Linkage
);
ASSERT(pInterface->pGlobalFlow != NULL);
pFlow = pInterface->pGlobalFlow;
// Overwrite the old bandwidth limit but remember it
OldBandwidth = UL_GET_BW_FRM_FLOWSPEC(pFlow->GenFlow);
UL_SET_FLOWSPEC(pFlow->GenFlow, NewBandwidth,pFlow->pInterface->MTUSize);
// Pass it down to low level modifier
Status = UlpModifyFlow(pInterface, pFlow);
if (!NT_SUCCESS(Status))
{
// Whine about it, but still continue
UlTrace(TC,("Ul!UlTcModifyGlobalFlows: FAILURE %08lx \n",Status));
// Restore the original flowspec back
UL_SET_FLOWSPEC(pFlow->GenFlow,OldBandwidth,pFlow->pInterface->MTUSize);
}
UL_DUMP_TC_FLOW(pFlow);
// Proceed to the next interface's global_flow
pInterfaceEntry = pInterfaceEntry->Flink;
}
UlReleaseResource(&g_pUlNonpagedData->TciIfcResource);
UlTrace( TC, ("Ul!UlTcModifyGlobalFlows: Modified.\n" ));
return Status;
}
/***************************************************************************++
Routine Description:
UlTcRemoveGlobalFlows :
Add a flow on existing QoS Enabled interface
--***************************************************************************/
NTSTATUS
UlTcRemoveGlobalFlows(
VOID
)
{
NTSTATUS Status;
PLIST_ENTRY pInterfaceEntry;
PUL_TCI_INTERFACE pInterface;
PUL_TCI_FLOW pFlow;
//
// Sanity check first
//
Status = STATUS_SUCCESS;
//
// If we have been called w/o being initialized
//
ASSERT(g_InitTciCalled);
UlTrace(TC,("Ul!UlTcRemoveGlobalFlows: ...\n"));
//
// To ensure no new filters can get attached to the global flows
//
UL_DISABLE_GLOBAL_THROTTLING();
//
// Remove each interface's global flow
//
UlAcquireResourceExclusive(&g_pUlNonpagedData->TciIfcResource, TRUE);
pInterfaceEntry = g_TciIfcListHead.Flink;
while ( pInterfaceEntry != &g_TciIfcListHead )
{
pInterface = CONTAINING_RECORD(
pInterfaceEntry,
UL_TCI_INTERFACE,
Linkage
);
pFlow = pInterface->pGlobalFlow;
if (pFlow)
{
ASSERT(IS_VALID_TCI_FLOW(pFlow));
// Remove from interface list and make the TC call down.
Status = UlpTcDeleteFlow(pFlow);
ASSERT(NT_SUCCESS(Status));
// No more global flow
pInterface->pGlobalFlow = NULL;
}
// Goto the next interface on the list
pInterfaceEntry = pInterfaceEntry->Flink;
}
UlReleaseResource(&g_pUlNonpagedData->TciIfcResource);
UlTrace(TC,("Ul!UlTcRemoveGlobalFlows: Flows are removed.\n" ));
return Status;
}
/***************************************************************************++
Routine Description:
UlTcAddFilter :
//
// There are two possibilities. The request could be served frm
// cache or can be routed to the user. In either case we need a
// flow installed if the BW is enabled for this request's site
// and there's no filter installed for this connection yet. We
// will remove the filter as soon as the connection dropped. But
// yes there's always a but,if the client is attempting to make
// requests to different sites using the same connection then we
// need to drop the filter frm the old site and move it to the
// newly requested site. This is a rare case but lets handle it
// anyway.
//
It's callers responsibility to ensure proper removal of the filter,
after it's done.
Algorithm:
1. Find the flow from the flow list of cgroup (or from global flows)
2. Add filter to that flow
Arguments:
pHttpConnection - required - Filter will be attached for this connection
pCgroup - optional - NULL means add to the global flow
Return Values:
STATUS_INVALID_DEVICE_REQUEST- If TC not initialized
STATUS_NOT_SUPPORTED - For attempts on Local Loopback
STATUS_OBJECT_NAME_NOT_FOUND - If flow has not been found for the cgroup
STATUS_SUCCESS - In other cases
--***************************************************************************/
NTSTATUS
UlTcAddFilter(
IN PUL_HTTP_CONNECTION pHttpConnection,
IN PUL_CONFIG_GROUP_OBJECT pCgroup OPTIONAL
)
{
NTSTATUS Status;
ULONG IpAddress;
ULONG IpAddressTemp;
TC_GEN_FILTER TcGenericFilter;
PUL_TCI_FLOW pFlow;
PUL_TCI_INTERFACE pInterface;
IP_PATTERN Pattern;
IP_PATTERN Mask;
PUL_TCI_FILTER pFilter;
PUL_CONFIG_GROUP_OBJECT pOldCgroup;
//
// A lot of sanity & early checks
//
Status = STATUS_SUCCESS;
ASSERT(UL_IS_VALID_HTTP_CONNECTION(pHttpConnection));
//
// If we have been called w/o being initialized
//
ASSERT(g_InitTciCalled);
//
// If BWT or GBWT disabled then just bail out.
//
if ((pCgroup == NULL && !UlTcGlobalThrottlingEnabled()) ||
(pCgroup != NULL && (pCgroup->MaxBandwidth.Flags.Present == 0 ||
pCgroup->MaxBandwidth.MaxBandwidth == HTTP_LIMIT_INFINITE))
)
{
return Status;
}
//
// We need the local & remote IPs. Connection keeps the ip and port in
// host order, but we need in network order here. To be competible with
// the rest of the BWT code. Convert it back.
//
IpAddress = SWAP_LONG( pHttpConnection->pConnection->LocalAddress );
IpAddressTemp = pHttpConnection->pConnection->LocalAddress;
UlTrace(TC,("Ul!UlTcAddFilter: Local %d.%d.%d.%d:%d\n",
(UCHAR)(IpAddressTemp >> 24),
(UCHAR)(IpAddressTemp >> 16),
(UCHAR)(IpAddressTemp >> 8),
(UCHAR)(IpAddressTemp >> 0),
pHttpConnection->pConnection->LocalPort
));
IpAddressTemp = pHttpConnection->pConnection->RemoteAddress;
UlTrace(TC,("Ul!UlTcAddFilter: Remote %d.%d.%d.%d:%d\n",
(UCHAR)(IpAddressTemp >> 24),
(UCHAR)(IpAddressTemp >> 16),
(UCHAR)(IpAddressTemp >> 8),
(UCHAR)(IpAddressTemp >> 0),
pHttpConnection->pConnection->RemotePort
));
if ( IpAddress == LOOPBACK_ADDR )
{
//
// Make sure that new Filter is not trying to go the Local_loopback
// if that is the case skip this. There's no qos on local_loopbacks
// PSched doesn't receive any packets for local_loopback address.
//
UlTrace( TC,
("Ul!UlTcAddFilter: LocalLoopback not supported."
"Not adding filter for pHttpConnection %p\n",
pHttpConnection
));
return STATUS_NOT_SUPPORTED;
}
//
// At this point we will be refering to the flows & filters
// in our list therefore we need to acquire the lock
//
UlAcquireResourceShared(&g_pUlNonpagedData->TciIfcResource, TRUE);
// If connection already has a filter attached
if (pHttpConnection->pFlow)
{
ASSERT(IS_VALID_TCI_FLOW(pHttpConnection->pFlow));
ASSERT(IS_VALID_TCI_FILTER(pHttpConnection->pFilter));
// To see if we have a new cgroup, if that's the case then
// we have to go to a new flow. If pCgroup is null we will
// still skip adding the same global filter again
pOldCgroup = pHttpConnection->pFlow->pConfigGroup;
if (pOldCgroup == pCgroup)
{
// No need to add a new filter we are done
UlTrace( TC,
("Ul!UlTcAddFilter: Skipping same pFlow %p & pFilter %p already exist\n",
pHttpConnection->pFlow,
pHttpConnection->pFilter,
pHttpConnection
));
goto end;
}
else
{
//
// If there was another filter before and this newly coming request
// is being going to a different site/flow. Then move the filter frm
// old one to the new flow.
//
UlpTcDeleteFilter(pHttpConnection->pFlow, pHttpConnection->pFilter);
}
}
//
// Search through the cgroup's flowlist to find the one we need. The one
// on the interface we want.
//
pFlow = UlpFindFlow( pCgroup, IpAddress );
if ( pFlow == NULL )
{
IpAddressTemp = SWAP_LONG(IpAddress);
UlTrace( TC,
("Ul!UlTcAddFilter: Unable to find interface (%x) %d.%d.%d.%d \n",
IpAddress,
(UCHAR)(IpAddressTemp >> 24),
(UCHAR)(IpAddressTemp >> 16),
(UCHAR)(IpAddressTemp >> 8),
(UCHAR)(IpAddressTemp >> 0)
));
// It's possible that we might not find out a flow
// after all the interfaces went down, even though
// qos configured on the cgroup.
// ASSERT(FALSE);
Status = STATUS_OBJECT_NAME_NOT_FOUND;
goto end;
}
// Some more initialization
pFilter = NULL;
pInterface = pFlow->pInterface;
RtlZeroMemory( &Pattern, sizeof(IP_PATTERN) );
RtlZeroMemory( &Mask, sizeof(IP_PATTERN) );
//
// Everything is fine.
// Now setup the filter with proper pattern & mask
//
Pattern.SrcAddr = SWAP_LONG(pHttpConnection->pConnection->LocalAddress);
Pattern.S_un.S_un_ports.s_srcport = SWAP_SHORT(pHttpConnection->pConnection->LocalPort);
Pattern.DstAddr = SWAP_LONG(pHttpConnection->pConnection->RemoteAddress);
Pattern.S_un.S_un_ports.s_dstport = SWAP_SHORT(pHttpConnection->pConnection->RemotePort);
Pattern.ProtocolId = IPPROTO_TCP;
/* Mask */
RtlFillMemory(&Mask, sizeof(IP_PATTERN), 0xff);
//Mask.SrcAddr = 0x00000000;
//Mask.S_un.S_un_ports.s_srcport = 0; // WHY ??
//Mask.DstAddr = 0x00000000;
//Mask.S_un.S_un_ports.s_dstport = 0; // WHY ??
TcGenericFilter.AddressType = NDIS_PROTOCOL_ID_TCP_IP;
TcGenericFilter.PatternSize = sizeof( IP_PATTERN );
TcGenericFilter.Pattern = &Pattern;
TcGenericFilter.Mask = &Mask;
Status = UlpTcAddFilter(
pFlow,
&TcGenericFilter,
&pFilter
);
if (!NT_SUCCESS(Status))
{
UlTrace( TC,
("Ul!UlTcAddFilter: Unable to add filter for;\n"
"\t pInterface : %p\n"
"\t pFlow : %p\n",
pInterface,
pFlow
));
goto end;
}
//
// Update the connection's pointers here.
//
pHttpConnection->pFlow = pFlow;
pHttpConnection->pFilter = pFilter;
pHttpConnection->BandwidthThrottlingEnabled = 1;
//
// Remember the connection for cleanup. If flow & filter get
// removed aynscly when connection still pointing to them
// we can go and null the connection's private pointers as
// well
//
pFilter->pHttpConnection = pHttpConnection;
//
// Sweet smell of success !
//
UlTrace(TC,
("Ul!UlTcAddFilter: Success for;\n"
"\t pInterface : %p\n"
"\t pFlow : %p\n",
pInterface,
pFlow
));
UL_DUMP_TC_FILTER(pFilter);
end:
if (!NT_SUCCESS(Status))
{
UlTrace( TC, ("Ul!UlTcAddFilter: FAILURE %08lx \n", Status ));
}
UlReleaseResource(&g_pUlNonpagedData->TciIfcResource);
return Status;
}
/***************************************************************************++
Routine Description:
UlpTcAddFilter :
Add a filter on existing flow
Arguments:
Return Value:
NTSTATUS - Completion status.
--***************************************************************************/
NTSTATUS
UlpTcAddFilter(
IN PUL_TCI_FLOW pFlow,
IN PTC_GEN_FILTER pGenericFilter,
OUT PUL_TCI_FILTER *ppFilter
)
{
NTSTATUS Status;
PGPC_ADD_PATTERN_REQ pGpcReq;
GPC_ADD_PATTERN_RES GpcRes;
ULONG InBuffSize;
ULONG OutBuffSize;
ULONG PatternSize;
IO_STATUS_BLOCK IoStatBlock;
ULONG IfIndex;
PUCHAR pTemp;
PIP_PATTERN pIpPattern;
HANDLE RetHandle;
PUL_TCI_FILTER pFilter;
//
// Sanity check
//
Status = STATUS_SUCCESS;
pGpcReq = NULL;
if ( !pGenericFilter || !pFlow || !g_GpcClientHandle )
{
return STATUS_INVALID_PARAMETER;
}
// Allocate a space for the filter
pFilter = UL_ALLOCATE_STRUCT(
NonPagedPool,
UL_TCI_FILTER,
UL_TCI_FILTER_POOL_TAG
);
if ( pFilter == NULL )
{
Status = STATUS_NO_MEMORY;
goto end;
}
pFilter->Signature = UL_TCI_FILTER_POOL_TAG;
// Buffer monkeying
PatternSize = sizeof(IP_PATTERN);
InBuffSize = sizeof(GPC_ADD_PATTERN_REQ) + (2 * PatternSize);
OutBuffSize = sizeof(GPC_ADD_PATTERN_RES);
pGpcReq = UL_ALLOCATE_STRUCT_WITH_SPACE(
PagedPool,
GPC_ADD_PATTERN_REQ,
(2 * PatternSize),
UL_TCI_GENERIC_POOL_TAG
);
if (pGpcReq == NULL)
{
Status = STATUS_NO_MEMORY;
goto end;
}
RtlZeroMemory( pGpcReq, InBuffSize);
RtlZeroMemory( &GpcRes, OutBuffSize);
pGpcReq->ClientHandle = g_GpcClientHandle;
pGpcReq->GpcCfInfoHandle = pFlow->FlowHandle;
pGpcReq->PatternSize = PatternSize;
pGpcReq->ProtocolTemplate = GPC_PROTOCOL_TEMPLATE_IP;
pTemp = (PUCHAR) &pGpcReq->PatternAndMask;
// Fill in the IP Pattern first
RtlCopyMemory( pTemp, pGenericFilter->Pattern, PatternSize );
pIpPattern = (PIP_PATTERN) pTemp;
//
// According to QoS Tc.dll ;
// This is a work around so that TCPIP wil not to find the index/link
// for ICMP/IGMP packets
//
pIpPattern->Reserved1 = pFlow->pInterface->IfIndex;
pIpPattern->Reserved2 = pFlow->pInterface->SpecificLinkCtx;
pIpPattern->Reserved3[0] = pIpPattern->Reserved3[1] = pIpPattern->Reserved3[2] = 0;
// Fill in the mask
pTemp += PatternSize;
RtlCopyMemory( pTemp, pGenericFilter->Mask, PatternSize );
pIpPattern = (PIP_PATTERN) pTemp;
pIpPattern->Reserved1 = pIpPattern->Reserved2 = 0xffffffff;
pIpPattern->Reserved3[0] = pIpPattern->Reserved3[1] = pIpPattern->Reserved3[2] = 0xff;
// Time to invoke Gpsy
Status = UlpTcDeviceControl( g_GpcFileHandle,
NULL,
NULL,
NULL,
&IoStatBlock,
IOCTL_GPC_ADD_PATTERN,
pGpcReq,
InBuffSize,
&GpcRes,
OutBuffSize);
if (!NT_SUCCESS(Status))
goto end;
//
// Insert the freshly created filter to flow
//
pFilter->FilterHandle = (HANDLE) GpcRes.GpcPatternHandle;
UlpInsertFilterEntry( pFilter, pFlow );
//
// Success!
//
*ppFilter = pFilter;
end:
if (!NT_SUCCESS(Status))
{
UlTrace( TC, ("Ul!UlpTcAddFilter: FAILURE %08lx \n", Status ));
// Cleanup filter only if we failed, otherwise it will go to
// the filterlist of the flow.
if (pFilter)
{
UL_FREE_POOL( pFilter, UL_TCI_FILTER_POOL_TAG );
}
}
// Cleanup the temp Gpc buffer which we used to pass down filter info
// to GPC. We don't need it anymore.
if (pGpcReq)
{
UL_FREE_POOL( pGpcReq, UL_TCI_GENERIC_POOL_TAG );
}
return Status;
}
/***************************************************************************++
Routine Description:
UlTcDeleteFilter :
Connection only deletes the filter prior to deleting itself. Any
operation initiated by the connection requires tc resource shared
and none of those cause race condition.
Anything other than this, such as flow & filter removal because of
BW disabling on the site will acquire the lock exclusively. Hence
the pFlow & pFilter are safe as long as we acquire the tc resource
shared.
Arguments:
connection object to get the flow & filter after we acquire the tc lock
--***************************************************************************/
NTSTATUS
UlTcDeleteFilter(
IN PUL_HTTP_CONNECTION pHttpConnection
)
{
NTSTATUS Status;
//
// Sanity check
//
Status = STATUS_SUCCESS;
//
// If we have been called w/o being initialized
//
ASSERT(g_InitTciCalled);
UlTrace(TC,("Ul!UlTcDeleteFilter: for connection %p\n", pHttpConnection));
UlAcquireResourceShared(&g_pUlNonpagedData->TciIfcResource, TRUE);
if (pHttpConnection->pFlow)
{
Status = UlpTcDeleteFilter(
pHttpConnection->pFlow,
pHttpConnection->pFilter
);
}
UlReleaseResource(&g_pUlNonpagedData->TciIfcResource);
return Status;
}
/***************************************************************************++
Routine Description:
UlpTcRemoveFilter :
Arguments:
flow & filter
--***************************************************************************/
NTSTATUS
UlpTcDeleteFilter(
IN PUL_TCI_FLOW pFlow,
IN PUL_TCI_FILTER pFilter
)
{
NTSTATUS Status;
HANDLE FilterHandle;
//
// Sanity check
//
Status = STATUS_SUCCESS;
ASSERT(IS_VALID_TCI_FLOW(pFlow));
ASSERT(IS_VALID_TCI_FILTER(pFilter));
if (pFlow == NULL || pFilter == NULL)
{
return STATUS_INVALID_PARAMETER;
}
FilterHandle = pFilter->FilterHandle;
pFilter->pHttpConnection->pFlow = NULL;
pFilter->pHttpConnection->pFilter = NULL;
//
// Now call the actual worker for us
//
UlpRemoveFilterEntry( pFilter, pFlow );
Status = UlpTcDeleteGpcFilter( FilterHandle );
if (!NT_SUCCESS(Status))
{
UlTrace( TC, ("Ul!UlpTcDeleteFilter: FAILURE %08lx \n", Status ));
}
return Status;
}
/***************************************************************************++
Routine Description:
UlpTcRemoveFilter :
This procedure builds up the structure necessary to delete a filter.
It then calls a routine to pass this info to the GPC.
Arguments:
FilterHandle - Handle of the filter to be deleted
--***************************************************************************/
NTSTATUS
UlpTcDeleteGpcFilter(
IN HANDLE FilterHandle
)
{
NTSTATUS Status;
ULONG InBuffSize;
ULONG OutBuffSize;
GPC_REMOVE_PATTERN_REQ GpcReq;
GPC_REMOVE_PATTERN_RES GpcRes;
IO_STATUS_BLOCK IoStatBlock;
Status = STATUS_SUCCESS;
ASSERT(FilterHandle != NULL);
InBuffSize = sizeof(GPC_REMOVE_PATTERN_REQ);
OutBuffSize = sizeof(GPC_REMOVE_PATTERN_RES);
GpcReq.ClientHandle = g_GpcClientHandle;
GpcReq.GpcPatternHandle = FilterHandle;
ASSERT(g_GpcFileHandle);
ASSERT(GpcReq.ClientHandle);
ASSERT(GpcReq.GpcPatternHandle);
Status = UlpTcDeviceControl( g_GpcFileHandle,
NULL,
NULL,
NULL,
&IoStatBlock,
IOCTL_GPC_REMOVE_PATTERN,
&GpcReq,
InBuffSize,
&GpcRes,
OutBuffSize
);
if (!NT_SUCCESS(Status))
{
UlTrace( TC, ("Ul!UlpTcDeleteGpcFilter: FAILURE %08lx \n", Status ));
}
else
{
UlTrace( TC, ("Ul!UlpTcDeleteGpcFilter: FilterHandle %d deleted in TC as well.\n",
FilterHandle
));
}
return Status;
}
/***************************************************************************++
Routine Description:
UlpInsertFilterEntry :
Inserts a filter entry to the filter list of the flow.
Arguments:
pEntry - The filter entry to be added to the flow list
--***************************************************************************/
VOID
UlpInsertFilterEntry(
IN PUL_TCI_FILTER pEntry,
IN OUT PUL_TCI_FLOW pFlow
)
{
LONGLONG listSize;
KIRQL oldIrql;
//
// Sanity check.
//
ASSERT(pEntry);
ASSERT(IS_VALID_TCI_FILTER(pEntry));
ASSERT(pFlow);
//
// add to the list
//
UlAcquireSpinLock( &pFlow->FilterListSpinLock, &oldIrql );
InsertHeadList( &pFlow->FilterList, &pEntry->Linkage );
pFlow->FilterListSize += 1;
listSize = pFlow->FilterListSize;
UlReleaseSpinLock( &pFlow->FilterListSpinLock, oldIrql );
ASSERT( listSize >= 1);
}
/***************************************************************************++
Routine Description:
UlRemoveFilterEntry :
Removes a filter entry frm the filter list of the flow.
Arguments:
pEntry - The filter entry to be removed from the flow list
--***************************************************************************/
VOID
UlpRemoveFilterEntry(
IN PUL_TCI_FILTER pEntry,
IN OUT PUL_TCI_FLOW pFlow
)
{
LONGLONG listSize;
KIRQL oldIrql;
//
// Sanity check.
//
ASSERT(IS_VALID_TCI_FLOW(pFlow));
ASSERT(IS_VALID_TCI_FILTER(pEntry));
//
// And the work
//
UlAcquireSpinLock( &pFlow->FilterListSpinLock, &oldIrql );
RemoveEntryList( &pEntry->Linkage );
pFlow->FilterListSize -= 1;
listSize = pFlow->FilterListSize;
pEntry->Linkage.Flink = pEntry->Linkage.Blink = NULL;
UlReleaseSpinLock( &pFlow->FilterListSpinLock, oldIrql );
ASSERT( listSize >= 0 );
UlTrace( TC, ("Ul!UlpRemoveFilterEntry: FilterEntry %p removed/deleted.\n",
pEntry
));
UL_FREE_POOL_WITH_SIG( pEntry, UL_TCI_FILTER_POOL_TAG );
}
//
// Various helpful utilities for TCI module
//
/***************************************************************************++
Routine Description:
UlpFindFlow :
Find the flow in the cgroups flow list by looking at the IP address
of each flows interface. The rule is cgroup will install one flow
on each interface available.
By having a flow list in each cgroup we are able to do a faster
flow lookup. This is more scalable than doing a linear search for
all the flows of the interface.
Arguments:
pCGroup - The config group of the site
IpAddress - The address we are searching for
Return Value:
NTSTATUS - Completion status.
--***************************************************************************/
PUL_TCI_FLOW
UlpFindFlow(
IN PUL_CONFIG_GROUP_OBJECT pCgroup OPTIONAL,
IN ULONG IpAddress
)
{
PLIST_ENTRY pFlowEntry;
PUL_TCI_FLOW pFlow;
PLIST_ENTRY pInterfaceEntry;
PUL_TCI_INTERFACE pInterface;
//
// Drop the lookup if the IP is zero. Basically by doing this
// we are rejecting any filters attached to flows on interfaces
// with zero IPs. The interface list may have interface(s)
// with zero IPs. They are just idle until ip change notification
// comes.
//
if (IpAddress == 0)
{
return NULL;
}
//
// Otherwise proceed with the flow lookup on current interfaces.
//
if (pCgroup)
{
// Look in the cgroup's flows
ASSERT(IS_VALID_CONFIG_GROUP(pCgroup));
pFlowEntry = pCgroup->FlowListHead.Flink;
while ( pFlowEntry != &pCgroup->FlowListHead )
{
pFlow = CONTAINING_RECORD(
pFlowEntry,
UL_TCI_FLOW,
Siblings
);
if (pFlow->pInterface->IpAddr == IpAddress)
{
return pFlow;
}
pFlowEntry = pFlowEntry->Flink;
}
}
else
{
// Or go through the interface list to find the global flow
pInterfaceEntry = g_TciIfcListHead.Flink;
while ( pInterfaceEntry != &g_TciIfcListHead )
{
pInterface = CONTAINING_RECORD(
pInterfaceEntry,
UL_TCI_INTERFACE,
Linkage
);
if (pInterface->IpAddr == IpAddress)
{
return pInterface->pGlobalFlow;
}
pInterfaceEntry = pInterfaceEntry->Flink;
}
}
return NULL;
}
/***************************************************************************++
Routine Description:
UlpFindInterface :
Find the interface in our global link list by looking at its IP
Arguments:
IpAddr - The ip address to be find among the interfaces
Return Value:
Pointer to interface if found or else null
--***************************************************************************/
PUL_TCI_INTERFACE
UlpFindInterface(
IN ULONG IpAddr
)
{
PLIST_ENTRY pEntry;
PUL_TCI_INTERFACE pInterface;
pEntry = g_TciIfcListHead.Flink;
while ( pEntry != &g_TciIfcListHead )
{
pInterface = CONTAINING_RECORD(
pEntry,
UL_TCI_INTERFACE,
Linkage
);
if ( pInterface->IpAddr == IpAddr )
{
return pInterface;
}
pEntry = pEntry->Flink;
}
return NULL;
}
/***************************************************************************++
Routine Description:
UlpTcDeviceControl :
Arguments:
As usual
Return Value:
NTSTATUS - Completion status.
--***************************************************************************/
NTSTATUS
UlpTcDeviceControl(
IN HANDLE FileHandle,
IN HANDLE EventHandle,
IN PIO_APC_ROUTINE ApcRoutine,
IN PVOID ApcContext,
OUT PIO_STATUS_BLOCK pIoStatusBlock,
IN ULONG Ioctl,
IN PVOID InBuffer,
IN ULONG InBufferSize,
IN PVOID OutBuffer,
IN ULONG OutBufferSize
)
{
NTSTATUS Status;
//
// Sanity check.
//
PAGED_CODE();
Status = STATUS_SUCCESS;
UlAttachToSystemProcess();
Status = ZwDeviceIoControlFile(
FileHandle, // FileHandle
NULL, // Event
NULL, // ApcRoutine
NULL, // ApcContext
pIoStatusBlock, // IoStatusBlock
Ioctl, // IoControlCode
InBuffer, // InputBuffer
InBufferSize, // InputBufferLength
OutBuffer, // OutputBuffer
OutBufferSize // OutputBufferLength
);
if (Status == STATUS_PENDING)
{
Status = ZwWaitForSingleObject(
FileHandle, // Handle
TRUE, // Alertable
NULL // Timeout
);
Status = pIoStatusBlock->Status;
}
UlDetachFromSystemProcess();
return Status;
}
/***************************************************************************++
Routine Description:
UlDumpTCInterface :
Helper utility to display interface content.
Arguments:
PUL_TCI_INTERFACE - TC Interface to be dumped
--***************************************************************************/
VOID
UlDumpTCInterface(
IN PUL_TCI_INTERFACE pTcIfc
)
{
ULONG IpAddress;
ASSERT(IS_VALID_TCI_INTERFACE(pTcIfc));
IpAddress = SWAP_LONG(pTcIfc->IpAddr);
UlTrace( TC,("Ul!UlDumpTCInterface: \n pTcIfc @ %p\n"
"\t Signature = %08lx \n",
pTcIfc, pTcIfc->Signature));
UlTrace( TC,(
"\t IsQoSEnabled: = %u \n"
"\t IpAddr: = (%x) %d.%d.%d.%d \n"
"\t IfIndex: = %d \n"
"\t SpecificLinkCtx: = %d \n"
"\t MTUSize: = %d \n"
"\t NameLength: = %u \n"
"\t Name: = %ws \n"
"\t InstanceIDLength: = %u \n"
"\t InstanceID: = %ws \n"
"\t pGlobalFlow = %p \n"
"\t FlowListSize: = %d \n"
"\t AddrListBytesCount: = %d \n"
"\t pAddressListDesc: = %p \n",
pTcIfc->IsQoSEnabled,
pTcIfc->IpAddr,
(UCHAR)(IpAddress >> 24),
(UCHAR)(IpAddress >> 16),
(UCHAR)(IpAddress >> 8),
(UCHAR)(IpAddress >> 0),
pTcIfc->IfIndex,
pTcIfc->SpecificLinkCtx,
pTcIfc->MTUSize,
pTcIfc->NameLength,
pTcIfc->Name,
pTcIfc->InstanceIDLength,
pTcIfc->InstanceID,
pTcIfc->pGlobalFlow,
pTcIfc->FlowListSize,
pTcIfc->AddrListBytesCount,
pTcIfc->pAddressListDesc
));
}
/***************************************************************************++
Routine Description:
UlDumpTCFlow :
Helper utility to display interface content.
Arguments:
PUL_TCI_FLOW - TC Flow to be dumped
--***************************************************************************/
VOID
UlDumpTCFlow(
IN PUL_TCI_FLOW pFlow
)
{
ASSERT(IS_VALID_TCI_FLOW(pFlow));
UlTrace( TC,
("Ul!UlDumpTCFlow: \n"
" pFlow @ %p\n"
"\t Signature = %08lx \n"
"\t pInterface @ %p \n"
"\t FlowHandle = %d \n"
"\t GenFlow @ %p \n"
"\t FlowRate KB/s = %d \n"
"\t FilterListSize = %I64d \n"
"\t pConfigGroup = %p \n"
,
pFlow,
pFlow->Signature,
pFlow->pInterface,
pFlow->FlowHandle,
&pFlow->GenFlow,
pFlow->GenFlow.SendingFlowspec.TokenRate / 1024,
pFlow->FilterListSize,
pFlow->pConfigGroup
));
}
/***************************************************************************++
Routine Description:
UlDumpTCFilter :
Helper utility to display filter structure content.
Arguments:
PUL_TCI_FILTER pFilter
--***************************************************************************/
VOID
UlDumpTCFilter(
IN PUL_TCI_FILTER pFilter
)
{
ASSERT(IS_VALID_TCI_FILTER(pFilter));
UlTrace( TC,
("Ul!UlDumpTCFilter: \n"
" pFilter @ %p\n"
"\t Signature = %08lx \n"
"\t pHttpConnection = %p \n"
"\t FilterHandle = %d \n",
pFilter,
pFilter->Signature,
pFilter->pHttpConnection,
pFilter->FilterHandle
));
}