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

两种方法各有优缺点。

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

一般来说,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() 函数获取详细的错误码。

CE6.0 下获得 SD 卡序列号的方法的更多相关文章

  1. Ubuntu下查看SD卡设备名的几个方法

    Ubuntu下使用SD卡查询SD卡的设备文件名:sudo fdisk -leg:Disk /dev/sdb:14.9 GiB,15931539456 字节,31116288 个扇区单元:扇区 / 1 ...

  2. Linux的启动SD卡的格式化方法

    要在OMAP3530上运行Linux,首先要知道如何启动OMAP3530,并且将MLO,XDLR,UBOOT,UImage以及文件系统等镜像程序下载到OMAP3530的芯片中去. OMAP3530提供 ...

  3. 【译】如何在 Android 5.0 上获取 SD卡 的读写权限

    因为最近项目需要,涉及到 SD卡 的读写操作,然而申请 <!-- 读写权限 --> <uses-permission android:name="android.permi ...

  4. 基于stm32f103zet6的FAT16文件系统学习0(读SD卡扇区)

    SD卡已经看了两天了,主要是因为测试出来的卡容量不对,所以一直找原因,最终还是发现了,总比不过是单位上面出现了问题,或许是之前没有接触到SD的缘故吧,所以对其中的一些寄存器很不了解,一切都是重新开始, ...

  5. Mac下给SD卡烧录树莓派系统

    1.mac 磁盘工具 抹掉sd卡 为fat 2. df -h 看清sd卡号 3. 卸载sd卡 diskutil unmount /dev/disk2s2 Volume UNTITLED on disk ...

  6. 关于ARM Linux下的SD卡及U盘的挂载问题

    内核配置并运行后,挂载SD卡,出现问题: zynq> mount -t /dev/mmcblk1 /mntmount: mounting /dev/mmcblk0 on /mnt failed: ...

  7. Linux环境下挂载SD卡的教程

    1.插入SD卡 如果系统能够识别SD卡,则会打印一些信息: 2.查看系统给SD卡分配的设备名 命令如下: fdisk -l 命令 说明:通常是根据SD卡的存储容量来确定的. 比如下面的信息: 3.挂载 ...

  8. SPI模式下MCU对SD卡的控制及操作命令

    一.前言 SD 卡有两个可选的通讯协议:SD 模式和 SPI模式 SD 模式是SD 卡标准的读写方式,但是在选用SD 模式时,往往需要选择带有SD 卡控制器接口的 MCU,或者必须加入额外的SD卡控制 ...

  9. SPI模式下MCU对SD卡的控制及操作命令(转)

    源:SPI模式下MCU对SD卡的控制及操作命令 一.前言 SD 卡有两个可选的通讯协议:SD 模式和 SPI模式 SD 模式是SD 卡标准的读写方式,但是在选用SD 模式时,往往需要选择带有SD 卡控 ...

随机推荐

  1. Bootstrap3 代码-用户输入

    通过 <kbd> 标签标记用户通过键盘输入的内容. To switch directories, type cd followed by the name of the directory ...

  2. Hadoop的RPC通信原理

    RPC调用: RPC(remote procedure call)远程过程调用: 不同java进程间的对象方法的调用. 一方称作服务端(server),一方称为客户端(client): server端 ...

  3. ECharts, PHP, MySQL, Ajax, JQuery 实现前后端数据可视化

    ECharts 下载js代码 工作原理浅析 在项目中引入ECharts 后台处理 数据库端MySQL PHP端 JQuery Ajax处理 ECharts 端处理 前端全部代码 演示结果 总结 最近要 ...

  4. Python 函数参数传递机制.

    learning python,5e中讲到.Python的函数参数传递机制是对象引用. Arguments are passed by assignment (object reference). I ...

  5. 侧滑面板(对viewGroup的自定义)

    额,好吧,最近一直在做侧滑的事情,到目前为止一共是学了三种方法了,一个是直接加第三方开源框架SlidingMenu,第二给是用DrawerLayout,今天这个是用谷歌官方提供的在新的support- ...

  6. 【移动开发】plurals

    不同的语言对数量的语法规定有不同的规则.在英语里面,例如,1是特例.我们会直接写1book,而针对一个以上的我们会在book后加复数形式.这种区别对单数和复数来说是很普遍的,但是其他的语言做了更好的区 ...

  7. 【一天一道LeetCode】#219. Contains Duplicate II

    一天一道LeetCode 本系列文章已全部上传至我的github,地址:ZeeCoder's Github 欢迎大家关注我的新浪微博,我的新浪微博 欢迎转载,转载请注明出处 (一)题目 Given a ...

  8. 【一天一道LeetCode】#299. Bulls and Cows

    一天一道LeetCode 本系列文章已全部上传至我的github,地址:ZeeCoder's Github 欢迎大家关注我的新浪微博,我的新浪微博 欢迎转载,转载请注明出处 (一)题目 You are ...

  9. 最简单的基于librtmp的示例:接收(RTMP保存为FLV)

    ===================================================== 最简单的基于libRTMP的示例系列文章列表: 最简单的基于librtmp的示例:接收(RT ...

  10. 定义范围中的备选方案生成、横向思维、创建WBS、工作包定义、WBS、确认范围过程和实施质量过程的关系、联合应用设计和质量功能展开QFD