|
|
/*
************************************************************************
Copyright (c) 1996-1997 Microsoft Corporation
Module Name:
gpcdb.c
Abstract:
This file contains specific patterns database routines
Author:
Ofer Bar -April 15, 1997
Environment:
Kernel mode
Revision History:
************************************************************************ */
#include "gpcpre.h"
//
// locals
//
/*
************************************************************************
CreateNewCfBlock -
create and return a new CF block for internal use
Arguments
CfId - MaxPriorities -
Returns pointer to the new CF block or NULL for no memory resources
************************************************************************ */ PCF_BLOCK CreateNewCfBlock( IN ULONG CfId, IN ULONG MaxPriorities ) { PCF_BLOCK pCf; int i; GPC_STATUS Status;
GpcAllocFromLL(&pCf, &ClassificationFamilyLL, ClassificationFamilyTag);
if (pCf) {
//
// reset all
//
NdisZeroMemory(pCf, sizeof(CF_BLOCK));
InitializeListHead(&pCf->ClientList); InitializeListHead(&pCf->BlobList); NDIS_INIT_LOCK(&pCf->Lock); //pCf->ClientSync = 0;
//INIT_LOCK(&pCf->ClientSync);
NDIS_INIT_LOCK(&pCf->ClientSync);
pCf->NumberOfClients = 0; pCf->AssignedIndex = CfId; pCf->MaxPriorities = MaxPriorities; ReferenceInit(&pCf->RefCount, pCf, ReleaseCfBlock); REFADD(&pCf->RefCount, 'CFBK');
//
// for each protocol, reset the generic pattern
// this will be dynamically allocated later
// when pattern are installed
//
for (i = 0; i < GPC_PROTOCOL_TEMPLATE_MAX; i++) {
Status = InitializeGenericDb(&pCf->arpGenericDb[i], MaxPriorities, glData.pProtocols[i].PatternSize); if (Status != GPC_STATUS_SUCCESS) { REFDEL(&pCf->RefCount, 'CFBK'); pCf = NULL; break; } } }
return pCf; }
/*
************************************************************************
ReleaseCfBlock -
Release the CF block
Arguments
pCf - the CF block to release
Returns void
************************************************************************ */ VOID ReleaseCfBlock( IN PCF_BLOCK pCf ) { int i;
ASSERT(pCf); ASSERT(IsListEmpty(&pCf->ClientList)); ASSERT(IsListEmpty(&pCf->BlobList)); ASSERT(pCf->NumberOfClients == 0);
for (i = 0; i < GPC_PROTOCOL_TEMPLATE_MAX; i++) { if (pCf->arpGenericDb[i]) { UninitializeGenericDb(&pCf->arpGenericDb[i], pCf->MaxPriorities ); } } NdisFreeSpinLock(&pCf->Lock);
//
// just free the memory
//
GpcFreeToLL(pCf, &ClassificationFamilyLL, ClassificationFamilyTag); }
/*
************************************************************************
CreateNewClientBlock -
create and return a new client block for internal use
Arguments none
Returns pointer to the new client block or NULL for no memory resources
************************************************************************ */ PCLIENT_BLOCK CreateNewClientBlock(VOID) { PCLIENT_BLOCK pClient;
GpcAllocFromLL(&pClient, &ClientLL, ClientTag);
if (pClient) {
//
// reset all
//
NdisZeroMemory(pClient, sizeof(CLIENT_BLOCK));
pClient->ObjectType = GPC_ENUM_CLIENT_TYPE; pClient->ClHandle = NULL;
NDIS_INIT_LOCK(&pClient->Lock); InitializeListHead(&pClient->BlobList); ReferenceInit(&pClient->RefCount, pClient, DereferenceClient); REFADD(&pClient->RefCount, 'CLNT'); }
return pClient; }
/*
************************************************************************
CreateNewPatternBlock -
create and return a new pattern block for internal use
Arguments none
Returns pointer to the new pattern block or NULL for no memory resources
************************************************************************ */ PPATTERN_BLOCK CreateNewPatternBlock( IN ULONG Flags ) { PPATTERN_BLOCK pPattern; int i;
GpcAllocFromLL(&pPattern, &PatternLL, PatternTag);
if (pPattern) {
//
// reset all
//
TRACE(PATTERN, pPattern, sizeof(PATTERN_BLOCK), "CreateNewPatternBlock");
NdisZeroMemory(pPattern, sizeof(PATTERN_BLOCK));
pPattern->ObjectType = GPC_ENUM_PATTERN_TYPE; for (i = 0; i < GPC_CF_MAX; i++) { InitializeListHead(&pPattern->BlobLinkage[i]); }
InitializeListHead(&pPattern->TimerLinkage); NDIS_INIT_LOCK(&pPattern->Lock); pPattern->Flags = Flags; ReferenceInit(&pPattern->RefCount, pPattern, DereferencePattern); REFADD(&pPattern->RefCount, 'FILT'); pPattern->ClientRefCount = 1; pPattern->State = GPC_STATE_INIT;
//AllocateHandle(&pPattern->ClHandle, (PVOID)pPattern);
}
return pPattern; }
/*
************************************************************************
CreateNewBlobBlock -
create and return a new blob block for internal use. A copy of ClientData is being place pointed by the new blob
Arguments none
Returns pointer to the new blob block or NULL for no memory resources
************************************************************************ */ PBLOB_BLOCK CreateNewBlobBlock( IN ULONG ClientDataSize, IN PVOID pClientData, BOOLEAN fChargeQuota // Should I charge Quota?
) { // Quota Charging should be done only if we have a proces context
PBLOB_BLOCK pBlob;
if (fChargeQuota){ GpcAllocMemWithQuota(&pBlob,CfInfoLLSize,CfInfoTag); } else{ GpcAllocMem(&pBlob,CfInfoLLSize,CfInfoTag); }
if (pBlob) {
//
// reset all
//
NdisZeroMemory(pBlob, sizeof(BLOB_BLOCK));
if (fChargeQuota){ GpcAllocMemWithQuota(&pBlob->pClientData, ClientDataSize, CfInfoDataTag); } else{ GpcAllocMem(&pBlob->pClientData, ClientDataSize, CfInfoDataTag); }
if (pBlob->pClientData) {
pBlob->ObjectType = GPC_ENUM_CFINFO_TYPE; pBlob->ClHandle = NULL; InitializeListHead(&pBlob->ClientLinkage); InitializeListHead(&pBlob->PatternList); InitializeListHead(&pBlob->CfLinkage);
pBlob->State = GPC_STATE_INIT; pBlob->Flags = 0; ReferenceInit(&pBlob->RefCount, pBlob, DereferenceBlob); REFADD(&pBlob->RefCount, 'BLOB'); pBlob->ClientDataSize = ClientDataSize; pBlob->LastStatus = GPC_STATUS_SUCCESS; NdisMoveMemory(pBlob->pClientData, pClientData, ClientDataSize); NDIS_INIT_LOCK(&pBlob->Lock);
//
// that's for the notified client about the CfInfo Add
//
pBlob->pNotifiedClient = NULL; pBlob->NotifiedClientCtx = NULL;
} else {
//GpcFreeToLL(pBlob, &CfInfoLL, CfInfoTag);
GpcFreeMem(pBlob,CfInfoTag);
pBlob = NULL; } }
return pBlob; }
/*
************************************************************************
AssignNewClientIndex -
Finds and returns a new index for a client on the CF. It also sets the index as busy in the CF block.
Arguments pClient - poinetr to the client block
Returns a client index or (-1) for none
************************************************************************ */ ULONG AssignNewClientIndex( IN PCF_BLOCK pCfBlock ) { ULONG i;
for (i = 0; i < MAX_CLIENTS_CTX_PER_BLOB; i++) {
if (TEST_BIT_OFF(pCfBlock->ClientIndexes, (1 << i))) break; }
if (i < MAX_CLIENTS_CTX_PER_BLOB) {
//
// found a zero bit, set it on
//
pCfBlock->ClientIndexes |= (1 << i);
} else { i = (-1); }
return i; }
/*
************************************************************************
ReleaseClientBlock -
Release the client block
Arguments pClientBlock - the client block to release
Returns void
************************************************************************ */ VOID ReleaseClientBlock( IN PCLIENT_BLOCK pClientBlock ) { NdisFreeSpinLock(&pClientBlock->Lock);
//
// just free the memory
//
GpcFreeToLL(pClientBlock, &ClientLL, ClientTag); }
/*
************************************************************************
ReleasePatternBlock -
Release the pattern block
Arguments pPatternBlock - the pattern block to release
Returns void
************************************************************************ */ VOID ReleasePatternBlock( IN PPATTERN_BLOCK pPatternBlock ) {
#if DBG
pPatternBlock->TimeToLive = 0xdeadbeef; #endif
//
// just free the memory
//
GpcFreeToLL(pPatternBlock, &PatternLL, PatternTag); }
/*
************************************************************************
ReleaseBlobBlock -
Release the blob block
Arguments pBlobBlock - the blob block to release
Returns void
************************************************************************ */ VOID ReleaseBlobBlock( IN PBLOB_BLOCK pBlobBlock ) {
//
// just free the memory
//
GpcFreeMem(pBlobBlock->pClientData, CfInfoDataTag); ASSERT(pBlobBlock->pNewClientData == NULL); //GpcFreeToLL(pBlobBlock, &CfInfoLL, CfInfoTag);
GpcFreeMem(pBlobBlock,CfInfoTag); }
/*
************************************************************************
CreateNewClassificationBlock -
create and return a new classification block for internal use
Arguments NumEntries - number of entries
Returns pointer to the new classification block or NULL for no memory resources
************************************************************************ */ PCLASSIFICATION_BLOCK CreateNewClassificationBlock( IN ULONG NumEntries ) { PCLASSIFICATION_BLOCK pClassification; ULONG l;
l = sizeof(CLASSIFICATION_BLOCK) + sizeof(PBLOB_BLOCK) * (NumEntries-1); GpcAllocMem(&pClassification, l, ClassificationBlockTag);
if (pClassification) {
//
// reset all
//
NdisZeroMemory(pClassification, l);
pClassification->NumberOfElements = NumEntries; }
return pClassification; }
/*
************************************************************************
ReleaseClassificationBlock -
Release the Classification block
Arguments pClassificationBlock - the Classification block to release
Returns void
************************************************************************ */ VOID ReleaseClassificationBlock( IN PCLASSIFICATION_BLOCK pClassificationBlock ) { if (pClassificationBlock) {
//
// release the memory block
//
GpcFreeMem(pClassificationBlock, ClassificationBlockTag); } }
/*
************************************************************************
GpcCalcHash -
Calculate the hash table key for a specific pattern
Arguments ProtocolTempId - The protocol template pPattern - a pointer to the pattern
Returns ULONG - the hash key, (-1) if Protocol value is illegal
************************************************************************ */ ULONG GpcCalcHash( IN ULONG ProtocolTempId, IN PUCHAR pPattern ) { ULONG Key = (-1); ULONG temp; PULONG Cheat; PGPC_IP_PATTERN pIpPatt; PGPC_IPX_PATTERN pIpxPatt; const ULONG MagicNumber = 0x9e4155b9; // magic number - hocus pocus
TRACE(LOOKUP, ProtocolTempId, pPattern, "GpcClacHash");
ASSERT(pPattern);
switch (ProtocolTempId) {
case GPC_PROTOCOL_TEMPLATE_IP:
//
// the IP protocol template, this function was contributed by
// JohnDo
//
pIpPatt = (PGPC_IP_PATTERN)pPattern; temp = (pIpPatt->SrcAddr << 16) ^ (pIpPatt->SrcAddr >> 16) ^ pIpPatt->DstAddr ^ pIpPatt->ProtocolId ^ pIpPatt->gpcSpi; Key = temp * MagicNumber; break;
case GPC_PROTOCOL_TEMPLATE_IPX:
//
// the IPX protocol template, this function was contributed by
// JohnDo
//
Cheat = (PULONG)pPattern; temp = (Cheat[0] << 16) ^ (Cheat[0] >> 16) ^ (Cheat[1] << 16) ^ (Cheat[1] >> 16) ^ (Cheat[2] << 16) ^ (Cheat[2] >> 16) ^ Cheat[3] ^ Cheat[4] ^ Cheat[5];
Key = temp * MagicNumber; break;
default:
//
// illegal value
//
ASSERT(0); }
//
// -1 is a bad key
//
if (Key == (-1)) Key = 0;
TRACE(LOOKUP, Key, 0, "GpcClacHash==>");
return Key; }
/*
************************************************************************
DereferencePattern -
Decrement the RefCount and deletes the pattern block if the count reaches zero.
Arguments pPattern - a pointer to the pattern
Returns void
************************************************************************ */ VOID DereferencePattern( IN PPATTERN_BLOCK pPattern ) { ASSERT(pPattern); //ASSERT(pPattern->RefCount.Count > 0);
TRACE(PATTERN, pPattern, pPattern->DbCtx, "DereferencePattern");
ProtocolStatInc(pPattern->ProtocolTemplate, DerefPattern2Zero);
ASSERT(IsListEmpty(&pPattern->TimerLinkage));
//
// no longer do we need this CB
//
ReleaseClassificationBlock(pPattern->pClassificationBlock);
//
// time to remove the pattern
//
ReleasePatternBlock(pPattern);
}
/*
************************************************************************
DereferenceBlob -
Decrement the RefCount and deletes the blob block if the count reaches zero.
Arguments pBlob - a pointer to the blob
Returns void
************************************************************************ */ VOID DereferenceBlob( IN PBLOB_BLOCK pBlob ) { ASSERT(pBlob); //
// If there is a File Object, dereference it.
//
if (pBlob->FileObject) { ObDereferenceObject(pBlob->FileObject); pBlob->FileObject = NULL; }
if (pBlob->Pattern) { GpcFreeMem(pBlob->Pattern,TcpPatternTag); pBlob->Pattern = NULL; }
//ASSERT(*ppBlob);
//TRACE(BLOB, *ppBlob, (*ppBlob)->RefCount, "DereferenceBlob");
//ASSERT((*ppBlob)->RefCount.Count > 0);
CfStatInc(pBlob->pOwnerClient->pCfBlock->AssignedIndex,DerefBlobs2Zero);
//
// time to remove the blob
//
ReleaseBlobBlock(pBlob); }
/*
************************************************************************
DereferenceClient -
Decrement the RefCount and deletes the client block if the count reaches zero.
Arguments pClient - pointer to the client block
Returns void
************************************************************************ */ VOID DereferenceClient( IN PCLIENT_BLOCK pClient ) { PCF_BLOCK pCf; KIRQL irql;
ASSERT(pClient);
//TRACE(CLIENT, pClient, pClient->RefCount, "DereferenceClient");
//ASSERT(pClient->RefCount.Count > 0);
pCf = pClient->pCfBlock;
RSC_WRITE_LOCK(&pCf->ClientSync, &irql);
//
// time to remove the client
//
GpcRemoveEntryList(&pClient->ClientLinkage); ReleaseClientIndex(pCf->ClientIndexes, pClient->AssignedIndex);
ReleaseClientBlock(pClient);
RSC_WRITE_UNLOCK(&pCf->ClientSync, irql); }
/*
************************************************************************
ClientAddCfInfo -
Arguments
Returns
************************************************************************ */ GPC_STATUS ClientAddCfInfo( IN PCLIENT_BLOCK pClient, IN PBLOB_BLOCK pBlob, OUT PGPC_CLIENT_HANDLE pClientCfInfoContext ) { GPC_STATUS Status = GPC_STATUS_SUCCESS; DEFINE_KIRQL(dbgIrql);
TRACE(PATTERN, pClient, pBlob, "ClientAddCfInfo");
*pClientCfInfoContext = NULL;
if (pClient->State == GPC_STATE_READY) {
if (pClient->FuncList.ClAddCfInfoNotifyHandler) { NdisInterlockedIncrement(&pBlob->ClientStatusCountDown);
GET_IRQL(dbgIrql);
TRACE(PATTERN, pClient, dbgIrql, "ClientAddCfInfo (2)");
// Pass in link id if a client of the new interface registers
if (IS_USERMODE_CLIENT_EX(pBlob->pOwnerClient)){ Status = (pClient->FuncList.ClAddCfInfoNotifyHandler) ( pClient->ClientCtx, (GPC_HANDLE)pBlob, &(pBlob->Pattern->InterfaceId), pBlob->ClientDataSize, pBlob->pClientData, pClientCfInfoContext ); } else{ Status = (pClient->FuncList.ClAddCfInfoNotifyHandler) ( pClient->ClientCtx, (GPC_HANDLE)pBlob, NULL, pBlob->ClientDataSize, pBlob->pClientData, pClientCfInfoContext ); }
VERIFY_IRQL(dbgIrql);
TRACE(PATTERN, *pClientCfInfoContext, Status, "ClientAddCfInfo (3)");
if (Status != GPC_STATUS_PENDING) {
NdisInterlockedDecrement(&pBlob->ClientStatusCountDown); }
} }
return Status; }
/*
************************************************************************
ClientAddCfInfoComplete -
Arguments
Returns
************************************************************************ */ VOID ClientAddCfInfoComplete( IN PCLIENT_BLOCK pClient, IN PBLOB_BLOCK pBlob, IN GPC_STATUS Status ) { ULONG CfIndex; DEFINE_KIRQL(dbgIrql);
TRACE(PATTERN, pClient, pBlob, "ClientAddCfInfoComplete");
ASSERT(Status != GPC_STATUS_PENDING); ASSERT(pClient); ASSERT(pBlob);
if (pClient->State == GPC_STATE_READY) {
#if NO_USER_PENDING
//
// the user is blocking on this call
//
CTESignal(&pBlob->WaitBlock, Status);
#else
CfIndex = pClient->pCfBlock->AssignedIndex;
if (NT_SUCCESS(Status)) { CfStatInc(CfIndex,CreatedBlobs); CfStatInc(CfIndex,CurrentBlobs); } else { CfStatInc(CfIndex,RejectedBlobs); }
if (pClient->FuncList.ClAddCfInfoCompleteHandler) { GET_IRQL(dbgIrql);
TRACE(PATTERN, pClient, dbgIrql, "ClientAddCfInfoComplete (2)"); TRACE(PATTERN, pClient, pBlob->arClientCtx[pClient->AssignedIndex], "ClientAddCfInfoComplete (3)");
(pClient->FuncList.ClAddCfInfoCompleteHandler) ( pClient->ClientCtx, pBlob->arClientCtx[pClient->AssignedIndex], Status );
VERIFY_IRQL(dbgIrql);
TRACE(PATTERN, pClient, Status, "ClientAddCfInfoComplete (4)");
} else if (IS_USERMODE_CLIENT(pClient) && pClient == pBlob->pOwnerClient ) {
GET_IRQL(dbgIrql);
TRACE(PATTERN, pClient, dbgIrql, "ClientAddCfInfoComplete (5)");
//
// this is a user mode client - call the specific routine
// to complete a pending IRP, but only if the client is the
// blob owner
//
UMCfInfoComplete( OP_ADD_CFINFO, pClient, pBlob, Status ); VERIFY_IRQL(dbgIrql);
TRACE(PATTERN, pClient, Status, "ClientAddCfInfoComplete (6)");
}
if (NT_SUCCESS(Status)) { pBlob->State = GPC_STATE_READY; } else { //
// remove the blob
//
REFDEL(&pBlob->RefCount, 'BLOB');
} #endif
} }
/*
************************************************************************
ClientModifyCfInfoComplete -
Arguments
Returns
************************************************************************ */ VOID ClientModifyCfInfoComplete( IN PCLIENT_BLOCK pClient, IN PBLOB_BLOCK pBlob, IN GPC_STATUS Status ) { DEFINE_KIRQL(dbgIrql);
TRACE(PATTERN, pClient, pBlob, "ClientModifyCfInfoComplete");
ASSERT(Status != GPC_STATUS_PENDING);
if (pClient->State == GPC_STATE_READY) {
if (pClient->FuncList.ClModifyCfInfoCompleteHandler) { GET_IRQL(dbgIrql);
TRACE(PATTERN, pClient, dbgIrql, "ClientModifyCfInfoComplete (2)");
(pClient->FuncList.ClModifyCfInfoCompleteHandler) ( pClient->ClientCtx, pBlob->arClientCtx[pClient->AssignedIndex], Status );
VERIFY_IRQL(dbgIrql);
TRACE(PATTERN, pBlob->arClientCtx[pClient->AssignedIndex], Status, "ClientModifyCfInfoComplete (3)");
} else if (IS_USERMODE_CLIENT(pClient)) {
GET_IRQL(dbgIrql);
TRACE(PATTERN, pClient, dbgIrql, "ClientModifyCfInfoComplete (4)");
UMCfInfoComplete( OP_MODIFY_CFINFO, pClient, pBlob, Status ); VERIFY_IRQL(dbgIrql);
TRACE(PATTERN, pBlob->arClientCtx[pClient->AssignedIndex], Status, "ClientModifyCfInfoComplete (5)"); }
} }
/*
************************************************************************
ClientModifyCfInfo -
Arguments
Returns
************************************************************************ */ GPC_STATUS ClientModifyCfInfo( IN PCLIENT_BLOCK pClient, IN PBLOB_BLOCK pBlob, IN ULONG CfInfoSize, IN PVOID pClientData ) { GPC_STATUS Status = GPC_STATUS_IGNORED; DEFINE_KIRQL(dbgIrql);
TRACE(PATTERN, pClient, pBlob, "ClientModifyCfInfo");
if (pClient->State == GPC_STATE_READY) {
if (pClient->FuncList.ClModifyCfInfoNotifyHandler) {
NdisInterlockedIncrement(&pBlob->ClientStatusCountDown);
GET_IRQL(dbgIrql);
TRACE(PATTERN, pClient, dbgIrql, "ClientModifyCfInfo (2)");
Status = (pClient->FuncList.ClModifyCfInfoNotifyHandler) (pClient->ClientCtx, pBlob->arClientCtx[pClient->AssignedIndex], CfInfoSize, pClientData );
VERIFY_IRQL(dbgIrql);
TRACE(PATTERN, pBlob->arClientCtx[pClient->AssignedIndex], Status, "ClientModifyCfInfo (3)");
if (Status != GPC_STATUS_PENDING) { NdisInterlockedDecrement(&pBlob->ClientStatusCountDown); }
} }
return Status; }
/*
************************************************************************
ClientRemoveCfInfoComplete -
Arguments
Returns
************************************************************************ */ VOID ClientRemoveCfInfoComplete( IN PCLIENT_BLOCK pClient, IN PBLOB_BLOCK pBlob, IN GPC_STATUS Status ) { ULONG CfIndex; DEFINE_KIRQL(dbgIrql);
TRACE(PATTERN, pClient, pBlob, "ClientRemoveCfInfoComplete");
ASSERT(Status != GPC_STATUS_PENDING); ASSERT(pClient); ASSERT(pBlob);
#if NO_USER_PENDING
//
// the user is blocking on this call
//
CTESignal(&pBlob->WaitBlock, Status);
#else
CfIndex = pClient->pCfBlock->AssignedIndex;
if (NT_SUCCESS(Status)) { CfStatInc(CfIndex,DeletedBlobs); CfStatDec(CfIndex,CurrentBlobs); } if (pClient->FuncList.ClRemoveCfInfoCompleteHandler) { GET_IRQL(dbgIrql);
TRACE(PATTERN, pClient, pBlob->arClientCtx[pClient->AssignedIndex], "ClientRemoveCfInfoComplete (2)");
(pClient->FuncList.ClRemoveCfInfoCompleteHandler) ( pClient->ClientCtx, pBlob->arClientCtx[pClient->AssignedIndex], Status ); VERIFY_IRQL(dbgIrql);
TRACE(PATTERN, pClient, Status, "ClientRemoveCfInfoComplete (3)");
} else if (IS_USERMODE_CLIENT(pClient)) {
GET_IRQL(dbgIrql);
TRACE(PATTERN, pClient, pBlob->arClientCtx[pClient->AssignedIndex], "ClientRemoveCfInfoComplete (4)");
UMCfInfoComplete( OP_REMOVE_CFINFO, pClient, pBlob, Status ); VERIFY_IRQL(dbgIrql);
TRACE(PATTERN, pClient, Status, "ClientRemoveCfInfoComplete (5)"); }
#endif
}
/*
************************************************************************
ClientRemoveCfInfo -
Arguments
Returns
************************************************************************ */ GPC_STATUS ClientRemoveCfInfo( IN PCLIENT_BLOCK pClient, IN PBLOB_BLOCK pBlob, IN GPC_CLIENT_HANDLE ClientCfInfoContext ) { GPC_STATUS Status = GPC_STATUS_SUCCESS; DEFINE_KIRQL(dbgIrql);
TRACE(PATTERN, pClient, pBlob, "ClientRemoveCfInfo");
if (pClient->State == GPC_STATE_READY) {
if (pClient->FuncList.ClRemoveCfInfoNotifyHandler) {
NdisInterlockedIncrement(&pBlob->ClientStatusCountDown);
GET_IRQL(dbgIrql);
TRACE(PATTERN, pClient, ClientCfInfoContext, "ClientRemoveCfInfo (2)");
Status = (pClient->FuncList.ClRemoveCfInfoNotifyHandler) (pClient->ClientCtx, ClientCfInfoContext );
VERIFY_IRQL(dbgIrql);
TRACE(PATTERN, pClient, Status, "ClientRemoveCfInfo (3)");
if (Status != GPC_STATUS_PENDING) { NdisInterlockedDecrement(&pBlob->ClientStatusCountDown); }
} else if (IS_USERMODE_CLIENT(pClient) //For client using IOCTL interface
&& !IS_USERMODE_CLIENT_EX(pClient)//But Not for new IOCTL interfaces
&& pClient == pBlob->pOwnerClient) {// If the client owns the Blob
GET_IRQL(dbgIrql);
TRACE(PATTERN, pClient, ClientCfInfoContext, "ClientRemoveCfInfo (4)");
//
// the notified client installed the blob
// and it is a user mode client
// we need to make a special call to dequeue a pending IRP
// and complete it with the specified data
//
UMClientRemoveCfInfoNotify(pClient, pBlob);
VERIFY_IRQL(dbgIrql);
TRACE(PATTERN, pClient, Status, "ClientRemoveCfInfo (5)"); } }
return Status; }
/*
************************************************************************
ClearPatternLinks -
Arguments
Returns
************************************************************************ */ VOID ClearPatternLinks( IN PPATTERN_BLOCK pPattern, IN PPROTOCOL_BLOCK pProtocol, IN ULONG CfIndex ) { PBLOB_BLOCK *ppBlob;
//
// Remove the pattern from the blob list
// and NULL the pointer to the blob in the pattern block
//
ppBlob = & GetBlobFromPattern(pPattern, CfIndex);
if (*ppBlob) {
NDIS_LOCK(&(*ppBlob)->Lock);
GpcRemoveEntryList(&pPattern->BlobLinkage[CfIndex]);
NDIS_UNLOCK(&(*ppBlob)->Lock);
*ppBlob = NULL;
}
}
/*
************************************************************************
ModifyCompleteClients -
Arguments
Returns
************************************************************************ */ VOID ModifyCompleteClients( IN PCLIENT_BLOCK pClient, IN PBLOB_BLOCK pBlob ) { uint i; PCLIENT_BLOCK pNotifyClient; KIRQL irql;
if (NT_SUCCESS(pBlob->LastStatus)) { //
// in case of successful completion, assign the
// new client data
//
NDIS_LOCK(&pBlob->Lock); WRITE_LOCK(&glData.ChLock, &irql);
GpcFreeMem(pBlob->pClientData, CfInfoDataTag); pBlob->ClientDataSize = pBlob->NewClientDataSize; pBlob->pClientData = pBlob->pNewClientData;
WRITE_UNLOCK(&glData.ChLock, irql); NDIS_UNLOCK(&pBlob->Lock);
} else { GpcFreeMem(pBlob->pNewClientData, CfInfoDataTag); }
pBlob->NewClientDataSize = 0; pBlob->pNewClientData = NULL; //
// notify each client (except the caller) that successfully
// modified the blob about the status.
// it can be SUCCESS or some failure
//
for (i = 0; i < MAX_CLIENTS_CTX_PER_BLOB; i++) { //
// only clients with none zero entries
// have been succefully modified the blob
//
if (pNotifyClient = pBlob->arpClientStatus[i]) { //
// complete here
//
ClientModifyCfInfoComplete( pNotifyClient, pBlob, pBlob->LastStatus ); //
// release the one we got earlier
//
//DereferenceClient(pNotifyClient);
} } // for
#if 0
//
// now, complete the call back to the calling client
//
ClientModifyCfInfoComplete( pBlob->pCallingClient, pBlob, pBlob->LastStatus ); #endif
pBlob->State = GPC_STATE_READY; }
#if 0
/*
************************************************************************
GetClassificationHandle -
Get the classification handle from the blob. If none is available creates a new one. '0' is not a valid CH!
Arguments pClient - The calling client pPattern - The pattern that refers to the CH
Returns A classification handle
************************************************************************ */ CLASSIFICATION_HANDLE GetClassificationHandle( IN PCLIENT_BLOCK pClient, IN PPATTERN_BLOCK pPattern ) { HFHandle hfh; ULONG t; PCLASSIFICATION_BLOCK pCB;
TRACE(CLASSIFY, pClient, pPattern, "GetClassificationHandle:");
if (TEST_BIT_ON(pPattern->Flags,PATTERN_SPECIFIC)) { //
// this is a specific pattern
//
//
// get the CH from it
//
ASSERT(pPattern->pClassificationBlock); hfh = pPattern->pClassificationBlock->ClassificationHandle;
//
// check if CH is valid
//
t = pPattern->ProtocolTemplate; pCB = (PCLASSIFICATION_BLOCK)dereference_HF_handle(glData.pCHTable, hfh); TRACE(CLASSHAND, pCB, hfh, "GetClassificationHandle (~)");
if (pCB != pPattern->pClassificationBlock) {
//
// the CH is invalid, release it and get a new one
//
NDIS_LOCK(&glData.Lock);
release_HF_handle(glData.pCHTable, hfh);
ProtocolStatInc(pPattern->ProtocolTemplate, RemovedCH); TRACE(CLASSHAND, glData.pCHTable, hfh, "GetClassificationHandle (-)");
hfh = assign_HF_handle(glData.pCHTable, (void *)pPattern->pClassificationBlock);
ProtocolStatInc(pPattern->ProtocolTemplate, InsertedCH);
NDIS_UNLOCK(&glData.Lock);
TRACE(CLASSIFY, pPattern, hfh, "GetClassificationHandle (+)");
pPattern->pClassificationBlock->ClassificationHandle = hfh; }
} else {
//
// this is a generic pattern
//
hfh = 0;
}
TRACE(CLASSIFY, pPattern, hfh, "GetClassificationHandle==>");
return (CLASSIFICATION_HANDLE)hfh; } #endif
/*
************************************************************************
FreeClassificationHandle -
Free the classification handle. It will invalidate the entry in the index table and make it avaiable for future use.
Arguments pClient - The calling client CH - The classification handle
Returns void
************************************************************************ */ VOID FreeClassificationHandle( IN PCLIENT_BLOCK pClient, IN CLASSIFICATION_HANDLE CH ) { KIRQL CHirql;
WRITE_LOCK(&glData.ChLock, &CHirql);
release_HF_handle(glData.pCHTable, CH);
TRACE(CLASSHAND, glData.pCHTable, CH, "FreeClassificationHandle");
WRITE_UNLOCK(&glData.ChLock, CHirql); }
GPC_STATUS CleanupBlobs( IN PCLIENT_BLOCK pClient ) { PBLOB_BLOCK pBlob; //PPATTERN_BLOCK pPattern;
//ULONG CfIndex = pClient->pCfBlock->AssignedIndex;
GPC_STATUS Status = GPC_STATUS_SUCCESS;
NDIS_LOCK(&pClient->Lock);
//
// scan blobs
//
while (!IsListEmpty(&pClient->BlobList)) {
//
// get the blob
//
pBlob = CONTAINING_RECORD(pClient->BlobList.Flink, BLOB_BLOCK, ClientLinkage);
pBlob->Flags |= PATTERN_REMOVE_CB_BLOB;
NDIS_UNLOCK(&pClient->Lock);
//
// remove the blob
//
Status = GpcRemoveCfInfo((GPC_HANDLE)pClient, (GPC_HANDLE)pBlob );
NDIS_LOCK(&pClient->Lock); }
NDIS_UNLOCK(&pClient->Lock);
return GPC_STATUS_SUCCESS; }
VOID CloseAllObjects( IN PFILE_OBJECT FileObject, IN PIRP Irp ) { PLIST_ENTRY pEntry0, pHead0; PLIST_ENTRY pEntry, pHead; PCLIENT_BLOCK pClient = NULL; PCF_BLOCK pCf; PPROTOCOL_BLOCK pProtocol; //int i,j;
//NTSTATUS NtStatus;
//QUEUED_COMPLETION QItem;
KIRQL irql;
NDIS_LOCK(&glData.Lock);
pHead0 = &glData.CfList; pEntry0 = pHead0->Flink;
while (pEntry0 != pHead0 && pClient == NULL) {
pCf = CONTAINING_RECORD(pEntry0, CF_BLOCK, Linkage); pEntry0 = pEntry0->Flink; RSC_READ_LOCK(&pCf->ClientSync, &irql);
pHead = &pCf->ClientList; pEntry = pHead->Flink; while (pEntry != pHead && pClient == NULL) {
pClient = CONTAINING_RECORD(pEntry,CLIENT_BLOCK,ClientLinkage); pEntry = pEntry->Flink;
if (pClient->pFileObject == FileObject) {
REFADD(&pClient->RefCount, 'CAOB');
} else {
pClient = NULL; } }
RSC_READ_UNLOCK(&pCf->ClientSync, irql); } // while (...)
NDIS_UNLOCK(&glData.Lock);
if (pClient) {
//
// clear all the blobs on the client
//
CleanupBlobs(pClient); //
// deregister the client
//
GpcDeregisterClient((GPC_HANDLE)pClient);
//
// release the previous ref count
//
REFDEL(&pClient->RefCount, 'CAOB');
}
}
// Cool new feature - Timer Wheels [ShreeM]
// We maintain N lists of patterns, one for each "timertick". The Pattern
// Expiry routine will clean up one of the lists every time it is invoked.
// It then makes a note to cleanup the next list on the wheel, the next time
// it is invoked.
// The Timer Wheels reduces spin lock contention between inserts and deletes.
VOID PatternTimerExpired( IN PVOID SystemSpecific1, IN PVOID FunctionContext, IN PVOID SystemSpecific2, IN PVOID SystemSpecific3 ) { PLIST_ENTRY pEntry; PPATTERN_BLOCK pPattern; ULONG CleanupWheelIndex = 0, NewWheelIndex; PPROTOCOL_BLOCK pProtocol = &glData.pProtocols[PtrToUlong(FunctionContext)];
TRACE(PAT_TIMER, FunctionContext, 0, "PatternTimerExpired");
DBGPRINT(PAT_TIMER, ("PatternTimerExpired: Timer expired, protocol=%d \n", PtrToUlong(FunctionContext)));
//
// Which of the timer wheels do we want to cleanup this time?
// Remember that we Increment the current index pointer into the wheels
// All the wheel index calculations are protected by gldata->lock.
NDIS_LOCK(&glData.Lock);
CleanupWheelIndex = pProtocol->CurrentWheelIndex; pProtocol->CurrentWheelIndex += 1; //
// Make sure we wrap around.
//
pProtocol->CurrentWheelIndex %= NUMBER_OF_WHEELS; NDIS_UNLOCK(&glData.Lock); NDIS_LOCK(&pProtocol->PatternTimerLock[CleanupWheelIndex]);
while (!IsListEmpty(&pProtocol->TimerPatternList[CleanupWheelIndex])) {
PCLIENT_BLOCK pClient;
pEntry = RemoveHeadList(&pProtocol->TimerPatternList[CleanupWheelIndex]);
pPattern = CONTAINING_RECORD(pEntry, PATTERN_BLOCK, TimerLinkage);
InitializeListHead(&pPattern->TimerLinkage);
NDIS_UNLOCK(&pProtocol->PatternTimerLock[CleanupWheelIndex]); ASSERT(pPattern->TimeToLive != 0xdeadbeef); ASSERT(TEST_BIT_ON( pPattern->Flags, PATTERN_AUTO ));
NDIS_LOCK(&pPattern->Lock); pPattern->Flags |= ~PATTERN_AUTO; NDIS_UNLOCK(&pPattern->Lock);
TRACE(PAT_TIMER, pPattern, pPattern->RefCount.Count, "PatternTimerExpired: del");
ProtocolStatInc(pPattern->ProtocolTemplate, DeletedAp);
ProtocolStatDec(pPattern->ProtocolTemplate, CurrentAp);
//
// actually remove the pattern
//
DBGPRINT(PAT_TIMER, ("PatternTimerExpired: removing pattern=%X, ref=%d, client=%X \n", pPattern, pPattern->RefCount, pPattern->pAutoClient));
pClient = pPattern->pAutoClient;
GpcRemovePattern((GPC_HANDLE)pPattern->pAutoClient, (GPC_HANDLE)pPattern);
REFDEL(&(pClient->RefCount), 'CLNT'); InterlockedDecrement(&pProtocol->AutoSpecificPatternCount);
NDIS_LOCK(&pProtocol->PatternTimerLock[CleanupWheelIndex]);
} NDIS_UNLOCK(&pProtocol->PatternTimerLock[CleanupWheelIndex]);
//
// If there are any Auto Specific patterns around restart the timer.
//
if(InterlockedExchangeAdd(&pProtocol->AutoSpecificPatternCount, 0) > 0) { NdisSetTimer(&pProtocol->PatternTimer, PATTERN_TIMEOUT); DBGPRINT(PAT_TIMER, ("PatternTimer restarted\n")); } else {
DBGPRINT(PAT_TIMER, ("Zero Auto Patterns - Timer NOT restarted\n"));
}
}
GPC_STATUS AddSpecificPatternWithTimer( IN PCLIENT_BLOCK pClient, IN ULONG ProtocolTemplate, IN PVOID PatternKey, OUT PPATTERN_BLOCK *ppPattern, OUT PCLASSIFICATION_HANDLE pClassificationHandle ) { GPC_STATUS Status=GPC_STATUS_SUCCESS; PPATTERN_BLOCK pPattern=NULL, pCreatedPattern=NULL; PPROTOCOL_BLOCK pProtocol = &glData.pProtocols[ProtocolTemplate]; UCHAR Mask[MAX_PATTERN_SIZE]; ULONG WheelIndex = 0; ULONG ulCurrentAutoPatternCount=0;
ulCurrentAutoPatternCount=InterlockedExchangeAdd(&pProtocol->AutoSpecificPatternCount,0); DBGPRINT(PAT_TIMER,("MSGPC:Creating New AutoPattern: Number of AutoPatterns" "Before Creation = %ul \n", ulCurrentAutoPatternCount));
if (ulCurrentAutoPatternCount > glData.AutoPatternLimit) { DBGPRINT(PAT_TIMER,("MSGPCHit AutoPatternLimit : Num AutoPatterns = %ul :" "Limit =%ul \n",ulCurrentAutoPatternCount, glData.AutoPatternLimit)); Status = GPC_STATUS_NO_MEMORY; goto exit; } TRACE(PAT_TIMER, pClient, PatternKey, "AddSpecificPatternWithTimer");
#if DBG
{ PGPC_IP_PATTERN pIp = (PGPC_IP_PATTERN)PatternKey; DBGPRINT(PAT_TIMER, ("AddSpecificPatternWithTimer: Client=%X \n", pClient)); #if INTERFACE_ID
DBGPRINT(PAT_TIMER, ("IP: ifc={%d,%d} src=%08X:%04x, dst=%08X:%04x, prot=%d rsv=%x,%x,%x\n", pIp->InterfaceId.InterfaceId, pIp->InterfaceId.LinkId, pIp->SrcAddr, pIp->gpcSrcPort, pIp->DstAddr, pIp->gpcDstPort, pIp->ProtocolId, pIp->Reserved[0], pIp->Reserved[1], pIp->Reserved[2] )); #endif
} #endif
RtlFillMemory(Mask, sizeof(Mask), 0xff);
pPattern = CreateNewPatternBlock(PATTERN_SPECIFIC); pCreatedPattern = pPattern; if (pPattern) { //
// setup the pattern fields and add it
//
//pPattern->RefCount++;
pPattern->Priority = 0; pPattern->ProtocolTemplate = ProtocolTemplate; pPattern->Flags |= PATTERN_AUTO; // This flag indicates that the auto pattern is not on
// timer list yet . This means
// the pattern is not ready yet though it may or may
// not have been inserted into the patternhashtable
pPattern->Flags |= PATTERN_AUTO_NOT_READY; pPattern->pAutoClient = pClient;
Status = AddSpecificPattern( pClient, PatternKey, Mask, NULL, pProtocol, &pPattern, // output pattern pointer
pClassificationHandle );
} else {
Status = GPC_STATUS_NO_MEMORY;
} if (NT_SUCCESS(Status)) { //
// we didn't get an already existing pattern
//
//ASSERT(*pClassificationHandle);
*ppPattern = pPattern; // Figure out which wheel to stick this pattern on.
NDIS_LOCK(&glData.Lock); WheelIndex = pProtocol->CurrentWheelIndex; NDIS_UNLOCK(&glData.Lock); WheelIndex += (NUMBER_OF_WHEELS -1); WheelIndex %= NUMBER_OF_WHEELS;
//
// we must lock this pattern since we look at the timer linkage
//
NDIS_LOCK(&pPattern->Lock);
//
// set the AUTO flag again, since we might have got
// a pattern that already exist
//
pPattern->Flags |= PATTERN_AUTO; pPattern->pAutoClient = pClient; pPattern->WheelIndex = WheelIndex;
//
// this pattern has not been on any the timer list yet
//
if (IsListEmpty(&pPattern->TimerLinkage)) {
//
// We need to insert this in the TimerWheel which is (NUMBER_OF_WHEELS - 1)
// away from the current, so that it spends enough time on the list.
//
NDIS_DPR_LOCK(&pProtocol->PatternTimerLock[WheelIndex]); //
// If the AutoSpecificPatternCount was zero earlier, then we need
// to a) start the timer and b) increment this count.
//
if (1 == InterlockedIncrement(&pProtocol->AutoSpecificPatternCount)) { //
// restart the timer for the first auto pattern
//
NdisSetTimer(&pProtocol->PatternTimer, PATTERN_TIMEOUT); TRACE(PAT_TIMER, pPattern, PATTERN_TIMEOUT, "Starting Pattern Timer\n AddSpecificPatternWithTimer: (1)"); } GpcInsertHeadList(&pProtocol->TimerPatternList[WheelIndex], &pPattern->TimerLinkage);
//
// don't refer to pPattern after it has been placed on the timer list
// since the timer may expire any time and remove it from there...
//
NDIS_DPR_UNLOCK(&pProtocol->PatternTimerLock[WheelIndex]);
}
//
// This is a specific pattern, so lets increment the count [ShreeM].
InterlockedIncrement(&pProtocol->SpecificPatternCount); // Pattern is now ready. It has been inserted
// into the hash table and has been put into the
// timer list.
pPattern->Flags &= ~PATTERN_AUTO_NOT_READY;
NDIS_UNLOCK(&pPattern->Lock); ProtocolStatInc(ProtocolTemplate, CreatedAp); ProtocolStatInc(ProtocolTemplate, CurrentAp); } else { goto exit; }
exit: if (!NT_SUCCESS(Status)){ *ppPattern = NULL; *pClassificationHandle = 0; ProtocolStatInc(ProtocolTemplate, RejectedAp); } else{ REFADD(&pClient->RefCount, 'CLNT'); } if (pPattern) {
//
// release the reference count to this pattern
// in case of an error, this will also release
// the data block
//
REFDEL(&pCreatedPattern->RefCount, 'FILT'); }
TRACE(PAT_TIMER, pPattern, Status, "AddSpecificPatternWithTimer==>"); DBGPRINT(CLASSIFY, ("AddSpecificPatternWithTimer: pClient=%X Pattern=%X Status=%X\n", pClient, pPattern, Status)); return Status; }
// SS202 CR
//
NTSTATUS InitPatternTimer( IN ULONG ProtocolTemplate ) { ULONG i=0;
//
// We increase the granularity of when a "Automatic" Pattern gets cleaned
// out by using timer wheels, but they are more efficient for inserting and
// removing (in terms of locks).
//
for (i = 0; i < NUMBER_OF_WHEELS; i++) { NDIS_INIT_LOCK(&glData.pProtocols[ProtocolTemplate].PatternTimerLock[i]);
InitializeListHead(&glData.pProtocols[ProtocolTemplate].TimerPatternList[i]);
}
glData.pProtocols[ProtocolTemplate].CurrentWheelIndex = 0;
NdisInitializeTimer(&glData.pProtocols[ProtocolTemplate].PatternTimer, PatternTimerExpired, (PVOID) ULongToPtr(ProtocolTemplate));
return STATUS_SUCCESS; }
//
// Some CRT and RTL functions that cant be used at DISPATHC_LEVEL IRQL are being
// cut/paste here.
//
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; }
/*++
Routine Description:
Reads a REG_DWORD value from registry and init the variable passed in..
Arguments:
- Open handle to the parent key of the value to read. param - 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.
--*/ VOID GPC_REG_READ_DWORD(HANDLE hRegKey, PWCHAR pwcName, PULONG pulData, ULONG ulDefault,ULONG ulMax, ULONG ulMin) { NTSTATUS status; status = GetRegDWORDValue(hRegKey, pwcName, (pulData)); if (!NT_SUCCESS(status)) { *(pulData) = ulDefault; } else if (*(pulData) > ulMax) { *(pulData) = ulMax; } else if (*(pulData) < ulMin) { *(pulData) = ulMin; } }
VOID GpcReadRegistry() /*++
Routine Description:
Reads config info from registry into glData
Arguments:
Return Value:
status of the read.
--*/ { NTSTATUS status; HANDLE hRegKey; WCHAR GpcParametersRegistryKey[] = GPC_REG_KEY; glData.SystemSizeHint = MmQuerySystemSize(); switch (glData.SystemSizeHint) { case MmSmallSystem: glData.AutoPatternLimit = DEFAULT_SMALL_SYSTEM_AUTO_PATTERN_LIMIT; break; case MmMediumSystem: glData.AutoPatternLimit = DEFAULT_MEDIUM_SYSTEM_AUTO_PATTERN_LIMIT; break; case MmLargeSystem: glData.AutoPatternLimit = DEFAULT_LARGE_SYSTEM_AUTO_PATTERN_LIMIT; break; default: break; }
status = OpenRegKey(&hRegKey, GpcParametersRegistryKey);
if (NT_SUCCESS(status)) { //
// Expected configuration values. We use reasonable defaults if they
// aren't available for some reason.
//
GPC_REG_READ_DWORD( hRegKey, GPC_REG_AUTO_PATTERN_LIMIT, &glData.AutoPatternLimit, glData.AutoPatternLimit, GPC_AUTO_PATTERN_MAX, GPC_AUTO_PATTERN_MIN); } ZwClose(hRegKey); }
|