再談WinIO初始化異常


再談WinIO初始化異常  

 

 
 
前段時間WinIO在我的新項目中總是初始化失敗,有時候又是好好的,很讓人費解。修改了源代碼顯示了很多調試信息后,也沒有什么太多的收獲。由於我們的工控卡必須要用這個庫,所以沒辦法,只得停下腳步,細細研究一下問題所在。

初始化的時候調用的是InitializeWinIo()函數:

bool _stdcall InitializeWinIo()
{
bool bResult;
DWORD dwBytesReturned;


IsNT = IsWinNT();

if (IsNT)
{
hDriver = CreateFile("\\\\.\\WINIO",
GENERIC_READ | GENERIC_WRITE,
0,
NULL,
OPEN_EXISTING,
FILE_ATTRIBUTE_NORMAL,
NULL);

// If the driver is not running, install it

if (hDriver == INVALID_HANDLE_VALUE)
{
GetDriverPath();

bResult = InstallWinIoDriver(szWinIoDriverPath, true);

if (!bResult)
return false;

bResult = StartWinIoDriver();

if (!bResult)
return false;

hDriver = CreateFile("\\\\.\\WINIO",
GENERIC_READ | GENERIC_WRITE,
0,
NULL,
OPEN_EXISTING,
FILE_ATTRIBUTE_NORMAL,
NULL);

if (hDriver == INVALID_HANDLE_VALUE)
return false;
}

// Enable I/O port access for this process

if (!DeviceIoControl(hDriver, IOCTL_WINIO_ENABLEDIRECTIO, NULL,
0, NULL, 0, &dwBytesReturned, NULL))
return false;

}
else
{
VxDCall = (DWORD (WINAPI *)(DWORD,DWORD,DWORD))GetK32ProcAddress(1);

hDriver = CreateFile("\\\\.\\WINIO.VXD", 0, 0, 0, CREATE_NEW, FILE_FLAG_DELETE_ON_CLOSE, 0);

if (hDriver == INVALID_HANDLE_VALUE)
return false;
}

IsWinIoInitialized = true;

return true;
}

函數首先查看是不是NT系統,如果是,就創建設備驅動的handle,如果沒有的話,就安裝一個。有的話就直接使用。

仔細看看這個函數值后就知道其實引起初始化失敗的原因很多。我遇到的問題就是有時候異常退出了,第二次就無法正常啟動WinIO了,只能按照這樣的方式來進行:

InitWinIO->成功->程序異常退出->InitWinIO->失敗->ShutdownWinIO->InitWinIO->成功

也就是說重新啟動兩次程序才行。由於異常退出保留在內存的驅動第二次InitWinIO會失敗,然后ShutdownWinIO就會卸載這個驅動,再進行InitWinIO就正確了。

那么直接在應用程序啟動前無論怎么就運行ShutdownWinIO先關閉一下可以么?看看代碼似乎是可以的:

void _stdcall ShutdownWinIo()
{
DWORD dwBytesReturned;


if (IsNT)
{
if (hDriver != INVALID_HANDLE_VALUE)
{
// Disable I/O port access

DeviceIoControl(hDriver, IOCTL_WINIO_DISABLEDIRECTIO, NULL,
0, NULL, 0, &dwBytesReturned, NULL);

CloseHandle(hDriver);

}

RemoveWinIoDriver();
}
else
CloseHandle(hDriver);

IsWinIoInitialized = false;
}

我測試過,直接在程序一開始就執行ShutdownWinIo()然后在初始化WinIO,一樣可能提示出錯。什么原因呢?

其實,WinIO的ShutdownWinIo這里有一個bug:

if(IsNT)

如果你沒有調用過InitializeWinIo()的話,全局變量IsNT是為false的,他就沒有正確的執行nt平台的代碼。這就解釋了為什么沒有關閉成功。

所以,找到了問題所在,有兩種解決方法:

1、將IsNT替換為IsWinNT()的函數調用

2、在InitWinIO里面if(IsNT)以后,執行一次ShutdownWinIO即可。

這樣,我所遇到的初始化失敗問題終於得到了解決:)


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM