CPUIDopcode는 x86 아키텍처를 위한 프로세서 기계 명령어이다. (CPUID는 CPU IDentification에서 비롯한다.) 인텔이 펜티엄과 SL 강화 486 프로세서를 내세운 1993년에 도입하였다.[1]
CPUID opcode를 사용하여 소프트웨어는 프로세서 종류와 MMX/SSE와 같은 기능들을 결정할 수 있다. CPUID opcode는 0FA2h이며 EAX 레지스터 값은 어떠한 정보를 반환할지를 결정한다.
CPUID 명령을 일반적으로 사용하기까지 프로그래머들은 프로세서 메이커와 모델을 결정하기 위하여 CPU 동작에 사소한 차이를 불러일으킬 수 있는 비밀스러운 기계어를 기록하여야 했다.[2][3]
CPUID 호출
어셈블리어에서 CPUID 명령은 CPUID가 EAX 레지스터를 그대로 사용하므로 매개변수를 이용하지 않는다. EAX 레지스터는 어떠한 정보를 반환할지를 지정하는 값과 함께 로드되어야 한다. CPUID는 EAX = 0으로 먼저 호출되어야 하는데, 이는 CPU가 지원하는 가장 높은 호출 변수를 반환할 것이기에 그러한 것이다. 확장 정보를 가져오려면 CPUID는 비트 31의 EAX 집합을 사용하여 호출되어야 한다. 가장 높은 확장 명령 호출 변수를 결정하려면 EAX = 80000000h를 사용하여 CPUID를 호출하면 된다.
EAX=0: 제조업체 ID 가져오기
이것은 CPU의 제조업체 ID 문자열을 반환한다. 이 문자열은 EBX, EDX, ECX 순으로 저장된 12자리 아스키 문자열이다. 가장 높은 기본 호출 변수는 EAX로 반환된다.
이를테면 GenuineIntel 프로세서 값은 EBX로 0x756e6547를, EDX로 0x49656e69를 ECX로 0x6c65746e로 반환된다.
.section.datas0:.string"최대로 지원하는 표준 명령어 수: %i\n"s1:.string"업체 ID: %s\n".text.globalmain.typemain,@functionmain:pushrbpmovrbp,rspsubrsp,16xoreax,eaxmov[rsp+12],eaxcpuidmov[rsp],ebxmov[rsp+4],edxmov[rsp+8],ecxmovedi,$s0movesi,eaxxoreax,eaxcallprintfmovrdi,$s1movrsi,rspxoreax,eaxcallprintfxoreax,eaxmovrsp,rbppoprbpret
.section.datas0:.string"프로세서 브랜드 문자열: %s\n".text.globalmain.typemain,@functionmain:pushq%rbpmovq%rsp,%rbpsubq$48,%rspmovl$0x80000002,%eaxcpuidmovl%eax,(%rsp)movl%ebx,4(%rsp)movl%ecx,8(%rsp)movl%edx,12(%rsp)movl$0x80000003,%eaxcpuidmovl%eax,16(%rsp)movl%ebx,20(%rsp)movl%ecx,24(%rsp)movl%edx,28(%rsp)movl$0x80000004,%eaxcpuidmovl%eax,32(%rsp)movl%ebx,36(%rsp)movl%ecx,40(%rsp)movl%edx,44(%rsp)movl$s0,%edimovq%rsp,%rsisubl%eax,%eaxcallprintfsubl%eax,%eaxmovq%rbp,%rsppopq%rbpret
7:0 - 프로세서가 지원하는 물리적 주소 크기(MAXPHYADDR 이라고 하며 최대 52)
15:8 - 프로세서가 지원하는 선형 주소 너비
31:16 - 예약된 영역
다른 언어로부터 ID 접근
이 정보는 다른 언어로부터 접근하기 쉽다. 이를테면 아래의 C++ (gcc) 코드는 cpuid가 반환하는 처음 다섯 개의 값을 인쇄한다.
#include<iostream>intmain(intargc,char**argv){intb;for(inta=0;a<5;a++){asm("mov %1, %%eax; "// a → eax"cpuid;""mov %%eax, %0;"// eeax → b:"=r"(b)/* 출력 */:"r"(a)/* 입력 */:"%eax"/* 레지스터 표시 */);std::cout<<"코드 "<<a<<"은(는) 다음을 제공한다: "<<b<<std::endl;}return0;}
마이크로소프트 비주얼 C 컴파일러는 함수 __cpuid()를 내장하고 있으므로 cpuid 명령은 인라인 어셈블리를 사용하지 않아도 추가할 수 있다. x64 버전의 MSVC가 인라인 어셈블리를 아예 허용하지 않으므로 다루기 쉽다. MSVC의 경우 다음과 같이 하면 된다:
#include<iostream>#include<intrin.h>intmain(intargc,char**argv){intb[4];for(inta=0;a<5;a++){__cpuid(b,a);std::cout<<"코드 "<<a<<"은(는) 다음을 제공한다: "<<b[0]<<std::endl;}return0;}