1 /* 2 * Copyright (C) 2017 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 * Header file of an in-memory representation of DEX files. 17 */ 18 19 #ifndef ART_DEXLAYOUT_COMPACT_DEX_WRITER_H_ 20 #define ART_DEXLAYOUT_COMPACT_DEX_WRITER_H_ 21 22 #include <memory> // For unique_ptr 23 #include <unordered_map> 24 25 #include "base/utils.h" 26 #include "dex_writer.h" 27 28 namespace art { 29 30 // Compact dex writer for a single dex. 31 class CompactDexWriter : public DexWriter { 32 public: 33 explicit CompactDexWriter(DexLayout* dex_layout); 34 35 protected: 36 class Deduper { 37 public: 38 static const uint32_t kDidNotDedupe = 0; 39 40 // if not enabled, Dedupe will always return kDidNotDedupe. 41 explicit Deduper(bool enabled, DexContainer::Section* section); 42 43 // Deduplicate a blob of data that has been written to mem_map. 44 // Returns the offset of the deduplicated data or kDidNotDedupe did deduplication did not occur. 45 uint32_t Dedupe(uint32_t data_start, uint32_t data_end, uint32_t item_offset); 46 47 // Clear dedupe state to prevent deduplication against existing items in the future. Clear()48 void Clear() { 49 dedupe_map_.clear(); 50 } 51 52 private: 53 class HashedMemoryRange { 54 public: 55 uint32_t offset_; 56 uint32_t length_; 57 58 class HashEqual { 59 public: HashEqual(DexContainer::Section * section)60 explicit HashEqual(DexContainer::Section* section) : section_(section) {} 61 62 // Equal function. operator()63 bool operator()(const HashedMemoryRange& a, const HashedMemoryRange& b) const { 64 if (a.length_ != b.length_) { 65 return false; 66 } 67 const uint8_t* data = Data(); 68 DCHECK_LE(a.offset_ + a.length_, section_->Size()); 69 DCHECK_LE(b.offset_ + b.length_, section_->Size()); 70 return std::equal(data + a.offset_, data + a.offset_ + a.length_, data + b.offset_); 71 } 72 73 // Hash function. operator()74 size_t operator()(const HashedMemoryRange& range) const { 75 DCHECK_LE(range.offset_ + range.length_, section_->Size()); 76 return HashBytes(Data() + range.offset_, range.length_); 77 } 78 Data()79 ALWAYS_INLINE uint8_t* Data() const { 80 return section_->Begin(); 81 } 82 83 private: 84 DexContainer::Section* const section_; 85 }; 86 }; 87 88 const bool enabled_; 89 90 // Dedupe map. 91 std::unordered_map<HashedMemoryRange, 92 uint32_t, 93 HashedMemoryRange::HashEqual, 94 HashedMemoryRange::HashEqual> dedupe_map_; 95 }; 96 97 // Handles alignment and deduping of a data section item. 98 class ScopedDataSectionItem { 99 public: 100 ScopedDataSectionItem(Stream* stream, dex_ir::Item* item, size_t alignment, Deduper* deduper); 101 ~ScopedDataSectionItem(); 102 size_t Written() const; 103 104 private: 105 Stream* const stream_; 106 dex_ir::Item* const item_; 107 const size_t alignment_; 108 Deduper* deduper_; 109 const uint32_t start_offset_; 110 }; 111 112 public: 113 class Container : public DexContainer { 114 public: GetMainSection()115 Section* GetMainSection() OVERRIDE { 116 return &main_section_; 117 } 118 GetDataSection()119 Section* GetDataSection() OVERRIDE { 120 return &data_section_; 121 } 122 IsCompactDexContainer()123 bool IsCompactDexContainer() const OVERRIDE { 124 return true; 125 } 126 127 private: 128 explicit Container(bool dedupe_code_items); 129 130 VectorSection main_section_; 131 VectorSection data_section_; 132 Deduper code_item_dedupe_; 133 Deduper data_item_dedupe_; 134 135 friend class CompactDexWriter; 136 }; 137 138 protected: 139 // Return true if we can generate compact dex for the IR. 140 bool CanGenerateCompactDex(std::string* error_msg); 141 142 bool Write(DexContainer* output, std::string* error_msg) OVERRIDE; 143 144 std::unique_ptr<DexContainer> CreateDexContainer() const OVERRIDE; 145 146 void WriteHeader(Stream* stream) OVERRIDE; 147 148 size_t GetHeaderSize() const OVERRIDE; 149 150 uint32_t WriteDebugInfoOffsetTable(Stream* stream); 151 152 void WriteCodeItem(Stream* stream, dex_ir::CodeItem* code_item, bool reserve_only) OVERRIDE; 153 154 void WriteStringData(Stream* stream, dex_ir::StringData* string_data) OVERRIDE; 155 156 void WriteDebugInfoItem(Stream* stream, dex_ir::DebugInfoItem* debug_info) OVERRIDE; 157 158 void SortDebugInfosByMethodIndex(); 159 160 CompactDexLevel GetCompactDexLevel() const; 161 162 private: 163 // Position in the compact dex file for the debug info table data starts. 164 uint32_t debug_info_offsets_pos_ = 0u; 165 166 // Offset into the debug info table data where the lookup table is. 167 uint32_t debug_info_offsets_table_offset_ = 0u; 168 169 // Base offset of where debug info starts in the dex file. 170 uint32_t debug_info_base_ = 0u; 171 172 // Part of the shared data section owned by this file. 173 uint32_t owned_data_begin_ = 0u; 174 uint32_t owned_data_end_ = 0u; 175 176 // State for where we are deduping. 177 Deduper* code_item_dedupe_ = nullptr; 178 Deduper* data_item_dedupe_ = nullptr; 179 180 DISALLOW_COPY_AND_ASSIGN(CompactDexWriter); 181 }; 182 183 } // namespace art 184 185 #endif // ART_DEXLAYOUT_COMPACT_DEX_WRITER_H_ 186