|  | /* | 
|  | * Copyright (C) 2017 The Android Open Source Project | 
|  | * | 
|  | * Licensed under the Apache License, Version 2.0 (the "License"); | 
|  | * you may not use this file except in compliance with the License. | 
|  | * You may obtain a copy of the License at | 
|  | * | 
|  | *      http://www.apache.org/licenses/LICENSE-2.0 | 
|  | * | 
|  | * Unless required by applicable law or agreed to in writing, software | 
|  | * distributed under the License is distributed on an "AS IS" BASIS, | 
|  | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
|  | * See the License for the specific language governing permissions and | 
|  | * limitations under the License. | 
|  | */ | 
|  |  | 
|  | #ifndef ART_RUNTIME_DEX_REFERENCE_COLLECTION_H_ | 
|  | #define ART_RUNTIME_DEX_REFERENCE_COLLECTION_H_ | 
|  |  | 
|  | #include <map> | 
|  | #include <vector> | 
|  |  | 
|  | #include "base/macros.h" | 
|  |  | 
|  | namespace art { | 
|  |  | 
|  | class DexFile; | 
|  |  | 
|  | // Collection of dex references that is more memory efficient than a vector of <dex, index> pairs. | 
|  | // Also allows quick lookups of all of the references for a single dex. | 
|  | template <class IndexType, template<typename Type> class Allocator> | 
|  | class DexReferenceCollection { | 
|  | public: | 
|  | using VectorAllocator = Allocator<IndexType>; | 
|  | using IndexVector = std::vector<IndexType, VectorAllocator>; | 
|  | using MapAllocator = Allocator<std::pair<const DexFile*, IndexVector>>; | 
|  | using DexFileMap = std::map< | 
|  | const DexFile*, | 
|  | IndexVector, | 
|  | std::less<const DexFile*>, | 
|  | Allocator<std::pair<const DexFile* const, IndexVector>>>; | 
|  |  | 
|  | DexReferenceCollection(const MapAllocator& map_allocator = MapAllocator(), | 
|  | const VectorAllocator& vector_allocator = VectorAllocator()) | 
|  | : map_(map_allocator), | 
|  | vector_allocator_(vector_allocator) {} | 
|  |  | 
|  | void AddReference(const DexFile* dex, IndexType index) { | 
|  | GetOrInsertVector(dex)->push_back(index); | 
|  | } | 
|  |  | 
|  | DexFileMap& GetMap() { | 
|  | return map_; | 
|  | } | 
|  |  | 
|  | size_t NumReferences() const { | 
|  | size_t ret = 0; | 
|  | for (auto&& pair : map_) { | 
|  | ret += pair.second.size(); | 
|  | } | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | private: | 
|  | DexFileMap map_; | 
|  | const DexFile* current_dex_file_ = nullptr; | 
|  | IndexVector* current_vector_ = nullptr; | 
|  | VectorAllocator vector_allocator_; | 
|  |  | 
|  | ALWAYS_INLINE IndexVector* GetOrInsertVector(const DexFile* dex) { | 
|  | // Optimize for adding to same vector in succession, the cached dex file and vector aims to | 
|  | // prevent map lookups. | 
|  | if (UNLIKELY(current_dex_file_ != dex)) { | 
|  | // There is an assumption that constructing an empty vector wont do any allocations. If this | 
|  | // incorrect, this might leak for the arena case. | 
|  | current_vector_ = &map_.emplace(dex, IndexVector(vector_allocator_)).first->second; | 
|  | current_dex_file_ = dex; | 
|  | } | 
|  | return current_vector_; | 
|  | } | 
|  | }; | 
|  |  | 
|  | }  // namespace art | 
|  |  | 
|  | #endif  // ART_RUNTIME_DEX_REFERENCE_COLLECTION_H_ |