1 /* 2 * Copyright (C) 2014 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 #ifndef ART_DEX2OAT_UTILS_SWAP_SPACE_H_ 18 #define ART_DEX2OAT_UTILS_SWAP_SPACE_H_ 19 20 #include <stddef.h> 21 #include <stdint.h> 22 #include <cstdlib> 23 #include <list> 24 #include <set> 25 #include <vector> 26 27 #include <android-base/logging.h> 28 29 #include "base/logging.h" 30 #include "base/macros.h" 31 #include "base/mutex.h" 32 33 namespace art { 34 35 // An arena pool that creates arenas backed by an mmaped file. 36 class SwapSpace { 37 public: 38 SwapSpace(int fd, size_t initial_size); 39 ~SwapSpace(); 40 void* Alloc(size_t size) REQUIRES(!lock_); 41 void Free(void* ptr, size_t size) REQUIRES(!lock_); 42 GetSize()43 size_t GetSize() { 44 return size_; 45 } 46 47 private: 48 // Chunk of space. 49 struct SpaceChunk { 50 // We need mutable members as we keep these objects in a std::set<> (providing only const 51 // access) but we modify these members while carefully preserving the std::set<> ordering. 52 mutable uint8_t* ptr; 53 mutable size_t size; 54 StartSpaceChunk55 uintptr_t Start() const { 56 return reinterpret_cast<uintptr_t>(ptr); 57 } EndSpaceChunk58 uintptr_t End() const { 59 return reinterpret_cast<uintptr_t>(ptr) + size; 60 } 61 }; 62 63 class SortChunkByPtr { 64 public: operator()65 bool operator()(const SpaceChunk& a, const SpaceChunk& b) const { 66 return reinterpret_cast<uintptr_t>(a.ptr) < reinterpret_cast<uintptr_t>(b.ptr); 67 } 68 }; 69 70 using FreeByStartSet = std::set<SpaceChunk, SortChunkByPtr>; 71 72 // Map size to an iterator to free_by_start_'s entry. 73 struct FreeBySizeEntry { FreeBySizeEntryFreeBySizeEntry74 FreeBySizeEntry(size_t sz, FreeByStartSet::const_iterator entry) 75 : size(sz), free_by_start_entry(entry) { } 76 77 // We need mutable members as we keep these objects in a std::set<> (providing only const 78 // access) but we modify these members while carefully preserving the std::set<> ordering. 79 mutable size_t size; 80 mutable FreeByStartSet::const_iterator free_by_start_entry; 81 }; 82 struct FreeBySizeComparator { operatorFreeBySizeComparator83 bool operator()(const FreeBySizeEntry& lhs, const FreeBySizeEntry& rhs) const { 84 if (lhs.size != rhs.size) { 85 return lhs.size < rhs.size; 86 } else { 87 return lhs.free_by_start_entry->Start() < rhs.free_by_start_entry->Start(); 88 } 89 } 90 }; 91 using FreeBySizeSet = std::set<FreeBySizeEntry, FreeBySizeComparator>; 92 93 SpaceChunk NewFileChunk(size_t min_size) REQUIRES(lock_); 94 95 void RemoveChunk(FreeBySizeSet::const_iterator free_by_size_pos) REQUIRES(lock_); 96 void InsertChunk(const SpaceChunk& chunk) REQUIRES(lock_); 97 98 int fd_; 99 size_t size_; 100 101 // NOTE: Boost.Bimap would be useful for the two following members. 102 103 // Map start of a free chunk to its size. 104 FreeByStartSet free_by_start_ GUARDED_BY(lock_); 105 // Free chunks ordered by size. 106 FreeBySizeSet free_by_size_ GUARDED_BY(lock_); 107 108 mutable Mutex lock_ DEFAULT_MUTEX_ACQUIRED_AFTER; 109 DISALLOW_COPY_AND_ASSIGN(SwapSpace); 110 }; 111 112 template <typename T> class SwapAllocator; 113 114 template <> 115 class SwapAllocator<void> { 116 public: 117 using value_type = void; 118 using pointer = void*; 119 using const_pointer = const void*; 120 121 template <typename U> 122 struct rebind { 123 using other = SwapAllocator<U>; 124 }; 125 SwapAllocator(SwapSpace * swap_space)126 explicit SwapAllocator(SwapSpace* swap_space) : swap_space_(swap_space) {} 127 128 template <typename U> SwapAllocator(const SwapAllocator<U> & other)129 SwapAllocator(const SwapAllocator<U>& other) 130 : swap_space_(other.swap_space_) {} 131 132 SwapAllocator(const SwapAllocator& other) = default; 133 SwapAllocator& operator=(const SwapAllocator& other) = default; 134 ~SwapAllocator() = default; 135 136 private: 137 SwapSpace* swap_space_; 138 139 template <typename U> 140 friend class SwapAllocator; 141 142 template <typename U> 143 friend bool operator==(const SwapAllocator<U>& lhs, const SwapAllocator<U>& rhs); 144 }; 145 146 template <typename T> 147 class SwapAllocator { 148 public: 149 using value_type = T; 150 using pointer = T*; 151 using reference = T&; 152 using const_pointer = const T*; 153 using const_reference = const T&; 154 using size_type = size_t; 155 using difference_type = ptrdiff_t; 156 157 template <typename U> 158 struct rebind { 159 using other = SwapAllocator<U>; 160 }; 161 SwapAllocator(SwapSpace * swap_space)162 explicit SwapAllocator(SwapSpace* swap_space) : swap_space_(swap_space) {} 163 164 template <typename U> SwapAllocator(const SwapAllocator<U> & other)165 SwapAllocator(const SwapAllocator<U>& other) 166 : swap_space_(other.swap_space_) {} 167 168 SwapAllocator(const SwapAllocator& other) = default; 169 SwapAllocator& operator=(const SwapAllocator& other) = default; 170 ~SwapAllocator() = default; 171 max_size()172 size_type max_size() const { 173 return static_cast<size_type>(-1) / sizeof(T); 174 } 175 address(reference x)176 pointer address(reference x) const { return &x; } address(const_reference x)177 const_pointer address(const_reference x) const { return &x; } 178 179 pointer allocate(size_type n, [[maybe_unused]] SwapAllocator<void>::pointer hint = nullptr) { 180 DCHECK_LE(n, max_size()); 181 if (swap_space_ == nullptr) { 182 T* result = reinterpret_cast<T*>(malloc(n * sizeof(T))); 183 CHECK_IMPLIES(result == nullptr, n == 0u); // Abort if malloc() fails. 184 return result; 185 } else { 186 return reinterpret_cast<T*>(swap_space_->Alloc(n * sizeof(T))); 187 } 188 } deallocate(pointer p,size_type n)189 void deallocate(pointer p, size_type n) { 190 if (swap_space_ == nullptr) { 191 free(p); 192 } else { 193 swap_space_->Free(p, n * sizeof(T)); 194 } 195 } 196 construct(pointer p,const_reference val)197 void construct(pointer p, const_reference val) { 198 new (static_cast<void*>(p)) value_type(val); 199 } 200 template <class U, class... Args> construct(U * p,Args &&...args)201 void construct(U* p, Args&&... args) { 202 ::new (static_cast<void*>(p)) U(std::forward<Args>(args)...); 203 } destroy(pointer p)204 void destroy(pointer p) { 205 p->~value_type(); 206 } 207 208 inline bool operator==(SwapAllocator const& other) { 209 return swap_space_ == other.swap_space_; 210 } 211 inline bool operator!=(SwapAllocator const& other) { 212 return !operator==(other); 213 } 214 215 private: 216 SwapSpace* swap_space_; 217 218 template <typename U> 219 friend class SwapAllocator; 220 221 template <typename U> 222 friend bool operator==(const SwapAllocator<U>& lhs, const SwapAllocator<U>& rhs); 223 }; 224 225 template <typename T> 226 inline bool operator==(const SwapAllocator<T>& lhs, const SwapAllocator<T>& rhs) { 227 return lhs.swap_space_ == rhs.swap_space_; 228 } 229 230 template <typename T> 231 inline bool operator!=(const SwapAllocator<T>& lhs, const SwapAllocator<T>& rhs) { 232 return !(lhs == rhs); 233 } 234 235 template <typename T> 236 using SwapVector = std::vector<T, SwapAllocator<T>>; 237 template <typename T, typename Comparator> 238 using SwapSet = std::set<T, Comparator, SwapAllocator<T>>; 239 240 } // namespace art 241 242 #endif // ART_DEX2OAT_UTILS_SWAP_SPACE_H_ 243