array_mixed.hpp 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138
  1. /*************************************************************************
  2. *
  3. * Copyright 2018 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_MIXED_HPP
  19. #define REALM_ARRAY_MIXED_HPP
  20. #include <realm/data_type.hpp>
  21. #include <realm/mixed.hpp>
  22. #include <realm/obj.hpp>
  23. #include <realm/array_binary.hpp>
  24. #include <realm/array_string.hpp>
  25. #include <realm/array_timestamp.hpp>
  26. #include <realm/array_key.hpp>
  27. namespace realm {
  28. class ArrayMixed : public ArrayPayload, private Array {
  29. public:
  30. using value_type = Mixed;
  31. using Array::detach;
  32. using Array::is_attached;
  33. using Array::get_ref;
  34. using Array::set_parent;
  35. using Array::update_parent;
  36. using Array::get_parent;
  37. explicit ArrayMixed(Allocator&);
  38. static Mixed default_value(bool)
  39. {
  40. return Mixed{};
  41. }
  42. void create();
  43. void destroy()
  44. {
  45. Array::destroy_deep();
  46. }
  47. void init_from_mem(MemRef mem) noexcept;
  48. void init_from_ref(ref_type ref) noexcept override
  49. {
  50. init_from_mem(MemRef(m_alloc.translate(ref), ref, m_alloc));
  51. }
  52. void set_parent(ArrayParent* parent, size_t ndx_in_parent) noexcept override
  53. {
  54. Array::set_parent(parent, ndx_in_parent);
  55. }
  56. void init_from_parent()
  57. {
  58. ref_type ref = get_ref_from_parent();
  59. ArrayMixed::init_from_ref(ref);
  60. }
  61. size_t size() const
  62. {
  63. return m_composite.size();
  64. }
  65. void add(Mixed value);
  66. void set(size_t ndx, Mixed value);
  67. void set_null(size_t ndx);
  68. void insert(size_t ndx, Mixed value);
  69. Mixed get(size_t ndx) const;
  70. Mixed get_any(size_t ndx) const override
  71. {
  72. return get(ndx);
  73. }
  74. bool is_null(size_t ndx) const
  75. {
  76. return m_composite.get(ndx) == 0;
  77. }
  78. void clear();
  79. void erase(size_t ndx);
  80. void truncate_and_destroy_children(size_t ndx);
  81. void move(ArrayMixed& dst, size_t ndx);
  82. size_t find_first(Mixed value, size_t begin = 0, size_t end = realm::npos) const noexcept;
  83. void verify() const;
  84. private:
  85. enum { payload_idx_type, payload_idx_int, payload_idx_pair, payload_idx_str, payload_idx_size };
  86. static constexpr int64_t s_data_type_mask = 0b0001'1111;
  87. static constexpr int64_t s_payload_idx_mask = 0b1110'0000;
  88. static constexpr int64_t s_payload_idx_shift = 5;
  89. static constexpr int64_t s_data_shift = 8;
  90. // This primary array contains an aggregation of the actual value - which can be
  91. // either the value itself or an index into one of the payload arrays - the index
  92. // of the payload array and the data_type.
  93. //
  94. // value << s_data_shift | payload_idx << s_payload_idx_shift | data_type
  95. //
  96. // payload_idx one of PayloadIdx
  97. Array m_composite;
  98. // Used to store big ints, floats and doubles
  99. mutable Array m_ints;
  100. // Used to store timestamps
  101. mutable Array m_int_pairs;
  102. // Used to store String and Binary
  103. mutable ArrayString m_strings;
  104. DataType get_type(size_t ndx) const
  105. {
  106. return DataType((m_composite.get(ndx) & s_data_type_mask) - 1);
  107. }
  108. int64_t store(const Mixed&);
  109. void ensure_array_accessor(Array& arr, size_t ndx_in_parent) const;
  110. void ensure_int_array() const;
  111. void ensure_int_pair_array() const;
  112. void ensure_string_array() const;
  113. void replace_index(size_t old_ndx, size_t new_ndx, size_t payload_index);
  114. void erase_linked_payload(size_t ndx);
  115. };
  116. } // namespace realm
  117. #endif /* REALM_ARRAY_MIXED_HPP */