simd_t.cpp 9.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439
  1. /*
  2. * Copyright 2010-2016 Branimir Karadzic. All rights reserved.
  3. * License: https://github.com/bkaradzic/bx#license-bsd-2-clause
  4. */
  5. #include "test.h"
  6. #include <bx/simd_t.h>
  7. #include <bx/fpumath.h>
  8. #include <string.h>
  9. using namespace bx;
  10. union simd_cast
  11. {
  12. bx::simd256_t simd256;
  13. bx::simd128_t simd128;
  14. float f[8];
  15. uint32_t ui[8];
  16. int32_t i[8];
  17. char c[32];
  18. };
  19. void simd_check_bool(const char* _str, bool _a, bool _0)
  20. {
  21. DBG("%s %d == %d"
  22. , _str
  23. , _a
  24. , _0
  25. );
  26. CHECK_EQUAL(_a, _0);
  27. }
  28. void simd_check_int32(
  29. const char* _str
  30. , bx::simd128_t _a
  31. , int32_t _0
  32. , int32_t _1
  33. , int32_t _2
  34. , int32_t _3
  35. )
  36. {
  37. simd_cast c; c.simd128 = _a;
  38. DBG("%s (%d, %d, %d, %d) == (%d, %d, %d, %d)"
  39. , _str
  40. , c.i[0], c.i[1], c.i[2], c.i[3]
  41. , _0, _1, _2, _3
  42. );
  43. CHECK_EQUAL(c.i[0], _0);
  44. CHECK_EQUAL(c.i[1], _1);
  45. CHECK_EQUAL(c.i[2], _2);
  46. CHECK_EQUAL(c.i[3], _3);
  47. }
  48. #if 0
  49. void simd_check_int32(
  50. const char* _str
  51. , bx::simd256_t _a
  52. , int32_t _0
  53. , int32_t _1
  54. , int32_t _2
  55. , int32_t _3
  56. , int32_t _4
  57. , int32_t _5
  58. , int32_t _6
  59. , int32_t _7
  60. )
  61. {
  62. simd_cast c; c.simd256 = _a;
  63. DBG("%s (%d, %d, %d, %d, %d, %d, %d, %d) == (%d, %d, %d, %d, %d, %d, %d, %d)"
  64. , _str
  65. , c.i[0], c.i[1], c.i[2], c.i[3], c.i[4], c.i[5], c.i[6], c.i[7]
  66. , _0, _1, _2, _3, _4, _5, _6, _7
  67. );
  68. CHECK_EQUAL(c.i[0], _0);
  69. CHECK_EQUAL(c.i[1], _1);
  70. CHECK_EQUAL(c.i[2], _2);
  71. CHECK_EQUAL(c.i[3], _3);
  72. CHECK_EQUAL(c.i[4], _4);
  73. CHECK_EQUAL(c.i[5], _5);
  74. CHECK_EQUAL(c.i[6], _6);
  75. CHECK_EQUAL(c.i[7], _7);
  76. }
  77. #endif // 0
  78. void simd_check_uint32(
  79. const char* _str
  80. , bx::simd128_t _a
  81. , uint32_t _0
  82. , uint32_t _1
  83. , uint32_t _2
  84. , uint32_t _3
  85. )
  86. {
  87. simd_cast c; c.simd128 = _a;
  88. DBG("%s (0x%08x, 0x%08x, 0x%08x, 0x%08x) == (0x%08x, 0x%08x, 0x%08x, 0x%08x)"
  89. , _str
  90. , c.ui[0], c.ui[1], c.ui[2], c.ui[3]
  91. , _0, _1, _2, _3
  92. );
  93. CHECK_EQUAL(c.ui[0], _0);
  94. CHECK_EQUAL(c.ui[1], _1);
  95. CHECK_EQUAL(c.ui[2], _2);
  96. CHECK_EQUAL(c.ui[3], _3);
  97. }
  98. #if 0
  99. void simd_check_uint32(
  100. const char* _str
  101. , bx::simd256_t _a
  102. , uint32_t _0
  103. , uint32_t _1
  104. , uint32_t _2
  105. , uint32_t _3
  106. , uint32_t _4
  107. , uint32_t _5
  108. , uint32_t _6
  109. , uint32_t _7
  110. )
  111. {
  112. simd_cast c; c.simd256 = _a;
  113. DBG("%s (0x%08x, 0x%08x, 0x%08x, 0x%08x, 0x%08x, 0x%08x, 0x%08x, 0x%08x) == (0x%08x, 0x%08x, 0x%08x, 0x%08x, 0x%08x, 0x%08x, 0x%08x, 0x%08x)"
  114. , _str
  115. , c.ui[0], c.ui[1], c.ui[2], c.ui[3], c.ui[4], c.ui[5], c.ui[6], c.ui[7]
  116. , _0, _1, _2, _3, _4, _5, _6, _7
  117. );
  118. CHECK_EQUAL(c.ui[0], _0);
  119. CHECK_EQUAL(c.ui[1], _1);
  120. CHECK_EQUAL(c.ui[2], _2);
  121. CHECK_EQUAL(c.ui[3], _3);
  122. CHECK_EQUAL(c.ui[4], _4);
  123. CHECK_EQUAL(c.ui[5], _5);
  124. CHECK_EQUAL(c.ui[6], _6);
  125. CHECK_EQUAL(c.ui[7], _7);
  126. }
  127. #endif // 0
  128. void simd_check_float(
  129. const char* _str
  130. , bx::simd128_t _a
  131. , float _0
  132. , float _1
  133. , float _2
  134. , float _3
  135. )
  136. {
  137. simd_cast c; c.simd128 = _a;
  138. DBG("%s (%f, %f, %f, %f) == (%f, %f, %f, %f)"
  139. , _str
  140. , c.f[0], c.f[1], c.f[2], c.f[3]
  141. , _0, _1, _2, _3
  142. );
  143. CHECK(bx::fequal(c.f[0], _0, 0.0001f) );
  144. CHECK(bx::fequal(c.f[1], _1, 0.0001f) );
  145. CHECK(bx::fequal(c.f[2], _2, 0.0001f) );
  146. CHECK(bx::fequal(c.f[3], _3, 0.0001f) );
  147. }
  148. #if 0
  149. void simd_check_float(
  150. const char* _str
  151. , bx::simd256_t _a
  152. , float _0
  153. , float _1
  154. , float _2
  155. , float _3
  156. , float _4
  157. , float _5
  158. , float _6
  159. , float _7
  160. )
  161. {
  162. simd_cast c; c.simd256 = _a;
  163. DBG("%s (%f, %f, %f, %f, %f, %f, %f, %f) == (%f, %f, %f, %f, %f, %f, %f, %f)"
  164. , _str
  165. , c.f[0], c.f[1], c.f[2], c.f[3], c.f[4], c.f[5], c.f[6], c.f[7]
  166. , _0, _1, _2, _3, _4, _5, _6, _7
  167. );
  168. CHECK(bx::fequal(c.f[0], _0, 0.0001f) );
  169. CHECK(bx::fequal(c.f[1], _1, 0.0001f) );
  170. CHECK(bx::fequal(c.f[2], _2, 0.0001f) );
  171. CHECK(bx::fequal(c.f[3], _3, 0.0001f) );
  172. CHECK(bx::fequal(c.f[4], _4, 0.0001f) );
  173. CHECK(bx::fequal(c.f[5], _5, 0.0001f) );
  174. CHECK(bx::fequal(c.f[6], _6, 0.0001f) );
  175. CHECK(bx::fequal(c.f[7], _7, 0.0001f) );
  176. }
  177. #endif // 0
  178. void simd_check_string(const char* _str, bx::simd128_t _a)
  179. {
  180. simd_cast c; c.simd128 = _a;
  181. const char test[5] = { c.c[0], c.c[4], c.c[8], c.c[12], '\0' };
  182. DBG("%s %s", _str, test);
  183. CHECK(0 == strcmp(_str, test) );
  184. }
  185. TEST(simd_swizzle)
  186. {
  187. const simd128_t xyzw = simd_ild(0x78787878, 0x79797979, 0x7a7a7a7a, 0x77777777);
  188. #define ELEMx 0
  189. #define ELEMy 1
  190. #define ELEMz 2
  191. #define ELEMw 3
  192. #define BX_SIMD128_IMPLEMENT_SWIZZLE(_x, _y, _z, _w) \
  193. simd_check_string("" #_x #_y #_z #_w "", simd_swiz_##_x##_y##_z##_w(xyzw) ); \
  194. #include <bx/simd128_swizzle.inl>
  195. #undef BX_SIMD128_IMPLEMENT_SWIZZLE
  196. #undef ELEMw
  197. #undef ELEMz
  198. #undef ELEMy
  199. #undef ELEMx
  200. }
  201. TEST(simd_shuffle)
  202. {
  203. const simd128_t xyzw = simd_ild(0x78787878, 0x79797979, 0x7a7a7a7a, 0x77777777);
  204. const simd128_t ABCD = simd_ild(0x41414141, 0x42424242, 0x43434343, 0x44444444);
  205. simd_check_string("xyAB", simd_shuf_xyAB(xyzw, ABCD) );
  206. simd_check_string("ABxy", simd_shuf_ABxy(xyzw, ABCD) );
  207. simd_check_string("zwCD", simd_shuf_zwCD(xyzw, ABCD) );
  208. simd_check_string("CDzw", simd_shuf_CDzw(xyzw, ABCD) );
  209. simd_check_string("xAyB", simd_shuf_xAyB(xyzw, ABCD) );
  210. simd_check_string("zCwD", simd_shuf_zCwD(xyzw, ABCD) );
  211. simd_check_string("xAzC", simd_shuf_xAzC(xyzw, ABCD) );
  212. simd_check_string("yBwD", simd_shuf_yBwD(xyzw, ABCD) );
  213. simd_check_string("CzDw", simd_shuf_CzDw(xyzw, ABCD) );
  214. }
  215. TEST(simd_compare)
  216. {
  217. simd_check_uint32("cmpeq"
  218. , simd_cmpeq(simd_ld(1.0f, 2.0f, 3.0f, 4.0f), simd_ld(0.0f, 2.0f, 0.0f, 3.0f) )
  219. , 0, 0xffffffff, 0, 0
  220. );
  221. simd_check_uint32("cmplt"
  222. , simd_cmplt(simd_ld(1.0f, 2.0f, 3.0f, 4.0f), simd_ld(0.0f, 2.0f, 0.0f, 3.0f) )
  223. , 0, 0, 0, 0
  224. );
  225. simd_check_uint32("cmple"
  226. , simd_cmple(simd_ld(1.0f, 2.0f, 3.0f, 4.0f), simd_ld(0.0f, 2.0f, 0.0f, 3.0f) )
  227. , 0, 0xffffffff, 0, 0
  228. );
  229. simd_check_uint32("cmpgt"
  230. , simd_cmpgt(simd_ld(1.0f, 2.0f, 3.0f, 4.0f), simd_ld(0.0f, 2.0f, 0.0f, 3.0f) )
  231. , 0xffffffff, 0, 0xffffffff, 0xffffffff
  232. );
  233. simd_check_uint32("cmpge"
  234. , simd_cmpge(simd_ld(1.0f, 2.0f, 3.0f, 4.0f), simd_ld(0.0f, 2.0f, 0.0f, 3.0f) )
  235. , 0xffffffff, 0xffffffff, 0xffffffff, 0xffffffff
  236. );
  237. simd_check_uint32("icmpeq"
  238. , simd_icmpeq(simd_ild(0, 1, 2, 3), simd_ild(0, uint32_t(-2), 1, 3) )
  239. , 0xffffffff, 0, 0, 0xffffffff
  240. );
  241. simd_check_uint32("icmplt"
  242. , simd_icmplt(simd_ild(0, 1, 2, 3), simd_ild(0, uint32_t(-2), 1, 3) )
  243. , 0, 0, 0, 0
  244. );
  245. simd_check_uint32("icmpgt"
  246. , simd_icmpgt(simd_ild(0, 1, 2, 3), simd_ild(0, uint32_t(-2), 1, 3) )
  247. , 0, 0xffffffff, 0xffffffff, 0
  248. );
  249. }
  250. TEST(simd_test)
  251. {
  252. simd_check_bool("test_any_xyzw"
  253. , simd_test_any_xyzw(simd_ild(0xffffffff, 0, 0, 0) )
  254. , true
  255. );
  256. simd_check_bool("test_all_xyzw"
  257. , simd_test_all_xyzw(simd_ild(0xffffffff, 0, 0xffffffff, 0) )
  258. , false
  259. );
  260. simd_check_bool("test_all_xyzw"
  261. , simd_test_all_xyzw(simd_ild(0xffffffff, 0xffffffff, 0xffffffff, 0xffffffff) )
  262. , true
  263. );
  264. simd_check_bool("test_all_xw"
  265. , simd_test_all_xw(simd_ild(0xffffffff, 0, 0, 0xffffffff) )
  266. , true
  267. );
  268. simd_check_bool("test_all_xzw"
  269. , simd_test_all_xzw(simd_ild(0xffffffff, 0, 0, 0xffffffff) )
  270. , false
  271. );
  272. }
  273. TEST(simd_load)
  274. {
  275. simd_check_float("ld"
  276. , simd_ld(0.0f, 1.0f, 2.0f, 3.0f)
  277. , 0.0f, 1.0f, 2.0f, 3.0f
  278. );
  279. // simd_check_float("ld"
  280. // , simd_ld<simd256_t>(0.0f, 1.0f, 2.0f, 3.0f, 4.0f, 5.0f, 6.0f, 7.0f)
  281. // , 0.0f, 1.0f, 2.0f, 3.0f, 4.0f, 5.0f, 6.0f, 7.0f
  282. // );
  283. simd_check_int32("ild"
  284. , simd_ild(uint32_t(-1), 0, 1, 2)
  285. , uint32_t(-1), 0, 1, 2
  286. );
  287. // simd_check_int32("ild"
  288. // , simd_ild<simd256_t>(uint32_t(-1), 0, 1, 2, 3, 4, 5, 6)
  289. // , uint32_t(-1), 0, 1, 2, 3, 4, 5, 6
  290. // );
  291. simd_check_int32("ild"
  292. , simd_ild(uint32_t(-1), uint32_t(-2), uint32_t(-3), uint32_t(-4) )
  293. , uint32_t(-1), uint32_t(-2), uint32_t(-3), uint32_t(-4)
  294. );
  295. simd_check_uint32("zero", simd_zero()
  296. , 0, 0, 0, 0
  297. );
  298. simd_check_uint32("isplat", simd_isplat(0x80000001)
  299. , 0x80000001, 0x80000001, 0x80000001, 0x80000001
  300. );
  301. simd_check_float("isplat", simd_splat(1.0f)
  302. , 1.0f, 1.0f, 1.0f, 1.0f
  303. );
  304. }
  305. TEST(simd_arithmetic)
  306. {
  307. simd_check_float("madd"
  308. , simd_madd(simd_ld(0.0f, 1.0f, 2.0f, 3.0f), simd_ld(4.0f, 5.0f, 6.0f, 7.0f), simd_ld(8.0f, 9.0f, 10.0f, 11.0f) )
  309. , 8.0f, 14.0f, 22.0f, 32.0f
  310. );
  311. simd_check_float("cross3"
  312. , simd_cross3(simd_ld(1.0f, 0.0f, 0.0f, 0.0f), simd_ld(0.0f, 1.0f, 0.0f, 0.0f) )
  313. , 0.0f, 0.0f, 1.0f, 0.0f
  314. );
  315. }
  316. TEST(simd_sqrt)
  317. {
  318. simd_check_float("simd_sqrt"
  319. , simd_sqrt(simd_ld(1.0f, 16.0f, 65536.0f, 123456.0f) )
  320. , 1.0f, 4.0f, 256.0f, 351.363060096f
  321. );
  322. simd_check_float("simd_sqrt_nr_ni"
  323. , simd_sqrt_nr_ni(simd_ld(1.0f, 16.0f, 65536.0f, 123456.0f) )
  324. , 1.0f, 4.0f, 256.0f, 351.363060096f
  325. );
  326. simd_check_float("simd_sqrt_nr1_ni"
  327. , simd_sqrt_nr1_ni(simd_ld(1.0f, 16.0f, 65536.0f, 123456.0f) )
  328. , 1.0f, 4.0f, 256.0f, 351.363060096f
  329. );
  330. }
  331. TEST(float4)
  332. {
  333. const simd128_t isplat = simd_isplat(0x80000001);
  334. simd_check_uint32("sll"
  335. , simd_sll(isplat, 1)
  336. , 0x00000002, 0x00000002, 0x00000002, 0x00000002
  337. );
  338. simd_check_uint32("srl"
  339. , simd_srl(isplat, 1)
  340. , 0x40000000, 0x40000000, 0x40000000, 0x40000000
  341. );
  342. simd_check_uint32("sra"
  343. , simd_sra(isplat, 1)
  344. , 0xc0000000, 0xc0000000, 0xc0000000, 0xc0000000
  345. );
  346. simd_check_uint32("and"
  347. , simd_and(simd_isplat(0x55555555), simd_isplat(0xaaaaaaaa) )
  348. , 0, 0, 0, 0
  349. );
  350. simd_check_uint32("or "
  351. , simd_or(simd_isplat(0x55555555), simd_isplat(0xaaaaaaaa) )
  352. , uint32_t(-1), uint32_t(-1), uint32_t(-1), uint32_t(-1)
  353. );
  354. simd_check_uint32("xor"
  355. , simd_or(simd_isplat(0x55555555), simd_isplat(0xaaaaaaaa) )
  356. , uint32_t(-1), uint32_t(-1), uint32_t(-1), uint32_t(-1)
  357. );
  358. simd_check_int32("imin"
  359. , simd_imin(simd_ild(0, 1, 2, 3), simd_ild(uint32_t(-1), 2, uint32_t(-2), 1) )
  360. , uint32_t(-1), 1, uint32_t(-2), 1
  361. );
  362. simd_check_float("min"
  363. , simd_min(simd_ld(0.0f, 1.0f, 2.0f, 3.0f), simd_ld(-1.0f, 2.0f, -2.0f, 1.0f) )
  364. , -1.0f, 1.0f, -2.0f, 1.0f
  365. );
  366. simd_check_int32("imax"
  367. , simd_imax(simd_ild(0, 1, 2, 3), simd_ild(uint32_t(-1), 2, uint32_t(-2), 1) )
  368. , 0, 2, 2, 3
  369. );
  370. simd_check_float("max"
  371. , simd_max(simd_ld(0.0f, 1.0f, 2.0f, 3.0f), simd_ld(-1.0f, 2.0f, -2.0f, 1.0f) )
  372. , 0.0f, 2.0f, 2.0f, 3.0f
  373. );
  374. }