如何获取多核、多cpu系统中指定cpu的序列号

时间:2021-01-07 00:05:48

如何获取多核、多cpu系统中指定cpu的序列号
作者:SkyJacker
(转贴请保持完整并注明作者和出处)
http://www.cnpack.org
CnPack IV  QQ Group: 130970
2007-01-23

感谢:Passion,Bahamut,早安,空气,SkyJacker...
没有Bahamut的奇思妙点,就没有这片文章。

在多cpu、多核中,会随机的获得不同的序列号.这就为我们根据cpu序列号来制作注册机带来了很大的麻烦。
Windows 2000/xp允许设置进程和线程的亲缘性。换句话说,可以控制哪个 CPU 能够运行某些线程。这称为硬亲缘性。Windows提供了设置亲缘性的函数SetProcessAffinityMask 。使用它可控制获取指定cpu的序列号。

本文分为2部分:

1、如何获得cpu的序列号。
2、如何获取指定cpu或指定cpu核的序列号。

1、如何获得cpu的序列号。

使用cpuid指令来获取。
在调用CPUID之前,EAX中存放的是功能代码。在调用CPUID之后,EAX,EBX,ECX,EDX存放的是CPU的各种特征信息。这些信息也就是我们通常所说的CPU序列号。

mov  eax, 0  //获取制造商信息
  cpuid
    
  mov  eax,  1 //获得CPU的序列号
  cpuid    
    
以下三个函数,可供参考:

  1. function NewCPUID: string;
  2. const
  3. CPUINFO = 'CPU制造商: %S  序列号: %x';
  4. var
  5. s: array[0..19] of Char;
  6. MyCpuID: Integer;
  7. begin
  8. FillChar(s, 20, 0);
  9. asm
  10. push ebx
  11. push ecx
  12. push edx
  13. mov  eax, 0
  14. cpuid
  15. mov  dword  ptr  s[0],    ebx
  16. mov  dword  ptr  s[4],    edx
  17. mov  dword  ptr  s[8],    ecx
  18. mov  eax,  1
  19. cpuid
  20. mov  MyCpuID,  edx
  21. pop edx
  22. pop ecx
  23. pop ebx
  24. end;
  25. Result := Format(CPUINFO, [s, MyCpuID]);
  26. end;
  27. function GetCPUID: TCPUID; assembler; register;
  28. asm
  29. PUSH    EBX         {Save affected register}
  30. PUSH    EDI
  31. MOV     EDI, EAX    [email={@Resukt]{@Resukt[/email]}
  32. MOV     EAX, 1
  33. DW      $A20F       {CPUID Command}
  34. STOSD               {CPUID[1]}
  35. MOV     EAX, EBX
  36. STOSD               {CPUID[2]}
  37. MOV     EAX, ECX
  38. STOSD               {CPUID[3]}
  39. MOV     EAX, EDX
  40. STOSD               {CPUID[4]}
  41. POP     EDI         {Restore registers}
  42. POP     EBX
  43. end;
  44. //获取cpu的序列号:
  45. function GetCnCPUID(): string;
  46. const
  47. CPUINFO = '%.8x-%.8x-%.8x-%.8x';
  48. var
  49. iEax: Integer;
  50. iEbx: Integer;
  51. iEcx: Integer;
  52. iEdx: Integer;
  53. begin
  54. asm
  55. push ebx
  56. push ecx
  57. push edx
  58. mov  eax, 1
  59. DW $A20F//cpuid
  60. mov  iEax, eax
  61. mov  iEbx, ebx
  62. mov  iEcx, ecx
  63. mov  iEdx, edx
  64. pop edx
  65. pop ecx
  66. pop ebx
  67. end;
  68. Result := Format(CPUINFO, [iEax, iEbx, iEcx, iEdx]);
  69. end;

2、如何获取指定cpu或指定cpu核的序列号。

根据Windows可以设置进程和线程的亲缘性的特点,使用SetProcessAffinityMask函数,来控制哪个cpu来运行获取序列号的进程,因此也就获取了指定的cpu的序列号。为了和单cpu兼容,建议总是获取第一个cpu的序列号。

  1. procedure SetCPU(h: THandle;CpuNo: Integer);
  2. //CpuNo:决定了获得第几个cpu内核的第几个序列号。
  3. var
  4. ProcessAffinity: Cardinal;
  5. _SystemAffinity: Cardinal;
  6. begin
  7. GetProcessAffinityMask(h, ProcessAffinity, _SystemAffinity) ;
  8. ProcessAffinity := CpuNo; //this sets the process to only run on CPU 0
  9. //for CPU 1 only use 2 and for CPUs 1 & 2 use 3
  10. SetProcessAffinityMask(h, ProcessAffinity)
  11. end;

使用方法:

SetCPU(GetCurrentProcess,1); //第一个cpu的第一个cpu内核
ShowMessage(GetCnCPUID);

后记:这片文章源于CnPack 群里关于获取多核CPU的ID的问题讨论。
感谢CnPack 群里的所有成员。

BS下'早安空气',竟然说“你在你短暂的生命里花大量的时间去做这个没有价值的事情”-_-!!超级BS、BS、BS……^_^
for I:=0 to 10000000000000000000000000000 do
   BS;

附录“早安空气”语录:
'放弃吧,API不适合你'
'你在你短暂的生命里花大量的时间去做这个没有价值的事情'

参考:http://blog.csdn.net/iseekcode/article/details/5284803