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 #ifdef PNG_MMX_CODE_SUPPORTED // superfluous, but what the heck 21 __asm__ __volatile__ ( 22 #ifdef __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 #ifdef __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 104