ClickHouse/libs/libvectorclass/vectorf128.h
2014-12-02 20:25:10 +03:00

2620 lines
92 KiB
C++

/**************************** vectorf128.h *******************************
* Author: Agner Fog
* Date created: 2012-05-30
* Last modified: 2014-10-24
* Version: 1.16
* Project: vector classes
* Description:
* Header file defining floating point vector classes as interface to
* intrinsic functions in x86 microprocessors with SSE2 and later instruction
* sets up to AVX.
*
* Instructions:
* Use Gnu, Intel or Microsoft C++ compiler. Compile for the desired
* instruction set, which must be at least SSE2. Specify the supported
* instruction set by a command line define, e.g. __SSE4_1__ if the
* compiler does not automatically do so.
*
* The following vector classes are defined here:
* Vec4f Vector of 4 single precision floating point numbers
* Vec4fb Vector of 4 Booleans for use with Vec4f
* Vec2d Vector of 2 double precision floating point numbers
* Vec2db Vector of 2 Booleans for use with Vec2d
*
* Each vector object is represented internally in the CPU as a 128-bit register.
* This header file defines operators and functions for these vectors.
*
* For example:
* Vec2d a(1.0, 2.0), b(3.0, 4.0), c;
* c = a + b; // now c contains (4.0, 6.0)
*
* For detailed instructions, see VectorClass.pdf
*
* (c) Copyright 2012 - 2014 GNU General Public License http://www.gnu.org/licenses
*****************************************************************************/
#ifndef VECTORF128_H
#define VECTORF128_H
#include "vectori128.h" // Define integer vectors
/*****************************************************************************
*
* select functions
*
*****************************************************************************/
// Select between two __m128 sources, element by element. Used in various functions
// and operators. Corresponds to this pseudocode:
// for (int i = 0; i < 4; i++) result[i] = s[i] ? a[i] : b[i];
// Each element in s must be either 0 (false) or 0xFFFFFFFF (true). No other values are
// allowed. The implementation depends on the instruction set:
// If SSE4.1 is supported then only bit 31 in each dword of s is checked,
// otherwise all bits in s are used.
static inline __m128 selectf (__m128 const & s, __m128 const & a, __m128 const & b) {
#if INSTRSET >= 5 // SSE4.1 supported
return _mm_blendv_ps (b, a, s);
#else
return _mm_or_ps(
_mm_and_ps(s,a),
_mm_andnot_ps(s,b));
#endif
}
// Same, with two __m128d sources.
// and operators. Corresponds to this pseudocode:
// for (int i = 0; i < 2; i++) result[i] = s[i] ? a[i] : b[i];
// Each element in s must be either 0 (false) or 0xFFFFFFFFFFFFFFFF (true). No other
// values are allowed. The implementation depends on the instruction set:
// If SSE4.1 is supported then only bit 63 in each dword of s is checked,
// otherwise all bits in s are used.
static inline __m128d selectd (__m128d const & s, __m128d const & a, __m128d const & b) {
#if INSTRSET >= 5 // SSE4.1 supported
return _mm_blendv_pd (b, a, s);
#else
return _mm_or_pd(
_mm_and_pd(s,a),
_mm_andnot_pd(s,b));
#endif
}
/*****************************************************************************
*
* Vec4fb: Vector of 4 Booleans for use with Vec4f
*
*****************************************************************************/
class Vec4fb {
protected:
__m128 xmm; // Float vector
public:
// Default constructor:
Vec4fb() {
}
// Constructor to build from all elements:
Vec4fb(bool b0, bool b1, bool b2, bool b3) {
xmm = _mm_castsi128_ps(_mm_setr_epi32(-(int)b0, -(int)b1, -(int)b2, -(int)b3));
}
// Constructor to convert from type __m128 used in intrinsics:
Vec4fb(__m128 const & x) {
xmm = x;
}
// Assignment operator to convert from type __m128 used in intrinsics:
Vec4fb & operator = (__m128 const & x) {
xmm = x;
return *this;
}
// Constructor to broadcast scalar value:
Vec4fb(bool b) {
xmm = _mm_castsi128_ps(_mm_set1_epi32(-int32_t(b)));
}
// Assignment operator to broadcast scalar value:
Vec4fb & operator = (bool b) {
*this = Vec4fb(b);
return *this;
}
private: // Prevent constructing from int, etc.
Vec4fb(int b);
Vec4fb & operator = (int x);
public:
// Constructor to convert from type Vec4ib used as Boolean for integer vectors
Vec4fb(Vec4ib const & x) {
xmm = _mm_castsi128_ps(x);
}
// Assignment operator to convert from type Vec4ib used as Boolean for integer vectors
Vec4fb & operator = (Vec4ib const & x) {
xmm = _mm_castsi128_ps(x);
return *this;
}
// Type cast operator to convert to __m128 used in intrinsics
operator __m128() const {
return xmm;
}
#if defined (__clang__) && CLANG_VERSION < 30900 || defined(__apple_build_version__)
#define FIX_CLANG_VECTOR_ALIAS_AMBIGUITY // clang 3.3 - 3.5 has silent conversion between intrinsic vector types.
// I expected this to be fixed in version 3.4 but it still exists!
// http://llvm.org/bugs/show_bug.cgi?id=17164
// Problem: The version number is not consistent across platforms
// The Apple build has different version numbers. Too bad!
// http://llvm.org/bugs/show_bug.cgi?id=12643
#else
// Type cast operator to convert to type Vec4ib used as Boolean for integer vectors
operator Vec4ib() const {
return _mm_castps_si128(xmm);
}
#endif
// Member function to change a single element in vector
// Note: This function is inefficient. Use load function if changing more than one element
Vec4fb const & insert(uint32_t index, bool value) {
static const int32_t maskl[8] = {0,0,0,0,-1,0,0,0};
__m128 mask = _mm_loadu_ps((float const*)(maskl+4-(index & 3))); // mask with FFFFFFFF at index position
if (value) {
xmm = _mm_or_ps(xmm,mask);
}
else {
xmm = _mm_andnot_ps(mask,xmm);
}
return *this;
}
// Member function extract a single element from vector
bool extract(uint32_t index) const {
//return Vec4ib(*this).extract(index);
return Vec4ib(_mm_castps_si128(xmm)).extract(index);
}
// Extract a single element. Operator [] can only read an element, not write.
bool operator [] (uint32_t index) const {
return extract(index);
}
static int size() {
return 4;
}
};
/*****************************************************************************
*
* Operators for Vec4fb
*
*****************************************************************************/
// vector operator & : bitwise and
static inline Vec4fb operator & (Vec4fb const & a, Vec4fb const & b) {
return _mm_and_ps(a, b);
}
static inline Vec4fb operator && (Vec4fb const & a, Vec4fb const & b) {
return a & b;
}
// vector operator &= : bitwise and
static inline Vec4fb & operator &= (Vec4fb & a, Vec4fb const & b) {
a = a & b;
return a;
}
// vector operator | : bitwise or
static inline Vec4fb operator | (Vec4fb const & a, Vec4fb const & b) {
return _mm_or_ps(a, b);
}
static inline Vec4fb operator || (Vec4fb const & a, Vec4fb const & b) {
return a | b;
}
// vector operator |= : bitwise or
static inline Vec4fb & operator |= (Vec4fb & a, Vec4fb const & b) {
a = a | b;
return a;
}
// vector operator ^ : bitwise xor
static inline Vec4fb operator ^ (Vec4fb const & a, Vec4fb const & b) {
return _mm_xor_ps(a, b);
}
// vector operator ^= : bitwise xor
static inline Vec4fb & operator ^= (Vec4fb & a, Vec4fb const & b) {
a = a ^ b;
return a;
}
// vector operator ~ : bitwise not
static inline Vec4fb operator ~ (Vec4fb const & a) {
return _mm_xor_ps(a, _mm_castsi128_ps(_mm_set1_epi32(-1)));
}
// vector operator ! : logical not
// (operator ! is less efficient than operator ~. Use only where not
// all bits in an element are the same)
static inline Vec4fb operator ! (Vec4fb const & a) {
return Vec4fb( ! Vec4ib(a));
}
// Functions for Vec4fb
// andnot: a & ~ b
static inline Vec4fb andnot(Vec4fb const & a, Vec4fb const & b) {
return _mm_andnot_ps(b, a);
}
/*****************************************************************************
*
* Horizontal Boolean functions
*
*****************************************************************************/
// horizontal_and. Returns true if all bits are 1
static inline bool horizontal_and (Vec4fb const & a) {
return horizontal_and(Vec128b(_mm_castps_si128(a)));
}
// horizontal_or. Returns true if at least one bit is 1
static inline bool horizontal_or (Vec4fb const & a) {
return horizontal_or(Vec128b(_mm_castps_si128(a)));
}
/*****************************************************************************
*
* Vec2db: Vector of 2 Booleans for use with Vec2d
*
*****************************************************************************/
class Vec2db {
protected:
__m128d xmm; // Double vector
public:
// Default constructor:
Vec2db() {
}
// Constructor to broadcast the same value into all elements:
// Constructor to build from all elements:
Vec2db(bool b0, bool b1) {
xmm = _mm_castsi128_pd(_mm_setr_epi32(-(int)b0, -(int)b0, -(int)b1, -(int)b1));
}
// Constructor to convert from type __m128d used in intrinsics:
Vec2db(__m128d const & x) {
xmm = x;
}
// Assignment operator to convert from type __m128d used in intrinsics:
Vec2db & operator = (__m128d const & x) {
xmm = x;
return *this;
}
// Constructor to broadcast scalar value:
Vec2db(bool b) {
xmm = _mm_castsi128_pd(_mm_set1_epi32(-int32_t(b)));
}
// Assignment operator to broadcast scalar value:
Vec2db & operator = (bool b) {
*this = Vec2db(b);
return *this;
}
private: // Prevent constructing from int, etc.
Vec2db(int b);
Vec2db & operator = (int x);
public:
// Constructor to convert from type Vec2qb used as Boolean for integer vectors
Vec2db(Vec2qb const & x) {
xmm = _mm_castsi128_pd(x);
}
// Assignment operator to convert from type Vec2qb used as Boolean for integer vectors
Vec2db & operator = (Vec2qb const & x) {
xmm = _mm_castsi128_pd(x);
return *this;
}
// Type cast operator to convert to __m128d used in intrinsics
operator __m128d() const {
return xmm;
}
#ifndef FIX_CLANG_VECTOR_ALIAS_AMBIGUITY
// Type cast operator to convert to type Vec2qb used as Boolean for integer vectors
operator Vec2qb() const {
return _mm_castpd_si128(xmm);
}
#endif
// Member function to change a single element in vector
// Note: This function is inefficient. Use load function if changing more than one element
Vec2db const & insert(uint32_t index, bool value) {
static const int32_t maskl[8] = {0,0,0,0,-1,-1,0,0};
__m128 mask = _mm_loadu_ps((float const*)(maskl+4-(index&1)*2)); // mask with FFFFFFFFFFFFFFFF at index position
if (value) {
xmm = _mm_or_pd(xmm,_mm_castps_pd(mask));
}
else {
xmm = _mm_andnot_pd(_mm_castps_pd(mask),xmm);
}
return *this;
}
// Member function extract a single element from vector
bool extract(uint32_t index) const {
return Vec2qb(*this).extract(index);
}
// Extract a single element. Operator [] can only read an element, not write.
bool operator [] (uint32_t index) const {
return extract(index);
}
static int size() {
return 2;
}
};
/*****************************************************************************
*
* Operators for Vec2db
*
*****************************************************************************/
// vector operator & : bitwise and
static inline Vec2db operator & (Vec2db const & a, Vec2db const & b) {
return _mm_and_pd(a, b);
}
static inline Vec2db operator && (Vec2db const & a, Vec2db const & b) {
return a & b;
}
// vector operator &= : bitwise and
static inline Vec2db & operator &= (Vec2db & a, Vec2db const & b) {
a = a & b;
return a;
}
// vector operator | : bitwise or
static inline Vec2db operator | (Vec2db const & a, Vec2db const & b) {
return _mm_or_pd(a, b);
}
static inline Vec2db operator || (Vec2db const & a, Vec2db const & b) {
return a | b;
}
// vector operator |= : bitwise or
static inline Vec2db & operator |= (Vec2db & a, Vec2db const & b) {
a = a | b;
return a;
}
// vector operator ^ : bitwise xor
static inline Vec2db operator ^ (Vec2db const & a, Vec2db const & b) {
return _mm_xor_pd(a, b);
}
// vector operator ^= : bitwise xor
static inline Vec2db & operator ^= (Vec2db & a, Vec2db const & b) {
a = a ^ b;
return a;
}
// vector operator ~ : bitwise not
static inline Vec2db operator ~ (Vec2db const & a) {
return _mm_xor_pd(a, _mm_castsi128_pd(_mm_set1_epi32(-1)));
}
// vector operator ! : logical not
// (operator ! is less efficient than operator ~. Use only where not
// all bits in an element are the same)
static inline Vec2db operator ! (Vec2db const & a) {
return Vec2db (! Vec2qb(a));
}
// Functions for Vec2db
// andnot: a & ~ b
static inline Vec2db andnot(Vec2db const & a, Vec2db const & b) {
return _mm_andnot_pd(b, a);
}
/*****************************************************************************
*
* Horizontal Boolean functions
*
*****************************************************************************/
// horizontal_and. Returns true if all bits are 1
static inline bool horizontal_and (Vec2db const & a) {
return horizontal_and(Vec128b(_mm_castpd_si128(a)));
}
// horizontal_or. Returns true if at least one bit is 1
static inline bool horizontal_or (Vec2db const & a) {
return horizontal_or(Vec128b(_mm_castpd_si128(a)));
}
/*****************************************************************************
*
* Vec4f: Vector of 4 single precision floating point values
*
*****************************************************************************/
class Vec4f {
protected:
__m128 xmm; // Float vector
public:
// Default constructor:
Vec4f() {
}
// Constructor to broadcast the same value into all elements:
Vec4f(float f) {
xmm = _mm_set1_ps(f);
}
// Constructor to build from all elements:
Vec4f(float f0, float f1, float f2, float f3) {
xmm = _mm_setr_ps(f0, f1, f2, f3);
}
// Constructor to convert from type __m128 used in intrinsics:
Vec4f(__m128 const & x) {
xmm = x;
}
// Assignment operator to convert from type __m128 used in intrinsics:
Vec4f & operator = (__m128 const & x) {
xmm = x;
return *this;
}
// Type cast operator to convert to __m128 used in intrinsics
operator __m128() const {
return xmm;
}
// Member function to load from array (unaligned)
Vec4f & load(float const * p) {
xmm = _mm_loadu_ps(p);
return *this;
}
// Member function to load from array, aligned by 16
// "load_a" is faster than "load" on older Intel processors (Pentium 4, Pentium M, Core 1,
// Merom, Wolfdale) and Atom, but not on other processors from Intel, AMD or VIA.
// You may use load_a instead of load if you are certain that p points to an address
// divisible by 16.
Vec4f & load_a(float const * p) {
xmm = _mm_load_ps(p);
return *this;
}
// Member function to store into array (unaligned)
void store(float * p) const {
_mm_storeu_ps(p, xmm);
}
// Member function to store into array, aligned by 16
// "store_a" is faster than "store" on older Intel processors (Pentium 4, Pentium M, Core 1,
// Merom, Wolfdale) and Atom, but not on other processors from Intel, AMD or VIA.
// You may use store_a instead of store if you are certain that p points to an address
// divisible by 16.
void store_a(float * p) const {
_mm_store_ps(p, xmm);
}
// Partial load. Load n elements and set the rest to 0
Vec4f & load_partial(int n, float const * p) {
__m128 t1, t2;
switch (n) {
case 1:
xmm = _mm_load_ss(p); break;
case 2:
xmm = _mm_castpd_ps(_mm_load_sd((double*)p)); break;
case 3:
t1 = _mm_castpd_ps(_mm_load_sd((double*)p));
t2 = _mm_load_ss(p + 2);
xmm = _mm_movelh_ps(t1, t2); break;
case 4:
load(p); break;
default:
xmm = _mm_setzero_ps();
}
return *this;
}
// Partial store. Store n elements
void store_partial(int n, float * p) const {
__m128 t1;
switch (n) {
case 1:
_mm_store_ss(p, xmm); break;
case 2:
_mm_store_sd((double*)p, _mm_castps_pd(xmm)); break;
case 3:
_mm_store_sd((double*)p, _mm_castps_pd(xmm));
t1 = _mm_movehl_ps(xmm,xmm);
_mm_store_ss(p + 2, t1); break;
case 4:
store(p); break;
default:;
}
}
// cut off vector to n elements. The last 4-n elements are set to zero
Vec4f & cutoff(int n) {
if (uint32_t(n) >= 4) return *this;
static const union {
int32_t i[8];
float f[8];
} mask = {{1,-1,-1,-1,0,0,0,0}};
xmm = _mm_and_ps(xmm, Vec4f().load(mask.f + 4 - n));
return *this;
}
// Member function to change a single element in vector
// Note: This function is inefficient. Use load function if changing more than one element
Vec4f const & insert(uint32_t index, float value) {
#if INSTRSET >= 5 // SSE4.1 supported
switch (index & 3) {
case 0:
xmm = _mm_insert_ps(xmm, _mm_set_ss(value), 0 << 4); break;
case 1:
xmm = _mm_insert_ps(xmm, _mm_set_ss(value), 1 << 4); break;
case 2:
xmm = _mm_insert_ps(xmm, _mm_set_ss(value), 2 << 4); break;
default:
xmm = _mm_insert_ps(xmm, _mm_set_ss(value), 3 << 4); break;
}
#else
static const int32_t maskl[8] = {0,0,0,0,-1,0,0,0};
__m128 broad = _mm_set1_ps(value); // broadcast value into all elements
__m128 mask = _mm_loadu_ps((float const*)(maskl+4-(index & 3))); // mask with FFFFFFFF at index position
xmm = selectf(mask,broad,xmm);
#endif
return *this;
};
// Member function extract a single element from vector
float extract(uint32_t index) const {
float x[4];
store(x);
return x[index & 3];
}
// Extract a single element. Use store function if extracting more than one element.
// Operator [] can only read an element, not write.
float operator [] (uint32_t index) const {
return extract(index);
}
static int size() {
return 4;
}
};
/*****************************************************************************
*
* Operators for Vec4f
*
*****************************************************************************/
// vector operator + : add element by element
static inline Vec4f operator + (Vec4f const & a, Vec4f const & b) {
return _mm_add_ps(a, b);
}
// vector operator + : add vector and scalar
static inline Vec4f operator + (Vec4f const & a, float b) {
return a + Vec4f(b);
}
static inline Vec4f operator + (float a, Vec4f const & b) {
return Vec4f(a) + b;
}
// vector operator += : add
static inline Vec4f & operator += (Vec4f & a, Vec4f const & b) {
a = a + b;
return a;
}
// postfix operator ++
static inline Vec4f operator ++ (Vec4f & a, int) {
Vec4f a0 = a;
a = a + 1.0f;
return a0;
}
// prefix operator ++
static inline Vec4f & operator ++ (Vec4f & a) {
a = a + 1.0f;
return a;
}
// vector operator - : subtract element by element
static inline Vec4f operator - (Vec4f const & a, Vec4f const & b) {
return _mm_sub_ps(a, b);
}
// vector operator - : subtract vector and scalar
static inline Vec4f operator - (Vec4f const & a, float b) {
return a - Vec4f(b);
}
static inline Vec4f operator - (float a, Vec4f const & b) {
return Vec4f(a) - b;
}
// vector operator - : unary minus
// Change sign bit, even for 0, INF and NAN
static inline Vec4f operator - (Vec4f const & a) {
return _mm_xor_ps(a, _mm_castsi128_ps(_mm_set1_epi32(0x80000000)));
}
// vector operator -= : subtract
static inline Vec4f & operator -= (Vec4f & a, Vec4f const & b) {
a = a - b;
return a;
}
// postfix operator --
static inline Vec4f operator -- (Vec4f & a, int) {
Vec4f a0 = a;
a = a - 1.0f;
return a0;
}
// prefix operator --
static inline Vec4f & operator -- (Vec4f & a) {
a = a - 1.0f;
return a;
}
// vector operator * : multiply element by element
static inline Vec4f operator * (Vec4f const & a, Vec4f const & b) {
return _mm_mul_ps(a, b);
}
// vector operator * : multiply vector and scalar
static inline Vec4f operator * (Vec4f const & a, float b) {
return a * Vec4f(b);
}
static inline Vec4f operator * (float a, Vec4f const & b) {
return Vec4f(a) * b;
}
// vector operator *= : multiply
static inline Vec4f & operator *= (Vec4f & a, Vec4f const & b) {
a = a * b;
return a;
}
// vector operator / : divide all elements by same integer
static inline Vec4f operator / (Vec4f const & a, Vec4f const & b) {
return _mm_div_ps(a, b);
}
// vector operator / : divide vector and scalar
static inline Vec4f operator / (Vec4f const & a, float b) {
return a / Vec4f(b);
}
static inline Vec4f operator / (float a, Vec4f const & b) {
return Vec4f(a) / b;
}
// vector operator /= : divide
static inline Vec4f & operator /= (Vec4f & a, Vec4f const & b) {
a = a / b;
return a;
}
// vector operator == : returns true for elements for which a == b
static inline Vec4fb operator == (Vec4f const & a, Vec4f const & b) {
return _mm_cmpeq_ps(a, b);
}
// vector operator != : returns true for elements for which a != b
static inline Vec4fb operator != (Vec4f const & a, Vec4f const & b) {
return _mm_cmpneq_ps(a, b);
}
// vector operator < : returns true for elements for which a < b
static inline Vec4fb operator < (Vec4f const & a, Vec4f const & b) {
return _mm_cmplt_ps(a, b);
}
// vector operator <= : returns true for elements for which a <= b
static inline Vec4fb operator <= (Vec4f const & a, Vec4f const & b) {
return _mm_cmple_ps(a, b);
}
// vector operator > : returns true for elements for which a > b
static inline Vec4fb operator > (Vec4f const & a, Vec4f const & b) {
return b < a;
}
// vector operator >= : returns true for elements for which a >= b
static inline Vec4fb operator >= (Vec4f const & a, Vec4f const & b) {
return b <= a;
}
// Bitwise logical operators
// vector operator & : bitwise and
static inline Vec4f operator & (Vec4f const & a, Vec4f const & b) {
return _mm_and_ps(a, b);
}
// vector operator &= : bitwise and
static inline Vec4f & operator &= (Vec4f & a, Vec4f const & b) {
a = a & b;
return a;
}
// vector operator & : bitwise and of Vec4f and Vec4fb
static inline Vec4f operator & (Vec4f const & a, Vec4fb const & b) {
return _mm_and_ps(a, b);
}
static inline Vec4f operator & (Vec4fb const & a, Vec4f const & b) {
return _mm_and_ps(a, b);
}
// vector operator | : bitwise or
static inline Vec4f operator | (Vec4f const & a, Vec4f const & b) {
return _mm_or_ps(a, b);
}
// vector operator |= : bitwise or
static inline Vec4f & operator |= (Vec4f & a, Vec4f const & b) {
a = a | b;
return a;
}
// vector operator ^ : bitwise xor
static inline Vec4f operator ^ (Vec4f const & a, Vec4f const & b) {
return _mm_xor_ps(a, b);
}
// vector operator ^= : bitwise xor
static inline Vec4f & operator ^= (Vec4f & a, Vec4f const & b) {
a = a ^ b;
return a;
}
// vector operator ! : logical not. Returns Boolean vector
static inline Vec4fb operator ! (Vec4f const & a) {
return a == Vec4f(0.0f);
}
/*****************************************************************************
*
* Functions for Vec4f
*
*****************************************************************************/
// Select between two operands. Corresponds to this pseudocode:
// for (int i = 0; i < 4; i++) result[i] = s[i] ? a[i] : b[i];
// Each byte in s must be either 0 (false) or 0xFFFFFFFF (true). No other values are allowed.
static inline Vec4f select (Vec4fb const & s, Vec4f const & a, Vec4f const & b) {
return selectf(s,a,b);
}
// Conditional add: For all vector elements i: result[i] = f[i] ? (a[i] + b[i]) : a[i]
static inline Vec4f if_add (Vec4fb const & f, Vec4f const & a, Vec4f const & b) {
return a + (Vec4f(f) & b);
}
// Conditional multiply: For all vector elements i: result[i] = f[i] ? (a[i] * b[i]) : a[i]
static inline Vec4f if_mul (Vec4fb const & f, Vec4f const & a, Vec4f const & b) {
return a * select(f, b, 1.f);
}
// General arithmetic functions, etc.
// Horizontal add: Calculates the sum of all vector elements.
static inline float horizontal_add (Vec4f const & a) {
#if INSTRSET >= 3 // SSE3
__m128 t1 = _mm_hadd_ps(a,a);
__m128 t2 = _mm_hadd_ps(t1,t1);
return _mm_cvtss_f32(t2);
#else
__m128 t1 = _mm_movehl_ps(a,a);
__m128 t2 = _mm_add_ps(a,t1);
__m128 t3 = _mm_shuffle_ps(t2,t2,1);
__m128 t4 = _mm_add_ss(t2,t3);
return _mm_cvtss_f32(t4);
#endif
}
// function max: a > b ? a : b
static inline Vec4f max(Vec4f const & a, Vec4f const & b) {
return _mm_max_ps(a,b);
}
// function min: a < b ? a : b
static inline Vec4f min(Vec4f const & a, Vec4f const & b) {
return _mm_min_ps(a,b);
}
// function abs: absolute value
// Removes sign bit, even for -0.0f, -INF and -NAN
static inline Vec4f abs(Vec4f const & a) {
__m128 mask = _mm_castsi128_ps(_mm_set1_epi32(0x7FFFFFFF));
return _mm_and_ps(a,mask);
}
// function sqrt: square root
static inline Vec4f sqrt(Vec4f const & a) {
return _mm_sqrt_ps(a);
}
// function square: a * a
static inline Vec4f square(Vec4f const & a) {
return a * a;
}
// pow(vector,int) function template
template <typename VTYPE>
static inline VTYPE pow_template_i(VTYPE const & x0, int n) {
VTYPE x = x0; // a^(2^i)
VTYPE y(1.0f); // accumulator
if (n >= 0) { // make sure n is not negative
while (true) { // loop for each bit in n
if (n & 1) y *= x; // multiply if bit = 1
n >>= 1; // get next bit of n
if (n == 0) return y; // finished
x *= x; // x = a^2, a^4, a^8, etc.
}
}
else { // n < 0
return VTYPE(1.0f)/pow_template_i<VTYPE>(x0,-n); // reciprocal
}
}
// pow(Vec4f, int):
// The purpose of this template is to prevent implicit conversion of a float
// exponent to int when calling pow(vector, float) and vectormath_exp.h is
// not included
template <typename TT> static Vec4f pow(Vec4f const & a, TT n);
// Raise floating point numbers to integer power n
template <>
inline Vec4f pow<int>(Vec4f const & x0, int n) {
return pow_template_i<Vec4f>(x0, n);
}
// allow conversion from unsigned int
template <>
inline Vec4f pow<uint32_t>(Vec4f const & x0, uint32_t n) {
return pow_template_i<Vec4f>(x0, (int)n);
}
// Raise floating point numbers to integer power n, where n is a compile-time constant
template <int n>
static inline Vec4f pow_n(Vec4f const & a) {
if (n < 0) return Vec4f(1.0f) / pow_n<-n>(a);
if (n == 0) return Vec4f(1.0f);
if (n >= 256) return pow(a, n);
Vec4f x = a; // a^(2^i)
Vec4f y; // accumulator
const int lowest = n - (n & (n-1));// lowest set bit in n
if (n & 1) y = x;
if (n < 2) return y;
x = x*x; // x^2
if (n & 2) {
if (lowest == 2) y = x; else y *= x;
}
if (n < 4) return y;
x = x*x; // x^4
if (n & 4) {
if (lowest == 4) y = x; else y *= x;
}
if (n < 8) return y;
x = x*x; // x^8
if (n & 8) {
if (lowest == 8) y = x; else y *= x;
}
if (n < 16) return y;
x = x*x; // x^16
if (n & 16) {
if (lowest == 16) y = x; else y *= x;
}
if (n < 32) return y;
x = x*x; // x^32
if (n & 32) {
if (lowest == 32) y = x; else y *= x;
}
if (n < 64) return y;
x = x*x; // x^64
if (n & 64) {
if (lowest == 64) y = x; else y *= x;
}
if (n < 128) return y;
x = x*x; // x^128
if (n & 128) {
if (lowest == 128) y = x; else y *= x;
}
return y;
}
// implement as function pow(vector, const_int)
template <int n>
static inline Vec4f pow(Vec4f const & a, Const_int_t<n>) {
return pow_n<n>(a);
}
// implement the same as macro pow_const(vector, int)
#define pow_const(x,n) pow_n<n>(x)
// avoid unsafe optimization in function round
#if defined(__GNUC__) && !defined(__INTEL_COMPILER) && !defined(__clang__) && INSTRSET < 5
static inline Vec4f round(Vec4f const & a) __attribute__ ((optimize("-fno-unsafe-math-optimizations")));
#elif defined(__clang__) && INSTRSET < 5
// static inline Vec4f round(Vec4f const & a) __attribute__ ((optnone));
// This doesn't work, but current versions of Clang (3.5) don't optimize away signedmagic, even with -funsafe-math-optimizations
// Add volatile to b if future versions fail
#elif defined (_MSC_VER) || defined(__INTEL_COMPILER) && INSTRSET < 5
#pragma float_control(push)
#pragma float_control(precise,on)
#define FLOAT_CONTROL_PRECISE_FOR_ROUND
#endif
// function round: round to nearest integer (even). (result as float vector)
static inline Vec4f round(Vec4f const & a) {
#if INSTRSET >= 5 // SSE4.1 supported
return _mm_round_ps(a, 0);
#else // SSE2. Use magic number method
// Note: assume MXCSR control register is set to rounding
// (don't use conversion to int, it will limit the value to +/- 2^31)
Vec4f signmask = _mm_castsi128_ps(constant4i<(int)0x80000000,(int)0x80000000,(int)0x80000000,(int)0x80000000>()); // -0.0
Vec4f magic = _mm_castsi128_ps(constant4i<0x4B000000,0x4B000000,0x4B000000,0x4B000000>()); // magic number = 2^23
Vec4f sign = _mm_and_ps(a, signmask); // signbit of a
Vec4f signedmagic = _mm_or_ps(magic, sign); // magic number with sign of a
// volatile
Vec4f b = a + signedmagic; // round by adding magic number
return b - signedmagic; // .. and subtracting it again
#endif
}
#ifdef FLOAT_CONTROL_PRECISE_FOR_ROUND
#pragma float_control(pop)
#endif
// function truncate: round towards zero. (result as float vector)
static inline Vec4f truncate(Vec4f const & a) {
#if INSTRSET >= 5 // SSE4.1 supported
return _mm_round_ps(a, 3);
#else // SSE2. Use magic number method (conversion to int would limit the value to 2^31)
uint32_t t1 = _mm_getcsr(); // MXCSR
uint32_t t2 = t1 | (3 << 13); // bit 13-14 = 11
_mm_setcsr(t2); // change MXCSR
Vec4f r = round(a); // use magic number method
_mm_setcsr(t1); // restore MXCSR
return r;
#endif
}
// function floor: round towards minus infinity. (result as float vector)
static inline Vec4f floor(Vec4f const & a) {
#if INSTRSET >= 5 // SSE4.1 supported
return _mm_round_ps(a, 1);
#else // SSE2. Use magic number method (conversion to int would limit the value to 2^31)
uint32_t t1 = _mm_getcsr(); // MXCSR
uint32_t t2 = t1 | (1 << 13); // bit 13-14 = 01
_mm_setcsr(t2); // change MXCSR
Vec4f r = round(a); // use magic number method
_mm_setcsr(t1); // restore MXCSR
return r;
#endif
}
// function ceil: round towards plus infinity. (result as float vector)
static inline Vec4f ceil(Vec4f const & a) {
#if INSTRSET >= 5 // SSE4.1 supported
return _mm_round_ps(a, 2);
#else // SSE2. Use magic number method (conversion to int would limit the value to 2^31)
uint32_t t1 = _mm_getcsr(); // MXCSR
uint32_t t2 = t1 | (2 << 13); // bit 13-14 = 10
_mm_setcsr(t2); // change MXCSR
Vec4f r = round(a); // use magic number method
_mm_setcsr(t1); // restore MXCSR
return r;
#endif
}
// function round_to_int: round to nearest integer (even). (result as integer vector)
static inline Vec4i round_to_int(Vec4f const & a) {
// Note: assume MXCSR control register is set to rounding
return _mm_cvtps_epi32(a);
}
// function truncate_to_int: round towards zero. (result as integer vector)
static inline Vec4i truncate_to_int(Vec4f const & a) {
return _mm_cvttps_epi32(a);
}
// function to_float: convert integer vector to float vector
static inline Vec4f to_float(Vec4i const & a) {
return _mm_cvtepi32_ps(a);
}
// Approximate math functions
// approximate reciprocal (Faster than 1.f / a. relative accuracy better than 2^-11)
static inline Vec4f approx_recipr(Vec4f const & a) {
return _mm_rcp_ps(a);
}
// approximate reciprocal squareroot (Faster than 1.f / sqrt(a). Relative accuracy better than 2^-11)
static inline Vec4f approx_rsqrt(Vec4f const & a) {
return _mm_rsqrt_ps(a);
}
// Fused multiply and add functions
// Multiply and add
static inline Vec4f mul_add(Vec4f const & a, Vec4f const & b, Vec4f const & c) {
#ifdef __FMA__
return _mm_fmadd_ps(a, b, c);
#elif defined (__FMA4__)
return _mm_macc_ps(a, b, c);
#else
return a * b + c;
#endif
}
// Multiply and subtract
static inline Vec4f mul_sub(Vec4f const & a, Vec4f const & b, Vec4f const & c) {
#ifdef __FMA__
return _mm_fmsub_ps(a, b, c);
#elif defined (__FMA4__)
return _mm_msub_ps(a, b, c);
#else
return a * b - c;
#endif
}
// Multiply and inverse subtract
static inline Vec4f nmul_add(Vec4f const & a, Vec4f const & b, Vec4f const & c) {
#ifdef __FMA__
return _mm_fnmadd_ps(a, b, c);
#elif defined (__FMA4__)
return _mm_nmacc_ps(a, b, c);
#else
return c - a * b;
#endif
}
// Multiply and subtract with extra precision on the intermediate calculations,
// even if FMA instructions not supported, using Veltkamp-Dekker split
static inline Vec4f mul_sub_x(Vec4f const & a, Vec4f const & b, Vec4f const & c) {
#ifdef __FMA__
return _mm_fmsub_ps(a, b, c);
#elif defined (__FMA4__)
return _mm_msub_ps(a, b, c);
#else
// calculate a * b - c with extra precision
Vec4i upper_mask = -(1 << 12); // mask to remove lower 12 bits
Vec4f a_high = a & Vec4f(_mm_castsi128_ps(upper_mask));// split into high and low parts
Vec4f b_high = b & Vec4f(_mm_castsi128_ps(upper_mask));
Vec4f a_low = a - a_high;
Vec4f b_low = b - b_high;
Vec4f r1 = a_high * b_high; // this product is exact
Vec4f r2 = r1 - c; // subtract c from high product
Vec4f r3 = r2 + (a_high * b_low + b_high * a_low) + a_low * b_low; // add rest of product
return r3; // + ((r2 - r1) + c);
#endif
}
// Math functions using fast bit manipulation
// Extract the exponent as an integer
// exponent(a) = floor(log2(abs(a)));
// exponent(1.0f) = 0, exponent(0.0f) = -127, exponent(INF) = +128, exponent(NAN) = +128
static inline Vec4i exponent(Vec4f const & a) {
Vec4ui t1 = _mm_castps_si128(a); // reinterpret as 32-bit integer
Vec4ui t2 = t1 << 1; // shift out sign bit
Vec4ui t3 = t2 >> 24; // shift down logical to position 0
Vec4i t4 = Vec4i(t3) - 0x7F; // subtract bias from exponent
return t4;
}
// Extract the fraction part of a floating point number
// a = 2^exponent(a) * fraction(a), except for a = 0
// fraction(1.0f) = 1.0f, fraction(5.0f) = 1.25f
static inline Vec4f fraction(Vec4f const & a) {
Vec4ui t1 = _mm_castps_si128(a); // reinterpret as 32-bit integer
Vec4ui t2 = Vec4ui((t1 & 0x007FFFFF) | 0x3F800000); // set exponent to 0 + bias
return _mm_castsi128_ps(t2);
}
// Fast calculation of pow(2,n) with n integer
// n = 0 gives 1.0f
// n >= 128 gives +INF
// n <= -127 gives 0.0f
// This function will never produce denormals, and never raise exceptions
static inline Vec4f exp2(Vec4i const & n) {
Vec4i t1 = max(n, -0x7F); // limit to allowed range
Vec4i t2 = min(t1, 0x80);
Vec4i t3 = t2 + 0x7F; // add bias
Vec4i t4 = t3 << 23; // put exponent into position 23
return _mm_castsi128_ps(t4); // reinterpret as float
}
//static Vec4f exp2(Vec4f const & x); // defined in vectormath_exp.h
// Control word manipulaton
// ------------------------
// The MXCSR control word has the following bits:
// 0: Invalid Operation Flag
// 1: Denormal Flag (=subnormal)
// 2: Divide-by-Zero Flag
// 3: Overflow Flag
// 4: Underflow Flag
// 5: Precision Flag
// 6: Denormals Are Zeros (=subnormals)
// 7: Invalid Operation Mask
// 8: Denormal Operation Mask (=subnormal)
// 9: Divide-by-Zero Mask
// 10: Overflow Mask
// 11: Underflow Mask
// 12: Precision Mask
// 13-14: Rounding control
// 00: round to nearest or even
// 01: round down towards -infinity
// 10: round up towards +infinity
// 11: round towards zero (truncate)
// 15: Flush to Zero
// Function get_control_word:
// Read the MXCSR control word
static inline uint32_t get_control_word() {
return _mm_getcsr();
}
// Function set_control_word:
// Write the MXCSR control word
static inline void set_control_word(uint32_t w) {
_mm_setcsr(w);
}
// Function no_subnormals:
// Set "Denormals Are Zeros" and "Flush to Zero" mode to avoid the extremely
// time-consuming denormals in case of underflow
static inline void no_subnormals() {
uint32_t t1 = get_control_word();
t1 |= (1 << 6) | (1 << 15); // set bit 6 and 15 in MXCSR
set_control_word(t1);
}
// Function reset_control_word:
// Set the MXCSR control word to the default value 0x1F80.
// This will mask floating point exceptions, set rounding mode to nearest (or even),
// and allow denormals.
static inline void reset_control_word() {
set_control_word(0x1F80);
}
// Categorization functions
// Function sign_bit: gives true for elements that have the sign bit set
// even for -0.0f, -INF and -NAN
// Note that sign_bit(Vec4f(-0.0f)) gives true, while Vec4f(-0.0f) < Vec4f(0.0f) gives false
// (the underscore in the name avoids a conflict with a macro in Intel's mathimf.h)
static inline Vec4fb sign_bit(Vec4f const & a) {
Vec4i t1 = _mm_castps_si128(a); // reinterpret as 32-bit integer
Vec4i t2 = t1 >> 31; // extend sign bit
return _mm_castsi128_ps(t2); // reinterpret as 32-bit Boolean
}
// Function sign_combine: changes the sign of a when b has the sign bit set
// same as select(sign_bit(b), -a, a)
static inline Vec4f sign_combine(Vec4f const & a, Vec4f const & b) {
Vec4f signmask = _mm_castsi128_ps(constant4i<(int)0x80000000,(int)0x80000000,(int)0x80000000,(int)0x80000000>()); // -0.0
return a ^ (b & signmask);
}
// Function is_finite: gives true for elements that are normal, denormal or zero,
// false for INF and NAN
// (the underscore in the name avoids a conflict with a macro in Intel's mathimf.h)
static inline Vec4fb is_finite(Vec4f const & a) {
Vec4i t1 = _mm_castps_si128(a); // reinterpret as 32-bit integer
Vec4i t2 = t1 << 1; // shift out sign bit
Vec4i t3 = Vec4i(t2 & 0xFF000000) != 0xFF000000; // exponent field is not all 1s
return Vec4ib(t3);
}
// Function is_inf: gives true for elements that are +INF or -INF
// false for finite numbers and NAN
// (the underscore in the name avoids a conflict with a macro in Intel's mathimf.h)
static inline Vec4fb is_inf(Vec4f const & a) {
Vec4i t1 = _mm_castps_si128(a); // reinterpret as 32-bit integer
Vec4i t2 = t1 << 1; // shift out sign bit
return t2 == Vec4i(0xFF000000); // exponent is all 1s, fraction is 0
}
// Function is_nan: gives true for elements that are +NAN or -NAN
// false for finite numbers and +/-INF
// (the underscore in the name avoids a conflict with a macro in Intel's mathimf.h)
static inline Vec4fb is_nan(Vec4f const & a) {
Vec4i t1 = _mm_castps_si128(a); // reinterpret as 32-bit integer
Vec4i t2 = t1 << 1; // shift out sign bit
Vec4i t3 = 0xFF000000; // exponent mask
Vec4i t4 = t2 & t3; // exponent
Vec4i t5 = _mm_andnot_si128(t3,t2);// fraction
return Vec4ib((t4 == t3) & (t5 != 0));// exponent = all 1s and fraction != 0
}
// Function is_subnormal: gives true for elements that are denormal (subnormal)
// false for finite numbers, zero, NAN and INF
static inline Vec4fb is_subnormal(Vec4f const & a) {
Vec4i t1 = _mm_castps_si128(a); // reinterpret as 32-bit integer
Vec4i t2 = t1 << 1; // shift out sign bit
Vec4i t3 = 0xFF000000; // exponent mask
Vec4i t4 = t2 & t3; // exponent
Vec4i t5 = _mm_andnot_si128(t3,t2);// fraction
return Vec4ib((t4 == 0) & (t5 != 0));// exponent = 0 and fraction != 0
}
// Function is_zero_or_subnormal: gives true for elements that are zero or subnormal (denormal)
// false for finite numbers, NAN and INF
static inline Vec4fb is_zero_or_subnormal(Vec4f const & a) {
Vec4i t = _mm_castps_si128(a); // reinterpret as 32-bit integer
t &= 0x7F800000; // isolate exponent
return t == 0; // exponent = 0
}
// Function infinite4f: returns a vector where all elements are +INF
static inline Vec4f infinite4f() {
return _mm_castsi128_ps(_mm_set1_epi32(0x7F800000));
}
// Function nan4f: returns a vector where all elements are NAN (quiet)
static inline Vec4f nan4f(int n = 0x10) {
return _mm_castsi128_ps(_mm_set1_epi32(0x7FC00000 + n));
}
/*****************************************************************************
*
* Vector Vec4f permute and blend functions
*
******************************************************************************
*
* The permute function can reorder the elements of a vector and optionally
* set some elements to zero.
*
* The indexes are inserted as template parameters in <>. These indexes must be
* constants. Each template parameter is an index to the element you want to
* select. A negative index will generate zero.
*
* Example:
* Vec4f a(10.f,11.f,12.f,13.f); // a is (10,11,12,13)
* Vec4f b, c;
* b = permute4f<0,0,2,2>(a); // b is (10,10,12,12)
* c = permute4f<3,2,-1,-1>(a); // c is (13,12, 0, 0)
*
*
* The blend function can mix elements from two different vectors and
* optionally set some elements to zero.
*
* The indexes are inserted as template parameters in <>. These indexes must be
* constants. Each template parameter is an index to the element you want to
* select, where indexes 0 - 3 indicate an element from the first source
* vector and indexes 4 - 7 indicate an element from the second source vector.
* A negative index will generate zero.
*
*
* Example:
* Vec4f a(10.f,11.f,12.f,13.f); // a is (10, 11, 12, 13)
* Vec4f b(20.f,21.f,22.f,23.f); // b is (20, 21, 22, 23)
* Vec4f c;
* c = blend4f<1,4,-1,7> (a,b); // c is (11, 20, 0, 23)
*
* Don't worry about the complicated code for these functions. Most of the
* code is resolved at compile time to generate only a few instructions.
*****************************************************************************/
// permute vector Vec4f
template <int i0, int i1, int i2, int i3>
static inline Vec4f permute4f(Vec4f const & a) {
// is shuffling needed
const bool do_shuffle = (i0 > 0) || (i1 != 1 && i1 >= 0) || (i2 != 2 && i2 >= 0) || (i3 != 3 && i3 >= 0);
// is zeroing needed
const bool do_zero = (i0 | i1 | i2 | i3) < 0 && ((i0 | i1 | i2 | i3) & 0x80);
if (!do_shuffle && !do_zero) {
return a; // trivial case: do nothing
}
if (do_zero && !do_shuffle) { // zeroing, not shuffling
if ((i0 & i1 & i2 & i3) < 0) return _mm_setzero_ps(); // zero everything
// zero some elements
__m128i mask1 = constant4i< -int(i0>=0), -int(i1>=0), -int(i2>=0), -int(i3>=0) >();
return _mm_and_ps(a,_mm_castsi128_ps(mask1)); // zero with AND mask
}
if (do_shuffle && !do_zero) { // shuffling, not zeroing
return _mm_shuffle_ps(a, a, (i0&3) | (i1&3)<<2 | (i2&3)<<4 | (i3&3)<<6);
}
// both shuffle and zero
if ((i0 & i1) < 0 && (i2 | i3) >= 0) { // zero low half, shuffle high half
return _mm_shuffle_ps(_mm_setzero_ps(), a, (i2&3)<<4 | (i3&3)<<6);
}
if ((i0 | i1) >= 0 && (i2 & i3) < 0) { // shuffle low half, zero high half
return _mm_shuffle_ps(a, _mm_setzero_ps(), (i0&3) | (i1&3)<<2);
}
#if INSTRSET >= 4 // SSSE3
// With SSSE3 we can do both with the PSHUFB instruction
const int j0 = (i0 & 3) << 2;
const int j1 = (i1 & 3) << 2;
const int j2 = (i2 & 3) << 2;
const int j3 = (i3 & 3) << 2;
__m128i mask2 = constant4i <
i0 < 0 ? -1 : j0 | (j0+1)<<8 | (j0+2)<<16 | (j0+3) << 24,
i1 < 0 ? -1 : j1 | (j1+1)<<8 | (j1+2)<<16 | (j1+3) << 24,
i2 < 0 ? -1 : j2 | (j2+1)<<8 | (j2+2)<<16 | (j2+3) << 24,
i3 < 0 ? -1 : j3 | (j3+1)<<8 | (j3+2)<<16 | (j3+3) << 24 > ();
return _mm_castsi128_ps(_mm_shuffle_epi8(_mm_castps_si128(a),mask2));
#else
__m128 t1 = _mm_shuffle_ps(a, a, (i0&3) | (i1&3)<<2 | (i2&3)<<4 | (i3&3)<<6); // shuffle
__m128i mask3 = constant4i< -int(i0>=0), -int(i1>=0), -int(i2>=0), -int(i3>=0) >();
return _mm_and_ps(t1,_mm_castsi128_ps(mask3)); // zero with AND mask
#endif
}
// blend vectors Vec4f
template <int i0, int i1, int i2, int i3>
static inline Vec4f blend4f(Vec4f const & a, Vec4f const & b) {
// Combine all the indexes into a single bitfield, with 8 bits for each
const int m1 = (i0&7) | (i1&7)<<8 | (i2&7)<<16 | (i3&7)<<24;
// Mask to zero out negative indexes
const int m2 = (i0<0?0:0xFF) | (i1<0?0:0xFF)<<8 | (i2<0?0:0xFF)<<16 | (i3<0?0:0xFF)<<24;
if ((m1 & 0x04040404 & m2) == 0) {
// no elements from b
return permute4f<i0,i1,i2,i3>(a);
}
if (((m1^0x04040404) & 0x04040404 & m2) == 0) {
// no elements from a
return permute4f<i0&~4, i1&~4, i2&~4, i3&~4>(b);
}
if (((m1 & ~0x04040404) ^ 0x03020100) == 0 && m2 == -1) {
// selecting without shuffling or zeroing
__m128i sel = constant4i <i0 & 4 ? 0 : -1, i1 & 4 ? 0 : -1, i2 & 4 ? 0 : -1, i3 & 4 ? 0 : -1> ();
return selectf(_mm_castsi128_ps(sel), a, b);
}
#ifdef __XOP__ // Use AMD XOP instruction PPERM
__m128i maska = constant4i <
i0 < 0 ? 0x80808080 : (i0*4 & 31) + (((i0*4 & 31) + 1) << 8) + (((i0*4 & 31) + 2) << 16) + (((i0*4 & 31) + 3) << 24),
i1 < 0 ? 0x80808080 : (i1*4 & 31) + (((i1*4 & 31) + 1) << 8) + (((i1*4 & 31) + 2) << 16) + (((i1*4 & 31) + 3) << 24),
i2 < 0 ? 0x80808080 : (i2*4 & 31) + (((i2*4 & 31) + 1) << 8) + (((i2*4 & 31) + 2) << 16) + (((i2*4 & 31) + 3) << 24),
i3 < 0 ? 0x80808080 : (i3*4 & 31) + (((i3*4 & 31) + 1) << 8) + (((i3*4 & 31) + 2) << 16) + (((i3*4 & 31) + 3) << 24) > ();
return _mm_castsi128_ps(_mm_perm_epi8(_mm_castps_si128(a), _mm_castps_si128(b), maska));
#else
if ((((m1 & ~0x04040404) ^ 0x03020100) & m2) == 0) {
// selecting and zeroing, not shuffling
__m128i sel1 = constant4i <i0 & 4 ? 0 : -1, i1 & 4 ? 0 : -1, i2 & 4 ? 0 : -1, i3 & 4 ? 0 : -1> ();
__m128i mask1 = constant4i< -int(i0>=0), -int(i1>=0), -int(i2>=0), -int(i3>=0) >();
__m128 t1 = selectf(_mm_castsi128_ps(sel1), a, b); // select
return _mm_and_ps(t1, _mm_castsi128_ps(mask1)); // zero
}
// special cases unpckhps, unpcklps, shufps
Vec4f t;
if (((m1 ^ 0x05010400) & m2) == 0) {
t = _mm_unpacklo_ps(a, b);
goto DOZERO;
}
if (((m1 ^ 0x01050004) & m2) == 0) {
t = _mm_unpacklo_ps(b, a);
goto DOZERO;
}
if (((m1 ^ 0x07030602) & m2) == 0) {
t = _mm_unpackhi_ps(a, b);
goto DOZERO;
}
if (((m1 ^ 0x03070206) & m2) == 0) {
t = _mm_unpackhi_ps(b, a);
goto DOZERO;
}
// first two elements from a, last two from b
if (((m1^0x04040000) & 0x04040404 & m2) == 0) {
t = _mm_shuffle_ps(a, b, (i0&3) + ((i1&3)<<2) + ((i2&3)<<4) + ((i3&3)<<6));
goto DOZERO;
}
// first two elements from b, last two from a
if (((m1^0x00000404) & 0x04040404 & m2) == 0) {
t = _mm_shuffle_ps(b, a, (i0&3) + ((i1&3)<<2) + ((i2&3)<<4) + ((i3&3)<<6));
goto DOZERO;
}
{ // general case. combine two permutes
__m128 a1 = permute4f <
(uint32_t)i0 < 4 ? i0 : -1,
(uint32_t)i1 < 4 ? i1 : -1,
(uint32_t)i2 < 4 ? i2 : -1,
(uint32_t)i3 < 4 ? i3 : -1 > (a);
__m128 b1 = permute4f <
(uint32_t)(i0^4) < 4 ? (i0^4) : -1,
(uint32_t)(i1^4) < 4 ? (i1^4) : -1,
(uint32_t)(i2^4) < 4 ? (i2^4) : -1,
(uint32_t)(i3^4) < 4 ? (i3^4) : -1 > (b);
return _mm_or_ps(a1,b1);
}
DOZERO:
if ((i0|i1|i2|i3) & 0x80) {
// zero some elements
__m128i mask1 = constant4i< -int(i0>=0), -int(i1>=0), -int(i2>=0), -int(i3>=0) >();
t = _mm_and_ps(t,_mm_castsi128_ps(mask1)); // zero with AND mask
}
return t;
#endif // __XOP__
}
// change signs on vectors Vec4f
// Each index i0 - i3 is 1 for changing sign on the corresponding element, 0 for no change
template <int i0, int i1, int i2, int i3>
static inline Vec4f change_sign(Vec4f const & a) {
if ((i0 | i1 | i2 | i3) == 0) return a;
__m128i mask = constant4i<i0 ? 0x80000000 : 0, i1 ? 0x80000000 : 0, i2 ? 0x80000000 : 0, i3 ? 0x80000000 : 0>();
return _mm_xor_ps(a, _mm_castsi128_ps(mask)); // flip sign bits
}
/*****************************************************************************
*
* Vec2d: Vector of 2 double precision floating point values
*
*****************************************************************************/
class Vec2d {
protected:
__m128d xmm; // double vector
public:
// Default constructor:
Vec2d() {
}
// Constructor to broadcast the same value into all elements:
Vec2d(double d) {
xmm = _mm_set1_pd(d);
}
// Constructor to build from all elements:
Vec2d(double d0, double d1) {
xmm = _mm_setr_pd(d0, d1);
}
// Constructor to convert from type __m128d used in intrinsics:
Vec2d(__m128d const & x) {
xmm = x;
}
// Assignment operator to convert from type __m128d used in intrinsics:
Vec2d & operator = (__m128d const & x) {
xmm = x;
return *this;
}
// Type cast operator to convert to __m128d used in intrinsics
operator __m128d() const {
return xmm;
}
// Member function to load from array (unaligned)
Vec2d & load(double const * p) {
xmm = _mm_loadu_pd(p);
return *this;
}
// Member function to load from array, aligned by 16
// "load_a" is faster than "load" on older Intel processors (Pentium 4, Pentium M, Core 1,
// Merom, Wolfdale) and Atom, but not on other processors from Intel, AMD or VIA.
// You may use load_a instead of load if you are certain that p points to an address
// divisible by 16.
Vec2d const & load_a(double const * p) {
xmm = _mm_load_pd(p);
return *this;
}
// Member function to store into array (unaligned)
void store(double * p) const {
_mm_storeu_pd(p, xmm);
}
// Member function to store into array, aligned by 16
// "store_a" is faster than "store" on older Intel processors (Pentium 4, Pentium M, Core 1,
// Merom, Wolfdale) and Atom, but not on other processors from Intel, AMD or VIA.
// You may use store_a instead of store if you are certain that p points to an address
// divisible by 16.
void store_a(double * p) const {
_mm_store_pd(p, xmm);
}
// Partial load. Load n elements and set the rest to 0
Vec2d & load_partial(int n, double const * p) {
if (n == 1) {
xmm = _mm_load_sd(p);
}
else if (n == 2) {
load(p);
}
else {
xmm = _mm_setzero_pd();
}
return *this;
}
// Partial store. Store n elements
void store_partial(int n, double * p) const {
if (n == 1) {
_mm_store_sd(p, xmm);
}
else if (n == 2) {
store(p);
}
}
// cut off vector to n elements. The last 4-n elements are set to zero
Vec2d & cutoff(int n) {
xmm = _mm_castps_pd(Vec4f(_mm_castpd_ps(xmm)).cutoff(n*2));
return *this;
}
// Member function to change a single element in vector
// Note: This function is inefficient. Use load function if changing more than one element
Vec2d const & insert(uint32_t index, double value) {
__m128d v2 = _mm_set_sd(value);
if (index == 0) {
xmm = _mm_shuffle_pd(v2,xmm,2);
}
else {
xmm = _mm_shuffle_pd(xmm,v2,0);
}
return *this;
};
// Member function extract a single element from vector
double extract(uint32_t index) const {
double x[2];
store(x);
return x[index & 1];
}
// Extract a single element. Use store function if extracting more than one element.
// Operator [] can only read an element, not write.
double operator [] (uint32_t index) const {
return extract(index);
}
static int size() {
return 2;
}
};
/*****************************************************************************
*
* Operators for Vec2d
*
*****************************************************************************/
// vector operator + : add element by element
static inline Vec2d operator + (Vec2d const & a, Vec2d const & b) {
return _mm_add_pd(a, b);
}
// vector operator + : add vector and scalar
static inline Vec2d operator + (Vec2d const & a, double b) {
return a + Vec2d(b);
}
static inline Vec2d operator + (double a, Vec2d const & b) {
return Vec2d(a) + b;
}
// vector operator += : add
static inline Vec2d & operator += (Vec2d & a, Vec2d const & b) {
a = a + b;
return a;
}
// postfix operator ++
static inline Vec2d operator ++ (Vec2d & a, int) {
Vec2d a0 = a;
a = a + 1.0;
return a0;
}
// prefix operator ++
static inline Vec2d & operator ++ (Vec2d & a) {
a = a + 1.0;
return a;
}
// vector operator - : subtract element by element
static inline Vec2d operator - (Vec2d const & a, Vec2d const & b) {
return _mm_sub_pd(a, b);
}
// vector operator - : subtract vector and scalar
static inline Vec2d operator - (Vec2d const & a, double b) {
return a - Vec2d(b);
}
static inline Vec2d operator - (double a, Vec2d const & b) {
return Vec2d(a) - b;
}
// vector operator - : unary minus
// Change sign bit, even for 0, INF and NAN
static inline Vec2d operator - (Vec2d const & a) {
return _mm_xor_pd(a, _mm_castsi128_pd(_mm_setr_epi32(0,0x80000000,0,0x80000000)));
}
// vector operator -= : subtract
static inline Vec2d & operator -= (Vec2d & a, Vec2d const & b) {
a = a - b;
return a;
}
// postfix operator --
static inline Vec2d operator -- (Vec2d & a, int) {
Vec2d a0 = a;
a = a - 1.0;
return a0;
}
// prefix operator --
static inline Vec2d & operator -- (Vec2d & a) {
a = a - 1.0;
return a;
}
// vector operator * : multiply element by element
static inline Vec2d operator * (Vec2d const & a, Vec2d const & b) {
return _mm_mul_pd(a, b);
}
// vector operator * : multiply vector and scalar
static inline Vec2d operator * (Vec2d const & a, double b) {
return a * Vec2d(b);
}
static inline Vec2d operator * (double a, Vec2d const & b) {
return Vec2d(a) * b;
}
// vector operator *= : multiply
static inline Vec2d & operator *= (Vec2d & a, Vec2d const & b) {
a = a * b;
return a;
}
// vector operator / : divide all elements by same integer
static inline Vec2d operator / (Vec2d const & a, Vec2d const & b) {
return _mm_div_pd(a, b);
}
// vector operator / : divide vector and scalar
static inline Vec2d operator / (Vec2d const & a, double b) {
return a / Vec2d(b);
}
static inline Vec2d operator / (double a, Vec2d const & b) {
return Vec2d(a) / b;
}
// vector operator /= : divide
static inline Vec2d & operator /= (Vec2d & a, Vec2d const & b) {
a = a / b;
return a;
}
// vector operator == : returns true for elements for which a == b
static inline Vec2db operator == (Vec2d const & a, Vec2d const & b) {
return _mm_cmpeq_pd(a, b);
}
// vector operator != : returns true for elements for which a != b
static inline Vec2db operator != (Vec2d const & a, Vec2d const & b) {
return _mm_cmpneq_pd(a, b);
}
// vector operator < : returns true for elements for which a < b
static inline Vec2db operator < (Vec2d const & a, Vec2d const & b) {
return _mm_cmplt_pd(a, b);
}
// vector operator <= : returns true for elements for which a <= b
static inline Vec2db operator <= (Vec2d const & a, Vec2d const & b) {
return _mm_cmple_pd(a, b);
}
// vector operator > : returns true for elements for which a > b
static inline Vec2db operator > (Vec2d const & a, Vec2d const & b) {
return b < a;
}
// vector operator >= : returns true for elements for which a >= b
static inline Vec2db operator >= (Vec2d const & a, Vec2d const & b) {
return b <= a;
}
// Bitwise logical operators
// vector operator & : bitwise and
static inline Vec2d operator & (Vec2d const & a, Vec2d const & b) {
return _mm_and_pd(a, b);
}
// vector operator &= : bitwise and
static inline Vec2d & operator &= (Vec2d & a, Vec2d const & b) {
a = a & b;
return a;
}
// vector operator & : bitwise and of Vec2d and Vec2db
static inline Vec2d operator & (Vec2d const & a, Vec2db const & b) {
return _mm_and_pd(a, b);
}
static inline Vec2d operator & (Vec2db const & a, Vec2d const & b) {
return _mm_and_pd(a, b);
}
// vector operator | : bitwise or
static inline Vec2d operator | (Vec2d const & a, Vec2d const & b) {
return _mm_or_pd(a, b);
}
// vector operator |= : bitwise or
static inline Vec2d & operator |= (Vec2d & a, Vec2d const & b) {
a = a | b;
return a;
}
// vector operator ^ : bitwise xor
static inline Vec2d operator ^ (Vec2d const & a, Vec2d const & b) {
return _mm_xor_pd(a, b);
}
// vector operator ^= : bitwise xor
static inline Vec2d & operator ^= (Vec2d & a, Vec2d const & b) {
a = a ^ b;
return a;
}
// vector operator ! : logical not. Returns Boolean vector
static inline Vec2db operator ! (Vec2d const & a) {
return a == Vec2d(0.0);
}
/*****************************************************************************
*
* Functions for Vec2d
*
*****************************************************************************/
// Select between two operands. Corresponds to this pseudocode:
// for (int i = 0; i < 2; i++) result[i] = s[i] ? a[i] : b[i];
// Each byte in s must be either 0 (false) or 0xFFFFFFFFFFFFFFFF (true).
// No other values are allowed.
static inline Vec2d select (Vec2db const & s, Vec2d const & a, Vec2d const & b) {
return selectd(s,a,b);
}
// Conditional add: For all vector elements i: result[i] = f[i] ? (a[i] + b[i]) : a[i]
static inline Vec2d if_add (Vec2db const & f, Vec2d const & a, Vec2d const & b) {
return a + (Vec2d(f) & b);
}
// Conditional multiply: For all vector elements i: result[i] = f[i] ? (a[i] * b[i]) : a[i]
static inline Vec2d if_mul (Vec2db const & f, Vec2d const & a, Vec2d const & b) {
return a * select(f, b, 1.);
}
// General arithmetic functions, etc.
// Horizontal add: Calculates the sum of all vector elements.
static inline double horizontal_add (Vec2d const & a) {
#if INSTRSET >= 3 // SSE3
__m128d t1 = _mm_hadd_pd(a,a);
return _mm_cvtsd_f64(t1);
#else
__m128 t0 = _mm_castpd_ps(a);
__m128d t1 = _mm_castps_pd(_mm_movehl_ps(t0,t0));
__m128d t2 = _mm_add_sd(a,t1);
return _mm_cvtsd_f64(t2);
#endif
}
// function max: a > b ? a : b
static inline Vec2d max(Vec2d const & a, Vec2d const & b) {
return _mm_max_pd(a,b);
}
// function min: a < b ? a : b
static inline Vec2d min(Vec2d const & a, Vec2d const & b) {
return _mm_min_pd(a,b);
}
// function abs: absolute value
// Removes sign bit, even for -0.0f, -INF and -NAN
static inline Vec2d abs(Vec2d const & a) {
__m128d mask = _mm_castsi128_pd(_mm_setr_epi32(-1,0x7FFFFFFF,-1,0x7FFFFFFF));
return _mm_and_pd(a,mask);
}
// function sqrt: square root
static inline Vec2d sqrt(Vec2d const & a) {
return _mm_sqrt_pd(a);
}
// function square: a * a
static inline Vec2d square(Vec2d const & a) {
return a * a;
}
// pow(Vec2d, int):
// The purpose of this template is to prevent implicit conversion of a float
// exponent to int when calling pow(vector, float) and vectormath_exp.h is
// not included
template <typename TT> static Vec2d pow(Vec2d const & a, TT n);
// Raise floating point numbers to integer power n
template <>
inline Vec2d pow<int>(Vec2d const & x0, int n) {
return pow_template_i<Vec2d>(x0, n);
}
// allow conversion from unsigned int
template <>
inline Vec2d pow<uint32_t>(Vec2d const & x0, uint32_t n) {
return pow_template_i<Vec2d>(x0, (int)n);
}
// Raise floating point numbers to integer power n, where n is a compile-time constant
template <int n>
static inline Vec2d pow_n(Vec2d const & a) {
if (n < 0) return Vec2d(1.0) / pow_n<-n>(a);
if (n == 0) return Vec2d(1.0);
if (n >= 256) return pow(a, n);
Vec2d x = a; // a^(2^i)
Vec2d y; // accumulator
const int lowest = n - (n & (n-1));// lowest set bit in n
if (n & 1) y = x;
if (n < 2) return y;
x = x*x; // x^2
if (n & 2) {
if (lowest == 2) y = x; else y *= x;
}
if (n < 4) return y;
x = x*x; // x^4
if (n & 4) {
if (lowest == 4) y = x; else y *= x;
}
if (n < 8) return y;
x = x*x; // x^8
if (n & 8) {
if (lowest == 8) y = x; else y *= x;
}
if (n < 16) return y;
x = x*x; // x^16
if (n & 16) {
if (lowest == 16) y = x; else y *= x;
}
if (n < 32) return y;
x = x*x; // x^32
if (n & 32) {
if (lowest == 32) y = x; else y *= x;
}
if (n < 64) return y;
x = x*x; // x^64
if (n & 64) {
if (lowest == 64) y = x; else y *= x;
}
if (n < 128) return y;
x = x*x; // x^128
if (n & 128) {
if (lowest == 128) y = x; else y *= x;
}
return y;
}
template <int n>
static inline Vec2d pow(Vec2d const & a, Const_int_t<n>) {
return pow_n<n>(a);
}
// avoid unsafe optimization in function round
#if defined(__GNUC__) && !defined(__INTEL_COMPILER) && !defined(__clang__) && INSTRSET < 5
static inline Vec4f round(Vec4f const & a) __attribute__ ((optimize("-fno-unsafe-math-optimizations")));
#elif defined (FLOAT_CONTROL_PRECISE_FOR_ROUND)
#pragma float_control(push)
#pragma float_control(precise,on)
#endif
// function round: round to nearest integer (even). (result as double vector)
static inline Vec2d round(Vec2d const & a) {
#if INSTRSET >= 5 // SSE4.1 supported
return _mm_round_pd(a, 0);
#else // SSE2. Use magic number method
// Note: assume MXCSR control register is set to rounding
// (don't use conversion to int, it will limit the value to +/- 2^31)
Vec2d signmask = _mm_castsi128_pd(constant4i<0,(int)0x80000000,0,(int)0x80000000>()); // -0.0
Vec2d magic = _mm_castsi128_pd(constant4i<0,0x43300000,0,0x43300000>()); // magic number = 2^52
Vec2d sign = _mm_and_pd(a, signmask); // signbit of a
Vec2d signedmagic = _mm_or_pd(magic, sign); // magic number with sign of a
return a + signedmagic - signedmagic; // round by adding magic number
#endif
}
#if defined (FLOAT_CONTROL_PRECISE_FOR_ROUND)
#pragma float_control(pop)
#endif
// function truncate: round towards zero. (result as double vector)
static inline Vec2d truncate(Vec2d const & a) {
// (note: may fail on MS Visual Studio 2008, works in later versions)
#if INSTRSET >= 5 // SSE4.1 supported
return _mm_round_pd(a, 3);
#else // SSE2. Use magic number method (conversion to int would limit the value to 2^31)
uint32_t t1 = _mm_getcsr(); // MXCSR
uint32_t t2 = t1 | (3 << 13); // bit 13-14 = 11
_mm_setcsr(t2); // change MXCSR
Vec2d r = round(a); // use magic number method
_mm_setcsr(t1); // restore MXCSR
return r;
#endif
}
// function floor: round towards minus infinity. (result as double vector)
// (note: may fail on MS Visual Studio 2008, works in later versions)
static inline Vec2d floor(Vec2d const & a) {
#if INSTRSET >= 5 // SSE4.1 supported
return _mm_round_pd(a, 1);
#else // SSE2. Use magic number method (conversion to int would limit the value to 2^31)
uint32_t t1 = _mm_getcsr(); // MXCSR
uint32_t t2 = t1 | (1 << 13); // bit 13-14 = 01
_mm_setcsr(t2); // change MXCSR
Vec2d r = round(a); // use magic number method
_mm_setcsr(t1); // restore MXCSR
return r;
#endif
}
// function ceil: round towards plus infinity. (result as double vector)
static inline Vec2d ceil(Vec2d const & a) {
#if INSTRSET >= 5 // SSE4.1 supported
return _mm_round_pd(a, 2);
#else // SSE2. Use magic number method (conversion to int would limit the value to 2^31)
uint32_t t1 = _mm_getcsr(); // MXCSR
uint32_t t2 = t1 | (2 << 13); // bit 13-14 = 10
_mm_setcsr(t2); // change MXCSR
Vec2d r = round(a); // use magic number method
_mm_setcsr(t1); // restore MXCSR
return r;
#endif
}
// function truncate_to_int: round towards zero.
static inline Vec4i truncate_to_int(Vec2d const & a, Vec2d const & b) {
Vec4i t1 = _mm_cvttpd_epi32(a);
Vec4i t2 = _mm_cvttpd_epi32(b);
return blend4i<0,1,4,5> (t1, t2);
}
// function truncate_to_int64: round towards zero. (inefficient)
static inline Vec2q truncate_to_int64(Vec2d const & a) {
double aa[2];
a.store(aa);
return Vec2q(int64_t(aa[0]), int64_t(aa[1]));
}
// function truncate_to_int64_limited: round towards zero. (inefficient)
// result as 64-bit integer vector, but with limited range
static inline Vec2q truncate_to_int64_limited(Vec2d const & a) {
// Note: assume MXCSR control register is set to rounding
Vec4i t1 = _mm_cvttpd_epi32(a);
return extend_low(t1);
}
// function round_to_int: round to nearest integer (even).
// result as 32-bit integer vector
static inline Vec4i round_to_int(Vec2d const & a, Vec2d const & b) {
// Note: assume MXCSR control register is set to rounding
Vec4i t1 = _mm_cvtpd_epi32(a);
Vec4i t2 = _mm_cvtpd_epi32(b);
return blend4i<0,1,4,5> (t1, t2);
}
// function round_to_int: round to nearest integer (even).
// result as 32-bit integer vector. Upper two values of result are 0
static inline Vec4i round_to_int(Vec2d const & a) {
Vec4i t1 = _mm_cvtpd_epi32(a);
return t1;
}
// function round_to_int64: round to nearest or even. (inefficient)
static inline Vec2q round_to_int64(Vec2d const & a) {
return truncate_to_int64(round(a));
}
// function round_to_int: round to nearest integer (even)
// result as 64-bit integer vector, but with limited range
static inline Vec2q round_to_int64_limited(Vec2d const & a) {
// Note: assume MXCSR control register is set to rounding
Vec4i t1 = _mm_cvtpd_epi32(a);
return extend_low(t1);
}
// function to_double: convert integer vector elements to double vector (inefficient)
static inline Vec2d to_double(Vec2q const & a) {
int64_t aa[2];
a.store(aa);
return Vec2d(double(aa[0]), double(aa[1]));
}
// function to_double_limited: convert integer vector elements to double vector
// limited to abs(x) < 2^31
static inline Vec2d to_double_limited(Vec2q const & x) {
Vec4i compressed = permute4i<0,2,-256,-256>(Vec4i(x));
return _mm_cvtepi32_pd(compressed);
}
// function to_double_low: convert integer vector elements [0] and [1] to double vector
static inline Vec2d to_double_low(Vec4i const & a) {
return _mm_cvtepi32_pd(a);
}
// function to_double_high: convert integer vector elements [2] and [3] to double vector
static inline Vec2d to_double_high(Vec4i const & a) {
return to_double_low(_mm_srli_si128(a,8));
}
// function compress: convert two Vec2d to one Vec4f
static inline Vec4f compress (Vec2d const & low, Vec2d const & high) {
Vec4f t1 = _mm_cvtpd_ps(low);
Vec4f t2 = _mm_cvtpd_ps(high);
return blend4f<0,1,4,5> (t1, t2);
}
// Function extend_low : convert Vec4f vector elements [0] and [1] to Vec2d
static inline Vec2d extend_low (Vec4f const & a) {
return _mm_cvtps_pd(a);
}
// Function extend_high : convert Vec4f vector elements [2] and [3] to Vec2d
static inline Vec2d extend_high (Vec4f const & a) {
return _mm_cvtps_pd(_mm_movehl_ps(a,a));
}
// Fused multiply and add functions
// Multiply and add
static inline Vec2d mul_add(Vec2d const & a, Vec2d const & b, Vec2d const & c) {
#ifdef __FMA__
return _mm_fmadd_pd(a, b, c);
#elif defined (__FMA4__)
return _mm_macc_pd(a, b, c);
#else
return a * b + c;
#endif
}
// Multiply and subtract
static inline Vec2d mul_sub(Vec2d const & a, Vec2d const & b, Vec2d const & c) {
#ifdef __FMA__
return _mm_fmsub_pd(a, b, c);
#elif defined (__FMA4__)
return _mm_msub_pd(a, b, c);
#else
return a * b - c;
#endif
}
// Multiply and inverse subtract
static inline Vec2d nmul_add(Vec2d const & a, Vec2d const & b, Vec2d const & c) {
#ifdef __FMA__
return _mm_fnmadd_pd(a, b, c);
#elif defined (__FMA4__)
return _mm_nmacc_pd(a, b, c);
#else
return c - a * b;
#endif
}
// Multiply and subtract with extra precision on the intermediate calculations,
// even if FMA instructions not supported, using Veltkamp-Dekker split
static inline Vec2d mul_sub_x(Vec2d const & a, Vec2d const & b, Vec2d const & c) {
#ifdef __FMA__
return _mm_fmsub_pd(a, b, c);
#elif defined (__FMA4__)
return _mm_msub_pd(a, b, c);
#else
// calculate a * b - c with extra precision
Vec2q upper_mask = -(1LL << 27); // mask to remove lower 27 bits
Vec2d a_high = a & Vec2d(_mm_castsi128_pd(upper_mask));// split into high and low parts
Vec2d b_high = b & Vec2d(_mm_castsi128_pd(upper_mask));
Vec2d a_low = a - a_high;
Vec2d b_low = b - b_high;
Vec2d r1 = a_high * b_high; // this product is exact
Vec2d r2 = r1 - c; // subtract c from high product
Vec2d r3 = r2 + (a_high * b_low + b_high * a_low) + a_low * b_low; // add rest of product
return r3; // + ((r2 - r1) + c);
#endif
}
// Math functions using fast bit manipulation
// Extract the exponent as an integer
// exponent(a) = floor(log2(abs(a)));
// exponent(1.0) = 0, exponent(0.0) = -1023, exponent(INF) = +1024, exponent(NAN) = +1024
static inline Vec2q exponent(Vec2d const & a) {
Vec2uq t1 = _mm_castpd_si128(a); // reinterpret as 64-bit integer
Vec2uq t2 = t1 << 1; // shift out sign bit
Vec2uq t3 = t2 >> 53; // shift down logical to position 0
Vec2q t4 = Vec2q(t3) - 0x3FF; // subtract bias from exponent
return t4;
}
// Extract the fraction part of a floating point number
// a = 2^exponent(a) * fraction(a), except for a = 0
// fraction(1.0) = 1.0, fraction(5.0) = 1.25
static inline Vec2d fraction(Vec2d const & a) {
Vec2uq t1 = _mm_castpd_si128(a); // reinterpret as 64-bit integer
Vec2uq t2 = Vec2uq((t1 & 0x000FFFFFFFFFFFFFll) | 0x3FF0000000000000ll); // set exponent to 0 + bias
return _mm_castsi128_pd(t2);
}
// Fast calculation of pow(2,n) with n integer
// n = 0 gives 1.0
// n >= 1024 gives +INF
// n <= -1023 gives 0.0
// This function will never produce denormals, and never raise exceptions
static inline Vec2d exp2(Vec2q const & n) {
Vec2q t1 = max(n, -0x3FF); // limit to allowed range
Vec2q t2 = min(t1, 0x400);
Vec2q t3 = t2 + 0x3FF; // add bias
Vec2q t4 = t3 << 52; // put exponent into position 52
return _mm_castsi128_pd(t4); // reinterpret as double
}
//static Vec2d exp2(Vec2d const & x); // defined in vectormath_exp.h
// Categorization functions
// Function sign_bit: gives true for elements that have the sign bit set
// even for -0.0, -INF and -NAN
// Note that sign_bit(Vec2d(-0.0)) gives true, while Vec2d(-0.0) < Vec2d(0.0) gives false
static inline Vec2db sign_bit(Vec2d const & a) {
Vec2q t1 = _mm_castpd_si128(a); // reinterpret as 64-bit integer
Vec2q t2 = t1 >> 63; // extend sign bit
return _mm_castsi128_pd(t2); // reinterpret as 64-bit Boolean
}
// Function sign_combine: changes the sign of a when b has the sign bit set
// same as select(sign_bit(b), -a, a)
static inline Vec2d sign_combine(Vec2d const & a, Vec2d const & b) {
Vec2d signmask = _mm_castsi128_pd(constant4i<0,(int)0x80000000,0,(int)0x80000000>()); // -0.0
return a ^ (b & signmask);
}
// Function is_finite: gives true for elements that are normal, denormal or zero,
// false for INF and NAN
static inline Vec2db is_finite(Vec2d const & a) {
Vec2q t1 = _mm_castpd_si128(a); // reinterpret as integer
Vec2q t2 = t1 << 1; // shift out sign bit
Vec2q t3 = 0xFFE0000000000000ll; // exponent mask
Vec2qb t4 = Vec2q(t2 & t3) != t3; // exponent field is not all 1s
return t4;
}
// Function is_inf: gives true for elements that are +INF or -INF
// false for finite numbers and NAN
static inline Vec2db is_inf(Vec2d const & a) {
Vec2q t1 = _mm_castpd_si128(a); // reinterpret as integer
Vec2q t2 = t1 << 1; // shift out sign bit
return t2 == 0xFFE0000000000000ll; // exponent is all 1s, fraction is 0
}
// Function is_nan: gives true for elements that are +NAN or -NAN
// false for finite numbers and +/-INF
static inline Vec2db is_nan(Vec2d const & a) {
Vec2q t1 = _mm_castpd_si128(a); // reinterpret as integer
Vec2q t2 = t1 << 1; // shift out sign bit
Vec2q t3 = 0xFFE0000000000000ll; // exponent mask
Vec2q t4 = t2 & t3; // exponent
Vec2q t5 = _mm_andnot_si128(t3,t2);// fraction
return Vec2qb((t4==t3) & (t5!=0)); // exponent = all 1s and fraction != 0
}
// Function is_subnormal: gives true for elements that are subnormal (denormal)
// false for finite numbers, zero, NAN and INF
static inline Vec2db is_subnormal(Vec2d const & a) {
Vec2q t1 = _mm_castpd_si128(a); // reinterpret as 32-bit integer
Vec2q t2 = t1 << 1; // shift out sign bit
Vec2q t3 = 0xFFE0000000000000ll; // exponent mask
Vec2q t4 = t2 & t3; // exponent
Vec2q t5 = _mm_andnot_si128(t3,t2);// fraction
return Vec2qb((t4==0) & (t5!=0)); // exponent = 0 and fraction != 0
}
// Function is_zero_or_subnormal: gives true for elements that are zero or subnormal (denormal)
// false for finite numbers, NAN and INF
static inline Vec2db is_zero_or_subnormal(Vec2d const & a) {
Vec2q t = _mm_castpd_si128(a); // reinterpret as 32-bit integer
t &= 0x7FF0000000000000ll; // isolate exponent
return t == 0; // exponent = 0
}
// Function infinite2d: returns a vector where all elements are +INF
static inline Vec2d infinite2d() {
return _mm_castsi128_pd(_mm_setr_epi32(0,0x7FF00000,0,0x7FF00000));
}
// Function nan2d: returns a vector where all elements are +NAN (quiet)
static inline Vec2d nan2d(int n = 0x10) {
return _mm_castsi128_pd(_mm_setr_epi32(n, 0x7FF80000, n, 0x7FF80000));
}
/*****************************************************************************
*
* Functions for reinterpretation between vector types
*
*****************************************************************************/
static inline __m128i reinterpret_i (__m128i const & x) {
return x;
}
static inline __m128i reinterpret_i (__m128 const & x) {
return _mm_castps_si128(x);
}
static inline __m128i reinterpret_i (__m128d const & x) {
return _mm_castpd_si128(x);
}
static inline __m128 reinterpret_f (__m128i const & x) {
return _mm_castsi128_ps(x);
}
static inline __m128 reinterpret_f (__m128 const & x) {
return x;
}
static inline __m128 reinterpret_f (__m128d const & x) {
return _mm_castpd_ps(x);
}
static inline __m128d reinterpret_d (__m128i const & x) {
return _mm_castsi128_pd(x);
}
static inline __m128d reinterpret_d (__m128 const & x) {
return _mm_castps_pd(x);
}
static inline __m128d reinterpret_d (__m128d const & x) {
return x;
}
/*****************************************************************************
*
* Vector permute and blend functions
*
******************************************************************************
*
* The permute function can reorder the elements of a vector and optionally
* set some elements to zero.
*
* The indexes are inserted as template parameters in <>. These indexes must be
* constants. Each template parameter is an index to the element you want to
* select. An index of -1 will generate zero. An index of -256 means don't care.
*
* Example:
* Vec2d a(10., 11.); // a is (10, 11)
* Vec2d b, c;
* b = permute2d<1,1>(a); // b is (11, 11)
* c = permute2d<-1,0>(a); // c is ( 0, 10)
*
*
* The blend function can mix elements from two different vectors and
* optionally set some elements to zero.
*
* The indexes are inserted as template parameters in <>. These indexes must be
* constants. Each template parameter is an index to the element you want to
* select, where indexes 0 - 1 indicate an element from the first source
* vector and indexes 2 - 3 indicate an element from the second source vector.
* An index of -1 will generate zero.
*
*
* Example:
* Vec2d a(10., 11.); // a is (10, 11)
* Vec2d b(20., 21.); // b is (20, 21)
* Vec2d c;
* c = blend2d<0,3> (a,b); // c is (10, 21)
*
* A lot of the code here is metaprogramming aiming to find the instructions
* that best fit the template parameters and instruction set. The metacode
* will be reduced out to leave only a few vector instructions in release
* mode with optimization on.
*****************************************************************************/
// permute vector Vec2d
template <int i0, int i1>
static inline Vec2d permute2d(Vec2d const & a) {
// is shuffling needed
const bool do_shuffle = (i0 > 0) || (i1 != 1 && i1 >= 0);
// is zeroing needed
const bool do_zero = ((i0 | i1) < 0 && (i0 | i1) & 0x80);
if (do_zero && !do_shuffle) { // zeroing, not shuffling
if ((i0 & i1) < 0) return _mm_setzero_pd(); // zero everything
// zero some elements
__m128i mask1 = constant4i< -int(i0>=0), -int(i0>=0), -int(i1>=0), -int(i1>=0) >();
return _mm_and_pd(a,_mm_castsi128_pd(mask1)); // zero with AND mask
}
else if (do_shuffle && !do_zero) { // shuffling, not zeroing
return _mm_shuffle_pd(a, a, (i0&1) | (i1&1)<<1);
}
else if (do_shuffle && do_zero) { // shuffling and zeroing
// both shuffle and zero
if (i0 < 0 && i1 >= 0) { // zero low half, shuffle high half
return _mm_shuffle_pd(_mm_setzero_pd(), a, (i1 & 1) << 1);
}
if (i0 >= 0 && i1 < 0) { // shuffle low half, zero high half
return _mm_shuffle_pd(a, _mm_setzero_pd(), i0 & 1);
}
}
return a; // trivial case: do nothing
}
// blend vectors Vec2d
template <int i0, int i1>
static inline Vec2d blend2d(Vec2d const & a, Vec2d const & b) {
// Combine all the indexes into a single bitfield, with 8 bits for each
const int m1 = (i0 & 3) | (i1 & 3) << 8;
// Mask to zero out negative indexes
const int m2 = (i0 < 0 ? 0 : 0xFF) | (i1 < 0 ? 0 : 0xFF) << 8;
if ((m1 & 0x0202 & m2) == 0) {
// no elements from b, only elements from a and possibly zero
return permute2d <i0, i1> (a);
}
if (((m1^0x0202) & 0x0202 & m2) == 0) {
// no elements from a, only elements from b and possibly zero
return permute2d <i0 & ~2, i1 & ~2> (b);
}
// selecting from both a and b without zeroing
if ((i0 & 2) == 0) { // first element from a, second element from b
return _mm_shuffle_pd(a, b, (i0 & 1) | (i1 & 1) << 1);
}
else { // first element from b, second element from a
return _mm_shuffle_pd(b, a, (i0 & 1) | (i1 & 1) << 1);
}
}
// change signs on vectors Vec4f
// Each index i0 - i1 is 1 for changing sign on the corresponding element, 0 for no change
template <int i0, int i1>
static inline Vec2d change_sign(Vec2d const & a) {
if ((i0 | i1) == 0) return a;
__m128i mask = constant4i<0, i0 ? 0x80000000 : 0, 0, i1 ? 0x80000000 : 0> ();
return _mm_xor_pd(a, _mm_castsi128_pd(mask)); // flip sign bits
}
/*****************************************************************************
*
* Vector lookup functions
*
******************************************************************************
*
* These functions use vector elements as indexes into a table.
* The table is given as one or more vectors or as an array.
*
* This can be used for several purposes:
* - table lookup
* - permute or blend with variable indexes
* - blend from more than two sources
* - gather non-contiguous data
*
* An index out of range may produce any value - the actual value produced is
* implementation dependent and may be different for different instruction
* sets. An index out of range does not produce an error message or exception.
*
* Example:
* Vec4i a(2,0,0,3); // index a is ( 2, 0, 0, 3)
* Vec4f b(1.0f,1.1f,1.2f,1.3f); // table b is (1.0, 1.1, 1.2, 1.3)
* Vec4f c;
* c = lookup4 (a,b); // result c is (1.2, 1.0, 1.0, 1.3)
*
*****************************************************************************/
static inline Vec4f lookup4(Vec4i const & index, Vec4f const & table) {
#if INSTRSET >= 7 // AVX
return _mm_permutevar_ps(table, index);
#else
int32_t ii[4];
float tt[6];
table.store(tt); (index & 3).store(ii);
__m128 r01 = _mm_loadh_pi(_mm_load_ss(&tt[ii[0]]), (const __m64 *)&tt[ii[1]]);
__m128 r23 = _mm_loadh_pi(_mm_load_ss(&tt[ii[2]]), (const __m64 *)&tt[ii[3]]);
return _mm_shuffle_ps(r01, r23, 0x88);
#endif
}
static inline Vec4f lookup8(Vec4i const & index, Vec4f const & table0, Vec4f const & table1) {
#if INSTRSET >= 8 // AVX2
__m256 tt = _mm256_insertf128_ps(_mm256_castps128_ps256(table0), table1, 1); // combine tables
#if defined (_MSC_VER) && _MSC_VER < 1700 && ! defined(__INTEL_COMPILER)
// bug in MS VS 11 beta: operands in wrong order
__m128 r = _mm256_castps256_ps128(_mm256_permutevar8x32_ps(_mm256_castsi256_ps(_mm256_castsi128_si256(index)), _mm256_castps_si256(tt)));
r = _mm_and_ps(r,r); // fix another bug in VS 11 beta (would store r as 256 bits aligned by 16)
#elif defined (GCC_VERSION) && GCC_VERSION <= 40700 && !defined(__INTEL_COMPILER) && !defined(__clang__)
// Gcc 4.7.0 has wrong parameter type and operands in wrong order
__m128 r = _mm256_castps256_ps128(_mm256_permutevar8x32_ps(_mm256_castsi256_ps(_mm256_castsi128_si256(index)), tt));
#else
// no bug version
__m128 r = _mm256_castps256_ps128(_mm256_permutevar8x32_ps(tt, _mm256_castsi128_si256(index)));
#endif
return r;
#elif INSTRSET >= 7 // AVX
__m128 r0 = _mm_permutevar_ps(table0, index);
__m128 r1 = _mm_permutevar_ps(table1, index);
__m128i i4 = _mm_slli_epi32(index, 29);
return _mm_blendv_ps(r0, r1, _mm_castsi128_ps(i4));
#elif INSTRSET >= 5 // SSE4.1
Vec4f r0 = lookup4(index, table0);
Vec4f r1 = lookup4(index, table1);
__m128i i4 = _mm_slli_epi32(index, 29);
return _mm_blendv_ps(r0, r1, _mm_castsi128_ps(i4));
#else // SSE2
Vec4f r0 = lookup4(index, table0);
Vec4f r1 = lookup4(index, table1);
__m128i i4 = _mm_srai_epi32(_mm_slli_epi32(index, 29), 31);
return selectf(_mm_castsi128_ps(i4), r1, r0);
#endif
}
template <int n>
static inline Vec4f lookup(Vec4i const & index, float const * table) {
if (n <= 0) return 0.0f;
if (n <= 4) return lookup4(index, Vec4f().load(table));
if (n <= 8) {
#if INSTRSET >= 8 // AVX2
__m256 tt = _mm256_loadu_ps(table);
#if defined (_MSC_VER) && _MSC_VER < 1700 && ! defined(__INTEL_COMPILER)
// bug in MS VS 11 beta: operands in wrong order
__m128 r = _mm256_castps256_ps128(_mm256_permutevar8x32_ps(_mm256_castsi256_ps(_mm256_castsi128_si256(index)), _mm256_castps_si256(tt)));
r = _mm_and_ps(r,r); // fix another bug in VS 11 beta (would store r as 256 bits aligned by 16)
#elif defined (GCC_VERSION) && GCC_VERSION <= 40700 && !defined(__INTEL_COMPILER) && !defined(__clang__)
// Gcc 4.7.0 has wrong parameter type and operands in wrong order
__m128 r = _mm256_castps256_ps128(_mm256_permutevar8x32_ps(_mm256_castsi256_ps(_mm256_castsi128_si256(index)), tt));
#else
// no bug version
__m128 r = _mm256_castps256_ps128(_mm256_permutevar8x32_ps(tt, _mm256_castsi128_si256(index)));
#endif
return r;
#else // not AVX2
return lookup8(index, Vec4f().load(table), Vec4f().load(table+4));
#endif // INSTRSET
}
// n > 8. Limit index
Vec4ui index1;
if ((n & (n-1)) == 0) {
// n is a power of 2, make index modulo n
index1 = Vec4ui(index) & (n-1);
}
else {
// n is not a power of 2, limit to n-1
index1 = min(Vec4ui(index), n-1);
}
#if INSTRSET >= 8 // AVX2
return _mm_i32gather_ps(table, index1, 4);
#else
uint32_t ii[4]; index1.store(ii);
return Vec4f(table[ii[0]], table[ii[1]], table[ii[2]], table[ii[3]]);
#endif
}
static inline Vec2d lookup2(Vec2q const & index, Vec2d const & table) {
#if INSTRSET >= 7 // AVX
return _mm_permutevar_pd(table, index + index);
#else
int32_t ii[4];
double tt[2];
table.store(tt); (index & 1).store(ii);
return Vec2d(tt[ii[0]], tt[ii[2]]);
#endif
}
static inline Vec2d lookup4(Vec2q const & index, Vec2d const & table0, Vec2d const & table1) {
#if INSTRSET >= 7 // AVX
Vec2q index2 = index + index; // index << 1
__m128d r0 = _mm_permutevar_pd(table0, index2);
__m128d r1 = _mm_permutevar_pd(table1, index2);
__m128i i4 = _mm_slli_epi64(index, 62);
return _mm_blendv_pd(r0, r1, _mm_castsi128_pd(i4));
#else
int32_t ii[4];
double tt[4];
table0.store(tt); table1.store(tt + 2);
(index & 3).store(ii);
return Vec2d(tt[ii[0]], tt[ii[2]]);
#endif
}
template <int n>
static inline Vec2d lookup(Vec2q const & index, double const * table) {
if (n <= 0) return 0.0;
if (n <= 2) return lookup2(index, Vec2d().load(table));
#if INSTRSET < 8 // not AVX2
if (n <= 4) return lookup4(index, Vec2d().load(table), Vec2d().load(table + 2));
#endif
// Limit index
Vec2uq index1;
if ((n & (n-1)) == 0) {
// n is a power of 2, make index modulo n
index1 = Vec2uq(index) & (n-1);
}
else {
// n is not a power of 2, limit to n-1
index1 = min(Vec2uq(index), n-1);
}
#if INSTRSET >= 8 // AVX2
return _mm_i64gather_pd(table, index1, 8);
#else
uint32_t ii[4]; index1.store(ii);
return Vec2d(table[ii[0]], table[ii[2]]);
#endif
}
/*****************************************************************************
*
* Gather functions with fixed indexes
*
*****************************************************************************/
// Load elements from array a with indices i0, i1, i2, i3
template <int i0, int i1, int i2, int i3>
static inline Vec4f gather4f(void const * a) {
return reinterpret_f(gather4i<i0, i1, i2, i3>(a));
}
// Load elements from array a with indices i0, i1
template <int i0, int i1>
static inline Vec2d gather2d(void const * a) {
return reinterpret_d(gather2q<i0, i1>(a));
}
/*****************************************************************************
*
* Horizontal scan functions
*
*****************************************************************************/
// Get index to the first element that is true. Return -1 if all are false
static inline int horizontal_find_first(Vec4fb const & x) {
return horizontal_find_first(Vec4ib(x));
}
static inline int horizontal_find_first(Vec2db const & x) {
return horizontal_find_first(Vec2qb(x));
}
// Count the number of elements that are true
static inline uint32_t horizontal_count(Vec4fb const & x) {
return horizontal_count(Vec4ib(x));
}
static inline uint32_t horizontal_count(Vec2db const & x) {
return horizontal_count(Vec2qb(x));
}
/*****************************************************************************
*
* Boolean <-> bitfield conversion functions
*
*****************************************************************************/
// to_bits: convert boolean vector to integer bitfield
static inline uint8_t to_bits(Vec4fb const & x) {
return to_bits(Vec4ib(x));
}
// to_Vec4fb: convert integer bitfield to boolean vector
static inline Vec4fb to_Vec4fb(uint8_t x) {
return Vec4fb(to_Vec4ib(x));
}
// to_bits: convert boolean vector to integer bitfield
static inline uint8_t to_bits(Vec2db const & x) {
return to_bits(Vec2qb(x));
}
// to_Vec2db: convert integer bitfield to boolean vector
static inline Vec2db to_Vec2db(uint8_t x) {
return Vec2db(to_Vec2qb(x));
}
#endif // VECTORF128_H