Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

1137 lines
40 KiB

#include <precomp.h>
#include "rpcsrv.h"
#include "utils.h"
#include "intflist.h"
#include "deviceio.h"
#include "wzcsvc.h"
#include "notify.h"
#include "storage.h"
#include "tracing.h"
//taroonm
#include <wifipol.h>
#define WZEROCONF_SERVICE TEXT("wzcsvc")
#define EAPOL_LINKED
SERVICE_STATUS g_WZCSvcStatus;
SERVICE_STATUS_HANDLE g_WZCSvcStatusHandle = NULL;
HDEVNOTIFY g_WZCSvcDeviceNotif = NULL;
UINT g_nThreads = 0;
HINSTANCE g_hInstance = NULL;
//context of users preferences
WZC_INTERNAL_CONTEXT g_wzcInternalCtxt = {0};
BOOL WINAPI DllMain(
HINSTANCE hinstDLL, // handle to the DLL module
DWORD fdwReason, // reason for calling function
LPVOID lpvReserved // reserved
)
{
if (g_hInstance == NULL)
g_hInstance = hinstDLL;
return TRUE;
}
//-----------------------------------------------------------
VOID WINAPI
WZCSvcMain(
IN DWORD dwArgc,
IN LPWSTR *lpwszArgv)
{
DWORD dwError = ERROR_SUCCESS;
DEV_BROADCAST_DEVICEINTERFACE PnPFilter;
BOOL bLogEnabled = FALSE;
// Initialize the workstation to receive service requests
// by registering the service control handler.
g_WZCSvcStatusHandle = RegisterServiceCtrlHandlerEx(
WZEROCONF_SERVICE,
WZCSvcControlHandler,
NULL);
if (g_WZCSvcStatusHandle == (SERVICE_STATUS_HANDLE)NULL)
return;
// this is the first thread to run
InterlockedIncrement(&g_nThreads);
// Initialize all the status fields so that the subsequent calls
// to SetServiceStatus need to only update fields that changed.
g_WZCSvcStatus.dwServiceType = SERVICE_WIN32_SHARE_PROCESS;
g_WZCSvcStatus.dwCurrentState = SERVICE_START_PENDING;
g_WZCSvcStatus.dwControlsAccepted = 0;
g_WZCSvcStatus.dwCheckPoint = 1;
g_WZCSvcStatus.dwWaitHint = 4000;
g_WZCSvcStatus.dwWin32ExitCode = ERROR_SUCCESS;
g_WZCSvcStatus.dwServiceSpecificExitCode = 0;
// update status to START_PENDING
WZCSvcUpdateStatus();
// Initialize Tracing
TrcInitialize();
DbgPrint((TRC_TRACK,"**** [WZCSvcMain - Service Start Pending"));
// Initialize global hashes. If this fails it means the most important
// critical section failed to initialize - no point in going further.
dwError = HshInitialize(&g_hshHandles);
if (dwError != ERROR_SUCCESS)
goto exit;
dwError = LstInitIntfHashes();
if (dwError != ERROR_SUCCESS)
goto exit;
//Initialize the service's context
dwError = WZCContextInit(&g_wzcInternalCtxt);
if (dwError != ERROR_SUCCESS)
goto exit;
// TODO: the block below should be moved to a function responsible for
// loading the whole g_wzcInternalCtxt from the persistent storage
{
// load the service's context from the registry
dwError = StoLoadWZCContext(NULL, &(g_wzcInternalCtxt.wzcContext));
if (ERROR_SUCCESS != dwError)
goto exit;
// load the global interface template from the registry.
dwError = StoLoadIntfConfig(NULL, g_wzcInternalCtxt.pIntfTemplate);
DbgAssert((dwError == ERROR_SUCCESS,"Err %d loading the template interface from the registry"));
}
// Open log database if logging enabled
EnterCriticalSection(&g_wzcInternalCtxt.csContext);
bLogEnabled = ((g_wzcInternalCtxt.wzcContext.dwFlags & WZC_CTXT_LOGGING_ON) != 0);
LeaveCriticalSection(&g_wzcInternalCtxt.csContext);
if (bLogEnabled == TRUE)
{
dwError = InitWZCDbGlobals();
if ((INT)dwError < 0)
dwError = ERROR_DATABASE_FAILURE;
}
if (ERROR_SUCCESS != dwError)
goto exit;
dwError = LstInitTimerQueue();
if (dwError != ERROR_SUCCESS)
goto exit;
#ifdef EAPOL_LINKED
// Start EAPOL/802.1X
EAPOLServiceMain (dwArgc, NULL);
#endif
// load the interfaces list
dwError = LstLoadInterfaces();
DbgAssert((dwError == ERROR_SUCCESS,"LstLoadInterfaces failed with error %d", dwError));
// register for service control notifications
ZeroMemory (&PnPFilter, sizeof(PnPFilter));
PnPFilter.dbcc_size = sizeof(PnPFilter);
PnPFilter.dbcc_devicetype = DBT_DEVTYP_DEVICEINTERFACE;
PnPFilter.dbcc_classguid = GUID_NDIS_LAN_CLASS;
// NOTE: EAPOL service is only working with ANSI strings, hence the ANSI calls
g_WZCSvcDeviceNotif = RegisterDeviceNotificationA(
(HANDLE)g_WZCSvcStatusHandle,
&PnPFilter,
DEVICE_NOTIFY_SERVICE_HANDLE );
DbgAssert((g_WZCSvcDeviceNotif != (HDEVNOTIFY) NULL,
"Registering for device notifications failed with error %d", GetLastError));
// register with WMI for device notifications
dwError = WZCSvcWMINotification(TRUE);
DbgAssert((dwError == ERROR_SUCCESS,"WZCSvcRegisterWMINotif failed with error %d", dwError));
// Start the RPC Server.
dwError = WZCSvcStartRPCServer();
DbgAssert((dwError == ERROR_SUCCESS,"WZCStartRPCServer failed with error %d", dwError));
//taroonM: Policy Engine Init
dwError = InitPolicyEngine(dwPolicyEngineParam, &hPolicyEngineThread);
DbgAssert((dwError == ERROR_SUCCESS,"InitPolicyEngine failed with error %d", dwError));
exit:
if (dwError == ERROR_SUCCESS)
{
g_WZCSvcStatus.dwCurrentState = SERVICE_RUNNING;
g_WZCSvcStatus.dwControlsAccepted = SERVICE_ACCEPT_STOP | SERVICE_ACCEPT_SHUTDOWN | SERVICE_ACCEPT_SESSIONCHANGE;
g_WZCSvcStatus.dwCheckPoint = 0;
g_WZCSvcStatus.dwWaitHint = 0;
// update status to RUNNING
WZCSvcUpdateStatus();
DbgPrint((TRC_TRACK,"**** WZCSvcMain - Service Running"));
DbLogWzcInfo(WZCSVC_SERVICE_STARTED, NULL);
}
else
{
DbgPrint((TRC_TRACK,"**** WZCSvcMain - Service Failed to start"));
// stop the WZC engine
WZCSvcShutdown(dwError);
// stop the EAP engine
EAPOLCleanUp (dwError);
// destroy the handles hash
HshDestroy(&g_hshHandles);
// if database has been opened, close it here since there is no one else using it
DeInitWZCDbGlobals();
// finaly destroy the WZC context
WZCContextDestroy(&g_wzcInternalCtxt);
TrcTerminate();
// if we did successfully register with SCM, then indicate the service has stopped
if (g_WZCSvcStatusHandle != (SERVICE_STATUS_HANDLE)NULL)
{
g_WZCSvcStatus.dwCurrentState = SERVICE_STOPPED;
g_WZCSvcStatus.dwControlsAccepted = 0;
g_WZCSvcStatus.dwCheckPoint = 0;
g_WZCSvcStatus.dwWaitHint = 0;
g_WZCSvcStatus.dwWin32ExitCode = ERROR_SERVICE_SPECIFIC_ERROR;
g_WZCSvcStatus.dwServiceSpecificExitCode = dwError;
WZCSvcUpdateStatus();
}
}
InterlockedDecrement(&g_nThreads);
return;
}
// global array for the GUIDs for the WMI notifications
LPGUID g_lpGuidWmiNotif[] = {
(LPGUID)&GUID_NDIS_NOTIFY_BIND,
(LPGUID)&GUID_NDIS_NOTIFY_UNBIND,
(LPGUID)&GUID_NDIS_STATUS_MEDIA_CONNECT,
(LPGUID)&GUID_NDIS_STATUS_MEDIA_DISCONNECT
};
//-----------------------------------------------------------
// Handles all WMI registration and de-registration
DWORD
WZCSvcWMINotification(BOOL bRegister)
{
DWORD dwErr = ERROR_SUCCESS;
INT nIdx;
DbgPrint((TRC_TRACK,"[WZCSvcWMINotification(%d)", bRegister));
// do the requested action - registration / deregistration
// if registering for notifications, break the loop first time a registration failed
// if de-registering, ignore the errors and go on deregistering the remaining notifications
for (nIdx = 0;
nIdx < sizeof(g_lpGuidWmiNotif)/sizeof(LPGUID) && (!bRegister || dwErr == ERROR_SUCCESS);
nIdx++)
{
dwErr = WmiNotificationRegistrationW(
g_lpGuidWmiNotif[nIdx],
(BOOLEAN)bRegister,
(PVOID)WZCSvcWMINotificationHandler,
(ULONG_PTR)NULL,
NOTIFICATION_CALLBACK_DIRECT);
DbgAssert((dwErr == 0, "Failed to %s notif index %d",
bRegister ? "register" : "de-register",
nIdx));
}
// in case a registration was requested and one of the Wmi calls above failed,
// rollback the action by deregistering for whatever was registered successfully
if (bRegister && dwErr != ERROR_SUCCESS)
{
DbgPrint((TRC_GENERIC,"Rollback WmiNotif for %d Guids", nIdx));
for(nIdx--; nIdx>=0; nIdx--)
{
WmiNotificationRegistrationW(
g_lpGuidWmiNotif[nIdx],
(BOOLEAN)FALSE,
(PVOID)WZCSvcWMINotificationHandler,
(ULONG_PTR)NULL,
NOTIFICATION_CALLBACK_DIRECT);
}
}
DbgPrint((TRC_TRACK, "WZCSvcWMINotification]=%d", dwErr));
return dwErr;
}
//-----------------------------------------------------------
DWORD
WZCSvcUpdateStatus()
{
DbgPrint((TRC_TRACK,"[WZCSvcUpdateStatus(%d)]", g_WZCSvcStatus.dwCurrentState));
return SetServiceStatus(g_WZCSvcStatusHandle, &g_WZCSvcStatus) ?
ERROR_SUCCESS : GetLastError();
}
//-----------------------------------------------------------
DWORD
WZCSvcControlHandler(
IN DWORD dwControl,
IN DWORD dwEventType,
IN PVOID pEventData,
IN PVOID pContext)
{
DWORD dwRetCode = NO_ERROR;
BOOL bDecrement = TRUE;
InterlockedIncrement(&g_nThreads);
DbgPrint((TRC_TRACK|TRC_NOTIF,"[WZCSvcControlHandler(%d,%d,..)", dwControl, dwEventType));
DbgPrint((TRC_NOTIF,"SCM Notification: Control=0x%x; EventType=0x%04x", dwControl, dwEventType));
switch (dwControl)
{
case SERVICE_CONTROL_DEVICEEVENT:
if (g_WZCSvcStatus.dwCurrentState == SERVICE_RUNNING &&
pEventData != NULL)
{
PDEV_BROADCAST_DEVICEINTERFACE pInfo = (DEV_BROADCAST_DEVICEINTERFACE *)pEventData;
if (pInfo->dbcc_devicetype == DBT_DEVTYP_DEVICEINTERFACE)
{
PWZC_DEVICE_NOTIF pDevNotif;
pDevNotif = MemCAlloc(FIELD_OFFSET(WZC_DEVICE_NOTIF, dbDeviceIntf) + pInfo->dbcc_size);
DbgAssert((pDevNotif != NULL, "Not enough memory?"));
if (pDevNotif != NULL)
{
// build up the notification information
switch(dwEventType)
{
case DBT_DEVICEARRIVAL:
pDevNotif->dwEventType = WZCNOTIF_DEVICE_ARRIVAL;
break;
case DBT_DEVICEREMOVECOMPLETE:
pDevNotif->dwEventType = WZCNOTIF_DEVICE_REMOVAL;
break;
default:
pDevNotif->dwEventType = WZCNOTIF_UNKNOWN;
DbgPrint((TRC_ERR,"SCM Notification %d is not recognized", dwEventType));
break;
}
// pass down notification only if it is recognized at this level
if (pDevNotif->dwEventType != WZCNOTIF_UNKNOWN)
{
memcpy(&(pDevNotif->dbDeviceIntf), pInfo, pInfo->dbcc_size);
// pDevNotif will be MemFree-ed by the worker thread
if (QueueUserWorkItem(
(LPTHREAD_START_ROUTINE)WZCWrkDeviceNotifHandler,
(LPVOID)pDevNotif,
WT_EXECUTELONGFUNCTION))
{
bDecrement = FALSE;
}
}
else
{
MemFree(pDevNotif);
}
}
}
}
break;
case SERVICE_CONTROL_STOP:
case SERVICE_CONTROL_SHUTDOWN:
// make sure the control is not sent twice forcing the service to lock
// a critical section already destroyed! (see WZCSvcShutdown->StoSaveConfig)
if (g_WZCSvcStatus.dwCurrentState != SERVICE_STOPPED)
{
g_WZCSvcStatus.dwCurrentState = SERVICE_STOP_PENDING;
g_WZCSvcStatus.dwCheckPoint = 1;
g_WZCSvcStatus.dwWaitHint = 60000;
WZCSvcUpdateStatus();
// Shutdown Zero Conf
WZCSvcShutdown(ERROR_SUCCESS);
// Shutdown EAPOL/802.1X
EAPOLCleanUp (NO_ERROR);
DbgPrint((TRC_TRACK,"EAPOLCleanUp done!"));
// destroy the handles hash
HshDestroy(&g_hshHandles);
DbgPrint((TRC_TRACK,"Hashes Destroyed!"));
DeInitWZCDbGlobals();
// clean up the service's context
WZCContextDestroy(&g_wzcInternalCtxt);
TrcTerminate();
//WZCSvcUpdateStatus();
g_WZCSvcStatus.dwCurrentState = SERVICE_STOPPED;
g_WZCSvcStatus.dwControlsAccepted = 0;
g_WZCSvcStatus.dwCheckPoint = 0;
g_WZCSvcStatus.dwWaitHint = 0;
g_WZCSvcStatus.dwWin32ExitCode = ERROR_SUCCESS;
g_WZCSvcStatus.dwServiceSpecificExitCode = 0;
WZCSvcUpdateStatus();
// this is the last thread of the service (guaranteed by WZCSvcShutdown)
// all datastructures have been closed, tracing has been disabled. No need
// to decrement the thread counter since no one uses it anymore.
// just set it to 0 to play safe
bDecrement = FALSE;
g_nThreads = 0;
}
break;
case SERVICE_CONTROL_SESSIONCHANGE:
// 802.1X session change handler
ElSessionChangeHandler (
pEventData,
dwEventType
);
break;
}
if (bDecrement)
{
DbgPrint((TRC_TRACK|TRC_NOTIF,"WZCSvcControlHandler]"));
InterlockedDecrement(&g_nThreads);
}
return ERROR_SUCCESS;
}
//-----------------------------------------------------------
// WZCSvcWMINotificationHandler: Callback function called by WMI on any registered
// notification (as of 01/19/01: bind/unbind/connect/disconnect)
VOID CALLBACK
WZCSvcWMINotificationHandler(
IN PWNODE_HEADER pWnodeHdr,
IN UINT_PTR uiNotificationContext)
{
DWORD dwErr = ERROR_SUCCESS;
PWZC_DEVICE_NOTIF pDevNotif = NULL;
PWNODE_SINGLE_INSTANCE pWnode = (PWNODE_SINGLE_INSTANCE)pWnodeHdr;
LPWSTR wszTransport;
BOOL bDecrement = TRUE;
// increment the thread counter
InterlockedIncrement(&g_nThreads);
DbgPrint((TRC_TRACK|TRC_NOTIF, "[WZCSvcWMIHandler(0x%p)", pWnodeHdr));
// check if the service is still running, otherwise ignore
// the notification
if (g_WZCSvcStatus.dwCurrentState != SERVICE_RUNNING)
goto exit;
// check if we have valid notification data from WMI
if (pWnodeHdr == NULL)
{
dwErr = ERROR_INVALID_PARAMETER;
goto exit;
}
// allocate memory for the WZC Notification Structure.
pDevNotif = MemCAlloc(FIELD_OFFSET(WZC_DEVICE_NOTIF, wmiNodeHdr) + pWnodeHdr->BufferSize);
if (pDevNotif == NULL)
{
dwErr = GetLastError();
goto exit;
}
// translate the specific WMI notification code to WZC notification
else if (!memcmp( &(pWnodeHdr->Guid), &GUID_NDIS_STATUS_MEDIA_CONNECT, sizeof(GUID)))
pDevNotif->dwEventType = WZCNOTIF_MEDIA_CONNECT;
else if (!memcmp( &(pWnodeHdr->Guid), &GUID_NDIS_STATUS_MEDIA_DISCONNECT, sizeof(GUID)))
pDevNotif->dwEventType = WZCNOTIF_MEDIA_DISCONNECT;
else if (!memcmp( &(pWnodeHdr->Guid), &GUID_NDIS_NOTIFY_BIND, sizeof(GUID)))
pDevNotif->dwEventType = WZCNOTIF_ADAPTER_BIND;
else if (!memcmp( &(pWnodeHdr->Guid), &GUID_NDIS_NOTIFY_UNBIND, sizeof(GUID)))
pDevNotif->dwEventType = WZCNOTIF_ADAPTER_UNBIND;
else
{
pDevNotif->dwEventType = WZCNOTIF_UNKNOWN;
DbgPrint((TRC_ERR,"WMI Notification GUID is not recognized"));
goto exit;
}
// copy the WMI notification data into the local buffer
memcpy(&(pDevNotif->wmiNodeHdr), pWnodeHdr, pWnodeHdr->BufferSize);
// pDevNotif will be MemFree-ed by the worker thread
if (!QueueUserWorkItem(
(LPTHREAD_START_ROUTINE)WZCWrkDeviceNotifHandler,
(LPVOID)pDevNotif,
WT_EXECUTELONGFUNCTION))
{
dwErr = GetLastError();
goto exit;
}
bDecrement = FALSE;
// since the working thread has been created successfully, the notification
// structure allocated above will be free-ed by the thread. Set the local
// pointer to NULL to prevent the memory from being free-ed prematurely
// (MemFree does nothing when pointer is NULL)
pDevNotif = NULL;
exit:
MemFree(pDevNotif);
DbgPrint((TRC_TRACK|TRC_NOTIF, "WZCSvcWMIHandler=%d]", dwErr));
if (bDecrement)
InterlockedDecrement(&g_nThreads);
}
//-----------------------------------------------------------
VOID
WZCSvcShutdown(IN DWORD dwErrorCode)
{
DbgPrint((TRC_TRACK,"[WZCSvcShutdown(%d)", dwErrorCode));
//taroonm:
TerminatePolicyEngine(hPolicyEngineThread);
if (g_WZCSvcDeviceNotif != NULL && !UnregisterDeviceNotification(g_WZCSvcDeviceNotif))
{
DbgPrint((TRC_ERR,"Err: UnregisterDeviceNotification->%d", GetLastError()));
}
// reset the notification handler since it has already been unregistered
g_WZCSvcDeviceNotif = NULL;
// unregister from WMI
WZCSvcWMINotification(FALSE);
// stop first the RPC server
WZCSvcStopRPCServer();
// all the notification registrations have been removed.
// Block here until all worker/rpc threads terminate.
DbgPrint((TRC_SYNC,"Waiting for %d thread(s) to terminate", g_nThreads));
while(g_nThreads != 1)
{
Sleep(1000);
DbgPrint((TRC_SYNC,"Waiting for %d more thread(s).", g_nThreads));
}
// save the configuration to the registry
StoSaveConfig();
// destroy all the hashes related to the list of interfaces
// (included the list itself)
LstDestroyIntfHashes();
// destroy the timer queue. At this point there should be no timer queued so
// there is no point in waiting for any completion
LstDestroyTimerQueue();
DbgPrint((TRC_TRACK,"WZCSvcShutdown]"));
}
//-----------------------------------------------------------
VOID
WZCWrkDeviceNotifHandler(
IN LPVOID pvData)
{
DWORD dwErr = ERROR_SUCCESS;
PWZC_DEVICE_NOTIF pDevNotif = (PWZC_DEVICE_NOTIF)pvData;
PWNODE_SINGLE_INSTANCE pWnode = (PWNODE_SINGLE_INSTANCE)&(pDevNotif->wmiNodeHdr);
LPBYTE pbDeviceKey = NULL; // pointer in the notification data where the dev key is
UINT nDeviceKeyLen; // number of bytes for the dev key
LPWSTR pwszDeviceKey = NULL;
PINTF_CONTEXT pIntfContext = NULL;
PHASH_NODE *ppHash = NULL; // reference to the hash to use for getting the INTF_CONTEXT
PHASH_NODE pHashNode = NULL; // interface's node in the hash
BOOL bForwardUp = FALSE; // specifies whether the notification needs to passed to the upper layers
DbgPrint((TRC_TRACK,"[WZCWrkDeviceNotifHandler(wzcnotif %d)", pDevNotif->dwEventType));
DbgAssert((pDevNotif != NULL, "(null) device notification info!"));
// don't do one single bit of work if the service is not in the running state
if (g_WZCSvcStatus.dwCurrentState != SERVICE_RUNNING)
goto exit;
// get the Device Key information for the event (either the GUID or the Description)
switch(pDevNotif->dwEventType)
{
case WZCNOTIF_DEVICE_REMOVAL:
case WZCNOTIF_ADAPTER_UNBIND:
// do forward up device removal or unbind no matter what
bForwardUp = TRUE;
// no break.
case WZCNOTIF_DEVICE_ARRIVAL:
case WZCNOTIF_ADAPTER_BIND:
{
LPBYTE pbDeviceKeyEnd = NULL;
if (pDevNotif->dwEventType == WZCNOTIF_DEVICE_ARRIVAL ||
pDevNotif->dwEventType == WZCNOTIF_DEVICE_REMOVAL)
{
// if the notification comes from SCM, get the pointer to the "\device\{guid}"
// string from the dbcc_name field
pbDeviceKey = (LPBYTE)(pDevNotif->dbDeviceIntf.dbcc_name);
}
else
{
// check if the notification refers to the NDISUIO transport
pbDeviceKey = RtlOffsetToPointer(pWnode, pWnode->DataBlockOffset);
// if this is not a notification for the NDISUIO transport, ignore it.
if (wcsncmp ((LPWSTR)pbDeviceKey, L"NDISUIO", 7))
{
DbgPrint((TRC_NOTIF,"Ignore WMI Notif %d for Transport %S",
pDevNotif->dwEventType,
pbDeviceKey));
goto exit;
}
// get first the pointer to the Transport Name
pbDeviceKey = RtlOffsetToPointer(pWnode, pWnode->DataBlockOffset);
// skip to the "\device\{guid}" string
pbDeviceKey += (wcslen((LPWSTR)pbDeviceKey) + 1) * sizeof(WCHAR);
}
// build now the actual "{guid}" string from the L"\DEVICE\{guid}" string
// pointed by wszGuid
pbDeviceKey = (LPBYTE)wcsrchr( (LPWSTR)pbDeviceKey, L'{' );
if (pbDeviceKey != NULL)
pbDeviceKeyEnd = (LPBYTE)wcsrchr( (LPWSTR)pbDeviceKey, L'}' );
if (pbDeviceKey == NULL || pbDeviceKeyEnd == NULL)
{
DbgPrint((TRC_ERR,"Err: Mal-formed dbcc_name"));
goto exit;
}
// include the closing curved bracket in the GUID string
pbDeviceKeyEnd += sizeof(WCHAR);
nDeviceKeyLen = (UINT)(pbDeviceKeyEnd - pbDeviceKey);
// get the reference to the GUID hash. This will be used in order to locate
// the interface context. This reference is guaranteed to exist since it is a static
// global variable.
ppHash = &g_lstIntfHashes.pHnGUID;
break;
}
case WZCNOTIF_MEDIA_DISCONNECT:
case WZCNOTIF_MEDIA_CONNECT:
{
LPBYTE pbDeviceKeyEnd = NULL;
// disconnect should be forwarded up no matter what
// for connects, we assume we'll forward it up. After dispatching the event
// to the state machine, if this resulted in a WZC notification we'll block
// the forwarding.
bForwardUp = TRUE;
// for MEDIA_CONNECT / DISCONNECT events, we also get the adapter's GUID
pbDeviceKey = RtlOffsetToPointer(pWnode, pWnode->DataBlockOffset);
// build now the actual "{guid}" string from the L"\DEVICE\{guid}" string
// pointed by wszGuid
pbDeviceKey = (LPBYTE)wcsrchr( (LPWSTR)pbDeviceKey, L'{' );
if (pbDeviceKey != NULL)
pbDeviceKeyEnd = (LPBYTE)wcsrchr( (LPWSTR)pbDeviceKey,L'}' );
if (pbDeviceKey == NULL || pbDeviceKeyEnd == NULL)
{
DbgPrint((TRC_ERR,"Err: Mal-formed device name"));
goto exit;
}
pbDeviceKeyEnd += sizeof(WCHAR);
nDeviceKeyLen = (UINT)(pbDeviceKeyEnd - pbDeviceKey);
// get the reference to the GUID hash. This will be used in order to locate
// the interface context. This reference is guaranteed to exist since it is a static
// global variable.
ppHash = &g_lstIntfHashes.pHnGUID;
break;
}
// no need to specify "default:" as the event type has been already filtered
// out to one of the valid events
}
// get memory for GUID (add space for the null terminator)
pwszDeviceKey = (LPWSTR)MemCAlloc(nDeviceKeyLen + sizeof(WCHAR));
if (pwszDeviceKey == NULL)
{
dwErr = GetLastError();
goto exit;
}
// copy the GUID string in the key (because of the CAlloc, the '\0' is already there)
memcpy(pwszDeviceKey, pbDeviceKey, nDeviceKeyLen);
// locate now the INTF_CONTEXT structure related to this notification
// and keep the lock on the hashes all the time (since the PnP event mostly sure
// results in removing / adding an interface context which means altering the
// hash)
EnterCriticalSection(&(g_lstIntfHashes.csMutex));
dwErr = HshQueryObjectRef(
*ppHash,
pwszDeviceKey,
&pHashNode);
if (dwErr == ERROR_SUCCESS)
{
pIntfContext = (PINTF_CONTEXT)pHashNode->pObject;
// at this point we know the notification type, the key info and
// the INTERFACE_CONTEXT object (if any) for the device.
DbgPrint((TRC_NOTIF,"WZCNotif %d for Device Key \"%S\". Context=0x%p",
pDevNotif->dwEventType,
pwszDeviceKey,
pIntfContext));
}
// forward now the notification for processing
dwErr = LstNotificationHandler(&pIntfContext, pDevNotif->dwEventType, pwszDeviceKey);
// At this point, bForwardUp is FALSE only for ADAPTER_BIND or DEVICE_ARRIVAL
// If this is a new adapter, but not a wireless one, pass up the notification.
if (pDevNotif->dwEventType == WZCNOTIF_ADAPTER_BIND || pDevNotif->dwEventType == WZCNOTIF_DEVICE_ARRIVAL)
bForwardUp = bForwardUp || (dwErr == ERROR_MEDIA_INCOMPATIBLE);
// At this point, bForwardUp is FALSE only for ADAPTER_BIND or DEVICE_ARRIVAL for a wireless adapter
// If WZC doesn't seem to be enabled for this adapter, pass up the notification
if (dwErr == ERROR_SUCCESS && pIntfContext != NULL)
bForwardUp = bForwardUp || ((pIntfContext->dwCtlFlags & INTFCTL_ENABLED) == 0);
// At this point, bForwardUp is FALSE only for ADAPTER_BIND or DEVICE_ARRIVAL for a wireless adapter
// on which WZC is enabled. For all the other cases, notification is passed through.
// All that remains is to block the pass-through of the notification if we deal with a valid context
// for which INTFCTL_INTERNAL_BLK_MEDIACONN bit is set. This bit is being set in StateNotifyFn
// and it is reset after processing each event / command so it is only a narrow case where we really
// need to block. We do block it because one WZC notification must have already been sent up so
// there is totally redundant to pass another one up.
if (pIntfContext != NULL)
{
bForwardUp = bForwardUp && !(pIntfContext->dwCtlFlags & INTFCTL_INTERNAL_BLK_MEDIACONN);
// now, since we determined whether we need or not to pass up the PnP notification
// we can (and must) clear up the INTFCTL_INTERNAL_BLK_MEDIACONN bit.
pIntfContext->dwCtlFlags &= ~INTFCTL_INTERNAL_BLK_MEDIACONN;
}
// for all the remaining cases, the notification is going to be substituted
// later (in {SN} state) with a "WZCNOTIF_WZC_CONNECT" notification. Assume something goes wrong
// and the interface never gets to {SN}, it means there is no reason for the upper layer to act
// on an interface for which the underlying layer (Zero Conf) failed.
// unlock the hashes at the end
LeaveCriticalSection(&(g_lstIntfHashes.csMutex));
if (bForwardUp)
{
// Notify upper level app (802.1x) that the selected
// 802.11 configuration is successful.
DbgPrint((TRC_NOTIF, "Passing through notification %d",pDevNotif->dwEventType));
dwErr = ElMediaEventsHandler(pDevNotif);
DbgAssert((dwErr == ERROR_SUCCESS, "Error or Exception 0x%x when passing through notification", dwErr));
}
exit:
MemFree(pwszDeviceKey);
MemFree(pDevNotif);
DbgPrint((TRC_TRACK,"WZCWrkDeviceNotifHandler=%d]", dwErr));
// decrement the thread counter
InterlockedDecrement(&g_nThreads);
}
//-----------------------------------------------------------
// WZCTimeoutCallback: timer callback routine. It should not lock any cs, but just spawn
// the timer handler routine after referencing the context (to avoid premature deletion)
VOID WINAPI
WZCTimeoutCallback(
IN PVOID pvData,
IN BOOL fTimerOrWaitFired)
{
DWORD dwErr = ERROR_SUCCESS;
PINTF_CONTEXT pIntfContext = (PINTF_CONTEXT)pvData;
BOOL bDecrement = TRUE;
// increment the thread counter the first thing!
InterlockedIncrement(&g_nThreads);
DbgPrint((TRC_TRACK|TRC_NOTIF, "[WZCTimeoutCallback(0x%p)", pIntfContext));
DbgAssert((pIntfContext != NULL, "Invalid (null) context in timer callback!"));
// reference the context to make sure no one will delete it unexpectedly.
LstRccsReference(pIntfContext);
if (!QueueUserWorkItem(
(LPTHREAD_START_ROUTINE)WZCSvcTimeoutHandler,
(LPVOID)pIntfContext,
WT_EXECUTELONGFUNCTION))
{
dwErr = GetLastError();
goto exit;
}
bDecrement = FALSE;
exit:
DbgPrint((TRC_TRACK|TRC_NOTIF, "WZCTimeoutCallback=%d]", dwErr));
if (bDecrement)
{
// getting here would be really bad - it would mean we failed to spawn the
// timer handler. We need to make sure we not unbalance the reference counter
// on the context. The counter can't get to 0 since the device notification thread
// is waiting for all the timer routines to complete.
InterlockedDecrement(&(pIntfContext->rccs.nRefCount));
InterlockedDecrement(&g_nThreads);
}
}
//-----------------------------------------------------------
VOID
WZCSvcTimeoutHandler(
IN PVOID pvData)
{
DWORD dwErr = ERROR_SUCCESS;
PINTF_CONTEXT pIntfContext = (PINTF_CONTEXT)pvData;
DbgPrint((TRC_TRACK,"[WZCSvcTimeoutHandler(0x%p)", pIntfContext));
// lock the context here (it has been referenced in the timer callback!)
LstRccsLock(pIntfContext);
// the timer handler should be a noop in all the following cases:
// - the service doesn't look to be either starting or running
// - the timer handler is invalid. This is an indication the context is being destroyed
// - the timer flag is not set! Same indication the context is being destroyed.
if ((g_WZCSvcStatus.dwCurrentState == SERVICE_RUNNING || g_WZCSvcStatus.dwCurrentState == SERVICE_START_PENDING) &&
(pIntfContext->hTimer != INVALID_HANDLE_VALUE) &&
(pIntfContext->dwCtlFlags & INTFCTL_INTERNAL_TM_ON))
{
// since the timer fired already for this event and we only deal with
// one time timers, reset the TIMER_ON flag for this context:
pIntfContext->dwCtlFlags &= ~INTFCTL_INTERNAL_TM_ON;
// dispatch the timeout event
dwErr = StateDispatchEvent(
eEventTimeout,
pIntfContext,
NULL);
// clear up the INTFCTL_INTERNAL_BLK_MEDIACONN bit since this is not a media sense handler
pIntfContext->dwCtlFlags &= ~INTFCTL_INTERNAL_BLK_MEDIACONN;
DbgAssert((dwErr == ERROR_SUCCESS,
"Dispatching timeout event failed for context 0x%p\n",
pIntfContext));
}
// unlock unref the context!
LstRccsUnlockUnref(pIntfContext);
DbgPrint((TRC_TRACK,"WZCSvcTimeoutHandler=%d]", dwErr));
InterlockedDecrement(&g_nThreads);
}
//-----------------------------------------------------------
VOID
WZCWrkWzcSendNotif(
IN LPVOID pvData)
{
DWORD dwErr = ERROR_SUCCESS;
DbgPrint((TRC_TRACK,"[WZCWrkWzcSendNotif(0x%p)", pvData));
DbgPrint((TRC_NOTIF, "Sending WZC_CONFIG_NOTIF to upper level apps"));
dwErr = ElMediaEventsHandler(pvData);
DbgPrint((TRC_TRACK,"WZCWrkWzcSendNotif]=%d", dwErr));
InterlockedDecrement(&g_nThreads);
}
//-----------------------------------------------------------
// EAPOLQueryGUIDNCSState
// Called by Netman module query the ncs state of the GUID under 802.1X control
// Arguments:
// pGuidConn - Interface GUID
// pncs - NCS status of the interface
// Returns:
// S_OK - network connection status is reported by WZC
// S_FALSE - status is not controlled by WZC
HRESULT
WZCQueryGUIDNCSState (
IN GUID *pGuidConn,
OUT NETCON_STATUS *pncs)
{
HRESULT hr = S_FALSE;
InterlockedIncrement(&g_nThreads);
// check if the service is still running, return the call instantly
if (g_WZCSvcStatus.dwCurrentState == SERVICE_RUNNING ||
g_WZCSvcStatus.dwCurrentState == SERVICE_START_PENDING)
{
// check what Zero Config says about the adapter's status
if (hr == S_FALSE)
{
WCHAR wszGuid[64]; // enough for "{xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx}"
// convert the GUID to "{xxx...}" and get the ncStatus from the interface context (if any)
if (StringFromGUID2(pGuidConn, wszGuid, 64) != 0)
hr = LstQueryGUIDNCStatus(wszGuid, pncs);
}
// check what 802.1x says about the adapter's status
if (hr == S_FALSE)
{
hr = EAPOLQueryGUIDNCSState (
pGuidConn,
pncs);
}
}
InterlockedDecrement(&g_nThreads);
return hr;
}
//-----------------------------------------------------------
// EAPOLQueryGUIDNCSState
// WZCTrayIconReady
//
// Purpose: Called by Netman module to inform about Tray being
// ready for notifications from WZCSVC
// Arguments:
// pszUserName - Username of the user logged in on the desktop
// Returns:
// None
VOID
WZCTrayIconReady (
IN const WCHAR * pszUserName)
{
EAPOLTrayIconReady(pszUserName);
}
//-----------------------------------------------------------
// WZCContextInit
// Description:Initialises an internal context with default values
// Parameters: pointer to an internal context with storage pre allocated
// Returns: win32 error code
DWORD WZCContextInit(PWZC_INTERNAL_CONTEXT pwzcICtxt)
{
DWORD dwErr = ERROR_SUCCESS;
DbgPrint((TRC_TRACK,"[WZCContextInit(%p=%d)", pwzcICtxt, pwzcICtxt->bValid));
if (pwzcICtxt->bValid)
{
dwErr = ERROR_ALREADY_INITIALIZED;
}
else
{
PWZC_CONTEXT pwzcCtxt = &(pwzcICtxt->wzcContext);
DbgPrint((TRC_TRACK,"Initializing wzc context"));
// null out the service's context
ZeroMemory(pwzcCtxt, sizeof(WZC_CONTEXT));
__try
{
//set defaults
pwzcCtxt->tmTr = TMMS_DEFAULT_TR;
pwzcCtxt->tmTc = TMMS_DEFAULT_TC;
pwzcCtxt->tmTp = TMMS_DEFAULT_TP;
pwzcCtxt->tmTf = TMMS_DEFAULT_TF;
pwzcCtxt->tmTd = TMMS_DEFAULT_TD;
//Init CriticalSection
InitializeCriticalSection(&(pwzcICtxt->csContext));
DbgPrint((TRC_TRACK,"Critical section initialized successfully"));
// mark the context is valid
pwzcICtxt->bValid = TRUE;
}
__except(EXCEPTION_EXECUTE_HANDLER)
{
dwErr = GetExceptionCode();
}
if (dwErr == ERROR_SUCCESS)
{
dwErr = LstConstructIntfContext(
NULL,
&(pwzcICtxt->pIntfTemplate));
}
}
DbgPrint((TRC_TRACK,"WZCContextInit]=%d", dwErr));
return dwErr;
}
//-----------------------------------------------------------
// WZCContextDestroy
// Description:Destroys an internal context with default values
// Parameters: pointer to an internal context initialised using WZCContextInit
// Returns: win32 error code
DWORD WZCContextDestroy(PWZC_INTERNAL_CONTEXT pwzcICtxt)
{
DWORD dwErr = ERROR_SUCCESS;
PWZC_CONTEXT pwzcCtxt = NULL;
DbgPrint((TRC_TRACK,"[WZCContextDestroy"));
if (pwzcICtxt->bValid)
{
// destroy the global context here
LstDestroyIntfContext(pwzcICtxt->pIntfTemplate);
pwzcICtxt->pIntfTemplate = NULL;
pwzcICtxt->bValid = FALSE;
//Destroy critical section
DeleteCriticalSection(&(pwzcICtxt->csContext));
}
DbgPrint((TRC_TRACK,"WZCContextDestroy]=%d", dwErr));
return dwErr;
}
//-----------------------------------------------------------
// WzcContextQuery
// Description: Queries specified params in the global context and sends the
// values back to the client.
// Parameters:
// [in] dwInFlags - Bitmask of the WZC_CTL_* flags, indicates the
// appropriate parameter.
// [out] pContext - Holds current values requested by user
// [out] pdwOutFlags - Indicates the values that were successfully returned
// Returns: win32 error code
DWORD WzcContextQuery(
DWORD dwInFlags,
PWZC_CONTEXT pContext,
LPDWORD pdwOutFlags)
{
DWORD dwErr = ERROR_SUCCESS;
DWORD dwOutFlags = 0;
PWZC_CONTEXT pwzcCtxt = &g_wzcInternalCtxt.wzcContext;
DbgPrint((TRC_TRACK, "[WzcContextQuery(%d)", dwInFlags));
if (FALSE == g_wzcInternalCtxt.bValid)
{
dwErr = ERROR_ARENA_TRASHED;
goto exit;
}
EnterCriticalSection(&g_wzcInternalCtxt.csContext);
if (dwInFlags & WZC_CONTEXT_CTL_LOG)
{
pContext->dwFlags |= (DWORD)(pwzcCtxt->dwFlags & WZC_CTXT_LOGGING_ON);
dwOutFlags |= WZC_CONTEXT_CTL_LOG;
}
if (dwInFlags & WZC_CONTEXT_CTL_TIMER_TR)
{
pContext->tmTr = pwzcCtxt->tmTr;
dwOutFlags |= WZC_CONTEXT_CTL_TIMER_TR;
}
if (dwInFlags & WZC_CONTEXT_CTL_TIMER_TC)
{
pContext->tmTc = pwzcCtxt->tmTc;
dwOutFlags |= WZC_CONTEXT_CTL_TIMER_TC;
}
if (dwInFlags & WZC_CONTEXT_CTL_TIMER_TP)
{
pContext->tmTp = pwzcCtxt->tmTp;
dwOutFlags |= WZC_CONTEXT_CTL_TIMER_TP;
}
if (dwInFlags & WZC_CONTEXT_CTL_TIMER_TF)
{
pContext->tmTf = pwzcCtxt->tmTf;
dwOutFlags |= WZC_CONTEXT_CTL_TIMER_TF;
}
if (dwInFlags & WZC_CONTEXT_CTL_TIMER_TD)
{
pContext->tmTd = pwzcCtxt->tmTd;
dwOutFlags |= WZC_CONTEXT_CTL_TIMER_TD;
}
LeaveCriticalSection(&g_wzcInternalCtxt.csContext);
exit:
if (pdwOutFlags != NULL)
*pdwOutFlags = dwOutFlags;
DbgPrint((TRC_TRACK, "WzcContextQuery(out: 0x%08x)]=%d", dwOutFlags, dwErr));
return dwErr;
}
//-----------------------------------------------------------
// WzcContextSet
// Description: Sets specified params in the global context to the values
// passed in by the client.
// Parameters:
// [in] dwInFlags - Bitmask of the WZC_CTL_* flags, indicates the
// appropriate parameter.
// [in] pContext - Should point to the user specified values
// [out] pdwOutFlags - Indicates the values that were successfully set
// Returns: win32 error code
DWORD WzcContextSet(
DWORD dwInFlags,
PWZC_CONTEXT pContext,
LPDWORD pdwOutFlags)
{
DWORD dwErr = ERROR_SUCCESS;
DWORD dwOutFlags = 0;
PWZC_CONTEXT pwzcCtxt = &g_wzcInternalCtxt.wzcContext;
DbgPrint((TRC_TRACK, "[WzcContextSet(%d)", dwInFlags));
if (FALSE == g_wzcInternalCtxt.bValid)
{
dwErr = ERROR_ARENA_TRASHED;
goto exit;
}
EnterCriticalSection(&g_wzcInternalCtxt.csContext);
//Set appropriate entries
if (dwInFlags & WZC_CONTEXT_CTL_LOG)
{
if (pContext->dwFlags & WZC_CTXT_LOGGING_ON)
pwzcCtxt->dwFlags |= WZC_CONTEXT_CTL_LOG;
else
pwzcCtxt->dwFlags &= ~WZC_CONTEXT_CTL_LOG;
dwOutFlags |= WZC_CONTEXT_CTL_LOG;
}
if (dwInFlags & WZC_CONTEXT_CTL_TIMER_TR)
{
pwzcCtxt->tmTr = pContext->tmTr;
dwOutFlags |= WZC_CONTEXT_CTL_TIMER_TR;
}
if (dwInFlags & WZC_CONTEXT_CTL_TIMER_TC)
{
pwzcCtxt->tmTc = pContext->tmTc;
dwOutFlags |= WZC_CONTEXT_CTL_TIMER_TC;
}
if (dwInFlags & WZC_CONTEXT_CTL_TIMER_TP)
{
pwzcCtxt->tmTp = pContext->tmTp;
dwOutFlags |= WZC_CONTEXT_CTL_TIMER_TP;
}
if (dwInFlags & WZC_CONTEXT_CTL_TIMER_TF)
{
pwzcCtxt->tmTf = pContext->tmTf;
dwOutFlags |= WZC_CONTEXT_CTL_TIMER_TF;
}
if (dwInFlags & WZC_CONTEXT_CTL_TIMER_TD)
{
pwzcCtxt->tmTd = pContext->tmTd;
dwOutFlags |= WZC_CONTEXT_CTL_TIMER_TD;
}
//Save into the registry
dwErr = StoSaveWZCContext(NULL, &g_wzcInternalCtxt.wzcContext);
DbgAssert((ERROR_SUCCESS == dwErr, "Error saving context to registry %d",dwErr));
LeaveCriticalSection(&g_wzcInternalCtxt.csContext);
exit:
if (pdwOutFlags != NULL)
*pdwOutFlags = dwOutFlags;
DbgPrint((TRC_TRACK, "WzcContextSet(out: 0x%08x)]=%d", dwOutFlags, dwErr));
return dwErr;
}