Mercurial > libavutil.hg
annotate x86/cpu.c @ 1010:6138233957fe libavutil
Clean up av_get_cpu_flag()
Instead of defining functions in per-arch header files included
by the main cpu.c, define them normally and call them from the
generic one.
author | mru |
---|---|
date | Thu, 09 Sep 2010 18:51:45 +0000 |
parents | x86/cpu.h@40b8596460af |
children | c915dc284b9a |
rev | line source |
---|---|
1009
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
1 /* |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
2 * CPU detection code, extracted from mmx.h |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
3 * (c)1997-99 by H. Dietz and R. Fisher |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
4 * Converted to C and improved by Fabrice Bellard. |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
5 * |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
6 * This file is part of FFmpeg. |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
7 * |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
8 * FFmpeg is free software; you can redistribute it and/or |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
9 * modify it under the terms of the GNU Lesser General Public |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
10 * License as published by the Free Software Foundation; either |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
11 * version 2.1 of the License, or (at your option) any later version. |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
12 * |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
13 * FFmpeg is distributed in the hope that it will be useful, |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
16 * Lesser General Public License for more details. |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
17 * |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
18 * You should have received a copy of the GNU Lesser General Public |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
19 * License along with FFmpeg; if not, write to the Free Software |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
20 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
21 */ |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
22 |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
23 #include <stdlib.h> |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
24 #include "libavutil/x86_cpu.h" |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
25 #include "libavutil/cpu.h" |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
26 |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
27 /* ebx saving is necessary for PIC. gcc seems unable to see it alone */ |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
28 #define cpuid(index,eax,ebx,ecx,edx)\ |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
29 __asm__ volatile\ |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
30 ("mov %%"REG_b", %%"REG_S"\n\t"\ |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
31 "cpuid\n\t"\ |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
32 "xchg %%"REG_b", %%"REG_S\ |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
33 : "=a" (eax), "=S" (ebx),\ |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
34 "=c" (ecx), "=d" (edx)\ |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
35 : "0" (index)); |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
36 |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
37 /* Function to test if multimedia instructions are supported... */ |
1010 | 38 int ff_get_cpu_flags_x86(void) |
1009
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
39 { |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
40 int rval = 0; |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
41 int eax, ebx, ecx, edx; |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
42 int max_std_level, max_ext_level, std_caps=0, ext_caps=0; |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
43 int family=0, model=0; |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
44 union { int i[3]; char c[12]; } vendor; |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
45 |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
46 #if ARCH_X86_32 |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
47 x86_reg a, c; |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
48 __asm__ volatile ( |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
49 /* See if CPUID instruction is supported ... */ |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
50 /* ... Get copies of EFLAGS into eax and ecx */ |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
51 "pushfl\n\t" |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
52 "pop %0\n\t" |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
53 "mov %0, %1\n\t" |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
54 |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
55 /* ... Toggle the ID bit in one copy and store */ |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
56 /* to the EFLAGS reg */ |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
57 "xor $0x200000, %0\n\t" |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
58 "push %0\n\t" |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
59 "popfl\n\t" |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
60 |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
61 /* ... Get the (hopefully modified) EFLAGS */ |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
62 "pushfl\n\t" |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
63 "pop %0\n\t" |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
64 : "=a" (a), "=c" (c) |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
65 : |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
66 : "cc" |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
67 ); |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
68 |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
69 if (a == c) |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
70 return 0; /* CPUID not supported */ |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
71 #endif |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
72 |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
73 cpuid(0, max_std_level, vendor.i[0], vendor.i[2], vendor.i[1]); |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
74 |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
75 if(max_std_level >= 1){ |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
76 cpuid(1, eax, ebx, ecx, std_caps); |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
77 family = ((eax>>8)&0xf) + ((eax>>20)&0xff); |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
78 model = ((eax>>4)&0xf) + ((eax>>12)&0xf0); |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
79 if (std_caps & (1<<23)) |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
80 rval |= AV_CPU_FLAG_MMX; |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
81 if (std_caps & (1<<25)) |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
82 rval |= AV_CPU_FLAG_MMX2 |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
83 #if HAVE_SSE |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
84 | AV_CPU_FLAG_SSE; |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
85 if (std_caps & (1<<26)) |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
86 rval |= AV_CPU_FLAG_SSE2; |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
87 if (ecx & 1) |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
88 rval |= AV_CPU_FLAG_SSE3; |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
89 if (ecx & 0x00000200 ) |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
90 rval |= AV_CPU_FLAG_SSSE3; |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
91 if (ecx & 0x00080000 ) |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
92 rval |= AV_CPU_FLAG_SSE4; |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
93 if (ecx & 0x00100000 ) |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
94 rval |= AV_CPU_FLAG_SSE42; |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
95 #endif |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
96 ; |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
97 } |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
98 |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
99 cpuid(0x80000000, max_ext_level, ebx, ecx, edx); |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
100 |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
101 if(max_ext_level >= 0x80000001){ |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
102 cpuid(0x80000001, eax, ebx, ecx, ext_caps); |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
103 if (ext_caps & (1<<31)) |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
104 rval |= AV_CPU_FLAG_3DNOW; |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
105 if (ext_caps & (1<<30)) |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
106 rval |= AV_CPU_FLAG_3DNOWEXT; |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
107 if (ext_caps & (1<<23)) |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
108 rval |= AV_CPU_FLAG_MMX; |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
109 if (ext_caps & (1<<22)) |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
110 rval |= AV_CPU_FLAG_MMX2; |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
111 } |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
112 |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
113 if (!strncmp(vendor.c, "GenuineIntel", 12) && |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
114 family == 6 && (model == 9 || model == 13 || model == 14)) { |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
115 /* 6/9 (pentium-m "banias"), 6/13 (pentium-m "dothan"), and 6/14 (core1 "yonah") |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
116 * theoretically support sse2, but it's usually slower than mmx, |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
117 * so let's just pretend they don't. */ |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
118 if (rval & AV_CPU_FLAG_SSE2) rval ^= AV_CPU_FLAG_SSE2SLOW|AV_CPU_FLAG_SSE2; |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
119 if (rval & AV_CPU_FLAG_SSE3) rval ^= AV_CPU_FLAG_SSE3SLOW|AV_CPU_FLAG_SSE3; |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
120 } |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
121 |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
122 return rval; |
40b8596460af
Move mm_support() from libavcodec to libavutil, make it a public
stefano
parents:
diff
changeset
|
123 } |