blob: dd7d699eee638b119c563c4491a43c8661ee7d88 [file] [log] [blame]
Brian Carlstrom7940e442013-07-12 13:46:57 -07001/*
2 * Copyright (C) 2011 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
Brian Carlstromfc0e3212013-07-17 14:40:12 -070017#ifndef ART_COMPILER_OAT_WRITER_H_
18#define ART_COMPILER_OAT_WRITER_H_
Brian Carlstrom7940e442013-07-12 13:46:57 -070019
20#include <stdint.h>
Brian Carlstrom7940e442013-07-12 13:46:57 -070021#include <cstddef>
Ian Rogers700a4022014-05-19 16:49:03 -070022#include <memory>
Brian Carlstrom7940e442013-07-12 13:46:57 -070023
David Brazdild9c90372016-09-14 16:53:55 +010024#include "base/array_ref.h"
Vladimir Marko49b0f452015-12-10 13:49:19 +000025#include "base/dchecked_vector.h"
Vladimir Markob163bb72015-03-31 21:49:49 +010026#include "linker/relative_patcher.h" // For linker::RelativePatcherTargetProvider.
Brian Carlstrom7940e442013-07-12 13:46:57 -070027#include "mem_map.h"
Vladimir Markof4da6752014-08-01 19:04:18 +010028#include "method_reference.h"
Brian Carlstrom7940e442013-07-12 13:46:57 -070029#include "mirror/class.h"
Artem Udovichenkod9786b02015-10-14 16:36:55 +030030#include "oat.h"
Vladimir Marko9bdf1082016-01-21 12:15:52 +000031#include "os.h"
Brian Carlstrom7940e442013-07-12 13:46:57 -070032#include "safe_map.h"
Brian Carlstrom7940e442013-07-12 13:46:57 -070033
34namespace art {
35
Brian Carlstromba150c32013-08-27 17:31:03 -070036class BitVector;
Andreas Gampe79273802014-08-05 20:21:05 -070037class CompiledMethod;
Vladimir Marko20f85592015-03-19 10:07:02 +000038class CompilerDriver;
Vladimir Markof4da6752014-08-01 19:04:18 +010039class ImageWriter;
Brian Carlstrom7940e442013-07-12 13:46:57 -070040class OutputStream;
Vladimir Marko20f85592015-03-19 10:07:02 +000041class TimingLogger;
Artem Udovichenkod9786b02015-10-14 16:36:55 +030042class TypeLookupTable;
Vladimir Marko9bdf1082016-01-21 12:15:52 +000043class ZipEntry;
Brian Carlstrom7940e442013-07-12 13:46:57 -070044
David Srbeckyc5bfa972016-02-05 15:49:10 +000045namespace debug {
Vladimir Marko10c13562015-11-25 14:33:36 +000046struct MethodDebugInfo;
David Srbeckyc5bfa972016-02-05 15:49:10 +000047} // namespace debug
Vladimir Marko10c13562015-11-25 14:33:36 +000048
Vladimir Marko944da602016-02-19 12:27:55 +000049namespace linker {
50class MultiOatRelativePatcher;
51} // namespace linker
52
Brian Carlstrom7940e442013-07-12 13:46:57 -070053// OatHeader variable length with count of D OatDexFiles
54//
55// OatDexFile[0] one variable sized OatDexFile with offsets to Dex and OatClasses
56// OatDexFile[1]
57// ...
58// OatDexFile[D]
59//
Artem Udovichenkod9786b02015-10-14 16:36:55 +030060// TypeLookupTable[0] one descriptor to class def index hash table for each OatDexFile.
61// TypeLookupTable[1]
62// ...
63// TypeLookupTable[D]
64//
Vladimir Marko9bdf1082016-01-21 12:15:52 +000065// ClassOffsets[0] one table of OatClass offsets for each class def for each OatDexFile.
66// ClassOffsets[1]
67// ...
68// ClassOffsets[D]
69//
Brian Carlstrom7940e442013-07-12 13:46:57 -070070// OatClass[0] one variable sized OatClass for each of C DexFile::ClassDefs
71// OatClass[1] contains OatClass entries with class status, offsets to code, etc.
72// ...
73// OatClass[C]
74//
Vladimir Marko96c6ab92014-04-08 14:00:50 +010075// GcMap one variable sized blob with GC map.
76// GcMap GC maps are deduplicated.
77// ...
78// GcMap
79//
80// VmapTable one variable sized VmapTable blob (quick compiler only).
81// VmapTable VmapTables are deduplicated.
82// ...
83// VmapTable
84//
85// MappingTable one variable sized blob with MappingTable (quick compiler only).
86// MappingTable MappingTables are deduplicated.
87// ...
88// MappingTable
89//
Brian Carlstrom7940e442013-07-12 13:46:57 -070090// padding if necessary so that the following code will be page aligned
91//
Vladimir Marko96c6ab92014-04-08 14:00:50 +010092// OatMethodHeader fixed size header for a CompiledMethod including the size of the MethodCode.
93// MethodCode one variable sized blob with the code of a CompiledMethod.
94// OatMethodHeader (OatMethodHeader, MethodCode) pairs are deduplicated.
95// MethodCode
Brian Carlstrom7940e442013-07-12 13:46:57 -070096// ...
Vladimir Marko96c6ab92014-04-08 14:00:50 +010097// OatMethodHeader
98// MethodCode
Brian Carlstrom7940e442013-07-12 13:46:57 -070099//
100class OatWriter {
101 public:
Vladimir Marko9bdf1082016-01-21 12:15:52 +0000102 enum class CreateTypeLookupTable {
103 kCreate,
104 kDontCreate,
105 kDefault = kCreate
106 };
107
108 OatWriter(bool compiling_boot_image, TimingLogger* timings);
109
110 // To produce a valid oat file, the user must first add sources with any combination of
111 // - AddDexFileSource(),
112 // - AddZippedDexFilesSource(),
113 // - AddRawDexFileSource().
114 // Then the user must call in order
115 // - WriteAndOpenDexFiles()
116 // - PrepareLayout(),
117 // - WriteRodata(),
118 // - WriteCode(),
119 // - WriteHeader().
120
121 // Add dex file source(s) from a file, either a plain dex file or
122 // a zip file with one or more dex files.
123 bool AddDexFileSource(
124 const char* filename,
125 const char* location,
126 CreateTypeLookupTable create_type_lookup_table = CreateTypeLookupTable::kDefault);
127 // Add dex file source(s) from a zip file specified by a file handle.
128 bool AddZippedDexFilesSource(
Andreas Gampe43e10b02016-07-15 17:17:34 -0700129 File&& zip_fd,
Vladimir Marko9bdf1082016-01-21 12:15:52 +0000130 const char* location,
131 CreateTypeLookupTable create_type_lookup_table = CreateTypeLookupTable::kDefault);
132 // Add dex file source from raw memory.
133 bool AddRawDexFileSource(
134 const ArrayRef<const uint8_t>& data,
135 const char* location,
136 uint32_t location_checksum,
137 CreateTypeLookupTable create_type_lookup_table = CreateTypeLookupTable::kDefault);
138 dchecked_vector<const char*> GetSourceLocations() const;
139
David Brazdil7b49e6c2016-09-01 11:06:18 +0100140 // Write raw dex files to the vdex file, mmap the file and open the dex files from it.
141 // Supporting data structures are written into the .rodata section of the oat file.
142 // The `verify` setting dictates whether the dex file verifier should check the dex files.
143 // This is generally the case, and should only be false for tests.
144 bool WriteAndOpenDexFiles(File* vdex_file,
145 OutputStream* oat_rodata,
Vladimir Marko9bdf1082016-01-21 12:15:52 +0000146 InstructionSet instruction_set,
147 const InstructionSetFeatures* instruction_set_features,
148 SafeMap<std::string, std::string>* key_value_store,
Andreas Gampe3a2bd292016-01-26 17:23:47 -0800149 bool verify,
Vladimir Marko9bdf1082016-01-21 12:15:52 +0000150 /*out*/ std::unique_ptr<MemMap>* opened_dex_files_map,
151 /*out*/ std::vector<std::unique_ptr<const DexFile>>* opened_dex_files);
152 // Prepare layout of remaining data.
153 void PrepareLayout(const CompilerDriver* compiler,
154 ImageWriter* image_writer,
Vladimir Marko944da602016-02-19 12:27:55 +0000155 const std::vector<const DexFile*>& dex_files,
156 linker::MultiOatRelativePatcher* relative_patcher);
Vladimir Marko9bdf1082016-01-21 12:15:52 +0000157 // Write the rest of .rodata section (ClassOffsets[], OatClass[], maps).
158 bool WriteRodata(OutputStream* out);
159 // Write the code to the .text section.
160 bool WriteCode(OutputStream* out);
161 // Write the oat header. This finalizes the oat file.
162 bool WriteHeader(OutputStream* out,
163 uint32_t image_file_location_oat_checksum,
164 uintptr_t image_file_location_oat_begin,
165 int32_t image_patch_delta);
Brian Carlstromc50d8e12013-07-23 22:35:16 -0700166
Vladimir Marko09d09432015-09-08 13:47:48 +0100167 // Returns whether the oat file has an associated image.
168 bool HasImage() const {
169 // Since the image is being created at the same time as the oat file,
170 // check if there's an image writer.
171 return image_writer_ != nullptr;
172 }
173
Mathieu Chartierda5b28a2015-11-05 08:03:47 -0800174 bool HasBootImage() const {
175 return compiling_boot_image_;
176 }
177
Brian Carlstromc50d8e12013-07-23 22:35:16 -0700178 const OatHeader& GetOatHeader() const {
179 return *oat_header_;
180 }
181
David Brazdil7b49e6c2016-09-01 11:06:18 +0100182 size_t GetOatSize() const {
183 return oat_size_;
Brian Carlstromc50d8e12013-07-23 22:35:16 -0700184 }
185
Vladimir Marko5c42c292015-02-25 12:02:49 +0000186 size_t GetBssSize() const {
187 return bss_size_;
188 }
189
Vladimir Marko944da602016-02-19 12:27:55 +0000190 size_t GetOatDataOffset() const {
191 return oat_data_offset_;
192 }
193
Vladimir Marko49b0f452015-12-10 13:49:19 +0000194 ArrayRef<const uintptr_t> GetAbsolutePatchLocations() const {
195 return ArrayRef<const uintptr_t>(absolute_patch_locations_);
Vladimir Markof4da6752014-08-01 19:04:18 +0100196 }
197
Brian Carlstrom7940e442013-07-12 13:46:57 -0700198 ~OatWriter();
199
David Srbecky09c2a6b2016-03-11 17:11:44 +0000200 void AddMethodDebugInfos(const std::vector<debug::MethodDebugInfo>& infos) {
201 method_info_.insert(method_info_.end(), infos.begin(), infos.end());
202 }
203
David Srbeckyc5bfa972016-02-05 15:49:10 +0000204 ArrayRef<const debug::MethodDebugInfo> GetMethodDebugInfo() const {
205 return ArrayRef<const debug::MethodDebugInfo>(method_info_);
Mark Mendellae9fd932014-02-10 16:14:35 -0800206 }
207
Vladimir Markob163bb72015-03-31 21:49:49 +0100208 const CompilerDriver* GetCompilerDriver() {
209 return compiler_driver_;
210 }
211
Brian Carlstromc50d8e12013-07-23 22:35:16 -0700212 private:
Vladimir Marko9bdf1082016-01-21 12:15:52 +0000213 class DexFileSource;
Vladimir Marko49b0f452015-12-10 13:49:19 +0000214 class OatClass;
215 class OatDexFile;
216
Vladimir Marko96c6ab92014-04-08 14:00:50 +0100217 // The function VisitDexMethods() below iterates through all the methods in all
218 // the compiled dex files in order of their definitions. The method visitor
219 // classes provide individual bits of processing for each of the passes we need to
220 // first collect the data we want to write to the oat file and then, in later passes,
221 // to actually write it.
222 class DexMethodVisitor;
223 class OatDexMethodVisitor;
224 class InitOatClassesMethodVisitor;
225 class InitCodeMethodVisitor;
Vladimir Marko96c6ab92014-04-08 14:00:50 +0100226 class InitMapMethodVisitor;
227 class InitImageMethodVisitor;
228 class WriteCodeMethodVisitor;
Vladimir Marko96c6ab92014-04-08 14:00:50 +0100229 class WriteMapMethodVisitor;
230
231 // Visit all the methods in all the compiled dex files in their definition order
232 // with a given DexMethodVisitor.
233 bool VisitDexMethods(DexMethodVisitor* visitor);
234
David Brazdil7b49e6c2016-09-01 11:06:18 +0100235 bool WriteVdexHeader(OutputStream* vdex_out);
236
237 bool WriteDexFiles(OutputStream* out, File* file);
238 bool WriteDexFile(OutputStream* out, File* file, OatDexFile* oat_dex_file);
239 bool SeekToDexFile(OutputStream* out, File* file, OatDexFile* oat_dex_file);
240 bool WriteDexFile(OutputStream* out,
241 File* file,
242 OatDexFile* oat_dex_file,
243 ZipEntry* dex_file);
244 bool WriteDexFile(OutputStream* out,
245 File* file,
246 OatDexFile* oat_dex_file,
247 File* dex_file);
248 bool WriteDexFile(OutputStream* out, OatDexFile* oat_dex_file, const uint8_t* dex_file);
249 bool OpenDexFiles(File* file,
250 bool verify,
251 /*out*/ std::unique_ptr<MemMap>* opened_dex_files_map,
252 /*out*/ std::vector<std::unique_ptr<const DexFile>>* opened_dex_files);
253
Vladimir Marko9bdf1082016-01-21 12:15:52 +0000254 size_t InitOatHeader(InstructionSet instruction_set,
255 const InstructionSetFeatures* instruction_set_features,
256 uint32_t num_dex_files,
257 SafeMap<std::string, std::string>* key_value_store);
Brian Carlstrom7940e442013-07-12 13:46:57 -0700258 size_t InitOatDexFiles(size_t offset);
Brian Carlstrom7940e442013-07-12 13:46:57 -0700259 size_t InitOatClasses(size_t offset);
Vladimir Marko96c6ab92014-04-08 14:00:50 +0100260 size_t InitOatMaps(size_t offset);
Vladimir Marko49b0f452015-12-10 13:49:19 +0000261 size_t InitOatCode(size_t offset);
262 size_t InitOatCodeDexFiles(size_t offset);
Brian Carlstrom7940e442013-07-12 13:46:57 -0700263
Vladimir Marko9bdf1082016-01-21 12:15:52 +0000264 bool WriteClassOffsets(OutputStream* out);
265 bool WriteClasses(OutputStream* out);
Vladimir Marko96c6ab92014-04-08 14:00:50 +0100266 size_t WriteMaps(OutputStream* out, const size_t file_offset, size_t relative_offset);
267 size_t WriteCode(OutputStream* out, const size_t file_offset, size_t relative_offset);
Ian Rogers3d504072014-03-01 09:16:49 -0800268 size_t WriteCodeDexFiles(OutputStream* out, const size_t file_offset, size_t relative_offset);
Brian Carlstrom7940e442013-07-12 13:46:57 -0700269
Vladimir Marko944da602016-02-19 12:27:55 +0000270 bool RecordOatDataOffset(OutputStream* out);
David Brazdil7b49e6c2016-09-01 11:06:18 +0100271 bool ReadDexFileHeader(File* oat_file, OatDexFile* oat_dex_file);
Vladimir Marko9bdf1082016-01-21 12:15:52 +0000272 bool ValidateDexFileHeader(const uint8_t* raw_header, const char* location);
David Brazdil7b49e6c2016-09-01 11:06:18 +0100273 bool WriteOatDexFiles(OutputStream* oat_rodata);
274 bool WriteTypeLookupTables(OutputStream* oat_rodata,
Vladimir Marko9bdf1082016-01-21 12:15:52 +0000275 const std::vector<std::unique_ptr<const DexFile>>& opened_dex_files);
Vladimir Markof4da6752014-08-01 19:04:18 +0100276 bool WriteCodeAlignment(OutputStream* out, uint32_t aligned_code_delta);
Vladimir Marko944da602016-02-19 12:27:55 +0000277 void SetMultiOatRelativePatcherAdjustment();
Vladimir Markof4da6752014-08-01 19:04:18 +0100278
Vladimir Marko9bdf1082016-01-21 12:15:52 +0000279 enum class WriteState {
280 kAddingDexFileSources,
281 kPrepareLayout,
282 kWriteRoData,
283 kWriteText,
284 kWriteHeader,
285 kDone
286 };
287
288 WriteState write_state_;
289 TimingLogger* timings_;
290
291 std::vector<std::unique_ptr<File>> raw_dex_files_;
292 std::vector<std::unique_ptr<ZipArchive>> zip_archives_;
293 std::vector<std::unique_ptr<ZipEntry>> zipped_dex_files_;
294
295 // Using std::list<> which doesn't move elements around on push/emplace_back().
296 // We need this because we keep plain pointers to the strings' c_str().
297 std::list<std::string> zipped_dex_file_locations_;
298
David Srbeckyc5bfa972016-02-05 15:49:10 +0000299 dchecked_vector<debug::MethodDebugInfo> method_info_;
Mark Mendellae9fd932014-02-10 16:14:35 -0800300
Vladimir Marko9bdf1082016-01-21 12:15:52 +0000301 const CompilerDriver* compiler_driver_;
302 ImageWriter* image_writer_;
Mathieu Chartierda5b28a2015-11-05 08:03:47 -0800303 const bool compiling_boot_image_;
Brian Carlstrom7940e442013-07-12 13:46:57 -0700304
305 // note OatFile does not take ownership of the DexFiles
306 const std::vector<const DexFile*>* dex_files_;
307
David Brazdil7b49e6c2016-09-01 11:06:18 +0100308 // Size required for Vdex data structures.
309 size_t vdex_size_;
310
311 // Offset of section holding Dex files inside Vdex.
312 size_t vdex_dex_files_offset_;
313
Brian Carlstromc50d8e12013-07-23 22:35:16 -0700314 // Size required for Oat data structures.
David Brazdil7b49e6c2016-09-01 11:06:18 +0100315 size_t oat_size_;
Brian Carlstromc50d8e12013-07-23 22:35:16 -0700316
Vladimir Marko5c42c292015-02-25 12:02:49 +0000317 // The size of the required .bss section holding the DexCache data.
318 size_t bss_size_;
319
Vladimir Marko09d09432015-09-08 13:47:48 +0100320 // Offsets of the dex cache arrays for each app dex file. For the
321 // boot image, this information is provided by the ImageWriter.
Vladimir Marko06d7aaa2015-10-16 11:23:41 +0100322 SafeMap<const DexFile*, size_t> dex_cache_arrays_offsets_; // DexFiles not owned.
Vladimir Marko09d09432015-09-08 13:47:48 +0100323
Vladimir Markof4da6752014-08-01 19:04:18 +0100324 // Offset of the oat data from the start of the mmapped region of the elf file.
325 size_t oat_data_offset_;
326
Brian Carlstrom7940e442013-07-12 13:46:57 -0700327 // data to write
Vladimir Marko49b0f452015-12-10 13:49:19 +0000328 std::unique_ptr<OatHeader> oat_header_;
329 dchecked_vector<OatDexFile> oat_dex_files_;
330 dchecked_vector<OatClass> oat_classes_;
Ian Rogers700a4022014-05-19 16:49:03 -0700331 std::unique_ptr<const std::vector<uint8_t>> jni_dlsym_lookup_;
Ian Rogers700a4022014-05-19 16:49:03 -0700332 std::unique_ptr<const std::vector<uint8_t>> quick_generic_jni_trampoline_;
333 std::unique_ptr<const std::vector<uint8_t>> quick_imt_conflict_trampoline_;
334 std::unique_ptr<const std::vector<uint8_t>> quick_resolution_trampoline_;
335 std::unique_ptr<const std::vector<uint8_t>> quick_to_interpreter_bridge_;
Brian Carlstrom7940e442013-07-12 13:46:57 -0700336
337 // output stats
David Brazdil7b49e6c2016-09-01 11:06:18 +0100338 uint32_t size_vdex_header_;
Brian Carlstrom7940e442013-07-12 13:46:57 -0700339 uint32_t size_dex_file_alignment_;
340 uint32_t size_executable_offset_alignment_;
341 uint32_t size_oat_header_;
Andreas Gampe22f8e5c2014-07-09 11:38:21 -0700342 uint32_t size_oat_header_key_value_store_;
Brian Carlstrom7940e442013-07-12 13:46:57 -0700343 uint32_t size_dex_file_;
Ian Rogers468532e2013-08-05 10:56:33 -0700344 uint32_t size_interpreter_to_interpreter_bridge_;
345 uint32_t size_interpreter_to_compiled_code_bridge_;
346 uint32_t size_jni_dlsym_lookup_;
Andreas Gampe2da88232014-02-27 12:26:20 -0800347 uint32_t size_quick_generic_jni_trampoline_;
Jeff Hao88474b42013-10-23 16:24:40 -0700348 uint32_t size_quick_imt_conflict_trampoline_;
Brian Carlstrom7940e442013-07-12 13:46:57 -0700349 uint32_t size_quick_resolution_trampoline_;
Ian Rogers468532e2013-08-05 10:56:33 -0700350 uint32_t size_quick_to_interpreter_bridge_;
351 uint32_t size_trampoline_alignment_;
Vladimir Marko96c6ab92014-04-08 14:00:50 +0100352 uint32_t size_method_header_;
Brian Carlstrom7940e442013-07-12 13:46:57 -0700353 uint32_t size_code_;
354 uint32_t size_code_alignment_;
Vladimir Markof4da6752014-08-01 19:04:18 +0100355 uint32_t size_relative_call_thunks_;
Vladimir Markoc74658b2015-03-31 10:26:41 +0100356 uint32_t size_misc_thunks_;
Brian Carlstrom7940e442013-07-12 13:46:57 -0700357 uint32_t size_vmap_table_;
Brian Carlstrom7940e442013-07-12 13:46:57 -0700358 uint32_t size_oat_dex_file_location_size_;
359 uint32_t size_oat_dex_file_location_data_;
360 uint32_t size_oat_dex_file_location_checksum_;
361 uint32_t size_oat_dex_file_offset_;
Vladimir Marko9bdf1082016-01-21 12:15:52 +0000362 uint32_t size_oat_dex_file_class_offsets_offset_;
Vladimir Marko49b0f452015-12-10 13:49:19 +0000363 uint32_t size_oat_dex_file_lookup_table_offset_;
Vladimir Marko49b0f452015-12-10 13:49:19 +0000364 uint32_t size_oat_lookup_table_alignment_;
365 uint32_t size_oat_lookup_table_;
Vladimir Marko9bdf1082016-01-21 12:15:52 +0000366 uint32_t size_oat_class_offsets_alignment_;
367 uint32_t size_oat_class_offsets_;
Brian Carlstromba150c32013-08-27 17:31:03 -0700368 uint32_t size_oat_class_type_;
Brian Carlstrom7940e442013-07-12 13:46:57 -0700369 uint32_t size_oat_class_status_;
Brian Carlstromba150c32013-08-27 17:31:03 -0700370 uint32_t size_oat_class_method_bitmaps_;
Brian Carlstrom7940e442013-07-12 13:46:57 -0700371 uint32_t size_oat_class_method_offsets_;
372
Vladimir Marko944da602016-02-19 12:27:55 +0000373 // The helper for processing relative patches is external so that we can patch across oat files.
374 linker::MultiOatRelativePatcher* relative_patcher_;
Vladimir Markof4da6752014-08-01 19:04:18 +0100375
David Srbeckyf8980872015-05-22 17:04:47 +0100376 // The locations of absolute patches relative to the start of the executable section.
Vladimir Marko9bdf1082016-01-21 12:15:52 +0000377 dchecked_vector<uintptr_t> absolute_patch_locations_;
Vladimir Markof4da6752014-08-01 19:04:18 +0100378
Brian Carlstrom7940e442013-07-12 13:46:57 -0700379 DISALLOW_COPY_AND_ASSIGN(OatWriter);
380};
381
382} // namespace art
383
Brian Carlstromfc0e3212013-07-17 14:40:12 -0700384#endif // ART_COMPILER_OAT_WRITER_H_