emscripten/tests/test_sse1_full.cpp

144 строки
4.7 KiB
C++

// This file uses SSE1 by calling different functions with different interesting inputs and prints the results.
// Use a diff tool to compare the results between platforms.
#include <xmmintrin.h>
#include "test_sse_full.h"
int main()
{
float *interesting_floats = get_interesting_floats();
int numInterestingFloats = sizeof(interesting_floats_)/sizeof(interesting_floats_[0]);
assert(numInterestingFloats % 4 == 0);
uint32_t *interesting_ints = get_interesting_ints();
int numInterestingInts = sizeof(interesting_ints_)/sizeof(interesting_ints_[0]);
assert(numInterestingInts % 4 == 0);
// SSE1 Arithmetic instructions:
Ret_M128_M128(__m128, _mm_add_ps);
Ret_M128_M128(__m128, _mm_add_ss);
Ret_M128_M128(__m128, _mm_div_ps);
Ret_M128_M128(__m128, _mm_div_ss);
Ret_M128_M128(__m128, _mm_mul_ps);
Ret_M128_M128(__m128, _mm_mul_ss);
Ret_M128_M128(__m128, _mm_sub_ps);
Ret_M128_M128(__m128, _mm_sub_ss);
// SSE1 Elementary Math functions:
#if 0 // TODO: Precision differs in SIMD.js and native. Test differently. See https://github.com/kripken/emscripten/issues/3049
Ret_M128(__m128, _mm_rcp_ps);
Ret_M128(__m128, _mm_rcp_ss);
Ret_M128(__m128, _mm_rsqrt_ps);
Ret_M128(__m128, _mm_rsqrt_ss);
Ret_M128(__m128, _mm_sqrt_ps);
Ret_M128(__m128, _mm_sqrt_ss);
#endif
// SSE1 Logical instructions:
Ret_M128_M128(__m128, _mm_and_ps);
Ret_M128_M128(__m128, _mm_andnot_ps);
Ret_M128_M128(__m128, _mm_or_ps);
Ret_M128_M128(__m128, _mm_xor_ps);
// SSE1 Compare instructions:
Ret_M128_M128(__m128, _mm_cmpeq_ps);
Ret_M128_M128(__m128, _mm_cmpeq_ss);
Ret_M128_M128(__m128, _mm_cmpge_ps);
Ret_M128_M128(__m128, _mm_cmpge_ss);
Ret_M128_M128(__m128, _mm_cmpgt_ps);
Ret_M128_M128(__m128, _mm_cmpgt_ss);
Ret_M128_M128(__m128, _mm_cmple_ps);
Ret_M128_M128(__m128, _mm_cmple_ss);
Ret_M128_M128(__m128, _mm_cmplt_ps);
Ret_M128_M128(__m128, _mm_cmplt_ss);
Ret_M128_M128(__m128, _mm_cmpneq_ps);
Ret_M128_M128(__m128, _mm_cmpneq_ss);
Ret_M128_M128(__m128, _mm_cmpnge_ps);
Ret_M128_M128(__m128, _mm_cmpnge_ss);
Ret_M128_M128(__m128, _mm_cmpngt_ps);
Ret_M128_M128(__m128, _mm_cmpngt_ss);
Ret_M128_M128(__m128, _mm_cmpnle_ps);
Ret_M128_M128(__m128, _mm_cmpnle_ss);
Ret_M128_M128(__m128, _mm_cmpnlt_ps);
Ret_M128_M128(__m128, _mm_cmpnlt_ss);
Ret_M128_M128(__m128, _mm_cmpord_ps);
Ret_M128_M128(__m128, _mm_cmpord_ss);
Ret_M128_M128(__m128, _mm_cmpunord_ps);
Ret_M128_M128(__m128, _mm_cmpunord_ss);
Ret_M128_M128(int, _mm_comieq_ss);
Ret_M128_M128(int, _mm_comige_ss);
Ret_M128_M128(int, _mm_comigt_ss);
Ret_M128_M128(int, _mm_comile_ss);
Ret_M128_M128(int, _mm_comilt_ss);
Ret_M128_M128(int, _mm_comineq_ss);
Ret_M128_M128(int, _mm_ucomieq_ss);
Ret_M128_M128(int, _mm_ucomige_ss);
Ret_M128_M128(int, _mm_ucomigt_ss);
Ret_M128_M128(int, _mm_ucomile_ss);
Ret_M128_M128(int, _mm_ucomilt_ss);
Ret_M128_M128(int, _mm_ucomineq_ss);
// SSE1 Convert instructions:
Ret_M128_int(__m128, _mm_cvt_si2ss);
Ret_M128(int, _mm_cvt_ss2si);
Ret_M128_int(__m128, _mm_cvtsi32_ss);
Ret_M128(float, _mm_cvtss_f32);
Ret_M128(int, _mm_cvtss_si32);
Ret_M128(int64_t, _mm_cvtss_si64);
Ret_M128(int, _mm_cvtt_ss2si);
Ret_M128(int, _mm_cvttss_si32);
Ret_M128(int64_t, _mm_cvttss_si64);
// SSE1 Load functions:
Ret_FloatPtr(__m128, _mm_load_ps, 4, 4);
Ret_FloatPtr(__m128, _mm_load_ps1, 1, 1);
Ret_FloatPtr(__m128, _mm_load_ss, 1, 1);
Ret_FloatPtr(__m128, _mm_load1_ps, 1, 1);
Ret_M128_FloatPtr(__m128, _mm_loadh_pi, __m64*, 2, 1);
Ret_M128_FloatPtr(__m128, _mm_loadl_pi, __m64*, 2, 1);
Ret_FloatPtr(__m128, _mm_loadr_ps, 4, 4);
Ret_FloatPtr(__m128, _mm_loadu_ps, 4, 1);
// SSE1 Miscellaneous functions:
Ret_M128(int, _mm_movemask_ps);
// SSE1 Move functions:
Ret_M128_M128(__m128, _mm_move_ss);
Ret_M128_M128(__m128, _mm_movehl_ps);
Ret_M128_M128(__m128, _mm_movelh_ps);
/*
// SSE1 Set functions:
_mm_set_ps
_mm_set_ps1
_mm_set_ss
_mm_set1_ps
_mm_setr_ps
_mm_setzero_ps
*/
// SSE1 Special Math instructions:
Ret_M128_M128(__m128, _mm_max_ps);
Ret_M128_M128(__m128, _mm_max_ss);
Ret_M128_M128(__m128, _mm_min_ps);
Ret_M128_M128(__m128, _mm_min_ss);
// SSE1 Store instructions:
void_OutFloatPtr_M128(_mm_store_ps, float*, 16, 16);
void_OutFloatPtr_M128(_mm_store_ps1, float*, 16, 16);
void_OutFloatPtr_M128(_mm_store_ss, float*, 4, 1);
void_OutFloatPtr_M128(_mm_store1_ps, float*, 16, 16);
void_OutFloatPtr_M128(_mm_storeh_pi, __m64*, 8, 1);
void_OutFloatPtr_M128(_mm_storel_pi, __m64*, 8, 1);
void_OutFloatPtr_M128(_mm_storer_ps, float*, 16, 16);
void_OutFloatPtr_M128(_mm_storeu_ps, float*, 16, 1);
void_OutFloatPtr_M128(_mm_stream_ps, float*, 16, 16);
// SSE1 Swizzle instructions:
Ret_M128_M128_Tint(__m128, _mm_shuffle_ps);
// _MM_TRANSPOSE4_PS
Ret_M128_M128(__m128, _mm_unpackhi_ps);
Ret_M128_M128(__m128, _mm_unpacklo_ps);
}