base2_exponential_histogram_indexer_test.cc 8.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180
  1. // Copyright The OpenTelemetry Authors
  2. // SPDX-License-Identifier: Apache-2.0
  3. #include <gtest/gtest.h>
  4. #include <stdlib.h>
  5. #include <limits>
  6. #include "opentelemetry/sdk/metrics/aggregation/base2_exponential_histogram_indexer.h"
  7. using namespace opentelemetry::sdk::metrics;
  8. TEST(Base2ExponentialHistogramIndexerTest, ScaleOne)
  9. {
  10. const Base2ExponentialHistogramIndexer indexer{1};
  11. auto compute_index = [indexer](double value) { return indexer.ComputeIndex(value); };
  12. EXPECT_EQ(compute_index(std::numeric_limits<double>::max()), 2047);
  13. EXPECT_EQ(compute_index(strtod("0x1p1023", nullptr)), 2045);
  14. EXPECT_EQ(compute_index(strtod("0x1.1p1023", nullptr)), 2046);
  15. EXPECT_EQ(compute_index(strtod("0x1p1022", nullptr)), 2043);
  16. EXPECT_EQ(compute_index(strtod("0x1.1p1022", nullptr)), 2044);
  17. EXPECT_EQ(compute_index(strtod("0x1p-1022", nullptr)), -2045);
  18. EXPECT_EQ(compute_index(strtod("0x1.1p-1022", nullptr)), -2044);
  19. EXPECT_EQ(compute_index(strtod("0x1p-1021", nullptr)), -2043);
  20. EXPECT_EQ(compute_index(strtod("0x1.1p-1021", nullptr)), -2042);
  21. EXPECT_EQ(compute_index(std::numeric_limits<double>::min()), -2045);
  22. EXPECT_EQ(compute_index(std::numeric_limits<double>::denorm_min()), -2149);
  23. EXPECT_EQ(compute_index(15.0), 7);
  24. EXPECT_EQ(compute_index(9.0), 6);
  25. EXPECT_EQ(compute_index(7.0), 5);
  26. EXPECT_EQ(compute_index(5.0), 4);
  27. EXPECT_EQ(compute_index(3.0), 3);
  28. EXPECT_EQ(compute_index(2.5), 2);
  29. EXPECT_EQ(compute_index(1.5), 1);
  30. EXPECT_EQ(compute_index(1.2), 0);
  31. EXPECT_EQ(compute_index(1.0), -1);
  32. EXPECT_EQ(compute_index(0.75), -1);
  33. EXPECT_EQ(compute_index(0.55), -2);
  34. EXPECT_EQ(compute_index(0.45), -3);
  35. }
  36. TEST(Base2ExponentialHistogramIndexerTest, ScaleZero)
  37. {
  38. const Base2ExponentialHistogramIndexer indexer{0};
  39. auto compute_index = [indexer](double value) { return indexer.ComputeIndex(value); };
  40. // Near +Inf.
  41. // Use constant for exp, as max_exponent constant includes bias.
  42. EXPECT_EQ(compute_index(std::numeric_limits<double>::max()), 1023);
  43. EXPECT_EQ(compute_index(strtod("0x1p1023", nullptr)), 1022);
  44. EXPECT_EQ(compute_index(strtod("0x1.1p1023", nullptr)), 1023);
  45. EXPECT_EQ(compute_index(strtod("0x1p1022", nullptr)), 1021);
  46. EXPECT_EQ(compute_index(strtod("0x1.1p1022", nullptr)), 1022);
  47. // Near 0.
  48. EXPECT_EQ(compute_index(strtod("0x1p-1022", nullptr)), -1023);
  49. EXPECT_EQ(compute_index(strtod("0x1.1p-1022", nullptr)), -1022);
  50. EXPECT_EQ(compute_index(strtod("0x1p-1021", nullptr)), -1022);
  51. EXPECT_EQ(compute_index(strtod("0x1.1p-1021", nullptr)), -1021);
  52. EXPECT_EQ(compute_index(std::numeric_limits<double>::min()), -1023);
  53. EXPECT_EQ(compute_index(std::numeric_limits<double>::denorm_min()), -1075);
  54. // Near 1.
  55. EXPECT_EQ(compute_index(4.0), 1);
  56. EXPECT_EQ(compute_index(3.0), 1);
  57. EXPECT_EQ(compute_index(2.0), 0);
  58. EXPECT_EQ(compute_index(1.5), 0);
  59. EXPECT_EQ(compute_index(1.0), -1);
  60. EXPECT_EQ(compute_index(0.75), -1);
  61. EXPECT_EQ(compute_index(0.51), -1);
  62. EXPECT_EQ(compute_index(0.5), -2);
  63. EXPECT_EQ(compute_index(0.26), -2);
  64. EXPECT_EQ(compute_index(0.25), -3);
  65. EXPECT_EQ(compute_index(0.126), -3);
  66. EXPECT_EQ(compute_index(0.125), -4);
  67. }
  68. TEST(Base2ExponentialHistogramIndexerTest, ScaleNegativeOne)
  69. {
  70. const Base2ExponentialHistogramIndexer indexer{-1};
  71. auto compute_index = [indexer](double value) { return indexer.ComputeIndex(value); };
  72. EXPECT_EQ(compute_index(17.0), 2);
  73. EXPECT_EQ(compute_index(16.0), 1);
  74. EXPECT_EQ(compute_index(15.0), 1);
  75. EXPECT_EQ(compute_index(9.0), 1);
  76. EXPECT_EQ(compute_index(8.0), 1);
  77. EXPECT_EQ(compute_index(5.0), 1);
  78. EXPECT_EQ(compute_index(4.0), 0);
  79. EXPECT_EQ(compute_index(3.0), 0);
  80. EXPECT_EQ(compute_index(2.0), 0);
  81. EXPECT_EQ(compute_index(1.5), 0);
  82. EXPECT_EQ(compute_index(1.0), -1);
  83. EXPECT_EQ(compute_index(0.75), -1);
  84. EXPECT_EQ(compute_index(0.5), -1);
  85. EXPECT_EQ(compute_index(0.25), -2);
  86. EXPECT_EQ(compute_index(0.20), -2);
  87. EXPECT_EQ(compute_index(0.13), -2);
  88. EXPECT_EQ(compute_index(0.125), -2);
  89. EXPECT_EQ(compute_index(0.10), -2);
  90. EXPECT_EQ(compute_index(0.0625), -3);
  91. EXPECT_EQ(compute_index(0.06), -3);
  92. }
  93. TEST(Base2ExponentialHistogramIndexerTest, ScaleNegativeFour)
  94. {
  95. const Base2ExponentialHistogramIndexer indexer{-4};
  96. auto compute_index = [indexer](double value) { return indexer.ComputeIndex(value); };
  97. EXPECT_EQ(compute_index(strtod("0x1p0", nullptr)), -1);
  98. EXPECT_EQ(compute_index(strtod("0x10p0", nullptr)), 0);
  99. EXPECT_EQ(compute_index(strtod("0x100p0", nullptr)), 0);
  100. EXPECT_EQ(compute_index(strtod("0x1000p0", nullptr)), 0);
  101. EXPECT_EQ(compute_index(strtod("0x10000p0", nullptr)), 0); // Base == 2**16
  102. EXPECT_EQ(compute_index(strtod("0x100000p0", nullptr)), 1);
  103. EXPECT_EQ(compute_index(strtod("0x1000000p0", nullptr)), 1);
  104. EXPECT_EQ(compute_index(strtod("0x10000000p0", nullptr)), 1);
  105. EXPECT_EQ(compute_index(strtod("0x100000000p0", nullptr)), 1); // == 2**32
  106. EXPECT_EQ(compute_index(strtod("0x1000000000p0", nullptr)), 2);
  107. EXPECT_EQ(compute_index(strtod("0x10000000000p0", nullptr)), 2);
  108. EXPECT_EQ(compute_index(strtod("0x100000000000p0", nullptr)), 2);
  109. EXPECT_EQ(compute_index(strtod("0x1000000000000p0", nullptr)), 2); // 2**48
  110. EXPECT_EQ(compute_index(strtod("0x10000000000000p0", nullptr)), 3);
  111. EXPECT_EQ(compute_index(strtod("0x100000000000000p0", nullptr)), 3);
  112. EXPECT_EQ(compute_index(strtod("0x1000000000000000p0", nullptr)), 3);
  113. EXPECT_EQ(compute_index(strtod("0x10000000000000000p0", nullptr)), 3); // 2**64
  114. EXPECT_EQ(compute_index(strtod("0x100000000000000000p0", nullptr)), 4);
  115. EXPECT_EQ(compute_index(strtod("0x1000000000000000000p0", nullptr)), 4);
  116. EXPECT_EQ(compute_index(strtod("0x10000000000000000000p0", nullptr)), 4);
  117. EXPECT_EQ(compute_index(strtod("0x100000000000000000000p0", nullptr)), 4);
  118. EXPECT_EQ(compute_index(strtod("0x1000000000000000000000p0", nullptr)), 5);
  119. EXPECT_EQ(compute_index(1 / strtod("0x1p0", nullptr)), -1);
  120. EXPECT_EQ(compute_index(1 / strtod("0x10p0", nullptr)), -1);
  121. EXPECT_EQ(compute_index(1 / strtod("0x100p0", nullptr)), -1);
  122. EXPECT_EQ(compute_index(1 / strtod("0x1000p0", nullptr)), -1);
  123. EXPECT_EQ(compute_index(1 / strtod("0x10000p0", nullptr)), -2); // 2**-16
  124. EXPECT_EQ(compute_index(1 / strtod("0x100000p0", nullptr)), -2);
  125. EXPECT_EQ(compute_index(1 / strtod("0x1000000p0", nullptr)), -2);
  126. EXPECT_EQ(compute_index(1 / strtod("0x10000000p0", nullptr)), -2);
  127. EXPECT_EQ(compute_index(1 / strtod("0x100000000p0", nullptr)), -3); // 2**-32
  128. EXPECT_EQ(compute_index(1 / strtod("0x1000000000p0", nullptr)), -3);
  129. EXPECT_EQ(compute_index(1 / strtod("0x10000000000p0", nullptr)), -3);
  130. EXPECT_EQ(compute_index(1 / strtod("0x100000000000p0", nullptr)), -3);
  131. EXPECT_EQ(compute_index(1 / strtod("0x1000000000000p0", nullptr)), -4); // 2**-48
  132. EXPECT_EQ(compute_index(1 / strtod("0x10000000000000p0", nullptr)), -4);
  133. EXPECT_EQ(compute_index(1 / strtod("0x100000000000000p0", nullptr)), -4);
  134. EXPECT_EQ(compute_index(1 / strtod("0x1000000000000000p0", nullptr)), -4);
  135. EXPECT_EQ(compute_index(1 / strtod("0x10000000000000000p0", nullptr)), -5); // 2**-64
  136. EXPECT_EQ(compute_index(1 / strtod("0x100000000000000000p0", nullptr)), -5);
  137. // Max values.
  138. EXPECT_EQ(compute_index(0x1.FFFFFFFFFFFFFp1023), 63);
  139. EXPECT_EQ(compute_index(strtod("0x1p1023", nullptr)), 63);
  140. EXPECT_EQ(compute_index(strtod("0x1p1019", nullptr)), 63);
  141. EXPECT_EQ(compute_index(strtod("0x1p1009", nullptr)), 63);
  142. EXPECT_EQ(compute_index(strtod("0x1p1008", nullptr)), 62);
  143. EXPECT_EQ(compute_index(strtod("0x1p1007", nullptr)), 62);
  144. EXPECT_EQ(compute_index(strtod("0x1p1000", nullptr)), 62);
  145. EXPECT_EQ(compute_index(strtod("0x1p0993", nullptr)), 62);
  146. EXPECT_EQ(compute_index(strtod("0x1p0992", nullptr)), 61);
  147. EXPECT_EQ(compute_index(strtod("0x1p0991", nullptr)), 61);
  148. // Min and subnormal values.
  149. EXPECT_EQ(compute_index(strtod("0x1p-1074", nullptr)), -68);
  150. EXPECT_EQ(compute_index(strtod("0x1p-1073", nullptr)), -68);
  151. EXPECT_EQ(compute_index(strtod("0x1p-1072", nullptr)), -68);
  152. EXPECT_EQ(compute_index(strtod("0x1p-1057", nullptr)), -67);
  153. EXPECT_EQ(compute_index(strtod("0x1p-1056", nullptr)), -67);
  154. EXPECT_EQ(compute_index(strtod("0x1p-1041", nullptr)), -66);
  155. EXPECT_EQ(compute_index(strtod("0x1p-1040", nullptr)), -66);
  156. EXPECT_EQ(compute_index(strtod("0x1p-1025", nullptr)), -65);
  157. EXPECT_EQ(compute_index(strtod("0x1p-1024", nullptr)), -65);
  158. EXPECT_EQ(compute_index(strtod("0x1p-1023", nullptr)), -64);
  159. EXPECT_EQ(compute_index(strtod("0x1p-1022", nullptr)), -64);
  160. EXPECT_EQ(compute_index(strtod("0x1p-1009", nullptr)), -64);
  161. EXPECT_EQ(compute_index(strtod("0x1p-1008", nullptr)), -64);
  162. EXPECT_EQ(compute_index(strtod("0x1p-1007", nullptr)), -63);
  163. EXPECT_EQ(compute_index(strtod("0x1p-0993", nullptr)), -63);
  164. EXPECT_EQ(compute_index(strtod("0x1p-0992", nullptr)), -63);
  165. EXPECT_EQ(compute_index(strtod("0x1p-0991", nullptr)), -62);
  166. EXPECT_EQ(compute_index(strtod("0x1p-0977", nullptr)), -62);
  167. EXPECT_EQ(compute_index(strtod("0x1p-0976", nullptr)), -62);
  168. EXPECT_EQ(compute_index(strtod("0x1p-0975", nullptr)), -61);
  169. }