cluster_tree.hpp 6.7 KB

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