Skip to content

Latest commit

 

History

History
446 lines (412 loc) · 12.9 KB

SDL_cpuinfo.c

File metadata and controls

446 lines (412 loc) · 12.9 KB
 
1
2
/*
SDL - Simple DirectMedia Layer
Feb 1, 2006
Feb 1, 2006
3
Copyright (C) 1997-2006 Sam Lantinga
4
5
This library is free software; you can redistribute it and/or
Feb 1, 2006
Feb 1, 2006
6
modify it under the terms of the GNU Lesser General Public
7
License as published by the Free Software Foundation; either
Feb 1, 2006
Feb 1, 2006
8
version 2.1 of the License, or (at your option) any later version.
9
10
11
12
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
Feb 1, 2006
Feb 1, 2006
13
Lesser General Public License for more details.
Feb 1, 2006
Feb 1, 2006
15
16
17
You should have received a copy of the GNU Lesser General Public
License along with this library; if not, write to the Free Software
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
18
19
20
21
Sam Lantinga
slouken@libsdl.org
*/
Feb 21, 2006
Feb 21, 2006
22
#include "SDL_config.h"
23
24
25
/* CPU feature detection for SDL */
Feb 16, 2006
Feb 16, 2006
26
27
28
#include "SDL.h"
#include "SDL_cpuinfo.h"
Feb 21, 2006
Feb 21, 2006
29
#ifdef __MACOSX__
Jan 6, 2004
Jan 6, 2004
30
#include <sys/sysctl.h> /* For AltiVec check */
Mar 9, 2006
Mar 9, 2006
31
32
33
#elif SDL_ALTIVEC_BLITTERS && HAVE_SETJMP
#include <signal.h>
#include <setjmp.h>
Jan 6, 2004
Jan 6, 2004
34
35
#endif
Nov 24, 2003
Nov 24, 2003
36
37
#define CPU_HAS_RDTSC 0x00000001
#define CPU_HAS_MMX 0x00000002
Jan 24, 2004
Jan 24, 2004
38
39
40
41
42
43
#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
Mar 9, 2006
Mar 9, 2006
45
#if SDL_ALTIVEC_BLITTERS && HAVE_SETJMP && !__MACOSX__
Jan 29, 2004
Jan 29, 2004
46
47
48
49
50
51
52
53
/* 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);
}
Feb 16, 2006
Feb 16, 2006
54
#endif /* HAVE_SETJMP */
Jan 29, 2004
Jan 29, 2004
55
Feb 24, 2006
Feb 24, 2006
56
static __inline__ int CPU_haveCPUID(void)
Nov 24, 2003
Nov 24, 2003
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
{
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
73
: "=m" (has_CPUID)
Nov 24, 2003
Nov 24, 2003
74
75
76
:
: "%eax", "%ecx"
);
Apr 11, 2004
Apr 11, 2004
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
#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"
);
Feb 26, 2006
Feb 26, 2006
97
#elif (defined(_MSC_VER) && defined(_M_IX86)) || defined(__WATCOMC__)
Nov 24, 2003
Nov 24, 2003
98
__asm {
Nov 24, 2003
Nov 24, 2003
99
100
101
102
103
104
105
106
107
108
109
110
111
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:
}
Jan 5, 2006
Jan 5, 2006
112
#elif defined(__sun) && defined(__x86)
Feb 16, 2006
Feb 16, 2006
113
__asm (
Jan 5, 2006
Jan 5, 2006
114
" pushfl \n"
Feb 16, 2006
Feb 16, 2006
115
116
117
118
119
120
121
122
123
124
" popl %eax \n"
" movl %eax,%ecx \n"
" xorl $0x200000,%eax \n"
" pushl %eax \n"
" popfl \n"
" pushfl \n"
" popl %eax \n"
" xorl %ecx,%eax \n"
" jz 1f \n"
" movl $1,-8(%ebp) \n"
Jan 5, 2006
Jan 5, 2006
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
"1: \n"
);
#elif defined(__sun) && defined(__amd64)
__asm (
" pushfq \n"
" popq %rax \n"
" movq %rax,%rcx \n"
" xorl $0x200000,%eax \n"
" pushq %rax \n"
" popfq \n"
" pushfq \n"
" popq %rax \n"
" xorl %ecx,%eax \n"
" jz 1f \n"
" movl $1,-8(%rbp) \n"
"1: \n"
);
Nov 24, 2003
Nov 24, 2003
142
143
144
145
#endif
return has_CPUID;
}
Feb 24, 2006
Feb 24, 2006
146
static __inline__ int CPU_getCPUIDFeatures(void)
Nov 24, 2003
Nov 24, 2003
147
148
{
int features = 0;
Apr 11, 2004
Apr 11, 2004
149
#if defined(__GNUC__) && ( defined(i386) || defined(__x86_64__) )
Nov 24, 2003
Nov 24, 2003
150
__asm__ (
Jan 24, 2004
Jan 24, 2004
151
" movl %%ebx,%%edi\n"
Nov 24, 2003
Nov 24, 2003
152
153
154
155
156
157
158
159
160
" 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
161
" movl %%edi,%%ebx\n"
Jan 17, 2004
Jan 17, 2004
162
: "=m" (features)
Nov 24, 2003
Nov 24, 2003
163
:
May 16, 2004
May 16, 2004
164
: "%eax", "%ecx", "%edx", "%edi"
Nov 24, 2003
Nov 24, 2003
165
);
Feb 26, 2006
Feb 26, 2006
166
#elif (defined(_MSC_VER) && defined(_M_IX86)) || defined(__WATCOMC__)
Nov 24, 2003
Nov 24, 2003
167
__asm {
Nov 24, 2003
Nov 24, 2003
168
169
170
171
172
173
174
175
176
177
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:
}
Jan 5, 2006
Jan 5, 2006
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
#elif defined(__sun) && (defined(__x86) || defined(__amd64))
__asm(
" movl %ebx,%edi\n"
" xorl %eax,%eax \n"
" cpuid \n"
" cmpl $1,%eax \n"
" jl 1f \n"
" xorl %eax,%eax \n"
" incl %eax \n"
" cpuid \n"
#ifdef __i386
" movl %edx,-8(%ebp) \n"
#else
" movl %edx,-8(%rbp) \n"
#endif
"1: \n"
" movl %edi,%ebx\n" );
Nov 24, 2003
Nov 24, 2003
195
196
197
198
#endif
return features;
}
Feb 24, 2006
Feb 24, 2006
199
static __inline__ int CPU_getCPUIDFeaturesExt(void)
Nov 24, 2003
Nov 24, 2003
200
{
Jan 24, 2004
Jan 24, 2004
201
int features = 0;
Apr 11, 2004
Apr 11, 2004
202
#if defined(__GNUC__) && (defined(i386) || defined (__x86_64__) )
Nov 24, 2003
Nov 24, 2003
203
__asm__ (
Jan 24, 2004
Jan 24, 2004
204
" movl %%ebx,%%edi\n"
Nov 24, 2003
Nov 24, 2003
205
206
207
" movl $0x80000000,%%eax # Query for extended functions \n"
" cpuid # Get extended function limit \n"
" cmpl $0x80000001,%%eax \n"
Jan 24, 2004
Jan 24, 2004
208
" jl 1f # Nope, we dont have function 800000001h\n"
Nov 24, 2003
Nov 24, 2003
209
210
" movl $0x80000001,%%eax # Setup extended function 800000001h\n"
" cpuid # and get the information \n"
Jan 24, 2004
Jan 24, 2004
211
" movl %%edx,%0 \n"
Nov 24, 2003
Nov 24, 2003
212
"1: \n"
Jan 24, 2004
Jan 24, 2004
213
214
" movl %%edi,%%ebx\n"
: "=m" (features)
Nov 24, 2003
Nov 24, 2003
215
:
May 16, 2004
May 16, 2004
216
: "%eax", "%ecx", "%edx", "%edi"
Nov 24, 2003
Nov 24, 2003
217
);
Feb 26, 2006
Feb 26, 2006
218
#elif (defined(_MSC_VER) && defined(_M_IX86)) || defined(__WATCOMC__)
Nov 24, 2003
Nov 24, 2003
219
__asm {
Nov 24, 2003
Nov 24, 2003
220
221
222
mov eax,80000000h ; Query for extended functions
cpuid ; Get extended function limit
cmp eax,80000001h
Jan 24, 2004
Jan 24, 2004
223
jl done ; Nope, we dont have function 800000001h
Nov 24, 2003
Nov 24, 2003
224
225
mov eax,80000001h ; Setup extended function 800000001h
cpuid ; and get the information
Jan 24, 2004
Jan 24, 2004
226
mov features,edx
Nov 24, 2003
Nov 24, 2003
227
228
done:
}
Jan 5, 2006
Jan 5, 2006
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
#elif defined(__sun) && ( defined(__i386) || defined(__amd64) )
__asm (
" movl %ebx,%edi\n"
" movl $0x80000000,%eax \n"
" cpuid \n"
" cmpl $0x80000001,%eax \n"
" jl 1f \n"
" movl $0x80000001,%eax \n"
" cpuid \n"
#ifdef __i386
" movl %edx,-8(%ebp) \n"
#else
" movl %edx,-8(%rbp) \n"
#endif
"1: \n"
" movl %edi,%ebx\n"
);
Nov 24, 2003
Nov 24, 2003
246
#endif
Jan 24, 2004
Jan 24, 2004
247
248
249
return features;
}
Feb 24, 2006
Feb 24, 2006
250
static __inline__ int CPU_haveRDTSC(void)
Jan 24, 2004
Jan 24, 2004
251
252
253
254
255
256
257
{
if ( CPU_haveCPUID() ) {
return (CPU_getCPUIDFeatures() & 0x00000010);
}
return 0;
}
Feb 24, 2006
Feb 24, 2006
258
static __inline__ int CPU_haveMMX(void)
Jan 24, 2004
Jan 24, 2004
259
260
261
262
263
264
265
{
if ( CPU_haveCPUID() ) {
return (CPU_getCPUIDFeatures() & 0x00800000);
}
return 0;
}
Feb 24, 2006
Feb 24, 2006
266
static __inline__ int CPU_haveMMXExt(void)
Jan 24, 2004
Jan 24, 2004
267
268
269
270
271
272
273
{
if ( CPU_haveCPUID() ) {
return (CPU_getCPUIDFeaturesExt() & 0x00400000);
}
return 0;
}
Feb 24, 2006
Feb 24, 2006
274
static __inline__ int CPU_have3DNow(void)
Jan 24, 2004
Jan 24, 2004
275
276
277
278
279
280
281
{
if ( CPU_haveCPUID() ) {
return (CPU_getCPUIDFeaturesExt() & 0x80000000);
}
return 0;
}
Feb 24, 2006
Feb 24, 2006
282
static __inline__ int CPU_have3DNowExt(void)
Jan 24, 2004
Jan 24, 2004
283
284
285
286
287
{
if ( CPU_haveCPUID() ) {
return (CPU_getCPUIDFeaturesExt() & 0x40000000);
}
return 0;
Nov 24, 2003
Nov 24, 2003
288
289
}
Feb 24, 2006
Feb 24, 2006
290
static __inline__ int CPU_haveSSE(void)
Nov 24, 2003
Nov 24, 2003
291
292
293
294
295
296
{
if ( CPU_haveCPUID() ) {
return (CPU_getCPUIDFeatures() & 0x02000000);
}
return 0;
}
Feb 24, 2006
Feb 24, 2006
298
static __inline__ int CPU_haveSSE2(void)
Jan 24, 2004
Jan 24, 2004
299
300
301
302
303
304
305
{
if ( CPU_haveCPUID() ) {
return (CPU_getCPUIDFeatures() & 0x04000000);
}
return 0;
}
Feb 24, 2006
Feb 24, 2006
306
static __inline__ int CPU_haveAltiVec(void)
Jan 6, 2004
Jan 6, 2004
307
{
Jan 31, 2004
Jan 31, 2004
308
volatile int altivec = 0;
Feb 21, 2006
Feb 21, 2006
309
#ifdef __MACOSX__
Jan 6, 2004
Jan 6, 2004
310
311
312
313
314
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
315
altivec = (hasVectorUnit != 0);
Feb 16, 2006
Feb 16, 2006
316
#elif SDL_ALTIVEC_BLITTERS && HAVE_SETJMP
Jan 29, 2004
Jan 29, 2004
317
318
319
320
321
322
323
324
325
326
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
327
#endif
Jan 29, 2004
Jan 29, 2004
328
return altivec;
Jan 6, 2004
Jan 6, 2004
329
330
}
331
332
static Uint32 SDL_CPUFeatures = 0xFFFFFFFF;
Feb 24, 2006
Feb 24, 2006
333
static Uint32 SDL_GetCPUFeatures(void)
334
335
336
{
if ( SDL_CPUFeatures == 0xFFFFFFFF ) {
SDL_CPUFeatures = 0;
Nov 24, 2003
Nov 24, 2003
337
338
339
if ( CPU_haveRDTSC() ) {
SDL_CPUFeatures |= CPU_HAS_RDTSC;
}
340
341
342
if ( CPU_haveMMX() ) {
SDL_CPUFeatures |= CPU_HAS_MMX;
}
Jan 24, 2004
Jan 24, 2004
343
344
345
if ( CPU_haveMMXExt() ) {
SDL_CPUFeatures |= CPU_HAS_MMXEXT;
}
346
347
348
if ( CPU_have3DNow() ) {
SDL_CPUFeatures |= CPU_HAS_3DNOW;
}
Jan 24, 2004
Jan 24, 2004
349
350
351
if ( CPU_have3DNowExt() ) {
SDL_CPUFeatures |= CPU_HAS_3DNOWEXT;
}
352
353
354
if ( CPU_haveSSE() ) {
SDL_CPUFeatures |= CPU_HAS_SSE;
}
Jan 24, 2004
Jan 24, 2004
355
356
357
if ( CPU_haveSSE2() ) {
SDL_CPUFeatures |= CPU_HAS_SSE2;
}
Jan 6, 2004
Jan 6, 2004
358
359
360
if ( CPU_haveAltiVec() ) {
SDL_CPUFeatures |= CPU_HAS_ALTIVEC;
}
361
362
363
364
}
return SDL_CPUFeatures;
}
Feb 24, 2006
Feb 24, 2006
365
SDL_bool SDL_HasRDTSC(void)
Nov 24, 2003
Nov 24, 2003
366
367
368
369
370
371
372
{
if ( SDL_GetCPUFeatures() & CPU_HAS_RDTSC ) {
return SDL_TRUE;
}
return SDL_FALSE;
}
Feb 24, 2006
Feb 24, 2006
373
SDL_bool SDL_HasMMX(void)
374
375
376
377
378
379
380
{
if ( SDL_GetCPUFeatures() & CPU_HAS_MMX ) {
return SDL_TRUE;
}
return SDL_FALSE;
}
Feb 24, 2006
Feb 24, 2006
381
SDL_bool SDL_HasMMXExt(void)
Feb 10, 2004
Feb 10, 2004
383
if ( SDL_GetCPUFeatures() & CPU_HAS_MMXEXT ) {
384
385
386
387
388
return SDL_TRUE;
}
return SDL_FALSE;
}
Feb 24, 2006
Feb 24, 2006
389
SDL_bool SDL_Has3DNow(void)
Feb 10, 2004
Feb 10, 2004
391
if ( SDL_GetCPUFeatures() & CPU_HAS_3DNOW ) {
392
393
394
395
396
return SDL_TRUE;
}
return SDL_FALSE;
}
Feb 24, 2006
Feb 24, 2006
397
SDL_bool SDL_Has3DNowExt(void)
Jan 6, 2004
Jan 6, 2004
398
{
Feb 10, 2004
Feb 10, 2004
399
if ( SDL_GetCPUFeatures() & CPU_HAS_3DNOWEXT ) {
Jan 6, 2004
Jan 6, 2004
400
401
402
403
404
return SDL_TRUE;
}
return SDL_FALSE;
}
Feb 24, 2006
Feb 24, 2006
405
SDL_bool SDL_HasSSE(void)
Jan 24, 2004
Jan 24, 2004
406
{
Feb 10, 2004
Feb 10, 2004
407
408
409
410
if ( SDL_GetCPUFeatures() & CPU_HAS_SSE ) {
return SDL_TRUE;
}
return SDL_FALSE;
Jan 24, 2004
Jan 24, 2004
411
412
}
Feb 24, 2006
Feb 24, 2006
413
SDL_bool SDL_HasSSE2(void)
Jan 24, 2004
Jan 24, 2004
414
{
Feb 10, 2004
Feb 10, 2004
415
416
417
418
if ( SDL_GetCPUFeatures() & CPU_HAS_SSE2 ) {
return SDL_TRUE;
}
return SDL_FALSE;
Jan 24, 2004
Jan 24, 2004
419
420
}
Feb 24, 2006
Feb 24, 2006
421
SDL_bool SDL_HasAltiVec(void)
Jan 24, 2004
Jan 24, 2004
422
{
Feb 10, 2004
Feb 10, 2004
423
424
425
426
if ( SDL_GetCPUFeatures() & CPU_HAS_ALTIVEC ) {
return SDL_TRUE;
}
return SDL_FALSE;
Jan 24, 2004
Jan 24, 2004
427
428
}
429
430
431
432
433
434
#ifdef TEST_MAIN
#include <stdio.h>
int main()
{
Jan 6, 2004
Jan 6, 2004
435
printf("RDTSC: %d\n", SDL_HasRDTSC());
436
printf("MMX: %d\n", SDL_HasMMX());
Jan 24, 2004
Jan 24, 2004
437
printf("MMXExt: %d\n", SDL_HasMMXExt());
438
printf("3DNow: %d\n", SDL_Has3DNow());
Jan 24, 2004
Jan 24, 2004
439
printf("3DNowExt: %d\n", SDL_Has3DNowExt());
440
printf("SSE: %d\n", SDL_HasSSE());
Jan 24, 2004
Jan 24, 2004
441
printf("SSE2: %d\n", SDL_HasSSE2());
Jan 6, 2004
Jan 6, 2004
442
printf("AltiVec: %d\n", SDL_HasAltiVec());
Nov 24, 2003
Nov 24, 2003
443
return 0;
444
445
446
}
#endif /* TEST_MAIN */