一、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章节:
https://developer.amd.com/wp-content/resources/55803_B0_PUB_0_91.pdf
三、CPUID编程实现
在windows下可以调用__cpuid和__cpuidex这两个函数,__cpuid函数在VS2005中就已经开始支持了。 https://msdn.microsoft.com/en-us/library/hskdteyh(v=vs.140).aspx
下面是简单的调用代码,在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\n",cpu[0],cpu[1],cpu[2],cpu[3]);
return 0;
}
参考链接:
《CPUID讲解》
《CPUID指令简单调用》
《CPUID 指令的使用》
《linux cpuid指令,通过CPUID指令获取CPU信息》
若对你有帮助,欢迎点赞、收藏、评论,你的支持就是我的最大动力!!!
同时,阿超为大家准备了丰富的学习资料,欢迎关注公众号“超哥学编程”,即可领取。