c2372d9463679a208e15023e470b47ae61b7f24f
[flac.git] / src / libFLAC / include / private / cpu.h
1 /* libFLAC - Free Lossless Audio Codec library
2  * Copyright (C) 2001-2009  Josh Coalson
3  * Copyright (C) 2011-2016  Xiph.Org Foundation
4  *
5  * Redistribution and use in source and binary forms, with or without
6  * modification, are permitted provided that the following conditions
7  * are met:
8  *
9  * - Redistributions of source code must retain the above copyright
10  * notice, this list of conditions and the following disclaimer.
11  *
12  * - Redistributions in binary form must reproduce the above copyright
13  * notice, this list of conditions and the following disclaimer in the
14  * documentation and/or other materials provided with the distribution.
15  *
16  * - Neither the name of the Xiph.org Foundation nor the names of its
17  * contributors may be used to endorse or promote products derived from
18  * this software without specific prior written permission.
19  *
20  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
21  * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
22  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
23  * A PARTICULAR PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE FOUNDATION OR
24  * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
25  * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
26  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
27  * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
28  * LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
29  * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
30  * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
31  */
32
33 #ifndef FLAC__PRIVATE__CPU_H
34 #define FLAC__PRIVATE__CPU_H
35
36 #include "FLAC/ordinals.h"
37
38 #ifdef HAVE_CONFIG_H
39 #include <config.h>
40 #endif
41
42 #ifndef FLAC__CPU_X86_64
43
44 #if defined(__amd64__) || defined(__amd64) || defined(__x86_64__) || defined(__x86_64) || defined(_M_X64) || defined(_M_AMD64)
45 #define FLAC__CPU_X86_64
46 #endif
47
48 #endif
49
50 #ifndef FLAC__CPU_IA32
51
52 #if defined(__i386__) || defined(__i486__) || defined(__i586__) || defined(__i686__) ||defined( __i386) || defined(_M_IX86)
53 #define FLAC__CPU_IA32
54 #endif
55
56 #endif
57
58
59 #if FLAC__HAS_X86INTRIN
60 /* SSE intrinsics support by ICC/MSVC/GCC */
61 #if defined __INTEL_COMPILER
62   #define FLAC__SSE_TARGET(x)
63   #define FLAC__SSE_SUPPORTED 1
64   #define FLAC__SSE2_SUPPORTED 1
65   #if (__INTEL_COMPILER >= 1000) /* Intel C++ Compiler 10.0 */
66     #define FLAC__SSSE3_SUPPORTED 1
67     #define FLAC__SSE4_1_SUPPORTED 1
68   #endif
69   #if (__INTEL_COMPILER >= 1110) /* Intel C++ Compiler 11.1 */
70     #define FLAC__AVX_SUPPORTED 1
71   #endif
72   #if (__INTEL_COMPILER >= 1300) /* Intel C++ Compiler 13.0 */
73     #define FLAC__AVX2_SUPPORTED 1
74     #define FLAC__FMA_SUPPORTED 1
75   #endif
76 #elif defined _MSC_VER
77   #define FLAC__SSE_TARGET(x)
78   #define FLAC__SSE_SUPPORTED 1
79   #define FLAC__SSE2_SUPPORTED 1
80   #if (_MSC_VER >= 1500) /* MS Visual Studio 2008 */
81     #define FLAC__SSSE3_SUPPORTED 1
82     #define FLAC__SSE4_1_SUPPORTED 1
83   #endif
84   #if (_MSC_FULL_VER >= 160040219) /* MS Visual Studio 2010 SP1 */
85     #define FLAC__AVX_SUPPORTED 1
86   #endif
87   #if (_MSC_VER >= 1700) /* MS Visual Studio 2012 */
88     #define FLAC__AVX2_SUPPORTED 1
89     #define FLAC__FMA_SUPPORTED 1
90   #endif
91 #elif defined __GNUC__ && defined __clang__
92   #if defined __clang__ && __has_attribute(__target__) /* clang */
93     #define FLAC__SSE_TARGET(x) __attribute__ ((__target__ (x)))
94     #if __has_builtin(__builtin_ia32_maxps)
95       #define FLAC__SSE_SUPPORTED 1
96     #endif
97     #if __has_builtin(__builtin_ia32_pmuludq128)
98       #define FLAC__SSE2_SUPPORTED 1
99     #endif
100     #if __has_builtin(__builtin_ia32_pabsd128)
101       #define FLAC__SSSE3_SUPPORTED 1
102     #endif
103     #if __has_builtin(__builtin_ia32_pmuldq128)
104       #define FLAC__SSE4_1_SUPPORTED 1
105     #endif
106     #if __has_builtin(__builtin_ia32_pabsd256)
107       #define FLAC__AVX2_SUPPORTED 1
108     #endif
109 #elif defined (__GNUC__) &&  (__GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 9)) /* GCC 4.9+ */
110     #define FLAC__SSE_TARGET(x) __attribute__ ((__target__ (x)))
111     #define FLAC__SSE_SUPPORTED 1
112     #define FLAC__SSE2_SUPPORTED 1
113     #define FLAC__SSSE3_SUPPORTED 1
114     #define FLAC__SSE4_1_SUPPORTED 1
115 #ifdef FLAC__USE_AVX
116     #define FLAC__AVX_SUPPORTED 1
117     #define FLAC__AVX2_SUPPORTED 1
118     #define FLAC__FMA_SUPPORTED 1
119 #endif
120   #else /* older GCC and clang */
121     #define FLAC__SSE_TARGET(x)
122     #ifdef __SSE__
123       #define FLAC__SSE_SUPPORTED 1
124     #endif
125     #ifdef __SSE2__
126       #define FLAC__SSE2_SUPPORTED 1
127     #endif
128     #ifdef __SSSE3__
129       #define FLAC__SSSE3_SUPPORTED 1
130     #endif
131     #ifdef __SSE4_1__
132       #define FLAC__SSE4_1_SUPPORTED 1
133     #endif
134     #ifdef __AVX__
135       #define FLAC__AVX_SUPPORTED 1
136     #endif
137     #ifdef __AVX2__
138       #define FLAC__AVX2_SUPPORTED 1
139     #endif
140     #ifdef __FMA__
141       #define FLAC__FMA_SUPPORTED 1
142     #endif
143   #endif
144 #endif /* compiler version */
145 #endif /* intrinsics support */
146
147
148 #ifndef FLAC__AVX_SUPPORTED
149 #define FLAC__AVX_SUPPORTED 0
150 #endif
151
152 typedef enum {
153         FLAC__CPUINFO_TYPE_IA32,
154         FLAC__CPUINFO_TYPE_X86_64,
155         FLAC__CPUINFO_TYPE_UNKNOWN
156 } FLAC__CPUInfo_Type;
157
158 typedef struct {
159         FLAC__bool intel;
160
161         FLAC__bool cmov;
162         FLAC__bool mmx;
163         FLAC__bool sse;
164         FLAC__bool sse2;
165
166         FLAC__bool sse3;
167         FLAC__bool ssse3;
168         FLAC__bool sse41;
169         FLAC__bool sse42;
170         FLAC__bool avx;
171         FLAC__bool avx2;
172         FLAC__bool fma;
173 } FLAC__CPUInfo_IA32;
174
175 typedef struct {
176         FLAC__bool intel;
177
178         FLAC__bool sse3;
179         FLAC__bool ssse3;
180         FLAC__bool sse41;
181         FLAC__bool sse42;
182         FLAC__bool avx;
183         FLAC__bool avx2;
184         FLAC__bool fma;
185 } FLAC__CPUInfo_x86;
186
187
188 typedef struct {
189         FLAC__bool use_asm;
190         FLAC__CPUInfo_Type type;
191         FLAC__CPUInfo_IA32 ia32;
192         FLAC__CPUInfo_x86 x86;
193 } FLAC__CPUInfo;
194
195 void FLAC__cpu_info(FLAC__CPUInfo *info);
196
197 FLAC__uint32 FLAC__cpu_have_cpuid_asm_ia32(void);
198
199 void         FLAC__cpu_info_asm_ia32(FLAC__uint32 *flags_edx, FLAC__uint32 *flags_ecx);
200
201 void         FLAC__cpu_info_x86(FLAC__uint32 level, FLAC__uint32 *eax, FLAC__uint32 *ebx, FLAC__uint32 *ecx, FLAC__uint32 *edx);
202
203 #endif