You can not select more than 25 topics Topics must start with a chinese character,a letter or number, can include dashes ('-') and can be up to 35 characters long.

rng.h 7.8 kB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272
  1. /**
  2. * \file dnn/test/common/rng.h
  3. * MegEngine is Licensed under the Apache License, Version 2.0 (the "License")
  4. *
  5. * Copyright (c) 2014-2021 Megvii Inc. All rights reserved.
  6. *
  7. * Unless required by applicable law or agreed to in writing,
  8. * software distributed under the License is distributed on an
  9. * "AS IS" BASIS, WITHOUT ARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  10. */
  11. #pragma once
  12. #include "megdnn/dtype.h"
  13. #include <random>
  14. #include <set>
  15. #include "test/common/random_state.h"
  16. #include "test/common/utils.h"
  17. namespace megdnn {
  18. namespace test {
  19. #if __cplusplus >= 201703L
  20. #define COMPAT_RANDOM(begin, end) \
  21. { \
  22. std::default_random_engine rng_engine; \
  23. std::shuffle(begin, end, rng_engine); \
  24. }
  25. #else
  26. #define COMPAT_RANDOM(begin, end) std::random_shuffle(begin, end);
  27. #endif
  28. class RNG {
  29. protected:
  30. class RNGxorshf;
  31. public:
  32. virtual void gen(const TensorND& tensor) = 0;
  33. virtual ~RNG() = default;
  34. };
  35. class Float16PeriodicalRNG : public RNG {
  36. public:
  37. Float16PeriodicalRNG();
  38. Float16PeriodicalRNG(size_t range);
  39. void gen(const TensorND& tensor) override;
  40. dt_float16 get_single_val();
  41. private:
  42. void gen_all_valid_float16();
  43. size_t m_offset;
  44. std::vector<dt_float16> m_sequence;
  45. };
  46. class BFloat16PeriodicalRNG : public RNG {
  47. public:
  48. BFloat16PeriodicalRNG() {
  49. size_t bits = sizeof(dt_bfloat16) * 8;
  50. size_t mantissa_bits = std::numeric_limits<dt_bfloat16>::digits - 1;
  51. size_t exponent_bits = bits - mantissa_bits - 1;
  52. for (size_t exp = 1u << (exponent_bits - 2);
  53. exp < (1u << exponent_bits) - (1u << (exponent_bits - 2)); ++exp) {
  54. for (size_t x = 0; x < 1u << mantissa_bits; ++x) {
  55. size_t pos_num = (exp << mantissa_bits) + x;
  56. size_t neg_num = (1u << (bits - 1)) + (exp << mantissa_bits) + x;
  57. union U {
  58. U() {}
  59. uint16_t i;
  60. dt_bfloat16 f;
  61. } i2f;
  62. i2f.i = static_cast<uint16_t>(pos_num);
  63. m_sequence.push_back(i2f.f);
  64. i2f.i = static_cast<uint16_t>(neg_num);
  65. m_sequence.push_back(i2f.f);
  66. }
  67. }
  68. std::shuffle(m_sequence.begin(), m_sequence.end(), RandomState::generator());
  69. }
  70. void gen(const TensorND& tensor) override {
  71. megdnn_assert(tensor.layout.dtype.enumv() == DTypeTrait<dt_bfloat16>::enumv);
  72. size_t nr_elems = tensor.layout.span().dist_elem();
  73. auto offset = tensor.layout.span().low_elem;
  74. for (size_t i = 0; i < nr_elems; ++i) {
  75. tensor.ptr<dt_bfloat16>()[offset + i] = get_single_val();
  76. }
  77. }
  78. dt_bfloat16 get_single_val() {
  79. if (m_offset >= m_sequence.size()) {
  80. m_offset = 0;
  81. }
  82. return m_sequence[m_offset++];
  83. }
  84. private:
  85. size_t m_offset = 0;
  86. std::vector<dt_bfloat16> m_sequence;
  87. };
  88. class IIDRNG : public RNG {
  89. public:
  90. void gen(const TensorND& tensor) override;
  91. virtual dt_float32 gen_single_val() = 0;
  92. virtual bool output_is_float() { return true; }
  93. protected:
  94. virtual bool has_fast_float32();
  95. virtual void fill_fast_float32(dt_float32* dest, size_t size);
  96. };
  97. class NormalRNG final : public IIDRNG {
  98. public:
  99. NormalRNG(dt_float32 mean = 0.0f, dt_float32 stddev = 1.0f)
  100. : m_dist(mean, stddev) {}
  101. void fill_fast_float32(dt_float32* dest, size_t size) override;
  102. protected:
  103. dt_float32 gen_single_val() override;
  104. private:
  105. std::normal_distribution<dt_float32> m_dist;
  106. bool has_fast_float32() override;
  107. };
  108. class ConstValue final : public IIDRNG {
  109. public:
  110. ConstValue(dt_float32 value = 0.0f) : value_(value) {}
  111. void fill_fast_float32(dt_float32* dest, size_t size) override;
  112. protected:
  113. dt_float32 gen_single_val() override { return value_; }
  114. private:
  115. dt_float32 value_;
  116. bool has_fast_float32() override { return true; }
  117. };
  118. class UniformIntRNG : public IIDRNG {
  119. public:
  120. UniformIntRNG(dt_int32 a, dt_int32 b) : m_dist(a, b) {}
  121. dt_float32 gen_single_val() override;
  122. bool output_is_float() override { return false; }
  123. protected:
  124. std::uniform_int_distribution<dt_int32> m_dist;
  125. };
  126. //! range must be positive; each value would be negated with prob 0.5
  127. class UniformIntNonZeroRNG : public UniformIntRNG {
  128. std::uniform_int_distribution<dt_int32> m_dist_flip{0, 1};
  129. public:
  130. UniformIntNonZeroRNG(int a, int b) : UniformIntRNG(a, b) {
  131. megdnn_assert(a > 0 && b > a);
  132. }
  133. dt_float32 gen_single_val() override;
  134. };
  135. class UniformFloatRNG : public IIDRNG {
  136. public:
  137. UniformFloatRNG(dt_float32 a, dt_float32 b) : m_dist(a, b) {}
  138. dt_float32 gen_single_val() override;
  139. protected:
  140. std::uniform_real_distribution<dt_float32> m_dist;
  141. bool has_fast_float32() override;
  142. void fill_fast_float32(dt_float32* dest, size_t size) override;
  143. };
  144. //! range must be positive; each value would be negated with prob 0.5
  145. class UniformFloatNonZeroRNG : public UniformFloatRNG {
  146. std::uniform_int_distribution<dt_int32> m_dist_flip{0, 1};
  147. public:
  148. UniformFloatNonZeroRNG(float a, float b) : UniformFloatRNG(a, b) {
  149. megdnn_assert(a > 0 && b > a);
  150. }
  151. dt_float32 gen_single_val() override;
  152. void fill_fast_float32(dt_float32* dest, size_t size) override;
  153. };
  154. class UniformFloatWithValueRNG : public UniformFloatRNG {
  155. public:
  156. UniformFloatWithValueRNG(
  157. dt_float32 a, dt_float32 b, float val_proportion, float val)
  158. : UniformFloatRNG(a, b), val_(val) {
  159. if (val_proportion < 0.f)
  160. val_proportion_ = 0.f;
  161. else if (val_proportion > 1.f)
  162. val_proportion_ = 1.f;
  163. else
  164. val_proportion_ = val_proportion;
  165. }
  166. private:
  167. float val_proportion_, val_;
  168. void fill_fast_float32(dt_float32* dest, size_t size) override;
  169. };
  170. class UniformFloatWithZeroRNG final : public UniformFloatWithValueRNG {
  171. public:
  172. UniformFloatWithZeroRNG(dt_float32 a, dt_float32 b, float zero_val_proportion)
  173. : UniformFloatWithValueRNG(a, b, zero_val_proportion, 0.f) {}
  174. };
  175. class BernoulliRNG final : public IIDRNG {
  176. public:
  177. BernoulliRNG(dt_float32 probability_);
  178. dt_float32 gen_single_val() override;
  179. private:
  180. dt_float32 m_probability;
  181. std::uniform_real_distribution<dt_float32> m_dist;
  182. };
  183. /**
  184. * \brief RNG without replacement, so that no two values in the tensor are
  185. * equal.
  186. *
  187. * Each value is generated repeatedly by IIDRNG, until the newly-generated value
  188. * differs from any previous value.
  189. */
  190. class NoReplacementRNG final : public RNG {
  191. private:
  192. IIDRNG* m_iid_rng;
  193. public:
  194. NoReplacementRNG(IIDRNG* iid_rng) : m_iid_rng(iid_rng) {}
  195. void gen(const TensorND& tensor) override;
  196. };
  197. //! generate a batch of matrices that are likely to have a small condition num
  198. class InvertibleMatrixRNG final : public RNG {
  199. std::unique_ptr<RNGxorshf> m_rng;
  200. public:
  201. InvertibleMatrixRNG();
  202. ~InvertibleMatrixRNG() noexcept;
  203. void gen(const TensorND& tensor) override;
  204. private:
  205. template <typename ctype>
  206. void do_gen(ctype* ptr, size_t batch, size_t n);
  207. };
  208. //! generate a continuous number of delta, start from value
  209. class ConsecutiveRNG final : public IIDRNG {
  210. public:
  211. ConsecutiveRNG(dt_float32 value = 0.0f, dt_float32 delta = 1.0f)
  212. : value_(value), delta_(delta) {}
  213. void fill_fast_float32(dt_float32* dest, size_t size) override;
  214. protected:
  215. dt_float32 gen_single_val() override {
  216. auto res = value_;
  217. value_ += delta_;
  218. return res;
  219. }
  220. private:
  221. dt_float32 value_, delta_;
  222. bool has_fast_float32() override { return true; }
  223. };
  224. } // namespace test
  225. } // namespace megdnn
  226. // vim: syntax=cpp.doxygen