Chinaunix首页 | 论坛 | 博客
  • 博客访问: 1636971
  • 博文数量: 584
  • 博客积分: 13857
  • 博客等级: 上将
  • 技术积分: 11883
  • 用 户 组: 普通用户
  • 注册时间: 2009-12-16 09:34

分类: WINDOWS

2011-09-20 09:48:27

  1.   CPU ID 指用户计算机当今的信息处理器的信息。 信息包括型号,信息处理器家庭,高速缓存尺寸,钟速度和制造厂codename 等。 通过查询可以知道一些信息:晶体管数,针脚类型,尺寸等。
  2.   1. 什么是cpuid指令
  3.   CPUID指令是intel IA32架构下获得CPU信息的汇编指令,可以得到CPU类型,型号,制造商信息,商标信息,序列号,缓存等一系列CPU相关的东西。
  4.   2. cpuid指令的使用
  5.   cpuid使用eax作为输入参数,eax,ebx,ecx,edx作为输出参数,举个例子:
  6.   __asm
  7.   {
  8.   mov eax, 1
  9.   cpuid
  10.   ...
  11.   }
  12.   以上代码以1为输入参数,执行cpuid后,所有寄存器的值都被返回值填充。针对不同的输入参数eax的值,输出参数的意义都不相同。
  13.   为了更好的在C++中使用cpuid指令,可以使用类对指令进行封装,在类中定义一个专门的函数负责cpuid的执行,他需要一个输入参数。还需要定义四个成员变量存储cpuid执行以后返回来的值。由于这四个寄存器都是32位长的,可以使用unsinged long 类型变量存储。
  14.   typedef unsigned long DWORD
  15.   class CPUID
  16.   {
  17.   public:
  18.   ...
  19.   private:
  20.   void Executecpuid(DWORD eax); // 用来实现cpuid
  21.   DWORD m_eax; // 存储返回的eax
  22.   DWORD m_ebx; // 存储返回的ebx
  23.   DWORD m_ecx; // 存储返回的ecx
  24.   DWORD m_edx; // 存储返回的edx
  25.   ...
  26.   }
  27.   void CPUID::Executecpuid(DWORD veax)
  28.   {
  29.   // 因为嵌入式的汇编代码不能识别 类成员变量
  30.   // 所以定义四个临时变量作为过渡
  31.   DWORD deax;
  32.   DWORD debx;
  33.   DWORD decx;
  34.   DWORD dedx;
  35.   __asm
  36.   {
  37.   mov eax, veax ;将输入参数移入eax
  38.   cpuid ;执行cpuid
  39.   mov deax, eax ;以下四行代码把寄存器中的变量存入临时变量
  40.   mov debx, ebx
  41.   mov decx, ecx
  42.   mov dedx, edx
  43.   }
  44.   m_eax = deax; // 把临时变量中的内容放入类成员变量
  45.   m_ebx = debx;
  46.   m_ecx = decx;
  47.   m_edx = dedx;
  48.   }
  49.   这样就可以通过直接调用Executecupid()函数的方式来执行cpuid指令了,返回值存在类成员变量m_eax, m_ebx, m_ecx和m_edx中。
  50.   3. 获得CPU的制造商信息(Vender ID String)
  51.   把eax = 0作为输入参数,可以得到CPU的制造商信息。
  52.   cpuid指令执行以后,会返回一个12字符的制造商信息,前四个字符的ASC码按低位到高位放在ebx,中间四个放在edx,最后四个字符放在ecx。比如说,对于intel的cpu,会返回一个“GenuineIntel”的字符串,返回值的存储格式为:
  53.   31 23 15 07 00
  54.   EBX| u (75)| n (6E)| e (65)| G (47)
  55.   EDX| I (49)| e (65)| n (6E)| i (69)
  56.   ECX| l (6C)| e (65)| t (74)| n (6E)
  57.   因此可以这样实现他:
  58.   string CPUID::GetVID()
  59.   {
  60.   char cVID[13]; // 字符串,用来存储制造商信息
  61.   memset(cVID, 0, 13); // 把数组清0
  62.   Executecpuid(0); // 执行cpuid指令,使用输入参数 eax = 0
  63.   memcpy(cVID, &m_ebx, 4); // 复制前四个字符到数组
  64.   memcpy(cVID+4, &m_edx, 4); // 复制中间四个字符到数组
  65.   memcpy(cVID+8, &m_ecx, 4); // 复制最后四个字符到数组
  66.   return string(cVID); // 以string的形式返回
  67.   }
  68.   4. 获得CPU商标信息(Brand String)
  69.   在我的电脑上点击右键,选择属性,可以在窗口的下面看到一条CPU的信息,这就是CPU的商标字符串。CPU的商标字符串也是通过cpuid得到的。由于商标的字符串很长(48个字符),所以不能在一次cpuid指令执行时全部得到,所以intel把它分成了3个操作,eax的输入参数分别是0x80000002,0x80000003,0x80000004,每次返回的16个字符,按照从低位到高位的顺序依次放在eax, ebx, ecx, edx。因此,可以用循环的方式,每次执行完以后保存结果,然后执行下一次cpuid。
  70.   string CPUID::GetBrand()
  71.   {
  72.   const DWORD BRANDID = 0x80000002; // 从0x80000002开始,到0x80000004结束
  73.   char cBrand[49]; // 用来存储商标字符串,48个字符
  74.   memset(cBrand, 0, 49); // 初始化为0
  75.   for (DWORD i = 0; i < 3; i++) // 依次执行3个指令
  76.   {
  77.   Executecpuid(BRANDID + i);
  78.   memcpy(cBrand + i*16, &m_eax, 16); // 每次执行结束后,保存四个寄存器里的asc码到数组
  79.   } // 由于在内存中,m_eax, m_ebx, m_ecx, m_edx是连续排列
  80.   // 所以可以直接以内存copy的方式进行保存
  81.   return string(cBrand); // 以string的形式返回
  82.   }
  83.   5. 检测CPU特性(CPU feature)
  84.   我98年初买第一台电脑的时候,CPU能支持MMX就很了不起了。现在的intel CPU,台式机的好点的都支持Hyper-Threading了,移动的要支持Speed Sted。这些都是CPU的特性。CPU的特性可以通过cpuid获得,参数是eax = 1,返回值放在edx和ecx,通过验证edx或者ecx的某一个bit,可以获得CPU的一个特性是否被支持。比如说,edx的bit 32代表是否支持MMX,edx的bit 28代表是否支持Hyper-Threading,ecx的bit 7代表是否支持speed sted。下面就是获得CPU特性的例子:
  85.   bool CPUID::IsHyperThreading() // 判断是否支持hyper-threading
  86.   {
  87.   Executecpuid(1); // 执行cpuid指令,使用输入参数 eax = 1
  88.   return m_edx & (1 < <28); // 返回edx的bit 28
  89.   }
  90.   bool CPUID::IsEST() // 判断是否支持speed step
  91.   {
  92.   Executecpuid(1); // 执行cpuid指令,使用输入参数 eax = 1
  93.   return m_ecx & (1 < <7); // 返回ecx的bit 7
  94.   }
  95.   bool CPUID::IsMMX() // 判断是否支持MMX
  96.   {
  97.   Executecpuid(1); // 执行cpuid指令,使用输入参数 eax = 1
  98.   return m_edx & (1 < <23); // 返回edx的bit 23
  99.   }
  100.   CPU的特性还有很多,这只是平时我们听到比较多的三个,更多的特性请参考intel的资料。
  101.   6. 获得CPU的缓存(cache)
  102.   缓存,就是CACHE,已经成为判断CPU性能的一项大指标。缓存信息包括:第几级缓存(level),缓存大小(size),通道数(way),吞吐量(line size)。因此可以使用一个结构体来存储缓存信息。
  103.   struct CacheInfo
  104.   {
  105.   int level; // 第几级缓存
  106.   int size; // 缓存大小,单位KB
  107.   int way; // 通道数
  108.   int linesize; // 吞吐量
  109.   CacheInfo() // 构造函数
  110.   {
  111.   level = 0;
  112.   size = 0;
  113.   way = 0;
  114.   linesize = 0;
  115.   }
  116.   CacheInfo(int clevel, int csize, int cway, int clinesize) // 构造函数
  117.   {
  118.   level = clevel;
  119.   size = csize;
  120.   way = cway;
  121.   linesize = clinesize;
  122.   }
  123.   };
  124.   缓存信息可以通过eax = 2的cpuid来得到(得到的不光有cache信息,还有其他的一些信息),返回值在eax(高24位), ebx, ecx和edx,总共15个BYTE的信息,每个BYTE的值不同,代表的意义也不同,所以需要用一个哈希表存储各种不同BYTE的定义,可以定义一个map类型的类成员存储这些资料。我把资料上和缓存有关的信息存储如下:
  125.   m_cache[0x06] = CacheInfo(1, 8, 4, 32);
  126.   m_cache[0x08] = CacheInfo(1, 16, 4, 32);
  127.   m_cache[0x0a] = CacheInfo(1, 8, 2, 32);
  128.   m_cache[0x0c] = CacheInfo(1, 16, 4, 32);
  129.   m_cache[0x2c] = CacheInfo(1, 32, 8, 64);
  130.   m_cache[0x30] = CacheInfo(1, 32, 8, 64);
  131.   m_cache[0x60] = CacheInfo(1, 16, 8, 64);
  132.   m_cache[0x66] = CacheInfo(1, 8, 4, 64);
  133.   m_cache[0x67] = CacheInfo(1, 16, 4, 64);
  134.   m_cache[0x68] = CacheInfo(1, 32, 4, 64);
  135.   m_cache[0x39] = CacheInfo(2, 128, 4, 64);
  136.   m_cache[0x3b] = CacheInfo(2, 128, 2, 64);
  137.   m_cache[0x3c] = CacheInfo(2, 256, 4, 64);
  138.   m_cache[0x41] = CacheInfo(2, 128, 4, 32);
  139.   m_cache[0x42] = CacheInfo(2, 256, 4, 32);
  140.   m_cache[0x43] = CacheInfo(2, 512, 4, 32);
  141.   m_cache[0x44] = CacheInfo(2, 1024, 4, 32);
  142.   m_cache[0x45] = CacheInfo(2, 2048, 4, 32);
  143.   m_cache[0x79] = CacheInfo(2, 128, 8, 64);
  144.   m_cache[0x7a] = CacheInfo(2, 256, 8, 64);
  145.   m_cache[0x7b] = CacheInfo(2, 512, 8, 64);
  146.   m_cache[0x7c] = CacheInfo(2, 1024, 8, 64);
  147.   m_cache[0x82] = CacheInfo(2, 256, 8, 32);
  148.   m_cache[0x83] = CacheInfo(2, 512, 8, 32);
  149.   m_cache[0x84] = CacheInfo(2, 1024, 8, 32);
  150.   m_cache[0x85] = CacheInfo(2, 2048, 8, 32);
  151.   m_cache[0x86] = CacheInfo(2, 512, 4, 64);
  152.   m_cache[0x87] = CacheInfo(2, 1024, 8, 64);
  153.   m_cache[0x22] = CacheInfo(3, 512, 4, 64);
  154.   m_cache[0x23] = CacheInfo(3, 1024, 8, 64);
  155.   m_cache[0x25] = CacheInfo(3, 2048, 8, 64);
  156.   m_cache[0x29] = CacheInfo(3, 4096, 8, 64);
  157.   m_cache是类成员,定义如下:
  158.   map<int, CacheInfo> m_cache; // Cache information table
  159.   在得到返回值以后,只需要遍历每一个BYTE的值,找到在m_cache中存在的元素,就可以得到cache信息了。代码如下:
  160.   typedef unsigned char BYTE;
  161.   DWORD CPUID::GetCacheInfo(CacheInfo& L1, CacheInfo& L2, CacheInfo& L3)
  162.   {
  163.   BYTE cValues[16]; // 存储返回的16个byte值
  164.   DWORD result = 0; // 记录发现的缓存数量
  165.   Executecpuid(2); // 执行cpuid,参数为eax = 2
  166.   memcpy(cValues, &m_eax, 16); // 把m_eax, m_ebx, m_ecx和m_edx存储到cValue
  167.   for (int i = 1; i < 16; i++) // 开始遍历,注意eax的第一个byte没有意义,需要跳过
  168.   {
  169.   if (m_cache.find(cValues[i]) != m_cache.end()) // 从表中查找此信息是否代表缓存
  170.   {
  171.   switch (m_cache[cValues[i]].level) // 对号入座,保存缓存信息
  172.   {
  173.   case 1: // L1 cache
  174.   L1 = m_cache[cValues[i]];
  175.   break;
  176.   case 2: // L2 cache
  177.   L2 = m_cache[cValues[i]];
  178.   break;
  179.   case 3: // L3 cache
  180.   L3 = m_cache[cValues[i]];
  181.   break;
  182.   default:
  183.   break;
  184.   }
  185.   result++;
  186.   }
  187.   }
  188.   return result;
  189.   }
  190.   7. 获得CPU的序列号
  191.   序列号无处不在!!CPU的序列号用一个96bit的串表示,格式是连续的6个WORD值:XXXX-XXXX-XXXX-XXX-XXXX-XXXX。WORD是16个bit长的数据,可以用unsigned short模拟:
  192.   typedef unsigned short WORD;
  193.   获得序列号需要两个步骤,首先用eax = 1做参数,返回的eax中存储序列号的高两个WORD。用eax = 3做参数,返回ecx和edx按从低位到高位的顺序存储前4个WORD。实现如下:
  194.   bool CPUID::GetSerialNumber(SerialNumber& serial)
  195.   {
  196.   Executecpuid(1); // 执行cpuid,参数为 eax = 1
  197.   bool isSupport = m_edx & (1<<18); // edx是否为1代表CPU是否存在序列号
  198.   if (false == isSupport) // 不支持,返回false
  199.   {
  200.   return false;
  201.   }
  202.   memcpy(&serial.nibble[4], &m_eax, 4); // eax为最高位的两个WORD
  203.   Executecpuid(3); // 执行cpuid,参数为 eax = 3
  204.   memcpy(&serial.nibble[0], &m_ecx, 8); // ecx 和 edx为低位的4个WORD
  205.   return true;
  206.   }
阅读(1248) | 评论(0) | 转发(1) |
给主人留下些什么吧!~~