compare256_rle.h 3.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134
  1. /* compare256_rle.h -- 256 byte run-length encoding comparison
  2. * Copyright (C) 2022 Nathan Moinvaziri
  3. * For conditions of distribution and use, see copyright notice in zlib.h
  4. */
  5. #include "zbuild.h"
  6. #include "fallback_builtins.h"
  7. typedef uint32_t (*compare256_rle_func)(const uint8_t* src0, const uint8_t* src1);
  8. /* ALIGNED, byte comparison */
  9. static inline uint32_t compare256_rle_c(const uint8_t *src0, const uint8_t *src1) {
  10. uint32_t len = 0;
  11. do {
  12. if (*src0 != *src1)
  13. return len;
  14. src1 += 1, len += 1;
  15. if (*src0 != *src1)
  16. return len;
  17. src1 += 1, len += 1;
  18. if (*src0 != *src1)
  19. return len;
  20. src1 += 1, len += 1;
  21. if (*src0 != *src1)
  22. return len;
  23. src1 += 1, len += 1;
  24. if (*src0 != *src1)
  25. return len;
  26. src1 += 1, len += 1;
  27. if (*src0 != *src1)
  28. return len;
  29. src1 += 1, len += 1;
  30. if (*src0 != *src1)
  31. return len;
  32. src1 += 1, len += 1;
  33. if (*src0 != *src1)
  34. return len;
  35. src1 += 1, len += 1;
  36. } while (len < 256);
  37. return 256;
  38. }
  39. #ifdef UNALIGNED_OK
  40. /* 16-bit unaligned integer comparison */
  41. static inline uint32_t compare256_rle_unaligned_16(const uint8_t *src0, const uint8_t *src1) {
  42. uint32_t len = 0;
  43. uint16_t src0_cmp, src1_cmp;
  44. memcpy(&src0_cmp, src0, sizeof(src0_cmp));
  45. do {
  46. memcpy(&src1_cmp, src1, sizeof(src1_cmp));
  47. if (src0_cmp != src1_cmp)
  48. return len + (*src0 == *src1);
  49. src1 += 2, len += 2;
  50. memcpy(&src1_cmp, src1, sizeof(src1_cmp));
  51. if (src0_cmp != src1_cmp)
  52. return len + (*src0 == *src1);
  53. src1 += 2, len += 2;
  54. memcpy(&src1_cmp, src1, sizeof(src1_cmp));
  55. if (src0_cmp != src1_cmp)
  56. return len + (*src0 == *src1);
  57. src1 += 2, len += 2;
  58. memcpy(&src1_cmp, src1, sizeof(src1_cmp));
  59. if (src0_cmp != src1_cmp)
  60. return len + (*src0 == *src1);
  61. src1 += 2, len += 2;
  62. } while (len < 256);
  63. return 256;
  64. }
  65. #ifdef HAVE_BUILTIN_CTZ
  66. /* 32-bit unaligned integer comparison */
  67. static inline uint32_t compare256_rle_unaligned_32(const uint8_t *src0, const uint8_t *src1) {
  68. uint32_t sv, len = 0;
  69. uint16_t src0_cmp;
  70. memcpy(&src0_cmp, src0, sizeof(src0_cmp));
  71. sv = ((uint32_t)src0_cmp << 16) | src0_cmp;
  72. do {
  73. uint32_t mv, diff;
  74. memcpy(&mv, src1, sizeof(mv));
  75. diff = sv ^ mv;
  76. if (diff) {
  77. uint32_t match_byte = __builtin_ctz(diff) / 8;
  78. return len + match_byte;
  79. }
  80. src1 += 4, len += 4;
  81. } while (len < 256);
  82. return 256;
  83. }
  84. #endif
  85. #if defined(UNALIGNED64_OK) && defined(HAVE_BUILTIN_CTZLL)
  86. /* 64-bit unaligned integer comparison */
  87. static inline uint32_t compare256_rle_unaligned_64(const uint8_t *src0, const uint8_t *src1) {
  88. uint32_t src0_cmp32, len = 0;
  89. uint16_t src0_cmp;
  90. uint64_t sv;
  91. memcpy(&src0_cmp, src0, sizeof(src0_cmp));
  92. src0_cmp32 = ((uint32_t)src0_cmp << 16) | src0_cmp;
  93. sv = ((uint64_t)src0_cmp32 << 32) | src0_cmp32;
  94. do {
  95. uint64_t mv, diff;
  96. memcpy(&mv, src1, sizeof(mv));
  97. diff = sv ^ mv;
  98. if (diff) {
  99. uint64_t match_byte = __builtin_ctzll(diff) / 8;
  100. return len + (uint32_t)match_byte;
  101. }
  102. src1 += 8, len += 8;
  103. } while (len < 256);
  104. return 256;
  105. }
  106. #endif
  107. #endif