Add Clang support for FLAC__SSE_TARGET
[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 #ifndef __has_attribute
59 #define __has_attribute(x) 0
60 #endif
61
62 #if FLAC__HAS_X86INTRIN
63 /* SSE intrinsics support by ICC/MSVC/GCC */
64 #if defined __INTEL_COMPILER
65   #define FLAC__SSE_TARGET(x)
66   #define FLAC__SSE_SUPPORTED 1
67   #define FLAC__SSE2_SUPPORTED 1
68   #if (__INTEL_COMPILER >= 1000) /* Intel C++ Compiler 10.0 */
69     #define FLAC__SSSE3_SUPPORTED 1
70     #define FLAC__SSE4_1_SUPPORTED 1
71   #endif
72   #if (__INTEL_COMPILER >= 1110) /* Intel C++ Compiler 11.1 */
73     #define FLAC__AVX_SUPPORTED 1
74   #endif
75   #if (__INTEL_COMPILER >= 1300) /* Intel C++ Compiler 13.0 */
76     #define FLAC__AVX2_SUPPORTED 1
77     #define FLAC__FMA_SUPPORTED 1
78   #endif
79 #elif defined _MSC_VER
80   #define FLAC__SSE_TARGET(x)
81   #define FLAC__SSE_SUPPORTED 1
82   #define FLAC__SSE2_SUPPORTED 1
83   #if (_MSC_VER >= 1500) /* MS Visual Studio 2008 */
84     #define FLAC__SSSE3_SUPPORTED 1
85     #define FLAC__SSE4_1_SUPPORTED 1
86   #endif
87   #if (_MSC_FULL_VER >= 160040219) /* MS Visual Studio 2010 SP1 */
88     #define FLAC__AVX_SUPPORTED 1
89   #endif
90   #if (_MSC_VER >= 1700) /* MS Visual Studio 2012 */
91     #define FLAC__AVX2_SUPPORTED 1
92     #define FLAC__FMA_SUPPORTED 1
93   #endif
94 #elif defined __clang__ && __has_attribute(__target__) /* clang */
95   #define FLAC__SSE_TARGET(x) __attribute__ ((__target__ (x)))
96   #if __has_builtin(__builtin_ia32_maxps)
97     #define FLAC__SSE_SUPPORTED 1
98   #endif
99   #if __has_builtin(__builtin_ia32_pmuludq128)
100     #define FLAC__SSE2_SUPPORTED 1
101   #endif
102   #if __has_builtin(__builtin_ia32_pabsd128)
103     #define FLAC__SSSE3_SUPPORTED 1
104   #endif
105   #if __has_builtin(__builtin_ia32_pmuldq128)
106     #define FLAC__SSE4_1_SUPPORTED 1
107   #endif
108   #if __has_builtin(__builtin_ia32_pabsd256)
109     #define FLAC__AVX2_SUPPORTED 1
110   #endif
111 #elif defined __GNUC__ && ((__GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 9)) /* GCC 4.9+ */ \
112   || (defined __clang__ && (__clang_major__ > 3) || ( __clang_major__ == 3 || __clang_minor__ >= 8)) /* clang 3.8+ */)
113   #define FLAC__SSE_TARGET(x) __attribute__ ((__target__ (x)))
114   #define FLAC__SSE_SUPPORTED 1
115   #define FLAC__SSE2_SUPPORTED 1
116   #define FLAC__SSSE3_SUPPORTED 1
117   #define FLAC__SSE4_1_SUPPORTED 1
118   #ifdef FLAC__USE_AVX
119     #define FLAC__AVX_SUPPORTED 1
120     #define FLAC__AVX2_SUPPORTED 1
121     #define FLAC__FMA_SUPPORTED 1
122   #endif
123 #else
124   #define FLAC__SSE_TARGET(x)
125   #ifdef __SSE__
126     #define FLAC__SSE_SUPPORTED 1
127   #endif
128   #ifdef __SSE2__
129     #define FLAC__SSE2_SUPPORTED 1
130   #endif
131   #ifdef __SSSE3__
132     #define FLAC__SSSE3_SUPPORTED 1
133   #endif
134   #ifdef __SSE4_1__
135     #define FLAC__SSE4_1_SUPPORTED 1
136   #endif
137   #ifdef __AVX__
138     #define FLAC__AVX_SUPPORTED 1
139   #endif
140   #ifdef __AVX2__
141     #define FLAC__AVX2_SUPPORTED 1
142   #endif
143   #ifdef __FMA__
144     #define FLAC__FMA_SUPPORTED 1
145   #endif
146 #endif /* compiler version */
147 #endif /* intrinsics support */
148
149
150 #ifndef FLAC__AVX_SUPPORTED
151 #define FLAC__AVX_SUPPORTED 0
152 #endif
153
154 typedef enum {
155         FLAC__CPUINFO_TYPE_IA32,
156         FLAC__CPUINFO_TYPE_X86_64,
157         FLAC__CPUINFO_TYPE_UNKNOWN
158 } FLAC__CPUInfo_Type;
159
160 typedef struct {
161         FLAC__bool intel;
162
163         FLAC__bool cmov;
164         FLAC__bool mmx;
165         FLAC__bool sse;
166         FLAC__bool sse2;
167
168         FLAC__bool sse3;
169         FLAC__bool ssse3;
170         FLAC__bool sse41;
171         FLAC__bool sse42;
172         FLAC__bool avx;
173         FLAC__bool avx2;
174         FLAC__bool fma;
175 } FLAC__CPUInfo_x86;
176
177
178 typedef struct {
179         FLAC__bool use_asm;
180         FLAC__CPUInfo_Type type;
181         FLAC__CPUInfo_x86 x86;
182 } FLAC__CPUInfo;
183
184 void FLAC__cpu_info(FLAC__CPUInfo *info);
185
186 FLAC__uint32 FLAC__cpu_have_cpuid_asm_ia32(void);
187
188 void         FLAC__cpu_info_asm_ia32(FLAC__uint32 level, FLAC__uint32 *eax, FLAC__uint32 *ebx, FLAC__uint32 *ecx, FLAC__uint32 *edx);
189
190 #endif