1bad58c9cSBen Gras #include <sys/types.h>
2bad58c9cSBen Gras #include <stdint.h>
3bad58c9cSBen Gras #include <minix/minlib.h>
4bad58c9cSBen Gras #include <minix/cpufeature.h>
5bad58c9cSBen Gras #include <machine/vm.h>
6bad58c9cSBen Gras #include <string.h>
7bad58c9cSBen Gras
_cpufeature(int cpufeature)8bad58c9cSBen Gras int _cpufeature(int cpufeature)
9bad58c9cSBen Gras {
10bad58c9cSBen Gras u32_t eax, ebx, ecx, edx;
11bad58c9cSBen Gras u32_t ef_eax = 0, ef_ebx = 0, ef_ecx = 0, ef_edx = 0;
12bad58c9cSBen Gras unsigned int family, model, stepping;
13bad58c9cSBen Gras int is_intel = 0, is_amd = 0;
14bad58c9cSBen Gras
15bad58c9cSBen Gras eax = ebx = ecx = edx = 0;
16bad58c9cSBen Gras
17bad58c9cSBen Gras /* We assume >= pentium for cpuid */
18bad58c9cSBen Gras eax = 0;
19bad58c9cSBen Gras _cpuid(&eax, &ebx, &ecx, &edx);
20bad58c9cSBen Gras if(eax > 0) {
21bad58c9cSBen Gras char vendor[12];
22bad58c9cSBen Gras memcpy(vendor, &ebx, sizeof(ebx));
23bad58c9cSBen Gras memcpy(vendor+4, &edx, sizeof(edx));
24bad58c9cSBen Gras memcpy(vendor+8, &ecx, sizeof(ecx));
25bad58c9cSBen Gras if(!strncmp(vendor, "GenuineIntel", sizeof(vendor)))
26bad58c9cSBen Gras is_intel = 1;
27bad58c9cSBen Gras if(!strncmp(vendor, "AuthenticAMD", sizeof(vendor)))
28bad58c9cSBen Gras is_amd = 1;
29bad58c9cSBen Gras eax = 1;
30bad58c9cSBen Gras _cpuid(&eax, &ebx, &ecx, &edx);
31bad58c9cSBen Gras } else return 0;
32bad58c9cSBen Gras
33bad58c9cSBen Gras stepping = eax & 0xf;
34bad58c9cSBen Gras model = (eax >> 4) & 0xf;
35bad58c9cSBen Gras
36bad58c9cSBen Gras if(model == 0xf || model == 0x6) {
37bad58c9cSBen Gras model += ((eax >> 16) & 0xf) << 4;
38bad58c9cSBen Gras }
39bad58c9cSBen Gras
40bad58c9cSBen Gras family = (eax >> 8) & 0xf;
41bad58c9cSBen Gras
42bad58c9cSBen Gras if(family == 0xf) {
43bad58c9cSBen Gras family += (eax >> 20) & 0xff;
44bad58c9cSBen Gras }
45bad58c9cSBen Gras
46bad58c9cSBen Gras if(is_amd) {
47bad58c9cSBen Gras ef_eax = 0x80000001;
48bad58c9cSBen Gras _cpuid(&ef_eax, &ef_ebx, &ef_ecx, &ef_edx);
49bad58c9cSBen Gras }
50bad58c9cSBen Gras
51bad58c9cSBen Gras switch(cpufeature) {
52bad58c9cSBen Gras case _CPUF_I386_PSE:
53bad58c9cSBen Gras return edx & CPUID1_EDX_PSE;
54*bf609e10Srlfnb case _CPUF_I386_PAE:
55*bf609e10Srlfnb return edx & CPUID1_EDX_PAE;
56bad58c9cSBen Gras case _CPUF_I386_PGE:
57bad58c9cSBen Gras return edx & CPUID1_EDX_PGE;
58bad58c9cSBen Gras case _CPUF_I386_APIC_ON_CHIP:
59bad58c9cSBen Gras return edx & CPUID1_EDX_APIC_ON_CHIP;
60bad58c9cSBen Gras case _CPUF_I386_TSC:
61bad58c9cSBen Gras return edx & CPUID1_EDX_TSC;
62bad58c9cSBen Gras case _CPUF_I386_FPU:
63bad58c9cSBen Gras return edx & CPUID1_EDX_FPU;
64bad58c9cSBen Gras #define SSE_FULL_EDX (CPUID1_EDX_FXSR | CPUID1_EDX_SSE | CPUID1_EDX_SSE2)
65bad58c9cSBen Gras #define SSE_FULL_ECX (CPUID1_ECX_SSE3 | CPUID1_ECX_SSSE3 | \
66bad58c9cSBen Gras CPUID1_ECX_SSE4_1 | CPUID1_ECX_SSE4_2)
67bad58c9cSBen Gras case _CPUF_I386_SSE1234_12:
68bad58c9cSBen Gras return (edx & SSE_FULL_EDX) == SSE_FULL_EDX &&
69bad58c9cSBen Gras (ecx & SSE_FULL_ECX) == SSE_FULL_ECX;
70bad58c9cSBen Gras case _CPUF_I386_FXSR:
71bad58c9cSBen Gras return edx & CPUID1_EDX_FXSR;
72bad58c9cSBen Gras case _CPUF_I386_SSE:
73bad58c9cSBen Gras return edx & CPUID1_EDX_SSE;
74bad58c9cSBen Gras case _CPUF_I386_SSE2:
75bad58c9cSBen Gras return edx & CPUID1_EDX_SSE2;
76bad58c9cSBen Gras case _CPUF_I386_SSE3:
77bad58c9cSBen Gras return ecx & CPUID1_ECX_SSE3;
78bad58c9cSBen Gras case _CPUF_I386_SSSE3:
79bad58c9cSBen Gras return ecx & CPUID1_ECX_SSSE3;
80bad58c9cSBen Gras case _CPUF_I386_SSE4_1:
81bad58c9cSBen Gras return ecx & CPUID1_ECX_SSE4_1;
82bad58c9cSBen Gras case _CPUF_I386_SSE4_2:
83bad58c9cSBen Gras return ecx & CPUID1_ECX_SSE4_2;
84bad58c9cSBen Gras case _CPUF_I386_HTT:
85bad58c9cSBen Gras return edx & CPUID1_EDX_HTT;
86bad58c9cSBen Gras case _CPUF_I386_HTT_MAX_NUM:
87bad58c9cSBen Gras return (ebx >> 16) & 0xff;
88bad58c9cSBen Gras case _CPUF_I386_SYSENTER:
89bad58c9cSBen Gras if(!is_intel) return 0;
90bad58c9cSBen Gras if(!(edx & CPUID1_EDX_SYSENTER)) return 0;
91bad58c9cSBen Gras if(family == 6 && model < 3 && stepping < 3) return 0;
92bad58c9cSBen Gras return 1;
93bad58c9cSBen Gras case _CPUF_I386_SYSCALL:
94bad58c9cSBen Gras if(!is_amd) return 0;
95bad58c9cSBen Gras if(!(ef_edx & CPUID_EF_EDX_SYSENTER)) return 0;
96bad58c9cSBen Gras return 1;
97bad58c9cSBen Gras }
98bad58c9cSBen Gras
99bad58c9cSBen Gras return 0;
100bad58c9cSBen Gras }
101bad58c9cSBen Gras
102