Skip to content
This repository has been archived by the owner on Feb 11, 2021. It is now read-only.

Latest commit

 

History

History
473 lines (438 loc) · 13.5 KB

SDL_cpuinfo.c

File metadata and controls

473 lines (438 loc) · 13.5 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"
Mar 22, 2006
Mar 22, 2006
29
#if defined(__MACOSX__) && defined(__ppc__)
Jul 10, 2006
Jul 10, 2006
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
/* This is the brute force way of detecting instruction sets...
the idea is borrowed from the libmpeg2 library - thanks!
*/
static jmp_buf jmpbuf;
Jul 10, 2006
Jul 10, 2006
50
51
static void
illegal_instruction(int sig)
Jan 29, 2004
Jan 29, 2004
52
{
Jul 10, 2006
Jul 10, 2006
53
longjmp(jmpbuf, 1);
Jan 29, 2004
Jan 29, 2004
54
}
Feb 16, 2006
Feb 16, 2006
55
#endif /* HAVE_SETJMP */
Jan 29, 2004
Jan 29, 2004
56
Jul 10, 2006
Jul 10, 2006
57
58
static __inline__ int
CPU_haveCPUID(void)
Nov 24, 2003
Nov 24, 2003
59
{
Jul 10, 2006
Jul 10, 2006
60
61
int has_CPUID = 0;
/* *INDENT-OFF* */
Nov 24, 2003
Nov 24, 2003
62
63
64
65
66
67
68
69
70
71
72
73
74
75
#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
76
: "=m" (has_CPUID)
Nov 24, 2003
Nov 24, 2003
77
78
79
:
: "%eax", "%ecx"
);
Apr 11, 2004
Apr 11, 2004
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
#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
100
#elif (defined(_MSC_VER) && defined(_M_IX86)) || defined(__WATCOMC__)
Nov 24, 2003
Nov 24, 2003
101
__asm {
Nov 24, 2003
Nov 24, 2003
102
103
104
105
106
107
108
109
110
111
112
113
114
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:
}
Jun 20, 2006
Jun 20, 2006
115
#elif defined(__sun) && defined(__i386)
Feb 16, 2006
Feb 16, 2006
116
__asm (
Jan 5, 2006
Jan 5, 2006
117
" pushfl \n"
Feb 16, 2006
Feb 16, 2006
118
119
120
121
122
123
124
125
126
127
" 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
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
"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
145
#endif
Jul 10, 2006
Jul 10, 2006
146
147
/* *INDENT-ON* */
return has_CPUID;
Nov 24, 2003
Nov 24, 2003
148
149
}
Jul 10, 2006
Jul 10, 2006
150
151
static __inline__ int
CPU_getCPUIDFeatures(void)
Nov 24, 2003
Nov 24, 2003
152
{
Jul 10, 2006
Jul 10, 2006
153
154
int features = 0;
/* *INDENT-OFF* */
Apr 11, 2004
Apr 11, 2004
155
#if defined(__GNUC__) && ( defined(i386) || defined(__x86_64__) )
Nov 24, 2003
Nov 24, 2003
156
__asm__ (
Jan 24, 2004
Jan 24, 2004
157
" movl %%ebx,%%edi\n"
Nov 24, 2003
Nov 24, 2003
158
159
160
161
162
163
164
165
166
" 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
167
" movl %%edi,%%ebx\n"
Jan 17, 2004
Jan 17, 2004
168
: "=m" (features)
Nov 24, 2003
Nov 24, 2003
169
:
May 16, 2004
May 16, 2004
170
: "%eax", "%ecx", "%edx", "%edi"
Nov 24, 2003
Nov 24, 2003
171
);
Feb 26, 2006
Feb 26, 2006
172
#elif (defined(_MSC_VER) && defined(_M_IX86)) || defined(__WATCOMC__)
Nov 24, 2003
Nov 24, 2003
173
__asm {
Nov 24, 2003
Nov 24, 2003
174
175
176
177
178
179
180
181
182
183
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:
}
Jun 20, 2006
Jun 20, 2006
184
#elif defined(__sun) && (defined(__i386) || defined(__amd64))
Jan 5, 2006
Jan 5, 2006
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
__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
201
#endif
Jul 10, 2006
Jul 10, 2006
202
203
/* *INDENT-ON* */
return features;
Nov 24, 2003
Nov 24, 2003
204
205
}
Jul 10, 2006
Jul 10, 2006
206
207
static __inline__ int
CPU_getCPUIDFeaturesExt(void)
Nov 24, 2003
Nov 24, 2003
208
{
Jul 10, 2006
Jul 10, 2006
209
210
int features = 0;
/* *INDENT-OFF* */
Apr 11, 2004
Apr 11, 2004
211
#if defined(__GNUC__) && (defined(i386) || defined (__x86_64__) )
Nov 24, 2003
Nov 24, 2003
212
__asm__ (
Jan 24, 2004
Jan 24, 2004
213
" movl %%ebx,%%edi\n"
Nov 24, 2003
Nov 24, 2003
214
215
216
" movl $0x80000000,%%eax # Query for extended functions \n"
" cpuid # Get extended function limit \n"
" cmpl $0x80000001,%%eax \n"
Jan 24, 2004
Jan 24, 2004
217
" jl 1f # Nope, we dont have function 800000001h\n"
Nov 24, 2003
Nov 24, 2003
218
219
" movl $0x80000001,%%eax # Setup extended function 800000001h\n"
" cpuid # and get the information \n"
Jan 24, 2004
Jan 24, 2004
220
" movl %%edx,%0 \n"
Nov 24, 2003
Nov 24, 2003
221
"1: \n"
Jan 24, 2004
Jan 24, 2004
222
223
" movl %%edi,%%ebx\n"
: "=m" (features)
Nov 24, 2003
Nov 24, 2003
224
:
May 16, 2004
May 16, 2004
225
: "%eax", "%ecx", "%edx", "%edi"
Nov 24, 2003
Nov 24, 2003
226
);
Feb 26, 2006
Feb 26, 2006
227
#elif (defined(_MSC_VER) && defined(_M_IX86)) || defined(__WATCOMC__)
Nov 24, 2003
Nov 24, 2003
228
__asm {
Nov 24, 2003
Nov 24, 2003
229
230
231
mov eax,80000000h ; Query for extended functions
cpuid ; Get extended function limit
cmp eax,80000001h
Jan 24, 2004
Jan 24, 2004
232
jl done ; Nope, we dont have function 800000001h
Nov 24, 2003
Nov 24, 2003
233
234
mov eax,80000001h ; Setup extended function 800000001h
cpuid ; and get the information
Jan 24, 2004
Jan 24, 2004
235
mov features,edx
Nov 24, 2003
Nov 24, 2003
236
237
done:
}
Jan 5, 2006
Jan 5, 2006
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
#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
255
#endif
Jul 10, 2006
Jul 10, 2006
256
257
/* *INDENT-ON* */
return features;
Jan 24, 2004
Jan 24, 2004
258
259
}
Jul 10, 2006
Jul 10, 2006
260
261
static __inline__ int
CPU_haveRDTSC(void)
Jan 24, 2004
Jan 24, 2004
262
{
Jul 10, 2006
Jul 10, 2006
263
264
265
266
if (CPU_haveCPUID()) {
return (CPU_getCPUIDFeatures() & 0x00000010);
}
return 0;
Jan 24, 2004
Jan 24, 2004
267
268
}
Jul 10, 2006
Jul 10, 2006
269
270
static __inline__ int
CPU_haveMMX(void)
Jan 24, 2004
Jan 24, 2004
271
{
Jul 10, 2006
Jul 10, 2006
272
273
274
275
if (CPU_haveCPUID()) {
return (CPU_getCPUIDFeatures() & 0x00800000);
}
return 0;
Jan 24, 2004
Jan 24, 2004
276
277
}
Jul 10, 2006
Jul 10, 2006
278
279
static __inline__ int
CPU_haveMMXExt(void)
Jan 24, 2004
Jan 24, 2004
280
{
Jul 10, 2006
Jul 10, 2006
281
282
283
284
if (CPU_haveCPUID()) {
return (CPU_getCPUIDFeaturesExt() & 0x00400000);
}
return 0;
Jan 24, 2004
Jan 24, 2004
285
286
}
Jul 10, 2006
Jul 10, 2006
287
288
static __inline__ int
CPU_have3DNow(void)
Jan 24, 2004
Jan 24, 2004
289
{
Jul 10, 2006
Jul 10, 2006
290
291
292
293
if (CPU_haveCPUID()) {
return (CPU_getCPUIDFeaturesExt() & 0x80000000);
}
return 0;
Jan 24, 2004
Jan 24, 2004
294
295
}
Jul 10, 2006
Jul 10, 2006
296
297
static __inline__ int
CPU_have3DNowExt(void)
Jan 24, 2004
Jan 24, 2004
298
{
Jul 10, 2006
Jul 10, 2006
299
300
301
302
if (CPU_haveCPUID()) {
return (CPU_getCPUIDFeaturesExt() & 0x40000000);
}
return 0;
Nov 24, 2003
Nov 24, 2003
303
304
}
Jul 10, 2006
Jul 10, 2006
305
306
static __inline__ int
CPU_haveSSE(void)
Nov 24, 2003
Nov 24, 2003
307
{
Jul 10, 2006
Jul 10, 2006
308
309
310
311
if (CPU_haveCPUID()) {
return (CPU_getCPUIDFeatures() & 0x02000000);
}
return 0;
Nov 24, 2003
Nov 24, 2003
312
}
Jul 10, 2006
Jul 10, 2006
314
315
static __inline__ int
CPU_haveSSE2(void)
Jan 24, 2004
Jan 24, 2004
316
{
Jul 10, 2006
Jul 10, 2006
317
318
319
320
if (CPU_haveCPUID()) {
return (CPU_getCPUIDFeatures() & 0x04000000);
}
return 0;
Jan 24, 2004
Jan 24, 2004
321
322
}
Jul 10, 2006
Jul 10, 2006
323
324
static __inline__ int
CPU_haveAltiVec(void)
Jan 6, 2004
Jan 6, 2004
325
{
Jul 10, 2006
Jul 10, 2006
326
volatile int altivec = 0;
Mar 22, 2006
Mar 22, 2006
327
#if defined(__MACOSX__) && defined(__ppc__)
Jul 10, 2006
Jul 10, 2006
328
329
330
331
332
333
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)
altivec = (hasVectorUnit != 0);
Feb 16, 2006
Feb 16, 2006
334
#elif SDL_ALTIVEC_BLITTERS && HAVE_SETJMP
Jul 10, 2006
Jul 10, 2006
335
336
337
338
339
340
341
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
342
#endif
Jul 10, 2006
Jul 10, 2006
343
return altivec;
Jan 6, 2004
Jan 6, 2004
344
345
}
346
347
static Uint32 SDL_CPUFeatures = 0xFFFFFFFF;
Jul 10, 2006
Jul 10, 2006
348
349
static Uint32
SDL_GetCPUFeatures(void)
Jul 10, 2006
Jul 10, 2006
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
if (SDL_CPUFeatures == 0xFFFFFFFF) {
SDL_CPUFeatures = 0;
if (CPU_haveRDTSC()) {
SDL_CPUFeatures |= CPU_HAS_RDTSC;
}
if (CPU_haveMMX()) {
SDL_CPUFeatures |= CPU_HAS_MMX;
}
if (CPU_haveMMXExt()) {
SDL_CPUFeatures |= CPU_HAS_MMXEXT;
}
if (CPU_have3DNow()) {
SDL_CPUFeatures |= CPU_HAS_3DNOW;
}
if (CPU_have3DNowExt()) {
SDL_CPUFeatures |= CPU_HAS_3DNOWEXT;
}
if (CPU_haveSSE()) {
SDL_CPUFeatures |= CPU_HAS_SSE;
}
if (CPU_haveSSE2()) {
SDL_CPUFeatures |= CPU_HAS_SSE2;
}
if (CPU_haveAltiVec()) {
SDL_CPUFeatures |= CPU_HAS_ALTIVEC;
}
}
return SDL_CPUFeatures;
Jul 10, 2006
Jul 10, 2006
381
382
SDL_bool
SDL_HasRDTSC(void)
Nov 24, 2003
Nov 24, 2003
383
{
Jul 10, 2006
Jul 10, 2006
384
385
386
387
if (SDL_GetCPUFeatures() & CPU_HAS_RDTSC) {
return SDL_TRUE;
}
return SDL_FALSE;
Nov 24, 2003
Nov 24, 2003
388
389
}
Jul 10, 2006
Jul 10, 2006
390
391
SDL_bool
SDL_HasMMX(void)
Jul 10, 2006
Jul 10, 2006
393
394
395
396
if (SDL_GetCPUFeatures() & CPU_HAS_MMX) {
return SDL_TRUE;
}
return SDL_FALSE;
Jul 10, 2006
Jul 10, 2006
399
400
SDL_bool
SDL_HasMMXExt(void)
Jul 10, 2006
Jul 10, 2006
402
403
404
405
if (SDL_GetCPUFeatures() & CPU_HAS_MMXEXT) {
return SDL_TRUE;
}
return SDL_FALSE;
Jul 10, 2006
Jul 10, 2006
408
409
SDL_bool
SDL_Has3DNow(void)
Jul 10, 2006
Jul 10, 2006
411
412
413
414
if (SDL_GetCPUFeatures() & CPU_HAS_3DNOW) {
return SDL_TRUE;
}
return SDL_FALSE;
Jul 10, 2006
Jul 10, 2006
417
418
SDL_bool
SDL_Has3DNowExt(void)
Jan 6, 2004
Jan 6, 2004
419
{
Jul 10, 2006
Jul 10, 2006
420
421
422
423
if (SDL_GetCPUFeatures() & CPU_HAS_3DNOWEXT) {
return SDL_TRUE;
}
return SDL_FALSE;
Jan 6, 2004
Jan 6, 2004
424
425
}
Jul 10, 2006
Jul 10, 2006
426
427
SDL_bool
SDL_HasSSE(void)
Jan 24, 2004
Jan 24, 2004
428
{
Jul 10, 2006
Jul 10, 2006
429
430
431
432
if (SDL_GetCPUFeatures() & CPU_HAS_SSE) {
return SDL_TRUE;
}
return SDL_FALSE;
Jan 24, 2004
Jan 24, 2004
433
434
}
Jul 10, 2006
Jul 10, 2006
435
436
SDL_bool
SDL_HasSSE2(void)
Jan 24, 2004
Jan 24, 2004
437
{
Jul 10, 2006
Jul 10, 2006
438
439
440
441
if (SDL_GetCPUFeatures() & CPU_HAS_SSE2) {
return SDL_TRUE;
}
return SDL_FALSE;
Jan 24, 2004
Jan 24, 2004
442
443
}
Jul 10, 2006
Jul 10, 2006
444
445
SDL_bool
SDL_HasAltiVec(void)
Jan 24, 2004
Jan 24, 2004
446
{
Jul 10, 2006
Jul 10, 2006
447
448
449
450
if (SDL_GetCPUFeatures() & CPU_HAS_ALTIVEC) {
return SDL_TRUE;
}
return SDL_FALSE;
Jan 24, 2004
Jan 24, 2004
451
452
}
453
454
455
456
#ifdef TEST_MAIN
#include <stdio.h>
Jul 10, 2006
Jul 10, 2006
457
458
int
main()
Jul 10, 2006
Jul 10, 2006
460
461
462
463
464
465
466
467
468
printf("RDTSC: %d\n", SDL_HasRDTSC());
printf("MMX: %d\n", SDL_HasMMX());
printf("MMXExt: %d\n", SDL_HasMMXExt());
printf("3DNow: %d\n", SDL_Has3DNow());
printf("3DNowExt: %d\n", SDL_Has3DNowExt());
printf("SSE: %d\n", SDL_HasSSE());
printf("SSE2: %d\n", SDL_HasSSE2());
printf("AltiVec: %d\n", SDL_HasAltiVec());
return 0;
469
470
471
}
#endif /* TEST_MAIN */
Jul 10, 2006
Jul 10, 2006
472
473
/* vi: set ts=4 sw=4 expandtab: */