CE6.0 下获得 SD 卡序列号的方法

时间:2022-05-14 22:53:05

经常在坛子里看到讨论软件加密的帖子,纯软件加密与读取硬件序列号加密是经常讨论到的。

两种方法各有优缺点。

在通过读取硬件序列号的方法来加密的方法,受硬件的限制。

一般来说,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);

源代码如下函数所示:

  1. BOOL GetStorageID(TCHAR *ptcCardName,TCHAR *ptcManufactureID,TCHAR *ptcSerialNum)
  2. {
  3. DWORD dwSize = 0;
  4. DWORD dwReqSize = 0;
  5. STORAGE_IDENTIFICATION StoreInfo;
  6. STORAGE_IDENTIFICATION StoreInfo2;
  7. HANDLE hVolume = NULL;
  8. BOOL bRet = FALSE;
  9. BYTE *pucSerialNo = NULL;
  10. BYTE *pucManuID = NULL;
  11. int i = 0;
  12. ZeroMemory(&StoreInfo,sizeof(STORAGE_IDENTIFICATION));
  13. hVolume = CreateFile(ptcCardName,GENERIC_READ | GENERIC_WRITE,0,NULL,OPEN_EXISTING,0,NULL);
  14. if(NULL == hVolume || INVALID_HANDLE_VALUE == hVolume)
  15. {
  16. // MessageBox(L"Open Partation failed!");
  17. RETAILMSG(1,(L"Open Partation failed!\r\n"));
  18. return FALSE;
  19. }
  20. bRet = DeviceIoControl(hVolume,IOCTL_DISK_GET_STORAGEID,
  21. NULL,0,(LPVOID)&StoreInfo,/*sizeof(STORAGE_IDENTIFICATION)*/3000,&dwSize,NULL);
  22. if(!bRet)
  23. {
  24. DWORD dwErr = GetLastError();
  25. // TCHAR tcError[64];
  26. // wsprintf(tcError,L"Device IO 1 failed: %d!",dwErr);
  27. // MessageBox(tcError);
  28. RETAILMSG(1,(L"Device IO 1 failed: %d!\r\n",dwErr));
  29. CloseHandle(hVolume);
  30. return FALSE;
  31. }
  32. dwReqSize = StoreInfo.dwSize;
  33. ASSERT(dwReqSize > 0);
  34. dwSize = 0;
  35. StoreInfo2.dwSize = dwReqSize;
  36. bRet = DeviceIoControl(hVolume,IOCTL_DISK_GET_STORAGEID,
  37. NULL,0,(LPVOID)&StoreInfo,dwReqSize,&dwSize,NULL);
  38. if(FALSE == bRet)
  39. {
  40. DWORD dwErr = GetLastError();
  41. // TCHAR tcError[64];
  42. // wsprintf(tcError,L"Device IO 2 failed: %d!",dwErr);
  43. // MessageBox(tcError);
  44. RETAILMSG(1,(L"Device IO 2 failed: %d!\r\n",dwErr));
  45. CloseHandle(hVolume);
  46. return FALSE;
  47. }
  48. pucSerialNo = (((BYTE *)&StoreInfo) + StoreInfo.dwSerialNumOffset);
  49. pucManuID = (((BYTE *)&StoreInfo) + StoreInfo.dwManufactureIDOffset);
  50. while(pucSerialNo[i] != 0 && i < 200 && i < (int)(dwSize - StoreInfo.dwSerialNumOffset))
  51. {
  52. ptcSerialNum[i] = pucSerialNo[i];
  53. i++;
  54. }
  55. pucSerialNo[i] = '\0 ';
  56. i = 0;
  57. while(pucManuID[i] != 0 && i < 200 && i < (int)(StoreInfo.dwSerialNumOffset - StoreInfo.dwManufactureIDOffset))
  58. {
  59. ptcManufactureID[i] = pucManuID[i];
  60. i++;
  61. }
  62. pucManuID[i] = '\0 ';
  63. CloseHandle(hVolume);
  64. return TRUE;
  65. }

调用示例如下:

  1. TCHAR tcSDSerial[256];
  2. TCHAR tcSDManu[256];
  3. ZeroMemory(tcSDSerial,sizeof(TCHAR) * 256);
  4. ZeroMemory(tcSDManu,sizeof(TCHAR) * 256);
  5. if(0 == GetStorageID(SD_PART_NAME,tcSDManu,tcSDSerial))
  6. {
  7. MessageBox(tcSDSerial);
  8. if(0 == wcsncmp(VALID_SD_SERIAL_1,tcSDSerial,wcslen(VALID_SD_SERIAL_1)))
  9. {
  10. }
  11. else
  12. {
  13. }
  14. }
  15. 实现过程中,遇到以下错误:
  16. (1) 当将调用代码修改为: if(GetStorageID(L"DSK1:",csManufactureID,csSerialID)) 时(DSK1 是存在的)产生如下错误:
  17. Error 50: 不支持请求。
  18. (2) 当将实现代码中的 DeviceIoControl()函数 修改为如下时:
  19. bRet = DeviceIoControl(hVolume,IOCTL_DISK_GET_STORAGEID,
  20. NULL,0,(LPVOID)&StoreInfo,sizeof(STORAGE_IDENTIFICATION),&dwSize,NULL);
  21. 产生如下错误:Error 122: 传递给系统调用的数据区域太小。所以,建议各位程序在编码时,尽量对函数的返回值进行判断。在出错的状态,一定要调用 GetLastError() 函数获取详细的错误码。