Pnp管理器(2)
來源:程序員人生 發布時間:2016-07-15 09:31:50 閱讀次數:4017次
前1篇Pnp管理器(1)提到了總線上裝備變化時,將產生Pnp消息并在Pnp管理器個組件間活動。消息傳遞的終究目的是通知某個組件加載驅動。本文看1下Pnp管理器收到消息后加載驅動的流程。
當PnpEventThread函數從等待阻塞中返回,判斷如果是添加裝備則創建1個DeviceInstallParams* Params變量
typedef struct
{
#ifdef HAVE_SLIST_ENTRY_IMPLEMENTED
SLIST_ENTRY ListEntry;
#else
LIST_ENTRY ListEntry;
#endif
WCHAR DeviceIds[1];
} DeviceInstallParams;
前面裝備樹結構變更時,取得了裝備的ID,現在將這個ID填入Params變量,并加入DeviceInstallListHead鏈表,然后通知Pnp管理器的DeviceInstallThread內核線程為新裝備安裝驅動。和PnpEventThread1樣DeviceInstallThread也是IO管理器初始化時創建的線程,循環往復的等待事件hDeviceInstallListNotEmpty被觸發。
static VOID CALLBACK
ServiceMain(DWORD argc, LPTSTR *argv)
{
...
hThread = CreateThread(NULL,
0,
PnpEventThread,
NULL,
0,
&dwThreadId);
if (hThread != NULL)
CloseHandle(hThread);
hThread = CreateThread(NULL,
0,
DeviceInstallThread,
NULL,
0,
&dwThreadId);
...
}
當DeviceInstallThread線程從阻塞中返回,從DeviceInstallListHead隊列中取得PnpEventThread線程中加入的DeviceInstallParams項,然后調用InstallDevice開始安裝驅動
static DWORD WINAPI
DeviceInstallThread(LPVOID lpParameter)
{
while (TRUE)
{
if ((BOOL)IsListEmpty(&DeviceInstallListHead))
ListEntry = NULL;
else
ListEntry = RemoveHeadList(&DeviceInstallListHead);
//從DeviceInstallListHead隊列中取得PnpEventThread線程中加入的DeviceInstallParams項
//ListEntry 收到通知時從阻塞中返回將進入到else分支,否則進入if分支
//繼續等待
if (ListEntry == NULL)
{
SetEvent(hNoPendingInstalls);
WaitForSingleObject(hDeviceInstallListNotEmpty, INFINITE);
}
else
{
ResetEvent(hNoPendingInstalls);
Params = CONTAINING_RECORD(ListEntry, DeviceInstallParams, ListEntry);
InstallDevice(Params->DeviceIds, setupActive);
}
}
}
ReactOS0.33中DevInstall的實現比較簡單:從newdev.dll中搜索DevInstallW函數,然后向DevInstallW輸入DevID就算完成驅動的安裝了。看這個函數的接口情勢有點像windows的UpdateDriverForPlugAndPlayDevices。后面有時間我就寫篇關于用UpdateDriverForPlugAndPlayDevices驅動安裝的博文
生活不易,碼農辛苦
如果您覺得本網站對您的學習有所幫助,可以手機掃描二維碼進行捐贈