1 /*
2 * xmmintrin.h
3 *
4 * This file is part of the ReactOS CRT package.
5 *
6 * Contributors:
7 * Timo Kreuzer (timo.kreuzer@reactos.org)
8 *
9 * THIS SOFTWARE IS NOT COPYRIGHTED
10 *
11 * This source code is offered for use in the public domain. You may
12 * use, modify or distribute it freely.
13 *
14 * This code is distributed in the hope that it will be useful but
15 * WITHOUT ANY WARRANTY. ALL WARRANTIES, EXPRESS OR IMPLIED ARE HEREBY
16 * DISCLAIMED. This includes but is not limited to warranties of
17 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
18 *
19 */
20
21 #pragma once
22 #ifndef _INCLUDED_MM2
23 #define _INCLUDED_MM2
24
25 #include <mmintrin.h>
26
27 #if defined(_MM2_FUNCTIONALITY) && !defined(_MM_FUNCTIONALITY)
28 #define _MM_FUNCTIONALITY
29 #endif
30
31 #if !defined _VCRT_BUILD && !defined _INC_MALLOC
32 #include <malloc.h> // For _mm_malloc() and _mm_free()
33 #endif
34
35 #ifdef __cplusplus
36 extern "C" {
37 #endif
38
39 #if defined(_MSC_VER) && !defined(__clang__)
40
41 typedef union _DECLSPEC_INTRIN_TYPE _CRT_ALIGN(16) __m128
42 {
43 float m128_f32[4];
44 unsigned __int64 m128_u64[2];
45 __int8 m128_i8[16];
46 __int16 m128_i16[8];
47 __int32 m128_i32[4];
48 __int64 m128_i64[2];
49 unsigned __int8 m128_u8[16];
50 unsigned __int16 m128_u16[8];
51 unsigned __int32 m128_u32[4];
52 } __m128;
53
54 #define __ATTRIBUTE_SSE__
55
56 #else /* _MSC_VER */
57
58 typedef float __v4sf __attribute__((__vector_size__(16)));
59 typedef signed int __v4si __attribute__((__vector_size__(16)));
60 typedef unsigned int __v4su __attribute__((__vector_size__(16)));
61 typedef float __m128_u __attribute__((__vector_size__(16), __aligned__(1)));
62
63 typedef float __m128 __attribute__((__vector_size__(16), __aligned__(16)));
64
65 #ifdef __clang__
66 #define __ATTRIBUTE_SSE__ __attribute__((__target__("sse"),__min_vector_width__(128)))
67 #else
68 #define __ATTRIBUTE_SSE__ __attribute__((__target__("sse")))
69 #endif
70 #define __INTRIN_INLINE_SSE __INTRIN_INLINE __ATTRIBUTE_SSE__
71
72 #endif /* _MSC_VER */
73
74 #define _MM_ALIGN16 _VCRT_ALIGN(16)
75
76 /* Constants for use with _mm_prefetch. */
77 #define _MM_HINT_NTA 0
78 #define _MM_HINT_T0 1
79 #define _MM_HINT_T1 2
80 #define _MM_HINT_T2 3
81 #define _MM_HINT_ENTA 4
82 #if 0 // Not supported yet
83 #define _MM_HINT_ET0 5
84 #define _MM_HINT_ET1 6
85 #define _MM_HINT_ET2 7
86 #endif
87
88 /* Create a selector for use with the SHUFPS instruction. */
89 #define _MM_SHUFFLE(fp3, fp2, fp1, fp0) \
90 (((fp3) << 6) | ((fp2) << 4) | ((fp1) << 2) | (fp0))
91
92 /* Bits in the MXCSR. */
93 #define _MM_EXCEPT_MASK 0x003f
94 #define _MM_EXCEPT_INVALID 0x0001
95 #define _MM_EXCEPT_DENORM 0x0002
96 #define _MM_EXCEPT_DIV_ZERO 0x0004
97 #define _MM_EXCEPT_OVERFLOW 0x0008
98 #define _MM_EXCEPT_UNDERFLOW 0x0010
99 #define _MM_EXCEPT_INEXACT 0x0020
100
101 #define _MM_MASK_MASK 0x1f80
102 #define _MM_MASK_INVALID 0x0080
103 #define _MM_MASK_DENORM 0x0100
104 #define _MM_MASK_DIV_ZERO 0x0200
105 #define _MM_MASK_OVERFLOW 0x0400
106 #define _MM_MASK_UNDERFLOW 0x0800
107 #define _MM_MASK_INEXACT 0x1000
108
109 #define _MM_ROUND_MASK 0x6000
110 #define _MM_ROUND_NEAREST 0x0000
111 #define _MM_ROUND_DOWN 0x2000
112 #define _MM_ROUND_UP 0x4000
113 #define _MM_ROUND_TOWARD_ZERO 0x6000
114
115 #define _MM_FLUSH_ZERO_MASK 0x8000
116 #define _MM_FLUSH_ZERO_ON 0x8000
117 #define _MM_FLUSH_ZERO_OFF 0x0000
118
119 #ifdef __ICL
120 void* __cdecl _mm_malloc(size_t Size, size_t Al);
121 void __cdecl _mm_free(void* P);
122 #endif
123
124 void _mm_prefetch(_In_ char const* p, _In_ int i);
125 __m128 _mm_setzero_ps(void);
126 __m128 _mm_add_ss(__m128 a, __m128 b);
127 __m128 _mm_sub_ss(__m128 a, __m128 b);
128 __m128 _mm_mul_ss(__m128 a, __m128 b);
129 __m128 _mm_div_ss(__m128 a, __m128 b);
130 __m128 _mm_sqrt_ss(__m128 a);
131 __m128 _mm_rcp_ss(__m128 a);
132 __m128 _mm_rsqrt_ss(__m128 a);
133 __m128 _mm_min_ss(__m128 a, __m128 b);
134 __m128 _mm_max_ss(__m128 a, __m128 b);
135 __m128 _mm_add_ps(__m128 a, __m128 b);
136 __m128 _mm_sub_ps(__m128 a, __m128 b);
137 __m128 _mm_mul_ps(__m128 a, __m128 b);
138 __m128 _mm_div_ps(__m128 a, __m128 b);
139 __m128 _mm_sqrt_ps(__m128 a);
140 __m128 _mm_rcp_ps(__m128 a);
141 __m128 _mm_rsqrt_ps(__m128 a);
142 __m128 _mm_min_ps(__m128 a, __m128 b);
143 __m128 _mm_max_ps(__m128 a, __m128 b);
144 __m128 _mm_and_ps(__m128 a, __m128 b);
145 __m128 _mm_andnot_ps(__m128 a, __m128 b);
146 __m128 _mm_or_ps(__m128 a, __m128 b);
147 __m128 _mm_xor_ps(__m128 a, __m128 b);
148 __m128 _mm_cmpeq_ss(__m128 a, __m128 b);
149 __m128 _mm_cmplt_ss(__m128 a, __m128 b);
150 __m128 _mm_cmple_ss(__m128 a, __m128 b);
151 __m128 _mm_cmpgt_ss(__m128 a, __m128 b);
152 __m128 _mm_cmpge_ss(__m128 a, __m128 b);
153 __m128 _mm_cmpneq_ss(__m128 a, __m128 b);
154 __m128 _mm_cmpnlt_ss(__m128 a, __m128 b);
155 __m128 _mm_cmpnle_ss(__m128 a, __m128 b);
156 __m128 _mm_cmpngt_ss(__m128 a, __m128 b);
157 __m128 _mm_cmpnge_ss(__m128 a, __m128 b);
158 __m128 _mm_cmpord_ss(__m128 a, __m128 b);
159 __m128 _mm_cmpunord_ss(__m128 a, __m128 b);
160 __m128 _mm_cmpeq_ps(__m128 a, __m128 b);
161 __m128 _mm_cmplt_ps(__m128 a, __m128 b);
162 __m128 _mm_cmple_ps(__m128 a, __m128 b);
163 __m128 _mm_cmpgt_ps(__m128 a, __m128 b);
164 __m128 _mm_cmpge_ps(__m128 a, __m128 b);
165 __m128 _mm_cmpneq_ps(__m128 a, __m128 b);
166 __m128 _mm_cmpnlt_ps(__m128 a, __m128 b);
167 __m128 _mm_cmpnle_ps(__m128 a, __m128 b);
168 __m128 _mm_cmpngt_ps(__m128 a, __m128 b);
169 __m128 _mm_cmpnge_ps(__m128 a, __m128 b);
170 __m128 _mm_cmpord_ps(__m128 a, __m128 b);
171 __m128 _mm_cmpunord_ps(__m128 a, __m128 b);
172 int _mm_comieq_ss(__m128 a, __m128 b);
173 int _mm_comilt_ss(__m128 a, __m128 b);
174 int _mm_comile_ss(__m128 a, __m128 b);
175 int _mm_comigt_ss(__m128 a, __m128 b);
176 int _mm_comige_ss(__m128 a, __m128 b);
177 int _mm_comineq_ss(__m128 a, __m128 b);
178 int _mm_ucomieq_ss(__m128 a, __m128 b);
179 int _mm_ucomilt_ss(__m128 a, __m128 b);
180 int _mm_ucomile_ss(__m128 a, __m128 b);
181 int _mm_ucomigt_ss(__m128 a, __m128 b);
182 int _mm_ucomige_ss(__m128 a, __m128 b);
183 int _mm_ucomineq_ss(__m128 a, __m128 b);
184 int _mm_cvt_ss2si(__m128 a);
185 int _mm_cvtt_ss2si(__m128 a);
186 __m128 _mm_cvt_si2ss(__m128 a, int b);
187 #ifdef _M_IX86
188 __m64 _mm_cvt_ps2pi(__m128 a);
189 __m64 _mm_cvtt_ps2pi(__m128 a);
190 __m128 _mm_cvt_pi2ps(__m128 a, __m64 b);
191 #endif
192 __m128 _mm_shuffle_ps(__m128 a, __m128 b, unsigned int imm8);
193 __m128 _mm_unpackhi_ps(__m128 a, __m128 b);
194 __m128 _mm_unpacklo_ps(__m128 a, __m128 b);
195 __m128 _mm_loadh_pi(__m128 a, __m64 const* p);
196 void _mm_storeh_pi(__m64* p, __m128 a);
197 __m128 _mm_movehl_ps(__m128 a, __m128 b);
198 __m128 _mm_movelh_ps(__m128 a, __m128 b);
199 __m128 _mm_loadl_pi(__m128 a, __m64 const* p);
200 void _mm_storel_pi(__m64* p, __m128 a);
201 int _mm_movemask_ps(__m128 a);
202 unsigned int _mm_getcsr(void);
203 void _mm_setcsr(unsigned int a);
204 __m128 _mm_set_ss(float a);
205 __m128 _mm_set_ps1(float a);
206 __m128 _mm_load_ss(float const* p);
207 __m128 _mm_load_ps1(float const* p);
208 __m128 _mm_load_ps(float const* p);
209 __m128 _mm_loadu_ps(float const* p);
210 __m128 _mm_loadr_ps(float const* p);
211 __m128 _mm_set_ps(float e3, float e2, float e1, float e0);
212 __m128 _mm_setr_ps(float e3, float e2, float e1, float e0);
213 void _mm_store_ss(float* p, __m128 a);
214 float _mm_cvtss_f32(__m128 a);
215 void _mm_store_ps(float* p, __m128 a);
216 void _mm_storeu_ps(float* p, __m128 a);
217 void _mm_store_ps1(float* p, __m128 a);
218 void _mm_storer_ps(float* p, __m128 a);
219 __m128 _mm_move_ss(__m128 a, __m128 b);
220 #ifdef _M_IX86
221 int _m_pextrw(__m64 a, int imm8);
222 __m64 _m_pinsrw(__m64 a, int i, int imm8);
223 __m64 _m_pmaxsw(__m64 a, __m64 b);
224 __m64 _m_pmaxub(__m64 a, __m64 b);
225 __m64 _m_pminsw(__m64 a, __m64 b);
226 __m64 _m_pminub(__m64 a, __m64 b);
227 int _m_pmovmskb(__m64 a);
228 __m64 _m_pmulhuw(__m64 a, __m64 b);
229 __m64 _m_pshufw(__m64 a, int imm8);
230 void _m_maskmovq(__m64 a, __m64 b, char*);
231 __m64 _m_pavgb(__m64 a, __m64 b);
232 __m64 _m_pavgw(__m64 a, __m64 b);
233 __m64 _m_psadbw(__m64 a, __m64 b);
234 void _mm_stream_pi(__m64* p, __m64 a);
235 #endif
236 void _mm_stream_ps(float* p, __m128 a);
237 void _mm_sfence(void);
238 #ifdef _M_AMD64
239 __int64 _mm_cvtss_si64(__m128 a);
240 __int64 _mm_cvttss_si64(__m128 a);
241 __m128 _mm_cvtsi64_ss(__m128 a, __int64 b);
242 #endif
243
244 /* Alternate names */
245 #define _mm_cvtss_si32 _mm_cvt_ss2si
246 #define _mm_cvttss_si32 _mm_cvtt_ss2si
247 #define _mm_cvtsi32_ss _mm_cvt_si2ss
248 #define _mm_set1_ps _mm_set_ps1
249 #define _mm_load1_ps _mm_load_ps1f
250 #define _mm_store1_ps _mm_store_ps1
251 #define _mm_cvtps_pi32 _mm_cvt_ps2pi
252 #define _mm_cvttps_pi32 _mm_cvtt_ps2pi
253 #define _mm_cvtpi32_ps _mm_cvt_pi2ps
254 #define _mm_extract_pi16 _m_pextrw
255 #define _mm_insert_pi16 _m_pinsrw
256 #define _mm_max_pi16 _m_pmaxsw
257 #define _mm_max_pu8 _m_pmaxub
258 #define _mm_min_pi16 _m_pminsw
259 #define _mm_min_pu8 _m_pminub
260 #define _mm_movemask_pi8 _m_pmovmskb
261 #define _mm_mulhi_pu16 _m_pmulhuw
262 #define _mm_shuffle_pi16 _m_pshufw
263 #define _mm_maskmove_si64 _m_maskmovq
264 #define _mm_avg_pu8 _m_pavgb
265 #define _mm_avg_pu16 _m_pavgw
266 #define _mm_sad_pu8 _m_psadbw
267
268 #ifdef _M_IX86
269 /* Inline functions from Clang: https://github.com/llvm/llvm-project/blob/main/clang/lib/Headers/xmmintrin.h */
270
271 __ATTRIBUTE_SSE__
_mm_cvtpi16_ps(__m64 __a)272 static __inline __m128 _mm_cvtpi16_ps(__m64 __a)
273 {
274 __m64 __b, __c;
275 __m128 __r;
276
277 __b = _mm_setzero_si64();
278 __b = _mm_cmpgt_pi16(__b, __a);
279 __c = _mm_unpackhi_pi16(__a, __b);
280 __r = _mm_setzero_ps();
281 __r = _mm_cvtpi32_ps(__r, __c);
282 __r = _mm_movelh_ps(__r, __r);
283 __c = _mm_unpacklo_pi16(__a, __b);
284 __r = _mm_cvtpi32_ps(__r, __c);
285
286 return __r;
287 }
288
289 __ATTRIBUTE_SSE__
_mm_cvtpu16_ps(__m64 __a)290 static __inline __m128 _mm_cvtpu16_ps(__m64 __a)
291 {
292 __m64 __b, __c;
293 __m128 __r;
294
295 __b = _mm_setzero_si64();
296 __c = _mm_unpackhi_pi16(__a, __b);
297 __r = _mm_setzero_ps();
298 __r = _mm_cvtpi32_ps(__r, __c);
299 __r = _mm_movelh_ps(__r, __r);
300 __c = _mm_unpacklo_pi16(__a, __b);
301 __r = _mm_cvtpi32_ps(__r, __c);
302
303 return __r;
304 }
305
306 __ATTRIBUTE_SSE__
_mm_cvtpi8_ps(__m64 __a)307 static __inline __m128 _mm_cvtpi8_ps(__m64 __a)
308 {
309 __m64 __b;
310
311 __b = _mm_setzero_si64();
312 __b = _mm_cmpgt_pi8(__b, __a);
313 __b = _mm_unpacklo_pi8(__a, __b);
314
315 return _mm_cvtpi16_ps(__b);
316 }
317
318 __ATTRIBUTE_SSE__
_mm_cvtpu8_ps(__m64 __a)319 static __inline __m128 _mm_cvtpu8_ps(__m64 __a)
320 {
321 __m64 __b;
322
323 __b = _mm_setzero_si64();
324 __b = _mm_unpacklo_pi8(__a, __b);
325
326 return _mm_cvtpi16_ps(__b);
327 }
328
329 __ATTRIBUTE_SSE__
_mm_cvtpi32x2_ps(__m64 __a,__m64 __b)330 static __inline __m128 _mm_cvtpi32x2_ps(__m64 __a, __m64 __b)
331 {
332 __m128 __c;
333
334 __c = _mm_setzero_ps();
335 __c = _mm_cvtpi32_ps(__c, __b);
336 __c = _mm_movelh_ps(__c, __c);
337
338 return _mm_cvtpi32_ps(__c, __a);
339 }
340
341 __ATTRIBUTE_SSE__
_mm_cvtps_pi16(__m128 __a)342 static __inline __m64 _mm_cvtps_pi16(__m128 __a)
343 {
344 __m64 __b, __c;
345
346 __b = _mm_cvtps_pi32(__a);
347 __a = _mm_movehl_ps(__a, __a);
348 __c = _mm_cvtps_pi32(__a);
349
350 return _mm_packs_pi32(__b, __c);
351 }
352
353 __ATTRIBUTE_SSE__
_mm_cvtps_pi8(__m128 __a)354 static __inline __m64 _mm_cvtps_pi8(__m128 __a)
355 {
356 __m64 __b, __c;
357
358 __b = _mm_cvtps_pi16(__a);
359 __c = _mm_setzero_si64();
360
361 return _mm_packs_pi16(__b, __c);
362 }
363
364 #endif /* _M_IX86 */
365
366 /* Transpose the 4x4 matrix composed of row[0-3]. */
367 #define _MM_TRANSPOSE4_PS(row0, row1, row2, row3) \
368 do { \
369 __m128 t0 = _mm_unpacklo_ps(row0, row1); \
370 __m128 t1 = _mm_unpacklo_ps(row2, row3); \
371 __m128 t2 = _mm_unpackhi_ps(row0, row1); \
372 __m128 t3 = _mm_unpackhi_ps(row2, row3); \
373 (row0) = _mm_movelh_ps(t0, t1); \
374 (row1) = _mm_movehl_ps(t1, t0); \
375 (row2) = _mm_movelh_ps(t2, t3); \
376 (row3) = _mm_movehl_ps(t3, t2); \
377 } while (0)
378
379 #define _MM_GET_EXCEPTION_STATE() \
380 (_mm_getcsr() & _MM_EXCEPT_MASK)
381
382 #define _MM_GET_EXCEPTION_MASK() \
383 (_mm_getcsr() & _MM_MASK_MASK)
384
385 #define _MM_GET_ROUNDING_MODE() \
386 (_mm_getcsr() & _MM_ROUND_MASK)
387
388 #define _MM_GET_FLUSH_ZERO_MODE() \
389 (_mm_getcsr() & _MM_FLUSH_ZERO_MASK)
390
391 #define _MM_SET_EXCEPTION_STATE(__mask) \
392 _mm_setcsr((_mm_getcsr() & ~_MM_EXCEPT_MASK) | (__mask))
393
394 #define _MM_SET_EXCEPTION_MASK(__mask) \
395 _mm_setcsr((_mm_getcsr() & ~_MM_MASK_MASK) | (__mask))
396
397 #define _MM_SET_ROUNDING_MODE(__mode) \
398 _mm_setcsr((_mm_getcsr() & ~_MM_ROUND_MASK) | (__mode))
399
400 #define _MM_SET_FLUSH_ZERO_MODE(__mode) \
401 _mm_setcsr((_mm_getcsr() & ~_MM_FLUSH_ZERO_MASK) | (__mode))
402
403 /* Use intrinsics on MSVC */
404 #if defined(_MSC_VER) && !defined(__clang__)
405 #pragma intrinsic(_mm_prefetch)
406 #pragma intrinsic(_mm_setzero_ps)
407 #pragma intrinsic(_mm_add_ss)
408 #pragma intrinsic(_mm_sub_ss)
409 #pragma intrinsic(_mm_mul_ss)
410 #pragma intrinsic(_mm_div_ss)
411 #pragma intrinsic(_mm_sqrt_ss)
412 #pragma intrinsic(_mm_rcp_ss)
413 #pragma intrinsic(_mm_rsqrt_ss)
414 #pragma intrinsic(_mm_min_ss)
415 #pragma intrinsic(_mm_max_ss)
416 #pragma intrinsic(_mm_add_ps)
417 #pragma intrinsic(_mm_sub_ps)
418 #pragma intrinsic(_mm_mul_ps)
419 #pragma intrinsic(_mm_div_ps)
420 #pragma intrinsic(_mm_sqrt_ps)
421 #pragma intrinsic(_mm_rcp_ps)
422 #pragma intrinsic(_mm_rsqrt_ps)
423 #pragma intrinsic(_mm_min_ps)
424 #pragma intrinsic(_mm_max_ps)
425 #pragma intrinsic(_mm_and_ps)
426 #pragma intrinsic(_mm_andnot_ps)
427 #pragma intrinsic(_mm_or_ps)
428 #pragma intrinsic(_mm_xor_ps)
429 #pragma intrinsic(_mm_cmpeq_ss)
430 #pragma intrinsic(_mm_cmplt_ss)
431 #pragma intrinsic(_mm_cmple_ss)
432 #pragma intrinsic(_mm_cmpgt_ss)
433 #pragma intrinsic(_mm_cmpge_ss)
434 #pragma intrinsic(_mm_cmpneq_ss)
435 #pragma intrinsic(_mm_cmpnlt_ss)
436 #pragma intrinsic(_mm_cmpnle_ss)
437 #pragma intrinsic(_mm_cmpngt_ss)
438 #pragma intrinsic(_mm_cmpnge_ss)
439 #pragma intrinsic(_mm_cmpord_ss)
440 #pragma intrinsic(_mm_cmpunord_ss)
441 #pragma intrinsic(_mm_cmpeq_ps)
442 #pragma intrinsic(_mm_cmplt_ps)
443 #pragma intrinsic(_mm_cmple_ps)
444 #pragma intrinsic(_mm_cmpgt_ps)
445 #pragma intrinsic(_mm_cmpge_ps)
446 #pragma intrinsic(_mm_cmpneq_ps)
447 #pragma intrinsic(_mm_cmpnlt_ps)
448 #pragma intrinsic(_mm_cmpnle_ps)
449 #pragma intrinsic(_mm_cmpngt_ps)
450 #pragma intrinsic(_mm_cmpnge_ps)
451 #pragma intrinsic(_mm_cmpord_ps)
452 #pragma intrinsic(_mm_cmpunord_ps)
453 #pragma intrinsic(_mm_comieq_ss)
454 #pragma intrinsic(_mm_comilt_ss)
455 #pragma intrinsic(_mm_comile_ss)
456 #pragma intrinsic(_mm_comigt_ss)
457 #pragma intrinsic(_mm_comige_ss)
458 #pragma intrinsic(_mm_comineq_ss)
459 #pragma intrinsic(_mm_ucomieq_ss)
460 #pragma intrinsic(_mm_ucomilt_ss)
461 #pragma intrinsic(_mm_ucomile_ss)
462 #pragma intrinsic(_mm_ucomigt_ss)
463 #pragma intrinsic(_mm_ucomige_ss)
464 #pragma intrinsic(_mm_ucomineq_ss)
465 #pragma intrinsic(_mm_cvt_ss2si)
466 #pragma intrinsic(_mm_cvtt_ss2si)
467 #pragma intrinsic(_mm_cvt_si2ss)
468 #ifdef _M_IX86
469 #pragma intrinsic(_mm_cvt_ps2pi)
470 #pragma intrinsic(_mm_cvtt_ps2pi)
471 #pragma intrinsic(_mm_cvt_pi2ps)
472 #endif // _M_IX86
473 #pragma intrinsic(_mm_shuffle_ps)
474 #pragma intrinsic(_mm_unpackhi_ps)
475 #pragma intrinsic(_mm_unpacklo_ps)
476 #pragma intrinsic(_mm_loadh_pi)
477 #pragma intrinsic(_mm_storeh_pi)
478 #pragma intrinsic(_mm_movehl_ps)
479 #pragma intrinsic(_mm_movelh_ps)
480 #pragma intrinsic(_mm_loadl_pi)
481 #pragma intrinsic(_mm_storel_pi)
482 #pragma intrinsic(_mm_movemask_ps)
483 #pragma intrinsic(_mm_getcsr)
484 #pragma intrinsic(_mm_setcsr)
485 #pragma intrinsic(_mm_set_ss)
486 #pragma intrinsic(_mm_set_ps1)
487 #pragma intrinsic(_mm_load_ss)
488 #pragma intrinsic(_mm_load_ps1)
489 #pragma intrinsic(_mm_load_ps)
490 #pragma intrinsic(_mm_loadu_ps)
491 #pragma intrinsic(_mm_loadr_ps)
492 #pragma intrinsic(_mm_set_ps)
493 #pragma intrinsic(_mm_setr_ps)
494 #pragma intrinsic(_mm_store_ss)
495 #pragma intrinsic(_mm_cvtss_f32)
496 #pragma intrinsic(_mm_store_ps)
497 #pragma intrinsic(_mm_storeu_ps)
498 #pragma intrinsic(_mm_store_ps1)
499 #pragma intrinsic(_mm_storer_ps)
500 #pragma intrinsic(_mm_move_ss)
501 #ifdef _M_IX86
502 #pragma intrinsic(_m_pextrw)
503 #pragma intrinsic(_m_pinsrw)
504 #pragma intrinsic(_m_pmaxsw)
505 #pragma intrinsic(_m_pmaxub)
506 #pragma intrinsic(_m_pminsw)
507 #pragma intrinsic(_m_pminub)
508 #pragma intrinsic(_m_pmovmskb)
509 #pragma intrinsic(_m_pmulhuw)
510 #pragma intrinsic(_m_pshufw)
511 #pragma intrinsic(_m_maskmovq)
512 #pragma intrinsic(_m_pavgb)
513 #pragma intrinsic(_m_pavgw)
514 #pragma intrinsic(_m_psadbw)
515 #pragma intrinsic(_mm_stream_pi)
516 #endif // _M_IX86
517 #pragma intrinsic(_mm_stream_ps)
518 #pragma intrinsic(_mm_sfence)
519 #ifdef _M_AMD64
520 #pragma intrinsic(_mm_cvtss_si64)
521 #pragma intrinsic(_mm_cvttss_si64)
522 #pragma intrinsic(_mm_cvtsi64_ss)
523 #endif // _M_AMD64
524
525 #else /* _MSC_VER */
526
527 /*
528 GCC: https://github.com/gcc-mirror/gcc/blob/master/gcc/config/i386/xmmintrin.h
529 Clang: https://github.com/llvm/llvm-project/blob/main/clang/lib/Headers/xmmintrin.h
530 */
531
532 /* Use inline functions on GCC/Clang */
533
534 #if !HAS_BUILTIN(_mm_getcsr) && !defined(_MSC_VER)
_mm_getcsr(void)535 __INTRIN_INLINE_SSE unsigned int _mm_getcsr(void)
536 {
537 return __builtin_ia32_stmxcsr();
538 }
539 #endif
540
541 #if !HAS_BUILTIN(_mm_setcsr) && !defined(_MSC_VER)
_mm_setcsr(unsigned int a)542 __INTRIN_INLINE_SSE void _mm_setcsr(unsigned int a)
543 {
544 __builtin_ia32_ldmxcsr(a);
545 }
546 #endif
547
_mm_add_ss(__m128 __a,__m128 __b)548 __INTRIN_INLINE_SSE __m128 _mm_add_ss(__m128 __a, __m128 __b)
549 {
550 __a[0] += __b[0];
551 return __a;
552 }
553
_mm_add_ps(__m128 __a,__m128 __b)554 __INTRIN_INLINE_SSE __m128 _mm_add_ps(__m128 __a, __m128 __b)
555 {
556 return (__m128)((__v4sf)__a + (__v4sf)__b);
557 }
558
_mm_sub_ss(__m128 __a,__m128 __b)559 __INTRIN_INLINE_SSE __m128 _mm_sub_ss(__m128 __a, __m128 __b)
560 {
561 __a[0] -= __b[0];
562 return __a;
563 }
564
_mm_sub_ps(__m128 __a,__m128 __b)565 __INTRIN_INLINE_SSE __m128 _mm_sub_ps(__m128 __a, __m128 __b)
566 {
567 return (__m128)((__v4sf)__a - (__v4sf)__b);
568 }
569
_mm_mul_ss(__m128 __a,__m128 __b)570 __INTRIN_INLINE_SSE __m128 _mm_mul_ss(__m128 __a, __m128 __b)
571 {
572 __a[0] *= __b[0];
573 return __a;
574 }
575
_mm_mul_ps(__m128 __a,__m128 __b)576 __INTRIN_INLINE_SSE __m128 _mm_mul_ps(__m128 __a, __m128 __b)
577 {
578 return (__m128)((__v4sf)__a * (__v4sf)__b);
579 }
580
_mm_div_ss(__m128 __a,__m128 __b)581 __INTRIN_INLINE_SSE __m128 _mm_div_ss(__m128 __a, __m128 __b)
582 {
583 __a[0] /= __b[0];
584 return __a;
585 }
586
_mm_div_ps(__m128 __a,__m128 __b)587 __INTRIN_INLINE_SSE __m128 _mm_div_ps(__m128 __a, __m128 __b)
588 {
589 return (__m128)((__v4sf)__a / (__v4sf)__b);
590 }
591
_mm_sqrt_ss(__m128 __a)592 __INTRIN_INLINE_SSE __m128 _mm_sqrt_ss(__m128 __a)
593 {
594 return (__m128)__builtin_ia32_sqrtss((__v4sf)__a);
595 }
596
_mm_sqrt_ps(__m128 __a)597 __INTRIN_INLINE_SSE __m128 _mm_sqrt_ps(__m128 __a)
598 {
599 return __builtin_ia32_sqrtps((__v4sf)__a);
600 }
601
_mm_rcp_ss(__m128 __a)602 __INTRIN_INLINE_SSE __m128 _mm_rcp_ss(__m128 __a)
603 {
604 return (__m128)__builtin_ia32_rcpss((__v4sf)__a);
605 }
606
_mm_rcp_ps(__m128 __a)607 __INTRIN_INLINE_SSE __m128 _mm_rcp_ps(__m128 __a)
608 {
609 return (__m128)__builtin_ia32_rcpps((__v4sf)__a);
610 }
611
_mm_rsqrt_ss(__m128 __a)612 __INTRIN_INLINE_SSE __m128 _mm_rsqrt_ss(__m128 __a)
613 {
614 return __builtin_ia32_rsqrtss((__v4sf)__a);
615 }
616
_mm_rsqrt_ps(__m128 __a)617 __INTRIN_INLINE_SSE __m128 _mm_rsqrt_ps(__m128 __a)
618 {
619 return __builtin_ia32_rsqrtps((__v4sf)__a);
620 }
621
_mm_min_ss(__m128 __a,__m128 __b)622 __INTRIN_INLINE_SSE __m128 _mm_min_ss(__m128 __a, __m128 __b)
623 {
624 return __builtin_ia32_minss((__v4sf)__a, (__v4sf)__b);
625 }
626
_mm_min_ps(__m128 __a,__m128 __b)627 __INTRIN_INLINE_SSE __m128 _mm_min_ps(__m128 __a, __m128 __b)
628 {
629 return __builtin_ia32_minps((__v4sf)__a, (__v4sf)__b);
630 }
631
_mm_max_ss(__m128 __a,__m128 __b)632 __INTRIN_INLINE_SSE __m128 _mm_max_ss(__m128 __a, __m128 __b)
633 {
634 return __builtin_ia32_maxss((__v4sf)__a, (__v4sf)__b);
635 }
636
_mm_max_ps(__m128 __a,__m128 __b)637 __INTRIN_INLINE_SSE __m128 _mm_max_ps(__m128 __a, __m128 __b)
638 {
639 return __builtin_ia32_maxps((__v4sf)__a, (__v4sf)__b);
640 }
641
_mm_and_ps(__m128 __a,__m128 __b)642 __INTRIN_INLINE_SSE __m128 _mm_and_ps(__m128 __a, __m128 __b)
643 {
644 return (__m128)((__v4su)__a & (__v4su)__b);
645 }
646
_mm_andnot_ps(__m128 __a,__m128 __b)647 __INTRIN_INLINE_SSE __m128 _mm_andnot_ps(__m128 __a, __m128 __b)
648 {
649 return (__m128)(~(__v4su)__a & (__v4su)__b);
650 }
651
_mm_or_ps(__m128 __a,__m128 __b)652 __INTRIN_INLINE_SSE __m128 _mm_or_ps(__m128 __a, __m128 __b)
653 {
654 return (__m128)((__v4su)__a | (__v4su)__b);
655 }
656
_mm_xor_ps(__m128 __a,__m128 __b)657 __INTRIN_INLINE_SSE __m128 _mm_xor_ps(__m128 __a, __m128 __b)
658 {
659 return (__m128)((__v4su)__a ^ (__v4su)__b);
660 }
661
_mm_cmpeq_ss(__m128 __a,__m128 __b)662 __INTRIN_INLINE_SSE __m128 _mm_cmpeq_ss(__m128 __a, __m128 __b)
663 {
664 return (__m128)__builtin_ia32_cmpeqss((__v4sf)__a, (__v4sf)__b);
665 }
666
_mm_cmpeq_ps(__m128 __a,__m128 __b)667 __INTRIN_INLINE_SSE __m128 _mm_cmpeq_ps(__m128 __a, __m128 __b)
668 {
669 return (__m128)__builtin_ia32_cmpeqps((__v4sf)__a, (__v4sf)__b);
670 }
671
_mm_cmplt_ss(__m128 __a,__m128 __b)672 __INTRIN_INLINE_SSE __m128 _mm_cmplt_ss(__m128 __a, __m128 __b)
673 {
674 return (__m128)__builtin_ia32_cmpltss((__v4sf)__a, (__v4sf)__b);
675 }
676
_mm_cmplt_ps(__m128 __a,__m128 __b)677 __INTRIN_INLINE_SSE __m128 _mm_cmplt_ps(__m128 __a, __m128 __b)
678 {
679 return (__m128)__builtin_ia32_cmpltps((__v4sf)__a, (__v4sf)__b);
680 }
681
_mm_cmple_ss(__m128 __a,__m128 __b)682 __INTRIN_INLINE_SSE __m128 _mm_cmple_ss(__m128 __a, __m128 __b)
683 {
684 return (__m128)__builtin_ia32_cmpless((__v4sf)__a, (__v4sf)__b);
685 }
686
_mm_cmple_ps(__m128 __a,__m128 __b)687 __INTRIN_INLINE_SSE __m128 _mm_cmple_ps(__m128 __a, __m128 __b)
688 {
689 return (__m128)__builtin_ia32_cmpleps((__v4sf)__a, (__v4sf)__b);
690 }
691
_mm_cmpgt_ss(__m128 __a,__m128 __b)692 __INTRIN_INLINE_SSE __m128 _mm_cmpgt_ss(__m128 __a, __m128 __b)
693 {
694 __v4sf temp = __builtin_ia32_cmpltss((__v4sf)__b, (__v4sf)__a);
695 #ifdef __clang__
696 return (__m128)__builtin_shufflevector((__v4sf)__a, temp, 4, 1, 2, 3);
697 #else
698 return (__m128)__builtin_ia32_movss((__v4sf)__a, temp);
699 #endif
700 }
701
_mm_cmpgt_ps(__m128 __a,__m128 __b)702 __INTRIN_INLINE_SSE __m128 _mm_cmpgt_ps(__m128 __a, __m128 __b)
703 {
704 return (__m128)__builtin_ia32_cmpltps((__v4sf)__b, (__v4sf)__a);
705 }
706
_mm_cmpge_ss(__m128 __a,__m128 __b)707 __INTRIN_INLINE_SSE __m128 _mm_cmpge_ss(__m128 __a, __m128 __b)
708 {
709 __v4sf temp = __builtin_ia32_cmpless((__v4sf)__b, (__v4sf)__a);
710 #ifdef __clang__
711 return (__m128)__builtin_shufflevector((__v4sf)__a, temp, 4, 1, 2, 3);
712 #else
713 return (__m128)__builtin_ia32_movss((__v4sf)__a, temp);
714 #endif
715 }
716
_mm_cmpge_ps(__m128 __a,__m128 __b)717 __INTRIN_INLINE_SSE __m128 _mm_cmpge_ps(__m128 __a, __m128 __b)
718 {
719 return (__m128)__builtin_ia32_cmpleps((__v4sf)__b, (__v4sf)__a);
720 }
721
_mm_cmpneq_ss(__m128 __a,__m128 __b)722 __INTRIN_INLINE_SSE __m128 _mm_cmpneq_ss(__m128 __a, __m128 __b)
723 {
724 return (__m128)__builtin_ia32_cmpneqss((__v4sf)__a, (__v4sf)__b);
725 }
726
_mm_cmpneq_ps(__m128 __a,__m128 __b)727 __INTRIN_INLINE_SSE __m128 _mm_cmpneq_ps(__m128 __a, __m128 __b)
728 {
729 return (__m128)__builtin_ia32_cmpneqps((__v4sf)__a, (__v4sf)__b);
730 }
731
_mm_cmpnlt_ss(__m128 __a,__m128 __b)732 __INTRIN_INLINE_SSE __m128 _mm_cmpnlt_ss(__m128 __a, __m128 __b)
733 {
734 return (__m128)__builtin_ia32_cmpnltss((__v4sf)__a, (__v4sf)__b);
735 }
736
_mm_cmpnlt_ps(__m128 __a,__m128 __b)737 __INTRIN_INLINE_SSE __m128 _mm_cmpnlt_ps(__m128 __a, __m128 __b)
738 {
739 return (__m128)__builtin_ia32_cmpnltps((__v4sf)__a, (__v4sf)__b);
740 }
741
_mm_cmpnle_ss(__m128 __a,__m128 __b)742 __INTRIN_INLINE_SSE __m128 _mm_cmpnle_ss(__m128 __a, __m128 __b)
743 {
744 return (__m128)__builtin_ia32_cmpnless((__v4sf)__a, (__v4sf)__b);
745 }
746
_mm_cmpnle_ps(__m128 __a,__m128 __b)747 __INTRIN_INLINE_SSE __m128 _mm_cmpnle_ps(__m128 __a, __m128 __b)
748 {
749 return (__m128)__builtin_ia32_cmpnleps((__v4sf)__a, (__v4sf)__b);
750 }
751
_mm_cmpngt_ss(__m128 __a,__m128 __b)752 __INTRIN_INLINE_SSE __m128 _mm_cmpngt_ss(__m128 __a, __m128 __b)
753 {
754 __v4sf temp = __builtin_ia32_cmpnltss((__v4sf)__b, (__v4sf)__a);
755 #ifdef __clang__
756 return (__m128)__builtin_shufflevector((__v4sf)__a, temp, 4, 1, 2, 3);
757 #else
758 return (__m128)__builtin_ia32_movss((__v4sf)__a, temp);
759 #endif
760 }
761
_mm_cmpngt_ps(__m128 __a,__m128 __b)762 __INTRIN_INLINE_SSE __m128 _mm_cmpngt_ps(__m128 __a, __m128 __b)
763 {
764 return (__m128)__builtin_ia32_cmpnltps((__v4sf)__b, (__v4sf)__a);
765 }
766
_mm_cmpnge_ss(__m128 __a,__m128 __b)767 __INTRIN_INLINE_SSE __m128 _mm_cmpnge_ss(__m128 __a, __m128 __b)
768 {
769 __v4sf temp = (__v4sf)__builtin_ia32_cmpnless((__v4sf)__b, (__v4sf)__a);
770 #ifdef __clang__
771 return (__m128)__builtin_shufflevector((__v4sf)__a, temp, 4, 1, 2, 3);
772 #else
773 return (__m128)__builtin_ia32_movss((__v4sf)__a, temp);
774 #endif
775 }
776
_mm_cmpnge_ps(__m128 __a,__m128 __b)777 __INTRIN_INLINE_SSE __m128 _mm_cmpnge_ps(__m128 __a, __m128 __b)
778 {
779 return (__m128)__builtin_ia32_cmpnleps((__v4sf)__b, (__v4sf)__a);
780 }
781
_mm_cmpord_ss(__m128 __a,__m128 __b)782 __INTRIN_INLINE_SSE __m128 _mm_cmpord_ss(__m128 __a, __m128 __b)
783 {
784 return (__m128)__builtin_ia32_cmpordss((__v4sf)__a, (__v4sf)__b);
785 }
786
_mm_cmpord_ps(__m128 __a,__m128 __b)787 __INTRIN_INLINE_SSE __m128 _mm_cmpord_ps(__m128 __a, __m128 __b)
788 {
789 return (__m128)__builtin_ia32_cmpordps((__v4sf)__a, (__v4sf)__b);
790 }
791
_mm_cmpunord_ss(__m128 __a,__m128 __b)792 __INTRIN_INLINE_SSE __m128 _mm_cmpunord_ss(__m128 __a, __m128 __b)
793 {
794 return (__m128)__builtin_ia32_cmpunordss((__v4sf)__a, (__v4sf)__b);
795 }
796
_mm_cmpunord_ps(__m128 __a,__m128 __b)797 __INTRIN_INLINE_SSE __m128 _mm_cmpunord_ps(__m128 __a, __m128 __b)
798 {
799 return (__m128)__builtin_ia32_cmpunordps((__v4sf)__a, (__v4sf)__b);
800 }
801
_mm_comieq_ss(__m128 __a,__m128 __b)802 __INTRIN_INLINE_SSE int _mm_comieq_ss(__m128 __a, __m128 __b)
803 {
804 return __builtin_ia32_comieq((__v4sf)__a, (__v4sf)__b);
805 }
806
_mm_comilt_ss(__m128 __a,__m128 __b)807 __INTRIN_INLINE_SSE int _mm_comilt_ss(__m128 __a, __m128 __b)
808 {
809 return __builtin_ia32_comilt((__v4sf)__a, (__v4sf)__b);
810 }
811
_mm_comile_ss(__m128 __a,__m128 __b)812 __INTRIN_INLINE_SSE int _mm_comile_ss(__m128 __a, __m128 __b)
813 {
814 return __builtin_ia32_comile((__v4sf)__a, (__v4sf)__b);
815 }
816
_mm_comigt_ss(__m128 __a,__m128 __b)817 __INTRIN_INLINE_SSE int _mm_comigt_ss(__m128 __a, __m128 __b)
818 {
819 return __builtin_ia32_comigt((__v4sf)__a, (__v4sf)__b);
820 }
821
_mm_comige_ss(__m128 __a,__m128 __b)822 __INTRIN_INLINE_SSE int _mm_comige_ss(__m128 __a, __m128 __b)
823 {
824 return __builtin_ia32_comige((__v4sf)__a, (__v4sf)__b);
825 }
826
_mm_comineq_ss(__m128 __a,__m128 __b)827 __INTRIN_INLINE_SSE int _mm_comineq_ss(__m128 __a, __m128 __b)
828 {
829 return __builtin_ia32_comineq((__v4sf)__a, (__v4sf)__b);
830 }
831
_mm_ucomieq_ss(__m128 __a,__m128 __b)832 __INTRIN_INLINE_SSE int _mm_ucomieq_ss(__m128 __a, __m128 __b)
833 {
834 return __builtin_ia32_ucomieq((__v4sf)__a, (__v4sf)__b);
835 }
836
_mm_ucomilt_ss(__m128 __a,__m128 __b)837 __INTRIN_INLINE_SSE int _mm_ucomilt_ss(__m128 __a, __m128 __b)
838 {
839 return __builtin_ia32_ucomilt((__v4sf)__a, (__v4sf)__b);
840 }
841
_mm_ucomile_ss(__m128 __a,__m128 __b)842 __INTRIN_INLINE_SSE int _mm_ucomile_ss(__m128 __a, __m128 __b)
843 {
844 return __builtin_ia32_ucomile((__v4sf)__a, (__v4sf)__b);
845 }
846
_mm_ucomigt_ss(__m128 __a,__m128 __b)847 __INTRIN_INLINE_SSE int _mm_ucomigt_ss(__m128 __a, __m128 __b)
848 {
849 return __builtin_ia32_ucomigt((__v4sf)__a, (__v4sf)__b);
850 }
851
_mm_ucomige_ss(__m128 __a,__m128 __b)852 __INTRIN_INLINE_SSE int _mm_ucomige_ss(__m128 __a, __m128 __b)
853 {
854 return __builtin_ia32_ucomige((__v4sf)__a, (__v4sf)__b);
855 }
856
_mm_ucomineq_ss(__m128 __a,__m128 __b)857 __INTRIN_INLINE_SSE int _mm_ucomineq_ss(__m128 __a, __m128 __b)
858 {
859 return __builtin_ia32_ucomineq((__v4sf)__a, (__v4sf)__b);
860 }
861
862 // _mm_cvt_ss2si
_mm_cvtss_si32(__m128 __a)863 __INTRIN_INLINE_SSE int _mm_cvtss_si32(__m128 __a)
864 {
865 return __builtin_ia32_cvtss2si((__v4sf)__a);
866 }
867
868 #ifdef _M_AMD64
_mm_cvtss_si64(__m128 __a)869 __INTRIN_INLINE_SSE long long _mm_cvtss_si64(__m128 __a)
870 {
871 return __builtin_ia32_cvtss2si64((__v4sf)__a);
872 }
873 #endif
874
875 // _mm_cvt_ps2pi
_mm_cvtps_pi32(__m128 __a)876 __INTRIN_INLINE_SSE __m64 _mm_cvtps_pi32(__m128 __a)
877 {
878 return (__m64)__builtin_ia32_cvtps2pi((__v4sf)__a);
879 }
880
881 // _mm_cvtt_ss2si
_mm_cvttss_si32(__m128 __a)882 __INTRIN_INLINE_SSE int _mm_cvttss_si32(__m128 __a)
883 {
884 return __builtin_ia32_cvttss2si((__v4sf)__a);
885 }
886
887 #ifdef _M_AMD64
_mm_cvttss_si64(__m128 __a)888 __INTRIN_INLINE_SSE long long _mm_cvttss_si64(__m128 __a)
889 {
890 return __builtin_ia32_cvttss2si64((__v4sf)__a);
891 }
892 #endif
893
894 // _mm_cvtt_ps2pi
_mm_cvttps_pi32(__m128 __a)895 __INTRIN_INLINE_SSE __m64 _mm_cvttps_pi32(__m128 __a)
896 {
897 return (__m64)__builtin_ia32_cvttps2pi((__v4sf)__a);
898 }
899
900 // _mm_cvt_si2ss
_mm_cvtsi32_ss(__m128 __a,int __b)901 __INTRIN_INLINE_SSE __m128 _mm_cvtsi32_ss(__m128 __a, int __b)
902 {
903 __a[0] = __b;
904 return __a;
905 }
906
907 #ifdef _M_AMD64
_mm_cvtsi64_ss(__m128 __a,long long __b)908 __INTRIN_INLINE_SSE __m128 _mm_cvtsi64_ss(__m128 __a, long long __b)
909 {
910 __a[0] = __b;
911 return __a;
912 }
913 #endif
914
915 // _mm_cvt_pi2ps
_mm_cvtpi32_ps(__m128 __a,__m64 __b)916 __INTRIN_INLINE_SSE __m128 _mm_cvtpi32_ps(__m128 __a, __m64 __b)
917 {
918 return __builtin_ia32_cvtpi2ps((__v4sf)__a, (__v2si)__b);
919 }
920
_mm_cvtss_f32(__m128 __a)921 __INTRIN_INLINE_SSE float _mm_cvtss_f32(__m128 __a)
922 {
923 return __a[0];
924 }
925
_mm_loadh_pi(__m128 __a,const __m64 * __p)926 __INTRIN_INLINE_SSE __m128 _mm_loadh_pi(__m128 __a, const __m64 *__p)
927 {
928 #ifdef __clang__
929 typedef float __mm_loadh_pi_v2f32 __attribute__((__vector_size__(8)));
930 struct __mm_loadh_pi_struct {
931 __mm_loadh_pi_v2f32 __u;
932 } __attribute__((__packed__, __may_alias__));
933 __mm_loadh_pi_v2f32 __b = ((const struct __mm_loadh_pi_struct*)__p)->__u;
934 __m128 __bb = __builtin_shufflevector(__b, __b, 0, 1, 0, 1);
935 return __builtin_shufflevector(__a, __bb, 0, 1, 4, 5);
936 #else
937 return (__m128)__builtin_ia32_loadhps(__a, __p);
938 #endif
939 }
940
_mm_loadl_pi(__m128 __a,const __m64 * __p)941 __INTRIN_INLINE_SSE __m128 _mm_loadl_pi(__m128 __a, const __m64 *__p)
942 {
943 #ifdef __clang__
944 typedef float __mm_loadl_pi_v2f32 __attribute__((__vector_size__(8)));
945 struct __mm_loadl_pi_struct {
946 __mm_loadl_pi_v2f32 __u;
947 } __attribute__((__packed__, __may_alias__));
948 __mm_loadl_pi_v2f32 __b = ((const struct __mm_loadl_pi_struct*)__p)->__u;
949 __m128 __bb = __builtin_shufflevector(__b, __b, 0, 1, 0, 1);
950 return __builtin_shufflevector(__a, __bb, 4, 5, 2, 3);
951 #else
952 return (__m128)__builtin_ia32_loadlps(__a, __p);
953 #endif
954 }
955
_mm_load_ss(const float * __p)956 __INTRIN_INLINE_SSE __m128 _mm_load_ss(const float *__p)
957 {
958 return _mm_set_ss(*__p);
959 }
960
961 // _mm_load_ps1
_mm_load1_ps(const float * __p)962 __INTRIN_INLINE_SSE __m128 _mm_load1_ps(const float *__p)
963 {
964 return _mm_set1_ps(*__p);
965 }
966
_mm_load_ps(const float * __p)967 __INTRIN_INLINE_SSE __m128 _mm_load_ps(const float *__p)
968 {
969 return *(const __m128*)__p;
970 }
971
_mm_loadu_ps(const float * __p)972 __INTRIN_INLINE_SSE __m128 _mm_loadu_ps(const float *__p)
973 {
974 struct __loadu_ps {
975 __m128_u __v;
976 } __attribute__((__packed__, __may_alias__));
977 return ((const struct __loadu_ps*)__p)->__v;
978 }
979
_mm_loadr_ps(const float * __p)980 __INTRIN_INLINE_SSE __m128 _mm_loadr_ps(const float *__p)
981 {
982 __m128 __a = _mm_load_ps(__p);
983 #ifdef __clang__
984 return __builtin_shufflevector((__v4sf)__a, (__v4sf)__a, 3, 2, 1, 0);
985 #else
986 return (__m128)__builtin_ia32_shufps(__a, __a, _MM_SHUFFLE(0,1,2,3));
987 #endif
988 }
989
_mm_undefined_ps(void)990 __INTRIN_INLINE_SSE __m128 _mm_undefined_ps(void)
991 {
992 #ifdef __clang__
993 return (__m128)__builtin_ia32_undef128();
994 #else
995 __m128 undef = undef;
996 return undef;
997 #endif
998 }
999
_mm_set_ss(float __w)1000 __INTRIN_INLINE_SSE __m128 _mm_set_ss(float __w)
1001 {
1002 return __extension__ (__m128){ __w, 0, 0, 0 };
1003 }
1004
1005 // _mm_set_ps1
_mm_set1_ps(float __w)1006 __INTRIN_INLINE_SSE __m128 _mm_set1_ps(float __w)
1007 {
1008 return __extension__ (__m128){ __w, __w, __w, __w };
1009 }
1010
_mm_set_ps(float __z,float __y,float __x,float __w)1011 __INTRIN_INLINE_SSE __m128 _mm_set_ps(float __z, float __y, float __x, float __w)
1012 {
1013 return __extension__ (__m128){ __w, __x, __y, __z };
1014 }
1015
_mm_setr_ps(float __z,float __y,float __x,float __w)1016 __INTRIN_INLINE_SSE __m128 _mm_setr_ps(float __z, float __y, float __x, float __w)
1017 {
1018 return __extension__ (__m128){ __z, __y, __x, __w };
1019 }
1020
_mm_setzero_ps(void)1021 __INTRIN_INLINE_SSE __m128 _mm_setzero_ps(void)
1022 {
1023 return __extension__ (__m128){ 0, 0, 0, 0 };
1024 }
1025
_mm_storeh_pi(__m64 * __p,__m128 __a)1026 __INTRIN_INLINE_SSE void _mm_storeh_pi(__m64 *__p, __m128 __a)
1027 {
1028 #ifdef __clang__
1029 typedef float __mm_storeh_pi_v2f32 __attribute__((__vector_size__(8)));
1030 struct __mm_storeh_pi_struct {
1031 __mm_storeh_pi_v2f32 __u;
1032 } __attribute__((__packed__, __may_alias__));
1033 ((struct __mm_storeh_pi_struct*)__p)->__u = __builtin_shufflevector(__a, __a, 2, 3);
1034 #else
1035 __builtin_ia32_storehps(__p, __a);
1036 #endif
1037 }
1038
_mm_storel_pi(__m64 * __p,__m128 __a)1039 __INTRIN_INLINE_SSE void _mm_storel_pi(__m64 *__p, __m128 __a)
1040 {
1041 #ifdef __clang__
1042 typedef float __mm_storeh_pi_v2f32 __attribute__((__vector_size__(8)));
1043 struct __mm_storeh_pi_struct {
1044 __mm_storeh_pi_v2f32 __u;
1045 } __attribute__((__packed__, __may_alias__));
1046 ((struct __mm_storeh_pi_struct*)__p)->__u = __builtin_shufflevector(__a, __a, 0, 1);
1047 #else
1048 __builtin_ia32_storelps(__p, __a);
1049 #endif
1050 }
1051
_mm_store_ss(float * __p,__m128 __a)1052 __INTRIN_INLINE_SSE void _mm_store_ss(float *__p, __m128 __a)
1053 {
1054 *__p = ((__v4sf)__a)[0];
1055 }
1056
_mm_storeu_ps(float * __p,__m128 __a)1057 __INTRIN_INLINE_SSE void _mm_storeu_ps(float *__p, __m128 __a)
1058 {
1059 *(__m128_u *)__p = __a;
1060 }
1061
_mm_store_ps(float * __p,__m128 __a)1062 __INTRIN_INLINE_SSE void _mm_store_ps(float *__p, __m128 __a)
1063 {
1064 *(__m128*)__p = __a;
1065 }
1066
1067 // _mm_store_ps1
_mm_store1_ps(float * __p,__m128 __a)1068 __INTRIN_INLINE_SSE void _mm_store1_ps(float *__p, __m128 __a)
1069 {
1070 // FIXME: Should we use a temp instead?
1071 #ifdef __clang__
1072 __a = __builtin_shufflevector((__v4sf)__a, (__v4sf)__a, 0, 0, 0, 0);
1073 #else
1074 __a = __builtin_ia32_shufps(__a, __a, _MM_SHUFFLE(0,0,0,0));
1075 #endif
1076 _mm_store_ps(__p, __a);
1077 }
1078
_mm_storer_ps(float * __p,__m128 __a)1079 __INTRIN_INLINE_SSE void _mm_storer_ps(float *__p, __m128 __a)
1080 {
1081 #ifdef __clang__
1082 __m128 __tmp = __builtin_shufflevector((__v4sf)__a, (__v4sf)__a, 3, 2, 1, 0);
1083 #else
1084 __m128 __tmp = __builtin_ia32_shufps(__a, __a, _MM_SHUFFLE(0,1,2,3));
1085 #endif
1086 _mm_store_ps(__p, __tmp);
1087 }
1088
1089 /* GCC / Clang specific consants */
1090 #define _MM_HINT_NTA_ALT 0
1091 #define _MM_HINT_T0_ALT 3
1092 #define _MM_HINT_T1_ALT 2
1093 #define _MM_HINT_T2_ALT 1
1094 #define _MM_HINT_ENTA_ALT 4
1095
1096 // These are not supported yet
1097 //#define _MM_HINT_ET0_ALT 7
1098 //#define _MM_HINT_ET1_ALT 6
1099 //#define _MM_HINT_ET2_ALT 5
1100
1101 #define _MM_HINT_MS_TO_ALT(sel) \
1102 (((sel) == _MM_HINT_NTA) ? _MM_HINT_NTA_ALT : \
1103 ((sel) == _MM_HINT_T0) ? _MM_HINT_T0_ALT : \
1104 ((sel) == _MM_HINT_T1) ? _MM_HINT_T1_ALT : \
1105 ((sel) == _MM_HINT_T2) ? _MM_HINT_T2_ALT : \
1106 ((sel) == _MM_HINT_ENTA) ? _MM_HINT_ENTA_ALT : 0)
1107
1108 #ifdef _MSC_VER1
1109
1110 /* On clang-cl we have an intrinsic, but the constants are different */
1111 #pragma intrinsic(_mm_prefetch)
1112 #define _mm_prefetch(p, sel) _mm_prefetch(p, _MM_HINT_MS_TO_ALT(sel))
1113
1114 #else /* _MSC_VER */
1115
1116 #define _mm_prefetch(p, sel) \
1117 __builtin_prefetch((const void *)(p), (_MM_HINT_MS_TO_ALT(sel) >> 2) & 1, _MM_HINT_MS_TO_ALT(sel) & 0x3)
1118
1119 #endif /* _MSC_VER */
1120
_mm_stream_pi(__m64 * __p,__m64 __a)1121 __INTRIN_INLINE_SSE void _mm_stream_pi(__m64 *__p, __m64 __a)
1122 {
1123 #ifdef __clang__
1124 __builtin_ia32_movntq((__v1di*)__p, __a);
1125 #else
1126 __builtin_ia32_movntq((long long unsigned int *)__p, (long long unsigned int)__a);
1127 #endif
1128 }
1129
_mm_stream_ps(float * __p,__m128 __a)1130 __INTRIN_INLINE_SSE void _mm_stream_ps(float *__p, __m128 __a)
1131 {
1132 #ifdef __clang__
1133 __builtin_nontemporal_store((__v4sf)__a, (__v4sf*)__p);
1134 #else
1135 __builtin_ia32_movntps(__p, (__v4sf)__a);
1136 #endif
1137 }
1138
1139 #if !HAS_BUILTIN(_mm_sfence) && !defined(_MSC_VER)
_mm_sfence(void)1140 __INTRIN_INLINE_SSE void _mm_sfence(void)
1141 {
1142 __builtin_ia32_sfence();
1143 }
1144 #endif
1145
1146 #ifdef __clang__
1147 #define _m_pextrw(a, n) \
1148 ((int)__builtin_ia32_vec_ext_v4hi((__v4hi)a, (int)n))
1149
1150 #define _m_pinsrw(a, d, n) \
1151 ((__m64)__builtin_ia32_vec_set_v4hi((__v4hi)a, (int)d, (int)n))
1152 #else
1153 // _m_pextrw
_mm_extract_pi16(__m64 const __a,int const __n)1154 __INTRIN_INLINE_SSE int _mm_extract_pi16(__m64 const __a, int const __n)
1155 {
1156 return (unsigned short)__builtin_ia32_vec_ext_v4hi((__v4hi)__a, __n);
1157 }
1158
1159 // _m_pinsrw
_mm_insert_pi16(__m64 const __a,int const __d,int const __n)1160 __INTRIN_INLINE_SSE __m64 _mm_insert_pi16 (__m64 const __a, int const __d, int const __n)
1161 {
1162 return (__m64)__builtin_ia32_vec_set_v4hi ((__v4hi)__a, __d, __n);
1163 }
1164
1165 #endif
1166
1167 // _m_pmaxsw
_mm_max_pi16(__m64 __a,__m64 __b)1168 __INTRIN_INLINE_SSE __m64 _mm_max_pi16(__m64 __a, __m64 __b)
1169 {
1170 return (__m64)__builtin_ia32_pmaxsw((__v4hi)__a, (__v4hi)__b);
1171 }
1172
1173 // _m_pmaxub
_mm_max_pu8(__m64 __a,__m64 __b)1174 __INTRIN_INLINE_SSE __m64 _mm_max_pu8(__m64 __a, __m64 __b)
1175 {
1176 return (__m64)__builtin_ia32_pmaxub((__v8qi)__a, (__v8qi)__b);
1177 }
1178
1179 // _m_pminsw
_mm_min_pi16(__m64 __a,__m64 __b)1180 __INTRIN_INLINE_SSE __m64 _mm_min_pi16(__m64 __a, __m64 __b)
1181 {
1182 return (__m64)__builtin_ia32_pminsw((__v4hi)__a, (__v4hi)__b);
1183 }
1184
1185 // _m_pminub
_mm_min_pu8(__m64 __a,__m64 __b)1186 __INTRIN_INLINE_SSE __m64 _mm_min_pu8(__m64 __a, __m64 __b)
1187 {
1188 return (__m64)__builtin_ia32_pminub((__v8qi)__a, (__v8qi)__b);
1189 }
1190
1191 // _m_pmovmskb
_mm_movemask_pi8(__m64 __a)1192 __INTRIN_INLINE_SSE int _mm_movemask_pi8(__m64 __a)
1193 {
1194 return __builtin_ia32_pmovmskb((__v8qi)__a);
1195 }
1196
1197 // _m_pmulhuw
_mm_mulhi_pu16(__m64 __a,__m64 __b)1198 __INTRIN_INLINE_SSE __m64 _mm_mulhi_pu16(__m64 __a, __m64 __b)
1199 {
1200 return (__m64)__builtin_ia32_pmulhuw((__v4hi)__a, (__v4hi)__b);
1201 }
1202
1203 #ifdef __clang__
1204 #define _m_pshufw(a, n) \
1205 ((__m64)__builtin_ia32_pshufw((__v4hi)(__m64)(a), (n)))
1206 #else
1207 // _m_pshufw
_mm_shuffle_pi16(__m64 __a,int const __n)1208 __INTRIN_INLINE_MMX __m64 _mm_shuffle_pi16 (__m64 __a, int const __n)
1209 {
1210 return (__m64) __builtin_ia32_pshufw ((__v4hi)__a, __n);
1211 }
1212 #endif
1213
1214 // _m_maskmovq
_mm_maskmove_si64(__m64 __d,__m64 __n,char * __p)1215 __INTRIN_INLINE_SSE void _mm_maskmove_si64(__m64 __d, __m64 __n, char *__p)
1216 {
1217 __builtin_ia32_maskmovq((__v8qi)__d, (__v8qi)__n, __p);
1218 }
1219
1220 // _m_pavgb
_mm_avg_pu8(__m64 __a,__m64 __b)1221 __INTRIN_INLINE_SSE __m64 _mm_avg_pu8(__m64 __a, __m64 __b)
1222 {
1223 return (__m64)__builtin_ia32_pavgb((__v8qi)__a, (__v8qi)__b);
1224 }
1225
1226 // _m_pavgw
_mm_avg_pu16(__m64 __a,__m64 __b)1227 __INTRIN_INLINE_SSE __m64 _mm_avg_pu16(__m64 __a, __m64 __b)
1228 {
1229 return (__m64)__builtin_ia32_pavgw((__v4hi)__a, (__v4hi)__b);
1230 }
1231
1232 // _m_psadbw
_mm_sad_pu8(__m64 __a,__m64 __b)1233 __INTRIN_INLINE_SSE __m64 _mm_sad_pu8(__m64 __a, __m64 __b)
1234 {
1235 return (__m64)__builtin_ia32_psadbw((__v8qi)__a, (__v8qi)__b);
1236 }
1237
1238 #endif // __GNUC__
1239
1240 #ifdef __cplusplus
1241 }
1242 #endif // __cplusplus
1243
1244 #endif /* _INCLUDED_MM2 */
1245