1 | /**
|
---|
2 | * This file has no copyright assigned and is placed in the Public Domain.
|
---|
3 | * This file is part of the mingw-w64 runtime package.
|
---|
4 | * No warranty is given; refer to the file DISCLAIMER.PD within this package.
|
---|
5 | */
|
---|
6 | #ifndef _FVEC_H_INCLUDED
|
---|
7 | #define _FVEC_H_INCLUDED
|
---|
8 |
|
---|
9 | #ifndef RC_INVOKED
|
---|
10 | #ifndef __cplusplus
|
---|
11 | #error ERROR: This file is only supported in C++ compilations!
|
---|
12 | #endif
|
---|
13 |
|
---|
14 | #include <intrin.h>
|
---|
15 | #include <assert.h>
|
---|
16 | #include <crtdefs.h>
|
---|
17 |
|
---|
18 | #if defined(_ENABLE_VEC_DEBUG)
|
---|
19 | #include <iostream>
|
---|
20 | #endif
|
---|
21 |
|
---|
22 | #pragma pack(push,_CRT_PACKING)
|
---|
23 |
|
---|
24 | #ifdef __SSE__
|
---|
25 |
|
---|
26 | #pragma pack(push,16)
|
---|
27 |
|
---|
28 | #define EXPLICIT explicit
|
---|
29 |
|
---|
30 | class F32vec4 {
|
---|
31 | protected:
|
---|
32 | __m128 vec;
|
---|
33 | public:
|
---|
34 | F32vec4() {}
|
---|
35 | F32vec4(__m128 m) { vec = m;}
|
---|
36 | F32vec4(float f3,float f2,float f1,float f0) { vec= _mm_set_ps(f3,f2,f1,f0); }
|
---|
37 | EXPLICIT F32vec4(float f) { vec = _mm_set_ps1(f); }
|
---|
38 | EXPLICIT F32vec4(double d) { vec = _mm_set_ps1((float) d); }
|
---|
39 | F32vec4& operator =(float f) { vec = _mm_set_ps1(f); return *this; }
|
---|
40 | F32vec4& operator =(double d) { vec = _mm_set_ps1((float) d); return *this; }
|
---|
41 | operator __m128() const { return vec; }
|
---|
42 | friend F32vec4 operator &(const F32vec4 &a,const F32vec4 &b) { return _mm_and_ps(a,b); }
|
---|
43 | friend F32vec4 operator |(const F32vec4 &a,const F32vec4 &b) { return _mm_or_ps(a,b); }
|
---|
44 | friend F32vec4 operator ^(const F32vec4 &a,const F32vec4 &b) { return _mm_xor_ps(a,b); }
|
---|
45 | friend F32vec4 operator +(const F32vec4 &a,const F32vec4 &b) { return _mm_add_ps(a,b); }
|
---|
46 | friend F32vec4 operator -(const F32vec4 &a,const F32vec4 &b) { return _mm_sub_ps(a,b); }
|
---|
47 | friend F32vec4 operator *(const F32vec4 &a,const F32vec4 &b) { return _mm_mul_ps(a,b); }
|
---|
48 | friend F32vec4 operator /(const F32vec4 &a,const F32vec4 &b) { return _mm_div_ps(a,b); }
|
---|
49 | F32vec4& operator =(const F32vec4 &a) { vec = a.vec; return *this; }
|
---|
50 | F32vec4& operator =(const __m128 &avec) { vec = avec; return *this; }
|
---|
51 | F32vec4& operator +=(F32vec4 &a) { return *this = _mm_add_ps(vec,a); }
|
---|
52 | F32vec4& operator -=(F32vec4 &a) { return *this = _mm_sub_ps(vec,a); }
|
---|
53 | F32vec4& operator *=(F32vec4 &a) { return *this = _mm_mul_ps(vec,a); }
|
---|
54 | F32vec4& operator /=(F32vec4 &a) { return *this = _mm_div_ps(vec,a); }
|
---|
55 | F32vec4& operator &=(F32vec4 &a) { return *this = _mm_and_ps(vec,a); }
|
---|
56 | F32vec4& operator |=(F32vec4 &a) { return *this = _mm_or_ps(vec,a); }
|
---|
57 | F32vec4& operator ^=(F32vec4 &a) { return *this = _mm_xor_ps(vec,a); }
|
---|
58 | friend float add_horizontal(F32vec4 &a) {
|
---|
59 | F32vec4 ftemp = _mm_add_ss(a,_mm_add_ss(_mm_shuffle_ps(a,a,1),_mm_add_ss(_mm_shuffle_ps(a,a,2),_mm_shuffle_ps(a,a,3))));
|
---|
60 | return ftemp[0];
|
---|
61 | }
|
---|
62 | friend F32vec4 sqrt(const F32vec4 &a) { return _mm_sqrt_ps(a); }
|
---|
63 | friend F32vec4 rcp(const F32vec4 &a) { return _mm_rcp_ps(a); }
|
---|
64 | friend F32vec4 rsqrt(const F32vec4 &a) { return _mm_rsqrt_ps(a); }
|
---|
65 | friend F32vec4 rcp_nr(const F32vec4 &a) {
|
---|
66 | F32vec4 Ra0 = _mm_rcp_ps(a);
|
---|
67 | return _mm_sub_ps(_mm_add_ps(Ra0,Ra0),_mm_mul_ps(_mm_mul_ps(Ra0,a),Ra0));
|
---|
68 | }
|
---|
69 | friend F32vec4 rsqrt_nr(const F32vec4 &a) {
|
---|
70 | static const F32vec4 fvecf0pt5(0.5f);
|
---|
71 | static const F32vec4 fvecf3pt0(3.0f);
|
---|
72 | F32vec4 Ra0 = _mm_rsqrt_ps(a);
|
---|
73 | return (fvecf0pt5 *Ra0) *(fvecf3pt0 - (a *Ra0) *Ra0);
|
---|
74 |
|
---|
75 | }
|
---|
76 | #define Fvec32s4_COMP(op) friend F32vec4 cmp##op (const F32vec4 &a,const F32vec4 &b) { return _mm_cmp##op##_ps(a,b); }
|
---|
77 | Fvec32s4_COMP(eq)
|
---|
78 | Fvec32s4_COMP(lt)
|
---|
79 | Fvec32s4_COMP(le)
|
---|
80 | Fvec32s4_COMP(gt)
|
---|
81 | Fvec32s4_COMP(ge)
|
---|
82 | Fvec32s4_COMP(neq)
|
---|
83 | Fvec32s4_COMP(nlt)
|
---|
84 | Fvec32s4_COMP(nle)
|
---|
85 | Fvec32s4_COMP(ngt)
|
---|
86 | Fvec32s4_COMP(nge)
|
---|
87 | #undef Fvec32s4_COMP
|
---|
88 |
|
---|
89 | friend F32vec4 simd_min(const F32vec4 &a,const F32vec4 &b) { return _mm_min_ps(a,b); }
|
---|
90 | friend F32vec4 simd_max(const F32vec4 &a,const F32vec4 &b) { return _mm_max_ps(a,b); }
|
---|
91 |
|
---|
92 | #if defined(_ENABLE_VEC_DEBUG)
|
---|
93 | friend std::ostream & operator<<(std::ostream & os,const F32vec4 &a) {
|
---|
94 | float *fp = (float*)&a;
|
---|
95 | os << "[3]:" << *(fp+3)
|
---|
96 | << " [2]:" << *(fp+2)
|
---|
97 | << " [1]:" << *(fp+1)
|
---|
98 | << " [0]:" << *fp;
|
---|
99 | return os;
|
---|
100 | }
|
---|
101 | #endif
|
---|
102 | const float& operator[](int i) const {
|
---|
103 | assert((0 <= i) && (i <= 3));
|
---|
104 | float *fp = (float*)&vec;
|
---|
105 | return *(fp+i);
|
---|
106 | }
|
---|
107 | float& operator[](int i) {
|
---|
108 | assert((0 <= i) && (i <= 3));
|
---|
109 | float *fp = (float*)&vec;
|
---|
110 | return *(fp+i);
|
---|
111 | }
|
---|
112 | };
|
---|
113 |
|
---|
114 | inline F32vec4 unpack_low(const F32vec4 &a,const F32vec4 &b) { return _mm_unpacklo_ps(a,b); }
|
---|
115 | inline F32vec4 unpack_high(const F32vec4 &a,const F32vec4 &b) { return _mm_unpackhi_ps(a,b); }
|
---|
116 | inline int move_mask(const F32vec4 &a) { return _mm_movemask_ps(a); }
|
---|
117 | inline void loadu(F32vec4 &a,float *p) { a = _mm_loadu_ps(p); }
|
---|
118 | inline void storeu(float *p,const F32vec4 &a) { _mm_storeu_ps(p,a); }
|
---|
119 | inline void store_nta(float *p,F32vec4 &a) { _mm_stream_ps(p,a); }
|
---|
120 |
|
---|
121 | #define Fvec32s4_SELECT(op) inline F32vec4 select_##op (const F32vec4 &a,const F32vec4 &b,const F32vec4 &c,const F32vec4 &d) { F32vec4 mask = _mm_cmp##op##_ps(a,b); return((mask & c) | F32vec4((_mm_andnot_ps(mask,d)))); }
|
---|
122 | Fvec32s4_SELECT(eq)
|
---|
123 | Fvec32s4_SELECT(lt)
|
---|
124 | Fvec32s4_SELECT(le)
|
---|
125 | Fvec32s4_SELECT(gt)
|
---|
126 | Fvec32s4_SELECT(ge)
|
---|
127 | Fvec32s4_SELECT(neq)
|
---|
128 | Fvec32s4_SELECT(nlt)
|
---|
129 | Fvec32s4_SELECT(nle)
|
---|
130 | Fvec32s4_SELECT(ngt)
|
---|
131 | Fvec32s4_SELECT(nge)
|
---|
132 | #undef Fvec32s4_SELECT
|
---|
133 |
|
---|
134 | #if 0 /* Commented until required types are defined */
|
---|
135 | inline Is16vec4 simd_max(const Is16vec4 &a,const Is16vec4 &b) { return _m_pmaxsw(a,b); }
|
---|
136 | inline Is16vec4 simd_min(const Is16vec4 &a,const Is16vec4 &b) { return _m_pminsw(a,b); }
|
---|
137 | inline Iu8vec8 simd_max(const Iu8vec8 &a,const Iu8vec8 &b) { return _m_pmaxub(a,b); }
|
---|
138 | inline Iu8vec8 simd_min(const Iu8vec8 &a,const Iu8vec8 &b) { return _m_pminub(a,b); }
|
---|
139 | inline Iu16vec4 simd_avg(const Iu16vec4 &a,const Iu16vec4 &b) { return _m_pavgw(a,b); }
|
---|
140 | inline Iu8vec8 simd_avg(const Iu8vec8 &a,const Iu8vec8 &b) { return _m_pavgb(a,b); }
|
---|
141 | inline int move_mask(const I8vec8 &a) { return _m_pmovmskb(a); }
|
---|
142 | inline Iu16vec4 mul_high(const Iu16vec4 &a,const Iu16vec4 &b) { return _m_pmulhuw(a,b); }
|
---|
143 | inline void mask_move(const I8vec8 &a,const I8vec8 &b,char *addr) { _m_maskmovq(a,b,addr); }
|
---|
144 | inline void store_nta(__m64 *p,M64 &a) { _mm_stream_pi(p,a); }
|
---|
145 | inline int F32vec4ToInt(const F32vec4 &a) { return _mm_cvtt_ss2si(a); }
|
---|
146 | inline Is32vec2 F32vec4ToIs32vec2 (const F32vec4 &a) {
|
---|
147 | __m64 result;
|
---|
148 | result = _mm_cvtt_ps2pi(a);
|
---|
149 | return Is32vec2(result);
|
---|
150 | }
|
---|
151 | #endif
|
---|
152 |
|
---|
153 | inline F32vec4 IntToF32vec4(const F32vec4 &a,int i) {
|
---|
154 | __m128 result;
|
---|
155 | result = _mm_cvt_si2ss(a,i);
|
---|
156 | return F32vec4(result);
|
---|
157 | }
|
---|
158 |
|
---|
159 | #if 0 /* Commented until required types are defined */
|
---|
160 | inline F32vec4 Is32vec2ToF32vec4(const F32vec4 &a,const Is32vec2 &b) {
|
---|
161 | __m128 result;
|
---|
162 | result = _mm_cvt_pi2ps(a,b);
|
---|
163 | return F32vec4(result);
|
---|
164 | }
|
---|
165 | #endif
|
---|
166 |
|
---|
167 | class F32vec1 {
|
---|
168 | protected:
|
---|
169 | __m128 vec;
|
---|
170 | public:
|
---|
171 | F32vec1() {}
|
---|
172 | F32vec1(int i) { vec = _mm_cvt_si2ss(vec,i);};
|
---|
173 | EXPLICIT F32vec1(float f) { vec = _mm_set_ss(f); }
|
---|
174 | EXPLICIT F32vec1(double d) { vec = _mm_set_ss((float) d); }
|
---|
175 | F32vec1(__m128 m) { vec = m; }
|
---|
176 | operator __m128() const { return vec; }
|
---|
177 | friend F32vec1 operator &(const F32vec1 &a,const F32vec1 &b) { return _mm_and_ps(a,b); }
|
---|
178 | friend F32vec1 operator |(const F32vec1 &a,const F32vec1 &b) { return _mm_or_ps(a,b); }
|
---|
179 | friend F32vec1 operator ^(const F32vec1 &a,const F32vec1 &b) { return _mm_xor_ps(a,b); }
|
---|
180 | friend F32vec1 operator +(const F32vec1 &a,const F32vec1 &b) { return _mm_add_ss(a,b); }
|
---|
181 | friend F32vec1 operator -(const F32vec1 &a,const F32vec1 &b) { return _mm_sub_ss(a,b); }
|
---|
182 | friend F32vec1 operator *(const F32vec1 &a,const F32vec1 &b) { return _mm_mul_ss(a,b); }
|
---|
183 | friend F32vec1 operator /(const F32vec1 &a,const F32vec1 &b) { return _mm_div_ss(a,b); }
|
---|
184 | F32vec1& operator +=(F32vec1 &a) { return *this = _mm_add_ss(vec,a); }
|
---|
185 | F32vec1& operator -=(F32vec1 &a) { return *this = _mm_sub_ss(vec,a); }
|
---|
186 | F32vec1& operator *=(F32vec1 &a) { return *this = _mm_mul_ss(vec,a); }
|
---|
187 | F32vec1& operator /=(F32vec1 &a) { return *this = _mm_div_ss(vec,a); }
|
---|
188 | F32vec1& operator &=(F32vec1 &a) { return *this = _mm_and_ps(vec,a); }
|
---|
189 | F32vec1& operator |=(F32vec1 &a) { return *this = _mm_or_ps(vec,a); }
|
---|
190 | F32vec1& operator ^=(F32vec1 &a) { return *this = _mm_xor_ps(vec,a); }
|
---|
191 | friend F32vec1 sqrt(const F32vec1 &a) { return _mm_sqrt_ss(a); }
|
---|
192 | friend F32vec1 rcp(const F32vec1 &a) { return _mm_rcp_ss(a); }
|
---|
193 | friend F32vec1 rsqrt(const F32vec1 &a) { return _mm_rsqrt_ss(a); }
|
---|
194 | friend F32vec1 rcp_nr(const F32vec1 &a) {
|
---|
195 | F32vec1 Ra0 = _mm_rcp_ss(a);
|
---|
196 | return _mm_sub_ss(_mm_add_ss(Ra0,Ra0),_mm_mul_ss(_mm_mul_ss(Ra0,a),Ra0));
|
---|
197 | }
|
---|
198 | friend F32vec1 rsqrt_nr(const F32vec1 &a) {
|
---|
199 | static const F32vec1 fvecf0pt5(0.5f);
|
---|
200 | static const F32vec1 fvecf3pt0(3.0f);
|
---|
201 | F32vec1 Ra0 = _mm_rsqrt_ss(a);
|
---|
202 | return (fvecf0pt5 *Ra0) *(fvecf3pt0 - (a *Ra0) *Ra0);
|
---|
203 | }
|
---|
204 | #define Fvec32s1_COMP(op) friend F32vec1 cmp##op (const F32vec1 &a,const F32vec1 &b) { return _mm_cmp##op##_ss(a,b); }
|
---|
205 | Fvec32s1_COMP(eq)
|
---|
206 | Fvec32s1_COMP(lt)
|
---|
207 | Fvec32s1_COMP(le)
|
---|
208 | Fvec32s1_COMP(gt)
|
---|
209 | Fvec32s1_COMP(ge)
|
---|
210 | Fvec32s1_COMP(neq)
|
---|
211 | Fvec32s1_COMP(nlt)
|
---|
212 | Fvec32s1_COMP(nle)
|
---|
213 | Fvec32s1_COMP(ngt)
|
---|
214 | Fvec32s1_COMP(nge)
|
---|
215 | #undef Fvec32s1_COMP
|
---|
216 |
|
---|
217 | friend F32vec1 simd_min(const F32vec1 &a,const F32vec1 &b) { return _mm_min_ss(a,b); }
|
---|
218 | friend F32vec1 simd_max(const F32vec1 &a,const F32vec1 &b) { return _mm_max_ss(a,b); }
|
---|
219 |
|
---|
220 | #if defined(_ENABLE_VEC_DEBUG)
|
---|
221 | friend std::ostream & operator<<(std::ostream & os,const F32vec1 &a) {
|
---|
222 | float *fp = (float*)&a;
|
---|
223 | os << "float:" << *fp;
|
---|
224 | return os;
|
---|
225 | }
|
---|
226 | #endif
|
---|
227 | };
|
---|
228 |
|
---|
229 | #define Fvec32s1_SELECT(op) inline F32vec1 select_##op (const F32vec1 &a,const F32vec1 &b,const F32vec1 &c,const F32vec1 &d) { F32vec1 mask = _mm_cmp##op##_ss(a,b); return((mask & c) | F32vec1((_mm_andnot_ps(mask,d)))); }
|
---|
230 | Fvec32s1_SELECT(eq)
|
---|
231 | Fvec32s1_SELECT(lt)
|
---|
232 | Fvec32s1_SELECT(le)
|
---|
233 | Fvec32s1_SELECT(gt)
|
---|
234 | Fvec32s1_SELECT(ge)
|
---|
235 | Fvec32s1_SELECT(neq)
|
---|
236 | Fvec32s1_SELECT(nlt)
|
---|
237 | Fvec32s1_SELECT(nle)
|
---|
238 | Fvec32s1_SELECT(ngt)
|
---|
239 | Fvec32s1_SELECT(nge)
|
---|
240 | #undef Fvec32s1_SELECT
|
---|
241 |
|
---|
242 | inline int F32vec1ToInt(const F32vec1 &a)
|
---|
243 | {
|
---|
244 | return _mm_cvtt_ss2si(a);
|
---|
245 | }
|
---|
246 |
|
---|
247 | #pragma pack(pop)
|
---|
248 |
|
---|
249 | #endif /* #ifdef __SSE__ */
|
---|
250 | #pragma pack(pop)
|
---|
251 |
|
---|
252 | #include <ivec.h>
|
---|
253 |
|
---|
254 | #endif
|
---|
255 | #endif
|
---|