1 | /* pnggccrd.c was removed from libpng-1.2.20. */
|
---|
2 |
|
---|
3 | /* This code snippet is for use by configure's compilation test. */
|
---|
4 |
|
---|
5 | #if (!defined _MSC_VER) && \
|
---|
6 | defined(PNG_ASSEMBLER_CODE_SUPPORTED) && \
|
---|
7 | defined(PNG_MMX_CODE_SUPPORTED)
|
---|
8 |
|
---|
9 | int PNGAPI png_dummy_mmx_support(void);
|
---|
10 |
|
---|
11 | static int _mmx_supported = 2; // 0: no MMX; 1: MMX supported; 2: not tested
|
---|
12 |
|
---|
13 | int PNGAPI
|
---|
14 | png_dummy_mmx_support(void) __attribute__((noinline));
|
---|
15 |
|
---|
16 | int PNGAPI
|
---|
17 | png_dummy_mmx_support(void)
|
---|
18 | {
|
---|
19 | int result;
|
---|
20 | #if defined(PNG_MMX_CODE_SUPPORTED) // superfluous, but what the heck
|
---|
21 | __asm__ __volatile__ (
|
---|
22 | #if defined(__x86_64__)
|
---|
23 | "pushq %%rbx \n\t" // rbx gets clobbered by CPUID instruction
|
---|
24 | "pushq %%rcx \n\t" // so does rcx...
|
---|
25 | "pushq %%rdx \n\t" // ...and rdx (but rcx & rdx safe on Linux)
|
---|
26 | "pushfq \n\t" // save Eflag to stack
|
---|
27 | "popq %%rax \n\t" // get Eflag from stack into rax
|
---|
28 | "movq %%rax, %%rcx \n\t" // make another copy of Eflag in rcx
|
---|
29 | "xorl $0x200000, %%eax \n\t" // toggle ID bit in Eflag (i.e., bit 21)
|
---|
30 | "pushq %%rax \n\t" // save modified Eflag back to stack
|
---|
31 | "popfq \n\t" // restore modified value to Eflag reg
|
---|
32 | "pushfq \n\t" // save Eflag to stack
|
---|
33 | "popq %%rax \n\t" // get Eflag from stack
|
---|
34 | "pushq %%rcx \n\t" // save original Eflag to stack
|
---|
35 | "popfq \n\t" // restore original Eflag
|
---|
36 | #else
|
---|
37 | "pushl %%ebx \n\t" // ebx gets clobbered by CPUID instruction
|
---|
38 | "pushl %%ecx \n\t" // so does ecx...
|
---|
39 | "pushl %%edx \n\t" // ...and edx (but ecx & edx safe on Linux)
|
---|
40 | "pushfl \n\t" // save Eflag to stack
|
---|
41 | "popl %%eax \n\t" // get Eflag from stack into eax
|
---|
42 | "movl %%eax, %%ecx \n\t" // make another copy of Eflag in ecx
|
---|
43 | "xorl $0x200000, %%eax \n\t" // toggle ID bit in Eflag (i.e., bit 21)
|
---|
44 | "pushl %%eax \n\t" // save modified Eflag back to stack
|
---|
45 | "popfl \n\t" // restore modified value to Eflag reg
|
---|
46 | "pushfl \n\t" // save Eflag to stack
|
---|
47 | "popl %%eax \n\t" // get Eflag from stack
|
---|
48 | "pushl %%ecx \n\t" // save original Eflag to stack
|
---|
49 | "popfl \n\t" // restore original Eflag
|
---|
50 | #endif
|
---|
51 | "xorl %%ecx, %%eax \n\t" // compare new Eflag with original Eflag
|
---|
52 | "jz 0f \n\t" // if same, CPUID instr. is not supported
|
---|
53 |
|
---|
54 | "xorl %%eax, %%eax \n\t" // set eax to zero
|
---|
55 | // ".byte 0x0f, 0xa2 \n\t" // CPUID instruction (two-byte opcode)
|
---|
56 | "cpuid \n\t" // get the CPU identification info
|
---|
57 | "cmpl $1, %%eax \n\t" // make sure eax return non-zero value
|
---|
58 | "jl 0f \n\t" // if eax is zero, MMX is not supported
|
---|
59 |
|
---|
60 | "xorl %%eax, %%eax \n\t" // set eax to zero and...
|
---|
61 | "incl %%eax \n\t" // ...increment eax to 1. This pair is
|
---|
62 | // faster than the instruction "mov eax, 1"
|
---|
63 | "cpuid \n\t" // get the CPU identification info again
|
---|
64 | "andl $0x800000, %%edx \n\t" // mask out all bits but MMX bit (23)
|
---|
65 | "cmpl $0, %%edx \n\t" // 0 = MMX not supported
|
---|
66 | "jz 0f \n\t" // non-zero = yes, MMX IS supported
|
---|
67 |
|
---|
68 | "movl $1, %%eax \n\t" // set return value to 1
|
---|
69 | "jmp 1f \n\t" // DONE: have MMX support
|
---|
70 |
|
---|
71 | "0: \n\t" // .NOT_SUPPORTED: target label for jump instructions
|
---|
72 | "movl $0, %%eax \n\t" // set return value to 0
|
---|
73 | "1: \n\t" // .RETURN: target label for jump instructions
|
---|
74 | #if defined(__x86_64__)
|
---|
75 | "popq %%rdx \n\t" // restore rdx
|
---|
76 | "popq %%rcx \n\t" // restore rcx
|
---|
77 | "popq %%rbx \n\t" // restore rbx
|
---|
78 | #else
|
---|
79 | "popl %%edx \n\t" // restore edx
|
---|
80 | "popl %%ecx \n\t" // restore ecx
|
---|
81 | "popl %%ebx \n\t" // restore ebx
|
---|
82 | #endif
|
---|
83 |
|
---|
84 | // "ret \n\t" // DONE: no MMX support
|
---|
85 | // (fall through to standard C "ret")
|
---|
86 |
|
---|
87 | : "=a" (result) // output list
|
---|
88 |
|
---|
89 | : // any variables used on input (none)
|
---|
90 |
|
---|
91 | // no clobber list
|
---|
92 | // , "%ebx", "%ecx", "%edx" // GRR: we handle these manually
|
---|
93 | // , "memory" // if write to a variable gcc thought was in a reg
|
---|
94 | // , "cc" // "condition codes" (flag bits)
|
---|
95 | );
|
---|
96 | _mmx_supported = result;
|
---|
97 | #else
|
---|
98 | _mmx_supported = 0;
|
---|
99 | #endif /* PNG_MMX_CODE_SUPPORTED */
|
---|
100 |
|
---|
101 | return _mmx_supported;
|
---|
102 | }
|
---|
103 | #endif
|
---|