1 /* 2 * Copyright (C) 2014 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_STACK_MAP_H_ 18 #define ART_RUNTIME_STACK_MAP_H_ 19 20 #include "base/bit_vector.h" 21 #include "base/bit_utils.h" 22 #include "memory_region.h" 23 #include "leb128.h" 24 25 namespace art { 26 27 class VariableIndentationOutputStream; 28 29 // Size of a frame slot, in bytes. This constant is a signed value, 30 // to please the compiler in arithmetic operations involving int32_t 31 // (signed) values. 32 static constexpr ssize_t kFrameSlotSize = 4; 33 34 // Size of Dex virtual registers. 35 static constexpr size_t kVRegSize = 4; 36 37 class CodeInfo; 38 class StackMapEncoding; 39 struct CodeInfoEncoding; 40 41 /** 42 * Classes in the following file are wrapper on stack map information backed 43 * by a MemoryRegion. As such they read and write to the region, they don't have 44 * their own fields. 45 */ 46 47 // Dex register location container used by DexRegisterMap and StackMapStream. 48 class DexRegisterLocation { 49 public: 50 /* 51 * The location kind used to populate the Dex register information in a 52 * StackMapStream can either be: 53 * - kStack: vreg stored on the stack, value holds the stack offset; 54 * - kInRegister: vreg stored in low 32 bits of a core physical register, 55 * value holds the register number; 56 * - kInRegisterHigh: vreg stored in high 32 bits of a core physical register, 57 * value holds the register number; 58 * - kInFpuRegister: vreg stored in low 32 bits of an FPU register, 59 * value holds the register number; 60 * - kInFpuRegisterHigh: vreg stored in high 32 bits of an FPU register, 61 * value holds the register number; 62 * - kConstant: value holds the constant; 63 * 64 * In addition, DexRegisterMap also uses these values: 65 * - kInStackLargeOffset: value holds a "large" stack offset (greater than 66 * or equal to 128 bytes); 67 * - kConstantLargeValue: value holds a "large" constant (lower than 0, or 68 * or greater than or equal to 32); 69 * - kNone: the register has no location, meaning it has not been set. 70 */ 71 enum class Kind : uint8_t { 72 // Short location kinds, for entries fitting on one byte (3 bits 73 // for the kind, 5 bits for the value) in a DexRegisterMap. 74 kInStack = 0, // 0b000 75 kInRegister = 1, // 0b001 76 kInRegisterHigh = 2, // 0b010 77 kInFpuRegister = 3, // 0b011 78 kInFpuRegisterHigh = 4, // 0b100 79 kConstant = 5, // 0b101 80 81 // Large location kinds, requiring a 5-byte encoding (1 byte for the 82 // kind, 4 bytes for the value). 83 84 // Stack location at a large offset, meaning that the offset value 85 // divided by the stack frame slot size (4 bytes) cannot fit on a 86 // 5-bit unsigned integer (i.e., this offset value is greater than 87 // or equal to 2^5 * 4 = 128 bytes). 88 kInStackLargeOffset = 6, // 0b110 89 90 // Large constant, that cannot fit on a 5-bit signed integer (i.e., 91 // lower than 0, or greater than or equal to 2^5 = 32). 92 kConstantLargeValue = 7, // 0b111 93 94 // Entries with no location are not stored and do not need own marker. 95 kNone = static_cast<uint8_t>(-1), 96 97 kLastLocationKind = kConstantLargeValue 98 }; 99 100 static_assert( 101 sizeof(Kind) == 1u, 102 "art::DexRegisterLocation::Kind has a size different from one byte."); 103 IsShortLocationKind(Kind kind)104 static bool IsShortLocationKind(Kind kind) { 105 switch (kind) { 106 case Kind::kInStack: 107 case Kind::kInRegister: 108 case Kind::kInRegisterHigh: 109 case Kind::kInFpuRegister: 110 case Kind::kInFpuRegisterHigh: 111 case Kind::kConstant: 112 return true; 113 114 case Kind::kInStackLargeOffset: 115 case Kind::kConstantLargeValue: 116 return false; 117 118 case Kind::kNone: 119 LOG(FATAL) << "Unexpected location kind"; 120 } 121 UNREACHABLE(); 122 } 123 124 // Convert `kind` to a "surface" kind, i.e. one that doesn't include 125 // any value with a "large" qualifier. 126 // TODO: Introduce another enum type for the surface kind? ConvertToSurfaceKind(Kind kind)127 static Kind ConvertToSurfaceKind(Kind kind) { 128 switch (kind) { 129 case Kind::kInStack: 130 case Kind::kInRegister: 131 case Kind::kInRegisterHigh: 132 case Kind::kInFpuRegister: 133 case Kind::kInFpuRegisterHigh: 134 case Kind::kConstant: 135 return kind; 136 137 case Kind::kInStackLargeOffset: 138 return Kind::kInStack; 139 140 case Kind::kConstantLargeValue: 141 return Kind::kConstant; 142 143 case Kind::kNone: 144 return kind; 145 } 146 UNREACHABLE(); 147 } 148 149 // Required by art::StackMapStream::LocationCatalogEntriesIndices. DexRegisterLocation()150 DexRegisterLocation() : kind_(Kind::kNone), value_(0) {} 151 DexRegisterLocation(Kind kind,int32_t value)152 DexRegisterLocation(Kind kind, int32_t value) : kind_(kind), value_(value) {} 153 None()154 static DexRegisterLocation None() { 155 return DexRegisterLocation(Kind::kNone, 0); 156 } 157 158 // Get the "surface" kind of the location, i.e., the one that doesn't 159 // include any value with a "large" qualifier. GetKind()160 Kind GetKind() const { 161 return ConvertToSurfaceKind(kind_); 162 } 163 164 // Get the value of the location. GetValue()165 int32_t GetValue() const { return value_; } 166 167 // Get the actual kind of the location. GetInternalKind()168 Kind GetInternalKind() const { return kind_; } 169 170 bool operator==(DexRegisterLocation other) const { 171 return kind_ == other.kind_ && value_ == other.value_; 172 } 173 174 bool operator!=(DexRegisterLocation other) const { 175 return !(*this == other); 176 } 177 178 private: 179 Kind kind_; 180 int32_t value_; 181 182 friend class DexRegisterLocationHashFn; 183 }; 184 185 std::ostream& operator<<(std::ostream& stream, const DexRegisterLocation::Kind& kind); 186 187 /** 188 * Store information on unique Dex register locations used in a method. 189 * The information is of the form: 190 * 191 * [DexRegisterLocation+]. 192 * 193 * DexRegisterLocations are either 1- or 5-byte wide (see art::DexRegisterLocation::Kind). 194 */ 195 class DexRegisterLocationCatalog { 196 public: DexRegisterLocationCatalog(MemoryRegion region)197 explicit DexRegisterLocationCatalog(MemoryRegion region) : region_(region) {} 198 199 // Short (compressed) location, fitting on one byte. 200 typedef uint8_t ShortLocation; 201 SetRegisterInfo(size_t offset,const DexRegisterLocation & dex_register_location)202 void SetRegisterInfo(size_t offset, const DexRegisterLocation& dex_register_location) { 203 DexRegisterLocation::Kind kind = ComputeCompressedKind(dex_register_location); 204 int32_t value = dex_register_location.GetValue(); 205 if (DexRegisterLocation::IsShortLocationKind(kind)) { 206 // Short location. Compress the kind and the value as a single byte. 207 if (kind == DexRegisterLocation::Kind::kInStack) { 208 // Instead of storing stack offsets expressed in bytes for 209 // short stack locations, store slot offsets. A stack offset 210 // is a multiple of 4 (kFrameSlotSize). This means that by 211 // dividing it by 4, we can fit values from the [0, 128) 212 // interval in a short stack location, and not just values 213 // from the [0, 32) interval. 214 DCHECK_EQ(value % kFrameSlotSize, 0); 215 value /= kFrameSlotSize; 216 } 217 DCHECK(IsShortValue(value)) << value; 218 region_.StoreUnaligned<ShortLocation>(offset, MakeShortLocation(kind, value)); 219 } else { 220 // Large location. Write the location on one byte and the value 221 // on 4 bytes. 222 DCHECK(!IsShortValue(value)) << value; 223 if (kind == DexRegisterLocation::Kind::kInStackLargeOffset) { 224 // Also divide large stack offsets by 4 for the sake of consistency. 225 DCHECK_EQ(value % kFrameSlotSize, 0); 226 value /= kFrameSlotSize; 227 } 228 // Data can be unaligned as the written Dex register locations can 229 // either be 1-byte or 5-byte wide. Use 230 // art::MemoryRegion::StoreUnaligned instead of 231 // art::MemoryRegion::Store to prevent unligned word accesses on ARM. 232 region_.StoreUnaligned<DexRegisterLocation::Kind>(offset, kind); 233 region_.StoreUnaligned<int32_t>(offset + sizeof(DexRegisterLocation::Kind), value); 234 } 235 } 236 237 // Find the offset of the location catalog entry number `location_catalog_entry_index`. FindLocationOffset(size_t location_catalog_entry_index)238 size_t FindLocationOffset(size_t location_catalog_entry_index) const { 239 size_t offset = kFixedSize; 240 // Skip the first `location_catalog_entry_index - 1` entries. 241 for (uint16_t i = 0; i < location_catalog_entry_index; ++i) { 242 // Read the first next byte and inspect its first 3 bits to decide 243 // whether it is a short or a large location. 244 DexRegisterLocation::Kind kind = ExtractKindAtOffset(offset); 245 if (DexRegisterLocation::IsShortLocationKind(kind)) { 246 // Short location. Skip the current byte. 247 offset += SingleShortEntrySize(); 248 } else { 249 // Large location. Skip the 5 next bytes. 250 offset += SingleLargeEntrySize(); 251 } 252 } 253 return offset; 254 } 255 256 // Get the internal kind of entry at `location_catalog_entry_index`. GetLocationInternalKind(size_t location_catalog_entry_index)257 DexRegisterLocation::Kind GetLocationInternalKind(size_t location_catalog_entry_index) const { 258 if (location_catalog_entry_index == kNoLocationEntryIndex) { 259 return DexRegisterLocation::Kind::kNone; 260 } 261 return ExtractKindAtOffset(FindLocationOffset(location_catalog_entry_index)); 262 } 263 264 // Get the (surface) kind and value of entry at `location_catalog_entry_index`. GetDexRegisterLocation(size_t location_catalog_entry_index)265 DexRegisterLocation GetDexRegisterLocation(size_t location_catalog_entry_index) const { 266 if (location_catalog_entry_index == kNoLocationEntryIndex) { 267 return DexRegisterLocation::None(); 268 } 269 size_t offset = FindLocationOffset(location_catalog_entry_index); 270 // Read the first byte and inspect its first 3 bits to get the location. 271 ShortLocation first_byte = region_.LoadUnaligned<ShortLocation>(offset); 272 DexRegisterLocation::Kind kind = ExtractKindFromShortLocation(first_byte); 273 if (DexRegisterLocation::IsShortLocationKind(kind)) { 274 // Short location. Extract the value from the remaining 5 bits. 275 int32_t value = ExtractValueFromShortLocation(first_byte); 276 if (kind == DexRegisterLocation::Kind::kInStack) { 277 // Convert the stack slot (short) offset to a byte offset value. 278 value *= kFrameSlotSize; 279 } 280 return DexRegisterLocation(kind, value); 281 } else { 282 // Large location. Read the four next bytes to get the value. 283 int32_t value = region_.LoadUnaligned<int32_t>(offset + sizeof(DexRegisterLocation::Kind)); 284 if (kind == DexRegisterLocation::Kind::kInStackLargeOffset) { 285 // Convert the stack slot (large) offset to a byte offset value. 286 value *= kFrameSlotSize; 287 } 288 return DexRegisterLocation(kind, value); 289 } 290 } 291 292 // Compute the compressed kind of `location`. ComputeCompressedKind(const DexRegisterLocation & location)293 static DexRegisterLocation::Kind ComputeCompressedKind(const DexRegisterLocation& location) { 294 DexRegisterLocation::Kind kind = location.GetInternalKind(); 295 switch (kind) { 296 case DexRegisterLocation::Kind::kInStack: 297 return IsShortStackOffsetValue(location.GetValue()) 298 ? DexRegisterLocation::Kind::kInStack 299 : DexRegisterLocation::Kind::kInStackLargeOffset; 300 301 case DexRegisterLocation::Kind::kInRegister: 302 case DexRegisterLocation::Kind::kInRegisterHigh: 303 DCHECK_GE(location.GetValue(), 0); 304 DCHECK_LT(location.GetValue(), 1 << kValueBits); 305 return kind; 306 307 case DexRegisterLocation::Kind::kInFpuRegister: 308 case DexRegisterLocation::Kind::kInFpuRegisterHigh: 309 DCHECK_GE(location.GetValue(), 0); 310 DCHECK_LT(location.GetValue(), 1 << kValueBits); 311 return kind; 312 313 case DexRegisterLocation::Kind::kConstant: 314 return IsShortConstantValue(location.GetValue()) 315 ? DexRegisterLocation::Kind::kConstant 316 : DexRegisterLocation::Kind::kConstantLargeValue; 317 318 case DexRegisterLocation::Kind::kConstantLargeValue: 319 case DexRegisterLocation::Kind::kInStackLargeOffset: 320 case DexRegisterLocation::Kind::kNone: 321 LOG(FATAL) << "Unexpected location kind " << kind; 322 } 323 UNREACHABLE(); 324 } 325 326 // Can `location` be turned into a short location? CanBeEncodedAsShortLocation(const DexRegisterLocation & location)327 static bool CanBeEncodedAsShortLocation(const DexRegisterLocation& location) { 328 DexRegisterLocation::Kind kind = location.GetInternalKind(); 329 switch (kind) { 330 case DexRegisterLocation::Kind::kInStack: 331 return IsShortStackOffsetValue(location.GetValue()); 332 333 case DexRegisterLocation::Kind::kInRegister: 334 case DexRegisterLocation::Kind::kInRegisterHigh: 335 case DexRegisterLocation::Kind::kInFpuRegister: 336 case DexRegisterLocation::Kind::kInFpuRegisterHigh: 337 return true; 338 339 case DexRegisterLocation::Kind::kConstant: 340 return IsShortConstantValue(location.GetValue()); 341 342 case DexRegisterLocation::Kind::kConstantLargeValue: 343 case DexRegisterLocation::Kind::kInStackLargeOffset: 344 case DexRegisterLocation::Kind::kNone: 345 LOG(FATAL) << "Unexpected location kind " << kind; 346 } 347 UNREACHABLE(); 348 } 349 EntrySize(const DexRegisterLocation & location)350 static size_t EntrySize(const DexRegisterLocation& location) { 351 return CanBeEncodedAsShortLocation(location) ? SingleShortEntrySize() : SingleLargeEntrySize(); 352 } 353 SingleShortEntrySize()354 static size_t SingleShortEntrySize() { 355 return sizeof(ShortLocation); 356 } 357 SingleLargeEntrySize()358 static size_t SingleLargeEntrySize() { 359 return sizeof(DexRegisterLocation::Kind) + sizeof(int32_t); 360 } 361 Size()362 size_t Size() const { 363 return region_.size(); 364 } 365 366 void Dump(VariableIndentationOutputStream* vios, const CodeInfo& code_info); 367 368 // Special (invalid) Dex register location catalog entry index meaning 369 // that there is no location for a given Dex register (i.e., it is 370 // mapped to a DexRegisterLocation::Kind::kNone location). 371 static constexpr size_t kNoLocationEntryIndex = -1; 372 373 private: 374 static constexpr int kFixedSize = 0; 375 376 // Width of the kind "field" in a short location, in bits. 377 static constexpr size_t kKindBits = 3; 378 // Width of the value "field" in a short location, in bits. 379 static constexpr size_t kValueBits = 5; 380 381 static constexpr uint8_t kKindMask = (1 << kKindBits) - 1; 382 static constexpr int32_t kValueMask = (1 << kValueBits) - 1; 383 static constexpr size_t kKindOffset = 0; 384 static constexpr size_t kValueOffset = kKindBits; 385 IsShortStackOffsetValue(int32_t value)386 static bool IsShortStackOffsetValue(int32_t value) { 387 DCHECK_EQ(value % kFrameSlotSize, 0); 388 return IsShortValue(value / kFrameSlotSize); 389 } 390 IsShortConstantValue(int32_t value)391 static bool IsShortConstantValue(int32_t value) { 392 return IsShortValue(value); 393 } 394 IsShortValue(int32_t value)395 static bool IsShortValue(int32_t value) { 396 return IsUint<kValueBits>(value); 397 } 398 MakeShortLocation(DexRegisterLocation::Kind kind,int32_t value)399 static ShortLocation MakeShortLocation(DexRegisterLocation::Kind kind, int32_t value) { 400 uint8_t kind_integer_value = static_cast<uint8_t>(kind); 401 DCHECK(IsUint<kKindBits>(kind_integer_value)) << kind_integer_value; 402 DCHECK(IsShortValue(value)) << value; 403 return (kind_integer_value & kKindMask) << kKindOffset 404 | (value & kValueMask) << kValueOffset; 405 } 406 ExtractKindFromShortLocation(ShortLocation location)407 static DexRegisterLocation::Kind ExtractKindFromShortLocation(ShortLocation location) { 408 uint8_t kind = (location >> kKindOffset) & kKindMask; 409 DCHECK_LE(kind, static_cast<uint8_t>(DexRegisterLocation::Kind::kLastLocationKind)); 410 // We do not encode kNone locations in the stack map. 411 DCHECK_NE(kind, static_cast<uint8_t>(DexRegisterLocation::Kind::kNone)); 412 return static_cast<DexRegisterLocation::Kind>(kind); 413 } 414 ExtractValueFromShortLocation(ShortLocation location)415 static int32_t ExtractValueFromShortLocation(ShortLocation location) { 416 return (location >> kValueOffset) & kValueMask; 417 } 418 419 // Extract a location kind from the byte at position `offset`. ExtractKindAtOffset(size_t offset)420 DexRegisterLocation::Kind ExtractKindAtOffset(size_t offset) const { 421 ShortLocation first_byte = region_.LoadUnaligned<ShortLocation>(offset); 422 return ExtractKindFromShortLocation(first_byte); 423 } 424 425 MemoryRegion region_; 426 427 friend class CodeInfo; 428 friend class StackMapStream; 429 }; 430 431 /* Information on Dex register locations for a specific PC, mapping a 432 * stack map's Dex register to a location entry in a DexRegisterLocationCatalog. 433 * The information is of the form: 434 * 435 * [live_bit_mask, entries*] 436 * 437 * where entries are concatenated unsigned integer values encoded on a number 438 * of bits (fixed per DexRegisterMap instances of a CodeInfo object) depending 439 * on the number of entries in the Dex register location catalog 440 * (see DexRegisterMap::SingleEntrySizeInBits). The map is 1-byte aligned. 441 */ 442 class DexRegisterMap { 443 public: DexRegisterMap(MemoryRegion region)444 explicit DexRegisterMap(MemoryRegion region) : region_(region) {} DexRegisterMap()445 DexRegisterMap() {} 446 IsValid()447 bool IsValid() const { return region_.pointer() != nullptr; } 448 449 // Get the surface kind of Dex register `dex_register_number`. GetLocationKind(uint16_t dex_register_number,uint16_t number_of_dex_registers,const CodeInfo & code_info,const CodeInfoEncoding & enc)450 DexRegisterLocation::Kind GetLocationKind(uint16_t dex_register_number, 451 uint16_t number_of_dex_registers, 452 const CodeInfo& code_info, 453 const CodeInfoEncoding& enc) const { 454 return DexRegisterLocation::ConvertToSurfaceKind( 455 GetLocationInternalKind(dex_register_number, number_of_dex_registers, code_info, enc)); 456 } 457 458 // Get the internal kind of Dex register `dex_register_number`. 459 DexRegisterLocation::Kind GetLocationInternalKind(uint16_t dex_register_number, 460 uint16_t number_of_dex_registers, 461 const CodeInfo& code_info, 462 const CodeInfoEncoding& enc) const; 463 464 // Get the Dex register location `dex_register_number`. 465 DexRegisterLocation GetDexRegisterLocation(uint16_t dex_register_number, 466 uint16_t number_of_dex_registers, 467 const CodeInfo& code_info, 468 const CodeInfoEncoding& enc) const; 469 GetStackOffsetInBytes(uint16_t dex_register_number,uint16_t number_of_dex_registers,const CodeInfo & code_info,const CodeInfoEncoding & enc)470 int32_t GetStackOffsetInBytes(uint16_t dex_register_number, 471 uint16_t number_of_dex_registers, 472 const CodeInfo& code_info, 473 const CodeInfoEncoding& enc) const { 474 DexRegisterLocation location = 475 GetDexRegisterLocation(dex_register_number, number_of_dex_registers, code_info, enc); 476 DCHECK(location.GetKind() == DexRegisterLocation::Kind::kInStack); 477 // GetDexRegisterLocation returns the offset in bytes. 478 return location.GetValue(); 479 } 480 GetConstant(uint16_t dex_register_number,uint16_t number_of_dex_registers,const CodeInfo & code_info,const CodeInfoEncoding & enc)481 int32_t GetConstant(uint16_t dex_register_number, 482 uint16_t number_of_dex_registers, 483 const CodeInfo& code_info, 484 const CodeInfoEncoding& enc) const { 485 DexRegisterLocation location = 486 GetDexRegisterLocation(dex_register_number, number_of_dex_registers, code_info, enc); 487 DCHECK_EQ(location.GetKind(), DexRegisterLocation::Kind::kConstant); 488 return location.GetValue(); 489 } 490 GetMachineRegister(uint16_t dex_register_number,uint16_t number_of_dex_registers,const CodeInfo & code_info,const CodeInfoEncoding & enc)491 int32_t GetMachineRegister(uint16_t dex_register_number, 492 uint16_t number_of_dex_registers, 493 const CodeInfo& code_info, 494 const CodeInfoEncoding& enc) const { 495 DexRegisterLocation location = 496 GetDexRegisterLocation(dex_register_number, number_of_dex_registers, code_info, enc); 497 DCHECK(location.GetInternalKind() == DexRegisterLocation::Kind::kInRegister || 498 location.GetInternalKind() == DexRegisterLocation::Kind::kInRegisterHigh || 499 location.GetInternalKind() == DexRegisterLocation::Kind::kInFpuRegister || 500 location.GetInternalKind() == DexRegisterLocation::Kind::kInFpuRegisterHigh) 501 << location.GetInternalKind(); 502 return location.GetValue(); 503 } 504 505 // Get the index of the entry in the Dex register location catalog 506 // corresponding to `dex_register_number`. GetLocationCatalogEntryIndex(uint16_t dex_register_number,uint16_t number_of_dex_registers,size_t number_of_location_catalog_entries)507 size_t GetLocationCatalogEntryIndex(uint16_t dex_register_number, 508 uint16_t number_of_dex_registers, 509 size_t number_of_location_catalog_entries) const { 510 if (!IsDexRegisterLive(dex_register_number)) { 511 return DexRegisterLocationCatalog::kNoLocationEntryIndex; 512 } 513 514 if (number_of_location_catalog_entries == 1) { 515 // We do not allocate space for location maps in the case of a 516 // single-entry location catalog, as it is useless. The only valid 517 // entry index is 0; 518 return 0; 519 } 520 521 // The bit offset of the beginning of the map locations. 522 size_t map_locations_offset_in_bits = 523 GetLocationMappingDataOffset(number_of_dex_registers) * kBitsPerByte; 524 size_t index_in_dex_register_map = GetIndexInDexRegisterMap(dex_register_number); 525 DCHECK_LT(index_in_dex_register_map, GetNumberOfLiveDexRegisters(number_of_dex_registers)); 526 // The bit size of an entry. 527 size_t map_entry_size_in_bits = SingleEntrySizeInBits(number_of_location_catalog_entries); 528 // The bit offset where `index_in_dex_register_map` is located. 529 size_t entry_offset_in_bits = 530 map_locations_offset_in_bits + index_in_dex_register_map * map_entry_size_in_bits; 531 size_t location_catalog_entry_index = 532 region_.LoadBits(entry_offset_in_bits, map_entry_size_in_bits); 533 DCHECK_LT(location_catalog_entry_index, number_of_location_catalog_entries); 534 return location_catalog_entry_index; 535 } 536 537 // Map entry at `index_in_dex_register_map` to `location_catalog_entry_index`. SetLocationCatalogEntryIndex(size_t index_in_dex_register_map,size_t location_catalog_entry_index,uint16_t number_of_dex_registers,size_t number_of_location_catalog_entries)538 void SetLocationCatalogEntryIndex(size_t index_in_dex_register_map, 539 size_t location_catalog_entry_index, 540 uint16_t number_of_dex_registers, 541 size_t number_of_location_catalog_entries) { 542 DCHECK_LT(index_in_dex_register_map, GetNumberOfLiveDexRegisters(number_of_dex_registers)); 543 DCHECK_LT(location_catalog_entry_index, number_of_location_catalog_entries); 544 545 if (number_of_location_catalog_entries == 1) { 546 // We do not allocate space for location maps in the case of a 547 // single-entry location catalog, as it is useless. 548 return; 549 } 550 551 // The bit offset of the beginning of the map locations. 552 size_t map_locations_offset_in_bits = 553 GetLocationMappingDataOffset(number_of_dex_registers) * kBitsPerByte; 554 // The bit size of an entry. 555 size_t map_entry_size_in_bits = SingleEntrySizeInBits(number_of_location_catalog_entries); 556 // The bit offset where `index_in_dex_register_map` is located. 557 size_t entry_offset_in_bits = 558 map_locations_offset_in_bits + index_in_dex_register_map * map_entry_size_in_bits; 559 region_.StoreBits(entry_offset_in_bits, location_catalog_entry_index, map_entry_size_in_bits); 560 } 561 SetLiveBitMask(uint16_t number_of_dex_registers,const BitVector & live_dex_registers_mask)562 void SetLiveBitMask(uint16_t number_of_dex_registers, 563 const BitVector& live_dex_registers_mask) { 564 size_t live_bit_mask_offset_in_bits = GetLiveBitMaskOffset() * kBitsPerByte; 565 for (uint16_t i = 0; i < number_of_dex_registers; ++i) { 566 region_.StoreBit(live_bit_mask_offset_in_bits + i, live_dex_registers_mask.IsBitSet(i)); 567 } 568 } 569 IsDexRegisterLive(uint16_t dex_register_number)570 bool IsDexRegisterLive(uint16_t dex_register_number) const { 571 size_t live_bit_mask_offset_in_bits = GetLiveBitMaskOffset() * kBitsPerByte; 572 return region_.LoadBit(live_bit_mask_offset_in_bits + dex_register_number); 573 } 574 GetNumberOfLiveDexRegisters(uint16_t number_of_dex_registers)575 size_t GetNumberOfLiveDexRegisters(uint16_t number_of_dex_registers) const { 576 size_t number_of_live_dex_registers = 0; 577 for (size_t i = 0; i < number_of_dex_registers; ++i) { 578 if (IsDexRegisterLive(i)) { 579 ++number_of_live_dex_registers; 580 } 581 } 582 return number_of_live_dex_registers; 583 } 584 GetLiveBitMaskOffset()585 static size_t GetLiveBitMaskOffset() { 586 return kFixedSize; 587 } 588 589 // Compute the size of the live register bit mask (in bytes), for a 590 // method having `number_of_dex_registers` Dex registers. GetLiveBitMaskSize(uint16_t number_of_dex_registers)591 static size_t GetLiveBitMaskSize(uint16_t number_of_dex_registers) { 592 return RoundUp(number_of_dex_registers, kBitsPerByte) / kBitsPerByte; 593 } 594 GetLocationMappingDataOffset(uint16_t number_of_dex_registers)595 static size_t GetLocationMappingDataOffset(uint16_t number_of_dex_registers) { 596 return GetLiveBitMaskOffset() + GetLiveBitMaskSize(number_of_dex_registers); 597 } 598 GetLocationMappingDataSize(uint16_t number_of_dex_registers,size_t number_of_location_catalog_entries)599 size_t GetLocationMappingDataSize(uint16_t number_of_dex_registers, 600 size_t number_of_location_catalog_entries) const { 601 size_t location_mapping_data_size_in_bits = 602 GetNumberOfLiveDexRegisters(number_of_dex_registers) 603 * SingleEntrySizeInBits(number_of_location_catalog_entries); 604 return RoundUp(location_mapping_data_size_in_bits, kBitsPerByte) / kBitsPerByte; 605 } 606 607 // Return the size of a map entry in bits. Note that if 608 // `number_of_location_catalog_entries` equals 1, this function returns 0, 609 // which is fine, as there is no need to allocate a map for a 610 // single-entry location catalog; the only valid location catalog entry index 611 // for a live register in this case is 0 and there is no need to 612 // store it. SingleEntrySizeInBits(size_t number_of_location_catalog_entries)613 static size_t SingleEntrySizeInBits(size_t number_of_location_catalog_entries) { 614 // Handle the case of 0, as we cannot pass 0 to art::WhichPowerOf2. 615 return number_of_location_catalog_entries == 0 616 ? 0u 617 : WhichPowerOf2(RoundUpToPowerOfTwo(number_of_location_catalog_entries)); 618 } 619 620 // Return the size of the DexRegisterMap object, in bytes. Size()621 size_t Size() const { 622 return region_.size(); 623 } 624 625 void Dump(VariableIndentationOutputStream* vios, 626 const CodeInfo& code_info, uint16_t number_of_dex_registers) const; 627 628 private: 629 // Return the index in the Dex register map corresponding to the Dex 630 // register number `dex_register_number`. GetIndexInDexRegisterMap(uint16_t dex_register_number)631 size_t GetIndexInDexRegisterMap(uint16_t dex_register_number) const { 632 if (!IsDexRegisterLive(dex_register_number)) { 633 return kInvalidIndexInDexRegisterMap; 634 } 635 return GetNumberOfLiveDexRegisters(dex_register_number); 636 } 637 638 // Special (invalid) Dex register map entry index meaning that there 639 // is no index in the map for a given Dex register (i.e., it must 640 // have been mapped to a DexRegisterLocation::Kind::kNone location). 641 static constexpr size_t kInvalidIndexInDexRegisterMap = -1; 642 643 static constexpr int kFixedSize = 0; 644 645 MemoryRegion region_; 646 647 friend class CodeInfo; 648 friend class StackMapStream; 649 }; 650 651 // Represents bit range of bit-packed integer field. 652 // We reuse the idea from ULEB128p1 to support encoding of -1 (aka 0xFFFFFFFF). 653 // If min_value is set to -1, we implicitly subtract one from any loaded value, 654 // and add one to any stored value. This is generalized to any negative values. 655 // In other words, min_value acts as a base and the stored value is added to it. 656 struct FieldEncoding { 657 FieldEncoding(size_t start_offset, size_t end_offset, int32_t min_value = 0) start_offset_FieldEncoding658 : start_offset_(start_offset), end_offset_(end_offset), min_value_(min_value) { 659 DCHECK_LE(start_offset_, end_offset_); 660 DCHECK_LE(BitSize(), 32u); 661 } 662 BitSizeFieldEncoding663 ALWAYS_INLINE size_t BitSize() const { return end_offset_ - start_offset_; } 664 LoadFieldEncoding665 ALWAYS_INLINE int32_t Load(const MemoryRegion& region) const { 666 DCHECK_LE(end_offset_, region.size_in_bits()); 667 const size_t bit_count = BitSize(); 668 if (bit_count == 0) { 669 // Do not touch any memory if the range is empty. 670 return min_value_; 671 } 672 uint8_t* address = region.start() + start_offset_ / kBitsPerByte; 673 const uint32_t shift = start_offset_ & (kBitsPerByte - 1); 674 // Load the value (reading only the strictly needed bytes). 675 const uint32_t load_bit_count = shift + bit_count; 676 uint32_t value = *address++ >> shift; 677 if (load_bit_count > 8) { 678 value |= static_cast<uint32_t>(*address++) << (8 - shift); 679 if (load_bit_count > 16) { 680 value |= static_cast<uint32_t>(*address++) << (16 - shift); 681 if (load_bit_count > 24) { 682 value |= static_cast<uint32_t>(*address++) << (24 - shift); 683 if (load_bit_count > 32) { 684 value |= static_cast<uint32_t>(*address++) << (32 - shift); 685 } 686 } 687 } 688 } 689 // Clear unwanted most significant bits. 690 uint32_t clear_bit_count = 32 - bit_count; 691 value = (value << clear_bit_count) >> clear_bit_count; 692 return value + min_value_; 693 } 694 StoreFieldEncoding695 ALWAYS_INLINE void Store(MemoryRegion region, int32_t value) const { 696 region.StoreBits(start_offset_, value - min_value_, BitSize()); 697 DCHECK_EQ(Load(region), value); 698 } 699 700 private: 701 size_t start_offset_; 702 size_t end_offset_; 703 int32_t min_value_; 704 }; 705 706 class StackMapEncoding { 707 public: StackMapEncoding()708 StackMapEncoding() {} 709 710 // Set stack map bit layout based on given sizes. 711 // Returns the size of stack map in bytes. SetFromSizes(size_t native_pc_max,size_t dex_pc_max,size_t dex_register_map_size,size_t inline_info_size,size_t register_mask_max,size_t stack_mask_bit_size)712 size_t SetFromSizes(size_t native_pc_max, 713 size_t dex_pc_max, 714 size_t dex_register_map_size, 715 size_t inline_info_size, 716 size_t register_mask_max, 717 size_t stack_mask_bit_size) { 718 size_t bit_offset = 0; 719 DCHECK_EQ(kNativePcBitOffset, bit_offset); 720 bit_offset += MinimumBitsToStore(native_pc_max); 721 722 dex_pc_bit_offset_ = dchecked_integral_cast<uint8_t>(bit_offset); 723 bit_offset += MinimumBitsToStore(1 /* kNoDexPc */ + dex_pc_max); 724 725 // We also need +1 for kNoDexRegisterMap, but since the size is strictly 726 // greater than any offset we might try to encode, we already implicitly have it. 727 dex_register_map_bit_offset_ = dchecked_integral_cast<uint8_t>(bit_offset); 728 bit_offset += MinimumBitsToStore(dex_register_map_size); 729 730 // We also need +1 for kNoInlineInfo, but since the inline_info_size is strictly 731 // greater than the offset we might try to encode, we already implicitly have it. 732 // If inline_info_size is zero, we can encode only kNoInlineInfo (in zero bits). 733 inline_info_bit_offset_ = dchecked_integral_cast<uint8_t>(bit_offset); 734 if (inline_info_size != 0) { 735 bit_offset += MinimumBitsToStore(dex_register_map_size + inline_info_size); 736 } 737 738 register_mask_bit_offset_ = dchecked_integral_cast<uint8_t>(bit_offset); 739 bit_offset += MinimumBitsToStore(register_mask_max); 740 741 stack_mask_bit_offset_ = dchecked_integral_cast<uint8_t>(bit_offset); 742 bit_offset += stack_mask_bit_size; 743 744 return RoundUp(bit_offset, kBitsPerByte) / kBitsPerByte; 745 } 746 GetNativePcEncoding()747 ALWAYS_INLINE FieldEncoding GetNativePcEncoding() const { 748 return FieldEncoding(kNativePcBitOffset, dex_pc_bit_offset_); 749 } GetDexPcEncoding()750 ALWAYS_INLINE FieldEncoding GetDexPcEncoding() const { 751 return FieldEncoding(dex_pc_bit_offset_, dex_register_map_bit_offset_, -1 /* min_value */); 752 } GetDexRegisterMapEncoding()753 ALWAYS_INLINE FieldEncoding GetDexRegisterMapEncoding() const { 754 return FieldEncoding(dex_register_map_bit_offset_, inline_info_bit_offset_, -1 /* min_value */); 755 } GetInlineInfoEncoding()756 ALWAYS_INLINE FieldEncoding GetInlineInfoEncoding() const { 757 return FieldEncoding(inline_info_bit_offset_, register_mask_bit_offset_, -1 /* min_value */); 758 } GetRegisterMaskEncoding()759 ALWAYS_INLINE FieldEncoding GetRegisterMaskEncoding() const { 760 return FieldEncoding(register_mask_bit_offset_, stack_mask_bit_offset_); 761 } GetStackMaskBitOffset()762 ALWAYS_INLINE size_t GetStackMaskBitOffset() const { 763 // The end offset is not encoded. It is implicitly the end of stack map entry. 764 return stack_mask_bit_offset_; 765 } 766 767 void Dump(VariableIndentationOutputStream* vios) const; 768 769 private: 770 static constexpr size_t kNativePcBitOffset = 0; 771 uint8_t dex_pc_bit_offset_; 772 uint8_t dex_register_map_bit_offset_; 773 uint8_t inline_info_bit_offset_; 774 uint8_t register_mask_bit_offset_; 775 uint8_t stack_mask_bit_offset_; 776 }; 777 778 /** 779 * A Stack Map holds compilation information for a specific PC necessary for: 780 * - Mapping it to a dex PC, 781 * - Knowing which stack entries are objects, 782 * - Knowing which registers hold objects, 783 * - Knowing the inlining information, 784 * - Knowing the values of dex registers. 785 * 786 * The information is of the form: 787 * 788 * [native_pc_offset, dex_pc, dex_register_map_offset, inlining_info_offset, register_mask, 789 * stack_mask]. 790 */ 791 class StackMap { 792 public: StackMap()793 StackMap() {} StackMap(MemoryRegion region)794 explicit StackMap(MemoryRegion region) : region_(region) {} 795 IsValid()796 ALWAYS_INLINE bool IsValid() const { return region_.pointer() != nullptr; } 797 GetDexPc(const StackMapEncoding & encoding)798 ALWAYS_INLINE uint32_t GetDexPc(const StackMapEncoding& encoding) const { 799 return encoding.GetDexPcEncoding().Load(region_); 800 } 801 SetDexPc(const StackMapEncoding & encoding,uint32_t dex_pc)802 ALWAYS_INLINE void SetDexPc(const StackMapEncoding& encoding, uint32_t dex_pc) { 803 encoding.GetDexPcEncoding().Store(region_, dex_pc); 804 } 805 GetNativePcOffset(const StackMapEncoding & encoding)806 ALWAYS_INLINE uint32_t GetNativePcOffset(const StackMapEncoding& encoding) const { 807 return encoding.GetNativePcEncoding().Load(region_); 808 } 809 SetNativePcOffset(const StackMapEncoding & encoding,uint32_t native_pc_offset)810 ALWAYS_INLINE void SetNativePcOffset(const StackMapEncoding& encoding, uint32_t native_pc_offset) { 811 encoding.GetNativePcEncoding().Store(region_, native_pc_offset); 812 } 813 GetDexRegisterMapOffset(const StackMapEncoding & encoding)814 ALWAYS_INLINE uint32_t GetDexRegisterMapOffset(const StackMapEncoding& encoding) const { 815 return encoding.GetDexRegisterMapEncoding().Load(region_); 816 } 817 SetDexRegisterMapOffset(const StackMapEncoding & encoding,uint32_t offset)818 ALWAYS_INLINE void SetDexRegisterMapOffset(const StackMapEncoding& encoding, uint32_t offset) { 819 encoding.GetDexRegisterMapEncoding().Store(region_, offset); 820 } 821 GetInlineDescriptorOffset(const StackMapEncoding & encoding)822 ALWAYS_INLINE uint32_t GetInlineDescriptorOffset(const StackMapEncoding& encoding) const { 823 return encoding.GetInlineInfoEncoding().Load(region_); 824 } 825 SetInlineDescriptorOffset(const StackMapEncoding & encoding,uint32_t offset)826 ALWAYS_INLINE void SetInlineDescriptorOffset(const StackMapEncoding& encoding, uint32_t offset) { 827 encoding.GetInlineInfoEncoding().Store(region_, offset); 828 } 829 GetRegisterMask(const StackMapEncoding & encoding)830 ALWAYS_INLINE uint32_t GetRegisterMask(const StackMapEncoding& encoding) const { 831 return encoding.GetRegisterMaskEncoding().Load(region_); 832 } 833 SetRegisterMask(const StackMapEncoding & encoding,uint32_t mask)834 ALWAYS_INLINE void SetRegisterMask(const StackMapEncoding& encoding, uint32_t mask) { 835 encoding.GetRegisterMaskEncoding().Store(region_, mask); 836 } 837 GetNumberOfStackMaskBits(const StackMapEncoding & encoding)838 ALWAYS_INLINE size_t GetNumberOfStackMaskBits(const StackMapEncoding& encoding) const { 839 return region_.size_in_bits() - encoding.GetStackMaskBitOffset(); 840 } 841 GetStackMaskBit(const StackMapEncoding & encoding,size_t index)842 ALWAYS_INLINE bool GetStackMaskBit(const StackMapEncoding& encoding, size_t index) const { 843 return region_.LoadBit(encoding.GetStackMaskBitOffset() + index); 844 } 845 SetStackMaskBit(const StackMapEncoding & encoding,size_t index,bool value)846 ALWAYS_INLINE void SetStackMaskBit(const StackMapEncoding& encoding, size_t index, bool value) { 847 region_.StoreBit(encoding.GetStackMaskBitOffset() + index, value); 848 } 849 HasDexRegisterMap(const StackMapEncoding & encoding)850 ALWAYS_INLINE bool HasDexRegisterMap(const StackMapEncoding& encoding) const { 851 return GetDexRegisterMapOffset(encoding) != kNoDexRegisterMap; 852 } 853 HasInlineInfo(const StackMapEncoding & encoding)854 ALWAYS_INLINE bool HasInlineInfo(const StackMapEncoding& encoding) const { 855 return GetInlineDescriptorOffset(encoding) != kNoInlineInfo; 856 } 857 Equals(const StackMap & other)858 ALWAYS_INLINE bool Equals(const StackMap& other) const { 859 return region_.pointer() == other.region_.pointer() && region_.size() == other.region_.size(); 860 } 861 862 void Dump(VariableIndentationOutputStream* vios, 863 const CodeInfo& code_info, 864 const CodeInfoEncoding& encoding, 865 uint32_t code_offset, 866 uint16_t number_of_dex_registers, 867 const std::string& header_suffix = "") const; 868 869 // Special (invalid) offset for the DexRegisterMapOffset field meaning 870 // that there is no Dex register map for this stack map. 871 static constexpr uint32_t kNoDexRegisterMap = -1; 872 873 // Special (invalid) offset for the InlineDescriptorOffset field meaning 874 // that there is no inline info for this stack map. 875 static constexpr uint32_t kNoInlineInfo = -1; 876 877 private: 878 static constexpr int kFixedSize = 0; 879 880 MemoryRegion region_; 881 882 friend class StackMapStream; 883 }; 884 885 class InlineInfoEncoding { 886 public: SetFromSizes(size_t method_index_max,size_t dex_pc_max,size_t invoke_type_max,size_t dex_register_map_size)887 void SetFromSizes(size_t method_index_max, 888 size_t dex_pc_max, 889 size_t invoke_type_max, 890 size_t dex_register_map_size) { 891 total_bit_size_ = kMethodIndexBitOffset; 892 total_bit_size_ += MinimumBitsToStore(method_index_max); 893 894 dex_pc_bit_offset_ = dchecked_integral_cast<uint8_t>(total_bit_size_); 895 total_bit_size_ += MinimumBitsToStore(1 /* kNoDexPc */ + dex_pc_max); 896 897 invoke_type_bit_offset_ = dchecked_integral_cast<uint8_t>(total_bit_size_); 898 total_bit_size_ += MinimumBitsToStore(invoke_type_max); 899 900 // We also need +1 for kNoDexRegisterMap, but since the size is strictly 901 // greater than any offset we might try to encode, we already implicitly have it. 902 dex_register_map_bit_offset_ = dchecked_integral_cast<uint8_t>(total_bit_size_); 903 total_bit_size_ += MinimumBitsToStore(dex_register_map_size); 904 } 905 GetMethodIndexEncoding()906 ALWAYS_INLINE FieldEncoding GetMethodIndexEncoding() const { 907 return FieldEncoding(kMethodIndexBitOffset, dex_pc_bit_offset_); 908 } GetDexPcEncoding()909 ALWAYS_INLINE FieldEncoding GetDexPcEncoding() const { 910 return FieldEncoding(dex_pc_bit_offset_, invoke_type_bit_offset_, -1 /* min_value */); 911 } GetInvokeTypeEncoding()912 ALWAYS_INLINE FieldEncoding GetInvokeTypeEncoding() const { 913 return FieldEncoding(invoke_type_bit_offset_, dex_register_map_bit_offset_); 914 } GetDexRegisterMapEncoding()915 ALWAYS_INLINE FieldEncoding GetDexRegisterMapEncoding() const { 916 return FieldEncoding(dex_register_map_bit_offset_, total_bit_size_, -1 /* min_value */); 917 } GetEntrySize()918 ALWAYS_INLINE size_t GetEntrySize() const { 919 return RoundUp(total_bit_size_, kBitsPerByte) / kBitsPerByte; 920 } 921 922 void Dump(VariableIndentationOutputStream* vios) const; 923 924 private: 925 static constexpr uint8_t kIsLastBitOffset = 0; 926 static constexpr uint8_t kMethodIndexBitOffset = 1; 927 uint8_t dex_pc_bit_offset_; 928 uint8_t invoke_type_bit_offset_; 929 uint8_t dex_register_map_bit_offset_; 930 uint8_t total_bit_size_; 931 }; 932 933 /** 934 * Inline information for a specific PC. The information is of the form: 935 * 936 * [is_last, method_index, dex_pc, invoke_type, dex_register_map_offset]+. 937 */ 938 class InlineInfo { 939 public: InlineInfo(MemoryRegion region)940 explicit InlineInfo(MemoryRegion region) : region_(region) { 941 } 942 GetDepth(const InlineInfoEncoding & encoding)943 ALWAYS_INLINE uint32_t GetDepth(const InlineInfoEncoding& encoding) const { 944 size_t depth = 0; 945 while (!GetRegionAtDepth(encoding, depth++).LoadBit(0)) { } // Check is_last bit. 946 return depth; 947 } 948 SetDepth(const InlineInfoEncoding & encoding,uint32_t depth)949 ALWAYS_INLINE void SetDepth(const InlineInfoEncoding& encoding, uint32_t depth) { 950 DCHECK_GT(depth, 0u); 951 for (size_t d = 0; d < depth; ++d) { 952 GetRegionAtDepth(encoding, d).StoreBit(0, d == depth - 1); // Set is_last bit. 953 } 954 } 955 GetMethodIndexAtDepth(const InlineInfoEncoding & encoding,uint32_t depth)956 ALWAYS_INLINE uint32_t GetMethodIndexAtDepth(const InlineInfoEncoding& encoding, 957 uint32_t depth) const { 958 return encoding.GetMethodIndexEncoding().Load(GetRegionAtDepth(encoding, depth)); 959 } 960 SetMethodIndexAtDepth(const InlineInfoEncoding & encoding,uint32_t depth,uint32_t index)961 ALWAYS_INLINE void SetMethodIndexAtDepth(const InlineInfoEncoding& encoding, 962 uint32_t depth, 963 uint32_t index) { 964 encoding.GetMethodIndexEncoding().Store(GetRegionAtDepth(encoding, depth), index); 965 } 966 GetDexPcAtDepth(const InlineInfoEncoding & encoding,uint32_t depth)967 ALWAYS_INLINE uint32_t GetDexPcAtDepth(const InlineInfoEncoding& encoding, 968 uint32_t depth) const { 969 return encoding.GetDexPcEncoding().Load(GetRegionAtDepth(encoding, depth)); 970 } 971 SetDexPcAtDepth(const InlineInfoEncoding & encoding,uint32_t depth,uint32_t dex_pc)972 ALWAYS_INLINE void SetDexPcAtDepth(const InlineInfoEncoding& encoding, 973 uint32_t depth, 974 uint32_t dex_pc) { 975 encoding.GetDexPcEncoding().Store(GetRegionAtDepth(encoding, depth), dex_pc); 976 } 977 GetInvokeTypeAtDepth(const InlineInfoEncoding & encoding,uint32_t depth)978 ALWAYS_INLINE uint32_t GetInvokeTypeAtDepth(const InlineInfoEncoding& encoding, 979 uint32_t depth) const { 980 return encoding.GetInvokeTypeEncoding().Load(GetRegionAtDepth(encoding, depth)); 981 } 982 SetInvokeTypeAtDepth(const InlineInfoEncoding & encoding,uint32_t depth,uint32_t invoke_type)983 ALWAYS_INLINE void SetInvokeTypeAtDepth(const InlineInfoEncoding& encoding, 984 uint32_t depth, 985 uint32_t invoke_type) { 986 encoding.GetInvokeTypeEncoding().Store(GetRegionAtDepth(encoding, depth), invoke_type); 987 } 988 GetDexRegisterMapOffsetAtDepth(const InlineInfoEncoding & encoding,uint32_t depth)989 ALWAYS_INLINE uint32_t GetDexRegisterMapOffsetAtDepth(const InlineInfoEncoding& encoding, 990 uint32_t depth) const { 991 return encoding.GetDexRegisterMapEncoding().Load(GetRegionAtDepth(encoding, depth)); 992 } 993 SetDexRegisterMapOffsetAtDepth(const InlineInfoEncoding & encoding,uint32_t depth,uint32_t offset)994 ALWAYS_INLINE void SetDexRegisterMapOffsetAtDepth(const InlineInfoEncoding& encoding, 995 uint32_t depth, 996 uint32_t offset) { 997 encoding.GetDexRegisterMapEncoding().Store(GetRegionAtDepth(encoding, depth), offset); 998 } 999 HasDexRegisterMapAtDepth(const InlineInfoEncoding & encoding,uint32_t depth)1000 ALWAYS_INLINE bool HasDexRegisterMapAtDepth(const InlineInfoEncoding& encoding, 1001 uint32_t depth) const { 1002 return GetDexRegisterMapOffsetAtDepth(encoding, depth) != StackMap::kNoDexRegisterMap; 1003 } 1004 1005 void Dump(VariableIndentationOutputStream* vios, 1006 const CodeInfo& info, 1007 uint16_t* number_of_dex_registers) const; 1008 1009 private: GetRegionAtDepth(const InlineInfoEncoding & encoding,uint32_t depth)1010 ALWAYS_INLINE MemoryRegion GetRegionAtDepth(const InlineInfoEncoding& encoding, 1011 uint32_t depth) const { 1012 size_t entry_size = encoding.GetEntrySize(); 1013 DCHECK_GT(entry_size, 0u); 1014 return region_.Subregion(depth * entry_size, entry_size); 1015 } 1016 1017 MemoryRegion region_; 1018 }; 1019 1020 // Most of the fields are encoded as ULEB128 to save space. 1021 struct CodeInfoEncoding { 1022 uint32_t non_header_size; 1023 uint32_t number_of_stack_maps; 1024 uint32_t stack_map_size_in_bytes; 1025 uint32_t number_of_location_catalog_entries; 1026 StackMapEncoding stack_map_encoding; 1027 InlineInfoEncoding inline_info_encoding; 1028 uint8_t header_size; 1029 CodeInfoEncodingCodeInfoEncoding1030 CodeInfoEncoding() { } 1031 CodeInfoEncodingCodeInfoEncoding1032 explicit CodeInfoEncoding(const void* data) { 1033 const uint8_t* ptr = reinterpret_cast<const uint8_t*>(data); 1034 non_header_size = DecodeUnsignedLeb128(&ptr); 1035 number_of_stack_maps = DecodeUnsignedLeb128(&ptr); 1036 stack_map_size_in_bytes = DecodeUnsignedLeb128(&ptr); 1037 number_of_location_catalog_entries = DecodeUnsignedLeb128(&ptr); 1038 static_assert(alignof(StackMapEncoding) == 1, 1039 "StackMapEncoding should not require alignment"); 1040 stack_map_encoding = *reinterpret_cast<const StackMapEncoding*>(ptr); 1041 ptr += sizeof(StackMapEncoding); 1042 if (stack_map_encoding.GetInlineInfoEncoding().BitSize() > 0) { 1043 static_assert(alignof(InlineInfoEncoding) == 1, 1044 "InlineInfoEncoding should not require alignment"); 1045 inline_info_encoding = *reinterpret_cast<const InlineInfoEncoding*>(ptr); 1046 ptr += sizeof(InlineInfoEncoding); 1047 } else { 1048 inline_info_encoding = InlineInfoEncoding{}; // NOLINT. 1049 } 1050 header_size = dchecked_integral_cast<uint8_t>(ptr - reinterpret_cast<const uint8_t*>(data)); 1051 } 1052 1053 template<typename Vector> CompressCodeInfoEncoding1054 void Compress(Vector* dest) const { 1055 EncodeUnsignedLeb128(dest, non_header_size); 1056 EncodeUnsignedLeb128(dest, number_of_stack_maps); 1057 EncodeUnsignedLeb128(dest, stack_map_size_in_bytes); 1058 EncodeUnsignedLeb128(dest, number_of_location_catalog_entries); 1059 const uint8_t* stack_map_ptr = reinterpret_cast<const uint8_t*>(&stack_map_encoding); 1060 dest->insert(dest->end(), stack_map_ptr, stack_map_ptr + sizeof(StackMapEncoding)); 1061 if (stack_map_encoding.GetInlineInfoEncoding().BitSize() > 0) { 1062 const uint8_t* inline_info_ptr = reinterpret_cast<const uint8_t*>(&inline_info_encoding); 1063 dest->insert(dest->end(), inline_info_ptr, inline_info_ptr + sizeof(InlineInfoEncoding)); 1064 } 1065 } 1066 }; 1067 1068 /** 1069 * Wrapper around all compiler information collected for a method. 1070 * The information is of the form: 1071 * 1072 * [CodeInfoEncoding, StackMap+, DexRegisterLocationCatalog+, DexRegisterMap+, InlineInfo*] 1073 * 1074 * where CodeInfoEncoding is of the form: 1075 * 1076 * [non_header_size, number_of_stack_maps, stack_map_size_in_bytes, 1077 * number_of_location_catalog_entries, StackMapEncoding] 1078 */ 1079 class CodeInfo { 1080 public: CodeInfo(MemoryRegion region)1081 explicit CodeInfo(MemoryRegion region) : region_(region) { 1082 } 1083 CodeInfo(const void * data)1084 explicit CodeInfo(const void* data) { 1085 CodeInfoEncoding encoding = CodeInfoEncoding(data); 1086 region_ = MemoryRegion(const_cast<void*>(data), 1087 encoding.header_size + encoding.non_header_size); 1088 } 1089 ExtractEncoding()1090 CodeInfoEncoding ExtractEncoding() const { 1091 return CodeInfoEncoding(region_.start()); 1092 } 1093 HasInlineInfo(const CodeInfoEncoding & encoding)1094 bool HasInlineInfo(const CodeInfoEncoding& encoding) const { 1095 return encoding.stack_map_encoding.GetInlineInfoEncoding().BitSize() > 0; 1096 } 1097 GetDexRegisterLocationCatalog(const CodeInfoEncoding & encoding)1098 DexRegisterLocationCatalog GetDexRegisterLocationCatalog(const CodeInfoEncoding& encoding) const { 1099 return DexRegisterLocationCatalog(region_.Subregion( 1100 GetDexRegisterLocationCatalogOffset(encoding), 1101 GetDexRegisterLocationCatalogSize(encoding))); 1102 } 1103 GetStackMapAt(size_t i,const CodeInfoEncoding & encoding)1104 StackMap GetStackMapAt(size_t i, const CodeInfoEncoding& encoding) const { 1105 size_t stack_map_size = encoding.stack_map_size_in_bytes; 1106 return StackMap(GetStackMaps(encoding).Subregion(i * stack_map_size, stack_map_size)); 1107 } 1108 GetNumberOfLocationCatalogEntries(const CodeInfoEncoding & encoding)1109 uint32_t GetNumberOfLocationCatalogEntries(const CodeInfoEncoding& encoding) const { 1110 return encoding.number_of_location_catalog_entries; 1111 } 1112 GetDexRegisterLocationCatalogSize(const CodeInfoEncoding & encoding)1113 uint32_t GetDexRegisterLocationCatalogSize(const CodeInfoEncoding& encoding) const { 1114 return ComputeDexRegisterLocationCatalogSize(GetDexRegisterLocationCatalogOffset(encoding), 1115 GetNumberOfLocationCatalogEntries(encoding)); 1116 } 1117 GetNumberOfStackMaps(const CodeInfoEncoding & encoding)1118 uint32_t GetNumberOfStackMaps(const CodeInfoEncoding& encoding) const { 1119 return encoding.number_of_stack_maps; 1120 } 1121 1122 // Get the size of all the stack maps of this CodeInfo object, in bytes. GetStackMapsSize(const CodeInfoEncoding & encoding)1123 size_t GetStackMapsSize(const CodeInfoEncoding& encoding) const { 1124 return encoding.stack_map_size_in_bytes * GetNumberOfStackMaps(encoding); 1125 } 1126 GetDexRegisterLocationCatalogOffset(const CodeInfoEncoding & encoding)1127 uint32_t GetDexRegisterLocationCatalogOffset(const CodeInfoEncoding& encoding) const { 1128 return GetStackMapsOffset(encoding) + GetStackMapsSize(encoding); 1129 } 1130 GetDexRegisterMapsOffset(const CodeInfoEncoding & encoding)1131 size_t GetDexRegisterMapsOffset(const CodeInfoEncoding& encoding) const { 1132 return GetDexRegisterLocationCatalogOffset(encoding) 1133 + GetDexRegisterLocationCatalogSize(encoding); 1134 } 1135 GetStackMapsOffset(const CodeInfoEncoding & encoding)1136 uint32_t GetStackMapsOffset(const CodeInfoEncoding& encoding) const { 1137 return encoding.header_size; 1138 } 1139 GetDexRegisterMapOf(StackMap stack_map,const CodeInfoEncoding & encoding,uint32_t number_of_dex_registers)1140 DexRegisterMap GetDexRegisterMapOf(StackMap stack_map, 1141 const CodeInfoEncoding& encoding, 1142 uint32_t number_of_dex_registers) const { 1143 if (!stack_map.HasDexRegisterMap(encoding.stack_map_encoding)) { 1144 return DexRegisterMap(); 1145 } else { 1146 uint32_t offset = GetDexRegisterMapsOffset(encoding) 1147 + stack_map.GetDexRegisterMapOffset(encoding.stack_map_encoding); 1148 size_t size = ComputeDexRegisterMapSizeOf(encoding, offset, number_of_dex_registers); 1149 return DexRegisterMap(region_.Subregion(offset, size)); 1150 } 1151 } 1152 1153 // Return the `DexRegisterMap` pointed by `inline_info` at depth `depth`. GetDexRegisterMapAtDepth(uint8_t depth,InlineInfo inline_info,const CodeInfoEncoding & encoding,uint32_t number_of_dex_registers)1154 DexRegisterMap GetDexRegisterMapAtDepth(uint8_t depth, 1155 InlineInfo inline_info, 1156 const CodeInfoEncoding& encoding, 1157 uint32_t number_of_dex_registers) const { 1158 if (!inline_info.HasDexRegisterMapAtDepth(encoding.inline_info_encoding, depth)) { 1159 return DexRegisterMap(); 1160 } else { 1161 uint32_t offset = GetDexRegisterMapsOffset(encoding) + 1162 inline_info.GetDexRegisterMapOffsetAtDepth(encoding.inline_info_encoding, depth); 1163 size_t size = ComputeDexRegisterMapSizeOf(encoding, offset, number_of_dex_registers); 1164 return DexRegisterMap(region_.Subregion(offset, size)); 1165 } 1166 } 1167 GetInlineInfoOf(StackMap stack_map,const CodeInfoEncoding & encoding)1168 InlineInfo GetInlineInfoOf(StackMap stack_map, const CodeInfoEncoding& encoding) const { 1169 DCHECK(stack_map.HasInlineInfo(encoding.stack_map_encoding)); 1170 uint32_t offset = stack_map.GetInlineDescriptorOffset(encoding.stack_map_encoding) 1171 + GetDexRegisterMapsOffset(encoding); 1172 return InlineInfo(region_.Subregion(offset, region_.size() - offset)); 1173 } 1174 GetStackMapForDexPc(uint32_t dex_pc,const CodeInfoEncoding & encoding)1175 StackMap GetStackMapForDexPc(uint32_t dex_pc, const CodeInfoEncoding& encoding) const { 1176 for (size_t i = 0, e = GetNumberOfStackMaps(encoding); i < e; ++i) { 1177 StackMap stack_map = GetStackMapAt(i, encoding); 1178 if (stack_map.GetDexPc(encoding.stack_map_encoding) == dex_pc) { 1179 return stack_map; 1180 } 1181 } 1182 return StackMap(); 1183 } 1184 1185 // Searches the stack map list backwards because catch stack maps are stored 1186 // at the end. GetCatchStackMapForDexPc(uint32_t dex_pc,const CodeInfoEncoding & encoding)1187 StackMap GetCatchStackMapForDexPc(uint32_t dex_pc, const CodeInfoEncoding& encoding) const { 1188 for (size_t i = GetNumberOfStackMaps(encoding); i > 0; --i) { 1189 StackMap stack_map = GetStackMapAt(i - 1, encoding); 1190 if (stack_map.GetDexPc(encoding.stack_map_encoding) == dex_pc) { 1191 return stack_map; 1192 } 1193 } 1194 return StackMap(); 1195 } 1196 GetOsrStackMapForDexPc(uint32_t dex_pc,const CodeInfoEncoding & encoding)1197 StackMap GetOsrStackMapForDexPc(uint32_t dex_pc, const CodeInfoEncoding& encoding) const { 1198 size_t e = GetNumberOfStackMaps(encoding); 1199 if (e == 0) { 1200 // There cannot be OSR stack map if there is no stack map. 1201 return StackMap(); 1202 } 1203 // Walk over all stack maps. If two consecutive stack maps are identical, then we 1204 // have found a stack map suitable for OSR. 1205 const StackMapEncoding& stack_map_encoding = encoding.stack_map_encoding; 1206 for (size_t i = 0; i < e - 1; ++i) { 1207 StackMap stack_map = GetStackMapAt(i, encoding); 1208 if (stack_map.GetDexPc(stack_map_encoding) == dex_pc) { 1209 StackMap other = GetStackMapAt(i + 1, encoding); 1210 if (other.GetDexPc(stack_map_encoding) == dex_pc && 1211 other.GetNativePcOffset(stack_map_encoding) == 1212 stack_map.GetNativePcOffset(stack_map_encoding)) { 1213 DCHECK_EQ(other.GetDexRegisterMapOffset(stack_map_encoding), 1214 stack_map.GetDexRegisterMapOffset(stack_map_encoding)); 1215 DCHECK(!stack_map.HasInlineInfo(stack_map_encoding)); 1216 if (i < e - 2) { 1217 // Make sure there are not three identical stack maps following each other. 1218 DCHECK_NE(stack_map.GetNativePcOffset(stack_map_encoding), 1219 GetStackMapAt(i + 2, encoding).GetNativePcOffset(stack_map_encoding)); 1220 } 1221 return stack_map; 1222 } 1223 } 1224 } 1225 return StackMap(); 1226 } 1227 GetStackMapForNativePcOffset(uint32_t native_pc_offset,const CodeInfoEncoding & encoding)1228 StackMap GetStackMapForNativePcOffset(uint32_t native_pc_offset, 1229 const CodeInfoEncoding& encoding) const { 1230 // TODO: Safepoint stack maps are sorted by native_pc_offset but catch stack 1231 // maps are not. If we knew that the method does not have try/catch, 1232 // we could do binary search. 1233 for (size_t i = 0, e = GetNumberOfStackMaps(encoding); i < e; ++i) { 1234 StackMap stack_map = GetStackMapAt(i, encoding); 1235 if (stack_map.GetNativePcOffset(encoding.stack_map_encoding) == native_pc_offset) { 1236 return stack_map; 1237 } 1238 } 1239 return StackMap(); 1240 } 1241 1242 // Dump this CodeInfo object on `os`. `code_offset` is the (absolute) 1243 // native PC of the compiled method and `number_of_dex_registers` the 1244 // number of Dex virtual registers used in this method. If 1245 // `dump_stack_maps` is true, also dump the stack maps and the 1246 // associated Dex register maps. 1247 void Dump(VariableIndentationOutputStream* vios, 1248 uint32_t code_offset, 1249 uint16_t number_of_dex_registers, 1250 bool dump_stack_maps) const; 1251 1252 private: GetStackMaps(const CodeInfoEncoding & encoding)1253 MemoryRegion GetStackMaps(const CodeInfoEncoding& encoding) const { 1254 return region_.size() == 0 1255 ? MemoryRegion() 1256 : region_.Subregion(GetStackMapsOffset(encoding), GetStackMapsSize(encoding)); 1257 } 1258 1259 // Compute the size of the Dex register map associated to the stack map at 1260 // `dex_register_map_offset_in_code_info`. ComputeDexRegisterMapSizeOf(const CodeInfoEncoding & encoding,uint32_t dex_register_map_offset_in_code_info,uint16_t number_of_dex_registers)1261 size_t ComputeDexRegisterMapSizeOf(const CodeInfoEncoding& encoding, 1262 uint32_t dex_register_map_offset_in_code_info, 1263 uint16_t number_of_dex_registers) const { 1264 // Offset where the actual mapping data starts within art::DexRegisterMap. 1265 size_t location_mapping_data_offset_in_dex_register_map = 1266 DexRegisterMap::GetLocationMappingDataOffset(number_of_dex_registers); 1267 // Create a temporary art::DexRegisterMap to be able to call 1268 // art::DexRegisterMap::GetNumberOfLiveDexRegisters and 1269 DexRegisterMap dex_register_map_without_locations( 1270 MemoryRegion(region_.Subregion(dex_register_map_offset_in_code_info, 1271 location_mapping_data_offset_in_dex_register_map))); 1272 size_t number_of_live_dex_registers = 1273 dex_register_map_without_locations.GetNumberOfLiveDexRegisters(number_of_dex_registers); 1274 size_t location_mapping_data_size_in_bits = 1275 DexRegisterMap::SingleEntrySizeInBits(GetNumberOfLocationCatalogEntries(encoding)) 1276 * number_of_live_dex_registers; 1277 size_t location_mapping_data_size_in_bytes = 1278 RoundUp(location_mapping_data_size_in_bits, kBitsPerByte) / kBitsPerByte; 1279 size_t dex_register_map_size = 1280 location_mapping_data_offset_in_dex_register_map + location_mapping_data_size_in_bytes; 1281 return dex_register_map_size; 1282 } 1283 1284 // Compute the size of a Dex register location catalog starting at offset `origin` 1285 // in `region_` and containing `number_of_dex_locations` entries. ComputeDexRegisterLocationCatalogSize(uint32_t origin,uint32_t number_of_dex_locations)1286 size_t ComputeDexRegisterLocationCatalogSize(uint32_t origin, 1287 uint32_t number_of_dex_locations) const { 1288 // TODO: Ideally, we would like to use art::DexRegisterLocationCatalog::Size or 1289 // art::DexRegisterLocationCatalog::FindLocationOffset, but the 1290 // DexRegisterLocationCatalog is not yet built. Try to factor common code. 1291 size_t offset = origin + DexRegisterLocationCatalog::kFixedSize; 1292 1293 // Skip the first `number_of_dex_locations - 1` entries. 1294 for (uint16_t i = 0; i < number_of_dex_locations; ++i) { 1295 // Read the first next byte and inspect its first 3 bits to decide 1296 // whether it is a short or a large location. 1297 DexRegisterLocationCatalog::ShortLocation first_byte = 1298 region_.LoadUnaligned<DexRegisterLocationCatalog::ShortLocation>(offset); 1299 DexRegisterLocation::Kind kind = 1300 DexRegisterLocationCatalog::ExtractKindFromShortLocation(first_byte); 1301 if (DexRegisterLocation::IsShortLocationKind(kind)) { 1302 // Short location. Skip the current byte. 1303 offset += DexRegisterLocationCatalog::SingleShortEntrySize(); 1304 } else { 1305 // Large location. Skip the 5 next bytes. 1306 offset += DexRegisterLocationCatalog::SingleLargeEntrySize(); 1307 } 1308 } 1309 size_t size = offset - origin; 1310 return size; 1311 } 1312 1313 MemoryRegion region_; 1314 friend class StackMapStream; 1315 }; 1316 1317 #undef ELEMENT_BYTE_OFFSET_AFTER 1318 #undef ELEMENT_BIT_OFFSET_AFTER 1319 1320 } // namespace art 1321 1322 #endif // ART_RUNTIME_STACK_MAP_H_ 1323