经常在坛子里看到讨论软件加密的帖子,纯软件加密与读取硬件序列号加密是经常讨论到的。
两种方法各有优缺点。
在通过读取硬件序列号的方法来加密的方法,受硬件的限制。
一般来说,CPU和T-Flash可能存在序列号。今天研究了一下 Windows CE 6.0 下的读取 SD 卡(T-Flash)的方法,以下将自己的实现过程列出来,供有需要的朋友一起学习。
函数的声明,在.H文件文件中:
#define SD_PART_NAME L"DSK2:"
#define VALID_SD_SERIAL_1 L"A7DFB784"
源代码如下函数所示:
BOOL GetStorageID(TCHAR *ptcCardName,TCHAR *ptcManufactureID,TCHAR *ptcSerialNum)
{
DWORD dwSize = 0;
DWORD dwReqSize = 0;
STORAGE_IDENTIFICATION StoreInfo;
STORAGE_IDENTIFICATION StoreInfo2;
HANDLE hVolume = NULL;
BOOL bRet = FALSE;
BYTE *pucSerialNo = NULL;
BYTE *pucManuID = NULL;
int i = 0;
ZeroMemory(&StoreInfo,sizeof(STORAGE_IDENTIFICATION));
hVolume = CreateFile(ptcCardName,GENERIC_READ | GENERIC_WRITE,0,NULL,OPEN_EXISTING,0,NULL);
if(NULL == hVolume || INVALID_HANDLE_VALUE == hVolume)
{
// MessageBox(L"Open Partation failed!");
RETAILMSG(1,(L"Open Partation failed!\r\n"));
return FALSE;
}
bRet = DeviceIoControl(hVolume,IOCTL_DISK_GET_STORAGEID,
NULL,0,(LPVOID)&StoreInfo,/*sizeof(STORAGE_IDENTIFICATION)*/3000,&dwSize,NULL);
if(!bRet)
{
DWORD dwErr = GetLastError();
// TCHAR tcError[64];
// wsprintf(tcError,L"Device IO 1 failed: %d!",dwErr);
// MessageBox(tcError);
RETAILMSG(1,(L"Device IO 1 failed: %d!\r\n",dwErr));
CloseHandle(hVolume);
return FALSE;
}
dwReqSize = StoreInfo.dwSize;
ASSERT(dwReqSize > 0);
dwSize = 0;
StoreInfo2.dwSize = dwReqSize;
bRet = DeviceIoControl(hVolume,IOCTL_DISK_GET_STORAGEID,
NULL,0,(LPVOID)&StoreInfo,dwReqSize,&dwSize,NULL);
if(FALSE == bRet)
{
DWORD dwErr = GetLastError();
// TCHAR tcError[64];
// wsprintf(tcError,L"Device IO 2 failed: %d!",dwErr);
// MessageBox(tcError);
RETAILMSG(1,(L"Device IO 2 failed: %d!\r\n",dwErr));
CloseHandle(hVolume);
return FALSE;
}
pucSerialNo = (((BYTE *)&StoreInfo) + StoreInfo.dwSerialNumOffset);
pucManuID = (((BYTE *)&StoreInfo) + StoreInfo.dwManufactureIDOffset);
while(pucSerialNo[i] != 0 && i < 200 && i < (int)(dwSize - StoreInfo.dwSerialNumOffset))
{
ptcSerialNum[i] = pucSerialNo[i];
i++;
}
pucSerialNo[i] = '\0 ';
i = 0;
while(pucManuID[i] != 0 && i < 200 && i < (int)(StoreInfo.dwSerialNumOffset - StoreInfo.dwManufactureIDOffset))
{
ptcManufactureID[i] = pucManuID[i];
i++;
}
pucManuID[i] = '\0 ';
CloseHandle(hVolume);
return TRUE;
}
调用示例如下:
TCHAR tcSDSerial[256];
TCHAR tcSDManu[256];
ZeroMemory(tcSDSerial,sizeof(TCHAR) * 256);
ZeroMemory(tcSDManu,sizeof(TCHAR) * 256);
if(0 == GetStorageID(SD_PART_NAME,tcSDManu,tcSDSerial))
{
MessageBox(tcSDSerial);
if(0 == wcsncmp(VALID_SD_SERIAL_1,tcSDSerial,wcslen(VALID_SD_SERIAL_1)))
{
}
else
{
}
}
实现过程中,遇到以下错误:
(1) 当将调用代码修改为: if(GetStorageID(L"DSK1:",csManufactureID,csSerialID)) 时(DSK1 是存在的)产生如下错误:
Error 50: 不支持请求。
(2) 当将实现代码中的 DeviceIoControl()函数 修改为如下时:
bRet = DeviceIoControl(hVolume,IOCTL_DISK_GET_STORAGEID,
NULL,0,(LPVOID)&StoreInfo,sizeof(STORAGE_IDENTIFICATION),&dwSize,NULL);
产生如下错误:Error 122: 传递给系统调用的数据区域太小。所以,建议各位程序在编码时,尽量对函数的返回值进行判断。在出错的状态,一定要调用 GetLastError() 函数获取详细的错误码。