array_integer.hpp 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275
  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::find_first;
  30. using Array::get;
  31. using Array::insert;
  32. using Array::move;
  33. using Array::set;
  34. explicit ArrayInteger(Allocator&) noexcept;
  35. ~ArrayInteger() noexcept override {}
  36. static value_type default_value(bool)
  37. {
  38. return 0;
  39. }
  40. void init_from_ref(ref_type ref) noexcept override
  41. {
  42. Array::init_from_ref(ref);
  43. }
  44. void set_parent(ArrayParent* parent, size_t ndx_in_parent) noexcept override
  45. {
  46. Array::set_parent(parent, ndx_in_parent);
  47. }
  48. // Disable copying, this is not allowed.
  49. ArrayInteger& operator=(const ArrayInteger&) = delete;
  50. ArrayInteger(const ArrayInteger&) = delete;
  51. void create()
  52. {
  53. Array::create(type_Normal, false, 0, 0);
  54. }
  55. Mixed get_any(size_t ndx) const override;
  56. bool is_null(size_t) const
  57. {
  58. return false;
  59. }
  60. template <class cond, class Callback>
  61. bool find(value_type value, size_t start, size_t end, QueryStateBase* state, Callback callback) const;
  62. };
  63. class ArrayIntNull : public Array, public ArrayPayload {
  64. public:
  65. using value_type = util::Optional<int64_t>;
  66. explicit ArrayIntNull(Allocator&) noexcept;
  67. ~ArrayIntNull() noexcept override;
  68. static value_type default_value(bool nullable)
  69. {
  70. return nullable ? util::none : util::Optional<int64_t>(0);
  71. }
  72. /// Construct an array of the specified type and size, and return just the
  73. /// reference to the underlying memory. All elements will be initialized to
  74. /// the specified value.
  75. static MemRef create_array(Type, bool context_flag, size_t size, Allocator&);
  76. void create()
  77. {
  78. MemRef r = create_array(type_Normal, false, 0, m_alloc);
  79. init_from_mem(r);
  80. }
  81. void init_from_ref(ref_type) noexcept override;
  82. void set_parent(ArrayParent* parent, size_t ndx_in_parent) noexcept override
  83. {
  84. Array::set_parent(parent, ndx_in_parent);
  85. }
  86. void init_from_mem(MemRef) noexcept;
  87. void init_from_parent() noexcept;
  88. size_t size() const noexcept;
  89. bool is_empty() const noexcept;
  90. void insert(size_t ndx, value_type value);
  91. void add(value_type value);
  92. void set(size_t ndx, value_type value);
  93. value_type get(size_t ndx) const noexcept;
  94. Mixed get_any(size_t ndx) const override;
  95. static value_type get(const char* header, size_t ndx) noexcept;
  96. void get_chunk(size_t ndx, value_type res[8]) const noexcept;
  97. void set_null(size_t ndx);
  98. bool is_null(size_t ndx) const noexcept;
  99. int64_t null_value() const noexcept;
  100. void erase(size_t ndx);
  101. void erase(size_t begin, size_t end);
  102. void move(ArrayIntNull& dst, size_t ndx);
  103. void clear();
  104. void move(size_t begin, size_t end, size_t dest_begin);
  105. bool find(int cond, value_type value, size_t start, size_t end, QueryStateBase* state) const;
  106. template <class cond, class Callback>
  107. bool find(value_type value, size_t start, size_t end, QueryStateBase* state, Callback callback) const;
  108. // Wrappers for backwards compatibility and for simple use without
  109. // setting up state initialization etc
  110. template <class cond>
  111. size_t find_first(value_type value, size_t start = 0, size_t end = npos) const;
  112. void find_all(IntegerColumn* result, value_type value, size_t col_offset = 0, size_t begin = 0,
  113. size_t end = npos) const;
  114. size_t find_first(value_type value, size_t begin = 0, size_t end = npos) const;
  115. protected:
  116. void avoid_null_collision(int64_t value);
  117. private:
  118. int_fast64_t choose_random_null(int64_t incoming) const;
  119. void replace_nulls_with(int64_t new_null);
  120. bool can_use_as_null(int64_t value) const;
  121. template <class Callback>
  122. bool find_impl(int cond, value_type value, size_t start, size_t end, QueryStateBase* state,
  123. Callback callback) const;
  124. template <class cond, class Callback>
  125. bool find_impl(value_type value, size_t start, size_t end, QueryStateBase* state, Callback callback) const;
  126. };
  127. // Implementation:
  128. inline ArrayInteger::ArrayInteger(Allocator& allocator) noexcept
  129. : Array(allocator)
  130. {
  131. m_is_inner_bptree_node = false;
  132. }
  133. inline ArrayIntNull::ArrayIntNull(Allocator& allocator) noexcept
  134. : Array(allocator)
  135. {
  136. }
  137. inline ArrayIntNull::~ArrayIntNull() noexcept {}
  138. inline size_t ArrayIntNull::size() const noexcept
  139. {
  140. return Array::size() - 1;
  141. }
  142. inline bool ArrayIntNull::is_empty() const noexcept
  143. {
  144. return size() == 0;
  145. }
  146. inline void ArrayIntNull::insert(size_t ndx, value_type value)
  147. {
  148. if (value) {
  149. avoid_null_collision(*value);
  150. Array::insert(ndx + 1, *value);
  151. }
  152. else {
  153. Array::insert(ndx + 1, null_value());
  154. }
  155. }
  156. inline void ArrayIntNull::add(value_type value)
  157. {
  158. if (value) {
  159. avoid_null_collision(*value);
  160. Array::add(*value);
  161. }
  162. else {
  163. Array::add(null_value());
  164. }
  165. }
  166. inline void ArrayIntNull::set(size_t ndx, value_type value)
  167. {
  168. if (value) {
  169. avoid_null_collision(*value);
  170. Array::set(ndx + 1, *value);
  171. }
  172. else {
  173. Array::set(ndx + 1, null_value());
  174. }
  175. }
  176. inline void ArrayIntNull::set_null(size_t ndx)
  177. {
  178. Array::set(ndx + 1, null_value());
  179. }
  180. inline ArrayIntNull::value_type ArrayIntNull::get(size_t ndx) const noexcept
  181. {
  182. int64_t value = Array::get(ndx + 1);
  183. if (value == null_value()) {
  184. return util::none;
  185. }
  186. return util::some<int64_t>(value);
  187. }
  188. inline ArrayIntNull::value_type ArrayIntNull::get(const char* header, size_t ndx) noexcept
  189. {
  190. int64_t null_value = Array::get(header, 0);
  191. int64_t value = Array::get(header, ndx + 1);
  192. if (value == null_value) {
  193. return util::none;
  194. }
  195. else {
  196. return util::some<int64_t>(value);
  197. }
  198. }
  199. inline bool ArrayIntNull::is_null(size_t ndx) const noexcept
  200. {
  201. return !get(ndx);
  202. }
  203. inline int64_t ArrayIntNull::null_value() const noexcept
  204. {
  205. return Array::get(0);
  206. }
  207. inline void ArrayIntNull::erase(size_t ndx)
  208. {
  209. Array::erase(ndx + 1);
  210. }
  211. inline void ArrayIntNull::erase(size_t begin, size_t end)
  212. {
  213. Array::erase(begin + 1, end + 1);
  214. }
  215. inline void ArrayIntNull::clear()
  216. {
  217. Array::truncate(0);
  218. Array::add(0);
  219. }
  220. inline void ArrayIntNull::move(size_t begin, size_t end, size_t dest_begin)
  221. {
  222. Array::move(begin + 1, end + 1, dest_begin + 1);
  223. }
  224. } // namespace realm
  225. #endif // REALM_ARRAY_INTEGER_HPP