通过CPUID指令读取处理器信息
一、CPUID简介
CPUID操作码是一个面向x86架构的处理器补充指令,它的名称派生自CPU识别,作用是允许软件通过CPUID指令读取处理器的详细信息。
二、CPUID基本原理
CPUID有很多function号,每个号代表一个功能。 CPUID指令的基本原理是:
-
1.先将function号,写入EAX寄存器; 2.再执行cpuid指令; 3.执行结果会被放入到EAX、EBX、ECX 及 EDX 寄存器,通过读取这4个寄存器,就可以获取到该function号对应的数据信息。
比如: EAX=0,获取CPU的Vendor ID EAX=1,处理器签名(Processor Signiture)和功能(Feature)位
有一部分function号下面,还有子页号,所以这种情况需要将function号和子页号,分别写入EAX和ECX寄存器,然后再执行cpuid指令,执行结果依然通过EAX、EBX、ECX、EDX进行返回。
关于CPUID指令有多少个function,可以参考PPR for amd family 17h model 31h b0.pdf中2.1.13.1 CPUID Instruction Functions章节:
三、CPUID编程实现
在windows下可以调用__cpuid和__cpuidex这两个函数,__cpuid函数在VS2005中就已经开始支持了。
下面是简单的调用代码,在windows下VS不支持x64项目嵌入asm,所以只能编译32位版本。
#include<stdio.h> #include<stdlib.h> // gcc cpuid.c -o cpuid // cl /DWIN32=1 cpuid.c static void cpuInfo(unsigned int cpuinfo[4], unsigned int fn_id) { unsigned int deax,debx,decx,dedx; #ifdef WIN32 __asm { mov eax,fn_id ; 将参数赋值给eax cpuid ; 执行cpuid指令 mov deax,eax ; 将寄存器值赋值给临时变量 mov debx,ebx mov decx,ecx mov dedx,edx } #elif defined(__GNUC__) __asm__ ("cpuid" :"=a"(deax), "=b"(debx), "=c"(decx), "=d"(dedx) :"a"(fn_id)); #endif cpuinfo[0]=deax; cpuinfo[1]=debx; cpuinfo[2]=dedx; cpuinfo[3]=decx; } int main(int argc,char** argv) { unsigned int fn_id = 0; if(argc == 2){ fn_id = atoi(argv[1]); } unsigned int cpu[4]; cpuInfo(cpu,fn_id); printf("%08X-%08X-%08X-%08X ",cpu[0],cpu[1],cpu[2],cpu[3]); return 0; }