/* * Copyright (C) 2007-2017 Gabest * http://www.gabest.org * * This Program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2, or (at your option) * any later version. * * This Program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with GNU Make; see the file COPYING. If not, write to * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA USA. * http://www.gnu.org/copyleft/gpl.html * */ class alignas(16) GSVector4i { static const GSVector4i m_xff[17]; static const GSVector4i m_x0f[17]; constexpr static __m128i cxpr_setr_epi32(int x, int y, int z, int w) { #ifdef __GNUC__ return (__m128i)(__v4si{x, y, z, w}); #else __m128i m = {}; m.m128i_i32[0] = x; m.m128i_i32[1] = y; m.m128i_i32[2] = z; m.m128i_i32[3] = w; return m; #endif } constexpr static __m128i cxpr_setr_epi8(char b0, char b1, char b2, char b3, char b4, char b5, char b6, char b7, char b8, char b9, char b10, char b11, char b12, char b13, char b14, char b15) { #ifdef __GNUC__ return (__m128i)__v16qi{b0, b1, b2, b3, b4, b5, b6, b7, b8, b9, b10, b11, b12, b13, b14, b15}; #else __m128i m = {}; m.m128i_i8[0] = b0; m.m128i_i8[1] = b1; m.m128i_i8[2] = b2; m.m128i_i8[3] = b3; m.m128i_i8[4] = b4; m.m128i_i8[5] = b5; m.m128i_i8[6] = b6; m.m128i_i8[7] = b7; m.m128i_i8[8] = b8; m.m128i_i8[9] = b9; m.m128i_i8[10] = b10; m.m128i_i8[11] = b11; m.m128i_i8[12] = b12; m.m128i_i8[13] = b13; m.m128i_i8[14] = b14; m.m128i_i8[15] = b15; return m; #endif } public: union { struct { int x, y, z, w; }; struct { int r, g, b, a; }; struct { int left, top, right, bottom; }; int v[4]; float f32[4]; int8 i8[16]; int16 i16[8]; int32 i32[4]; int64 i64[2]; uint8 u8[16]; uint16 u16[8]; uint32 u32[4]; uint64 u64[2]; __m128i m; }; __forceinline constexpr GSVector4i() : x(0), y(0), z(0), w(0) { } constexpr static GSVector4i cxpr(int x, int y, int z, int w) { return GSVector4i(cxpr_setr_epi32(x, y, z, w)); } constexpr static GSVector4i cxpr(int x) { return GSVector4i(cxpr_setr_epi32(x, x, x, x)); } __forceinline GSVector4i(int x, int y, int z, int w) { // 4 gprs // m = _mm_set_epi32(w, z, y, x); // 2 gprs GSVector4i xz = load(x).upl32(load(z)); GSVector4i yw = load(y).upl32(load(w)); *this = xz.upl32(yw); } __forceinline GSVector4i(int x, int y) { *this = load(x).upl32(load(y)); } __forceinline GSVector4i(short s0, short s1, short s2, short s3, short s4, short s5, short s6, short s7) { m = _mm_set_epi16(s7, s6, s5, s4, s3, s2, s1, s0); } constexpr GSVector4i(char b0, char b1, char b2, char b3, char b4, char b5, char b6, char b7, char b8, char b9, char b10, char b11, char b12, char b13, char b14, char b15) : m(cxpr_setr_epi8(b0, b1, b2, b3, b4, b5, b6, b7, b8, b9, b10, b11, b12, b13, b14, b15)) { } __forceinline GSVector4i(const GSVector4i& v) { m = v.m; } __forceinline explicit GSVector4i(const GSVector2i& v) { m = _mm_loadl_epi64((__m128i*)&v); } // MSVC has bad codegen for the constexpr version when applied to non-constexpr things (https://godbolt.org/z/h8qbn7), so leave the non-constexpr version default __forceinline explicit GSVector4i(int i) { *this = i; } __forceinline constexpr explicit GSVector4i(__m128i m) : m(m) { } __forceinline explicit GSVector4i(const GSVector4& v, bool truncate = true); __forceinline static GSVector4i cast(const GSVector4& v); #if _M_SSE >= 0x500 __forceinline static GSVector4i cast(const GSVector8& v); #endif #if _M_SSE >= 0x501 __forceinline static GSVector4i cast(const GSVector8i& v); #endif __forceinline void operator=(const GSVector4i& v) { m = v.m; } __forceinline void operator=(int i) { #if _M_SSE >= 0x501 m = _mm_broadcastd_epi32(_mm_cvtsi32_si128(i)); #else m = _mm_set1_epi32(i); #endif } __forceinline void operator=(__m128i m) { this->m = m; } __forceinline operator __m128i() const { return m; } // rect __forceinline int width() const { return right - left; } __forceinline int height() const { return bottom - top; } __forceinline GSVector4i rsize() const { return *this - xyxy(); // same as GSVector4i(0, 0, width(), height()); } __forceinline unsigned int rarea() const { return width() * height(); } __forceinline bool rempty() const { return (*this < zwzw()).mask() != 0x00ff; } __forceinline GSVector4i runion(const GSVector4i& a) const { int i = (upl64(a) < uph64(a)).mask(); if (i == 0xffff) { return runion_ordered(a); } if ((i & 0x00ff) == 0x00ff) { return *this; } if ((i & 0xff00) == 0xff00) { return a; } return GSVector4i::zero(); } __forceinline GSVector4i runion_ordered(const GSVector4i& a) const { return min_i32(a).upl64(max_i32(a).srl<8>()); } __forceinline GSVector4i rintersect(const GSVector4i& a) const { return sat_i32(a); } template __forceinline GSVector4i ralign(const GSVector2i& a) const { // a must be 1 << n GSVector4i mask = GSVector4i(a) - GSVector4i(1, 1); GSVector4i v; switch (mode) { case Align_Inside: v = *this + mask; break; case Align_Outside: v = *this + mask.zwxy(); break; case Align_NegInf: v = *this; break; case Align_PosInf: v = *this + mask.zwzw(); break; default: ASSERT(0); break; } return v.andnot(mask.xyxy()); } GSVector4i fit(int arx, int ary) const; GSVector4i fit(int preset) const; #ifdef _WIN32 __forceinline operator LPCRECT() const { return (LPCRECT)this; } __forceinline operator LPRECT() { return (LPRECT)this; } #endif // __forceinline uint32 rgba32() const { GSVector4i v = *this; v = v.ps32(v); v = v.pu16(v); return (uint32)store(v); } __forceinline GSVector4i sat_i8(const GSVector4i& a, const GSVector4i& b) const { return max_i8(a).min_i8(b); } __forceinline GSVector4i sat_i8(const GSVector4i& a) const { return max_i8(a.xyxy()).min_i8(a.zwzw()); } __forceinline GSVector4i sat_i16(const GSVector4i& a, const GSVector4i& b) const { return max_i16(a).min_i16(b); } __forceinline GSVector4i sat_i16(const GSVector4i& a) const { return max_i16(a.xyxy()).min_i16(a.zwzw()); } __forceinline GSVector4i sat_i32(const GSVector4i& a, const GSVector4i& b) const { return max_i32(a).min_i32(b); } __forceinline GSVector4i sat_i32(const GSVector4i& a) const { return max_i32(a.xyxy()).min_i32(a.zwzw()); } __forceinline GSVector4i sat_u8(const GSVector4i& a, const GSVector4i& b) const { return max_u8(a).min_u8(b); } __forceinline GSVector4i sat_u8(const GSVector4i& a) const { return max_u8(a.xyxy()).min_u8(a.zwzw()); } __forceinline GSVector4i sat_u16(const GSVector4i& a, const GSVector4i& b) const { return max_u16(a).min_u16(b); } __forceinline GSVector4i sat_u16(const GSVector4i& a) const { return max_u16(a.xyxy()).min_u16(a.zwzw()); } __forceinline GSVector4i sat_u32(const GSVector4i& a, const GSVector4i& b) const { return max_u32(a).min_u32(b); } __forceinline GSVector4i sat_u32(const GSVector4i& a) const { return max_u32(a.xyxy()).min_u32(a.zwzw()); } __forceinline GSVector4i min_i8(const GSVector4i& a) const { return GSVector4i(_mm_min_epi8(m, a)); } __forceinline GSVector4i max_i8(const GSVector4i& a) const { return GSVector4i(_mm_max_epi8(m, a)); } __forceinline GSVector4i min_i16(const GSVector4i& a) const { return GSVector4i(_mm_min_epi16(m, a)); } __forceinline GSVector4i max_i16(const GSVector4i& a) const { return GSVector4i(_mm_max_epi16(m, a)); } __forceinline GSVector4i min_i32(const GSVector4i& a) const { return GSVector4i(_mm_min_epi32(m, a)); } __forceinline GSVector4i max_i32(const GSVector4i& a) const { return GSVector4i(_mm_max_epi32(m, a)); } __forceinline GSVector4i min_u8(const GSVector4i& a) const { return GSVector4i(_mm_min_epu8(m, a)); } __forceinline GSVector4i max_u8(const GSVector4i& a) const { return GSVector4i(_mm_max_epu8(m, a)); } __forceinline GSVector4i min_u16(const GSVector4i& a) const { return GSVector4i(_mm_min_epu16(m, a)); } __forceinline GSVector4i max_u16(const GSVector4i& a) const { return GSVector4i(_mm_max_epu16(m, a)); } __forceinline GSVector4i min_u32(const GSVector4i& a) const { return GSVector4i(_mm_min_epu32(m, a)); } __forceinline GSVector4i max_u32(const GSVector4i& a) const { return GSVector4i(_mm_max_epu32(m, a)); } __forceinline static int min_i16(int a, int b) { return store(load(a).min_i16(load(b))); } __forceinline GSVector4i clamp8() const { return pu16().upl8(); } __forceinline GSVector4i blend8(const GSVector4i& a, const GSVector4i& mask) const { return GSVector4i(_mm_blendv_epi8(m, a, mask)); } template __forceinline GSVector4i blend16(const GSVector4i& a) const { return GSVector4i(_mm_blend_epi16(m, a, mask)); } #if _M_SSE >= 0x501 template __forceinline GSVector4i blend32(const GSVector4i& v) const { return GSVector4i(_mm_blend_epi32(m, v.m, mask)); } #endif __forceinline GSVector4i blend(const GSVector4i& a, const GSVector4i& mask) const { return GSVector4i(_mm_or_si128(_mm_andnot_si128(mask, m), _mm_and_si128(mask, a))); } __forceinline GSVector4i mix16(const GSVector4i& a) const { return blend16<0xaa>(a); } __forceinline GSVector4i shuffle8(const GSVector4i& mask) const { return GSVector4i(_mm_shuffle_epi8(m, mask)); } __forceinline GSVector4i ps16(const GSVector4i& a) const { return GSVector4i(_mm_packs_epi16(m, a)); } __forceinline GSVector4i ps16() const { return GSVector4i(_mm_packs_epi16(m, m)); } __forceinline GSVector4i pu16(const GSVector4i& a) const { return GSVector4i(_mm_packus_epi16(m, a)); } __forceinline GSVector4i pu16() const { return GSVector4i(_mm_packus_epi16(m, m)); } __forceinline GSVector4i ps32(const GSVector4i& a) const { return GSVector4i(_mm_packs_epi32(m, a)); } __forceinline GSVector4i ps32() const { return GSVector4i(_mm_packs_epi32(m, m)); } __forceinline GSVector4i pu32(const GSVector4i& a) const { return GSVector4i(_mm_packus_epi32(m, a)); } __forceinline GSVector4i pu32() const { return GSVector4i(_mm_packus_epi32(m, m)); } __forceinline GSVector4i upl8(const GSVector4i& a) const { return GSVector4i(_mm_unpacklo_epi8(m, a)); } __forceinline GSVector4i uph8(const GSVector4i& a) const { return GSVector4i(_mm_unpackhi_epi8(m, a)); } __forceinline GSVector4i upl16(const GSVector4i& a) const { return GSVector4i(_mm_unpacklo_epi16(m, a)); } __forceinline GSVector4i uph16(const GSVector4i& a) const { return GSVector4i(_mm_unpackhi_epi16(m, a)); } __forceinline GSVector4i upl32(const GSVector4i& a) const { return GSVector4i(_mm_unpacklo_epi32(m, a)); } __forceinline GSVector4i uph32(const GSVector4i& a) const { return GSVector4i(_mm_unpackhi_epi32(m, a)); } __forceinline GSVector4i upl64(const GSVector4i& a) const { return GSVector4i(_mm_unpacklo_epi64(m, a)); } __forceinline GSVector4i uph64(const GSVector4i& a) const { return GSVector4i(_mm_unpackhi_epi64(m, a)); } __forceinline GSVector4i upl8() const { #if 0 // _M_SSE >= 0x401 // TODO: compiler bug return GSVector4i(_mm_cvtepu8_epi16(m)); #else return GSVector4i(_mm_unpacklo_epi8(m, _mm_setzero_si128())); #endif } __forceinline GSVector4i uph8() const { return GSVector4i(_mm_unpackhi_epi8(m, _mm_setzero_si128())); } __forceinline GSVector4i upl16() const { #if 0 //_M_SSE >= 0x401 // TODO: compiler bug return GSVector4i(_mm_cvtepu16_epi32(m)); #else return GSVector4i(_mm_unpacklo_epi16(m, _mm_setzero_si128())); #endif } __forceinline GSVector4i uph16() const { return GSVector4i(_mm_unpackhi_epi16(m, _mm_setzero_si128())); } __forceinline GSVector4i upl32() const { #if 0 //_M_SSE >= 0x401 // TODO: compiler bug return GSVector4i(_mm_cvtepu32_epi64(m)); #else return GSVector4i(_mm_unpacklo_epi32(m, _mm_setzero_si128())); #endif } __forceinline GSVector4i uph32() const { return GSVector4i(_mm_unpackhi_epi32(m, _mm_setzero_si128())); } __forceinline GSVector4i upl64() const { return GSVector4i(_mm_unpacklo_epi64(m, _mm_setzero_si128())); } __forceinline GSVector4i uph64() const { return GSVector4i(_mm_unpackhi_epi64(m, _mm_setzero_si128())); } // WARNING!!! // // MSVC (2008, 2010 ctp) believes that there is a "mem, reg" form of the pmovz/sx* instructions, // turning these intrinsics into a minefield, don't spill regs when using them... __forceinline GSVector4i i8to16() const { return GSVector4i(_mm_cvtepi8_epi16(m)); } __forceinline GSVector4i u8to16() const { return GSVector4i(_mm_cvtepu8_epi16(m)); } __forceinline GSVector4i i8to32() const { return GSVector4i(_mm_cvtepi8_epi32(m)); } __forceinline GSVector4i u8to32() const { return GSVector4i(_mm_cvtepu8_epi32(m)); } __forceinline GSVector4i i8to64() const { return GSVector4i(_mm_cvtepi8_epi64(m)); } __forceinline GSVector4i u8to64() const { return GSVector4i(_mm_cvtepu16_epi64(m)); } __forceinline GSVector4i i16to32() const { return GSVector4i(_mm_cvtepi16_epi32(m)); } __forceinline GSVector4i u16to32() const { return GSVector4i(_mm_cvtepu16_epi32(m)); } __forceinline GSVector4i i16to64() const { return GSVector4i(_mm_cvtepi16_epi64(m)); } __forceinline GSVector4i u16to64() const { return GSVector4i(_mm_cvtepu16_epi64(m)); } __forceinline GSVector4i i32to64() const { return GSVector4i(_mm_cvtepi32_epi64(m)); } __forceinline GSVector4i u32to64() const { return GSVector4i(_mm_cvtepu32_epi64(m)); } template __forceinline GSVector4i srl() const { return GSVector4i(_mm_srli_si128(m, i)); } template __forceinline GSVector4i srl(const GSVector4i& v) { return GSVector4i(_mm_alignr_epi8(v.m, m, i)); } template __forceinline GSVector4i sll() const { return GSVector4i(_mm_slli_si128(m, i)); } __forceinline GSVector4i sra16(int i) const { return GSVector4i(_mm_srai_epi16(m, i)); } __forceinline GSVector4i sra16(__m128i i) const { return GSVector4i(_mm_sra_epi16(m, i)); } __forceinline GSVector4i sra32(int i) const { return GSVector4i(_mm_srai_epi32(m, i)); } __forceinline GSVector4i sra32(__m128i i) const { return GSVector4i(_mm_sra_epi32(m, i)); } __forceinline GSVector4i sll16(int i) const { return GSVector4i(_mm_slli_epi16(m, i)); } __forceinline GSVector4i sll16(__m128i i) const { return GSVector4i(_mm_sll_epi16(m, i)); } __forceinline GSVector4i sll32(int i) const { return GSVector4i(_mm_slli_epi32(m, i)); } __forceinline GSVector4i sll32(__m128i i) const { return GSVector4i(_mm_sll_epi32(m, i)); } #if _M_SSE >= 0x501 __forceinline GSVector4i sllv32(__m128i i) const { return GSVector4i(_mm_sllv_epi32(m, i)); } #endif __forceinline GSVector4i sll64(int i) const { return GSVector4i(_mm_slli_epi64(m, i)); } __forceinline GSVector4i sll64(__m128i i) const { return GSVector4i(_mm_sll_epi64(m, i)); } __forceinline GSVector4i srl16(int i) const { return GSVector4i(_mm_srli_epi16(m, i)); } __forceinline GSVector4i srl16(__m128i i) const { return GSVector4i(_mm_srl_epi16(m, i)); } __forceinline GSVector4i srl32(int i) const { return GSVector4i(_mm_srli_epi32(m, i)); } __forceinline GSVector4i srl32(__m128i i) const { return GSVector4i(_mm_srl_epi32(m, i)); } #if _M_SSE >= 0x501 __forceinline GSVector4i srlv32(__m128i i) const { return GSVector4i(_mm_srlv_epi32(m, i)); } #endif __forceinline GSVector4i srl64(int i) const { return GSVector4i(_mm_srli_epi64(m, i)); } __forceinline GSVector4i srl64(__m128i i) const { return GSVector4i(_mm_srl_epi64(m, i)); } __forceinline GSVector4i add8(const GSVector4i& v) const { return GSVector4i(_mm_add_epi8(m, v.m)); } __forceinline GSVector4i add16(const GSVector4i& v) const { return GSVector4i(_mm_add_epi16(m, v.m)); } __forceinline GSVector4i add32(const GSVector4i& v) const { return GSVector4i(_mm_add_epi32(m, v.m)); } __forceinline GSVector4i adds8(const GSVector4i& v) const { return GSVector4i(_mm_adds_epi8(m, v.m)); } __forceinline GSVector4i adds16(const GSVector4i& v) const { return GSVector4i(_mm_adds_epi16(m, v.m)); } __forceinline GSVector4i addus8(const GSVector4i& v) const { return GSVector4i(_mm_adds_epu8(m, v.m)); } __forceinline GSVector4i addus16(const GSVector4i& v) const { return GSVector4i(_mm_adds_epu16(m, v.m)); } __forceinline GSVector4i sub8(const GSVector4i& v) const { return GSVector4i(_mm_sub_epi8(m, v.m)); } __forceinline GSVector4i sub16(const GSVector4i& v) const { return GSVector4i(_mm_sub_epi16(m, v.m)); } __forceinline GSVector4i sub32(const GSVector4i& v) const { return GSVector4i(_mm_sub_epi32(m, v.m)); } __forceinline GSVector4i subs8(const GSVector4i& v) const { return GSVector4i(_mm_subs_epi8(m, v.m)); } __forceinline GSVector4i subs16(const GSVector4i& v) const { return GSVector4i(_mm_subs_epi16(m, v.m)); } __forceinline GSVector4i subus8(const GSVector4i& v) const { return GSVector4i(_mm_subs_epu8(m, v.m)); } __forceinline GSVector4i subus16(const GSVector4i& v) const { return GSVector4i(_mm_subs_epu16(m, v.m)); } __forceinline GSVector4i avg8(const GSVector4i& v) const { return GSVector4i(_mm_avg_epu8(m, v.m)); } __forceinline GSVector4i avg16(const GSVector4i& v) const { return GSVector4i(_mm_avg_epu16(m, v.m)); } __forceinline GSVector4i mul16hs(const GSVector4i& v) const { return GSVector4i(_mm_mulhi_epi16(m, v.m)); } __forceinline GSVector4i mul16hu(const GSVector4i& v) const { return GSVector4i(_mm_mulhi_epu16(m, v.m)); } __forceinline GSVector4i mul16l(const GSVector4i& v) const { return GSVector4i(_mm_mullo_epi16(m, v.m)); } __forceinline GSVector4i mul16hrs(const GSVector4i& v) const { return GSVector4i(_mm_mulhrs_epi16(m, v.m)); } GSVector4i madd(const GSVector4i& v) const { return GSVector4i(_mm_madd_epi16(m, v.m)); } template __forceinline GSVector4i lerp16(const GSVector4i& a, const GSVector4i& f) const { // (a - this) * f << shift + this return add16(a.sub16(*this).modulate16(f)); } template __forceinline static GSVector4i lerp16(const GSVector4i& a, const GSVector4i& b, const GSVector4i& c) { // (a - b) * c << shift return a.sub16(b).modulate16(c); } template __forceinline static GSVector4i lerp16(const GSVector4i& a, const GSVector4i& b, const GSVector4i& c, const GSVector4i& d) { // (a - b) * c << shift + d return d.add16(a.sub16(b).modulate16(c)); } __forceinline GSVector4i lerp16_4(const GSVector4i& a, const GSVector4i& f) const { // (a - this) * f >> 4 + this (a, this: 8-bit, f: 4-bit) return add16(a.sub16(*this).mul16l(f).sra16(4)); } template __forceinline GSVector4i modulate16(const GSVector4i& f) const { // a * f << shift #if _M_SSE >= 0x301 if (shift == 0) { return mul16hrs(f); } #endif return sll16(shift + 1).mul16hs(f); } __forceinline bool eq(const GSVector4i& v) const { // pxor, ptest, je GSVector4i t = *this ^ v; return _mm_testz_si128(t, t) != 0; } __forceinline GSVector4i eq8(const GSVector4i& v) const { return GSVector4i(_mm_cmpeq_epi8(m, v.m)); } __forceinline GSVector4i eq16(const GSVector4i& v) const { return GSVector4i(_mm_cmpeq_epi16(m, v.m)); } __forceinline GSVector4i eq32(const GSVector4i& v) const { return GSVector4i(_mm_cmpeq_epi32(m, v.m)); } __forceinline GSVector4i neq8(const GSVector4i& v) const { return ~eq8(v); } __forceinline GSVector4i neq16(const GSVector4i& v) const { return ~eq16(v); } __forceinline GSVector4i neq32(const GSVector4i& v) const { return ~eq32(v); } __forceinline GSVector4i gt8(const GSVector4i& v) const { return GSVector4i(_mm_cmpgt_epi8(m, v.m)); } __forceinline GSVector4i gt16(const GSVector4i& v) const { return GSVector4i(_mm_cmpgt_epi16(m, v.m)); } __forceinline GSVector4i gt32(const GSVector4i& v) const { return GSVector4i(_mm_cmpgt_epi32(m, v.m)); } __forceinline GSVector4i lt8(const GSVector4i& v) const { return GSVector4i(_mm_cmplt_epi8(m, v.m)); } __forceinline GSVector4i lt16(const GSVector4i& v) const { return GSVector4i(_mm_cmplt_epi16(m, v.m)); } __forceinline GSVector4i lt32(const GSVector4i& v) const { return GSVector4i(_mm_cmplt_epi32(m, v.m)); } __forceinline GSVector4i andnot(const GSVector4i& v) const { return GSVector4i(_mm_andnot_si128(v.m, m)); } __forceinline int mask() const { return _mm_movemask_epi8(m); } __forceinline bool alltrue() const { return mask() == 0xffff; } __forceinline bool allfalse() const { return _mm_testz_si128(m, m) != 0; } template __forceinline GSVector4i insert8(int a) const { return GSVector4i(_mm_insert_epi8(m, a, i)); } template __forceinline int extract8() const { return _mm_extract_epi8(m, i); } template __forceinline GSVector4i insert16(int a) const { return GSVector4i(_mm_insert_epi16(m, a, i)); } template __forceinline int extract16() const { return _mm_extract_epi16(m, i); } template __forceinline GSVector4i insert32(int a) const { return GSVector4i(_mm_insert_epi32(m, a, i)); } template __forceinline int extract32() const { if (i == 0) return GSVector4i::store(*this); return _mm_extract_epi32(m, i); } #ifdef _M_AMD64 template __forceinline GSVector4i insert64(int64 a) const { return GSVector4i(_mm_insert_epi64(m, a, i)); } template __forceinline int64 extract64() const { if (i == 0) return GSVector4i::storeq(*this); return _mm_extract_epi64(m, i); } #endif template __forceinline GSVector4i gather8_4(const T* ptr) const { GSVector4i v; v = load((int)ptr[extract8() & 0xf]); v = v.insert8<1>((int)ptr[extract8() >> 4]); v = v.insert8<2>((int)ptr[extract8() & 0xf]); v = v.insert8<3>((int)ptr[extract8() >> 4]); v = v.insert8<4>((int)ptr[extract8() & 0xf]); v = v.insert8<5>((int)ptr[extract8() >> 4]); v = v.insert8<6>((int)ptr[extract8() & 0xf]); v = v.insert8<7>((int)ptr[extract8() >> 4]); v = v.insert8<8>((int)ptr[extract8() & 0xf]); v = v.insert8<9>((int)ptr[extract8() >> 4]); v = v.insert8<10>((int)ptr[extract8() & 0xf]); v = v.insert8<11>((int)ptr[extract8() >> 4]); v = v.insert8<12>((int)ptr[extract8() & 0xf]); v = v.insert8<13>((int)ptr[extract8() >> 4]); v = v.insert8<14>((int)ptr[extract8() & 0xf]); v = v.insert8<15>((int)ptr[extract8() >> 4]); return v; } template __forceinline GSVector4i gather8_8(const T* ptr) const { GSVector4i v; v = load((int)ptr[extract8<0>()]); v = v.insert8<1>((int)ptr[extract8<1>()]); v = v.insert8<2>((int)ptr[extract8<2>()]); v = v.insert8<3>((int)ptr[extract8<3>()]); v = v.insert8<4>((int)ptr[extract8<4>()]); v = v.insert8<5>((int)ptr[extract8<5>()]); v = v.insert8<6>((int)ptr[extract8<6>()]); v = v.insert8<7>((int)ptr[extract8<7>()]); v = v.insert8<8>((int)ptr[extract8<8>()]); v = v.insert8<9>((int)ptr[extract8<9>()]); v = v.insert8<10>((int)ptr[extract8<10>()]); v = v.insert8<11>((int)ptr[extract8<11>()]); v = v.insert8<12>((int)ptr[extract8<12>()]); v = v.insert8<13>((int)ptr[extract8<13>()]); v = v.insert8<14>((int)ptr[extract8<14>()]); v = v.insert8<15>((int)ptr[extract8<15>()]); return v; } template __forceinline GSVector4i gather8_16(const T* ptr, const GSVector4i& a) const { GSVector4i v = a; v = v.insert8((int)ptr[extract16<0>()]); v = v.insert8((int)ptr[extract16<1>()]); v = v.insert8((int)ptr[extract16<2>()]); v = v.insert8((int)ptr[extract16<3>()]); v = v.insert8((int)ptr[extract16<4>()]); v = v.insert8((int)ptr[extract16<5>()]); v = v.insert8((int)ptr[extract16<6>()]); v = v.insert8((int)ptr[extract16<7>()]); return v; } template __forceinline GSVector4i gather8_32(const T* ptr, const GSVector4i& a) const { GSVector4i v = a; v = v.insert8((int)ptr[extract32<0>()]); v = v.insert8((int)ptr[extract32<1>()]); v = v.insert8((int)ptr[extract32<2>()]); v = v.insert8((int)ptr[extract32<3>()]); return v; } template __forceinline GSVector4i gather16_4(const T* ptr) const { GSVector4i v; v = load((int)ptr[extract8() & 0xf]); v = v.insert16<1>((int)ptr[extract8() >> 4]); v = v.insert16<2>((int)ptr[extract8() & 0xf]); v = v.insert16<3>((int)ptr[extract8() >> 4]); v = v.insert16<4>((int)ptr[extract8() & 0xf]); v = v.insert16<5>((int)ptr[extract8() >> 4]); v = v.insert16<6>((int)ptr[extract8() & 0xf]); v = v.insert16<7>((int)ptr[extract8() >> 4]); return v; } template __forceinline GSVector4i gather16_8(const T* ptr) const { GSVector4i v; v = load((int)ptr[extract8()]); v = v.insert16<1>((int)ptr[extract8()]); v = v.insert16<2>((int)ptr[extract8()]); v = v.insert16<3>((int)ptr[extract8()]); v = v.insert16<4>((int)ptr[extract8()]); v = v.insert16<5>((int)ptr[extract8()]); v = v.insert16<6>((int)ptr[extract8()]); v = v.insert16<7>((int)ptr[extract8()]); return v; } template __forceinline GSVector4i gather16_16(const T* ptr) const { GSVector4i v; v = load((int)ptr[extract16<0>()]); v = v.insert16<1>((int)ptr[extract16<1>()]); v = v.insert16<2>((int)ptr[extract16<2>()]); v = v.insert16<3>((int)ptr[extract16<3>()]); v = v.insert16<4>((int)ptr[extract16<4>()]); v = v.insert16<5>((int)ptr[extract16<5>()]); v = v.insert16<6>((int)ptr[extract16<6>()]); v = v.insert16<7>((int)ptr[extract16<7>()]); return v; } template __forceinline GSVector4i gather16_16(const T1* ptr1, const T2* ptr2) const { GSVector4i v; v = load((int)ptr2[ptr1[extract16<0>()]]); v = v.insert16<1>((int)ptr2[ptr1[extract16<1>()]]); v = v.insert16<2>((int)ptr2[ptr1[extract16<2>()]]); v = v.insert16<3>((int)ptr2[ptr1[extract16<3>()]]); v = v.insert16<4>((int)ptr2[ptr1[extract16<4>()]]); v = v.insert16<5>((int)ptr2[ptr1[extract16<5>()]]); v = v.insert16<6>((int)ptr2[ptr1[extract16<6>()]]); v = v.insert16<7>((int)ptr2[ptr1[extract16<7>()]]); return v; } template __forceinline GSVector4i gather16_32(const T* ptr, const GSVector4i& a) const { GSVector4i v = a; v = v.insert16((int)ptr[extract32<0>()]); v = v.insert16((int)ptr[extract32<1>()]); v = v.insert16((int)ptr[extract32<2>()]); v = v.insert16((int)ptr[extract32<3>()]); return v; } template __forceinline GSVector4i gather32_4(const T* ptr) const { GSVector4i v; v = load((int)ptr[extract8() & 0xf]); v = v.insert32<1>((int)ptr[extract8() >> 4]); v = v.insert32<2>((int)ptr[extract8() & 0xf]); v = v.insert32<3>((int)ptr[extract8() >> 4]); return v; } template __forceinline GSVector4i gather32_8(const T* ptr) const { GSVector4i v; v = load((int)ptr[extract8()]); v = v.insert32<1>((int)ptr[extract8()]); v = v.insert32<2>((int)ptr[extract8()]); v = v.insert32<3>((int)ptr[extract8()]); return v; } template __forceinline GSVector4i gather32_16(const T* ptr) const { GSVector4i v; v = load((int)ptr[extract16()]); v = v.insert32<1>((int)ptr[extract16()]); v = v.insert32<2>((int)ptr[extract16()]); v = v.insert32<3>((int)ptr[extract16()]); return v; } template __forceinline GSVector4i gather32_32(const T* ptr) const { GSVector4i v; v = load((int)ptr[extract32<0>()]); v = v.insert32<1>((int)ptr[extract32<1>()]); v = v.insert32<2>((int)ptr[extract32<2>()]); v = v.insert32<3>((int)ptr[extract32<3>()]); return v; } template __forceinline GSVector4i gather32_32(const T1* ptr1, const T2* ptr2) const { GSVector4i v; v = load((int)ptr2[ptr1[extract32<0>()]]); v = v.insert32<1>((int)ptr2[ptr1[extract32<1>()]]); v = v.insert32<2>((int)ptr2[ptr1[extract32<2>()]]); v = v.insert32<3>((int)ptr2[ptr1[extract32<3>()]]); return v; } #if defined(_M_AMD64) template __forceinline GSVector4i gather64_4(const T* ptr) const { GSVector4i v; v = loadq((int64)ptr[extract8() & 0xf]); v = v.insert64<1>((int64)ptr[extract8() >> 4]); return v; } template __forceinline GSVector4i gather64_8(const T* ptr) const { GSVector4i v; v = loadq((int64)ptr[extract8()]); v = v.insert64<1>((int64)ptr[extract8()]); return v; } template __forceinline GSVector4i gather64_16(const T* ptr) const { GSVector4i v; v = loadq((int64)ptr[extract16()]); v = v.insert64<1>((int64)ptr[extract16()]); return v; } template __forceinline GSVector4i gather64_32(const T* ptr) const { GSVector4i v; v = loadq((int64)ptr[extract32()]); v = v.insert64<1>((int64)ptr[extract32()]); return v; } template __forceinline GSVector4i gather64_64(const T* ptr) const { GSVector4i v; v = loadq((int64)ptr[extract64<0>()]); v = v.insert64<1>((int64)ptr[extract64<1>()]); return v; } #else template __forceinline GSVector4i gather64_4(const T* ptr) const { GSVector4i v; v = loadu(&ptr[extract8() & 0xf], &ptr[extract8() >> 4]); return v; } template __forceinline GSVector4i gather64_8(const T* ptr) const { GSVector4i v; v = load(&ptr[extract8()], &ptr[extract8()]); return v; } template __forceinline GSVector4i gather64_16(const T* ptr) const { GSVector4i v; v = load(&ptr[extract16()], &ptr[extract16()]); return v; } template __forceinline GSVector4i gather64_32(const T* ptr) const { GSVector4i v; v = load(&ptr[extract32()], &ptr[extract32()]); return v; } #endif template __forceinline void gather8_4(const T* RESTRICT ptr, GSVector4i* RESTRICT dst) const { dst[0] = gather8_4<0>(ptr); dst[1] = gather8_4<8>(ptr); } __forceinline void gather8_8(const uint8* RESTRICT ptr, GSVector4i* RESTRICT dst) const { dst[0] = gather8_8<>(ptr); } template __forceinline void gather16_4(const T* RESTRICT ptr, GSVector4i* RESTRICT dst) const { dst[0] = gather16_4<0>(ptr); dst[1] = gather16_4<4>(ptr); dst[2] = gather16_4<8>(ptr); dst[3] = gather16_4<12>(ptr); } template __forceinline void gather16_8(const T* RESTRICT ptr, GSVector4i* RESTRICT dst) const { dst[0] = gather16_8<0>(ptr); dst[1] = gather16_8<8>(ptr); } template __forceinline void gather16_16(const T* RESTRICT ptr, GSVector4i* RESTRICT dst) const { dst[0] = gather16_16<>(ptr); } template __forceinline void gather32_4(const T* RESTRICT ptr, GSVector4i* RESTRICT dst) const { dst[0] = gather32_4<0>(ptr); dst[1] = gather32_4<2>(ptr); dst[2] = gather32_4<4>(ptr); dst[3] = gather32_4<6>(ptr); dst[4] = gather32_4<8>(ptr); dst[5] = gather32_4<10>(ptr); dst[6] = gather32_4<12>(ptr); dst[7] = gather32_4<14>(ptr); } template __forceinline void gather32_8(const T* RESTRICT ptr, GSVector4i* RESTRICT dst) const { dst[0] = gather32_8<0>(ptr); dst[1] = gather32_8<4>(ptr); dst[2] = gather32_8<8>(ptr); dst[3] = gather32_8<12>(ptr); } template __forceinline void gather32_16(const T* RESTRICT ptr, GSVector4i* RESTRICT dst) const { dst[0] = gather32_16<0>(ptr); dst[1] = gather32_16<4>(ptr); } template __forceinline void gather32_32(const T* RESTRICT ptr, GSVector4i* RESTRICT dst) const { dst[0] = gather32_32<>(ptr); } template __forceinline void gather64_4(const T* RESTRICT ptr, GSVector4i* RESTRICT dst) const { dst[0] = gather64_4<0>(ptr); dst[1] = gather64_4<1>(ptr); dst[2] = gather64_4<2>(ptr); dst[3] = gather64_4<3>(ptr); dst[4] = gather64_4<4>(ptr); dst[5] = gather64_4<5>(ptr); dst[6] = gather64_4<6>(ptr); dst[7] = gather64_4<7>(ptr); dst[8] = gather64_4<8>(ptr); dst[9] = gather64_4<9>(ptr); dst[10] = gather64_4<10>(ptr); dst[11] = gather64_4<11>(ptr); dst[12] = gather64_4<12>(ptr); dst[13] = gather64_4<13>(ptr); dst[14] = gather64_4<14>(ptr); dst[15] = gather64_4<15>(ptr); } template __forceinline void gather64_8(const T* RESTRICT ptr, GSVector4i* RESTRICT dst) const { dst[0] = gather64_8<0>(ptr); dst[1] = gather64_8<2>(ptr); dst[2] = gather64_8<4>(ptr); dst[3] = gather64_8<6>(ptr); dst[4] = gather64_8<8>(ptr); dst[5] = gather64_8<10>(ptr); dst[6] = gather64_8<12>(ptr); dst[7] = gather64_8<14>(ptr); } template __forceinline void gather64_16(const T* RESTRICT ptr, GSVector4i* RESTRICT dst) const { dst[0] = gather64_16<0>(ptr); dst[1] = gather64_16<2>(ptr); dst[2] = gather64_16<4>(ptr); dst[3] = gather64_16<8>(ptr); } template __forceinline void gather64_32(const T* RESTRICT ptr, GSVector4i* RESTRICT dst) const { dst[0] = gather64_32<0>(ptr); dst[1] = gather64_32<2>(ptr); } #ifdef _M_AMD64 template __forceinline void gather64_64(const T* RESTRICT ptr, GSVector4i* RESTRICT dst) const { dst[0] = gather64_64<>(ptr); } #endif __forceinline static GSVector4i loadnt(const void* p) { return GSVector4i(_mm_stream_load_si128((__m128i*)p)); } __forceinline static GSVector4i loadl(const void* p) { return GSVector4i(_mm_loadl_epi64((__m128i*)p)); } __forceinline static GSVector4i loadh(const void* p) { return GSVector4i(_mm_castps_si128(_mm_loadh_pi(_mm_setzero_ps(), (__m64*)p))); } __forceinline static GSVector4i loadh(const void* p, const GSVector4i& v) { return GSVector4i(_mm_castps_si128(_mm_loadh_pi(_mm_castsi128_ps(v.m), (__m64*)p))); } __forceinline static GSVector4i load(const void* pl, const void* ph) { return loadh(ph, loadl(pl)); } /* __forceinline static GSVector4i load(const void* pl, const void* ph) { __m128i lo = _mm_loadl_epi64((__m128i*)pl); __m128i hi = _mm_loadl_epi64((__m128i*)ph); return GSVector4i(_mm_unpacklo_epi64(lo, hi)); } */ template __forceinline static GSVector4i load(const void* p) { return GSVector4i(aligned ? _mm_load_si128((__m128i*)p) : _mm_loadu_si128((__m128i*)p)); } __forceinline static GSVector4i load(int i) { return GSVector4i(_mm_cvtsi32_si128(i)); } #ifdef _M_AMD64 __forceinline static GSVector4i loadq(int64 i) { return GSVector4i(_mm_cvtsi64_si128(i)); } #endif __forceinline static void storent(void* p, const GSVector4i& v) { _mm_stream_si128((__m128i*)p, v.m); } __forceinline static void storel(void* p, const GSVector4i& v) { _mm_storel_epi64((__m128i*)p, v.m); } __forceinline static void storeh(void* p, const GSVector4i& v) { _mm_storeh_pi((__m64*)p, _mm_castsi128_ps(v.m)); } __forceinline static void store(void* pl, void* ph, const GSVector4i& v) { GSVector4i::storel(pl, v); GSVector4i::storeh(ph, v); } template __forceinline static void store(void* p, const GSVector4i& v) { if (aligned) _mm_store_si128((__m128i*)p, v.m); else _mm_storeu_si128((__m128i*)p, v.m); } __forceinline static int store(const GSVector4i& v) { return _mm_cvtsi128_si32(v.m); } #ifdef _M_AMD64 __forceinline static int64 storeq(const GSVector4i& v) { return _mm_cvtsi128_si64(v.m); } #endif __forceinline static void storent(void* RESTRICT dst, const void* RESTRICT src, size_t size) { const GSVector4i* s = (const GSVector4i*)src; GSVector4i* d = (GSVector4i*)dst; if (size == 0) return; size_t i = 0; size_t j = size >> 6; for (; i < j; i++, s += 4, d += 4) { storent(&d[0], s[0]); storent(&d[1], s[1]); storent(&d[2], s[2]); storent(&d[3], s[3]); } size &= 63; if (size == 0) return; memcpy(d, s, size); } __forceinline static void transpose(GSVector4i& a, GSVector4i& b, GSVector4i& c, GSVector4i& d) { _MM_TRANSPOSE4_SI128(a.m, b.m, c.m, d.m); } __forceinline static void sw4(GSVector4i& a, GSVector4i& b, GSVector4i& c, GSVector4i& d) { const __m128i epi32_0f0f0f0f = _mm_set1_epi32(0x0f0f0f0f); GSVector4i mask(epi32_0f0f0f0f); GSVector4i e = (b << 4).blend(a, mask); GSVector4i f = b.blend(a >> 4, mask); GSVector4i g = (d << 4).blend(c, mask); GSVector4i h = d.blend(c >> 4, mask); a = e.upl8(f); c = e.uph8(f); b = g.upl8(h); d = g.uph8(h); } __forceinline static void sw8(GSVector4i& a, GSVector4i& b, GSVector4i& c, GSVector4i& d) { GSVector4i e = a; GSVector4i f = c; a = e.upl8(b); c = e.uph8(b); b = f.upl8(d); d = f.uph8(d); } __forceinline static void sw16(GSVector4i& a, GSVector4i& b, GSVector4i& c, GSVector4i& d) { GSVector4i e = a; GSVector4i f = c; a = e.upl16(b); c = e.uph16(b); b = f.upl16(d); d = f.uph16(d); } __forceinline static void sw16rl(GSVector4i& a, GSVector4i& b, GSVector4i& c, GSVector4i& d) { GSVector4i e = a; GSVector4i f = c; a = b.upl16(e); c = e.uph16(b); b = d.upl16(f); d = f.uph16(d); } __forceinline static void sw16rh(GSVector4i& a, GSVector4i& b, GSVector4i& c, GSVector4i& d) { GSVector4i e = a; GSVector4i f = c; a = e.upl16(b); c = b.uph16(e); b = f.upl16(d); d = d.uph16(f); } __forceinline static void sw32(GSVector4i& a, GSVector4i& b, GSVector4i& c, GSVector4i& d) { GSVector4i e = a; GSVector4i f = c; a = e.upl32(b); c = e.uph32(b); b = f.upl32(d); d = f.uph32(d); } __forceinline static void sw64(GSVector4i& a, GSVector4i& b, GSVector4i& c, GSVector4i& d) { GSVector4i e = a; GSVector4i f = c; a = e.upl64(b); c = e.uph64(b); b = f.upl64(d); d = f.uph64(d); } __forceinline static bool compare16(const void* dst, const void* src, size_t size) { ASSERT((size & 15) == 0); size >>= 4; GSVector4i* s = (GSVector4i*)src; GSVector4i* d = (GSVector4i*)dst; for (size_t i = 0; i < size; i++) { if (!d[i].eq(s[i])) { return false; } } return true; } __forceinline static bool compare64(const void* dst, const void* src, size_t size) { ASSERT((size & 63) == 0); size >>= 6; GSVector4i* s = (GSVector4i*)src; GSVector4i* d = (GSVector4i*)dst; for (size_t i = 0; i < size; ++i) { GSVector4i v0 = (d[i * 4 + 0] == s[i * 4 + 0]); GSVector4i v1 = (d[i * 4 + 1] == s[i * 4 + 1]); GSVector4i v2 = (d[i * 4 + 2] == s[i * 4 + 2]); GSVector4i v3 = (d[i * 4 + 3] == s[i * 4 + 3]); v0 = v0 & v1; v2 = v2 & v3; if (!(v0 & v2).alltrue()) { return false; } } return true; } __forceinline static bool update(const void* dst, const void* src, size_t size) { ASSERT((size & 15) == 0); size >>= 4; GSVector4i* s = (GSVector4i*)src; GSVector4i* d = (GSVector4i*)dst; GSVector4i v = GSVector4i::xffffffff(); for (size_t i = 0; i < size; i++) { v &= d[i] == s[i]; d[i] = s[i]; } return v.alltrue(); } __forceinline void operator+=(const GSVector4i& v) { m = _mm_add_epi32(m, v); } __forceinline void operator-=(const GSVector4i& v) { m = _mm_sub_epi32(m, v); } __forceinline void operator+=(int i) { *this += GSVector4i(i); } __forceinline void operator-=(int i) { *this -= GSVector4i(i); } __forceinline void operator<<=(const int i) { m = _mm_slli_epi32(m, i); } __forceinline void operator>>=(const int i) { m = _mm_srli_epi32(m, i); } __forceinline void operator&=(const GSVector4i& v) { m = _mm_and_si128(m, v); } __forceinline void operator|=(const GSVector4i& v) { m = _mm_or_si128(m, v); } __forceinline void operator^=(const GSVector4i& v) { m = _mm_xor_si128(m, v); } __forceinline friend GSVector4i operator+(const GSVector4i& v1, const GSVector4i& v2) { return GSVector4i(_mm_add_epi32(v1, v2)); } __forceinline friend GSVector4i operator-(const GSVector4i& v1, const GSVector4i& v2) { return GSVector4i(_mm_sub_epi32(v1, v2)); } __forceinline friend GSVector4i operator+(const GSVector4i& v, int i) { return v + GSVector4i(i); } __forceinline friend GSVector4i operator-(const GSVector4i& v, int i) { return v - GSVector4i(i); } __forceinline friend GSVector4i operator<<(const GSVector4i& v, const int i) { return GSVector4i(_mm_slli_epi32(v, i)); } __forceinline friend GSVector4i operator>>(const GSVector4i& v, const int i) { return GSVector4i(_mm_srli_epi32(v, i)); } __forceinline friend GSVector4i operator&(const GSVector4i& v1, const GSVector4i& v2) { return GSVector4i(_mm_and_si128(v1, v2)); } __forceinline friend GSVector4i operator|(const GSVector4i& v1, const GSVector4i& v2) { return GSVector4i(_mm_or_si128(v1, v2)); } __forceinline friend GSVector4i operator^(const GSVector4i& v1, const GSVector4i& v2) { return GSVector4i(_mm_xor_si128(v1, v2)); } __forceinline friend GSVector4i operator&(const GSVector4i& v, int i) { return v & GSVector4i(i); } __forceinline friend GSVector4i operator|(const GSVector4i& v, int i) { return v | GSVector4i(i); } __forceinline friend GSVector4i operator^(const GSVector4i& v, int i) { return v ^ GSVector4i(i); } __forceinline friend GSVector4i operator~(const GSVector4i& v) { return v ^ (v == v); } __forceinline friend GSVector4i operator==(const GSVector4i& v1, const GSVector4i& v2) { return GSVector4i(_mm_cmpeq_epi32(v1, v2)); } __forceinline friend GSVector4i operator!=(const GSVector4i& v1, const GSVector4i& v2) { return ~(v1 == v2); } __forceinline friend GSVector4i operator>(const GSVector4i& v1, const GSVector4i& v2) { return GSVector4i(_mm_cmpgt_epi32(v1, v2)); } __forceinline friend GSVector4i operator<(const GSVector4i& v1, const GSVector4i& v2) { return GSVector4i(_mm_cmplt_epi32(v1, v2)); } __forceinline friend GSVector4i operator>=(const GSVector4i& v1, const GSVector4i& v2) { return (v1 > v2) | (v1 == v2); } __forceinline friend GSVector4i operator<=(const GSVector4i& v1, const GSVector4i& v2) { return (v1 < v2) | (v1 == v2); } // clang-format off #define VECTOR4i_SHUFFLE_4(xs, xn, ys, yn, zs, zn, ws, wn) \ __forceinline GSVector4i xs##ys##zs##ws() const {return GSVector4i(_mm_shuffle_epi32(m, _MM_SHUFFLE(wn, zn, yn, xn)));} \ __forceinline GSVector4i xs##ys##zs##ws##l() const {return GSVector4i(_mm_shufflelo_epi16(m, _MM_SHUFFLE(wn, zn, yn, xn)));} \ __forceinline GSVector4i xs##ys##zs##ws##h() const {return GSVector4i(_mm_shufflehi_epi16(m, _MM_SHUFFLE(wn, zn, yn, xn)));} \ __forceinline GSVector4i xs##ys##zs##ws##lh() const {return GSVector4i(_mm_shufflehi_epi16(_mm_shufflelo_epi16(m, _MM_SHUFFLE(wn, zn, yn, xn)), _MM_SHUFFLE(wn, zn, yn, xn)));} \ #define VECTOR4i_SHUFFLE_3(xs, xn, ys, yn, zs, zn) \ VECTOR4i_SHUFFLE_4(xs, xn, ys, yn, zs, zn, x, 0) \ VECTOR4i_SHUFFLE_4(xs, xn, ys, yn, zs, zn, y, 1) \ VECTOR4i_SHUFFLE_4(xs, xn, ys, yn, zs, zn, z, 2) \ VECTOR4i_SHUFFLE_4(xs, xn, ys, yn, zs, zn, w, 3) \ #define VECTOR4i_SHUFFLE_2(xs, xn, ys, yn) \ VECTOR4i_SHUFFLE_3(xs, xn, ys, yn, x, 0) \ VECTOR4i_SHUFFLE_3(xs, xn, ys, yn, y, 1) \ VECTOR4i_SHUFFLE_3(xs, xn, ys, yn, z, 2) \ VECTOR4i_SHUFFLE_3(xs, xn, ys, yn, w, 3) \ #define VECTOR4i_SHUFFLE_1(xs, xn) \ VECTOR4i_SHUFFLE_2(xs, xn, x, 0) \ VECTOR4i_SHUFFLE_2(xs, xn, y, 1) \ VECTOR4i_SHUFFLE_2(xs, xn, z, 2) \ VECTOR4i_SHUFFLE_2(xs, xn, w, 3) \ VECTOR4i_SHUFFLE_1(x, 0) VECTOR4i_SHUFFLE_1(y, 1) VECTOR4i_SHUFFLE_1(z, 2) VECTOR4i_SHUFFLE_1(w, 3) // clang-format on __forceinline static GSVector4i zero() { return GSVector4i(_mm_setzero_si128()); } __forceinline static GSVector4i xffffffff() { return zero() == zero(); } __forceinline static GSVector4i x00000001() { return xffffffff().srl32(31); } __forceinline static GSVector4i x00000003() { return xffffffff().srl32(30); } __forceinline static GSVector4i x00000007() { return xffffffff().srl32(29); } __forceinline static GSVector4i x0000000f() { return xffffffff().srl32(28); } __forceinline static GSVector4i x0000001f() { return xffffffff().srl32(27); } __forceinline static GSVector4i x0000003f() { return xffffffff().srl32(26); } __forceinline static GSVector4i x0000007f() { return xffffffff().srl32(25); } __forceinline static GSVector4i x000000ff() { return xffffffff().srl32(24); } __forceinline static GSVector4i x000001ff() { return xffffffff().srl32(23); } __forceinline static GSVector4i x000003ff() { return xffffffff().srl32(22); } __forceinline static GSVector4i x000007ff() { return xffffffff().srl32(21); } __forceinline static GSVector4i x00000fff() { return xffffffff().srl32(20); } __forceinline static GSVector4i x00001fff() { return xffffffff().srl32(19); } __forceinline static GSVector4i x00003fff() { return xffffffff().srl32(18); } __forceinline static GSVector4i x00007fff() { return xffffffff().srl32(17); } __forceinline static GSVector4i x0000ffff() { return xffffffff().srl32(16); } __forceinline static GSVector4i x0001ffff() { return xffffffff().srl32(15); } __forceinline static GSVector4i x0003ffff() { return xffffffff().srl32(14); } __forceinline static GSVector4i x0007ffff() { return xffffffff().srl32(13); } __forceinline static GSVector4i x000fffff() { return xffffffff().srl32(12); } __forceinline static GSVector4i x001fffff() { return xffffffff().srl32(11); } __forceinline static GSVector4i x003fffff() { return xffffffff().srl32(10); } __forceinline static GSVector4i x007fffff() { return xffffffff().srl32( 9); } __forceinline static GSVector4i x00ffffff() { return xffffffff().srl32( 8); } __forceinline static GSVector4i x01ffffff() { return xffffffff().srl32( 7); } __forceinline static GSVector4i x03ffffff() { return xffffffff().srl32( 6); } __forceinline static GSVector4i x07ffffff() { return xffffffff().srl32( 5); } __forceinline static GSVector4i x0fffffff() { return xffffffff().srl32( 4); } __forceinline static GSVector4i x1fffffff() { return xffffffff().srl32( 3); } __forceinline static GSVector4i x3fffffff() { return xffffffff().srl32( 2); } __forceinline static GSVector4i x7fffffff() { return xffffffff().srl32( 1); } __forceinline static GSVector4i x80000000() { return xffffffff().sll32(31); } __forceinline static GSVector4i xc0000000() { return xffffffff().sll32(30); } __forceinline static GSVector4i xe0000000() { return xffffffff().sll32(29); } __forceinline static GSVector4i xf0000000() { return xffffffff().sll32(28); } __forceinline static GSVector4i xf8000000() { return xffffffff().sll32(27); } __forceinline static GSVector4i xfc000000() { return xffffffff().sll32(26); } __forceinline static GSVector4i xfe000000() { return xffffffff().sll32(25); } __forceinline static GSVector4i xff000000() { return xffffffff().sll32(24); } __forceinline static GSVector4i xff800000() { return xffffffff().sll32(23); } __forceinline static GSVector4i xffc00000() { return xffffffff().sll32(22); } __forceinline static GSVector4i xffe00000() { return xffffffff().sll32(21); } __forceinline static GSVector4i xfff00000() { return xffffffff().sll32(20); } __forceinline static GSVector4i xfff80000() { return xffffffff().sll32(19); } __forceinline static GSVector4i xfffc0000() { return xffffffff().sll32(18); } __forceinline static GSVector4i xfffe0000() { return xffffffff().sll32(17); } __forceinline static GSVector4i xffff0000() { return xffffffff().sll32(16); } __forceinline static GSVector4i xffff8000() { return xffffffff().sll32(15); } __forceinline static GSVector4i xffffc000() { return xffffffff().sll32(14); } __forceinline static GSVector4i xffffe000() { return xffffffff().sll32(13); } __forceinline static GSVector4i xfffff000() { return xffffffff().sll32(12); } __forceinline static GSVector4i xfffff800() { return xffffffff().sll32(11); } __forceinline static GSVector4i xfffffc00() { return xffffffff().sll32(10); } __forceinline static GSVector4i xfffffe00() { return xffffffff().sll32( 9); } __forceinline static GSVector4i xffffff00() { return xffffffff().sll32( 8); } __forceinline static GSVector4i xffffff80() { return xffffffff().sll32( 7); } __forceinline static GSVector4i xffffffc0() { return xffffffff().sll32( 6); } __forceinline static GSVector4i xffffffe0() { return xffffffff().sll32( 5); } __forceinline static GSVector4i xfffffff0() { return xffffffff().sll32( 4); } __forceinline static GSVector4i xfffffff8() { return xffffffff().sll32( 3); } __forceinline static GSVector4i xfffffffc() { return xffffffff().sll32( 2); } __forceinline static GSVector4i xfffffffe() { return xffffffff().sll32( 1); } __forceinline static GSVector4i x0001() { return xffffffff().srl16(15); } __forceinline static GSVector4i x0003() { return xffffffff().srl16(14); } __forceinline static GSVector4i x0007() { return xffffffff().srl16(13); } __forceinline static GSVector4i x000f() { return xffffffff().srl16(12); } __forceinline static GSVector4i x001f() { return xffffffff().srl16(11); } __forceinline static GSVector4i x003f() { return xffffffff().srl16(10); } __forceinline static GSVector4i x007f() { return xffffffff().srl16( 9); } __forceinline static GSVector4i x00ff() { return xffffffff().srl16( 8); } __forceinline static GSVector4i x01ff() { return xffffffff().srl16( 7); } __forceinline static GSVector4i x03ff() { return xffffffff().srl16( 6); } __forceinline static GSVector4i x07ff() { return xffffffff().srl16( 5); } __forceinline static GSVector4i x0fff() { return xffffffff().srl16( 4); } __forceinline static GSVector4i x1fff() { return xffffffff().srl16( 3); } __forceinline static GSVector4i x3fff() { return xffffffff().srl16( 2); } __forceinline static GSVector4i x7fff() { return xffffffff().srl16( 1); } __forceinline static GSVector4i x8000() { return xffffffff().sll16(15); } __forceinline static GSVector4i xc000() { return xffffffff().sll16(14); } __forceinline static GSVector4i xe000() { return xffffffff().sll16(13); } __forceinline static GSVector4i xf000() { return xffffffff().sll16(12); } __forceinline static GSVector4i xf800() { return xffffffff().sll16(11); } __forceinline static GSVector4i xfc00() { return xffffffff().sll16(10); } __forceinline static GSVector4i xfe00() { return xffffffff().sll16( 9); } __forceinline static GSVector4i xff00() { return xffffffff().sll16( 8); } __forceinline static GSVector4i xff80() { return xffffffff().sll16( 7); } __forceinline static GSVector4i xffc0() { return xffffffff().sll16( 6); } __forceinline static GSVector4i xffe0() { return xffffffff().sll16( 5); } __forceinline static GSVector4i xfff0() { return xffffffff().sll16( 4); } __forceinline static GSVector4i xfff8() { return xffffffff().sll16( 3); } __forceinline static GSVector4i xfffc() { return xffffffff().sll16( 2); } __forceinline static GSVector4i xfffe() { return xffffffff().sll16( 1); } __forceinline static GSVector4i xffffffff(const GSVector4i& v) { return v == v; } __forceinline static GSVector4i x00000001(const GSVector4i& v) { return xffffffff(v).srl32(31); } __forceinline static GSVector4i x00000003(const GSVector4i& v) { return xffffffff(v).srl32(30); } __forceinline static GSVector4i x00000007(const GSVector4i& v) { return xffffffff(v).srl32(29); } __forceinline static GSVector4i x0000000f(const GSVector4i& v) { return xffffffff(v).srl32(28); } __forceinline static GSVector4i x0000001f(const GSVector4i& v) { return xffffffff(v).srl32(27); } __forceinline static GSVector4i x0000003f(const GSVector4i& v) { return xffffffff(v).srl32(26); } __forceinline static GSVector4i x0000007f(const GSVector4i& v) { return xffffffff(v).srl32(25); } __forceinline static GSVector4i x000000ff(const GSVector4i& v) { return xffffffff(v).srl32(24); } __forceinline static GSVector4i x000001ff(const GSVector4i& v) { return xffffffff(v).srl32(23); } __forceinline static GSVector4i x000003ff(const GSVector4i& v) { return xffffffff(v).srl32(22); } __forceinline static GSVector4i x000007ff(const GSVector4i& v) { return xffffffff(v).srl32(21); } __forceinline static GSVector4i x00000fff(const GSVector4i& v) { return xffffffff(v).srl32(20); } __forceinline static GSVector4i x00001fff(const GSVector4i& v) { return xffffffff(v).srl32(19); } __forceinline static GSVector4i x00003fff(const GSVector4i& v) { return xffffffff(v).srl32(18); } __forceinline static GSVector4i x00007fff(const GSVector4i& v) { return xffffffff(v).srl32(17); } __forceinline static GSVector4i x0000ffff(const GSVector4i& v) { return xffffffff(v).srl32(16); } __forceinline static GSVector4i x0001ffff(const GSVector4i& v) { return xffffffff(v).srl32(15); } __forceinline static GSVector4i x0003ffff(const GSVector4i& v) { return xffffffff(v).srl32(14); } __forceinline static GSVector4i x0007ffff(const GSVector4i& v) { return xffffffff(v).srl32(13); } __forceinline static GSVector4i x000fffff(const GSVector4i& v) { return xffffffff(v).srl32(12); } __forceinline static GSVector4i x001fffff(const GSVector4i& v) { return xffffffff(v).srl32(11); } __forceinline static GSVector4i x003fffff(const GSVector4i& v) { return xffffffff(v).srl32(10); } __forceinline static GSVector4i x007fffff(const GSVector4i& v) { return xffffffff(v).srl32( 9); } __forceinline static GSVector4i x00ffffff(const GSVector4i& v) { return xffffffff(v).srl32( 8); } __forceinline static GSVector4i x01ffffff(const GSVector4i& v) { return xffffffff(v).srl32( 7); } __forceinline static GSVector4i x03ffffff(const GSVector4i& v) { return xffffffff(v).srl32( 6); } __forceinline static GSVector4i x07ffffff(const GSVector4i& v) { return xffffffff(v).srl32( 5); } __forceinline static GSVector4i x0fffffff(const GSVector4i& v) { return xffffffff(v).srl32( 4); } __forceinline static GSVector4i x1fffffff(const GSVector4i& v) { return xffffffff(v).srl32( 3); } __forceinline static GSVector4i x3fffffff(const GSVector4i& v) { return xffffffff(v).srl32( 2); } __forceinline static GSVector4i x7fffffff(const GSVector4i& v) { return xffffffff(v).srl32( 1); } __forceinline static GSVector4i x80000000(const GSVector4i& v) { return xffffffff(v).sll32(31); } __forceinline static GSVector4i xc0000000(const GSVector4i& v) { return xffffffff(v).sll32(30); } __forceinline static GSVector4i xe0000000(const GSVector4i& v) { return xffffffff(v).sll32(29); } __forceinline static GSVector4i xf0000000(const GSVector4i& v) { return xffffffff(v).sll32(28); } __forceinline static GSVector4i xf8000000(const GSVector4i& v) { return xffffffff(v).sll32(27); } __forceinline static GSVector4i xfc000000(const GSVector4i& v) { return xffffffff(v).sll32(26); } __forceinline static GSVector4i xfe000000(const GSVector4i& v) { return xffffffff(v).sll32(25); } __forceinline static GSVector4i xff000000(const GSVector4i& v) { return xffffffff(v).sll32(24); } __forceinline static GSVector4i xff800000(const GSVector4i& v) { return xffffffff(v).sll32(23); } __forceinline static GSVector4i xffc00000(const GSVector4i& v) { return xffffffff(v).sll32(22); } __forceinline static GSVector4i xffe00000(const GSVector4i& v) { return xffffffff(v).sll32(21); } __forceinline static GSVector4i xfff00000(const GSVector4i& v) { return xffffffff(v).sll32(20); } __forceinline static GSVector4i xfff80000(const GSVector4i& v) { return xffffffff(v).sll32(19); } __forceinline static GSVector4i xfffc0000(const GSVector4i& v) { return xffffffff(v).sll32(18); } __forceinline static GSVector4i xfffe0000(const GSVector4i& v) { return xffffffff(v).sll32(17); } __forceinline static GSVector4i xffff0000(const GSVector4i& v) { return xffffffff(v).sll32(16); } __forceinline static GSVector4i xffff8000(const GSVector4i& v) { return xffffffff(v).sll32(15); } __forceinline static GSVector4i xffffc000(const GSVector4i& v) { return xffffffff(v).sll32(14); } __forceinline static GSVector4i xffffe000(const GSVector4i& v) { return xffffffff(v).sll32(13); } __forceinline static GSVector4i xfffff000(const GSVector4i& v) { return xffffffff(v).sll32(12); } __forceinline static GSVector4i xfffff800(const GSVector4i& v) { return xffffffff(v).sll32(11); } __forceinline static GSVector4i xfffffc00(const GSVector4i& v) { return xffffffff(v).sll32(10); } __forceinline static GSVector4i xfffffe00(const GSVector4i& v) { return xffffffff(v).sll32( 9); } __forceinline static GSVector4i xffffff00(const GSVector4i& v) { return xffffffff(v).sll32( 8); } __forceinline static GSVector4i xffffff80(const GSVector4i& v) { return xffffffff(v).sll32( 7); } __forceinline static GSVector4i xffffffc0(const GSVector4i& v) { return xffffffff(v).sll32( 6); } __forceinline static GSVector4i xffffffe0(const GSVector4i& v) { return xffffffff(v).sll32( 5); } __forceinline static GSVector4i xfffffff0(const GSVector4i& v) { return xffffffff(v).sll32( 4); } __forceinline static GSVector4i xfffffff8(const GSVector4i& v) { return xffffffff(v).sll32( 3); } __forceinline static GSVector4i xfffffffc(const GSVector4i& v) { return xffffffff(v).sll32( 2); } __forceinline static GSVector4i xfffffffe(const GSVector4i& v) { return xffffffff(v).sll32( 1); } __forceinline static GSVector4i x0001(const GSVector4i& v) { return xffffffff(v).srl16(15); } __forceinline static GSVector4i x0003(const GSVector4i& v) { return xffffffff(v).srl16(14); } __forceinline static GSVector4i x0007(const GSVector4i& v) { return xffffffff(v).srl16(13); } __forceinline static GSVector4i x000f(const GSVector4i& v) { return xffffffff(v).srl16(12); } __forceinline static GSVector4i x001f(const GSVector4i& v) { return xffffffff(v).srl16(11); } __forceinline static GSVector4i x003f(const GSVector4i& v) { return xffffffff(v).srl16(10); } __forceinline static GSVector4i x007f(const GSVector4i& v) { return xffffffff(v).srl16( 9); } __forceinline static GSVector4i x00ff(const GSVector4i& v) { return xffffffff(v).srl16( 8); } __forceinline static GSVector4i x01ff(const GSVector4i& v) { return xffffffff(v).srl16( 7); } __forceinline static GSVector4i x03ff(const GSVector4i& v) { return xffffffff(v).srl16( 6); } __forceinline static GSVector4i x07ff(const GSVector4i& v) { return xffffffff(v).srl16( 5); } __forceinline static GSVector4i x0fff(const GSVector4i& v) { return xffffffff(v).srl16( 4); } __forceinline static GSVector4i x1fff(const GSVector4i& v) { return xffffffff(v).srl16( 3); } __forceinline static GSVector4i x3fff(const GSVector4i& v) { return xffffffff(v).srl16( 2); } __forceinline static GSVector4i x7fff(const GSVector4i& v) { return xffffffff(v).srl16( 1); } __forceinline static GSVector4i x8000(const GSVector4i& v) { return xffffffff(v).sll16(15); } __forceinline static GSVector4i xc000(const GSVector4i& v) { return xffffffff(v).sll16(14); } __forceinline static GSVector4i xe000(const GSVector4i& v) { return xffffffff(v).sll16(13); } __forceinline static GSVector4i xf000(const GSVector4i& v) { return xffffffff(v).sll16(12); } __forceinline static GSVector4i xf800(const GSVector4i& v) { return xffffffff(v).sll16(11); } __forceinline static GSVector4i xfc00(const GSVector4i& v) { return xffffffff(v).sll16(10); } __forceinline static GSVector4i xfe00(const GSVector4i& v) { return xffffffff(v).sll16( 9); } __forceinline static GSVector4i xff00(const GSVector4i& v) { return xffffffff(v).sll16( 8); } __forceinline static GSVector4i xff80(const GSVector4i& v) { return xffffffff(v).sll16( 7); } __forceinline static GSVector4i xffc0(const GSVector4i& v) { return xffffffff(v).sll16( 6); } __forceinline static GSVector4i xffe0(const GSVector4i& v) { return xffffffff(v).sll16( 5); } __forceinline static GSVector4i xfff0(const GSVector4i& v) { return xffffffff(v).sll16( 4); } __forceinline static GSVector4i xfff8(const GSVector4i& v) { return xffffffff(v).sll16( 3); } __forceinline static GSVector4i xfffc(const GSVector4i& v) { return xffffffff(v).sll16( 2); } __forceinline static GSVector4i xfffe(const GSVector4i& v) { return xffffffff(v).sll16( 1); } __forceinline static GSVector4i xff(int n) { return m_xff[n]; } __forceinline static GSVector4i x0f(int n) { return m_x0f[n]; } };