group_writer.hpp 7.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199
  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_GROUP_WRITER_HPP
  19. #define REALM_GROUP_WRITER_HPP
  20. #include <cstdint> // unint8_t etc
  21. #include <utility>
  22. #include <map>
  23. #include <realm/util/file.hpp>
  24. #include <realm/alloc.hpp>
  25. #include <realm/array.hpp>
  26. #include <realm/impl/array_writer.hpp>
  27. #include <realm/db_options.hpp>
  28. namespace realm {
  29. // Pre-declarations
  30. class Group;
  31. class SlabAlloc;
  32. /// This class is not supposed to be reused for multiple write sessions. In
  33. /// particular, do not reuse it in case any of the functions throw.
  34. class GroupWriter : public _impl::ArrayWriterBase {
  35. public:
  36. // For groups in transactional mode (Group::m_is_shared), this constructor
  37. // must be called while a write transaction is in progress.
  38. //
  39. // The constructor adds free-space tracking information to the specified
  40. // group, if it is not already present (4th and 5th entry in
  41. // Group::m_top). If the specified group is in transactional mode
  42. // (Group::m_is_shared), the constructor also adds version tracking
  43. // information to the group, if it is not already present (6th and 7th entry
  44. // in Group::m_top).
  45. using Durability = DBOptions::Durability;
  46. GroupWriter(Group&, Durability dura = Durability::Full);
  47. ~GroupWriter();
  48. void set_versions(uint64_t current, uint64_t read_lock) noexcept;
  49. /// Write all changed array nodes into free space.
  50. ///
  51. /// Returns the new top ref. When in full durability mode, call
  52. /// commit() with the returned top ref.
  53. ref_type write_group();
  54. /// Flush changes to physical medium, then write the new top ref
  55. /// to the file header, then flush again. Pass the top ref
  56. /// returned by write_group().
  57. void commit(ref_type new_top_ref);
  58. size_t get_file_size() const noexcept;
  59. ref_type write_array(const char*, size_t, uint32_t) override;
  60. #ifdef REALM_DEBUG
  61. void dump();
  62. #endif
  63. size_t get_free_space_size() const
  64. {
  65. return m_free_space_size;
  66. }
  67. size_t get_locked_space_size() const
  68. {
  69. return m_locked_space_size;
  70. }
  71. private:
  72. class MapWindow;
  73. Group& m_group;
  74. SlabAlloc& m_alloc;
  75. Array m_free_positions; // 4th slot in Group::m_top
  76. Array m_free_lengths; // 5th slot in Group::m_top
  77. Array m_free_versions; // 6th slot in Group::m_top
  78. uint64_t m_current_version = 0;
  79. uint64_t m_readlock_version;
  80. size_t m_window_alignment;
  81. size_t m_free_space_size = 0;
  82. size_t m_locked_space_size = 0;
  83. Durability m_durability;
  84. struct FreeSpaceEntry {
  85. FreeSpaceEntry(size_t r, size_t s, uint64_t v)
  86. : ref(r)
  87. , size(s)
  88. , released_at_version(v)
  89. {
  90. }
  91. size_t ref;
  92. size_t size;
  93. uint64_t released_at_version;
  94. };
  95. class FreeList : public std::vector<FreeSpaceEntry> {
  96. public:
  97. FreeList() = default;
  98. // Merge adjacent chunks
  99. void merge_adjacent_entries_in_freelist();
  100. // Copy free space entries to structure where entries are sorted by size
  101. void move_free_in_file_to_size_map(std::multimap<size_t, size_t>& size_map);
  102. };
  103. // m_free_in_file;
  104. std::vector<FreeSpaceEntry> m_not_free_in_file;
  105. std::multimap<size_t, size_t> m_size_map;
  106. using FreeListElement = std::multimap<size_t, size_t>::iterator;
  107. void read_in_freelist();
  108. size_t recreate_freelist(size_t reserve_pos);
  109. // Currently cached memory mappings. We keep as many as 16 1MB windows
  110. // open for writing. The allocator will favor sequential allocation
  111. // from a modest number of windows, depending upon fragmentation, so
  112. // 16 windows should be more than enough. If more than 16 windows are
  113. // needed, the least recently used is sync'ed and closed to make room
  114. // for a new one. The windows are kept in MRU (most recently used) order.
  115. const static int num_map_windows = 16;
  116. std::vector<std::unique_ptr<MapWindow>> m_map_windows;
  117. // Get a suitable memory mapping for later access:
  118. // potentially adding it to the cache, potentially closing
  119. // the least recently used and sync'ing it to disk
  120. MapWindow* get_window(ref_type start_ref, size_t size);
  121. // Sync all cached memory mappings
  122. void sync_all_mappings();
  123. /// Allocate a chunk of free space of the specified size. The
  124. /// specified size must be 8-byte aligned. Extend the file if
  125. /// required. The returned chunk is removed from the amount of
  126. /// remaing free space. The returned chunk is guaranteed to be
  127. /// within a single contiguous memory mapping.
  128. ///
  129. /// \return The position within the database file of the allocated
  130. /// chunk.
  131. size_t get_free_space(size_t size);
  132. /// Find a block of free space that is at least as big as the
  133. /// specified size and which will allow an allocation that is mapped
  134. /// inside a contiguous address range. The specified size does not
  135. /// need to be 8-byte aligned. Extend the file if required.
  136. /// The returned chunk is not removed from the amount of remaing
  137. /// free space.
  138. ///
  139. /// \return A pair (`chunk_ndx`, `chunk_size`) where `chunk_ndx`
  140. /// is the index of a chunk whose size is at least the requestd
  141. /// size, and `chunk_size` is the size of that chunk.
  142. FreeListElement reserve_free_space(size_t size);
  143. FreeListElement search_free_space_in_free_list_element(FreeListElement element, size_t size);
  144. /// Search only a range of the free list for a block as big as the
  145. /// specified size. Return a pair with index and size of the found chunk.
  146. FreeListElement search_free_space_in_part_of_freelist(size_t size);
  147. /// Extend the file to ensure that a chunk of free space of the
  148. /// specified size is available. The specified size does not need
  149. /// to be 8-byte aligned. This function guarantees that it will
  150. /// add at most one entry to the free-lists.
  151. ///
  152. /// \return A pair (`chunk_ndx`, `chunk_size`) where `chunk_ndx`
  153. /// is the index of a chunk whose size is at least the requestd
  154. /// size, and `chunk_size` is the size of that chunk.
  155. FreeListElement extend_free_space(size_t requested_size);
  156. void write_array_at(MapWindow* window, ref_type, const char* data, size_t size);
  157. FreeListElement split_freelist_chunk(FreeListElement, size_t alloc_pos);
  158. };
  159. // Implementation:
  160. inline void GroupWriter::set_versions(uint64_t current, uint64_t read_lock) noexcept
  161. {
  162. REALM_ASSERT(read_lock <= current);
  163. m_current_version = current;
  164. m_readlock_version = read_lock;
  165. }
  166. } // namespace realm
  167. #endif // REALM_GROUP_WRITER_HPP