test_shuffle_roundtrip_avx2.c 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137
  1. /*********************************************************************
  2. Blosc - Blocked Shuffling and Compression Library
  3. Roundtrip tests for the AVX2-accelerated shuffle/unshuffle.
  4. Creation date: 2010-06-07
  5. Author: Francesc Alted <francesc@blosc.org>
  6. See LICENSES/BLOSC.txt for details about copyright and rights to use.
  7. **********************************************************************/
  8. #include "test_common.h"
  9. #include "../blosc/shuffle.h"
  10. #include "../blosc/shuffle-generic.h"
  11. /* Include accelerated shuffles if supported by this compiler.
  12. TODO: Need to also do run-time CPU feature support here. */
  13. #if defined(SHUFFLE_AVX2_ENABLED)
  14. #include "../blosc/shuffle-avx2.h"
  15. #else
  16. #if defined(_MSC_VER)
  17. #pragma message("AVX2 shuffle tests not enabled.")
  18. #else
  19. #warning AVX2 shuffle tests not enabled.
  20. #endif
  21. #endif /* defined(SHUFFLE_AVX2_ENABLED) */
  22. /** Roundtrip tests for the AVX2-accelerated shuffle/unshuffle. */
  23. static int test_shuffle_roundtrip_avx2(size_t type_size, size_t num_elements,
  24. size_t buffer_alignment, int test_type)
  25. {
  26. #if defined(SHUFFLE_AVX2_ENABLED)
  27. size_t buffer_size = type_size * num_elements;
  28. /* Allocate memory for the test. */
  29. void* original = blosc_test_malloc(buffer_alignment, buffer_size);
  30. void* shuffled = blosc_test_malloc(buffer_alignment, buffer_size);
  31. void* unshuffled = blosc_test_malloc(buffer_alignment, buffer_size);
  32. /* Fill the input data buffer with random values. */
  33. blosc_test_fill_random(original, buffer_size);
  34. /* Shuffle/unshuffle, selecting the implementations based on the test type. */
  35. switch(test_type)
  36. {
  37. case 0:
  38. /* avx2/avx2 */
  39. shuffle_avx2(type_size, buffer_size, original, shuffled);
  40. unshuffle_avx2(type_size, buffer_size, shuffled, unshuffled);
  41. break;
  42. case 1:
  43. /* generic/avx2 */
  44. shuffle_generic(type_size, buffer_size, original, shuffled);
  45. unshuffle_avx2(type_size, buffer_size, shuffled, unshuffled);
  46. break;
  47. case 2:
  48. /* avx2/generic */
  49. shuffle_avx2(type_size, buffer_size, original, shuffled);
  50. unshuffle_generic(type_size, buffer_size, shuffled, unshuffled);
  51. break;
  52. default:
  53. fprintf(stderr, "Invalid test type specified (%d).", test_type);
  54. return EXIT_FAILURE;
  55. }
  56. /* The round-tripped data matches the original data when the
  57. result of memcmp is 0. */
  58. int exit_code = memcmp(original, unshuffled, buffer_size) ?
  59. EXIT_FAILURE : EXIT_SUCCESS;
  60. /* Free allocated memory. */
  61. blosc_test_free(original);
  62. blosc_test_free(shuffled);
  63. blosc_test_free(unshuffled);
  64. return exit_code;
  65. #else
  66. return EXIT_SUCCESS;
  67. #endif /* defined(SHUFFLE_AVX2_ENABLED) */
  68. }
  69. /** Required number of arguments to this test, including the executable name. */
  70. #define TEST_ARG_COUNT 5
  71. int main(int argc, char **argv)
  72. {
  73. uint32_t type_size;
  74. uint32_t num_elements;
  75. uint32_t buffer_align_size;
  76. uint32_t test_type;
  77. /* argv[1]: sizeof(element type)
  78. argv[2]: number of elements
  79. argv[3]: buffer alignment
  80. argv[4]: test type
  81. */
  82. /* Verify the correct number of command-line args have been specified. */
  83. if (TEST_ARG_COUNT != argc)
  84. {
  85. blosc_test_print_bad_argcount_msg(TEST_ARG_COUNT, argc);
  86. return EXIT_FAILURE;
  87. }
  88. /* Parse arguments */
  89. if (!blosc_test_parse_uint32_t(argv[1], &type_size) || (type_size < 1))
  90. {
  91. blosc_test_print_bad_arg_msg(1);
  92. return EXIT_FAILURE;
  93. }
  94. if (!blosc_test_parse_uint32_t(argv[2], &num_elements) || (num_elements < 1))
  95. {
  96. blosc_test_print_bad_arg_msg(2);
  97. return EXIT_FAILURE;
  98. }
  99. if (!blosc_test_parse_uint32_t(argv[3], &buffer_align_size)
  100. || (buffer_align_size & (buffer_align_size - 1))
  101. || (buffer_align_size < sizeof(void*)))
  102. {
  103. blosc_test_print_bad_arg_msg(3);
  104. return EXIT_FAILURE;
  105. }
  106. if (!blosc_test_parse_uint32_t(argv[4], &test_type) || (test_type > 2))
  107. {
  108. blosc_test_print_bad_arg_msg(4);
  109. return EXIT_FAILURE;
  110. }
  111. /* Run the test. */
  112. return test_shuffle_roundtrip_avx2(type_size, num_elements, buffer_align_size, test_type);
  113. }