再談WinIO初始化異常
前段時間WinIO在我的新項目中總是初始化失敗,有時候又是好好的,很讓人費解。修改了源代碼顯示了很多調試信息后,也沒有什么太多的收獲。由於我們的工控卡必須要用這個庫,所以沒辦法,只得停下腳步,細細研究一下問題所在。
初始化的時候調用的是InitializeWinIo()函數:
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先關閉一下可以么?看看代碼似乎是可以的:
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即可。
這樣,我所遇到的初始化失敗問題終於得到了解決:)
初始化的時候調用的是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即可。
這樣,我所遇到的初始化失敗問題終於得到了解決:)