array_integer.hpp 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383
  1. /*************************************************************************
  2. *
  3. * Copyright 2016 Realm Inc.
  4. *
  5. * Licensed under the Apache License, Version 2.0 (the "License");
  6. * you may not use this file except in compliance with the License.
  7. * You may obtain a copy of the License at
  8. *
  9. * http://www.apache.org/licenses/LICENSE-2.0
  10. *
  11. * Unless required by applicable law or agreed to in writing, software
  12. * distributed under the License is distributed on an "AS IS" BASIS,
  13. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  14. * See the License for the specific language governing permissions and
  15. * limitations under the License.
  16. *
  17. **************************************************************************/
  18. #ifndef REALM_ARRAY_INTEGER_HPP
  19. #define REALM_ARRAY_INTEGER_HPP
  20. #include <realm/array.hpp>
  21. #include <realm/util/safe_int_ops.hpp>
  22. #include <realm/util/optional.hpp>
  23. #include <realm/array_key.hpp>
  24. namespace realm {
  25. class ArrayInteger : public Array, public ArrayPayload {
  26. public:
  27. using value_type = int64_t;
  28. using Array::add;
  29. using Array::get;
  30. using Array::insert;
  31. using Array::move;
  32. using Array::set;
  33. explicit ArrayInteger(Allocator&) noexcept;
  34. ~ArrayInteger() noexcept override {}
  35. static value_type default_value(bool)
  36. {
  37. return 0;
  38. }
  39. void init_from_ref(ref_type ref) noexcept override
  40. {
  41. Array::init_from_ref(ref);
  42. }
  43. void set_parent(ArrayParent* parent, size_t ndx_in_parent) noexcept override
  44. {
  45. Array::set_parent(parent, ndx_in_parent);
  46. }
  47. // Disable copying, this is not allowed.
  48. ArrayInteger& operator=(const ArrayInteger&) = delete;
  49. ArrayInteger(const ArrayInteger&) = delete;
  50. void create()
  51. {
  52. Array::create(type_Normal, false, 0, 0);
  53. }
  54. Mixed get_any(size_t ndx) const override;
  55. bool is_null(size_t) const
  56. {
  57. return false;
  58. }
  59. };
  60. class ArrayIntNull : public Array, public ArrayPayload {
  61. public:
  62. using value_type = util::Optional<int64_t>;
  63. explicit ArrayIntNull(Allocator&) noexcept;
  64. ~ArrayIntNull() noexcept override;
  65. static value_type default_value(bool nullable)
  66. {
  67. return nullable ? util::none : util::Optional<int64_t>(0);
  68. }
  69. /// Construct an array of the specified type and size, and return just the
  70. /// reference to the underlying memory. All elements will be initialized to
  71. /// the specified value.
  72. static MemRef create_array(Type, bool context_flag, size_t size, Allocator&);
  73. void create()
  74. {
  75. MemRef r = create_array(type_Normal, false, 0, m_alloc);
  76. init_from_mem(r);
  77. }
  78. void init_from_ref(ref_type) noexcept override;
  79. void set_parent(ArrayParent* parent, size_t ndx_in_parent) noexcept override
  80. {
  81. Array::set_parent(parent, ndx_in_parent);
  82. }
  83. void init_from_mem(MemRef) noexcept;
  84. void init_from_parent() noexcept;
  85. size_t size() const noexcept;
  86. bool is_empty() const noexcept;
  87. void insert(size_t ndx, value_type value);
  88. void add(value_type value);
  89. void set(size_t ndx, value_type value);
  90. value_type get(size_t ndx) const noexcept;
  91. Mixed get_any(size_t ndx) const override;
  92. static value_type get(const char* header, size_t ndx) noexcept;
  93. void get_chunk(size_t ndx, value_type res[8]) const noexcept;
  94. void set_null(size_t ndx);
  95. bool is_null(size_t ndx) const noexcept;
  96. int64_t null_value() const noexcept;
  97. void erase(size_t ndx);
  98. void erase(size_t begin, size_t end);
  99. void move(ArrayIntNull& dst, size_t ndx);
  100. void clear();
  101. void move(size_t begin, size_t end, size_t dest_begin);
  102. bool find(int cond, value_type value, size_t start, size_t end, size_t baseindex, QueryStateBase* state) const;
  103. // This is the one installed into the m_finder slots.
  104. template <class cond, size_t bitwidth>
  105. bool find(int64_t value, size_t start, size_t end, size_t baseindex, QueryStateBase* state) const;
  106. template <class cond, class Callback>
  107. bool find(value_type value, size_t start, size_t end, size_t baseindex, QueryStateBase* state,
  108. Callback callback) const;
  109. // Wrappers for backwards compatibility and for simple use without
  110. // setting up state initialization etc
  111. template <class cond>
  112. size_t find_first(value_type value, size_t start = 0, size_t end = npos) const;
  113. void find_all(IntegerColumn* result, value_type value, size_t col_offset = 0, size_t begin = 0,
  114. size_t end = npos) const;
  115. size_t find_first(value_type value, size_t begin = 0, size_t end = npos) const;
  116. protected:
  117. void avoid_null_collision(int64_t value);
  118. private:
  119. int_fast64_t choose_random_null(int64_t incoming) const;
  120. void replace_nulls_with(int64_t new_null);
  121. bool can_use_as_null(int64_t value) const;
  122. template <class Callback>
  123. bool find_impl(int cond, value_type value, size_t start, size_t end, size_t baseindex, QueryStateBase* state,
  124. Callback callback) const;
  125. template <class cond, class Callback>
  126. bool find_impl(value_type value, size_t start, size_t end, size_t baseindex, QueryStateBase* state,
  127. Callback callback) const;
  128. };
  129. // Implementation:
  130. inline ArrayInteger::ArrayInteger(Allocator& allocator) noexcept
  131. : Array(allocator)
  132. {
  133. m_is_inner_bptree_node = false;
  134. }
  135. inline ArrayIntNull::ArrayIntNull(Allocator& allocator) noexcept
  136. : Array(allocator)
  137. {
  138. }
  139. inline ArrayIntNull::~ArrayIntNull() noexcept {}
  140. inline size_t ArrayIntNull::size() const noexcept
  141. {
  142. return Array::size() - 1;
  143. }
  144. inline bool ArrayIntNull::is_empty() const noexcept
  145. {
  146. return size() == 0;
  147. }
  148. inline void ArrayIntNull::insert(size_t ndx, value_type value)
  149. {
  150. if (value) {
  151. avoid_null_collision(*value);
  152. Array::insert(ndx + 1, *value);
  153. }
  154. else {
  155. Array::insert(ndx + 1, null_value());
  156. }
  157. }
  158. inline void ArrayIntNull::add(value_type value)
  159. {
  160. if (value) {
  161. avoid_null_collision(*value);
  162. Array::add(*value);
  163. }
  164. else {
  165. Array::add(null_value());
  166. }
  167. }
  168. inline void ArrayIntNull::set(size_t ndx, value_type value)
  169. {
  170. if (value) {
  171. avoid_null_collision(*value);
  172. Array::set(ndx + 1, *value);
  173. }
  174. else {
  175. Array::set(ndx + 1, null_value());
  176. }
  177. }
  178. inline void ArrayIntNull::set_null(size_t ndx)
  179. {
  180. Array::set(ndx + 1, null_value());
  181. }
  182. inline ArrayIntNull::value_type ArrayIntNull::get(size_t ndx) const noexcept
  183. {
  184. int64_t value = Array::get(ndx + 1);
  185. if (value == null_value()) {
  186. return util::none;
  187. }
  188. return util::some<int64_t>(value);
  189. }
  190. inline ArrayIntNull::value_type ArrayIntNull::get(const char* header, size_t ndx) noexcept
  191. {
  192. int64_t null_value = Array::get(header, 0);
  193. int64_t value = Array::get(header, ndx + 1);
  194. if (value == null_value) {
  195. return util::none;
  196. }
  197. else {
  198. return util::some<int64_t>(value);
  199. }
  200. }
  201. inline bool ArrayIntNull::is_null(size_t ndx) const noexcept
  202. {
  203. return !get(ndx);
  204. }
  205. inline int64_t ArrayIntNull::null_value() const noexcept
  206. {
  207. return Array::get(0);
  208. }
  209. inline void ArrayIntNull::erase(size_t ndx)
  210. {
  211. Array::erase(ndx + 1);
  212. }
  213. inline void ArrayIntNull::erase(size_t begin, size_t end)
  214. {
  215. Array::erase(begin + 1, end + 1);
  216. }
  217. inline void ArrayIntNull::clear()
  218. {
  219. Array::truncate(0);
  220. Array::add(0);
  221. }
  222. inline void ArrayIntNull::move(size_t begin, size_t end, size_t dest_begin)
  223. {
  224. Array::move(begin + 1, end + 1, dest_begin + 1);
  225. }
  226. inline bool ArrayIntNull::find(int cond, value_type value, size_t start, size_t end, size_t baseindex,
  227. QueryStateBase* state) const
  228. {
  229. return find_impl(cond, value, start, end, baseindex, state, nullptr);
  230. }
  231. template <class cond, class Callback>
  232. inline bool ArrayIntNull::find(value_type value, size_t start, size_t end, size_t baseindex, QueryStateBase* state,
  233. Callback callback) const
  234. {
  235. return find_impl<cond, Callback>(value, start, end, baseindex, state, callback);
  236. }
  237. template <class Callback>
  238. inline bool ArrayIntNull::find_impl(int cond, value_type value, size_t start, size_t end, size_t baseindex,
  239. QueryStateBase* state, Callback callback) const
  240. {
  241. switch (cond) {
  242. case cond_Equal:
  243. return find_impl<Equal>(value, start, end, baseindex, state, callback);
  244. case cond_NotEqual:
  245. return find_impl<NotEqual>(value, start, end, baseindex, state, callback);
  246. case cond_Greater:
  247. return find_impl<Greater>(value, start, end, baseindex, state, callback);
  248. case cond_Less:
  249. return find_impl<Less>(value, start, end, baseindex, state, callback);
  250. case cond_None:
  251. return find_impl<None>(value, start, end, baseindex, state, callback);
  252. case cond_LeftNotNull:
  253. return find_impl<NotNull>(value, start, end, baseindex, state, callback);
  254. }
  255. REALM_ASSERT_DEBUG(false);
  256. return false;
  257. }
  258. template <class cond, class Callback>
  259. bool ArrayIntNull::find_impl(value_type opt_value, size_t start, size_t end, size_t baseindex, QueryStateBase* state,
  260. Callback callback) const
  261. {
  262. int64_t null_value = Array::get(0);
  263. bool find_null = !bool(opt_value);
  264. int64_t value;
  265. size_t end2 = (end == npos ? size() : end) + 1;
  266. size_t start2 = start + 1;
  267. size_t baseindex2 = baseindex - 1;
  268. if constexpr (std::is_same_v<cond, Equal>) {
  269. if (find_null) {
  270. value = null_value;
  271. }
  272. else {
  273. if (*opt_value == null_value) {
  274. // If the value to search for is equal to the null value, the value cannot be in the array
  275. return true;
  276. }
  277. else {
  278. value = *opt_value;
  279. }
  280. }
  281. // Fall back to plain Array find.
  282. return Array::find<cond>(value, start2, end2, baseindex2, state, callback);
  283. }
  284. else {
  285. cond c;
  286. if (opt_value) {
  287. value = *opt_value;
  288. }
  289. else {
  290. value = null_value;
  291. }
  292. for (size_t i = start2; i < end2; ++i) {
  293. int64_t v = Array::get(i);
  294. bool value_is_null = (v == null_value);
  295. if (c(v, value, value_is_null, find_null)) {
  296. util::Optional<int64_t> v2 = value_is_null ? util::none : util::make_optional(v);
  297. if (!Array::find_action(i + baseindex2, v2, state, callback)) {
  298. return false; // tell caller to stop aggregating/search
  299. }
  300. }
  301. }
  302. return true; // tell caller to continue aggregating/search (on next array leafs)
  303. }
  304. }
  305. template <class cond>
  306. size_t ArrayIntNull::find_first(value_type value, size_t start, size_t end) const
  307. {
  308. QueryStateFindFirst state;
  309. find_impl<cond>(value, start, end, 0, &state, nullptr);
  310. if (state.match_count() > 0)
  311. return to_size_t(state.m_state);
  312. else
  313. return not_found;
  314. }
  315. inline size_t ArrayIntNull::find_first(value_type value, size_t begin, size_t end) const
  316. {
  317. return find_first<Equal>(value, begin, end);
  318. }
  319. } // namespace realm
  320. #endif // REALM_ARRAY_INTEGER_HPP