通过CPUID指令读取处理器信息

本文介绍了CPUID指令的基本原理,包括如何通过function号获取处理器VendorID、签名和功能位,以及在Windows下的编程实现。通过__cpuid和__cpuidex函数展示了如何调用CPUID获取详细硬件信息。
摘要由CSDN通过智能技术生成

一、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信息》



若对你有帮助,欢迎点赞、收藏、评论,你的支持就是我的最大动力!!!

同时,阿超为大家准备了丰富的学习资料,欢迎关注公众号“超哥学编程”,即可领取。

在这里插入图片描述

### 回答1: 在Linux C中读取CPUID需要使用内嵌汇编来执行对应的CPUID指令。下面是一个简单的示例代码: ```c #include <stdio.h> void cpuid(unsigned int leaf, unsigned int subleaf, unsigned int *eax, unsigned int *ebx, unsigned int *ecx, unsigned int *edx) { asm volatile("cpuid" : "=a" (*eax), "=b" (*ebx), "=c" (*ecx), "=d" (*edx) : "a" (leaf), "c" (subleaf)); } int main() { unsigned int eax, ebx, ecx, edx; // 读取基本信息(leaf = 0x00) cpuid(0x00, 0x00, &eax, &ebx, &ecx, &edx); printf("Vendor ID: %.4s %.4s %.4s\n", (char *)&ebx, (char *)&edx, (char *)&ecx); // 读取扩展信息(leaf = 0x80000000) cpuid(0x80000000, 0x00, &eax, &ebx, &ecx, &edx); printf("CPU Manufacturer ID: %.12s\n", (char *)&ebx); return 0; } ``` 上述代码中的`cpuid`函数使用了内嵌汇编,通过将需要读取的寄存器和对应的标识符作为参数传递给`asm volatile`语句,然后使用`cpuid`指令来执行CPUID操作。`main`函数中先使用`cpuid`函数读取CPU的厂商信息,然后读取CPU的制造商信息。 这只是一个简单的示例代码,实际情况下可以根据需要读取更多的CPU信息,并对获取到的结果进行解析和处理。希望对你有所帮助! ### 回答2: 在Linux C中,要读取CPUID信息,可以使用cpuid指令来实现。CPUID指令是由x86架构的处理器提供的一条特殊指令,用于获取CPU的相关信息。 在C语言中,可以使用内联汇编的方式来执行CPUID指令并获取返回值。以下是一个实例: ```c #include <stdio.h> void getCPUID(unsigned int leaf, unsigned int* eax, unsigned int* ebx, unsigned int* ecx, unsigned int* edx) { asm volatile ( "cpuid" : "=a" (*eax), "=b" (*ebx), "=c" (*ecx), "=d" (*edx) : "a" (leaf) ); } int main() { unsigned int eax, ebx, ecx, edx; // 读取CPUID信息 getCPUID(1, &eax, &ebx, &ecx, &edx); printf("EAX: 0x%x\n", eax); printf("EBX: 0x%x\n", ebx); printf("ECX: 0x%x\n", ecx); printf("EDX: 0x%x\n", edx); return 0; } ``` 在上述代码中,我们定义了一个getCPUID函数,通过执行CPUID指令获取返回值,并将返回值保存到传入的参数中。然后在main函数中调用getCPUID函数,并打印获取到的返回值。 注意,内联汇编使用的是汇编指令,需要在编译器支持和目标平台上正确执行。此示例适用于x86架构的处理器。如果在其他平台上运行,可能需要进行相应的调整。 通过执行以上代码,我们就可以在Linux C中读取CPUID信息了。 ### 回答3: 在Linux下使用C语言读取CPUID可以通过调用x86架构提供的内嵌汇编来实现。首先,需要定义一个函数,可以使用__asm__关键字来嵌入汇编指令。以下是一个示例代码: ```c #include <stdio.h> void cpuid(int code, unsigned int *a, unsigned int *b, unsigned int *c, unsigned int *d) { __asm__( "cpuid;" : "=a"(*a), "=b"(*b), "=c"(*c), "=d"(*d) : "a"(code) ); } int main() { unsigned int a, b, c, d; cpuid(0, &a, &b, &c, &d); printf("EAX: 0x%08x\n", a); printf("EBX: 0x%08x\n", b); printf("ECX: 0x%08x\n", c); printf("EDX: 0x%08x\n", d); return 0; } ``` 在上述代码中,cpuid函数接收一个参数code,用于设置要读取的寄存器,然后将结果存储在传入的指针变量中。通过调用cpuid(0, &a, &b, &c, &d)来读取CPUID中的第一个返回码,并将返回结果打印出来。 请注意,此代码仅适用于x86架构的处理器,如果要在其他架构上运行,需要根据具体架构更改代码。此外,由于使用了内嵌汇编,需要确保正确的编译选项以启用对内嵌汇编的支持。 希望以上回答能对您有所帮助!
评论 14
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

百里杨

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值