0

获取 Windows 序列号的常规方法是 WMI。

 ManagementObjectSearcher mos = new ManagementObjectSearcher("Select * From Win32_OperatingSystem");
 // ...
 // Select number from managementobject mo["SerialNumber"]

我不想使用 WMI,因为紧凑的框架不支持它。该程序集必须在桌面和紧凑框架端工作,因此我无法添加参考。

如何使用 pinvoke 调用获得相同的结果?

4

2 回答 2

1

您需要为 WindowsCE 调用 KernelIOControl。

这是c++代码,没时间转换成c#

#include <WINIOCTL.H> 
extern "C" __declspec(dllimport) 
BOOL KernelIoControl( DWORD dwIoControlCode, LPVOID lpInBuf, DWORD nInBufSize, LPVOID lpOutBuf, DWORD nOutBufSize, LPDWORD lpBytesReturned ); 
#define IOCTL_HAL_GET_DEVICEID CTL_CODE(FILE_DEVICE_HAL, 21, METHOD_BUFFERED, FILE_ANY_ACCESS) 

CString GetSerialNumberFromKernelIoControl() { 
    DWORD dwOutBytes; 
    const int nBuffSize = 4096; 
    byte arrOutBuff[nBuffSize]; 
    BOOL bRes = ::KernelIoControl(IOCTL_HAL_GET_DEVICEID, 0, 0, arrOutBuff, nBuffSize, &dwOutBytes); 
    if (bRes) { CString strDeviceInfo; for (unsigned int i = 0; i<dwOutBytes; i++) { 
        CString strNextChar; strNextChar.Format(TEXT("%02X"), arrOutBuff[i]); strDeviceInfo += strNextChar; 
    } 
    CString strDeviceId = strDeviceInfo.Mid(40,2) + strDeviceInfo.Mid(45,9) + strDeviceInfo.Mid(70,6); 
    return strDeviceId; 
    } else { 
        return _T(""); 
    } 
} 

编辑:(pinvoke kernelIOControl c#)

[DllImport("coredll.dll")]
    public static extern bool KernelIoControl(long dwIoControlCode, IntPtr lpInBuff, long dwInBuffSize, IntPtr lpOutBuff, long dwOutBuffSize, IntPtr lpBytesReturned);
于 2009-02-02T10:20:38.113 回答
0

首先,您不会在桌面和设备上进行一次通话。只是不会发生。您可以做的是通过如下调用确定运行时环境:

if(Environment.OSVersion.Platform == PlatformID.WinCE) { ... }

这将使您分离桌面和设备。

然后你必须增加设备的复杂性,为此你需要了解你的目标硬件。对于 Windows Mobile 5.0 和更高版本,您想要调用GetDeviceUniqueID,因为 KernelIoControl 调用很可能会受到保护。对于 Pocket PC 2003 和更早版本,KernelIoControl P/Invoke 是合理的,尽管已知许多设备会呈现相同的结果,所以不能保证唯一。

对于通用 Windows CE 设备,它的种类要多得多。没有什么可以保证平台实现 IOCTL_HAL_GET_DEVICEID,因此您需要保护故障并找到一些其他机制(通常 OEM 实现自己的 ID API)。对于 CE 6.0,KernelIoControl 对应用程序非常有限,如果没有来自 OEM 的内核或驱动程序包装 API,您不太可能调用它。

于 2009-02-02T15:21:27.567 回答