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

经验分享 程序员 微信小程序 职场和发展