• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
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 
17 #ifndef ART_RUNTIME_IMAGE_H_
18 #define ART_RUNTIME_IMAGE_H_
19 
20 #include <string.h>
21 
22 #include "base/bit_utils.h"
23 #include "base/enums.h"
24 #include "globals.h"
25 #include "mirror/object.h"
26 
27 namespace art {
28 
29 class ArtField;
30 class ArtMethod;
31 
32 namespace linker {
33 class ImageWriter;
34 }  // namespace linker
35 
36 class ObjectVisitor {
37  public:
~ObjectVisitor()38   virtual ~ObjectVisitor() {}
39 
40   virtual void Visit(mirror::Object* object) = 0;
41 };
42 
43 class ArtMethodVisitor {
44  public:
~ArtMethodVisitor()45   virtual ~ArtMethodVisitor() {}
46 
47   virtual void Visit(ArtMethod* method) = 0;
48 };
49 
50 class ArtFieldVisitor {
51  public:
~ArtFieldVisitor()52   virtual ~ArtFieldVisitor() {}
53 
54   virtual void Visit(ArtField* method) = 0;
55 };
56 
57 class PACKED(4) ImageSection {
58  public:
ImageSection()59   ImageSection() : offset_(0), size_(0) { }
ImageSection(uint32_t offset,uint32_t size)60   ImageSection(uint32_t offset, uint32_t size) : offset_(offset), size_(size) { }
61   ImageSection(const ImageSection& section) = default;
62   ImageSection& operator=(const ImageSection& section) = default;
63 
Offset()64   uint32_t Offset() const {
65     return offset_;
66   }
67 
Size()68   uint32_t Size() const {
69     return size_;
70   }
71 
End()72   uint32_t End() const {
73     return Offset() + Size();
74   }
75 
Contains(uint64_t offset)76   bool Contains(uint64_t offset) const {
77     return offset - offset_ < size_;
78   }
79 
80  private:
81   uint32_t offset_;
82   uint32_t size_;
83 };
84 
85 // header of image files written by ImageWriter, read and validated by Space.
86 class PACKED(4) ImageHeader {
87  public:
88   enum StorageMode : uint32_t {
89     kStorageModeUncompressed,
90     kStorageModeLZ4,
91     kStorageModeLZ4HC,
92     kStorageModeCount,  // Number of elements in enum.
93   };
94   static constexpr StorageMode kDefaultStorageMode = kStorageModeUncompressed;
95 
ImageHeader()96   ImageHeader()
97       : image_begin_(0U),
98         image_size_(0U),
99         oat_checksum_(0U),
100         oat_file_begin_(0U),
101         oat_data_begin_(0U),
102         oat_data_end_(0U),
103         oat_file_end_(0U),
104         boot_image_begin_(0U),
105         boot_image_size_(0U),
106         boot_oat_begin_(0U),
107         boot_oat_size_(0U),
108         patch_delta_(0),
109         image_roots_(0U),
110         pointer_size_(0U),
111         compile_pic_(0),
112         is_pic_(0),
113         storage_mode_(kDefaultStorageMode),
114         data_size_(0) {}
115 
116   ImageHeader(uint32_t image_begin,
117               uint32_t image_size,
118               ImageSection* sections,
119               uint32_t image_roots,
120               uint32_t oat_checksum,
121               uint32_t oat_file_begin,
122               uint32_t oat_data_begin,
123               uint32_t oat_data_end,
124               uint32_t oat_file_end,
125               uint32_t boot_image_begin,
126               uint32_t boot_image_size,
127               uint32_t boot_oat_begin,
128               uint32_t boot_oat_size,
129               uint32_t pointer_size,
130               bool compile_pic,
131               bool is_pic,
132               StorageMode storage_mode,
133               size_t data_size);
134 
135   bool IsValid() const;
136   const char* GetMagic() const;
137 
GetImageBegin()138   uint8_t* GetImageBegin() const {
139     return reinterpret_cast<uint8_t*>(image_begin_);
140   }
141 
GetImageSize()142   size_t GetImageSize() const {
143     return static_cast<uint32_t>(image_size_);
144   }
145 
GetOatChecksum()146   uint32_t GetOatChecksum() const {
147     return oat_checksum_;
148   }
149 
SetOatChecksum(uint32_t oat_checksum)150   void SetOatChecksum(uint32_t oat_checksum) {
151     oat_checksum_ = oat_checksum;
152   }
153 
154   // The location that the oat file was expected to be when the image was created. The actual
155   // oat file may be at a different location for application images.
GetOatFileBegin()156   uint8_t* GetOatFileBegin() const {
157     return reinterpret_cast<uint8_t*>(oat_file_begin_);
158   }
159 
GetOatDataBegin()160   uint8_t* GetOatDataBegin() const {
161     return reinterpret_cast<uint8_t*>(oat_data_begin_);
162   }
163 
GetOatDataEnd()164   uint8_t* GetOatDataEnd() const {
165     return reinterpret_cast<uint8_t*>(oat_data_end_);
166   }
167 
GetOatFileEnd()168   uint8_t* GetOatFileEnd() const {
169     return reinterpret_cast<uint8_t*>(oat_file_end_);
170   }
171 
172   PointerSize GetPointerSize() const;
173 
GetPointerSizeUnchecked()174   uint32_t GetPointerSizeUnchecked() const {
175     return pointer_size_;
176   }
177 
GetPatchDelta()178   off_t GetPatchDelta() const {
179     return patch_delta_;
180   }
181 
SetPatchDelta(off_t patch_delta)182   void SetPatchDelta(off_t patch_delta) {
183     patch_delta_ = patch_delta;
184   }
185 
GetOatLocationFromImageLocation(const std::string & image)186   static std::string GetOatLocationFromImageLocation(const std::string& image) {
187     return GetLocationFromImageLocation(image, "oat");
188   }
189 
GetVdexLocationFromImageLocation(const std::string & image)190   static std::string GetVdexLocationFromImageLocation(const std::string& image) {
191     return GetLocationFromImageLocation(image, "vdex");
192   }
193 
194   enum ImageMethod {
195     kResolutionMethod,
196     kImtConflictMethod,
197     kImtUnimplementedMethod,
198     kSaveAllCalleeSavesMethod,
199     kSaveRefsOnlyMethod,
200     kSaveRefsAndArgsMethod,
201     kSaveEverythingMethod,
202     kSaveEverythingMethodForClinit,
203     kSaveEverythingMethodForSuspendCheck,
204     kImageMethodsCount,  // Number of elements in enum.
205   };
206 
207   enum ImageRoot {
208     kDexCaches,
209     kClassRoots,
210     kClassLoader,  // App image only.
211     kImageRootsMax,
212   };
213 
214   enum ImageSections {
215     kSectionObjects,
216     kSectionArtFields,
217     kSectionArtMethods,
218     kSectionRuntimeMethods,
219     kSectionImTables,
220     kSectionIMTConflictTables,
221     kSectionDexCacheArrays,
222     kSectionInternedStrings,
223     kSectionClassTable,
224     kSectionImageBitmap,
225     kSectionCount,  // Number of elements in enum.
226   };
227 
NumberOfImageRoots(bool app_image)228   static size_t NumberOfImageRoots(bool app_image) {
229     return app_image ? kImageRootsMax : kImageRootsMax - 1u;
230   }
231 
232   ArtMethod* GetImageMethod(ImageMethod index) const;
233   void SetImageMethod(ImageMethod index, ArtMethod* method);
234 
GetImageSection(ImageSections index)235   const ImageSection& GetImageSection(ImageSections index) const {
236     DCHECK_LT(static_cast<size_t>(index), kSectionCount);
237     return sections_[index];
238   }
239 
GetObjectsSection()240   const ImageSection& GetObjectsSection() const {
241     return GetImageSection(kSectionObjects);
242   }
243 
GetFieldsSection()244   const ImageSection& GetFieldsSection() const {
245     return GetImageSection(ImageHeader::kSectionArtFields);
246   }
247 
GetMethodsSection()248   const ImageSection& GetMethodsSection() const {
249     return GetImageSection(kSectionArtMethods);
250   }
251 
GetRuntimeMethodsSection()252   const ImageSection& GetRuntimeMethodsSection() const {
253     return GetImageSection(kSectionRuntimeMethods);
254   }
255 
GetImTablesSection()256   const ImageSection& GetImTablesSection() const {
257     return GetImageSection(kSectionImTables);
258   }
259 
GetIMTConflictTablesSection()260   const ImageSection& GetIMTConflictTablesSection() const {
261     return GetImageSection(kSectionIMTConflictTables);
262   }
263 
GetDexCacheArraysSection()264   const ImageSection& GetDexCacheArraysSection() const {
265     return GetImageSection(kSectionDexCacheArrays);
266   }
267 
GetInternedStringsSection()268   const ImageSection& GetInternedStringsSection() const {
269     return GetImageSection(kSectionInternedStrings);
270   }
271 
GetClassTableSection()272   const ImageSection& GetClassTableSection() const {
273     return GetImageSection(kSectionClassTable);
274   }
275 
GetImageBitmapSection()276   const ImageSection& GetImageBitmapSection() const {
277     return GetImageSection(kSectionImageBitmap);
278   }
279 
280   template <ReadBarrierOption kReadBarrierOption = kWithReadBarrier>
281   mirror::Object* GetImageRoot(ImageRoot image_root) const
282       REQUIRES_SHARED(Locks::mutator_lock_);
283 
284   template <ReadBarrierOption kReadBarrierOption = kWithReadBarrier>
285   mirror::ObjectArray<mirror::Object>* GetImageRoots() const
286       REQUIRES_SHARED(Locks::mutator_lock_);
287 
288   void RelocateImage(off_t delta);
289   void RelocateImageMethods(off_t delta);
290   void RelocateImageObjects(off_t delta);
291 
CompilePic()292   bool CompilePic() const {
293     return compile_pic_ != 0;
294   }
295 
IsPic()296   bool IsPic() const {
297     return is_pic_ != 0;
298   }
299 
GetBootImageBegin()300   uint32_t GetBootImageBegin() const {
301     return boot_image_begin_;
302   }
303 
GetBootImageSize()304   uint32_t GetBootImageSize() const {
305     return boot_image_size_;
306   }
307 
GetBootOatBegin()308   uint32_t GetBootOatBegin() const {
309     return boot_oat_begin_;
310   }
311 
GetBootOatSize()312   uint32_t GetBootOatSize() const {
313     return boot_oat_size_;
314   }
315 
GetStorageMode()316   StorageMode GetStorageMode() const {
317     return storage_mode_;
318   }
319 
GetDataSize()320   uint64_t GetDataSize() const {
321     return data_size_;
322   }
323 
IsAppImage()324   bool IsAppImage() const {
325     // App images currently require a boot image, if the size is non zero then it is an app image
326     // header.
327     return boot_image_size_ != 0u;
328   }
329 
GetBootImageConstantTablesOffset()330   uint32_t GetBootImageConstantTablesOffset() const {
331     // Interned strings table and class table for boot image are mmapped read only.
332     DCHECK(!IsAppImage());
333     const ImageSection& interned_strings = GetInternedStringsSection();
334     DCHECK_ALIGNED(interned_strings.Offset(), kPageSize);
335     return interned_strings.Offset();
336   }
337 
GetBootImageConstantTablesSize()338   uint32_t GetBootImageConstantTablesSize() const {
339     uint32_t start_offset = GetBootImageConstantTablesOffset();
340     const ImageSection& class_table = GetClassTableSection();
341     DCHECK_LE(start_offset, class_table.Offset());
342     size_t tables_size = class_table.Offset() + class_table.Size() - start_offset;
343     return RoundUp(tables_size, kPageSize);
344   }
345 
346   // Visit mirror::Objects in the section starting at base.
347   // TODO: Delete base parameter if it is always equal to GetImageBegin.
348   void VisitObjects(ObjectVisitor* visitor,
349                     uint8_t* base,
350                     PointerSize pointer_size) const
351       REQUIRES_SHARED(Locks::mutator_lock_);
352 
353   // Visit ArtMethods in the section starting at base. Includes runtime methods.
354   // TODO: Delete base parameter if it is always equal to GetImageBegin.
355   void VisitPackedArtMethods(ArtMethodVisitor* visitor,
356                              uint8_t* base,
357                              PointerSize pointer_size) const;
358 
359   // Visit ArtMethods in the section starting at base.
360   // TODO: Delete base parameter if it is always equal to GetImageBegin.
361   void VisitPackedArtFields(ArtFieldVisitor* visitor, uint8_t* base) const;
362 
363   template <typename Visitor>
364   void VisitPackedImTables(const Visitor& visitor,
365                            uint8_t* base,
366                            PointerSize pointer_size) const;
367 
368   template <typename Visitor>
369   void VisitPackedImtConflictTables(const Visitor& visitor,
370                                     uint8_t* base,
371                                     PointerSize pointer_size) const;
372 
373  private:
374   static const uint8_t kImageMagic[4];
375   static const uint8_t kImageVersion[4];
376 
GetLocationFromImageLocation(const std::string & image,const std::string & extension)377   static std::string GetLocationFromImageLocation(const std::string& image,
378                                                   const std::string& extension) {
379     std::string filename = image;
380     if (filename.length() <= 3) {
381       filename += "." + extension;
382     } else {
383       filename.replace(filename.length() - 3, 3, extension);
384     }
385     return filename;
386   }
387 
388   uint8_t magic_[4];
389   uint8_t version_[4];
390 
391   // Required base address for mapping the image.
392   uint32_t image_begin_;
393 
394   // Image size, not page aligned.
395   uint32_t image_size_;
396 
397   // Checksum of the oat file we link to for load time sanity check.
398   uint32_t oat_checksum_;
399 
400   // Start address for oat file. Will be before oat_data_begin_ for .so files.
401   uint32_t oat_file_begin_;
402 
403   // Required oat address expected by image Method::GetCode() pointers.
404   uint32_t oat_data_begin_;
405 
406   // End of oat data address range for this image file.
407   uint32_t oat_data_end_;
408 
409   // End of oat file address range. will be after oat_data_end_ for
410   // .so files. Used for positioning a following alloc spaces.
411   uint32_t oat_file_end_;
412 
413   // Boot image begin and end (app image headers only).
414   uint32_t boot_image_begin_;
415   uint32_t boot_image_size_;
416 
417   // Boot oat begin and end (app image headers only).
418   uint32_t boot_oat_begin_;
419   uint32_t boot_oat_size_;
420 
421   // TODO: We should probably insert a boot image checksum for app images.
422 
423   // The total delta that this image has been patched.
424   int32_t patch_delta_;
425 
426   // Absolute address of an Object[] of objects needed to reinitialize from an image.
427   uint32_t image_roots_;
428 
429   // Pointer size, this affects the size of the ArtMethods.
430   uint32_t pointer_size_;
431 
432   // Boolean (0 or 1) to denote if the image was compiled with --compile-pic option
433   const uint32_t compile_pic_;
434 
435   // Boolean (0 or 1) to denote if the image can be mapped at a random address, this only refers to
436   // the .art file. Currently, app oat files do not depend on their app image. There are no pointers
437   // from the app oat code to the app image.
438   const uint32_t is_pic_;
439 
440   // Image section sizes/offsets correspond to the uncompressed form.
441   ImageSection sections_[kSectionCount];
442 
443   // Image methods, may be inside of the boot image for app images.
444   uint64_t image_methods_[kImageMethodsCount];
445 
446   // Storage method for the image, the image may be compressed.
447   StorageMode storage_mode_;
448 
449   // Data size for the image data excluding the bitmap and the header. For compressed images, this
450   // is the compressed size in the file.
451   uint32_t data_size_;
452 
453   friend class linker::ImageWriter;
454 };
455 
456 std::ostream& operator<<(std::ostream& os, const ImageHeader::ImageMethod& policy);
457 std::ostream& operator<<(std::ostream& os, const ImageHeader::ImageRoot& policy);
458 std::ostream& operator<<(std::ostream& os, const ImageHeader::ImageSections& section);
459 std::ostream& operator<<(std::ostream& os, const ImageSection& section);
460 std::ostream& operator<<(std::ostream& os, const ImageHeader::StorageMode& mode);
461 
462 }  // namespace art
463 
464 #endif  // ART_RUNTIME_IMAGE_H_
465