rpcsx/rx/include/rx/simd.hpp
DH 7115851c82 Initial new PPU interpreter implementation
Replaced dangerous offset32 usages with safe OFFSET_OF macro
2025-04-24 13:41:04 +03:00

2237 lines
63 KiB
C++

#pragma once
#include "asm.hpp"
#include "types.hpp"
#include "v128.hpp"
#if defined(ARCH_X64)
#ifdef _MSC_VER
#include <intrin.h>
#else
#include <x86intrin.h>
#endif
#include <emmintrin.h>
#include <immintrin.h>
#endif
#if defined(ARCH_ARM64)
#include <arm_neon.h>
#endif
#include <algorithm>
#include <cmath>
#if defined(__clang__)
#pragma clang diagnostic push
#pragma clang diagnostic ignored "-Wunused-parameter"
#elif defined(__GNUC__)
#pragma GCC diagnostic push
#pragma GCC diagnostic ignored "-Wunused-parameter"
#endif
namespace rx {
inline v128 gv_select8(const v128 &_cmp, const v128 &_true, const v128 &_false);
inline v128 gv_signselect8(const v128 &bits, const v128 &_true,
const v128 &_false);
inline v128 gv_select16(const v128 &_cmp, const v128 &_true,
const v128 &_false);
inline v128 gv_select32(const v128 &_cmp, const v128 &_true,
const v128 &_false);
inline v128 gv_selectfs(const v128 &_cmp, const v128 &_true,
const v128 &_false);
inline void gv_set_zeroing_denormals() {
#if defined(ARCH_X64)
u32 cr = _mm_getcsr();
cr = (cr & ~_MM_FLUSH_ZERO_MASK) | _MM_FLUSH_ZERO_ON;
cr = (cr & ~_MM_DENORMALS_ZERO_MASK) | _MM_DENORMALS_ZERO_ON;
cr = (cr | _MM_MASK_INVALID);
_mm_setcsr(cr);
#elif defined(ARCH_ARM64)
u64 cr;
__asm__ volatile("mrs %0, FPCR" : "=r"(cr));
cr |= 0x1000000ull;
__asm__ volatile("msr FPCR, %0" ::"r"(cr));
#else
#error "Not implemented"
#endif
}
inline void gv_unset_zeroing_denormals() {
#if defined(ARCH_X64)
u32 cr = _mm_getcsr();
cr = (cr & ~_MM_FLUSH_ZERO_MASK) | _MM_FLUSH_ZERO_OFF;
cr = (cr & ~_MM_DENORMALS_ZERO_MASK) | _MM_DENORMALS_ZERO_OFF;
cr = (cr | _MM_MASK_INVALID);
_mm_setcsr(cr);
#elif defined(ARCH_ARM64)
u64 cr;
__asm__ volatile("mrs %0, FPCR" : "=r"(cr));
cr &= ~0x1000000ull;
__asm__ volatile("msr FPCR, %0" ::"r"(cr));
#else
#error "Not implemented"
#endif
}
inline v128 gv_bcst8(u8 value) {
#if defined(ARCH_X64)
return _mm_set1_epi8(value);
#elif defined(ARCH_ARM64)
return vdupq_n_s8(value);
#endif
}
inline v128 gv_bcst16(u16 value) {
#if defined(ARCH_X64)
return _mm_set1_epi16(value);
#elif defined(ARCH_ARM64)
return vdupq_n_s16(value);
#endif
}
inline v128 gv_bcst32(u32 value) {
#if defined(ARCH_X64)
return _mm_set1_epi32(value);
#elif defined(ARCH_ARM64)
return vdupq_n_s32(value);
#endif
}
inline v128 gv_bcst64(u64 value) {
#if defined(ARCH_X64)
return _mm_set1_epi64x(value);
#elif defined(ARCH_ARM64)
return vdupq_n_s64(value);
#endif
}
inline v128 gv_bcstfs(f32 value) {
#if defined(ARCH_X64)
return _mm_set1_ps(value);
#elif defined(ARCH_ARM64)
return vdupq_n_f32(value);
#endif
}
inline v128 gv_and32(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_and_si128(a, b);
#elif defined(ARCH_ARM64)
return vandq_s32(a, b);
#endif
}
inline v128 gv_andfs(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_and_ps(a, b);
#elif defined(ARCH_ARM64)
return vandq_s32(a, b);
#endif
}
inline v128 gv_andn32(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_andnot_si128(a, b);
#elif defined(ARCH_ARM64)
return vbicq_s32(b, a);
#endif
}
inline v128 gv_andnfs(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_andnot_ps(a, b);
#elif defined(ARCH_ARM64)
return vbicq_s32(b, a);
#endif
}
inline v128 gv_or32(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_or_si128(a, b);
#elif defined(ARCH_ARM64)
return vorrq_s32(a, b);
#endif
}
inline v128 gv_orfs(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_or_ps(a, b);
#elif defined(ARCH_ARM64)
return vorrq_s32(a, b);
#endif
}
inline v128 gv_xor32(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_xor_si128(a, b);
#elif defined(ARCH_ARM64)
return veorq_s32(a, b);
#endif
}
inline v128 gv_xorfs(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_xor_ps(a, b);
#elif defined(ARCH_ARM64)
return veorq_s32(a, b);
#endif
}
inline v128 gv_not32(const v128 &a) {
#if defined(ARCH_X64)
return _mm_xor_si128(a, _mm_set1_epi32(-1));
#elif defined(ARCH_ARM64)
return vmvnq_u32(a);
#endif
}
inline v128 gv_notfs(const v128 &a) {
#if defined(ARCH_X64)
return _mm_xor_ps(a, _mm_castsi128_ps(_mm_set1_epi32(-1)));
#elif defined(ARCH_ARM64)
return vmvnq_u32(a);
#endif
}
inline v128 gv_shl16(const v128 &a, u32 count) {
if (count >= 16)
return v128{};
#if defined(ARCH_X64)
return _mm_slli_epi16(a, count);
#elif defined(ARCH_ARM64)
return vshlq_s16(a, vdupq_n_s16(count));
#endif
}
inline v128 gv_shl32(const v128 &a, u32 count) {
if (count >= 32)
return v128{};
#if defined(ARCH_X64)
return _mm_slli_epi32(a, count);
#elif defined(ARCH_ARM64)
return vshlq_s32(a, vdupq_n_s32(count));
#endif
}
inline v128 gv_shl64(const v128 &a, u32 count) {
if (count >= 64)
return v128{};
#if defined(ARCH_X64)
return _mm_slli_epi64(a, count);
#elif defined(ARCH_ARM64)
return vshlq_s64(a, vdupq_n_s64(count));
#endif
}
inline v128 gv_shr16(const v128 &a, u32 count) {
if (count >= 16)
return v128{};
#if defined(ARCH_X64)
return _mm_srli_epi16(a, count);
#elif defined(ARCH_ARM64)
return vshlq_u16(a, vdupq_n_s16(0 - count));
#endif
}
inline v128 gv_shr32(const v128 &a, u32 count) {
if (count >= 32)
return v128{};
#if defined(ARCH_X64)
return _mm_srli_epi32(a, count);
#elif defined(ARCH_ARM64)
return vshlq_u32(a, vdupq_n_s32(0 - count));
#endif
}
inline v128 gv_shr64(const v128 &a, u32 count) {
if (count >= 64)
return v128{};
#if defined(ARCH_X64)
return _mm_srli_epi64(a, count);
#elif defined(ARCH_ARM64)
return vshlq_u64(a, vdupq_n_s64(0 - count));
#endif
}
inline v128 gv_sar16(const v128 &a, u32 count) {
if (count >= 16)
count = 15;
#if defined(ARCH_X64)
return _mm_srai_epi16(a, count);
#elif defined(ARCH_ARM64)
return vshlq_s16(a, vdupq_n_s16(0 - count));
#endif
}
inline v128 gv_sar32(const v128 &a, u32 count) {
if (count >= 32)
count = 31;
#if defined(ARCH_X64)
return _mm_srai_epi32(a, count);
#elif defined(ARCH_ARM64)
return vshlq_s32(a, vdupq_n_s32(0 - count));
#endif
}
inline v128 gv_sar64(const v128 &a, u32 count) {
if (count >= 64)
count = 63;
#if defined(__AVX512VL__)
return _mm_srai_epi64(a, count);
#elif defined(__SSE2__) && !defined(_M_X64)
return static_cast<__v2di>(a) >> count;
#elif defined(ARCH_ARM64)
return vshlq_s64(a, vdupq_n_s64(0 - count));
#else
v128 r;
r._s64[0] = a._s64[0] >> count;
r._s64[1] = a._s64[1] >> count;
return r;
#endif
}
inline v128 gv_add8(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_add_epi8(a, b);
#elif defined(ARCH_ARM64)
return vaddq_s8(a, b);
#endif
}
inline v128 gv_add16(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_add_epi16(a, b);
#elif defined(ARCH_ARM64)
return vaddq_s16(a, b);
#endif
}
inline v128 gv_add32(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_add_epi32(a, b);
#elif defined(ARCH_ARM64)
return vaddq_s32(a, b);
#endif
}
inline v128 gv_add64(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_add_epi64(a, b);
#elif defined(ARCH_ARM64)
return vaddq_s64(a, b);
#endif
}
inline v128 gv_adds_s8(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_adds_epi8(a, b);
#elif defined(ARCH_ARM64)
return vqaddq_s8(a, b);
#endif
}
inline v128 gv_adds_s16(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_adds_epi16(a, b);
#elif defined(ARCH_ARM64)
return vqaddq_s16(a, b);
#endif
}
inline v128 gv_adds_s32(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
const v128 s = _mm_add_epi32(a, b);
const v128 m = (a ^ s) & (b ^ s); // overflow bit
const v128 x = _mm_srai_epi32(m, 31); // saturation mask
const v128 y =
_mm_srai_epi32(_mm_and_si128(s, m), 31); // positive saturation mask
return _mm_xor_si128(_mm_xor_si128(_mm_srli_epi32(x, 1), y),
_mm_or_si128(s, x));
#elif defined(ARCH_ARM64)
return vqaddq_s32(a, b);
#endif
}
inline v128 gv_addus_u8(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_adds_epu8(a, b);
#elif defined(ARCH_ARM64)
return vqaddq_u8(a, b);
#endif
}
inline v128 gv_addus_u16(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_adds_epu16(a, b);
#elif defined(ARCH_ARM64)
return vqaddq_u16(a, b);
#endif
}
inline v128 gv_addus_u32(const v128 &a, const v128 &b) {
#if defined(__SSE4_1__)
return _mm_add_epi32(a, _mm_min_epu32(~a, b));
#elif defined(ARCH_X64)
const v128 s = _mm_add_epi32(a, b);
return _mm_or_si128(s,
_mm_cmpgt_epi32(_mm_xor_si128(b, _mm_set1_epi32(smin)),
_mm_xor_si128(a, _mm_set1_epi32(smax))));
#elif defined(ARCH_ARM64)
return vqaddq_u32(a, b);
#endif
}
inline v128 gv_addfs(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_add_ps(a, b);
#elif defined(ARCH_ARM64)
return vaddq_f32(a, b);
#endif
}
inline v128 gv_addfd(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_add_pd(a, b);
#elif defined(ARCH_ARM64)
return vaddq_f64(a, b);
#endif
}
inline v128 gv_sub8(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_sub_epi8(a, b);
#elif defined(ARCH_ARM64)
return vsubq_s8(a, b);
#endif
}
inline v128 gv_sub16(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_sub_epi16(a, b);
#elif defined(ARCH_ARM64)
return vsubq_s16(a, b);
#endif
}
inline v128 gv_sub32(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_sub_epi32(a, b);
#elif defined(ARCH_ARM64)
return vsubq_s32(a, b);
#endif
}
inline v128 gv_sub64(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_sub_epi64(a, b);
#elif defined(ARCH_ARM64)
return vsubq_s64(a, b);
#endif
}
inline v128 gv_subs_s8(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_subs_epi8(a, b);
#elif defined(ARCH_ARM64)
return vqsubq_s8(a, b);
#endif
}
inline v128 gv_subs_s16(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_subs_epi16(a, b);
#elif defined(ARCH_ARM64)
return vqsubq_s16(a, b);
#endif
}
inline v128 gv_subs_s32(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
const v128 d = _mm_sub_epi32(a, b);
const v128 m = (a ^ b) & (a ^ d); // overflow bit
const v128 x = _mm_srai_epi32(m, 31);
return _mm_or_si128(_mm_andnot_si128(x, d),
_mm_and_si128(x, _mm_xor_si128(_mm_srli_epi32(x, 1),
_mm_srai_epi32(a, 31))));
#elif defined(ARCH_ARM64)
return vqsubq_s32(a, b);
#endif
}
inline v128 gv_subus_u8(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_subs_epu8(a, b);
#elif defined(ARCH_ARM64)
return vqsubq_u8(a, b);
#endif
}
inline v128 gv_subus_u16(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_subs_epu16(a, b);
#elif defined(ARCH_ARM64)
return vqsubq_u16(a, b);
#endif
}
inline v128 gv_subus_u32(const v128 &a, const v128 &b) {
#if defined(__SSE4_1__)
return _mm_sub_epi32(a, _mm_min_epu32(a, b));
#elif defined(ARCH_X64)
const auto sign = _mm_set1_epi32(smin);
return _mm_andnot_si128(
_mm_cmpgt_epi32(_mm_xor_si128(b, sign), _mm_xor_si128(a, sign)),
_mm_sub_epi32(a, b));
#elif defined(ARCH_ARM64)
return vqsubq_u32(a, b);
#endif
}
inline v128 gv_subfs(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_sub_ps(a, b);
#elif defined(ARCH_ARM64)
return vsubq_f32(a, b);
#endif
}
inline v128 gv_subfd(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_sub_pd(a, b);
#elif defined(ARCH_ARM64)
return vsubq_f64(a, b);
#endif
}
inline v128 gv_maxu8(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_max_epu8(a, b);
#elif defined(ARCH_ARM64)
return vmaxq_u8(a, b);
#endif
}
inline v128 gv_maxu16(const v128 &a, const v128 &b) {
#if defined(__SSE4_1__)
return _mm_max_epu16(a, b);
#elif defined(ARCH_X64)
return _mm_add_epi16(_mm_subs_epu16(a, b), b);
#elif defined(ARCH_ARM64)
return vmaxq_u16(a, b);
#endif
}
inline v128 gv_maxu32(const v128 &a, const v128 &b) {
#if defined(__SSE4_1__)
return _mm_max_epu32(a, b);
#elif defined(ARCH_X64)
const __m128i s = _mm_set1_epi32(smin);
const __m128i m = _mm_cmpgt_epi32(_mm_xor_si128(a, s), _mm_xor_si128(b, s));
return _mm_or_si128(_mm_and_si128(m, a), _mm_andnot_si128(m, b));
#elif defined(ARCH_ARM64)
return vmaxq_u32(a, b);
#endif
}
inline v128 gv_maxs8(const v128 &a, const v128 &b) {
#if defined(__SSE4_1__)
return _mm_max_epi8(a, b);
#elif defined(ARCH_X64)
const __m128i m = _mm_cmpgt_epi8(a, b);
return _mm_or_si128(_mm_and_si128(m, a), _mm_andnot_si128(m, b));
#elif defined(ARCH_ARM64)
return vmaxq_s8(a, b);
#endif
}
inline v128 gv_maxs16(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_max_epi16(a, b);
#elif defined(ARCH_ARM64)
return vmaxq_s16(a, b);
#endif
}
inline v128 gv_maxs32(const v128 &a, const v128 &b) {
#if defined(__SSE4_1__)
return _mm_max_epi32(a, b);
#elif defined(ARCH_X64)
const __m128i m = _mm_cmpgt_epi32(a, b);
return _mm_or_si128(_mm_and_si128(m, a), _mm_andnot_si128(m, b));
#elif defined(ARCH_ARM64)
return vmaxq_s32(a, b);
#endif
}
inline v128 gv_maxfs(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_and_ps(_mm_max_ps(a, b), _mm_max_ps(b, a));
#elif defined(ARCH_ARM64)
return vmaxq_f32(a, b);
#endif
}
inline v128 gv_minu8(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_min_epu8(a, b);
#elif defined(ARCH_ARM64)
return vminq_u8(a, b);
#endif
}
inline v128 gv_minu16(const v128 &a, const v128 &b) {
#if defined(__SSE4_1__)
return _mm_min_epu16(a, b);
#elif defined(ARCH_X64)
return _mm_sub_epi16(a, _mm_subs_epu16(a, b));
#elif defined(ARCH_ARM64)
return vminq_u16(a, b);
#endif
}
inline v128 gv_minu32(const v128 &a, const v128 &b) {
#if defined(__SSE4_1__)
return _mm_min_epu32(a, b);
#elif defined(ARCH_X64)
const __m128i s = _mm_set1_epi32(smin);
const __m128i m = _mm_cmpgt_epi32(_mm_xor_si128(a, s), _mm_xor_si128(b, s));
return _mm_or_si128(_mm_andnot_si128(m, a), _mm_and_si128(m, b));
#elif defined(ARCH_ARM64)
return vminq_u32(a, b);
#endif
}
inline v128 gv_mins8(const v128 &a, const v128 &b) {
#if defined(__SSE4_1__)
return _mm_min_epi8(a, b);
#elif defined(ARCH_X64)
const __m128i m = _mm_cmpgt_epi8(a, b);
return _mm_or_si128(_mm_andnot_si128(m, a), _mm_and_si128(m, b));
#elif defined(ARCH_ARM64)
return vminq_s8(a, b);
#endif
}
inline v128 gv_mins16(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_min_epi16(a, b);
#elif defined(ARCH_ARM64)
return vminq_s16(a, b);
#endif
}
inline v128 gv_mins32(const v128 &a, const v128 &b) {
#if defined(__SSE4_1__)
return _mm_min_epi32(a, b);
#elif defined(ARCH_X64)
const __m128i m = _mm_cmpgt_epi32(a, b);
return _mm_or_si128(_mm_andnot_si128(m, a), _mm_and_si128(m, b));
#elif defined(ARCH_ARM64)
return vminq_s32(a, b);
#endif
}
inline v128 gv_minfs(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_or_ps(_mm_min_ps(a, b), _mm_min_ps(b, a));
#elif defined(ARCH_ARM64)
return vminq_f32(a, b);
#endif
}
inline v128 gv_eq8(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_cmpeq_epi8(a, b);
#elif defined(ARCH_ARM64)
return vceqq_s8(a, b);
#endif
}
inline v128 gv_eq16(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_cmpeq_epi16(a, b);
#elif defined(ARCH_ARM64)
return vceqq_s16(a, b);
#endif
}
inline v128 gv_eq32(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_cmpeq_epi32(a, b);
#elif defined(ARCH_ARM64)
return vceqq_s32(a, b);
#endif
}
// Ordered and equal
inline v128 gv_eqfs(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_cmpeq_ps(a, b);
#elif defined(ARCH_ARM64)
return vceqq_f32(a, b);
#endif
}
// Unordered or not equal
inline v128 gv_neqfs(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_cmpneq_ps(a, b);
#elif defined(ARCH_ARM64)
return ~vceqq_f32(a, b);
#endif
}
inline v128 gv_gtu8(const v128 &a, const v128 &b) {
#if defined(__AVX512VL__) && defined(__AVX512BW__)
return _mm_movm_epi8(_mm_cmpgt_epu8_mask(a, b));
#elif defined(ARCH_X64)
return _mm_cmpeq_epi8(_mm_cmpeq_epi8(a, _mm_min_epu8(a, b)),
_mm_setzero_si128());
#elif defined(ARCH_ARM64)
return vcgtq_u8(a, b);
#endif
}
inline v128 gv_gtu16(const v128 &a, const v128 &b) {
#if defined(__AVX512VL__) && defined(__AVX512BW__)
return _mm_movm_epi16(_mm_cmpgt_epu16_mask(a, b));
#elif defined(__SSE4_1__)
return _mm_cmpeq_epi16(_mm_cmpeq_epi16(a, _mm_min_epu16(a, b)),
_mm_setzero_si128());
#elif defined(ARCH_X64)
return _mm_cmpeq_epi16(
_mm_cmpeq_epi16(_mm_subs_epu16(a, b), _mm_setzero_si128()),
_mm_setzero_si128());
#elif defined(ARCH_ARM64)
return vcgtq_u16(a, b);
#endif
}
inline v128 gv_gtu32(const v128 &a, const v128 &b) {
#if defined(__AVX512VL__) && defined(__AVX512DQ__)
return _mm_movm_epi32(_mm_cmpgt_epu32_mask(a, b));
#elif defined(__SSE4_1__)
return _mm_cmpeq_epi32(_mm_cmpeq_epi32(a, _mm_min_epu32(a, b)),
_mm_setzero_si128());
#elif defined(ARCH_X64)
const auto sign = _mm_set1_epi32(smin);
return _mm_cmpgt_epi32(_mm_xor_si128(a, sign), _mm_xor_si128(b, sign));
#elif defined(ARCH_ARM64)
return vcgtq_u32(a, b);
#endif
}
// Ordered and greater than
inline v128 gv_gtfs(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_cmpgt_ps(a, b);
#elif defined(ARCH_ARM64)
return vcgtq_f32(a, b);
#endif
}
// Ordered and less than
inline v128 gv_ltfs(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_cmplt_ps(a, b);
#elif defined(ARCH_ARM64)
return vcltq_f32(a, b);
#endif
}
// Unordered or less or equal
inline v128 gv_ngtfs(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_cmpngt_ps(a, b);
#elif defined(ARCH_ARM64)
return ~vcgtq_f32(a, b);
#endif
}
// Unordered or greater or equal
inline v128 gv_nlefs(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_cmpnle_ps(a, b);
#elif defined(ARCH_ARM64)
return ~vcleq_f32(a, b);
#endif
}
inline v128 gv_geu8(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_cmpeq_epi8(b, _mm_min_epu8(a, b));
#elif defined(ARCH_ARM64)
return vcgeq_u8(a, b);
#endif
}
inline v128 gv_geu16(const v128 &a, const v128 &b) {
#if defined(__SSE4_1__)
return _mm_cmpeq_epi16(b, _mm_min_epu16(a, b));
#elif defined(ARCH_X64)
return _mm_cmpeq_epi16(_mm_subs_epu16(b, a), _mm_setzero_si128());
#elif defined(ARCH_ARM64)
return vcgeq_u16(a, b);
#endif
}
inline v128 gv_geu32(const v128 &a, const v128 &b) {
#if defined(__SSE4_1__)
return _mm_cmpeq_epi32(b, _mm_min_epu32(a, b));
#elif defined(ARCH_X64)
const auto sign = _mm_set1_epi32(smin);
return _mm_cmpeq_epi32(
_mm_cmpgt_epi32(_mm_xor_si128(b, sign), _mm_xor_si128(a, sign)),
_mm_setzero_si128());
#elif defined(ARCH_ARM64)
return vcgeq_u32(a, b);
#endif
}
// Ordered and not less than
inline v128 gv_gefs(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_cmpge_ps(a, b);
#elif defined(ARCH_ARM64)
return vcgeq_f32(a, b);
#endif
}
// Unordered or less than
inline v128 gv_ngefs(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_cmpnge_ps(a, b);
#elif defined(ARCH_ARM64)
return ~vcgeq_f32(a, b);
#endif
}
inline v128 gv_gts8(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_cmpgt_epi8(a, b);
#elif defined(ARCH_ARM64)
return vcgtq_s8(a, b);
#endif
}
inline v128 gv_gts16(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_cmpgt_epi16(a, b);
#elif defined(ARCH_ARM64)
return vcgtq_s16(a, b);
#endif
}
inline v128 gv_gts32(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_cmpgt_epi32(a, b);
#elif defined(ARCH_ARM64)
return vcgtq_s32(a, b);
#endif
}
inline v128 gv_avgu8(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_avg_epu8(a, b);
#elif defined(ARCH_ARM64)
return vrhaddq_u8(a, b);
#endif
}
inline v128 gv_avgu16(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_avg_epu16(a, b);
#elif defined(ARCH_ARM64)
return vrhaddq_u16(a, b);
#endif
}
inline v128 gv_avgu32(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
const auto ones = _mm_set1_epi32(-1);
const auto summ = gv_sub32(gv_add32(a, b), ones);
const auto carry = _mm_slli_epi32(gv_geu32(a, summ), 31);
return _mm_or_si128(carry, _mm_srli_epi32(summ, 1));
#elif defined(ARCH_ARM64)
return vrhaddq_u32(a, b);
#endif
}
inline v128 gv_avgs8(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
const v128 sign = _mm_set1_epi8(smin);
return gv_avgu8(a ^ sign, b ^ sign) ^ sign;
#elif defined(ARCH_ARM64)
return vrhaddq_s8(a, b);
#endif
}
inline v128 gv_avgs16(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
const v128 sign = _mm_set1_epi16(smin);
return gv_avgu16(a ^ sign, b ^ sign) ^ sign;
#elif defined(ARCH_ARM64)
return vrhaddq_s16(a, b);
#endif
}
inline v128 gv_avgs32(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
const v128 sign = _mm_set1_epi32(smin);
return gv_avgu32(a ^ sign, b ^ sign) ^ sign;
#elif defined(ARCH_ARM64)
return vrhaddq_s32(a, b);
#endif
}
inline v128 gv_divfs(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_div_ps(a, b);
#elif defined(ARCH_ARM64)
return vdivq_f32(a, b);
#endif
}
inline v128 gv_sqrtfs(const v128 &a) {
#if defined(ARCH_X64)
return _mm_sqrt_ps(a);
#elif defined(ARCH_ARM64)
return vsqrtq_f32(a);
#endif
}
inline v128 gv_fmafs(const v128 &a, const v128 &b, const v128 &c) {
#if defined(ARCH_X64) && defined(__FMA__)
return _mm_fmadd_ps(a, b, c);
#elif defined(__FMA4__)
return _mm_macc_ps(a, b, c);
#elif defined(ARCH_X64)
// This is inaccurate implementation
#ifdef __AVX__
const __m128 r = _mm256_cvtpd_ps(
_mm256_add_pd(_mm256_mul_pd(_mm256_cvtps_pd(a), _mm256_cvtps_pd(b)),
_mm256_cvtps_pd(c)));
#else
const __m128d a0 = _mm_cvtps_pd(a);
const __m128d a1 = _mm_cvtps_pd(_mm_movehl_ps(a, a));
const __m128d b0 = _mm_cvtps_pd(b);
const __m128d b1 = _mm_cvtps_pd(_mm_movehl_ps(b, b));
const __m128d c0 = _mm_cvtps_pd(c);
const __m128d c1 = _mm_cvtps_pd(_mm_movehl_ps(c, c));
const __m128d m0 = _mm_mul_pd(a0, b0);
const __m128d m1 = _mm_mul_pd(a1, b1);
const __m128d r0 = _mm_add_pd(m0, c0);
const __m128d r1 = _mm_add_pd(m1, c1);
const __m128 r = _mm_movelh_ps(_mm_cvtpd_ps(r0), _mm_cvtpd_ps(r1));
#endif
return r;
#elif defined(ARCH_ARM64)
return vfmaq_f32(c, a, b);
#else
v128 r;
for (int i = 0; i < 4; i++) {
r._f[i] = std::fmaf(a._f[i], b._f[i], c._f[i]);
}
return r;
#endif
}
// -> ssat((a * b * 2 + (c << 16) + 0x8000) >> 16)
inline v128 gv_rmuladds_hds16(const v128 &a, const v128 &b, const v128 &c) {
#if defined(ARCH_ARM64)
#if defined(__ARM_FEATURE_QRDMX)
return vqrdmlahq_s16(c, a, b);
#else
v128 d;
for (uint h = 0; h < 8; h++) {
s32 result = ((s32)a._s16[h] * (s32)b._s16[h]) + 0x4000;
result = (result >> 15) + (s32)c._s16[h];
if (result > INT16_MAX) {
d._s16[h] = (s16)INT16_MAX;
} else if (result < INT16_MIN) {
d._s16[h] = (s16)INT16_MIN;
} else {
d._s16[h] = (s16)result;
}
}
return d;
#endif
#elif defined(ARCH_X64)
const auto x80 =
_mm_set1_epi16(0x80); // 0x80 * 0x80 = 0x4000, add this to the product
const auto al = _mm_unpacklo_epi16(a, x80);
const auto ah = _mm_unpackhi_epi16(a, x80);
const auto bl = _mm_unpacklo_epi16(b, x80);
const auto bh = _mm_unpackhi_epi16(b, x80);
const auto ml = _mm_srai_epi32(_mm_madd_epi16(al, bl), 15);
const auto mh = _mm_srai_epi32(_mm_madd_epi16(ah, bh), 15);
const auto cl =
_mm_srai_epi32(_mm_unpacklo_epi16(_mm_setzero_si128(), c), 16);
const auto ch =
_mm_srai_epi32(_mm_unpackhi_epi16(_mm_setzero_si128(), c), 16);
const auto sl = _mm_add_epi32(ml, cl);
const auto sh = _mm_add_epi32(mh, ch);
return _mm_packs_epi32(sl, sh);
#endif
}
// -> ssat((a * b * 2 + 0x8000) >> 16)
inline v128 gv_rmuls_hds16(const v128 &a, const v128 &b) {
#if defined(ARCH_ARM64)
return vqrdmulhq_s16(a, b);
#elif defined(ARCH_X64)
const auto x80 =
_mm_set1_epi16(0x80); // 0x80 * 0x80 = 0x4000, add this to the product
const auto al = _mm_unpacklo_epi16(a, x80);
const auto ah = _mm_unpackhi_epi16(a, x80);
const auto bl = _mm_unpacklo_epi16(b, x80);
const auto bh = _mm_unpackhi_epi16(b, x80);
const auto ml = _mm_srai_epi32(_mm_madd_epi16(al, bl), 15);
const auto mh = _mm_srai_epi32(_mm_madd_epi16(ah, bh), 15);
return _mm_packs_epi32(ml, mh);
#endif
}
// -> ssat((a * b * 2) >> 16)
inline v128 gv_muls_hds16(const v128 &a, const v128 &b) {
#if defined(ARCH_ARM64)
return vqdmulhq_s16(a, b);
#elif defined(ARCH_X64)
const auto m = _mm_or_si128(_mm_srli_epi16(_mm_mullo_epi16(a, b), 15),
_mm_slli_epi16(_mm_mulhi_epi16(a, b), 1));
const auto s = _mm_cmpeq_epi16(
m, _mm_set1_epi16(-0x8000)); // detect special case (positive 0x8000)
return _mm_xor_si128(m, s);
#endif
}
inline v128 gv_muladd16(const v128 &a, const v128 &b, const v128 &c) {
#if defined(ARCH_X64)
return _mm_add_epi16(_mm_mullo_epi16(a, b), c);
#elif defined(ARCH_ARM64)
return vmlaq_s16(c, a, b);
#endif
}
inline v128 gv_mul16(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_mullo_epi16(a, b);
#elif defined(ARCH_ARM64)
return vmulq_s16(a, b);
#endif
}
inline v128 gv_mul32(const v128 &a, const v128 &b) {
#if defined(__SSE4_1__)
return _mm_mullo_epi32(a, b);
#elif defined(ARCH_X64)
const __m128i lows = _mm_shuffle_epi32(_mm_mul_epu32(a, b), 8);
const __m128i highs = _mm_shuffle_epi32(
_mm_mul_epu32(_mm_srli_epi64(a, 32), _mm_srli_epi64(b, 32)), 8);
return _mm_unpacklo_epi32(lows, highs);
#elif defined(ARCH_ARM64)
return vmulq_s32(a, b);
#endif
}
inline v128 gv_mulfs(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_mul_ps(a, b);
#elif defined(ARCH_ARM64)
return vmulq_f32(a, b);
#endif
}
inline v128 gv_mulfs(const v128 &a, f32 b) {
#if defined(ARCH_X64)
return _mm_mul_ps(a, _mm_set_ps1(b));
#elif defined(ARCH_ARM64)
return vmulq_n_f32(a, b);
#endif
}
inline v128 gv_hadds8x2(const v128 &a) {
#if defined(__SSSE3__)
return _mm_maddubs_epi16(_mm_set1_epi8(1), a);
#elif defined(ARCH_X64)
return _mm_add_epi16(_mm_srai_epi16(a, 8),
_mm_srai_epi16(_mm_slli_epi16(a, 8), 8));
#elif defined(ARCH_ARM64)
return vpaddlq_s8(a);
#endif
}
inline v128 gv_hadds8x4(const v128 &a, const v128 &c) {
#if (defined(__AVX512VL__) && defined(__AVX512VNNI__)) || defined(__AVXVNNI__)
return _mm_dpbusd_epi32(c, _mm_set1_epi8(1), a);
#elif defined(__SSSE3__)
return _mm_add_epi32(
_mm_madd_epi16(_mm_maddubs_epi16(_mm_set1_epi8(1), a), _mm_set1_epi16(1)),
c);
#elif defined(ARCH_X64)
return _mm_add_epi32(
_mm_madd_epi16(_mm_add_epi16(_mm_srai_epi16(a, 8),
_mm_srai_epi16(_mm_slli_epi16(a, 8), 8)),
_mm_set1_epi16(1)),
c);
#elif defined(ARCH_ARM64)
return vaddq_s32(vpaddlq_s16(vpaddlq_s8(a)), c);
#endif
}
inline v128 gv_haddu8x2(const v128 &a) {
#if defined(__SSSE3__)
return _mm_maddubs_epi16(a, _mm_set1_epi8(1));
#elif defined(ARCH_X64)
return _mm_add_epi16(_mm_srli_epi16(a, 8),
_mm_and_si128(a, _mm_set1_epi16(0x00ff)));
#elif defined(ARCH_ARM64)
return vpaddlq_u8(a);
#endif
}
inline v128 gv_haddu8x4(const v128 &a) {
#if (defined(__AVX512VL__) && defined(__AVX512VNNI__)) || defined(__AVXVNNI__)
return _mm_dpbusd_epi32(_mm_setzero_si128(), a, _mm_set1_epi8(1));
#elif defined(__SSSE3__)
return _mm_madd_epi16(_mm_maddubs_epi16(a, _mm_set1_epi8(1)),
_mm_set1_epi16(1));
#elif defined(ARCH_X64)
return _mm_madd_epi16(_mm_add_epi16(_mm_srli_epi16(a, 8),
_mm_and_si128(a, _mm_set1_epi16(0x00ff))),
_mm_set1_epi16(1));
#elif defined(ARCH_ARM64)
return vpaddlq_u16(vpaddlq_u8(a));
#endif
}
inline v128 gv_hadds16x2(const v128 &a, const v128 &c) {
#if (defined(__AVX512VL__) && defined(__AVX512VNNI__)) || defined(__AVXVNNI__)
return _mm_dpwssd_epi32(c, a, _mm_set1_epi8(1));
#elif defined(ARCH_X64)
return _mm_add_epi32(_mm_madd_epi16(a, _mm_set1_epi16(1)), c);
#elif defined(ARCH_ARM64)
return vaddq_s32(vpaddlq_s16(a), c);
#endif
}
// Unsigned bytes from a, signed bytes from b, 32-bit accumulator c
inline v128 gv_dotu8s8x4(const v128 &a, const v128 &b, const v128 &c) {
#if (defined(__AVX512VL__) && defined(__AVX512VNNI__)) || defined(__AVXVNNI__)
return _mm_dpbusd_epi32(c, a, b);
#elif defined(ARCH_X64)
const __m128i ah = _mm_srli_epi16(a, 8);
const __m128i al = _mm_and_si128(a, _mm_set1_epi16(0x00ff));
const __m128i bh = _mm_srai_epi16(b, 8);
const __m128i bl = _mm_srai_epi16(_mm_slli_epi16(b, 8), 8);
const __m128i mh = _mm_madd_epi16(ah, bh);
const __m128i ml = _mm_madd_epi16(al, bl);
const __m128i x = _mm_add_epi32(mh, ml);
return _mm_add_epi32(c, x);
#elif defined(__ARM_FEATURE_MATMUL_INT8)
return vusdotq_s32(c, a, b);
#elif defined(ARCH_ARM64)
const auto l =
vpaddlq_s16(vmulq_s16(vreinterpretq_s16_u16(vmovl_u8(vget_low_u8(a))),
vmovl_s8(vget_low_s8(b))));
const auto h =
vpaddlq_s16(vmulq_s16(vreinterpretq_s16_u16(vmovl_u8(vget_high_u8(a))),
vmovl_s8(vget_high_s8(b))));
return vaddq_s32(c, vaddq_s32(vuzp1q_s32(l, h), vuzp2q_s32(l, h)));
#endif
}
inline v128 gv_dotu8x4(const v128 &a, const v128 &b, const v128 &c) {
#if defined(ARCH_X64)
const __m128i ah = _mm_srli_epi16(a, 8);
const __m128i al = _mm_and_si128(a, _mm_set1_epi16(0x00ff));
const __m128i bh = _mm_srli_epi16(b, 8);
const __m128i bl = _mm_and_si128(b, _mm_set1_epi16(0x00ff));
const __m128i mh = _mm_madd_epi16(ah, bh);
const __m128i ml = _mm_madd_epi16(al, bl);
const __m128i x = _mm_add_epi32(mh, ml);
return _mm_add_epi32(c, x);
#elif defined(__ARM_FEATURE_DOTPROD)
return vdotq_u32(c, a, b);
#elif defined(ARCH_ARM64)
const auto l = vpaddlq_u16(
vmulq_u16(vmovl_u8(vget_low_u8(a)), vmovl_u8(vget_low_u8(b))));
const auto h = vpaddlq_u16(
vmulq_u16(vmovl_u8(vget_high_u8(a)), vmovl_u8(vget_high_u8(b))));
return vaddq_u32(c, vaddq_u32(vuzp1q_u32(l, h), vuzp2q_u32(l, h)));
#endif
}
inline v128 gv_dots16x2(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_madd_epi16(a, b);
#elif defined(ARCH_ARM64)
const auto ml = vmull_s16(vget_low_s16(a), vget_low_s16(b));
const auto mh = vmull_s16(vget_high_s16(a), vget_high_s16(b));
const auto sl = vpadd_s32(vget_low_s32(ml), vget_high_s32(ml));
const auto sh = vpadd_s32(vget_low_s32(mh), vget_high_s32(mh));
return vcombine_s32(sl, sh);
#endif
}
// Signed s16 from a and b, 32-bit accumulator c
inline v128 gv_dots16x2(const v128 &a, const v128 &b, const v128 &c) {
#if (defined(__AVX512VL__) && defined(__AVX512VNNI__)) || defined(__AVXVNNI__)
return _mm_dpwssd_epi32(c, a, b);
#else
return gv_add32(c, gv_dots16x2(a, b));
#endif
}
inline v128 gv_dotu16x2(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
const auto ml = _mm_mullo_epi16(a, b); // low results
const auto mh = _mm_mulhi_epu16(a, b); // high results
const auto ls = _mm_add_epi32(_mm_srli_epi32(ml, 16),
_mm_and_si128(ml, _mm_set1_epi32(0x0000ffff)));
const auto hs = _mm_add_epi32(_mm_slli_epi32(mh, 16),
_mm_and_si128(mh, _mm_set1_epi32(0xffff0000)));
return _mm_add_epi32(ls, hs);
#elif defined(ARCH_ARM64)
const auto ml = vmull_u16(vget_low_u16(a), vget_low_u16(b));
const auto mh = vmull_u16(vget_high_u16(a), vget_high_u16(b));
const auto sl = vpadd_u32(vget_low_u32(ml), vget_high_u32(ml));
const auto sh = vpadd_u32(vget_low_u32(mh), vget_high_u32(mh));
return vcombine_u32(sl, sh);
#endif
}
// Unsigned bytes from a, signed bytes from b, 32-bit accumulator c
inline v128 gv_dots_u8s8x4(const v128 &a, const v128 &b, const v128 &c) {
#if (defined(__AVX512VL__) && defined(__AVX512VNNI__)) || defined(__AVXVNNI__)
return _mm_dpbusds_epi32(c, a, b);
#elif defined(ARCH_X64)
const __m128i ah = _mm_srli_epi16(a, 8);
const __m128i al = _mm_and_si128(a, _mm_set1_epi16(0x00ff));
const __m128i bh = _mm_srai_epi16(b, 8);
const __m128i bl = _mm_srai_epi16(_mm_slli_epi16(b, 8), 8);
const __m128i mh = _mm_madd_epi16(ah, bh);
const __m128i ml = _mm_madd_epi16(al, bl);
return gv_adds_s32(c, _mm_add_epi32(mh, ml));
#elif defined(ARCH_ARM64)
const auto l =
vpaddlq_s16(vmulq_s16(vreinterpretq_s16_u16(vmovl_u8(vget_low_u8(a))),
vmovl_s8(vget_low_s8(b))));
const auto h =
vpaddlq_s16(vmulq_s16(vreinterpretq_s16_u16(vmovl_u8(vget_high_u8(a))),
vmovl_s8(vget_high_s8(b))));
return vqaddq_s32(c, vaddq_s32(vuzp1q_s32(l, h), vuzp2q_s32(l, h)));
#endif
}
// Signed s16 from a and b, 32-bit accumulator c; signed saturation
inline v128 gv_dots_s16x2(const v128 &a, const v128 &b, const v128 &c) {
#if (defined(__AVX512VL__) && defined(__AVX512VNNI__)) || defined(__AVXVNNI__)
return _mm_dpwssds_epi32(c, a, b);
#else
const auto ab = gv_dots16x2(a, b);
const auto s0 = gv_adds_s32(ab, c);
const auto s1 =
gv_eq32(ab, gv_bcst32(0x80000000)); // +0x80000000, negative c ->
// c^0x80000000; otherwise 0x7fffffff
const auto s2 =
gv_select32(gv_gts32(gv_bcst32(0), c), gv_xor32(c, gv_bcst32(0x80000000)),
gv_bcst32(0x7fffffff));
return gv_select32(s1, s2, s0);
#endif
}
// Multiply s16 elements 0, 2, 4, 6 to produce s32 results in corresponding
// lanes
inline v128 gv_mul_even_s16(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
const auto c = _mm_set1_epi32(0x0000ffff);
return _mm_madd_epi16(_mm_and_si128(a, c), _mm_and_si128(b, c));
#else
// TODO
return gv_mul32(gv_sar32(gv_shl32(a, 16), 16), gv_sar32(gv_shl32(b, 16), 16));
#endif
}
// Multiply u16 elements 0, 2, 4, 6 to produce u32 results in corresponding
// lanes
inline v128 gv_mul_even_u16(const v128 &a, const v128 &b) {
#if defined(__SSE4_1__) || defined(ARCH_ARM64)
const auto c = gv_bcst32(0x0000ffff);
return gv_mul32(a & c, b & c);
#elif defined(ARCH_X64)
const auto ml = _mm_mullo_epi16(a, b);
const auto mh = _mm_mulhi_epu16(a, b);
return _mm_or_si128(_mm_and_si128(ml, _mm_set1_epi32(0x0000ffff)),
_mm_slli_epi32(mh, 16));
#endif
}
// Multiply s16 elements 1, 3, 5, 7 to produce s32 results in corresponding
// lanes
inline v128 gv_mul_odds_s16(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_madd_epi16(_mm_srli_epi32(a, 16), _mm_srli_epi32(b, 16));
#else
return gv_mul32(gv_sar32(a, 16), gv_sar32(b, 16));
#endif
}
// Multiply u16 elements 1, 3, 5, 7 to produce u32 results in corresponding
// lanes
inline v128 gv_mul_odds_u16(const v128 &a, const v128 &b) {
#if defined(__SSE4_1__) || defined(ARCH_ARM64)
return gv_mul32(gv_shr32(a, 16), gv_shr32(b, 16));
#elif defined(ARCH_X64)
const auto ml = _mm_mullo_epi16(a, b);
const auto mh = _mm_mulhi_epu16(a, b);
return _mm_or_si128(_mm_and_si128(mh, _mm_set1_epi32(0xffff0000)),
_mm_srli_epi32(ml, 16));
#endif
}
inline v128 gv_cvts32_tofs(const v128 &src) {
#if defined(ARCH_X64)
return _mm_cvtepi32_ps(src);
#elif defined(ARCH_ARM64)
return vcvtq_f32_s32(src);
#endif
}
inline v128 gv_cvtu32_tofs(const v128 &src) {
#if defined(__AVX512VL__)
return _mm_cvtepu32_ps(src);
#elif defined(ARCH_X64)
const auto fix = _mm_and_ps(_mm_castsi128_ps(_mm_srai_epi32(src, 31)),
_mm_set1_ps(0x80000000));
return _mm_add_ps(
_mm_cvtepi32_ps(_mm_and_si128(src, _mm_set1_epi32(0x7fffffff))), fix);
#elif defined(ARCH_ARM64)
return vcvtq_f32_u32(src);
#endif
}
inline v128 gv_cvtfs_tos32(const v128 &src) {
#if defined(ARCH_X64)
return _mm_cvttps_epi32(src);
#elif defined(ARCH_ARM64)
return vcvtq_s32_f32(src);
#endif
}
inline v128 gv_cvtfs_tou32(const v128 &src) {
#if defined(__AVX512VL__)
return _mm_cvttps_epu32(src);
#elif defined(ARCH_X64)
const auto c1 = _mm_cvttps_epi32(src);
const auto s1 = _mm_srai_epi32(c1, 31);
const auto c2 = _mm_cvttps_epi32(_mm_sub_ps(src, _mm_set1_ps(2147483648.)));
return _mm_or_si128(c1, _mm_and_si128(c2, s1));
#elif defined(ARCH_ARM64)
return vcvtq_u32_f32(src);
#endif
}
inline f32 roundevenf32(f32 arg) {
u32 val = std::bit_cast<u32>(arg);
u32 exp = (val >> 23) & 0xff;
u32 abs = val & 0x7fffffff;
if (exp >= 127 + 23) {
// Big enough, NaN or INF
return arg;
}
if (exp >= 127) {
u32 int_pos = (127 + 23) - exp;
u32 half_pos = int_pos - 1;
u32 half_bit = 1u << half_pos;
u32 int_bit = 1u << int_pos;
if (val & (int_bit | (half_bit - 1)))
val += half_bit;
val &= ~(int_bit - 1);
} else if (exp == 126 && abs > 0x3f000000) {
val &= 0x80000000;
val |= 0x3f800000;
} else {
val &= 0x80000000;
}
return std::bit_cast<f32>(val);
}
#if defined(ARCH_X64)
enum class RoundMode { Even, Floor, Ceil, Trunc };
template <RoundMode Mode> __m128 sse41_roundf(__m128 a) {
v128 r = a;
for (u32 i = 0; i < 4; i++)
if constexpr (Mode == RoundMode::Even)
r._f[i] = roundevenf32(r._f[i]);
else if constexpr (Mode == RoundMode::Floor)
r._f[i] = ::floorf(r._f[i]);
else if constexpr (Mode == RoundMode::Ceil)
r._f[i] = ::ceilf(r._f[i]);
else if constexpr (Mode == RoundMode::Trunc)
r._f[i] = ::truncf(r._f[i]);
return r;
}
#endif
inline v128 gv_roundfs_even(const v128 &a) {
#if defined(__SSE4_1__)
return _mm_round_ps(a, 8 + 0);
#elif defined(ARCH_ARM64)
return vrndnq_f32(a);
#elif defined(ARCH_X64)
return sse41_roundf<RoundMode::Even>(a);
#else
v128 r;
for (u32 i = 0; i < 4; i++)
r._f[i] = roundevenf32(a._f[i]);
return r;
#endif
}
inline v128 gv_roundfs_ceil(const v128 &a) {
#if defined(__SSE4_1__)
return _mm_round_ps(a, 8 + 2);
#elif defined(ARCH_ARM64)
return vrndpq_f32(a);
#elif defined(ARCH_X64)
return sse41_roundf<RoundMode::Ceil>(a);
#else
v128 r;
for (u32 i = 0; i < 4; i++)
r._f[i] = ::ceilf(a._f[i]);
return r;
#endif
}
inline v128 gv_roundfs_floor(const v128 &a) {
#if defined(__SSE4_1__)
return _mm_round_ps(a, 8 + 1);
#elif defined(ARCH_ARM64)
return vrndmq_f32(a);
#elif defined(ARCH_X64)
return sse41_roundf<RoundMode::Floor>(a);
#else
v128 r;
for (u32 i = 0; i < 4; i++)
r._f[i] = ::floorf(a._f[i]);
return r;
#endif
}
inline v128 gv_roundfs_trunc(const v128 &a) {
#if defined(__SSE4_1__)
return _mm_round_ps(a, 8 + 3);
#elif defined(ARCH_ARM64)
return vrndq_f32(a);
#elif defined(ARCH_X64)
return sse41_roundf<RoundMode::Trunc>(a);
#else
v128 r;
for (u32 i = 0; i < 4; i++)
r._f[i] = ::truncf(a._f[i]);
return r;
#endif
}
inline bool gv_testz(const v128 &a) {
#if defined(__SSE4_1__)
return !!_mm_testz_si128(a, a);
#elif defined(ARCH_X64)
return _mm_cvtsi128_si64(_mm_packs_epi32(a, a)) == 0;
#elif defined(ARCH_ARM64)
return std::bit_cast<s64>(vqmovn_s32(a)) == 0;
#else
return !(a._u64[0] | a._u64[1]);
#endif
}
// Same as gv_testz but tuned for pairing with gv_testall1
inline bool gv_testall0(const v128 &a) {
#if defined(__SSE4_1__)
return !!_mm_testz_si128(a, _mm_set1_epi32(-1));
#elif defined(ARCH_X64)
return _mm_cvtsi128_si64(_mm_packs_epi32(a, a)) == 0;
#elif defined(ARCH_ARM64)
return std::bit_cast<s64>(vqmovn_s32(a)) == 0;
#else
return !(a._u64[0] | a._u64[1]);
#endif
}
inline bool gv_testall1(const v128 &a) {
#if defined(__SSE4_1__)
return !!_mm_test_all_ones(a);
#elif defined(ARCH_X64)
return _mm_cvtsi128_si64(_mm_packs_epi32(a, a)) == -1;
#elif defined(ARCH_ARM64)
return std::bit_cast<s64>(vqmovn_s32(a)) == -1;
#else
return (a._u64[0] & a._u64[1]) == UINT64_MAX;
#endif
}
// result = (~a) & (b)
inline v128 gv_andn(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_andnot_si128(a, b);
#elif defined(ARCH_ARM64)
return vbicq_s32(b, a);
#endif
}
// Select elements; _cmp must be result of SIMD comparison; undefined otherwise
FORCE_INLINE v128 gv_select8(const v128 &_cmp, const v128 &_true,
const v128 &_false) {
#if defined(__SSE4_1__)
return _mm_blendv_epi8(_false, _true, _cmp);
#elif defined(ARCH_ARM64)
return vbslq_u8(_cmp, _true, _false);
#else
return (_cmp & _true) | gv_andn(_cmp, _false);
#endif
}
// Select elements using sign bit only
FORCE_INLINE v128 gv_signselect8(const v128 &bits, const v128 &_true,
const v128 &_false) {
#if defined(__SSE4_1__)
return _mm_blendv_epi8(_false, _true, bits);
#else
return gv_select8(gv_gts8(gv_bcst8(0), bits), _true, _false);
#endif
}
// Select elements; _cmp must be result of SIMD comparison; undefined otherwise
inline v128 gv_select16(const v128 &_cmp, const v128 &_true,
const v128 &_false) {
#if defined(__SSE4_1__)
return _mm_blendv_epi8(_false, _true, _cmp);
#elif defined(ARCH_ARM64)
return vbslq_u16(_cmp, _true, _false);
#else
return (_cmp & _true) | gv_andn(_cmp, _false);
#endif
}
// Select elements; _cmp must be result of SIMD comparison; undefined otherwise
inline v128 gv_select32(const v128 &_cmp, const v128 &_true,
const v128 &_false) {
#if defined(__SSE4_1__)
return _mm_blendv_epi8(_false, _true, _cmp);
#elif defined(ARCH_ARM64)
return vbslq_u32(_cmp, _true, _false);
#else
return (_cmp & _true) | gv_andn(_cmp, _false);
#endif
}
// Select elements; _cmp must be result of SIMD comparison; undefined otherwise
inline v128 gv_selectfs(const v128 &_cmp, const v128 &_true,
const v128 &_false) {
#if defined(__SSE4_1__)
return _mm_blendv_ps(_false, _true, _cmp);
#elif defined(ARCH_ARM64)
return vbslq_f32(_cmp, _true, _false);
#else
return _mm_or_ps(_mm_and_ps(_cmp, _true), _mm_andnot_ps(_cmp, _false));
#endif
}
inline v128 gv_packss_s16(const v128 &low, const v128 &high) {
#if defined(ARCH_X64)
return _mm_packs_epi16(low, high);
#elif defined(ARCH_ARM64)
return vcombine_s8(vqmovn_s16(low), vqmovn_s16(high));
#endif
}
inline v128 gv_packus_s16(const v128 &low, const v128 &high) {
#if defined(ARCH_X64)
return _mm_packus_epi16(low, high);
#elif defined(ARCH_ARM64)
return vcombine_u8(vqmovun_s16(low), vqmovun_s16(high));
#endif
}
inline v128 gv_packus_u16(const v128 &low, const v128 &high) {
#if defined(__SSE4_1__)
return _mm_packus_epi16(_mm_min_epu16(low, _mm_set1_epi16(0xff)),
_mm_min_epu16(high, _mm_set1_epi16(0xff)));
#elif defined(ARCH_X64)
return _mm_packus_epi16(
_mm_sub_epi16(low, _mm_subs_epu16(low, _mm_set1_epi16(0xff))),
_mm_sub_epi16(high, _mm_subs_epu16(high, _mm_set1_epi16(0xff))));
#elif defined(ARCH_ARM64)
return vcombine_u8(vqmovn_u16(low), vqmovn_u16(high));
#endif
}
inline v128 gv_packtu16(const v128 &low, const v128 &high) {
#if defined(ARCH_X64)
return _mm_packus_epi16(low & _mm_set1_epi16(0xff),
high & _mm_set1_epi16(0xff));
#elif defined(ARCH_ARM64)
return vuzp1q_s8(low, high);
#endif
}
inline v128 gv_packss_s32(const v128 &low, const v128 &high) {
#if defined(ARCH_X64)
return _mm_packs_epi32(low, high);
#elif defined(ARCH_ARM64)
return vcombine_s16(vqmovn_s32(low), vqmovn_s32(high));
#endif
}
inline v128 gv_packus_s32(const v128 &low, const v128 &high) {
#if defined(__SSE4_1__)
return _mm_packus_epi32(low, high);
#elif defined(ARCH_X64)
const auto s = _mm_srai_epi16(_mm_packs_epi32(low, high), 15);
const auto r = gv_add16(_mm_packs_epi32(gv_sub32(low, gv_bcst32(0x8000)),
gv_sub32(high, gv_bcst32(0x8000))),
gv_bcst16(0x8000));
return gv_andn(s, r);
#elif defined(ARCH_ARM64)
return vcombine_u16(vqmovun_s32(low), vqmovun_s32(high));
#endif
}
inline v128 gv_packus_u32(const v128 &low, const v128 &high) {
#if defined(__SSE4_1__)
return _mm_packus_epi32(_mm_min_epu32(low, _mm_set1_epi32(0xffff)),
_mm_min_epu32(high, _mm_set1_epi32(0xffff)));
#elif defined(ARCH_X64)
const v128 s = _mm_cmpgt_epi16(
_mm_packs_epi32(_mm_srli_epi32(low, 16), _mm_srli_epi32(high, 16)),
_mm_setzero_si128());
const v128 r = _mm_packs_epi32(_mm_srai_epi32(_mm_slli_epi32(low, 16), 16),
_mm_srai_epi32(_mm_slli_epi32(high, 16), 16));
return _mm_or_si128(r, s);
#elif defined(ARCH_ARM64)
return vcombine_u16(vqmovn_u32(low), vqmovn_u32(high));
#endif
}
inline v128 gv_packtu32(const v128 &low, const v128 &high) {
#if defined(__SSE4_1__)
return _mm_packus_epi32(low & _mm_set1_epi32(0xffff),
high & _mm_set1_epi32(0xffff));
#elif defined(ARCH_X64)
return _mm_packs_epi32(_mm_srai_epi32(_mm_slli_epi32(low, 16), 16),
_mm_srai_epi32(_mm_slli_epi32(high, 16), 16));
#elif defined(ARCH_ARM64)
return vuzp1q_s16(low, high);
#endif
}
inline v128 gv_unpacklo8(const v128 &lows, const v128 &highs) {
#if defined(ARCH_X64)
return _mm_unpacklo_epi8(lows, highs);
#elif defined(ARCH_ARM64)
return vzip1q_s8(lows, highs);
#endif
}
inline v128 gv_extend_lo_s8(const v128 &vec) {
#if defined(__SSE4_1__)
return _mm_cvtepi8_epi16(vec);
#elif defined(ARCH_X64)
return _mm_srai_epi16(_mm_unpacklo_epi8(vec, vec), 8);
#elif defined(ARCH_ARM64)
return int16x8_t(vmovl_s8(vget_low_s8(vec)));
#endif
}
inline v128 gv_extend_hi_s8(const v128 &vec) {
#if defined(__SSE4_1__)
return _mm_cvtepi8_epi16(_mm_loadu_si64(vec._bytes + 8));
#elif defined(ARCH_X64)
return _mm_srai_epi16(_mm_unpackhi_epi8(vec, vec), 8);
#elif defined(ARCH_ARM64)
return int16x8_t(vmovl_s8(vget_high_s8(vec)));
#endif
}
inline v128 gv_unpacklo16(const v128 &lows, const v128 &highs) {
#if defined(ARCH_X64)
return _mm_unpacklo_epi16(lows, highs);
#elif defined(ARCH_ARM64)
return vzip1q_s16(lows, highs);
#endif
}
inline v128 gv_extend_lo_s16(const v128 &vec) {
#if defined(__SSE4_1__)
return _mm_cvtepi16_epi32(vec);
#elif defined(ARCH_X64)
return _mm_srai_epi32(_mm_unpacklo_epi16(vec, vec), 16);
#elif defined(ARCH_ARM64)
return int32x4_t(vmovl_s16(vget_low_s16(vec)));
#endif
}
inline v128 gv_extend_hi_s16(const v128 &vec) {
#if defined(__SSE4_1__)
return _mm_cvtepi16_epi32(_mm_loadu_si64(vec._bytes + 8));
#elif defined(ARCH_X64)
return _mm_srai_epi32(_mm_unpackhi_epi16(vec, vec), 16);
#elif defined(ARCH_ARM64)
return int32x4_t(vmovl_s16(vget_high_s16(vec)));
#endif
}
inline v128 gv_unpacklo32(const v128 &lows, const v128 &highs) {
#if defined(ARCH_X64)
return _mm_unpacklo_epi32(lows, highs);
#elif defined(ARCH_ARM64)
return vzip1q_s32(lows, highs);
#endif
}
inline v128 gv_unpackhi8(const v128 &lows, const v128 &highs) {
#if defined(ARCH_X64)
return _mm_unpackhi_epi8(lows, highs);
#elif defined(ARCH_ARM64)
return vzip2q_s8(lows, highs);
#endif
}
inline v128 gv_unpackhi16(const v128 &lows, const v128 &highs) {
#if defined(ARCH_X64)
return _mm_unpackhi_epi16(lows, highs);
#elif defined(ARCH_ARM64)
return vzip2q_s16(lows, highs);
#endif
}
inline v128 gv_unpackhi32(const v128 &lows, const v128 &highs) {
#if defined(ARCH_X64)
return _mm_unpackhi_epi32(lows, highs);
#elif defined(ARCH_ARM64)
return vzip2q_s32(lows, highs);
#endif
}
inline bool v128::operator==(const v128 &b) const {
#if defined(ARCH_X64)
return gv_testz(_mm_xor_si128(*this, b));
#else
return gv_testz(*this ^ b);
#endif
}
inline v128 v128::operator|(const v128 &rhs) const {
#if defined(ARCH_X64)
return _mm_or_si128(*this, rhs);
#elif defined(ARCH_ARM64)
return vorrq_s32(*this, rhs);
#endif
}
inline v128 v128::operator&(const v128 &rhs) const {
#if defined(ARCH_X64)
return _mm_and_si128(*this, rhs);
#elif defined(ARCH_ARM64)
return vandq_s32(*this, rhs);
#endif
}
inline v128 v128::operator^(const v128 &rhs) const {
#if defined(ARCH_X64)
return _mm_xor_si128(*this, rhs);
#elif defined(ARCH_ARM64)
return veorq_s32(*this, rhs);
#endif
}
inline v128 v128::operator~() const {
#if defined(ARCH_X64)
return _mm_xor_si128(*this, _mm_set1_epi32(-1));
#elif defined(ARCH_ARM64)
return vmvnq_u32(*this);
#endif
}
inline v128 gv_exp2_approxfs(const v128 &a) {
// TODO
#if 0
const auto x0 = _mm_max_ps(_mm_min_ps(a, _mm_set1_ps(127.4999961f)), _mm_set1_ps(-127.4999961f));
const auto x1 = _mm_add_ps(x0, _mm_set1_ps(0.5f));
const auto x2 = _mm_sub_epi32(_mm_cvtps_epi32(x1), _mm_and_si128(_mm_castps_si128(_mm_cmpnlt_ps(_mm_setzero_ps(), x1)), _mm_set1_epi32(1)));
const auto x3 = _mm_sub_ps(x0, _mm_cvtepi32_ps(x2));
const auto x4 = _mm_mul_ps(x3, x3);
const auto x5 = _mm_mul_ps(x3, _mm_add_ps(_mm_mul_ps(_mm_add_ps(_mm_mul_ps(x4, _mm_set1_ps(0.023093347705f)), _mm_set1_ps(20.20206567f)), x4), _mm_set1_ps(1513.906801f)));
const auto x6 = _mm_mul_ps(x5, _mm_rcp_ps(_mm_sub_ps(_mm_add_ps(_mm_mul_ps(_mm_set1_ps(233.1842117f), x4), _mm_set1_ps(4368.211667f)), x5)));
return _mm_mul_ps(_mm_add_ps(_mm_add_ps(x6, x6), _mm_set1_ps(1.0f)), _mm_castsi128_ps(_mm_slli_epi32(_mm_add_epi32(x2, _mm_set1_epi32(127)), 23)));
#else
v128 r;
for (u32 i = 0; i < 4; i++)
r._f[i] = std::exp2f(a._f[i]);
return r;
#endif
}
inline v128 gv_log2_approxfs(const v128 &a) {
// TODO
#if 0
const auto _1 = _mm_set1_ps(1.0f);
const auto _c = _mm_set1_ps(1.442695040f);
const auto x0 = _mm_max_ps(a, _mm_castsi128_ps(_mm_set1_epi32(0x00800000)));
const auto x1 = _mm_or_ps(_mm_and_ps(x0, _mm_castsi128_ps(_mm_set1_epi32(0x807fffff))), _1);
const auto x2 = _mm_rcp_ps(_mm_add_ps(x1, _1));
const auto x3 = _mm_mul_ps(_mm_sub_ps(x1, _1), x2);
const auto x4 = _mm_add_ps(x3, x3);
const auto x5 = _mm_mul_ps(x4, x4);
const auto x6 = _mm_add_ps(_mm_mul_ps(_mm_add_ps(_mm_mul_ps(_mm_set1_ps(-0.7895802789f), x5), _mm_set1_ps(16.38666457f)), x5), _mm_set1_ps(-64.1409953f));
const auto x7 = _mm_rcp_ps(_mm_add_ps(_mm_mul_ps(_mm_add_ps(_mm_mul_ps(_mm_set1_ps(-35.67227983f), x5), _mm_set1_ps(312.0937664f)), x5), _mm_set1_ps(-769.6919436f)));
const auto x8 = _mm_cvtepi32_ps(_mm_sub_epi32(_mm_srli_epi32(_mm_castps_si128(x0), 23), _mm_set1_epi32(127)));
return _mm_add_ps(_mm_mul_ps(_mm_mul_ps(_mm_mul_ps(_mm_mul_ps(x5, x6), x7), x4), _c), _mm_add_ps(_mm_mul_ps(x4, _c), x8));
#else
v128 r;
for (u32 i = 0; i < 4; i++)
r._f[i] = std::log2f(a._f[i]);
return r;
#endif
}
// For each 8-bit element, r = a << (b & 7)
inline v128 gv_shl8(const v128 &a, const v128 &b) {
#if defined(ARCH_ARM64)
return vshlq_u8(a, vandq_s8(b, gv_bcst8(7)));
#else
const v128 x1 = gv_add8(a, a); // shift left by 1
const v128 r1 = gv_signselect8(gv_shl64(b, 7), x1, a);
const v128 x2 = gv_and32(gv_shl64(r1, 2), gv_bcst8(0xfc)); // shift by 2
const v128 r2 = gv_signselect8(gv_shl64(b, 6), x2, r1);
const v128 x3 = gv_and32(gv_shl64(r2, 4), gv_bcst8(0xf0)); // shift by 4
return gv_signselect8(gv_shl64(b, 5), x3, r2);
#endif
}
// For each 16-bit element, r = a << (b & 15)
inline v128 gv_shl16(const v128 &a, const v128 &b) {
#if defined(__AVX512VL__) && defined(__AVX512BW__)
return _mm_sllv_epi16(a, _mm_and_si128(b, _mm_set1_epi16(15)));
#elif defined(ARCH_ARM64)
return vshlq_u16(a, vandq_s16(b, gv_bcst8(15)));
#else
v128 r;
for (u32 i = 0; i < 8; i++)
r._u16[i] = a._u16[i] << (b._u16[i] & 15);
return r;
#endif
}
// For each 32-bit element, r = a << (b & 31)
inline v128 gv_shl32(const v128 &a, const v128 &b) {
#if defined(__AVX2__)
return _mm_sllv_epi32(a, _mm_and_si128(b, _mm_set1_epi32(31)));
#elif defined(ARCH_ARM64)
return vshlq_u32(a, vandq_s32(b, gv_bcst8(31)));
#else
v128 r;
for (u32 i = 0; i < 4; i++)
r._u32[i] = a._u32[i] << (b._u32[i] & 31);
return r;
#endif
}
// For each unsigned 8-bit element, r = a >> (b & 7)
inline v128 gv_shr8(const v128 &a, const v128 &b) {
#if defined(ARCH_ARM64)
return vshlq_u8(a, vnegq_s8(vandq_s8(b, gv_bcst8(7))));
#else
const v128 x1 = gv_and32(gv_shr64(a, 1), gv_bcst8(0x7f)); // shift right by 1
const v128 r1 = gv_signselect8(gv_shl64(b, 7), x1, a);
const v128 x2 = gv_and32(gv_shr64(r1, 2), gv_bcst8(0x3f)); // shift by 2
const v128 r2 = gv_signselect8(gv_shl64(b, 6), x2, r1);
const v128 x3 = gv_and32(gv_shr64(r2, 4), gv_bcst8(0x0f)); // shift by 4
return gv_signselect8(gv_shl64(b, 5), x3, r2);
#endif
}
// For each unsigned 16-bit element, r = a >> (b & 15)
inline v128 gv_shr16(const v128 &a, const v128 &b) {
#if defined(__AVX512VL__) && defined(__AVX512BW__)
return _mm_srlv_epi16(a, _mm_and_si128(b, _mm_set1_epi16(15)));
#elif defined(ARCH_ARM64)
return vshlq_u16(a, vnegq_s16(vandq_s16(b, gv_bcst8(15))));
#else
v128 r;
for (u32 i = 0; i < 8; i++)
r._u16[i] = a._u16[i] >> (b._u16[i] & 15);
return r;
#endif
}
// For each unsigned 32-bit element, r = a >> (b & 31)
inline v128 gv_shr32(const v128 &a, const v128 &b) {
#if defined(__AVX2__)
return _mm_srlv_epi32(a, _mm_and_si128(b, _mm_set1_epi32(31)));
#elif defined(ARCH_ARM64)
return vshlq_u32(a, vnegq_s32(vandq_s32(b, gv_bcst8(31))));
#else
v128 r;
for (u32 i = 0; i < 4; i++)
r._u32[i] = a._u32[i] >> (b._u32[i] & 31);
return r;
#endif
}
// For each signed 8-bit element, r = a >> (b & 7)
inline v128 gv_sar8(const v128 &a, const v128 &b) {
#if defined(ARCH_ARM64)
return vshlq_s8(a, vnegq_s8(vandq_s8(b, gv_bcst8(7))));
#else
v128 r;
for (u32 i = 0; i < 16; i++)
r._s8[i] = a._s8[i] >> (b._s8[i] & 7);
return r;
#endif
}
// For each signed 16-bit element, r = a >> (b & 15)
inline v128 gv_sar16(const v128 &a, const v128 &b) {
#if defined(__AVX512VL__) && defined(__AVX512BW__)
return _mm_srav_epi16(a, _mm_and_si128(b, _mm_set1_epi16(15)));
#elif defined(ARCH_ARM64)
return vshlq_s16(a, vnegq_s16(vandq_s16(b, gv_bcst8(15))));
#else
v128 r;
for (u32 i = 0; i < 8; i++)
r._s16[i] = a._s16[i] >> (b._s16[i] & 15);
return r;
#endif
}
// For each signed 32-bit element, r = a >> (b & 31)
inline v128 gv_sar32(const v128 &a, const v128 &b) {
#if defined(__AVX2__)
return _mm_srav_epi32(a, _mm_and_si128(b, _mm_set1_epi32(31)));
#elif defined(ARCH_ARM64)
return vshlq_s32(a, vnegq_s32(vandq_s32(b, gv_bcst8(31))));
#else
v128 r;
for (u32 i = 0; i < 4; i++)
r._s32[i] = a._s32[i] >> (b._s32[i] & 31);
return r;
#endif
}
// For each 8-bit element, r = rotate a by b
inline v128 gv_rol8(const v128 &a, const v128 &b) {
#if defined(ARCH_ARM64)
const auto amt1 = vandq_s8(b, gv_bcst8(7));
const auto amt2 = vsubq_s8(amt1, gv_bcst8(8));
return vorrq_u8(vshlq_u8(a, amt1), vshlq_u8(a, amt2));
#else
const v128 x1 =
gv_sub8(gv_add8(a, a), gv_gts8(gv_bcst8(0), a)); // rotate left by 1
const v128 r1 = gv_signselect8(gv_shl64(b, 7), x1, a);
const v128 c2 = gv_bcst8(0x3);
const v128 x2 = gv_or32(gv_and32(gv_shr64(r1, 6), c2),
gv_andn32(c2, gv_shl64(r1, 2))); // rotate by 2
const v128 r2 = gv_signselect8(gv_shl64(b, 6), x2, r1);
const v128 c3 = gv_bcst8(0xf);
const v128 x3 = gv_or32(gv_and32(gv_shr64(r2, 4), c3),
gv_andn32(c3, gv_shl64(r2, 4))); // rotate by 4
return gv_signselect8(gv_shl64(b, 5), x3, r2);
#endif
}
// For each 16-bit element, r = rotate a by b
inline v128 gv_rol16(const v128 &a, const v128 &b) {
#if defined(ARCH_ARM64)
const auto amt1 = vandq_s16(b, gv_bcst16(15));
const auto amt2 = vsubq_s16(amt1, gv_bcst16(16));
return vorrq_u16(vshlq_u16(a, amt1), vshlq_u16(a, amt2));
#else
v128 r;
for (u32 i = 0; i < 8; i++)
r._u16[i] = rol16(a._u16[i], b._u16[i]);
return r;
#endif
}
// For each 16-bit element, r = rotate a by count
template <u8 Count> inline v128 gv_rol16(const v128 &a) {
constexpr u8 count = Count & 0xf;
#if defined(ARCH_X64)
return _mm_or_si128(_mm_srli_epi16(a, 16 - count), _mm_slli_epi16(a, count));
#elif defined(ARCH_ARM64)
return vorrq_u16(vshrq_n_u16(a, 16 - count), vshlq_n_u16(a, count));
#else
v128 r;
for (u32 i = 0; i < 8; i++)
r._u16[i] = std::rotl(a._u16[i], count);
return r;
#endif
}
// For each 32-bit element, r = rotate a by b
inline v128 gv_rol32(const v128 &a, const v128 &b) {
#if defined(__AVX512VL__)
return _mm_rolv_epi32(a, b);
#elif defined(ARCH_ARM64)
const auto amt1 = vandq_s32(b, gv_bcst32(31));
const auto amt2 = vsubq_s32(amt1, gv_bcst32(32));
return vorrq_u32(vshlq_u32(a, amt1), vshlq_u32(a, amt2));
#else
v128 r;
for (u32 i = 0; i < 4; i++)
r._u32[i] = rol32(a._u32[i], b._u32[i]);
return r;
#endif
}
// For each 32-bit element, r = rotate a by count
template <u8 Count> inline v128 gv_rol32(const v128 &a) {
constexpr u8 count = Count & 0x1f;
#if defined(__AVX512VL__)
return _mm_rol_epi32(a, count);
#elif defined(ARCH_X64)
return _mm_or_si128(_mm_srli_epi32(a, 32 - count), _mm_slli_epi32(a, count));
#elif defined(ARCH_ARM64)
return vorrq_u32(vshrq_n_u32(a, 32 - count), vshlq_n_u32(a, count));
#else
v128 r;
for (u32 i = 0; i < 4; i++)
r._u32[i] = rol32(a._u32[i], count);
return r;
#endif
}
// For each 8-bit element, r = (a << (c & 7)) | (b >> (~c & 7) >> 1)
inline auto gv_fshl8(const v128 &a, const v128 &b, const v128 &c) {
#if defined(ARCH_ARM64)
const auto amt1 = vandq_s8(c, gv_bcst8(7));
const auto amt2 = vsubq_s8(amt1, gv_bcst8(8));
return v128(vorrq_u8(vshlq_u8(a, amt1), vshlq_u8(b, amt2)));
#else
auto x1 = gv_sub8(gv_add8(a, a), gv_gts8(gv_bcst8(0), b));
auto s1 = gv_shl64(c, 7);
auto r1 = gv_signselect8(s1, x1, a);
auto b1 = gv_signselect8(s1, gv_shl64(b, 1), b);
auto c2 = gv_bcst8(0x3);
auto x2 = gv_and32(gv_shr64(b1, 6), c2);
x2 = gv_or32(x2, gv_andn32(c2, gv_shl64(r1, 2)));
auto s2 = gv_shl64(c, 6);
auto r2 = gv_signselect8(s2, x2, r1);
auto b2 = gv_signselect8(s2, gv_shl64(b1, 2), b1);
auto c3 = gv_bcst8(0xf);
auto x3 = gv_and32(gv_shr64(b2, 4), c3);
x3 = gv_or32(x3, gv_andn32(c3, gv_shl64(r2, 4)));
return gv_signselect8(gv_shl64(c, 5), x3, r2);
#endif
}
// For each 8-bit element, r = (b >> (c & 7)) | (a << (~c & 7) << 1)
inline auto gv_fshr8(const v128 &a, const v128 &b, const v128 &c) {
#if defined(ARCH_ARM64)
const auto amt1 = vandq_s8(c, gv_bcst8(7));
const auto amt2 = vsubq_s8(gv_bcst8(8), amt1);
return vorrq_u8(vshlq_u8(b, vnegq_s8(amt1)), vshlq_u8(a, amt2));
#else
auto c1 = gv_bcst8(0x7f);
auto x1 = gv_and32(gv_shr64(b, 1), c1);
x1 = gv_or32(x1, gv_andn32(c1, gv_shl64(a, 7)));
auto s1 = gv_shl64(c, 7);
auto r1 = gv_signselect8(s1, x1, b);
auto a1 = gv_signselect8(s1, gv_shr64(a, 1), a);
auto c2 = gv_bcst8(0x3f);
auto x2 = gv_and32(gv_shr64(r1, 2), c2);
x2 = gv_or32(x2, gv_andn32(c2, gv_shl64(a1, 6)));
auto s2 = gv_shl64(c, 6);
auto r2 = gv_signselect8(s2, x2, r1);
auto a2 = gv_signselect8(s2, gv_shr64(a1, 2), a1);
auto c3 = gv_bcst8(0x0f);
auto x3 = gv_and32(gv_shr64(r2, 4), c3);
x3 = gv_or32(x3, gv_andn32(c3, gv_shl64(a2, 4)));
return gv_signselect8(gv_shl64(c, 5), x3, r2);
#endif
}
// Shift left by byte amount
template <u32 Count> inline v128 gv_shuffle_left(const v128 &a) {
if (Count > 15)
return {};
#if defined(ARCH_X64)
return _mm_slli_si128(a, Count);
#elif defined(ARCH_ARM64)
v128 idx;
for (u32 i = 0; i < 16; i++)
idx._u8[i] = u8(i - Count);
return vqtbl1q_u8(a, idx);
#endif
}
// Shift right by byte amount
template <u32 Count> inline v128 gv_shuffle_right(const v128 &a) {
if (Count > 15)
return {};
#if defined(ARCH_X64)
return _mm_srli_si128(a, Count);
#elif defined(ARCH_ARM64)
v128 idx;
for (u32 i = 0; i < 16; i++)
idx._u8[i] = u8(i + Count);
return vqtbl1q_u8(a, idx);
#endif
}
// Load 32-bit integer into the first element of a new vector, set other
// elements to zero
inline v128 gv_loadu32(const void *ptr) {
#if defined(ARCH_X64)
return _mm_loadu_si32(ptr);
#elif defined(ARCH_ARM64)
return vld1q_lane_u32(static_cast<const u32 *>(ptr), vdupq_n_u32(0), 0);
#endif
}
// Load 16-bit integer into an existing vector at the position specified by
// Index
template <u8 Index> inline v128 gv_insert16(const v128 &vec, u16 value) {
#if defined(ARCH_X64)
return _mm_insert_epi16(vec, value, Index);
#elif defined(ARCH_ARM64)
return vsetq_lane_u16(value, vec, Index & 0x7);
#endif
}
// For each 8-bit element,
// if ctrl >= 0 && ctrl < 16 then r = vec[ctrl],
// else if ctrl < 0 then r = 0
inline v128 gv_shuffle8(const v128 &vec, const v128 &ctrl) {
AUDIT(
std::ranges::none_of(
ctrl._chars, [](s8 i) { return i >= static_cast<s8>(sizeof(v128)); }),
"All indices must be in the range [0, 15] or negative, since PSHUFB and "
"TBL behave differently otherwise");
#if defined(__SSSE3__)
return _mm_shuffle_epi8(vec, ctrl);
#elif defined(ARCH_ARM64)
return vqtbl1q_s8(vec, ctrl);
#else
v128 r;
for (s32 i = 0; i < 16; i++)
r._s8[i] = ctrl._s8[i] < 0 ? 0 : vec._s8[ctrl._s8[i] & 0xf];
return r;
#endif
}
// For each 2-bit index in Control, r = vec[index]
template <u8 Control> inline v128 gv_shuffle32(const v128 &vec) {
#if defined(ARCH_X64)
return _mm_shuffle_epi32(vec, Control);
#elif defined(ARCH_ARM64)
constexpr u8 idx0 = (Control & 3) * sizeof(s32);
constexpr u8 idx1 = (Control >> 2 & 3) * sizeof(s32);
constexpr u8 idx2 = (Control >> 4 & 3) * sizeof(s32);
constexpr u8 idx3 = (Control >> 6 & 3) * sizeof(s32);
constexpr uint8x16_t idx_vec = {
idx0, idx0 + 1, idx0 + 2, idx0 + 3, idx1, idx1 + 1, idx1 + 2, idx1 + 3,
idx2, idx2 + 1, idx2 + 2, idx2 + 3, idx3, idx3 + 1, idx3 + 2, idx3 + 3};
return vqtbl1q_s8(vec, idx_vec);
#endif
}
// For each index, r = vec[index & 3]
template <u8 Index0, u8 Index1, u8 Index2, u8 Index3>
inline v128 gv_shuffle32(const v128 &vec) {
#if defined(ARCH_X64)
return _mm_shuffle_epi32(vec, (Index0 & 3) | (Index1 & 3) << 2 |
(Index2 & 3) << 4 | (Index3 & 3) << 6);
#elif defined(ARCH_ARM64)
constexpr u8 idx0 = (Index0 & 3) * sizeof(s32);
constexpr u8 idx1 = (Index1 & 3) * sizeof(s32);
constexpr u8 idx2 = (Index2 & 3) * sizeof(s32);
constexpr u8 idx3 = (Index3 & 3) * sizeof(s32);
constexpr uint8x16_t idx_vec = {
idx0, idx0 + 1, idx0 + 2, idx0 + 3, idx1, idx1 + 1, idx1 + 2, idx1 + 3,
idx2, idx2 + 1, idx2 + 2, idx2 + 3, idx3, idx3 + 1, idx3 + 2, idx3 + 3};
return vqtbl1q_s8(vec, idx_vec);
#endif
}
// For the first two 2-bit indices in Control, r = a[index],
// for the last two indices, r = b[index]
template <u8 Control> inline v128 gv_shufflefs(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_shuffle_ps(a, b, Control);
#elif defined(ARCH_ARM64)
constexpr u8 idx0 = (Control & 3) * sizeof(s32);
constexpr u8 idx1 = (Control >> 2 & 3) * sizeof(s32);
constexpr u8 idx2 = (Control >> 4 & 3) * sizeof(s32) + sizeof(v128);
constexpr u8 idx3 = (Control >> 6 & 3) * sizeof(s32) + sizeof(v128);
constexpr uint8x16_t idx_vec = {
idx0, idx0 + 1, idx0 + 2, idx0 + 3, idx1, idx1 + 1, idx1 + 2, idx1 + 3,
idx2, idx2 + 1, idx2 + 2, idx2 + 3, idx3, idx3 + 1, idx3 + 2, idx3 + 3};
return vqtbl2q_s8({a, b}, idx_vec);
#endif
}
// For the first two indices, r = a[index & 3],
// for the last two indices, r = b[index & 3]
template <u8 Index0, u8 Index1, u8 Index2, u8 Index3>
inline v128 gv_shufflefs(const v128 &a, const v128 &b) {
#if defined(ARCH_X64)
return _mm_shuffle_ps(a, b,
(Index0 & 3) | (Index1 & 3) << 2 | (Index2 & 3) << 4 |
(Index3 & 3) << 6);
#elif defined(ARCH_ARM64)
constexpr u8 idx0 = (Index0 & 3) * sizeof(s32);
constexpr u8 idx1 = (Index1 & 3) * sizeof(s32);
constexpr u8 idx2 = (Index2 & 3) * sizeof(s32) + sizeof(v128);
constexpr u8 idx3 = (Index3 & 3) * sizeof(s32) + sizeof(v128);
constexpr uint8x16_t idx_vec = {
idx0, idx0 + 1, idx0 + 2, idx0 + 3, idx1, idx1 + 1, idx1 + 2, idx1 + 3,
idx2, idx2 + 1, idx2 + 2, idx2 + 3, idx3, idx3 + 1, idx3 + 2, idx3 + 3};
return vqtbl2q_s8({a, b}, idx_vec);
#endif
}
// For each 32-bit element, reverse byte order
inline v128 gv_rev32(const v128 &vec) {
#if defined(__SSSE3__)
return _mm_shuffle_epi8(
vec, _mm_setr_epi8(3, 2, 1, 0, 7, 6, 5, 4, 11, 10, 9, 8, 15, 14, 13, 12));
#elif defined(ARCH_ARM64)
return vrev32q_u8(vec);
#else
return gv_rol32<16>(gv_rol16<8>(vec));
#endif
}
// For each 32-bit element, convert between big-endian and native-endian
inline v128 gv_to_be32(const v128 &vec) {
if constexpr (std::endian::native == std::endian::little)
return gv_rev32(vec);
return vec;
}
#if defined(__clang__)
#pragma clang diagnostic pop
#elif defined(__GNUC__)
#pragma GCC diagnostic pop
#endif
} // namespace rx