blob: 6c76f4915e09d90c9d6bb50061cce54c80deb27f [file] [log] [blame]
Elliott Hughes2faa5f12012-01-30 14:42:07 -08001/*
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 */
Brian Carlstrom4a289ed2011-08-16 17:17:49 -070016
Brian Carlstromfc0e3212013-07-17 14:40:12 -070017#ifndef ART_RUNTIME_IMAGE_H_
18#define ART_RUNTIME_IMAGE_H_
Brian Carlstrom4a289ed2011-08-16 17:17:49 -070019
20#include <string.h>
21
Andreas Gampe542451c2016-07-26 09:02:02 -070022#include "base/enums.h"
Brian Carlstrom4a289ed2011-08-16 17:17:49 -070023#include "globals.h"
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080024#include "mirror/object.h"
Brian Carlstrom4a289ed2011-08-16 17:17:49 -070025
26namespace art {
27
Mathieu Chartier54d220e2015-07-30 16:20:06 -070028class ArtField;
29class ArtMethod;
30
31class ArtMethodVisitor {
32 public:
33 virtual ~ArtMethodVisitor() {}
34
35 virtual void Visit(ArtMethod* method) = 0;
36};
37
38class ArtFieldVisitor {
39 public:
40 virtual ~ArtFieldVisitor() {}
41
42 virtual void Visit(ArtField* method) = 0;
43};
44
Mathieu Chartiere401d142015-04-22 13:56:20 -070045class PACKED(4) ImageSection {
46 public:
47 ImageSection() : offset_(0), size_(0) { }
48 ImageSection(uint32_t offset, uint32_t size) : offset_(offset), size_(size) { }
49 ImageSection(const ImageSection& section) = default;
50 ImageSection& operator=(const ImageSection& section) = default;
51
52 uint32_t Offset() const {
53 return offset_;
54 }
55
56 uint32_t Size() const {
57 return size_;
58 }
59
60 uint32_t End() const {
61 return Offset() + Size();
62 }
63
64 bool Contains(uint64_t offset) const {
65 return offset - offset_ < size_;
66 }
67
68 private:
69 uint32_t offset_;
70 uint32_t size_;
71};
72
Brian Carlstrom4a289ed2011-08-16 17:17:49 -070073// header of image files written by ImageWriter, read and validated by Space.
Ian Rogersdf1ce912012-11-27 17:07:11 -080074class PACKED(4) ImageHeader {
Brian Carlstrom4a289ed2011-08-16 17:17:49 -070075 public:
Mathieu Chartierceb07b32015-12-10 09:33:21 -080076 enum StorageMode : uint32_t {
77 kStorageModeUncompressed,
78 kStorageModeLZ4,
Mathieu Chartiera6e81ed2016-02-25 13:52:10 -080079 kStorageModeLZ4HC,
Mathieu Chartierceb07b32015-12-10 09:33:21 -080080 kStorageModeCount, // Number of elements in enum.
81 };
82 static constexpr StorageMode kDefaultStorageMode = kStorageModeUncompressed;
83
Sebastien Hertzaa50d3a2015-08-25 15:25:41 +020084 ImageHeader()
Mathieu Chartierceb07b32015-12-10 09:33:21 -080085 : image_begin_(0U),
86 image_size_(0U),
87 oat_checksum_(0U),
88 oat_file_begin_(0U),
89 oat_data_begin_(0U),
90 oat_data_end_(0U),
91 oat_file_end_(0U),
Mathieu Chartierfbc31082016-01-24 11:59:56 -080092 boot_image_begin_(0U),
93 boot_image_size_(0U),
94 boot_oat_begin_(0U),
95 boot_oat_size_(0U),
Mathieu Chartierceb07b32015-12-10 09:33:21 -080096 patch_delta_(0),
97 image_roots_(0U),
98 pointer_size_(0U),
99 compile_pic_(0),
Mathieu Chartierfbc31082016-01-24 11:59:56 -0800100 is_pic_(0),
Mathieu Chartierceb07b32015-12-10 09:33:21 -0800101 storage_mode_(kDefaultStorageMode),
102 data_size_(0) {}
Brian Carlstrom4a289ed2011-08-16 17:17:49 -0700103
Ian Rogers30fab402012-01-23 15:43:46 -0800104 ImageHeader(uint32_t image_begin,
Mathieu Chartier763a31e2015-11-16 16:05:55 -0800105 uint32_t image_size,
Mathieu Chartiere401d142015-04-22 13:56:20 -0700106 ImageSection* sections,
Brian Carlstrome24fa612011-09-29 00:53:55 -0700107 uint32_t image_roots,
108 uint32_t oat_checksum,
Brian Carlstrom700c8d32012-11-05 10:42:02 -0800109 uint32_t oat_file_begin,
110 uint32_t oat_data_begin,
111 uint32_t oat_data_end,
Igor Murashkin46774762014-10-22 11:37:02 -0700112 uint32_t oat_file_end,
Mathieu Chartierfbc31082016-01-24 11:59:56 -0800113 uint32_t boot_image_begin,
114 uint32_t boot_image_size,
115 uint32_t boot_oat_begin,
116 uint32_t boot_oat_size,
Mathieu Chartiere401d142015-04-22 13:56:20 -0700117 uint32_t pointer_size,
Mathieu Chartierceb07b32015-12-10 09:33:21 -0800118 bool compile_pic,
Mathieu Chartierfbc31082016-01-24 11:59:56 -0800119 bool is_pic,
Mathieu Chartierceb07b32015-12-10 09:33:21 -0800120 StorageMode storage_mode,
121 size_t data_size);
Brian Carlstrom4a289ed2011-08-16 17:17:49 -0700122
Brian Carlstrom68708f52013-09-03 14:15:31 -0700123 bool IsValid() const;
124 const char* GetMagic() const;
Brian Carlstrom78128a62011-09-15 17:21:19 -0700125
Ian Rogers13735952014-10-08 12:43:28 -0700126 uint8_t* GetImageBegin() const {
127 return reinterpret_cast<uint8_t*>(image_begin_);
Brian Carlstrome24fa612011-09-29 00:53:55 -0700128 }
129
Mathieu Chartier31e89252013-08-28 11:29:12 -0700130 size_t GetImageSize() const {
131 return static_cast<uint32_t>(image_size_);
132 }
133
Brian Carlstrome24fa612011-09-29 00:53:55 -0700134 uint32_t GetOatChecksum() const {
135 return oat_checksum_;
136 }
137
Brian Carlstroma85b8372012-10-18 17:00:32 -0700138 void SetOatChecksum(uint32_t oat_checksum) {
139 oat_checksum_ = oat_checksum;
140 }
141
Mathieu Chartier5351da02016-02-17 16:19:53 -0800142 // The location that the oat file was expected to be when the image was created. The actual
143 // oat file may be at a different location for application images.
Ian Rogers13735952014-10-08 12:43:28 -0700144 uint8_t* GetOatFileBegin() const {
145 return reinterpret_cast<uint8_t*>(oat_file_begin_);
Brian Carlstrome24fa612011-09-29 00:53:55 -0700146 }
147
Ian Rogers13735952014-10-08 12:43:28 -0700148 uint8_t* GetOatDataBegin() const {
149 return reinterpret_cast<uint8_t*>(oat_data_begin_);
Brian Carlstrom700c8d32012-11-05 10:42:02 -0800150 }
151
Ian Rogers13735952014-10-08 12:43:28 -0700152 uint8_t* GetOatDataEnd() const {
153 return reinterpret_cast<uint8_t*>(oat_data_end_);
Brian Carlstrom700c8d32012-11-05 10:42:02 -0800154 }
155
Ian Rogers13735952014-10-08 12:43:28 -0700156 uint8_t* GetOatFileEnd() const {
157 return reinterpret_cast<uint8_t*>(oat_file_end_);
Brian Carlstrom4a289ed2011-08-16 17:17:49 -0700158 }
159
Andreas Gampebda1d602016-08-29 17:43:45 -0700160 PointerSize GetPointerSize() const;
Andreas Gampe542451c2016-07-26 09:02:02 -0700161
162 uint32_t GetPointerSizeUnchecked() const {
Mathieu Chartiere401d142015-04-22 13:56:20 -0700163 return pointer_size_;
164 }
165
Alex Light53cb16b2014-06-12 11:26:29 -0700166 off_t GetPatchDelta() const {
167 return patch_delta_;
168 }
169
Nicolas Geoffray9583fbc2014-02-28 15:21:07 +0000170 static std::string GetOatLocationFromImageLocation(const std::string& image) {
David Brazdil7b49e6c2016-09-01 11:06:18 +0100171 return GetLocationFromImageLocation(image, "oat");
172 }
173
174 static std::string GetVdexLocationFromImageLocation(const std::string& image) {
175 return GetLocationFromImageLocation(image, "vdex");
Nicolas Geoffray9583fbc2014-02-28 15:21:07 +0000176 }
177
Mathieu Chartiere401d142015-04-22 13:56:20 -0700178 enum ImageMethod {
Ian Rogers19846512012-02-24 11:42:47 -0800179 kResolutionMethod,
Jeff Hao88474b42013-10-23 16:24:40 -0700180 kImtConflictMethod,
Mathieu Chartier2d2621a2014-10-23 16:48:06 -0700181 kImtUnimplementedMethod,
Vladimir Markofd36f1f2016-08-03 18:49:58 +0100182 kSaveAllCalleeSavesMethod,
183 kSaveRefsOnlyMethod,
184 kSaveRefsAndArgsMethod,
Vladimir Marko952dbb12016-07-28 12:01:51 +0100185 kSaveEverythingMethod,
Mathieu Chartiere401d142015-04-22 13:56:20 -0700186 kImageMethodsCount, // Number of elements in enum.
187 };
188
189 enum ImageRoot {
Brian Carlstrom58ae9412011-10-04 00:56:06 -0700190 kDexCaches,
Brian Carlstrom34f426c2011-10-04 12:58:02 -0700191 kClassRoots,
Vladimir Markoeca3eda2016-11-09 16:26:44 +0000192 kClassLoader, // App image only.
Brian Carlstrom16192862011-09-12 17:50:06 -0700193 kImageRootsMax,
194 };
195
Mathieu Chartiere401d142015-04-22 13:56:20 -0700196 enum ImageSections {
197 kSectionObjects,
198 kSectionArtFields,
199 kSectionArtMethods,
Mathieu Chartiere42888f2016-04-14 10:49:19 -0700200 kSectionRuntimeMethods,
Artem Udovichenkoa62cb9b2016-06-30 09:18:25 +0000201 kSectionImTables,
Mathieu Chartiere42888f2016-04-14 10:49:19 -0700202 kSectionIMTConflictTables,
Vladimir Marko05792b92015-08-03 11:56:49 +0100203 kSectionDexCacheArrays,
Mathieu Chartierd39645e2015-06-09 17:50:29 -0700204 kSectionInternedStrings,
Mathieu Chartier208a5cb2015-12-02 15:44:07 -0800205 kSectionClassTable,
Mathieu Chartiere401d142015-04-22 13:56:20 -0700206 kSectionImageBitmap,
207 kSectionCount, // Number of elements in enum.
208 };
209
Vladimir Markoeca3eda2016-11-09 16:26:44 +0000210 static size_t NumberOfImageRoots(bool app_image) {
211 return app_image ? kImageRootsMax : kImageRootsMax - 1u;
212 }
213
Mathieu Chartiere401d142015-04-22 13:56:20 -0700214 ArtMethod* GetImageMethod(ImageMethod index) const;
215 void SetImageMethod(ImageMethod index, ArtMethod* method);
216
217 const ImageSection& GetImageSection(ImageSections index) const;
Mathieu Chartiere42888f2016-04-14 10:49:19 -0700218
Mathieu Chartiere401d142015-04-22 13:56:20 -0700219 const ImageSection& GetMethodsSection() const {
220 return GetImageSection(kSectionArtMethods);
221 }
222
Mathieu Chartiere42888f2016-04-14 10:49:19 -0700223 const ImageSection& GetRuntimeMethodsSection() const {
224 return GetImageSection(kSectionRuntimeMethods);
225 }
226
227 const ImageSection& GetFieldsSection() const {
228 return GetImageSection(ImageHeader::kSectionArtFields);
229 }
230
Mathieu Chartier4a26f172016-01-26 14:26:18 -0800231 template <ReadBarrierOption kReadBarrierOption = kWithReadBarrier>
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800232 mirror::Object* GetImageRoot(ImageRoot image_root) const
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700233 REQUIRES_SHARED(Locks::mutator_lock_);
Mathieu Chartier4a26f172016-01-26 14:26:18 -0800234
235 template <ReadBarrierOption kReadBarrierOption = kWithReadBarrier>
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -0800236 mirror::ObjectArray<mirror::Object>* GetImageRoots() const
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700237 REQUIRES_SHARED(Locks::mutator_lock_);
Brian Carlstromaded5f72011-10-07 17:15:04 -0700238
Alex Light53cb16b2014-06-12 11:26:29 -0700239 void RelocateImage(off_t delta);
Mathieu Chartierfbc31082016-01-24 11:59:56 -0800240 void RelocateImageMethods(off_t delta);
241 void RelocateImageObjects(off_t delta);
Alex Light53cb16b2014-06-12 11:26:29 -0700242
Igor Murashkin46774762014-10-22 11:37:02 -0700243 bool CompilePic() const {
244 return compile_pic_ != 0;
245 }
246
Mathieu Chartierfbc31082016-01-24 11:59:56 -0800247 bool IsPic() const {
248 return is_pic_ != 0;
249 }
250
251 uint32_t GetBootImageBegin() const {
252 return boot_image_begin_;
253 }
254
255 uint32_t GetBootImageSize() const {
256 return boot_image_size_;
257 }
258
259 uint32_t GetBootOatBegin() const {
260 return boot_oat_begin_;
261 }
262
263 uint32_t GetBootOatSize() const {
264 return boot_oat_size_;
265 }
266
Mathieu Chartierceb07b32015-12-10 09:33:21 -0800267 StorageMode GetStorageMode() const {
268 return storage_mode_;
269 }
270
271 uint64_t GetDataSize() const {
272 return data_size_;
273 }
274
Mathieu Chartierfbc31082016-01-24 11:59:56 -0800275 bool IsAppImage() const {
276 // App images currently require a boot image, if the size is non zero then it is an app image
277 // header.
278 return boot_image_size_ != 0u;
279 }
280
Mathieu Chartiere42888f2016-04-14 10:49:19 -0700281 // Visit ArtMethods in the section starting at base. Includes runtime methods.
282 // TODO: Delete base parameter if it is always equal to GetImageBegin.
Andreas Gampe542451c2016-07-26 09:02:02 -0700283 void VisitPackedArtMethods(ArtMethodVisitor* visitor,
284 uint8_t* base,
285 PointerSize pointer_size) const;
Mathieu Chartiere42888f2016-04-14 10:49:19 -0700286
287 // Visit ArtMethods in the section starting at base.
288 // TODO: Delete base parameter if it is always equal to GetImageBegin.
289 void VisitPackedArtFields(ArtFieldVisitor* visitor, uint8_t* base) const;
290
291 template <typename Visitor>
Artem Udovichenkoa62cb9b2016-06-30 09:18:25 +0000292 void VisitPackedImTables(const Visitor& visitor,
293 uint8_t* base,
Andreas Gampe542451c2016-07-26 09:02:02 -0700294 PointerSize pointer_size) const;
Artem Udovichenkoa62cb9b2016-06-30 09:18:25 +0000295
296 template <typename Visitor>
Mathieu Chartiere42888f2016-04-14 10:49:19 -0700297 void VisitPackedImtConflictTables(const Visitor& visitor,
298 uint8_t* base,
Andreas Gampe542451c2016-07-26 09:02:02 -0700299 PointerSize pointer_size) const;
Mathieu Chartiere42888f2016-04-14 10:49:19 -0700300
Alex Light53cb16b2014-06-12 11:26:29 -0700301 private:
Ian Rogers13735952014-10-08 12:43:28 -0700302 static const uint8_t kImageMagic[4];
303 static const uint8_t kImageVersion[4];
Brian Carlstrom4a289ed2011-08-16 17:17:49 -0700304
David Brazdil7b49e6c2016-09-01 11:06:18 +0100305 static std::string GetLocationFromImageLocation(const std::string& image,
306 const std::string& extension) {
307 std::string filename = image;
308 if (filename.length() <= 3) {
309 filename += "." + extension;
310 } else {
311 filename.replace(filename.length() - 3, 3, extension);
312 }
313 return filename;
314 }
315
Ian Rogers13735952014-10-08 12:43:28 -0700316 uint8_t magic_[4];
317 uint8_t version_[4];
Brian Carlstroma663ea52011-08-19 23:33:41 -0700318
Brian Carlstrom700c8d32012-11-05 10:42:02 -0800319 // Required base address for mapping the image.
Ian Rogers30fab402012-01-23 15:43:46 -0800320 uint32_t image_begin_;
Brian Carlstrome24fa612011-09-29 00:53:55 -0700321
Mathieu Chartier31e89252013-08-28 11:29:12 -0700322 // Image size, not page aligned.
323 uint32_t image_size_;
324
Brian Carlstrom700c8d32012-11-05 10:42:02 -0800325 // Checksum of the oat file we link to for load time sanity check.
Brian Carlstrome24fa612011-09-29 00:53:55 -0700326 uint32_t oat_checksum_;
327
Brian Carlstrom700c8d32012-11-05 10:42:02 -0800328 // Start address for oat file. Will be before oat_data_begin_ for .so files.
329 uint32_t oat_file_begin_;
Brian Carlstrome24fa612011-09-29 00:53:55 -0700330
Brian Carlstrom700c8d32012-11-05 10:42:02 -0800331 // Required oat address expected by image Method::GetCode() pointers.
332 uint32_t oat_data_begin_;
Brian Carlstrom16192862011-09-12 17:50:06 -0700333
Brian Carlstrom700c8d32012-11-05 10:42:02 -0800334 // End of oat data address range for this image file.
335 uint32_t oat_data_end_;
336
337 // End of oat file address range. will be after oat_data_end_ for
338 // .so files. Used for positioning a following alloc spaces.
339 uint32_t oat_file_end_;
340
Mathieu Chartierfbc31082016-01-24 11:59:56 -0800341 // Boot image begin and end (app image headers only).
342 uint32_t boot_image_begin_;
343 uint32_t boot_image_size_;
344
345 // Boot oat begin and end (app image headers only).
346 uint32_t boot_oat_begin_;
347 uint32_t boot_oat_size_;
348
349 // TODO: We should probably insert a boot image checksum for app images.
350
Alex Light53cb16b2014-06-12 11:26:29 -0700351 // The total delta that this image has been patched.
352 int32_t patch_delta_;
353
Brian Carlstrom700c8d32012-11-05 10:42:02 -0800354 // Absolute address of an Object[] of objects needed to reinitialize from an image.
Brian Carlstrom16192862011-09-12 17:50:06 -0700355 uint32_t image_roots_;
356
Mathieu Chartiere401d142015-04-22 13:56:20 -0700357 // Pointer size, this affects the size of the ArtMethods.
358 uint32_t pointer_size_;
359
Igor Murashkin46774762014-10-22 11:37:02 -0700360 // Boolean (0 or 1) to denote if the image was compiled with --compile-pic option
361 const uint32_t compile_pic_;
362
Mathieu Chartierfbc31082016-01-24 11:59:56 -0800363 // Boolean (0 or 1) to denote if the image can be mapped at a random address, this only refers to
364 // the .art file. Currently, app oat files do not depend on their app image. There are no pointers
365 // from the app oat code to the app image.
366 const uint32_t is_pic_;
367
Mathieu Chartier67ad20e2015-12-09 15:41:09 -0800368 // Image section sizes/offsets correspond to the uncompressed form.
Mathieu Chartiere401d142015-04-22 13:56:20 -0700369 ImageSection sections_[kSectionCount];
370
Mathieu Chartierfbc31082016-01-24 11:59:56 -0800371 // Image methods, may be inside of the boot image for app images.
Mathieu Chartiere401d142015-04-22 13:56:20 -0700372 uint64_t image_methods_[kImageMethodsCount];
373
Mathieu Chartierceb07b32015-12-10 09:33:21 -0800374 // Storage method for the image, the image may be compressed.
375 StorageMode storage_mode_;
376
377 // Data size for the image data excluding the bitmap and the header. For compressed images, this
378 // is the compressed size in the file.
379 uint32_t data_size_;
380
Brian Carlstrom16192862011-09-12 17:50:06 -0700381 friend class ImageWriter;
Brian Carlstrom4a289ed2011-08-16 17:17:49 -0700382};
383
Mathieu Chartiere401d142015-04-22 13:56:20 -0700384std::ostream& operator<<(std::ostream& os, const ImageHeader::ImageMethod& policy);
385std::ostream& operator<<(std::ostream& os, const ImageHeader::ImageRoot& policy);
386std::ostream& operator<<(std::ostream& os, const ImageHeader::ImageSections& section);
387std::ostream& operator<<(std::ostream& os, const ImageSection& section);
Mathieu Chartierceb07b32015-12-10 09:33:21 -0800388std::ostream& operator<<(std::ostream& os, const ImageHeader::StorageMode& mode);
Mathieu Chartiere401d142015-04-22 13:56:20 -0700389
Brian Carlstrom4a289ed2011-08-16 17:17:49 -0700390} // namespace art
391
Brian Carlstromfc0e3212013-07-17 14:40:12 -0700392#endif // ART_RUNTIME_IMAGE_H_