1
  
2
  
3
  
4
  
5
  
6
  
7
  
8
  
9
  
10
  
11
  
12
  
13
  
14
  
15
  
16
  
17
  
18
  
19
  
20
  
21
  
22
  
23
  
24
  
25
  
26
  
27
  
28
  
29
  
30
  
31
  
32
  
33
  
34
  
35
  
36
  
37
  
38
  
39
  
40
  
41
  
42
  
43
  
44
  
45
  
46
  
47
  
48
  
49
  
50
  
51
  
52
  
53
  
54
  
55
  
56
  
57
  
58
  
59
  
60
  
61
  
62
  
63
  
64
  
65
  
66
  
67
  
68
  
69
  
70
  
71
  
72
  
73
  
74
  
75
  
76
  
77
  
78
  
79
  
80
  
81
  
82
  
83
  
84
  
85
  
86
  
87
  
88
  
89
  
90
  
91
  
92
  
93
  
94
  
95
  
96
  
97
  
98
  
99
  
100
  
101
  
102
  
103
  
104
  
105
  
106
  
107
  
108
  
109
  
110
  
/* 
|| This file is part of Pike. For copyright information see COPYRIGHT. 
|| Pike is distributed under GPL, LGPL and MPL. See the file COPYING 
|| for more information. 
*/ 
 
#include "global.h" 
#include "pike_cpulib.h" 
#include "svalue.h" 
 
#ifdef HAVE_X86_ASM 
 
#if !defined (CL_X86_ASM_STYLE) && !defined (GCC_X86_ASM_STYLE) 
#error Dont know how to inline assembler with this compiler 
#endif 
 
PMOD_EXPORT void x86_get_cpuid(int oper, INT32 *cpuid_ptr) 
/* eax -> cpuid_ptr[0] 
 * ebx -> cpuid_ptr[1] 
 * edx -> cpuid_ptr[2] 
 * ecx -> cpuid_ptr[3] */ 
{ 
#ifdef HAVE_X86_64_ASM 
#define cpuid_supported 1 
#else  /* HAVE_IA32_ASM */ 
  static int cpuid_supported = 0; 
  if (!cpuid_supported) { 
    int fbits=0; 
#ifdef CL_X86_ASM_STYLE 
    __asm { 
      pushf 
      pop  eax 
      mov  ecx, eax 
      xor  eax, 00200000h 
      push eax 
      popf 
      pushf 
      pop  eax 
      xor  ecx, eax 
      mov  fbits, ecx 
    }; 
#else  /* GCC_X86_ASM_STYLE */ 
    /* Note: gcc swaps the argument order... */ 
    __asm__("pushf\n\t" 
            "pop  %%eax\n\t" 
            "movl %%eax, %%ecx\n\t" 
            "xorl $0x00200000, %%eax\n\t" 
            "push %%eax\n\t" 
            "popf\n\t" 
            "pushf\n\t" 
            "pop  %%eax\n\t" 
            "xorl %%eax, %%ecx\n\t" 
            "movl %%ecx, %0" 
            : "=m" (fbits) 
            : 
            : "cc", "eax", "ecx"); 
#endif 
    if (fbits & 0x00200000) { 
      cpuid_supported = 1; 
    } else { 
      cpuid_supported = -1; 
    } 
  } 
#endif        /* HAVE_IA32_ASM */ 
 
  if (cpuid_supported > 0) { 
#ifdef CL_X86_ASM_STYLE 
    __asm { 
      mov eax, oper; 
      mov edi, cpuid_ptr; 
      cpuid; 
      mov [edi], eax; 
      mov [edi+4], ebx; 
      mov [edi+8], edx; 
      mov [edi+12], ecx; 
    }; 
#else  /* GCC_X86_ASM_STYLE */ 
 
#if SIZEOF_CHAR_P == 4 
    __asm__ __volatile__("pushl %%ebx      \n\t" /* save %ebx */ 
                 "cpuid            \n\t" 
                 "movl %%ebx, %1   \n\t" /* save what cpuid just put in %ebx */ 
                 "popl %%ebx       \n\t" /* restore the old %ebx */ 
                 : "=a"(cpuid_ptr[0]), 
                   "=r"(cpuid_ptr[1]), 
                   "=d"(cpuid_ptr[2]), 
                   "=c"(cpuid_ptr[3]) 
                 : "0"(oper) 
                 : "cc"); 
#else 
    __asm__ __volatile__("push %%rbx      \n\t" /* save %rbx */ 
                 "cpuid            \n\t" 
                 "movl %%ebx, %1   \n\t" /* save what cpuid just put in %ebx */ 
                 "pop %%rbx       \n\t" /* restore the old %rbx */ 
                 : "=a"(cpuid_ptr[0]), 
                   "=r"(cpuid_ptr[1]), 
                   "=d"(cpuid_ptr[2]), 
                   "=c"(cpuid_ptr[3]) 
                 : "0"(oper) 
                 : "cc"); 
#endif 
#endif 
  } else { 
    cpuid_ptr[0] = cpuid_ptr[1] = cpuid_ptr[2] = cpuid_ptr[3] = 0; 
  } 
} 
 
#endif        /* HAVE_IA32_ASM */