| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671 |
- // ======================================================================== //
- // Copyright 2009-2017 Intel Corporation //
- // //
- // Licensed under the Apache License, Version 2.0 (the "License"); //
- // you may not use this file except in compliance with the License. //
- // You may obtain a copy of the License at //
- // //
- // http://www.apache.org/licenses/LICENSE-2.0 //
- // //
- // Unless required by applicable law or agreed to in writing, software //
- // distributed under the License is distributed on an "AS IS" BASIS, //
- // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. //
- // See the License for the specific language governing permissions and //
- // limitations under the License. //
- // ======================================================================== //
- #pragma once
- namespace embree
- {
- /* 16-wide AVX-512 float type */
- template<>
- struct vfloat<16>
- {
- typedef vboolf16 Bool;
- typedef vint16 Int;
- typedef vfloat16 Float;
- enum { size = 16 }; // number of SIMD elements
- union { // data
- __m512 v;
- float f[16];
- int i[16];
- };
- ////////////////////////////////////////////////////////////////////////////////
- /// Constructors, Assignment & Cast Operators
- ////////////////////////////////////////////////////////////////////////////////
-
- __forceinline vfloat() {}
- __forceinline vfloat(const vfloat16& t) { v = t; }
- __forceinline vfloat16& operator=(const vfloat16& f) { v = f.v; return *this; }
- __forceinline vfloat(const __m512& t) { v = t; }
- __forceinline operator __m512 () const { return v; }
- __forceinline operator __m256 () const { return _mm512_castps512_ps256(v); }
-
- __forceinline vfloat(const float& f) {
- v = _mm512_set1_ps(f);
- }
- __forceinline vfloat(const float& a, const float& b, const float& c, const float& d) {
- v = _mm512_set4_ps(a,b,c,d);
- }
- __forceinline vfloat(const vfloat4 &i) {
- v = _mm512_broadcast_f32x4(i);
- }
- __forceinline vfloat(const vfloat4 &a, const vfloat4 &b, const vfloat4 &c, const vfloat4 &d) {
- v = _mm512_broadcast_f32x4(a);
- v = _mm512_insertf32x4(v, b, 1);
- v = _mm512_insertf32x4(v, c, 2);
- v = _mm512_insertf32x4(v, d, 3);
- }
- __forceinline vfloat(const vboolf16& mask, const vfloat4 &a, const vfloat4 &b) {
- v = _mm512_broadcast_f32x4(a);
- v = _mm512_mask_broadcast_f32x4(v,mask,b);
- }
- __forceinline vfloat(const vfloat8 &i) {
- v = _mm512_castpd_ps(_mm512_broadcast_f64x4(_mm256_castps_pd(i)));
- }
- __forceinline vfloat(const vfloat8 &a, const vfloat8 &b) { // FIXME: optimize
- const vfloat aa = _mm512_castpd_ps(_mm512_broadcast_f64x4(_mm256_castps_pd(a)));
- const vfloat bb = _mm512_castpd_ps(_mm512_broadcast_f64x4(_mm256_castps_pd(b)));
- v = _mm512_mask_blend_ps(0xff, bb, aa);
- }
- /* WARNING: due to f64x4 the mask is considered as an 8bit mask */
- __forceinline vfloat(const vboolf16& mask, const vfloat8 &a, const vfloat8 &b) {
- __m512d aa = _mm512_broadcast_f64x4(_mm256_castps_pd(a));
- aa = _mm512_mask_broadcast_f64x4(aa,mask,_mm256_castps_pd(b));
- v = _mm512_castpd_ps(aa);
- }
-
- __forceinline explicit vfloat(const __m512i& a) {
- v = _mm512_cvtepi32_ps(a);
- }
- ////////////////////////////////////////////////////////////////////////////////
- /// Loads and Stores
- ////////////////////////////////////////////////////////////////////////////////
- static __forceinline vfloat16 load (const void* const ptr) { return _mm512_load_ps((float*)ptr); }
- static __forceinline vfloat16 loadu(const void* const ptr) { return _mm512_loadu_ps((float*)ptr); }
- static __forceinline vfloat16 load (const vboolf16& mask, const void* const ptr) { return _mm512_mask_load_ps (_mm512_setzero_ps(),mask,(float*)ptr); }
- static __forceinline vfloat16 loadu(const vboolf16& mask, const void* const ptr) { return _mm512_mask_loadu_ps(_mm512_setzero_ps(),mask,(float*)ptr); }
- static __forceinline void store (void* const ptr, const vfloat16& v) { _mm512_store_ps ((float*)ptr,v); }
- static __forceinline void storeu(void* const ptr, const vfloat16& v) { _mm512_storeu_ps((float*)ptr,v); }
- static __forceinline void store (const vboolf16& mask, void* ptr, const vfloat16& v) { _mm512_mask_store_ps ((float*)ptr,mask,v); }
- static __forceinline void storeu(const vboolf16& mask, void* ptr, const vfloat16& v) { _mm512_mask_storeu_ps((float*)ptr,mask,v); }
- static __forceinline void store_nt(void *__restrict__ ptr, const vfloat16& a) {
- _mm512_stream_ps((float*)ptr,a);
- }
- static __forceinline vfloat16 broadcast(const float *const f) {
- return _mm512_set1_ps(*f);
- }
- static __forceinline vfloat16 compact(const vboolf16& mask, vfloat16 &v) {
- return _mm512_mask_compress_ps(v,mask,v);
- }
- static __forceinline vfloat16 compact(const vboolf16& mask, const vfloat16 &a, vfloat16 &b) {
- return _mm512_mask_compress_ps(a,mask,b);
- }
- static __forceinline vfloat16 loadu_compact(const vboolf16& mask, const void *const ptr) {
- return _mm512_mask_expandloadu_ps(vfloat16::undefined(),mask,(float*)ptr);
- }
- static __forceinline void storeu_compact(const vboolf16& mask, float *addr, const vfloat16 reg) {
- _mm512_mask_compressstoreu_ps(addr,mask,reg);
- }
-
- static __forceinline void storeu_compact_single(const vboolf16& mask, float * addr, const vfloat16 ®) {
- //_mm512_mask_compressstoreu_ps(addr,mask,reg);
- *addr = mm512_cvtss_f32(_mm512_mask_compress_ps(reg,mask,reg));
- }
- ////////////////////////////////////////////////////////////////////////////////
- /// Constants
- ////////////////////////////////////////////////////////////////////////////////
-
- __forceinline vfloat( ZeroTy ) : v(_mm512_setzero_ps()) {}
- __forceinline vfloat( OneTy ) : v(_mm512_set1_ps(1.0f)) {}
- __forceinline vfloat( PosInfTy ) : v(_mm512_set1_ps(pos_inf)) {}
- __forceinline vfloat( NegInfTy ) : v(_mm512_set1_ps(neg_inf)) {}
- __forceinline vfloat( StepTy ) : v(_mm512_set_ps(15,14,13,12,11,10,9,8,7,6,5,4,3,2,1,0)) {}
- __forceinline vfloat( NaNTy ) : v(_mm512_set1_ps(nan)) {}
- __forceinline static vfloat16 undefined() { return _mm512_undefined(); }
- __forceinline static vfloat16 zero() { return _mm512_setzero_ps(); }
- __forceinline static vfloat16 one () { return _mm512_set1_ps(1.0f); }
- __forceinline static vfloat16 ulp () { return _mm512_set1_ps(embree::ulp); }
- __forceinline static vfloat16 inf () { return _mm512_set1_ps((float)pos_inf); }
- __forceinline static vfloat16 minus_inf () { return _mm512_set1_ps((float)neg_inf); }
- ////////////////////////////////////////////////////////////////////////////////
- /// Array Access
- ////////////////////////////////////////////////////////////////////////////////
-
- __forceinline float& operator [](const size_t index) { assert(index < 16); return f[index]; }
- __forceinline const float& operator [](const size_t index) const { assert(index < 16); return f[index]; }
- };
- ////////////////////////////////////////////////////////////////////////////////
- /// Unary Operators
- ////////////////////////////////////////////////////////////////////////////////
- __forceinline const vfloat16 asFloat ( const __m512i& a ) { return _mm512_castsi512_ps(a); }
- __forceinline const vfloat16 operator +( const vfloat16& a ) { return a; }
- __forceinline const vfloat16 operator -( const vfloat16& a ) { return _mm512_mul_ps(a,vfloat16(-1)); }
- __forceinline const vfloat16 abs ( const vfloat16& a ) { return _mm512_abs_ps(a); }
-
- __forceinline const vfloat16 signmsk ( const vfloat16& a ) { return _mm512_castsi512_ps(_mm512_and_epi32(_mm512_castps_si512(a),_mm512_set1_epi32(0x80000000))); }
- __forceinline const vfloat16 rcp(const vfloat16& a) {
- #if defined(__AVX512ER__)
- return _mm512_rcp28_ps(a);
- #else
- const vfloat16 r = _mm512_rcp14_ps(a.v);
- return _mm512_mul_ps(r, _mm512_fnmadd_ps(r, a, vfloat16(2.0f)));
- #endif
- }
- __forceinline const vfloat16 sqr ( const vfloat16& a ) { return _mm512_mul_ps(a,a); }
- __forceinline const vfloat16 sqrt ( const vfloat16& a ) { return _mm512_sqrt_ps(a); }
- __forceinline const vfloat16 rsqrt( const vfloat16& a )
- {
- #if defined(__AVX512VL__)
- const vfloat16 r = _mm512_rsqrt14_ps(a.v);
- return _mm512_fmadd_ps(_mm512_set1_ps(1.5f), r,
- _mm512_mul_ps(_mm512_mul_ps(_mm512_mul_ps(a, _mm512_set1_ps(-0.5f)), r), _mm512_mul_ps(r, r)));
- #else
- return _mm512_rsqrt28_ps(a.v);
- #endif
- }
- ////////////////////////////////////////////////////////////////////////////////
- /// Binary Operators
- ////////////////////////////////////////////////////////////////////////////////
- __forceinline const vfloat16 operator +( const vfloat16& a, const vfloat16& b ) { return _mm512_add_ps(a, b); }
- __forceinline const vfloat16 operator +( const vfloat16& a, const float& b ) { return a + vfloat16(b); }
- __forceinline const vfloat16 operator +( const float& a, const vfloat16& b ) { return vfloat16(a) + b; }
- __forceinline const vfloat16 operator -( const vfloat16& a, const vfloat16& b ) { return _mm512_sub_ps(a, b); }
- __forceinline const vfloat16 operator -( const vfloat16& a, const float& b ) { return a - vfloat16(b); }
- __forceinline const vfloat16 operator -( const float& a, const vfloat16& b ) { return vfloat16(a) - b; }
- __forceinline const vfloat16 operator *( const vfloat16& a, const vfloat16& b ) { return _mm512_mul_ps(a, b); }
- __forceinline const vfloat16 operator *( const vfloat16& a, const float& b ) { return a * vfloat16(b); }
- __forceinline const vfloat16 operator *( const float& a, const vfloat16& b ) { return vfloat16(a) * b; }
- __forceinline const vfloat16 operator /( const vfloat16& a, const vfloat16& b ) { return _mm512_div_ps(a,b); }
- __forceinline const vfloat16 operator /( const vfloat16& a, const float& b ) { return a/vfloat16(b); }
- __forceinline const vfloat16 operator /( const float& a, const vfloat16& b ) { return vfloat16(a)/b; }
-
- __forceinline const vfloat16 operator^(const vfloat16& a, const vfloat16& b) {
- return _mm512_castsi512_ps(_mm512_xor_epi32(_mm512_castps_si512(a),_mm512_castps_si512(b)));
- }
-
- __forceinline const vfloat16 min( const vfloat16& a, const vfloat16& b ) {
- return _mm512_min_ps(a,b);
- }
- __forceinline const vfloat16 min( const vfloat16& a, const float& b ) {
- return _mm512_min_ps(a,vfloat16(b));
- }
- __forceinline const vfloat16 min( const float& a, const vfloat16& b ) {
- return _mm512_min_ps(vfloat16(a),b);
- }
- __forceinline const vfloat16 max( const vfloat16& a, const vfloat16& b ) {
- return _mm512_max_ps(a,b);
- }
- __forceinline const vfloat16 max( const vfloat16& a, const float& b ) {
- return _mm512_max_ps(a,vfloat16(b));
- }
- __forceinline const vfloat16 max( const float& a, const vfloat16& b ) {
- return _mm512_max_ps(vfloat16(a),b);
- }
- __forceinline vfloat16 mask_add(const vboolf16& mask, const vfloat16& c, const vfloat16& a, const vfloat16& b) { return _mm512_mask_add_ps (c,mask,a,b); }
- __forceinline vfloat16 mask_min(const vboolf16& mask, const vfloat16& c, const vfloat16& a, const vfloat16& b) {
- return _mm512_mask_min_ps(c,mask,a,b);
- };
- __forceinline vfloat16 mask_max(const vboolf16& mask, const vfloat16& c, const vfloat16& a, const vfloat16& b) {
- return _mm512_mask_max_ps(c,mask,a,b);
- };
- __forceinline vfloat16 mini(const vfloat16& a, const vfloat16& b) {
- #if !defined(__AVX512ER__) // SKX
- const vint16 ai = _mm512_castps_si512(a);
- const vint16 bi = _mm512_castps_si512(b);
- const vint16 ci = _mm512_min_epi32(ai,bi);
- return _mm512_castsi512_ps(ci);
- #else // KNL
- return min(a,b);
- #endif
- }
- __forceinline vfloat16 maxi(const vfloat16& a, const vfloat16& b) {
- #if !defined(__AVX512ER__) // SKX
- const vint16 ai = _mm512_castps_si512(a);
- const vint16 bi = _mm512_castps_si512(b);
- const vint16 ci = _mm512_max_epi32(ai,bi);
- return _mm512_castsi512_ps(ci);
- #else // KNL
- return max(a,b);
- #endif
- }
- ////////////////////////////////////////////////////////////////////////////////
- /// Ternary Operators
- ////////////////////////////////////////////////////////////////////////////////
- __forceinline vfloat16 madd (const vfloat16& a, const vfloat16& b, const vfloat16& c) { return _mm512_fmadd_ps(a,b,c); }
- __forceinline vfloat16 msub (const vfloat16& a, const vfloat16& b, const vfloat16& c) { return _mm512_fmsub_ps(a,b,c); }
- __forceinline vfloat16 nmadd (const vfloat16& a, const vfloat16& b, const vfloat16& c) { return _mm512_fnmadd_ps(a,b,c); }
- __forceinline vfloat16 nmsub (const vfloat16& a, const vfloat16& b, const vfloat16& c) { return _mm512_fnmsub_ps(a,b,c); }
- __forceinline vfloat16 mask_msub (const vboolf16& mask,const vfloat16& a, const vfloat16& b, const vfloat16& c) { return _mm512_mask_fmsub_ps(a,mask,b,c); }
-
- __forceinline vfloat16 madd231 (const vfloat16& a, const vfloat16& b, const vfloat16& c) { return _mm512_fmadd_ps(c,b,a); }
- __forceinline vfloat16 msub213 (const vfloat16& a, const vfloat16& b, const vfloat16& c) { return _mm512_fmsub_ps(a,b,c); }
- __forceinline vfloat16 msub231 (const vfloat16& a, const vfloat16& b, const vfloat16& c) { return _mm512_fmsub_ps(c,b,a); }
- __forceinline vfloat16 msubr231(const vfloat16& a, const vfloat16& b, const vfloat16& c) { return _mm512_fnmadd_ps(c,b,a); }
- ////////////////////////////////////////////////////////////////////////////////
- /// Operators with rounding
- ////////////////////////////////////////////////////////////////////////////////
-
- __forceinline vfloat16 madd_round_down (const vfloat16& a, const vfloat16& b, const vfloat16& c) { return _mm512_fmadd_round_ps(a,b,c,_MM_FROUND_TO_NEG_INF | _MM_FROUND_NO_EXC); }
- __forceinline vfloat16 madd_round_up (const vfloat16& a, const vfloat16& b, const vfloat16& c) { return _mm512_fmadd_round_ps(a,b,c,_MM_FROUND_TO_POS_INF | _MM_FROUND_NO_EXC); }
- __forceinline vfloat16 mul_round_down (const vfloat16& a, const vfloat16& b) { return _mm512_mul_round_ps(a,b,_MM_FROUND_TO_NEG_INF | _MM_FROUND_NO_EXC); }
- __forceinline vfloat16 mul_round_up (const vfloat16& a, const vfloat16& b) { return _mm512_mul_round_ps(a,b,_MM_FROUND_TO_POS_INF | _MM_FROUND_NO_EXC); }
- __forceinline vfloat16 add_round_down (const vfloat16& a, const vfloat16& b) { return _mm512_add_round_ps(a,b,_MM_FROUND_TO_NEG_INF | _MM_FROUND_NO_EXC); }
- __forceinline vfloat16 add_round_up (const vfloat16& a, const vfloat16& b) { return _mm512_add_round_ps(a,b,_MM_FROUND_TO_POS_INF | _MM_FROUND_NO_EXC); }
- __forceinline vfloat16 sub_round_down (const vfloat16& a, const vfloat16& b) { return _mm512_sub_round_ps(a,b,_MM_FROUND_TO_NEG_INF | _MM_FROUND_NO_EXC); }
- __forceinline vfloat16 sub_round_up (const vfloat16& a, const vfloat16& b) { return _mm512_sub_round_ps(a,b,_MM_FROUND_TO_POS_INF | _MM_FROUND_NO_EXC); }
- __forceinline vfloat16 div_round_down (const vfloat16& a, const vfloat16& b) { return _mm512_div_round_ps(a,b,_MM_FROUND_TO_NEG_INF | _MM_FROUND_NO_EXC); }
- __forceinline vfloat16 div_round_up (const vfloat16& a, const vfloat16& b) { return _mm512_div_round_ps(a,b,_MM_FROUND_TO_POS_INF | _MM_FROUND_NO_EXC); }
- __forceinline vfloat16 mask_msub_round_down (const vboolf16& mask,const vfloat16& a, const vfloat16& b, const vfloat16& c) { return _mm512_mask_fmsub_round_ps(a,mask,b,c,_MM_FROUND_TO_NEG_INF | _MM_FROUND_NO_EXC); }
- __forceinline vfloat16 mask_msub_round_up (const vboolf16& mask,const vfloat16& a, const vfloat16& b, const vfloat16& c) { return _mm512_mask_fmsub_round_ps(a,mask,b,c,_MM_FROUND_TO_POS_INF | _MM_FROUND_NO_EXC); }
-
- __forceinline vfloat16 mask_mul_round_down (const vboolf16& mask,const vfloat16& a, const vfloat16& b, const vfloat16& c) { return _mm512_mask_mul_round_ps(a,mask,b,c,_MM_FROUND_TO_NEG_INF | _MM_FROUND_NO_EXC); }
- __forceinline vfloat16 mask_mul_round_up (const vboolf16& mask,const vfloat16& a, const vfloat16& b, const vfloat16& c) { return _mm512_mask_mul_round_ps(a,mask,b,c,_MM_FROUND_TO_POS_INF | _MM_FROUND_NO_EXC); }
- __forceinline vfloat16 mask_sub_round_down (const vboolf16& mask,const vfloat16& a, const vfloat16& b, const vfloat16& c) { return _mm512_mask_sub_round_ps(a,mask,b,c,_MM_FROUND_TO_NEG_INF | _MM_FROUND_NO_EXC); }
- __forceinline vfloat16 mask_sub_round_up (const vboolf16& mask,const vfloat16& a, const vfloat16& b, const vfloat16& c) { return _mm512_mask_sub_round_ps(a,mask,b,c,_MM_FROUND_TO_POS_INF | _MM_FROUND_NO_EXC); }
-
- ////////////////////////////////////////////////////////////////////////////////
- /// Assignment Operators
- ////////////////////////////////////////////////////////////////////////////////
- __forceinline vfloat16& operator +=( vfloat16& a, const vfloat16& b ) { return a = a + b; }
- __forceinline vfloat16& operator +=( vfloat16& a, const float& b ) { return a = a + b; }
-
- __forceinline vfloat16& operator -=( vfloat16& a, const vfloat16& b ) { return a = a - b; }
- __forceinline vfloat16& operator -=( vfloat16& a, const float& b ) { return a = a - b; }
-
- __forceinline vfloat16& operator *=( vfloat16& a, const vfloat16& b ) { return a = a * b; }
- __forceinline vfloat16& operator *=( vfloat16& a, const float& b ) { return a = a * b; }
- __forceinline vfloat16& operator /=( vfloat16& a, const vfloat16& b ) { return a = a / b; }
- __forceinline vfloat16& operator /=( vfloat16& a, const float& b ) { return a = a / b; }
- ////////////////////////////////////////////////////////////////////////////////
- /// Comparison Operators + Select
- ////////////////////////////////////////////////////////////////////////////////
- __forceinline const vboolf16 operator ==( const vfloat16& a, const vfloat16& b ) { return _mm512_cmp_ps_mask(a,b,_MM_CMPINT_EQ); }
- __forceinline const vboolf16 operator ==( const vfloat16& a, const float& b ) { return a == vfloat16(b); }
- __forceinline const vboolf16 operator ==( const float& a, const vfloat16& b ) { return vfloat16(a) == b; }
- __forceinline const vboolf16 operator !=( const vfloat16& a, const vfloat16& b ) { return _mm512_cmp_ps_mask(a,b,_MM_CMPINT_NE); }
- __forceinline const vboolf16 operator !=( const vfloat16& a, const float& b ) { return a != vfloat16(b); }
- __forceinline const vboolf16 operator !=( const float& a, const vfloat16& b ) { return vfloat16(a) != b; }
- __forceinline const vboolf16 operator < ( const vfloat16& a, const vfloat16& b ) { return _mm512_cmp_ps_mask(a,b,_MM_CMPINT_LT); }
- __forceinline const vboolf16 operator < ( const vfloat16& a, const float& b ) { return a < vfloat16(b); }
- __forceinline const vboolf16 operator < ( const float& a, const vfloat16& b ) { return vfloat16(a) < b; }
- __forceinline const vboolf16 operator >=( const vfloat16& a, const vfloat16& b ) { return _mm512_cmp_ps_mask(a,b,_MM_CMPINT_GE); }
- __forceinline const vboolf16 operator >=( const vfloat16& a, const float& b ) { return a >= vfloat16(b); }
- __forceinline const vboolf16 operator >=( const float& a, const vfloat16& b ) { return vfloat16(a) >= b; }
- __forceinline const vboolf16 operator > ( const vfloat16& a, const vfloat16& b ) { return _mm512_cmp_ps_mask(a,b,_MM_CMPINT_GT); }
- __forceinline const vboolf16 operator > ( const vfloat16& a, const float& b ) { return a > vfloat16(b); }
- __forceinline const vboolf16 operator > ( const float& a, const vfloat16& b ) { return vfloat16(a) > b; }
- __forceinline const vboolf16 operator <=( const vfloat16& a, const vfloat16& b ) { return _mm512_cmp_ps_mask(a,b,_MM_CMPINT_LE); }
- __forceinline const vboolf16 operator <=( const vfloat16& a, const float& b ) { return a <= vfloat16(b); }
- __forceinline const vboolf16 operator <=( const float& a, const vfloat16& b ) { return vfloat16(a) <= b; }
- __forceinline vboolf16 eq(const vfloat16& a, const vfloat16& b) { return _mm512_cmp_ps_mask(a,b,_MM_CMPINT_EQ); }
- __forceinline vboolf16 ne(const vfloat16& a, const vfloat16& b) { return _mm512_cmp_ps_mask(a,b,_MM_CMPINT_NE); }
- __forceinline vboolf16 lt(const vfloat16& a, const vfloat16& b) { return _mm512_cmp_ps_mask(a,b,_MM_CMPINT_LT); }
- __forceinline vboolf16 ge(const vfloat16& a, const vfloat16& b) { return _mm512_cmp_ps_mask(a,b,_MM_CMPINT_GE); }
- __forceinline vboolf16 gt(const vfloat16& a, const vfloat16& b) { return _mm512_cmp_ps_mask(a,b,_MM_CMPINT_GT); }
- __forceinline vboolf16 le(const vfloat16& a, const vfloat16& b) { return _mm512_cmp_ps_mask(a,b,_MM_CMPINT_LE); }
- __forceinline vboolf16 eq(const vboolf16& mask, const vfloat16& a, const vfloat16& b) { return _mm512_mask_cmp_ps_mask(mask,a,b,_MM_CMPINT_EQ); }
- __forceinline vboolf16 ne(const vboolf16& mask, const vfloat16& a, const vfloat16& b) { return _mm512_mask_cmp_ps_mask(mask,a,b,_MM_CMPINT_NE); }
- __forceinline vboolf16 lt(const vboolf16& mask, const vfloat16& a, const vfloat16& b) { return _mm512_mask_cmp_ps_mask(mask,a,b,_MM_CMPINT_LT); }
- __forceinline vboolf16 ge(const vboolf16& mask, const vfloat16& a, const vfloat16& b) { return _mm512_mask_cmp_ps_mask(mask,a,b,_MM_CMPINT_GE); }
- __forceinline vboolf16 gt(const vboolf16& mask, const vfloat16& a, const vfloat16& b) { return _mm512_mask_cmp_ps_mask(mask,a,b,_MM_CMPINT_GT); }
- __forceinline vboolf16 le(const vboolf16& mask, const vfloat16& a, const vfloat16& b) { return _mm512_mask_cmp_ps_mask(mask,a,b,_MM_CMPINT_LE); }
-
- __forceinline const vfloat16 select( const vboolf16& s, const vfloat16& t, const vfloat16& f ) {
- return _mm512_mask_blend_ps(s, f, t);
- }
- __forceinline vfloat16 lerp(const vfloat16& a, const vfloat16& b, const vfloat16& t) {
- return madd(t,b-a,a);
- }
- __forceinline void xchg(vboolf16 m, vfloat16& a, vfloat16& b)
- {
- vfloat16 c = a;
- a = select(m,b,a);
- b = select(m,c,b);
- }
- ////////////////////////////////////////////////////////////////////////////////
- /// Rounding Functions
- ////////////////////////////////////////////////////////////////////////////////
-
- __forceinline vfloat16 floor(const vfloat16& a) {
- return _mm512_floor_ps(a);
- }
- __forceinline vfloat16 ceil (const vfloat16& a) {
- return _mm512_ceil_ps(a);
- }
- #if !defined (__clang__) // FIXME: not yet supported in clang v4.0.0
- __forceinline vfloat16 trunc(const vfloat16& a) {
- return _mm512_trunc_ps(a);
- }
- __forceinline vfloat16 frac( const vfloat16& a ) {
- return a-trunc(a);
- }
- #endif
- __forceinline vint16 floori (const vfloat16& a) {
- return _mm512_cvt_roundps_epi32(a, _MM_FROUND_TO_NEG_INF | _MM_FROUND_NO_EXC);
- }
- ////////////////////////////////////////////////////////////////////////////////
- /// Movement/Shifting/Shuffling Functions
- ////////////////////////////////////////////////////////////////////////////////
- template<size_t i>
- __forceinline const vfloat16 shuffle( const vfloat16& a ) {
- return _mm512_permute_ps(a, _MM_SHUFFLE(i, i, i, i));
- }
- template<int A, int B, int C, int D>
- __forceinline vfloat16 shuffle (const vfloat16& v) {
- return _mm512_permute_ps(v,_MM_SHUFFLE(D,C,B,A));
- }
- template<int i>
- __forceinline vfloat16 shuffle4(const vfloat16& x) {
- return _mm512_shuffle_f32x4(x,x,_MM_SHUFFLE(i,i,i,i));
- }
- template<int A, int B, int C, int D>
- __forceinline vfloat16 shuffle4(const vfloat16& x) {
- return _mm512_shuffle_f32x4(x,x,_MM_SHUFFLE(D,C,B,A));
- }
- __forceinline vfloat16 permute(vfloat16 v,__m512i index)
- {
- return _mm512_castsi512_ps(_mm512_permutexvar_epi32(index,_mm512_castps_si512(v)));
- }
- __forceinline vfloat16 reverse(const vfloat16 &a)
- {
- return permute(a,_mm512_setr_epi32(15,14,13,12,11,10,9,8,7,6,5,4,3,2,1,0));
- }
- template<int i>
- __forceinline vfloat16 align_shift_right(const vfloat16 &a, const vfloat16 &b)
- {
- return _mm512_castsi512_ps(_mm512_alignr_epi32(_mm512_castps_si512(a),_mm512_castps_si512(b),i));
- };
- template<int i>
- __forceinline vfloat16 mask_align_shift_right(const vboolf16 &mask,vfloat16 &c,const vfloat16 &a, const vfloat16 &b)
- {
- return _mm512_castsi512_ps(_mm512_mask_alignr_epi32(_mm512_castps_si512(c),mask,_mm512_castps_si512(a),_mm512_castps_si512(b),i));
- };
-
- __forceinline vfloat16 shift_left_1(const vfloat16 &a) {
- vfloat16 z = vfloat16::zero();
- return mask_align_shift_right<15>(0xfffe,z,a,a);
- }
- __forceinline vfloat16 shift_right_1( const vfloat16& x) {
- return align_shift_right<1>(zero,x);
- }
- __forceinline float toScalar(const vfloat16& a) { return mm512_cvtss_f32(a); }
- template<int i> __forceinline const vfloat16 insert4(const vfloat16& a, const vfloat4& b) { return _mm512_insertf32x4(a, b, i); }
- template<int N, int i>
- vfloat<N> extractN(const vfloat16& v);
- template<> __forceinline vfloat4 extractN<4,0>(const vfloat16& v) { return _mm512_castps512_ps128(v); }
- template<> __forceinline vfloat4 extractN<4,1>(const vfloat16& v) { return _mm512_extractf32x4_ps(v, 1); }
- template<> __forceinline vfloat4 extractN<4,2>(const vfloat16& v) { return _mm512_extractf32x4_ps(v, 2); }
- template<> __forceinline vfloat4 extractN<4,3>(const vfloat16& v) { return _mm512_extractf32x4_ps(v, 3); }
- template<> __forceinline vfloat8 extractN<8,0>(const vfloat16& v) { return _mm512_castps512_ps256(v); }
- template<> __forceinline vfloat8 extractN<8,1>(const vfloat16& v) { return _mm512_extractf32x8_ps(v, 1); }
- template<int i> __forceinline vfloat4 extract4 (const vfloat16& v) { return _mm512_extractf32x4_ps(v, i); }
- template<> __forceinline vfloat4 extract4<0>(const vfloat16& v) { return _mm512_castps512_ps128(v); }
- template<int i> __forceinline vfloat8 extract8 (const vfloat16& v) { return _mm512_extractf32x8_ps(v, i); }
- template<> __forceinline vfloat8 extract8<0>(const vfloat16& v) { return _mm512_castps512_ps256(v); }
- ////////////////////////////////////////////////////////////////////////////////
- /// Reductions
- ////////////////////////////////////////////////////////////////////////////////
- __forceinline float reduce_add(const vfloat16 &a) { return _mm512_reduce_add_ps(a); }
- __forceinline float reduce_mul(const vfloat16 &a) { return _mm512_reduce_mul_ps(a); }
- __forceinline float reduce_min(const vfloat16 &a) { return _mm512_reduce_min_ps(a); }
- __forceinline float reduce_max(const vfloat16 &a) { return _mm512_reduce_max_ps(a); }
- __forceinline vfloat16 vreduce_add2(vfloat16 x) { return x + shuffle<1,0,3,2>(x); }
- __forceinline vfloat16 vreduce_add4(vfloat16 x) { x = vreduce_add2(x); return x + shuffle<2,3,0,1>(x); }
- __forceinline vfloat16 vreduce_add8(vfloat16 x) { x = vreduce_add4(x); return x + shuffle4<1,0,3,2>(x); }
- __forceinline vfloat16 vreduce_add (vfloat16 x) { x = vreduce_add8(x); return x + shuffle4<2,3,0,1>(x); }
- __forceinline vfloat16 vreduce_min2(vfloat16 x) { return min(x,shuffle<1,0,3,2>(x)); }
- __forceinline vfloat16 vreduce_min4(vfloat16 x) { x = vreduce_min2(x); return min(x,shuffle<2,3,0,1>(x)); }
- __forceinline vfloat16 vreduce_min8(vfloat16 x) { x = vreduce_min4(x); return min(x,shuffle4<1,0,3,2>(x)); }
- __forceinline vfloat16 vreduce_min (vfloat16 x) { x = vreduce_min8(x); return min(x,shuffle4<2,3,0,1>(x)); }
- __forceinline vfloat16 vreduce_max2(vfloat16 x) { return max(x,shuffle<1,0,3,2>(x)); }
- __forceinline vfloat16 vreduce_max4(vfloat16 x) { x = vreduce_max2(x); return max(x,shuffle<2,3,0,1>(x)); }
- __forceinline vfloat16 vreduce_max8(vfloat16 x) { x = vreduce_max4(x); return max(x,shuffle4<1,0,3,2>(x)); }
- __forceinline vfloat16 vreduce_max (vfloat16 x) { x = vreduce_max8(x); return max(x,shuffle4<2,3,0,1>(x)); }
- __forceinline size_t select_min(const vfloat16& v) {
- return __bsf(_mm512_kmov(_mm512_cmp_epi32_mask(_mm512_castps_si512(v),_mm512_castps_si512(vreduce_min(v)),_MM_CMPINT_EQ)));
- }
- __forceinline size_t select_max(const vfloat16& v) {
- return __bsf(_mm512_kmov(_mm512_cmp_epi32_mask(_mm512_castps_si512(v),_mm512_castps_si512(vreduce_max(v)),_MM_CMPINT_EQ)));
- }
- __forceinline size_t select_min(const vboolf16& valid, const vfloat16& v)
- {
- const vfloat16 a = select(valid,v,vfloat16(pos_inf));
- const vbool16 valid_min = valid & (a == vreduce_min(a));
- return __bsf(movemask(any(valid_min) ? valid_min : valid));
- }
- __forceinline size_t select_max(const vboolf16& valid, const vfloat16& v)
- {
- const vfloat16 a = select(valid,v,vfloat16(neg_inf));
- const vbool16 valid_max = valid & (a == vreduce_max(a));
- return __bsf(movemask(any(valid_max) ? valid_max : valid));
- }
-
- __forceinline vfloat16 prefix_sum(const vfloat16& a)
- {
- const vfloat16 z(zero);
- vfloat16 v = a;
- v = v + align_shift_right<16-1>(v,z);
- v = v + align_shift_right<16-2>(v,z);
- v = v + align_shift_right<16-4>(v,z);
- v = v + align_shift_right<16-8>(v,z);
- return v;
- }
- __forceinline vfloat16 reverse_prefix_sum(const vfloat16& a)
- {
- const vfloat16 z(zero);
- vfloat16 v = a;
- v = v + align_shift_right<1>(z,v);
- v = v + align_shift_right<2>(z,v);
- v = v + align_shift_right<4>(z,v);
- v = v + align_shift_right<8>(z,v);
- return v;
- }
- __forceinline vfloat16 prefix_min(const vfloat16& a)
- {
- const vfloat16 z(pos_inf);
- vfloat16 v = a;
- v = min(v,align_shift_right<16-1>(v,z));
- v = min(v,align_shift_right<16-2>(v,z));
- v = min(v,align_shift_right<16-4>(v,z));
- v = min(v,align_shift_right<16-8>(v,z));
- return v;
- }
- __forceinline vfloat16 prefix_max(const vfloat16& a)
- {
- const vfloat16 z(neg_inf);
- vfloat16 v = a;
- v = max(v,align_shift_right<16-1>(v,z));
- v = max(v,align_shift_right<16-2>(v,z));
- v = max(v,align_shift_right<16-4>(v,z));
- v = max(v,align_shift_right<16-8>(v,z));
- return v;
- }
- __forceinline vfloat16 reverse_prefix_min(const vfloat16& a)
- {
- const vfloat16 z(pos_inf);
- vfloat16 v = a;
- v = min(v,align_shift_right<1>(z,v));
- v = min(v,align_shift_right<2>(z,v));
- v = min(v,align_shift_right<4>(z,v));
- v = min(v,align_shift_right<8>(z,v));
- return v;
- }
- __forceinline vfloat16 reverse_prefix_max(const vfloat16& a)
- {
- const vfloat16 z(neg_inf);
- vfloat16 v = a;
- v = max(v,align_shift_right<1>(z,v));
- v = max(v,align_shift_right<2>(z,v));
- v = max(v,align_shift_right<4>(z,v));
- v = max(v,align_shift_right<8>(z,v));
- return v;
- }
- ////////////////////////////////////////////////////////////////////////////////
- /// Memory load and store operations
- ////////////////////////////////////////////////////////////////////////////////
- __forceinline void compactustore16f_low(const vboolf16& mask, float * addr, const vfloat16 ®) {
- _mm512_mask_compressstoreu_ps(addr,mask,reg);
- }
- template<int scale = 4>
- __forceinline vfloat16 gather16f(const vboolf16& mask, const float *const ptr, __m512i index) {
- vfloat16 r = vfloat16::undefined();
- return _mm512_mask_i32gather_ps(r,mask,index,ptr,scale);
- }
-
- template<int scale = 4>
- __forceinline void scatter16f(const vboolf16& mask,const float *const ptr, const __m512i index,const vfloat16 v) {
- _mm512_mask_i32scatter_ps((void*)ptr,mask,index,v,scale);
- }
- __forceinline vfloat16 loadAOS4to16f(const float& x,const float& y, const float& z)
- {
- vfloat16 f = vfloat16::zero();
- f = select(0x1111,vfloat16::broadcast(&x),f);
- f = select(0x2222,vfloat16::broadcast(&y),f);
- f = select(0x4444,vfloat16::broadcast(&z),f);
- return f;
- }
- __forceinline vfloat16 loadAOS4to16f(const unsigned int index,
- const vfloat16 &x,
- const vfloat16 &y,
- const vfloat16 &z)
- {
- vfloat16 f = vfloat16::zero();
- f = select(0x1111,vfloat16::broadcast((float*)&x + index),f);
- f = select(0x2222,vfloat16::broadcast((float*)&y + index),f);
- f = select(0x4444,vfloat16::broadcast((float*)&z + index),f);
- return f;
- }
- __forceinline vfloat16 loadAOS4to16f(const unsigned int index,
- const vfloat16 &x,
- const vfloat16 &y,
- const vfloat16 &z,
- const vfloat16 &fill)
- {
- vfloat16 f = fill;
- f = select(0x1111,vfloat16::broadcast((float*)&x + index),f);
- f = select(0x2222,vfloat16::broadcast((float*)&y + index),f);
- f = select(0x4444,vfloat16::broadcast((float*)&z + index),f);
- return f;
- }
- __forceinline vfloat16 rcp_safe( const vfloat16& a ) {
- return rcp(select(a != vfloat16::zero(),a,vfloat16(min_rcp_input)));
- }
- ////////////////////////////////////////////////////////////////////////////////
- /// Output Operators
- ////////////////////////////////////////////////////////////////////////////////
-
- __forceinline std::ostream &operator<<(std::ostream& cout, const vfloat16& v)
- {
- cout << "<" << v[0];
- for (int i=1; i<16; i++) cout << ", " << v[i];
- cout << ">";
- return cout;
- }
- }
|