From 62dd5c742f0073d5553d28acdc6fab8bce4fe0ce Mon Sep 17 00:00:00 2001 From: "Michael R. Crusoe" Date: Sat, 14 Sep 2024 14:14:00 +0200 Subject: [PATCH] x86: some better implementations for MSVC and others without SIMDE_STATEMENT_EXPR_ Closes: https://github.com/simd-everywhere/simde/issues/1219 --- simde/x86/avx.h | 24 ++++++++++--- simde/x86/avx512/cmp.h | 16 ++++++--- simde/x86/sse.h | 35 ++++++++++++------ simde/x86/sse2.h | 82 +++++++++++++++++++++++++++++++----------- 4 files changed, 116 insertions(+), 41 deletions(-) diff --git a/simde/x86/avx.h b/simde/x86/avx.h index 20c2e6b0b..bec901376 100644 --- a/simde/x86/avx.h +++ b/simde/x86/avx.h @@ -2083,7 +2083,11 @@ simde_mm256_round_ps (simde__m256 a, const int rounding) { simde__m256_private r_, a_ = simde__m256_to_private(a); - + #if SIMDE_NATURAL_VECTOR_SIZE_LE(128) && !defined(SIMDE_STATEMENT_EXPR_) + for (size_t i = 0 ; i < (sizeof(r_.m128) / sizeof(r_.m128[0])) ; i++) { + SIMDE_CONSTIFY_16_(simde_mm_round_ps, r_.m128[i], (HEDLEY_UNREACHABLE(), simde_mm_undefined_ps()), rounding, a_.m128[i]); + } + #else switch (rounding & ~SIMDE_MM_FROUND_NO_EXC) { #if defined(simde_math_nearbyintf) case SIMDE_MM_FROUND_CUR_DIRECTION: @@ -2128,7 +2132,7 @@ simde_mm256_round_ps (simde__m256 a, const int rounding) { default: HEDLEY_UNREACHABLE_RETURN(simde_mm256_undefined_ps()); } - + #endif return simde__m256_from_private(r_); } #if defined(SIMDE_X86_AVX_NATIVE) @@ -2157,6 +2161,11 @@ simde_mm256_round_pd (simde__m256d a, const int rounding) { simde__m256d_private r_, a_ = simde__m256d_to_private(a); + #if SIMDE_NATURAL_VECTOR_SIZE_LE(128) && !defined(SIMDE_STATEMENT_EXPR_) + for (size_t i = 0 ; i < (sizeof(r_.m128d) / sizeof(r_.m128d[0])) ; i++) { + SIMDE_CONSTIFY_16_(simde_mm_round_pd, r_.m128d[i], (HEDLEY_UNREACHABLE(), simde_mm_undefined_pd()), rounding, a_.m128d[i]); + } + #else switch (rounding & ~SIMDE_MM_FROUND_NO_EXC) { #if defined(simde_math_nearbyint) @@ -2202,7 +2211,7 @@ simde_mm256_round_pd (simde__m256d a, const int rounding) { default: HEDLEY_UNREACHABLE_RETURN(simde_mm256_undefined_pd()); } - + #endif return simde__m256d_from_private(r_); } #if defined(SIMDE_X86_AVX_NATIVE) @@ -2894,6 +2903,11 @@ simde_mm256_cmp_ps a_ = simde__m256_to_private(a), b_ = simde__m256_to_private(b); + #if defined(SIMDE_STATEMENT_EXPR_) && SIMDE_NATURAL_VECTOR_SIZE_LE(128) + for (size_t i = 0 ; i < (sizeof(r_.m128) / sizeof(r_.m128[0])) ; i++) { + SIMDE_CONSTIFY_32_(simde_mm_cmp_ps, r_.m128[i], (HEDLEY_UNREACHABLE(), simde_mm_undefined_ps()), imm8, a_.m128[i], b_.m128[i]); + } + #else switch (imm8) { case SIMDE_CMP_EQ_OQ: case SIMDE_CMP_EQ_OS: @@ -3076,7 +3090,7 @@ simde_mm256_cmp_ps default: HEDLEY_UNREACHABLE(); } - + #endif return simde__m256_from_private(r_); } #if defined(__clang__) && defined(__AVX512DQ__) @@ -3098,7 +3112,7 @@ simde_mm256_cmp_ps simde_mm256_cmp_ps_r; \ })) #elif defined(SIMDE_X86_AVX_NATIVE) - #define simde_mm256_cmp_ps(a, b, imm8) _mm256_cmp_ps(a, b, imm8) + #define simde_mm256_cmp_ps(a, b, imm8) _mm256_cmp_ps((a), (b), (imm8)) #elif defined(SIMDE_STATEMENT_EXPR_) && SIMDE_NATURAL_VECTOR_SIZE_LE(128) #define simde_mm256_cmp_ps(a, b, imm8) SIMDE_STATEMENT_EXPR_(({ \ simde__m256_private \ diff --git a/simde/x86/avx512/cmp.h b/simde/x86/avx512/cmp.h index 2a3b99c3b..a09cac538 100644 --- a/simde/x86/avx512/cmp.h +++ b/simde/x86/avx512/cmp.h @@ -248,7 +248,11 @@ simde_mm512_cmp_ps_mask (simde__m512 a, simde__m512 b, const int imm8) r_, a_ = simde__m512_to_private(a), b_ = simde__m512_to_private(b); - + #if !defined(SIMDE_STATEMENT_EXPR_) && SIMDE_NATURAL_VECTOR_SIZE_LE(128) + for (size_t i = 0 ; i < (sizeof(r_.m128) / sizeof(r_.m128[0])) ; i++) { + SIMDE_CONSTIFY_32_(simde_mm_cmp_ps, r_.m128[i], simde_mm_undefined_ps(), imm8, a_.m128[i], b_.m128[i]); + } + #else switch (imm8) { case SIMDE_CMP_EQ_OQ: case SIMDE_CMP_EQ_OS: @@ -431,7 +435,7 @@ simde_mm512_cmp_ps_mask (simde__m512 a, simde__m512 b, const int imm8) default: HEDLEY_UNREACHABLE(); } - + #endif return simde_mm512_movepi32_mask(simde_mm512_castps_si512(simde__m512_from_private(r_))); } #if defined(SIMDE_X86_AVX512F_NATIVE) @@ -496,7 +500,11 @@ simde_mm512_cmp_pd_mask (simde__m512d a, simde__m512d b, const int imm8) r_, a_ = simde__m512d_to_private(a), b_ = simde__m512d_to_private(b); - + #if !defined(SIMDE_STATEMENT_EXPR_) && SIMDE_NATURAL_VECTOR_SIZE_LE(128) + for (size_t i = 0 ; i < (sizeof(r_.m128d) / sizeof(r_.m128d[0])) ; i++) { + SIMDE_CONSTIFY_32_(simde_mm_cmp_pd, r_.m128d[i], simde_mm_undefined_pd(), imm8, a_.m128d[i], b_.m128d[i]); + } + #else switch (imm8) { case SIMDE_CMP_EQ_OQ: case SIMDE_CMP_EQ_OS: @@ -679,7 +687,7 @@ simde_mm512_cmp_pd_mask (simde__m512d a, simde__m512d b, const int imm8) default: HEDLEY_UNREACHABLE(); } - + #endif return simde_mm512_movepi64_mask(simde_mm512_castpd_si512(simde__m512d_from_private(r_))); } #if defined(SIMDE_X86_AVX512F_NATIVE) diff --git a/simde/x86/sse.h b/simde/x86/sse.h index e5b694369..cf0f65a0e 100644 --- a/simde/x86/sse.h +++ b/simde/x86/sse.h @@ -664,7 +664,7 @@ simde_x_mm_round_ps (simde__m128 a, int rounding, int lax_rounding) r_.f32[i] = simde_math_nearbyintf(a_.f32[i]); } #else - HEDLEY_UNREACHABLE_RETURN(simde_mm_undefined_pd()); + HEDLEY_UNREACHABLE_RETURN(simde_mm_undefined_ps()); #endif break; @@ -683,7 +683,7 @@ simde_x_mm_round_ps (simde__m128 a, int rounding, int lax_rounding) r_.f32[i] = simde_math_roundevenf(a_.f32[i]); } #else - HEDLEY_UNREACHABLE_RETURN(simde_mm_undefined_pd()); + HEDLEY_UNREACHABLE_RETURN(simde_mm_undefined_ps()); #endif break; @@ -702,7 +702,7 @@ simde_x_mm_round_ps (simde__m128 a, int rounding, int lax_rounding) r_.f32[i] = simde_math_floorf(a_.f32[i]); } #else - HEDLEY_UNREACHABLE_RETURN(simde_mm_undefined_pd()); + HEDLEY_UNREACHABLE_RETURN(simde_mm_undefined_ps()); #endif break; @@ -721,7 +721,7 @@ simde_x_mm_round_ps (simde__m128 a, int rounding, int lax_rounding) r_.f32[i] = simde_math_ceilf(a_.f32[i]); } #else - HEDLEY_UNREACHABLE_RETURN(simde_mm_undefined_pd()); + HEDLEY_UNREACHABLE_RETURN(simde_mm_undefined_ps()); #endif break; @@ -740,12 +740,12 @@ simde_x_mm_round_ps (simde__m128 a, int rounding, int lax_rounding) r_.f32[i] = simde_math_truncf(a_.f32[i]); } #else - HEDLEY_UNREACHABLE_RETURN(simde_mm_undefined_pd()); + HEDLEY_UNREACHABLE_RETURN(simde_mm_undefined_ps()); #endif break; default: - HEDLEY_UNREACHABLE_RETURN(simde_mm_undefined_pd()); + HEDLEY_UNREACHABLE_RETURN(simde_mm_undefined_ps()); } return simde__m128_from_private(r_); @@ -4122,11 +4122,24 @@ simde_mm_shuffle_ps (simde__m128 a, simde__m128 b, const int imm8) r_, a_ = simde__m128_to_private(a), b_ = simde__m128_to_private(b); - - r_.f32[0] = a_.f32[(imm8 >> 0) & 3]; - r_.f32[1] = a_.f32[(imm8 >> 2) & 3]; - r_.f32[2] = b_.f32[(imm8 >> 4) & 3]; - r_.f32[3] = b_.f32[(imm8 >> 6) & 3]; + #if defined(SIMDE_ARM_NEON_A32V7_NATIVE) && !defined(SIMDE_STATEMENT_EXPR_) + #include "../arm/neon/set_lane.h" + #include "../arm/neon/dup_n.h" + simde_float32 temp = 0.0f; + SIMDE_CONSTIFY_4(vgetq_lane_f32, temp, (HEDLEY_UNREACHABLE(), 0.0f), (imm8) & (0x3), a_); + r_.neon_f32 = vmovq_n_f32(temp); + SIMDE_CONSTIFY_4(vgetq_lane_f32, temp, (HEDLEY_UNREACHABLE(), 0.0f), (((imm8) >> 2) & 0x3), a_); + r_.neon_f32 = vsetq_lane_f32(temp, r_, 1); + SIMDE_CONSTIFY_4(vgetq_lane_f32, temp, (HEDLEY_UNREACHABLE(), 0.0f), (((imm8) >> 4) & 0x3), b_); + r_.neon_f32 = vsetq_lane_f32(temp, r_, 2); + SIMDE_CONSTIFY_4(vgetq_lane_f32, temp, (HEDLEY_UNREACHABLE(), 0.0f), (((imm8) >> 6) & 0x3), b_); + r_.neon_f32 = vsetq_lane_f32(temp, r_, 3); + #else + r_.f32[0] = a_.f32[(imm8 >> 0) & 3]; + r_.f32[1] = a_.f32[(imm8 >> 2) & 3]; + r_.f32[2] = b_.f32[(imm8 >> 4) & 3]; + r_.f32[3] = b_.f32[(imm8 >> 6) & 3]; + #endif return simde__m128_from_private(r_); } diff --git a/simde/x86/sse2.h b/simde/x86/sse2.h index 1d73c89a0..1dc746509 100644 --- a/simde/x86/sse2.h +++ b/simde/x86/sse2.h @@ -5504,10 +5504,23 @@ simde_mm_shuffle_epi32 (simde__m128i a, const int imm8) simde__m128i_private r_, a_ = simde__m128i_to_private(a); - - for (size_t i = 0 ; i < (sizeof(r_.i32) / sizeof(r_.i32[0])) ; i++) { - r_.i32[i] = a_.i32[(imm8 >> (i * 2)) & 3]; - } + #if defined(SIMDE_ARM_NEON_A32V7_NATIVE) && !defined(SIMDE_STATEMENT_EXPR_) + #include "../arm/neon/set_lane.h" + #include "../arm/neon/dup_n.h" + int32_t temp; + SIMDE_CONSTIFY_4(vgetq_lane_s32, temp, (HEDLEY_UNREACHABLE(), 0), (imm8) & (0x3), a_); + r_.neon_i32 = vmovq_n_s32(temp); + SIMDE_CONSTIFY_4(vgetq_lane_s32, temp, (HEDLEY_UNREACHABLE(), 0), (((imm8) >> 2) & 0x3), a_); + r_.neon_i32 = vsetq_lane_s32(temp, r_, 1); + SIMDE_CONSTIFY_4(vgetq_lane_s32, temp, (HEDLEY_UNREACHABLE(), 0), (((imm8) >> 4) & 0x3), a_); + r_.neon_i32 = vsetq_lane_s32(temp, r_, 2); + SIMDE_CONSTIFY_4(vgetq_lane_s32, temp, (HEDLEY_UNREACHABLE(), 0), (((imm8) >> 6) & 0x3), a_); + r_.neon_i32 = vsetq_lane_s32(temp, r_, 3); + #else + for (size_t i = 0 ; i < (sizeof(r_.i32) / sizeof(r_.i32[0])) ; i++) { + r_.i32[i] = a_.i32[(imm8 >> (i * 2)) & 3]; + } + #endif return simde__m128i_from_private(r_); } @@ -5587,15 +5600,28 @@ simde_mm_shufflehi_epi16 (simde__m128i a, const int imm8) simde__m128i_private r_, a_ = simde__m128i_to_private(a); - - SIMDE_VECTORIZE - for (size_t i = 0 ; i < ((sizeof(a_.i16) / sizeof(a_.i16[0])) / 2) ; i++) { - r_.i16[i] = a_.i16[i]; - } - for (size_t i = ((sizeof(a_.i16) / sizeof(a_.i16[0])) / 2) ; i < (sizeof(r_.i16) / sizeof(r_.i16[0])) ; i++) { - r_.i16[i] = a_.i16[((imm8 >> ((i - 4) * 2)) & 3) + 4]; - } - + #if defined(SIMDE_ARM_NEON_A32V7_NATIVE) && !defined(SIMDE_STATEMENT_EXPR_) + #include "../arm/neon/set_lane.h" + #include "../arm/neon/dup_n.h" + r_ = a_; + int16_t temp; + SIMDE_CONSTIFY_8(vgetq_lane_s16, temp1, (HEDLEY_UNREACHABLE(), 0), ((imm8) & 0x3) + 4, a_); + r_.neon_i16 = vsetq_lane_s16(temp, r_, 4); + SIMDE_CONSTIFY_8(vgetq_lane_s16, temp1, (HEDLEY_UNREACHABLE(), 0), (((imm8) >> 2) & 0x3) + 4, a_); + r_.neon_i16 = vsetq_lane_s16(temp, r_, 5); + SIMDE_CONSTIFY_8(vgetq_lane_s16, temp1, (HEDLEY_UNREACHABLE(), 0), (((imm8) >> 4) & 0x3) + 4, a_); + r_.neon_i16 = vsetq_lane_s16(temp, r_, 6); + SIMDE_CONSTIFY_8(vgetq_lane_s16, temp1, (HEDLEY_UNREACHABLE(), 0), (((imm8) >> 6) & 0x3) + 4, a_); + r_.neon_i16 = vsetq_lane_s16(temp, r_, 7); + #else + SIMDE_VECTORIZE + for (size_t i = 0 ; i < ((sizeof(a_.i16) / sizeof(a_.i16[0])) / 2) ; i++) { + r_.i16[i] = a_.i16[i]; + } + for (size_t i = ((sizeof(a_.i16) / sizeof(a_.i16[0])) / 2) ; i < (sizeof(r_.i16) / sizeof(r_.i16[0])) ; i++) { + r_.i16[i] = a_.i16[((imm8 >> ((i - 4) * 2)) & 3) + 4]; + } + #endif return simde__m128i_from_private(r_); } #if defined(SIMDE_X86_SSE2_NATIVE) @@ -5647,14 +5673,28 @@ simde_mm_shufflelo_epi16 (simde__m128i a, const int imm8) simde__m128i_private r_, a_ = simde__m128i_to_private(a); - - for (size_t i = 0 ; i < ((sizeof(r_.i16) / sizeof(r_.i16[0])) / 2) ; i++) { - r_.i16[i] = a_.i16[((imm8 >> (i * 2)) & 3)]; - } - SIMDE_VECTORIZE - for (size_t i = ((sizeof(a_.i16) / sizeof(a_.i16[0])) / 2) ; i < (sizeof(r_.i16) / sizeof(r_.i16[0])) ; i++) { - r_.i16[i] = a_.i16[i]; - } + #if defined(SIMDE_ARM_NEON_A32V7_NATIVE) && !defined(SIMDE_STATEMENT_EXPR_) + #include "../arm/neon/set_lane.h" + #include "../arm/neon/dup_n.h" + r_ = a_; + int16_t temp; + SIMDE_CONSTIFY_8(vgetq_lane_s16, temp1, (HEDLEY_UNREACHABLE(), 0), ((imm8) & 0x3), a_); + r_.neon_i16 = vsetq_lane_s16(temp, r_, 0); + SIMDE_CONSTIFY_8(vgetq_lane_s16, temp1, (HEDLEY_UNREACHABLE(), 0), (((imm8) >> 2) & 0x3), a_); + r_.neon_i16 = vsetq_lane_s16(temp, r_, 1); + SIMDE_CONSTIFY_8(vgetq_lane_s16, temp1, (HEDLEY_UNREACHABLE(), 0), (((imm8) >> 4) & 0x3), a_); + r_.neon_i16 = vsetq_lane_s16(temp, r_, 2); + SIMDE_CONSTIFY_8(vgetq_lane_s16, temp1, (HEDLEY_UNREACHABLE(), 0), (((imm8) >> 6) & 0x3), a_); + r_.neon_i16 = vsetq_lane_s16(temp, r_, 3); + #else + for (size_t i = 0 ; i < ((sizeof(r_.i16) / sizeof(r_.i16[0])) / 2) ; i++) { + r_.i16[i] = a_.i16[((imm8 >> (i * 2)) & 3)]; + } + SIMDE_VECTORIZE + for (size_t i = ((sizeof(a_.i16) / sizeof(a_.i16[0])) / 2) ; i < (sizeof(r_.i16) / sizeof(r_.i16[0])) ; i++) { + r_.i16[i] = a_.i16[i]; + } + #endif return simde__m128i_from_private(r_); }