cluster_tree.hpp 8.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287
  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_CLUSTER_TREE_HPP
  19. #define REALM_CLUSTER_TREE_HPP
  20. #include <realm/cluster.hpp>
  21. #include <realm/obj.hpp>
  22. #include <realm/util/function_ref.hpp>
  23. namespace realm {
  24. class Cluster;
  25. class ClusterTree {
  26. public:
  27. class Iterator;
  28. using TraverseFunction = util::FunctionRef<IteratorControl(const Cluster*)>;
  29. using UpdateFunction = util::FunctionRef<void(Cluster*)>;
  30. using ColIterateFunction = util::FunctionRef<IteratorControl(ColKey)>;
  31. ClusterTree(Table* owner, Allocator& alloc, size_t top_position_for_cluster_tree);
  32. virtual ~ClusterTree();
  33. ClusterTree(ClusterTree&&) = default;
  34. // Disable copying, this is not allowed.
  35. ClusterTree& operator=(const ClusterTree&) = delete;
  36. ClusterTree(const ClusterTree&) = delete;
  37. bool is_attached() const
  38. {
  39. return m_root && m_root->is_attached();
  40. }
  41. Allocator& get_alloc() const
  42. {
  43. return m_alloc;
  44. }
  45. /// Initialize the accessor from its slot in the `ArrayParent`. If the ref
  46. /// in the parent slot is zero, this returns false and leaves the
  47. /// `ClusterTree` in an unusable state.
  48. bool init_from_parent();
  49. void update_from_parent() noexcept;
  50. size_t size() const noexcept
  51. {
  52. return m_size;
  53. }
  54. size_t nb_columns() const
  55. {
  56. return m_root->nb_columns();
  57. }
  58. static size_t size_from_ref(ref_type, Allocator& alloc);
  59. void destroy()
  60. {
  61. m_root->destroy_deep();
  62. }
  63. void nullify_links(ObjKey, CascadeState&);
  64. bool is_empty() const noexcept
  65. {
  66. return size() == 0;
  67. }
  68. int64_t get_last_key_value() const
  69. {
  70. return m_root->get_last_key_value();
  71. }
  72. MemRef ensure_writeable(ObjKey k)
  73. {
  74. return m_root->ensure_writeable(k);
  75. }
  76. void update_ref_in_parent(ObjKey k, ref_type ref)
  77. {
  78. m_root->update_ref_in_parent(k, ref);
  79. }
  80. Array& get_fields_accessor(Array& fallback, MemRef mem) const
  81. {
  82. if (m_root->is_leaf()) {
  83. return *m_root;
  84. }
  85. fallback.init_from_mem(mem);
  86. return fallback;
  87. }
  88. uint64_t bump_content_version()
  89. {
  90. return m_alloc.bump_content_version();
  91. }
  92. void bump_storage_version()
  93. {
  94. m_alloc.bump_storage_version();
  95. }
  96. uint64_t get_content_version() const
  97. {
  98. return m_alloc.get_content_version();
  99. }
  100. uint64_t get_instance_version() const
  101. {
  102. return m_alloc.get_instance_version();
  103. }
  104. uint64_t get_storage_version(uint64_t inst_ver) const
  105. {
  106. return m_alloc.get_storage_version(inst_ver);
  107. }
  108. void insert_column(ColKey col)
  109. {
  110. m_root->insert_column(col);
  111. }
  112. void remove_column(ColKey col)
  113. {
  114. m_root->remove_column(col);
  115. }
  116. // Create and return object
  117. Obj insert(ObjKey k, const FieldValues& values);
  118. // Lookup and return object
  119. Obj get(ObjKey k) const
  120. {
  121. ClusterNode::State state;
  122. m_root->get(k, state);
  123. return Obj(get_table_ref(), state.mem, k, state.index);
  124. }
  125. Obj try_get_obj(ObjKey k) const noexcept
  126. {
  127. if (auto state = ClusterTree::try_get(k)) {
  128. return Obj(get_table_ref(), state.mem, k, state.index);
  129. }
  130. return {};
  131. }
  132. Obj get(size_t ndx) const
  133. {
  134. ObjKey k;
  135. auto state = ClusterTree::get(ndx, k);
  136. return Obj(get_table_ref(), state.mem, k, state.index);
  137. }
  138. void clear(CascadeState&);
  139. void enumerate_string_column(ColKey col_key);
  140. const Table* get_owning_table() const noexcept
  141. {
  142. return m_owner;
  143. }
  144. // Insert entry for object, but do not create and return the object accessor
  145. void insert_fast(ObjKey k, const FieldValues& init_values, ClusterNode::State& state);
  146. // Delete object with given key
  147. void erase(ObjKey k, CascadeState& state);
  148. // Check if an object with given key exists
  149. bool is_valid(ObjKey k) const noexcept;
  150. // Lookup and return object
  151. ClusterNode::State try_get(ObjKey k) const noexcept;
  152. // Lookup by index
  153. ClusterNode::State get(size_t ndx, ObjKey& k) const;
  154. // Get logical index of object identified by k
  155. size_t get_ndx(ObjKey k) const noexcept;
  156. // Find the leaf containing the requested object
  157. bool get_leaf(ObjKey key, ClusterNode::IteratorState& state) const noexcept;
  158. // Visit all leaves and call the supplied function. Stop when function returns IteratorControl::Stop.
  159. // Not allowed to modify the tree
  160. bool traverse(TraverseFunction func) const;
  161. // Visit all leaves and call the supplied function. The function can modify the leaf.
  162. void update(UpdateFunction func);
  163. void set_spec(ArrayPayload& arr, ColKey::Idx col_ndx) const;
  164. virtual std::unique_ptr<ClusterNode> get_root_from_parent();
  165. void dump_objects()
  166. {
  167. m_root->dump_objects(0, "");
  168. }
  169. void verify() const;
  170. protected:
  171. friend class Obj;
  172. friend class Cluster;
  173. friend class ClusterNodeInner;
  174. Allocator& m_alloc;
  175. Table* m_owner;
  176. size_t m_top_position_for_cluster_tree;
  177. std::unique_ptr<ClusterNode> m_root;
  178. size_t m_size = 0;
  179. void replace_root(std::unique_ptr<ClusterNode> leaf);
  180. std::unique_ptr<ClusterNode> create_root_from_parent(ArrayParent* parent, size_t ndx_in_parent);
  181. std::unique_ptr<ClusterNode> get_node(ArrayParent* parent, size_t ndx_in_parent) const;
  182. TableRef get_table_ref() const;
  183. bool is_string_enum_type(ColKey::Idx col_ndx) const;
  184. void remove_all_links(CascadeState&);
  185. };
  186. class ClusterTree::Iterator {
  187. public:
  188. typedef std::forward_iterator_tag iterator_category;
  189. typedef std::ptrdiff_t difference_type;
  190. typedef Obj value_type;
  191. typedef Obj* pointer;
  192. typedef Obj& reference;
  193. Iterator(const ClusterTree& t, size_t ndx);
  194. Iterator(const Iterator& other);
  195. Iterator& operator=(const Iterator& other)
  196. {
  197. REALM_ASSERT(&m_tree == &other.m_tree);
  198. m_position = other.m_position;
  199. m_key = other.m_key;
  200. m_leaf_invalid = true;
  201. return *this;
  202. }
  203. // Set the iterator to the given absolute position in the table.
  204. void go(size_t abs_pos);
  205. bool update() const;
  206. // Advance the iterator to the next object in the table. This also holds if the object
  207. // pointed to is deleted. That is - you will get the same result of advancing no matter
  208. // if the previous object is deleted or not.
  209. Iterator& operator++();
  210. Iterator& operator+=(ptrdiff_t adj);
  211. Iterator operator+(ptrdiff_t adj)
  212. {
  213. return Iterator(m_tree, get_position() + adj);
  214. }
  215. bool operator==(const Iterator& rhs) const
  216. {
  217. return m_key == rhs.m_key;
  218. }
  219. bool operator!=(const Iterator& rhs) const
  220. {
  221. return m_key != rhs.m_key;
  222. }
  223. // If the object pointed to by the iterator is deleted, you will get an exception if
  224. // you try to dereference the iterator before advancing it.
  225. reference operator*() const
  226. {
  227. return *operator->();
  228. }
  229. pointer operator->() const;
  230. protected:
  231. const ClusterTree& m_tree;
  232. mutable uint64_t m_storage_version = uint64_t(-1);
  233. mutable Cluster m_leaf;
  234. mutable ClusterNode::IteratorState m_state;
  235. mutable uint64_t m_instance_version = uint64_t(-1);
  236. ObjKey m_key;
  237. mutable bool m_leaf_invalid;
  238. mutable size_t m_position;
  239. mutable size_t m_leaf_start_pos = size_t(-1);
  240. mutable Obj m_obj;
  241. ObjKey load_leaf(ObjKey key) const;
  242. size_t get_position();
  243. };
  244. } // namespace realm
  245. #endif /* REALM_CLUSTER_TREE_HPP */