Elliott Hughes | 2faa5f1 | 2012-01-30 14:42:07 -0800 | [diff] [blame] | 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 | */ |
Brian Carlstrom | e24fa61 | 2011-09-29 00:53:55 -0700 | [diff] [blame] | 16 | |
| 17 | #include "oat.h" |
Ian Rogers | 2dd0e2c | 2013-01-24 12:42:14 -0800 | [diff] [blame] | 18 | #include "utils.h" |
Brian Carlstrom | e24fa61 | 2011-09-29 00:53:55 -0700 | [diff] [blame] | 19 | |
| 20 | #include <zlib.h> |
| 21 | |
| 22 | namespace art { |
| 23 | |
| 24 | const uint8_t OatHeader::kOatMagic[] = { 'o', 'a', 't', '\n' }; |
Jeff Hao | 0aba0ba | 2013-06-03 14:49:28 -0700 | [diff] [blame] | 25 | const uint8_t OatHeader::kOatVersion[] = { '0', '0', '6', '\0' }; |
Brian Carlstrom | e24fa61 | 2011-09-29 00:53:55 -0700 | [diff] [blame] | 26 | |
Elliott Hughes | a72ec82 | 2012-03-05 17:12:22 -0800 | [diff] [blame] | 27 | OatHeader::OatHeader() { |
| 28 | memset(this, 0, sizeof(*this)); |
| 29 | } |
| 30 | |
Brian Carlstrom | 81f3ca1 | 2012-03-17 00:27:35 -0700 | [diff] [blame] | 31 | OatHeader::OatHeader(InstructionSet instruction_set, |
| 32 | const std::vector<const DexFile*>* dex_files, |
Brian Carlstrom | 28db012 | 2012-10-18 16:20:41 -0700 | [diff] [blame] | 33 | uint32_t image_file_location_oat_checksum, |
Brian Carlstrom | 700c8d3 | 2012-11-05 10:42:02 -0800 | [diff] [blame] | 34 | uint32_t image_file_location_oat_data_begin, |
Brian Carlstrom | 81f3ca1 | 2012-03-17 00:27:35 -0700 | [diff] [blame] | 35 | const std::string& image_file_location) { |
Brian Carlstrom | e24fa61 | 2011-09-29 00:53:55 -0700 | [diff] [blame] | 36 | memcpy(magic_, kOatMagic, sizeof(kOatMagic)); |
| 37 | memcpy(version_, kOatVersion, sizeof(kOatVersion)); |
Brian Carlstrom | 81f3ca1 | 2012-03-17 00:27:35 -0700 | [diff] [blame] | 38 | |
Brian Carlstrom | e24fa61 | 2011-09-29 00:53:55 -0700 | [diff] [blame] | 39 | adler32_checksum_ = adler32(0L, Z_NULL, 0); |
Brian Carlstrom | 81f3ca1 | 2012-03-17 00:27:35 -0700 | [diff] [blame] | 40 | |
Brian Carlstrom | f852fb2 | 2012-10-19 11:01:58 -0700 | [diff] [blame] | 41 | CHECK_NE(instruction_set, kNone); |
Elliott Hughes | a72ec82 | 2012-03-05 17:12:22 -0800 | [diff] [blame] | 42 | instruction_set_ = instruction_set; |
| 43 | UpdateChecksum(&instruction_set_, sizeof(instruction_set_)); |
Brian Carlstrom | 81f3ca1 | 2012-03-17 00:27:35 -0700 | [diff] [blame] | 44 | |
Brian Carlstrom | e24fa61 | 2011-09-29 00:53:55 -0700 | [diff] [blame] | 45 | dex_file_count_ = dex_files->size(); |
| 46 | UpdateChecksum(&dex_file_count_, sizeof(dex_file_count_)); |
Brian Carlstrom | 81f3ca1 | 2012-03-17 00:27:35 -0700 | [diff] [blame] | 47 | |
Brian Carlstrom | 28db012 | 2012-10-18 16:20:41 -0700 | [diff] [blame] | 48 | image_file_location_oat_checksum_ = image_file_location_oat_checksum; |
| 49 | UpdateChecksum(&image_file_location_oat_checksum_, sizeof(image_file_location_oat_checksum_)); |
| 50 | |
Brian Carlstrom | 700c8d3 | 2012-11-05 10:42:02 -0800 | [diff] [blame] | 51 | CHECK(IsAligned<kPageSize>(image_file_location_oat_data_begin)); |
| 52 | image_file_location_oat_data_begin_ = image_file_location_oat_data_begin; |
| 53 | UpdateChecksum(&image_file_location_oat_data_begin_, sizeof(image_file_location_oat_data_begin_)); |
Brian Carlstrom | 81f3ca1 | 2012-03-17 00:27:35 -0700 | [diff] [blame] | 54 | |
| 55 | image_file_location_size_ = image_file_location.size(); |
| 56 | UpdateChecksum(&image_file_location_size_, sizeof(image_file_location_size_)); |
| 57 | UpdateChecksum(image_file_location.data(), image_file_location_size_); |
| 58 | |
Brian Carlstrom | e24fa61 | 2011-09-29 00:53:55 -0700 | [diff] [blame] | 59 | executable_offset_ = 0; |
Jeff Hao | 0aba0ba | 2013-06-03 14:49:28 -0700 | [diff] [blame] | 60 | interpreter_to_interpreter_entry_offset_ = 0; |
| 61 | interpreter_to_quick_entry_offset_ = 0; |
| 62 | portable_resolution_trampoline_offset_ = 0; |
| 63 | quick_resolution_trampoline_offset_ = 0; |
Brian Carlstrom | e24fa61 | 2011-09-29 00:53:55 -0700 | [diff] [blame] | 64 | } |
| 65 | |
| 66 | bool OatHeader::IsValid() const { |
Brian Carlstrom | f852fb2 | 2012-10-19 11:01:58 -0700 | [diff] [blame] | 67 | if (memcmp(magic_, kOatMagic, sizeof(kOatMagic)) != 0) { |
Brian Carlstrom | e24fa61 | 2011-09-29 00:53:55 -0700 | [diff] [blame] | 68 | return false; |
| 69 | } |
Brian Carlstrom | f852fb2 | 2012-10-19 11:01:58 -0700 | [diff] [blame] | 70 | if (memcmp(version_, kOatVersion, sizeof(kOatVersion)) != 0) { |
Brian Carlstrom | e24fa61 | 2011-09-29 00:53:55 -0700 | [diff] [blame] | 71 | return false; |
| 72 | } |
| 73 | return true; |
| 74 | } |
| 75 | |
| 76 | const char* OatHeader::GetMagic() const { |
| 77 | CHECK(IsValid()); |
| 78 | return reinterpret_cast<const char*>(magic_); |
| 79 | } |
| 80 | |
Brian Carlstrom | e24fa61 | 2011-09-29 00:53:55 -0700 | [diff] [blame] | 81 | uint32_t OatHeader::GetChecksum() const { |
| 82 | CHECK(IsValid()); |
| 83 | return adler32_checksum_; |
| 84 | } |
| 85 | |
| 86 | void OatHeader::UpdateChecksum(const void* data, size_t length) { |
| 87 | DCHECK(IsValid()); |
| 88 | const uint8_t* bytes = reinterpret_cast<const uint8_t*>(data); |
| 89 | adler32_checksum_ = adler32(adler32_checksum_, bytes, length); |
| 90 | } |
| 91 | |
Elliott Hughes | a72ec82 | 2012-03-05 17:12:22 -0800 | [diff] [blame] | 92 | InstructionSet OatHeader::GetInstructionSet() const { |
| 93 | CHECK(IsValid()); |
| 94 | return instruction_set_; |
| 95 | } |
| 96 | |
Brian Carlstrom | e24fa61 | 2011-09-29 00:53:55 -0700 | [diff] [blame] | 97 | uint32_t OatHeader::GetExecutableOffset() const { |
| 98 | DCHECK(IsValid()); |
Elliott Hughes | 06b37d9 | 2011-10-16 11:51:29 -0700 | [diff] [blame] | 99 | DCHECK_ALIGNED(executable_offset_, kPageSize); |
Brian Carlstrom | e24fa61 | 2011-09-29 00:53:55 -0700 | [diff] [blame] | 100 | CHECK_GT(executable_offset_, sizeof(OatHeader)); |
| 101 | return executable_offset_; |
| 102 | } |
| 103 | |
Jeff Hao | 0aba0ba | 2013-06-03 14:49:28 -0700 | [diff] [blame] | 104 | void OatHeader::SetExecutableOffset(uint32_t executable_offset) { |
| 105 | DCHECK_ALIGNED(executable_offset, kPageSize); |
| 106 | CHECK_GT(executable_offset, sizeof(OatHeader)); |
| 107 | DCHECK(IsValid()); |
| 108 | DCHECK_EQ(executable_offset_, 0U); |
| 109 | |
| 110 | executable_offset_ = executable_offset; |
| 111 | UpdateChecksum(&executable_offset_, sizeof(executable_offset)); |
| 112 | } |
| 113 | |
| 114 | const void* OatHeader::GetInterpreterToInterpreterEntry() const { |
| 115 | return reinterpret_cast<const uint8_t*>(this) + GetInterpreterToInterpreterEntryOffset(); |
| 116 | } |
| 117 | |
| 118 | uint32_t OatHeader::GetInterpreterToInterpreterEntryOffset() const { |
| 119 | DCHECK(IsValid()); |
| 120 | CHECK_GE(interpreter_to_interpreter_entry_offset_, executable_offset_); |
| 121 | return interpreter_to_interpreter_entry_offset_; |
| 122 | } |
| 123 | |
| 124 | void OatHeader::SetInterpreterToInterpreterEntryOffset(uint32_t offset) { |
| 125 | CHECK(offset == 0 || offset >= executable_offset_); |
| 126 | DCHECK(IsValid()); |
| 127 | DCHECK_EQ(interpreter_to_interpreter_entry_offset_, 0U) << offset; |
| 128 | |
| 129 | interpreter_to_interpreter_entry_offset_ = offset; |
| 130 | UpdateChecksum(&interpreter_to_interpreter_entry_offset_, sizeof(offset)); |
| 131 | } |
| 132 | |
| 133 | const void* OatHeader::GetInterpreterToQuickEntry() const { |
| 134 | return reinterpret_cast<const uint8_t*>(this) + GetInterpreterToQuickEntryOffset(); |
| 135 | } |
| 136 | |
| 137 | uint32_t OatHeader::GetInterpreterToQuickEntryOffset() const { |
| 138 | DCHECK(IsValid()); |
| 139 | CHECK_GE(interpreter_to_quick_entry_offset_, interpreter_to_interpreter_entry_offset_); |
| 140 | return interpreter_to_quick_entry_offset_; |
| 141 | } |
| 142 | |
| 143 | void OatHeader::SetInterpreterToQuickEntryOffset(uint32_t offset) { |
| 144 | CHECK(offset == 0 || offset >= interpreter_to_interpreter_entry_offset_); |
| 145 | DCHECK(IsValid()); |
| 146 | DCHECK_EQ(interpreter_to_quick_entry_offset_, 0U) << offset; |
| 147 | |
| 148 | interpreter_to_quick_entry_offset_ = offset; |
| 149 | UpdateChecksum(&interpreter_to_quick_entry_offset_, sizeof(offset)); |
| 150 | } |
| 151 | |
| 152 | const void* OatHeader::GetPortableResolutionTrampoline() const { |
| 153 | return reinterpret_cast<const uint8_t*>(this) + GetPortableResolutionTrampolineOffset(); |
| 154 | } |
| 155 | |
| 156 | uint32_t OatHeader::GetPortableResolutionTrampolineOffset() const { |
| 157 | DCHECK(IsValid()); |
| 158 | CHECK_GE(portable_resolution_trampoline_offset_, interpreter_to_quick_entry_offset_); |
| 159 | return portable_resolution_trampoline_offset_; |
| 160 | } |
| 161 | |
| 162 | void OatHeader::SetPortableResolutionTrampolineOffset(uint32_t offset) { |
| 163 | CHECK(offset == 0 || offset >= interpreter_to_quick_entry_offset_); |
| 164 | DCHECK(IsValid()); |
| 165 | DCHECK_EQ(portable_resolution_trampoline_offset_, 0U) << offset; |
| 166 | |
| 167 | portable_resolution_trampoline_offset_ = offset; |
| 168 | UpdateChecksum(&portable_resolution_trampoline_offset_, sizeof(offset)); |
| 169 | } |
| 170 | |
| 171 | const void* OatHeader::GetQuickResolutionTrampoline() const { |
| 172 | return reinterpret_cast<const uint8_t*>(this) + GetQuickResolutionTrampolineOffset(); |
| 173 | } |
| 174 | |
| 175 | uint32_t OatHeader::GetQuickResolutionTrampolineOffset() const { |
| 176 | DCHECK(IsValid()); |
| 177 | CHECK_GE(quick_resolution_trampoline_offset_, portable_resolution_trampoline_offset_); |
| 178 | return quick_resolution_trampoline_offset_; |
| 179 | } |
| 180 | |
| 181 | void OatHeader::SetQuickResolutionTrampolineOffset(uint32_t offset) { |
| 182 | CHECK(offset == 0 || offset >= portable_resolution_trampoline_offset_); |
| 183 | DCHECK(IsValid()); |
| 184 | DCHECK_EQ(quick_resolution_trampoline_offset_, 0U) << offset; |
| 185 | |
| 186 | quick_resolution_trampoline_offset_ = offset; |
| 187 | UpdateChecksum(&quick_resolution_trampoline_offset_, sizeof(offset)); |
| 188 | } |
| 189 | |
Brian Carlstrom | 28db012 | 2012-10-18 16:20:41 -0700 | [diff] [blame] | 190 | uint32_t OatHeader::GetImageFileLocationOatChecksum() const { |
Brian Carlstrom | 81f3ca1 | 2012-03-17 00:27:35 -0700 | [diff] [blame] | 191 | CHECK(IsValid()); |
Brian Carlstrom | 28db012 | 2012-10-18 16:20:41 -0700 | [diff] [blame] | 192 | return image_file_location_oat_checksum_; |
| 193 | } |
| 194 | |
Brian Carlstrom | 700c8d3 | 2012-11-05 10:42:02 -0800 | [diff] [blame] | 195 | uint32_t OatHeader::GetImageFileLocationOatDataBegin() const { |
Brian Carlstrom | 28db012 | 2012-10-18 16:20:41 -0700 | [diff] [blame] | 196 | CHECK(IsValid()); |
Brian Carlstrom | 700c8d3 | 2012-11-05 10:42:02 -0800 | [diff] [blame] | 197 | return image_file_location_oat_data_begin_; |
Brian Carlstrom | 81f3ca1 | 2012-03-17 00:27:35 -0700 | [diff] [blame] | 198 | } |
| 199 | |
| 200 | uint32_t OatHeader::GetImageFileLocationSize() const { |
| 201 | CHECK(IsValid()); |
| 202 | return image_file_location_size_; |
| 203 | } |
| 204 | |
| 205 | const uint8_t* OatHeader::GetImageFileLocationData() const { |
| 206 | CHECK(IsValid()); |
| 207 | return image_file_location_data_; |
| 208 | } |
| 209 | |
| 210 | std::string OatHeader::GetImageFileLocation() const { |
| 211 | CHECK(IsValid()); |
| 212 | return std::string(reinterpret_cast<const char*>(GetImageFileLocationData()), |
| 213 | GetImageFileLocationSize()); |
| 214 | } |
| 215 | |
Brian Carlstrom | 3320cf4 | 2011-10-04 14:58:28 -0700 | [diff] [blame] | 216 | OatMethodOffsets::OatMethodOffsets() |
Elliott Hughes | 362f9bc | 2011-10-17 18:56:41 -0700 | [diff] [blame] | 217 | : code_offset_(0), |
| 218 | frame_size_in_bytes_(0), |
Elliott Hughes | 362f9bc | 2011-10-17 18:56:41 -0700 | [diff] [blame] | 219 | core_spill_mask_(0), |
| 220 | fp_spill_mask_(0), |
| 221 | mapping_table_offset_(0), |
| 222 | vmap_table_offset_(0), |
Jeff Hao | 74180ca | 2013-03-27 15:29:11 -0700 | [diff] [blame] | 223 | gc_map_offset_(0) |
Logan Chien | ccb7bf1 | 2012-03-28 12:52:32 +0800 | [diff] [blame] | 224 | {} |
Brian Carlstrom | 3320cf4 | 2011-10-04 14:58:28 -0700 | [diff] [blame] | 225 | |
| 226 | OatMethodOffsets::OatMethodOffsets(uint32_t code_offset, |
| 227 | uint32_t frame_size_in_bytes, |
Brian Carlstrom | 3320cf4 | 2011-10-04 14:58:28 -0700 | [diff] [blame] | 228 | uint32_t core_spill_mask, |
| 229 | uint32_t fp_spill_mask, |
| 230 | uint32_t mapping_table_offset, |
| 231 | uint32_t vmap_table_offset, |
Jeff Hao | 74180ca | 2013-03-27 15:29:11 -0700 | [diff] [blame] | 232 | uint32_t gc_map_offset |
Logan Chien | ccb7bf1 | 2012-03-28 12:52:32 +0800 | [diff] [blame] | 233 | ) |
Elliott Hughes | 362f9bc | 2011-10-17 18:56:41 -0700 | [diff] [blame] | 234 | : code_offset_(code_offset), |
| 235 | frame_size_in_bytes_(frame_size_in_bytes), |
Elliott Hughes | 362f9bc | 2011-10-17 18:56:41 -0700 | [diff] [blame] | 236 | core_spill_mask_(core_spill_mask), |
| 237 | fp_spill_mask_(fp_spill_mask), |
| 238 | mapping_table_offset_(mapping_table_offset), |
| 239 | vmap_table_offset_(vmap_table_offset), |
Jeff Hao | 74180ca | 2013-03-27 15:29:11 -0700 | [diff] [blame] | 240 | gc_map_offset_(gc_map_offset) |
Logan Chien | ccb7bf1 | 2012-03-28 12:52:32 +0800 | [diff] [blame] | 241 | {} |
Brian Carlstrom | 3320cf4 | 2011-10-04 14:58:28 -0700 | [diff] [blame] | 242 | |
| 243 | OatMethodOffsets::~OatMethodOffsets() {} |
| 244 | |
Brian Carlstrom | e24fa61 | 2011-09-29 00:53:55 -0700 | [diff] [blame] | 245 | } // namespace art |