array_string.hpp 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166
  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_STRING_HPP
  19. #define REALM_ARRAY_STRING_HPP
  20. #include <realm/array_string_short.hpp>
  21. #include <realm/array_blobs_small.hpp>
  22. #include <realm/array_blobs_big.hpp>
  23. namespace realm {
  24. class Spec;
  25. class ArrayString : public ArrayPayload {
  26. public:
  27. using value_type = StringData;
  28. explicit ArrayString(Allocator&);
  29. static StringData default_value(bool nullable)
  30. {
  31. return nullable ? StringData{} : StringData{""};
  32. }
  33. // This is only used in the upgrade process
  34. void set_nullability(bool n)
  35. {
  36. m_nullable = n;
  37. }
  38. void create();
  39. bool is_attached() const
  40. {
  41. return m_arr->is_attached();
  42. }
  43. ref_type get_ref() const
  44. {
  45. return m_arr->get_ref();
  46. }
  47. ArrayParent* get_parent() const
  48. {
  49. return m_arr->get_parent();
  50. }
  51. size_t get_ndx_in_parent() const
  52. {
  53. return m_arr->get_ndx_in_parent();
  54. }
  55. void set_parent(ArrayParent* p, size_t n) noexcept override
  56. {
  57. m_arr->set_parent(p, n);
  58. }
  59. bool need_spec() const override
  60. {
  61. return true;
  62. }
  63. void set_spec(Spec* spec, size_t col_ndx) const override
  64. {
  65. m_spec = spec;
  66. m_col_ndx = col_ndx;
  67. }
  68. void update_parent()
  69. {
  70. m_arr->update_parent();
  71. }
  72. void init_from_mem(MemRef mem) noexcept;
  73. void init_from_ref(ref_type ref) noexcept override
  74. {
  75. init_from_mem(MemRef(m_alloc.translate(ref), ref, m_alloc));
  76. }
  77. void init_from_parent();
  78. void destroy() noexcept;
  79. void detach() noexcept;
  80. size_t size() const;
  81. void add(StringData value);
  82. void set(size_t ndx, StringData value);
  83. void set_null(size_t ndx)
  84. {
  85. set(ndx, StringData{});
  86. }
  87. void insert(size_t ndx, StringData value);
  88. StringData get(size_t ndx) const;
  89. StringData get_legacy(size_t ndx) const;
  90. Mixed get_any(size_t ndx) const override;
  91. bool is_null(size_t ndx) const;
  92. void erase(size_t ndx);
  93. void move(ArrayString& dst, size_t ndx);
  94. void clear();
  95. size_t find_first(StringData value, size_t begin, size_t end) const noexcept;
  96. size_t lower_bound(StringData value);
  97. /// Get the specified element without the cost of constructing an
  98. /// array instance. If an array instance is already available, or
  99. /// you need to get multiple values, then this method will be
  100. /// slower.
  101. static StringData get(const char* header, size_t ndx, Allocator& alloc) noexcept;
  102. void verify() const;
  103. private:
  104. static constexpr size_t small_string_max_size = 15; // ArrayStringShort
  105. static constexpr size_t medium_string_max_size = 63; // ArrayStringLong
  106. union Storage {
  107. std::aligned_storage<sizeof(ArrayStringShort), alignof(ArrayStringShort)>::type m_string_short;
  108. std::aligned_storage<sizeof(ArraySmallBlobs), alignof(ArraySmallBlobs)>::type m_string_long;
  109. std::aligned_storage<sizeof(ArrayBigBlobs), alignof(ArrayBigBlobs)>::type m_big_blobs;
  110. std::aligned_storage<sizeof(Array), alignof(Array)>::type m_enum;
  111. };
  112. enum class Type { small_strings, medium_strings, big_strings, enum_strings };
  113. Type m_type = Type::small_strings;
  114. Allocator& m_alloc;
  115. Storage m_storage;
  116. Array* m_arr;
  117. mutable Spec* m_spec = nullptr;
  118. mutable size_t m_col_ndx = realm::npos;
  119. bool m_nullable = true;
  120. std::unique_ptr<ArrayString> m_string_enum_values;
  121. Type upgrade_leaf(size_t value_size);
  122. };
  123. inline StringData ArrayString::get(const char* header, size_t ndx, Allocator& alloc) noexcept
  124. {
  125. bool long_strings = Array::get_hasrefs_from_header(header);
  126. if (!long_strings) {
  127. return ArrayStringShort::get(header, ndx, true);
  128. }
  129. else {
  130. bool is_big = Array::get_context_flag_from_header(header);
  131. if (!is_big) {
  132. return ArraySmallBlobs::get_string(header, ndx, alloc);
  133. }
  134. else {
  135. return ArrayBigBlobs::get_string(header, ndx, alloc);
  136. }
  137. }
  138. }
  139. }
  140. #endif /* REALM_ARRAY_STRING_HPP */