通过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;
}
