Skip to content

Latest commit

 

History

History
389 lines (353 loc) · 10.9 KB

SDL_cpuinfo.c

File metadata and controls

389 lines (353 loc) · 10.9 KB
 
1
2
/*
SDL - Simple DirectMedia Layer
Jan 4, 2004
Jan 4, 2004
3
Copyright (C) 1997-2004 Sam Lantinga
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
This library is free software; you can redistribute it and/or
modify it under the terms of the GNU Library General Public
License as published by the Free Software Foundation; either
version 2 of the License, or (at your option) any later version.
This library is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
Library General Public License for more details.
You should have received a copy of the GNU Library General Public
License along with this library; if not, write to the Free
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
Sam Lantinga
slouken@libsdl.org
*/
#ifdef SAVE_RCSID
static char rcsid =
"@(#) $Id$";
#endif
/* CPU feature detection for SDL */
Jan 29, 2004
Jan 29, 2004
30
31
32
33
34
35
#ifdef unix /* FIXME: Better setjmp detection? */
#define USE_SETJMP
#include <signal.h>
#include <setjmp.h>
#endif
Nov 24, 2003
Nov 24, 2003
37
#include "SDL_cpuinfo.h"
Jan 6, 2004
Jan 6, 2004
39
40
41
42
#ifdef MACOSX
#include <sys/sysctl.h> /* For AltiVec check */
#endif
Nov 24, 2003
Nov 24, 2003
43
44
#define CPU_HAS_RDTSC 0x00000001
#define CPU_HAS_MMX 0x00000002
Jan 24, 2004
Jan 24, 2004
45
46
47
48
49
50
#define CPU_HAS_MMXEXT 0x00000004
#define CPU_HAS_3DNOW 0x00000010
#define CPU_HAS_3DNOWEXT 0x00000020
#define CPU_HAS_SSE 0x00000040
#define CPU_HAS_SSE2 0x00000080
#define CPU_HAS_ALTIVEC 0x00000100
Jan 29, 2004
Jan 29, 2004
52
53
54
55
56
57
58
59
60
61
62
#ifdef USE_SETJMP
/* This is the brute force way of detecting instruction sets...
the idea is borrowed from the libmpeg2 library - thanks!
*/
static jmp_buf jmpbuf;
static void illegal_instruction(int sig)
{
longjmp(jmpbuf, 1);
}
#endif // USE_SETJMP
Nov 24, 2003
Nov 24, 2003
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
static __inline__ int CPU_haveCPUID()
{
int has_CPUID = 0;
#if defined(__GNUC__) && defined(i386)
__asm__ (
" pushfl # Get original EFLAGS \n"
" popl %%eax \n"
" movl %%eax,%%ecx \n"
" xorl $0x200000,%%eax # Flip ID bit in EFLAGS \n"
" pushl %%eax # Save new EFLAGS value on stack \n"
" popfl # Replace current EFLAGS value \n"
" pushfl # Get new EFLAGS \n"
" popl %%eax # Store new EFLAGS in EAX \n"
" xorl %%ecx,%%eax # Can not toggle ID bit, \n"
" jz 1f # Processor=80486 \n"
" movl $1,%0 # We have CPUID support \n"
"1: \n"
Jan 17, 2004
Jan 17, 2004
80
: "=m" (has_CPUID)
Nov 24, 2003
Nov 24, 2003
81
82
83
:
: "%eax", "%ecx"
);
Apr 11, 2004
Apr 11, 2004
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
#elif defined(__GNUC__) && defined(__x86_64__)
/* Technically, if this is being compiled under __x86_64__ then it has
CPUid by definition. But it's nice to be able to prove it. :) */
__asm__ (
" pushfq # Get original EFLAGS \n"
" popq %%rax \n"
" movq %%rax,%%rcx \n"
" xorl $0x200000,%%eax # Flip ID bit in EFLAGS \n"
" pushq %%rax # Save new EFLAGS value on stack \n"
" popfq # Replace current EFLAGS value \n"
" pushfq # Get new EFLAGS \n"
" popq %%rax # Store new EFLAGS in EAX \n"
" xorl %%ecx,%%eax # Can not toggle ID bit, \n"
" jz 1f # Processor=80486 \n"
" movl $1,%0 # We have CPUID support \n"
"1: \n"
: "=m" (has_CPUID)
:
: "%rax", "%rcx"
);
Sep 29, 2005
Sep 29, 2005
104
#elif defined(_MSC_VER) && (defined(_M_IX86) || defined(_X86_))
Nov 24, 2003
Nov 24, 2003
105
__asm {
Nov 24, 2003
Nov 24, 2003
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
pushfd ; Get original EFLAGS
pop eax
mov ecx, eax
xor eax, 200000h ; Flip ID bit in EFLAGS
push eax ; Save new EFLAGS value on stack
popfd ; Replace current EFLAGS value
pushfd ; Get new EFLAGS
pop eax ; Store new EFLAGS in EAX
xor eax, ecx ; Can not toggle ID bit,
jz done ; Processor=80486
mov has_CPUID,1 ; We have CPUID support
done:
}
#endif
return has_CPUID;
}
static __inline__ int CPU_getCPUIDFeatures()
{
int features = 0;
Apr 11, 2004
Apr 11, 2004
126
#if defined(__GNUC__) && ( defined(i386) || defined(__x86_64__) )
Nov 24, 2003
Nov 24, 2003
127
__asm__ (
Jan 24, 2004
Jan 24, 2004
128
" movl %%ebx,%%edi\n"
Nov 24, 2003
Nov 24, 2003
129
130
131
132
133
134
135
136
137
" xorl %%eax,%%eax # Set up for CPUID instruction \n"
" cpuid # Get and save vendor ID \n"
" cmpl $1,%%eax # Make sure 1 is valid input for CPUID\n"
" jl 1f # We dont have the CPUID instruction\n"
" xorl %%eax,%%eax \n"
" incl %%eax \n"
" cpuid # Get family/model/stepping/features\n"
" movl %%edx,%0 \n"
"1: \n"
Jan 24, 2004
Jan 24, 2004
138
" movl %%edi,%%ebx\n"
Jan 17, 2004
Jan 17, 2004
139
: "=m" (features)
Nov 24, 2003
Nov 24, 2003
140
:
May 16, 2004
May 16, 2004
141
: "%eax", "%ecx", "%edx", "%edi"
Nov 24, 2003
Nov 24, 2003
142
);
Sep 29, 2005
Sep 29, 2005
143
#elif defined(_MSC_VER) && (defined(_M_IX86) || defined(_X86_))
Nov 24, 2003
Nov 24, 2003
144
__asm {
Nov 24, 2003
Nov 24, 2003
145
146
147
148
149
150
151
152
153
154
155
156
157
158
xor eax, eax ; Set up for CPUID instruction
cpuid ; Get and save vendor ID
cmp eax, 1 ; Make sure 1 is valid input for CPUID
jl done ; We dont have the CPUID instruction
xor eax, eax
inc eax
cpuid ; Get family/model/stepping/features
mov features, edx
done:
}
#endif
return features;
}
Jan 24, 2004
Jan 24, 2004
159
static __inline__ int CPU_getCPUIDFeaturesExt()
Nov 24, 2003
Nov 24, 2003
160
{
Jan 24, 2004
Jan 24, 2004
161
int features = 0;
Apr 11, 2004
Apr 11, 2004
162
#if defined(__GNUC__) && (defined(i386) || defined (__x86_64__) )
Nov 24, 2003
Nov 24, 2003
163
__asm__ (
Jan 24, 2004
Jan 24, 2004
164
" movl %%ebx,%%edi\n"
Nov 24, 2003
Nov 24, 2003
165
166
167
" movl $0x80000000,%%eax # Query for extended functions \n"
" cpuid # Get extended function limit \n"
" cmpl $0x80000001,%%eax \n"
Jan 24, 2004
Jan 24, 2004
168
" jl 1f # Nope, we dont have function 800000001h\n"
Nov 24, 2003
Nov 24, 2003
169
170
" movl $0x80000001,%%eax # Setup extended function 800000001h\n"
" cpuid # and get the information \n"
Jan 24, 2004
Jan 24, 2004
171
" movl %%edx,%0 \n"
Nov 24, 2003
Nov 24, 2003
172
"1: \n"
Jan 24, 2004
Jan 24, 2004
173
174
" movl %%edi,%%ebx\n"
: "=m" (features)
Nov 24, 2003
Nov 24, 2003
175
:
May 16, 2004
May 16, 2004
176
: "%eax", "%ecx", "%edx", "%edi"
Nov 24, 2003
Nov 24, 2003
177
);
Sep 29, 2005
Sep 29, 2005
178
#elif defined(_MSC_VER) && (defined(_M_IX86) || defined(_X86_))
Nov 24, 2003
Nov 24, 2003
179
__asm {
Nov 24, 2003
Nov 24, 2003
180
181
182
mov eax,80000000h ; Query for extended functions
cpuid ; Get extended function limit
cmp eax,80000001h
Jan 24, 2004
Jan 24, 2004
183
jl done ; Nope, we dont have function 800000001h
Nov 24, 2003
Nov 24, 2003
184
185
mov eax,80000001h ; Setup extended function 800000001h
cpuid ; and get the information
Jan 24, 2004
Jan 24, 2004
186
mov features,edx
Nov 24, 2003
Nov 24, 2003
187
188
189
done:
}
#endif
Jan 24, 2004
Jan 24, 2004
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
return features;
}
static __inline__ int CPU_haveRDTSC()
{
if ( CPU_haveCPUID() ) {
return (CPU_getCPUIDFeatures() & 0x00000010);
}
return 0;
}
static __inline__ int CPU_haveMMX()
{
if ( CPU_haveCPUID() ) {
return (CPU_getCPUIDFeatures() & 0x00800000);
}
return 0;
}
static __inline__ int CPU_haveMMXExt()
{
if ( CPU_haveCPUID() ) {
return (CPU_getCPUIDFeaturesExt() & 0x00400000);
}
return 0;
}
static __inline__ int CPU_have3DNow()
{
if ( CPU_haveCPUID() ) {
return (CPU_getCPUIDFeaturesExt() & 0x80000000);
}
return 0;
}
static __inline__ int CPU_have3DNowExt()
{
if ( CPU_haveCPUID() ) {
return (CPU_getCPUIDFeaturesExt() & 0x40000000);
}
return 0;
Nov 24, 2003
Nov 24, 2003
231
232
233
234
235
236
237
238
239
}
static __inline__ int CPU_haveSSE()
{
if ( CPU_haveCPUID() ) {
return (CPU_getCPUIDFeatures() & 0x02000000);
}
return 0;
}
Jan 24, 2004
Jan 24, 2004
241
242
243
244
245
246
247
248
static __inline__ int CPU_haveSSE2()
{
if ( CPU_haveCPUID() ) {
return (CPU_getCPUIDFeatures() & 0x04000000);
}
return 0;
}
Jan 6, 2004
Jan 6, 2004
249
250
static __inline__ int CPU_haveAltiVec()
{
Jan 31, 2004
Jan 31, 2004
251
volatile int altivec = 0;
Jan 6, 2004
Jan 6, 2004
252
253
254
255
256
257
#ifdef MACOSX
int selectors[2] = { CTL_HW, HW_VECTORUNIT };
int hasVectorUnit = 0;
size_t length = sizeof(hasVectorUnit);
int error = sysctl(selectors, 2, &hasVectorUnit, &length, NULL, 0);
if( 0 == error )
Jan 29, 2004
Jan 29, 2004
258
altivec = (hasVectorUnit != 0);
Jan 29, 2004
Jan 29, 2004
259
#elif defined(USE_SETJMP) && defined(GCC_ALTIVEC)
Jan 29, 2004
Jan 29, 2004
260
261
262
263
264
265
266
267
268
269
void (*handler)(int sig);
handler = signal(SIGILL, illegal_instruction);
if ( setjmp(jmpbuf) == 0 ) {
asm volatile ("mtspr 256, %0\n\t"
"vand %%v0, %%v0, %%v0"
:
: "r" (-1));
altivec = 1;
}
signal(SIGILL, handler);
Jan 6, 2004
Jan 6, 2004
270
#endif
Jan 29, 2004
Jan 29, 2004
271
return altivec;
Jan 6, 2004
Jan 6, 2004
272
273
}
274
275
276
277
278
279
static Uint32 SDL_CPUFeatures = 0xFFFFFFFF;
static Uint32 SDL_GetCPUFeatures()
{
if ( SDL_CPUFeatures == 0xFFFFFFFF ) {
SDL_CPUFeatures = 0;
Nov 24, 2003
Nov 24, 2003
280
281
282
if ( CPU_haveRDTSC() ) {
SDL_CPUFeatures |= CPU_HAS_RDTSC;
}
283
284
285
if ( CPU_haveMMX() ) {
SDL_CPUFeatures |= CPU_HAS_MMX;
}
Jan 24, 2004
Jan 24, 2004
286
287
288
if ( CPU_haveMMXExt() ) {
SDL_CPUFeatures |= CPU_HAS_MMXEXT;
}
289
290
291
if ( CPU_have3DNow() ) {
SDL_CPUFeatures |= CPU_HAS_3DNOW;
}
Jan 24, 2004
Jan 24, 2004
292
293
294
if ( CPU_have3DNowExt() ) {
SDL_CPUFeatures |= CPU_HAS_3DNOWEXT;
}
295
296
297
if ( CPU_haveSSE() ) {
SDL_CPUFeatures |= CPU_HAS_SSE;
}
Jan 24, 2004
Jan 24, 2004
298
299
300
if ( CPU_haveSSE2() ) {
SDL_CPUFeatures |= CPU_HAS_SSE2;
}
Jan 6, 2004
Jan 6, 2004
301
302
303
if ( CPU_haveAltiVec() ) {
SDL_CPUFeatures |= CPU_HAS_ALTIVEC;
}
304
305
306
307
}
return SDL_CPUFeatures;
}
Nov 24, 2003
Nov 24, 2003
308
309
310
311
312
313
314
315
SDL_bool SDL_HasRDTSC()
{
if ( SDL_GetCPUFeatures() & CPU_HAS_RDTSC ) {
return SDL_TRUE;
}
return SDL_FALSE;
}
316
317
318
319
320
321
322
323
SDL_bool SDL_HasMMX()
{
if ( SDL_GetCPUFeatures() & CPU_HAS_MMX ) {
return SDL_TRUE;
}
return SDL_FALSE;
}
Feb 10, 2004
Feb 10, 2004
324
SDL_bool SDL_HasMMXExt()
Feb 10, 2004
Feb 10, 2004
326
if ( SDL_GetCPUFeatures() & CPU_HAS_MMXEXT ) {
327
328
329
330
331
return SDL_TRUE;
}
return SDL_FALSE;
}
Feb 10, 2004
Feb 10, 2004
332
SDL_bool SDL_Has3DNow()
Feb 10, 2004
Feb 10, 2004
334
if ( SDL_GetCPUFeatures() & CPU_HAS_3DNOW ) {
335
336
337
338
339
return SDL_TRUE;
}
return SDL_FALSE;
}
Feb 10, 2004
Feb 10, 2004
340
SDL_bool SDL_Has3DNowExt()
Jan 6, 2004
Jan 6, 2004
341
{
Feb 10, 2004
Feb 10, 2004
342
if ( SDL_GetCPUFeatures() & CPU_HAS_3DNOWEXT ) {
Jan 6, 2004
Jan 6, 2004
343
344
345
346
347
return SDL_TRUE;
}
return SDL_FALSE;
}
Feb 10, 2004
Feb 10, 2004
348
SDL_bool SDL_HasSSE()
Jan 24, 2004
Jan 24, 2004
349
{
Feb 10, 2004
Feb 10, 2004
350
351
352
353
if ( SDL_GetCPUFeatures() & CPU_HAS_SSE ) {
return SDL_TRUE;
}
return SDL_FALSE;
Jan 24, 2004
Jan 24, 2004
354
355
}
Feb 10, 2004
Feb 10, 2004
356
SDL_bool SDL_HasSSE2()
Jan 24, 2004
Jan 24, 2004
357
{
Feb 10, 2004
Feb 10, 2004
358
359
360
361
if ( SDL_GetCPUFeatures() & CPU_HAS_SSE2 ) {
return SDL_TRUE;
}
return SDL_FALSE;
Jan 24, 2004
Jan 24, 2004
362
363
}
Feb 10, 2004
Feb 10, 2004
364
SDL_bool SDL_HasAltiVec()
Jan 24, 2004
Jan 24, 2004
365
{
Feb 10, 2004
Feb 10, 2004
366
367
368
369
if ( SDL_GetCPUFeatures() & CPU_HAS_ALTIVEC ) {
return SDL_TRUE;
}
return SDL_FALSE;
Jan 24, 2004
Jan 24, 2004
370
371
}
372
373
374
375
376
377
#ifdef TEST_MAIN
#include <stdio.h>
int main()
{
Jan 6, 2004
Jan 6, 2004
378
printf("RDTSC: %d\n", SDL_HasRDTSC());
379
printf("MMX: %d\n", SDL_HasMMX());
Jan 24, 2004
Jan 24, 2004
380
printf("MMXExt: %d\n", SDL_HasMMXExt());
381
printf("3DNow: %d\n", SDL_Has3DNow());
Jan 24, 2004
Jan 24, 2004
382
printf("3DNowExt: %d\n", SDL_Has3DNowExt());
383
printf("SSE: %d\n", SDL_HasSSE());
Jan 24, 2004
Jan 24, 2004
384
printf("SSE2: %d\n", SDL_HasSSE2());
Jan 6, 2004
Jan 6, 2004
385
printf("AltiVec: %d\n", SDL_HasAltiVec());
Nov 24, 2003
Nov 24, 2003
386
return 0;
387
388
389
}
#endif /* TEST_MAIN */