/* * Copyright (C) 2016 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. * * Header file of an in-memory representation of DEX files. */ #include #include #include #include "dex_ir_builder.h" #include "dex/class_accessor-inl.h" #include "dex/code_item_accessors-inl.h" #include "dex/dex_file_exception_helpers.h" #include "dexlayout.h" namespace art { namespace dex_ir { static uint64_t ReadVarWidth(const uint8_t** data, uint8_t length, bool sign_extend) { uint64_t value = 0; for (uint32_t i = 0; i <= length; i++) { value |= static_cast(*(*data)++) << (i * 8); } if (sign_extend) { int shift = (7 - length) * 8; return (static_cast(value) << shift) >> shift; } return value; } static uint32_t GetDebugInfoStreamSize(const uint8_t* debug_info_stream) { const uint8_t* stream = debug_info_stream; DecodeUnsignedLeb128(&stream); // line_start uint32_t parameters_size = DecodeUnsignedLeb128(&stream); for (uint32_t i = 0; i < parameters_size; ++i) { DecodeUnsignedLeb128P1(&stream); // Parameter name. } for (;;) { uint8_t opcode = *stream++; switch (opcode) { case DexFile::DBG_END_SEQUENCE: return stream - debug_info_stream; // end of stream. case DexFile::DBG_ADVANCE_PC: DecodeUnsignedLeb128(&stream); // addr_diff break; case DexFile::DBG_ADVANCE_LINE: DecodeSignedLeb128(&stream); // line_diff break; case DexFile::DBG_START_LOCAL: DecodeUnsignedLeb128(&stream); // register_num DecodeUnsignedLeb128P1(&stream); // name_idx DecodeUnsignedLeb128P1(&stream); // type_idx break; case DexFile::DBG_START_LOCAL_EXTENDED: DecodeUnsignedLeb128(&stream); // register_num DecodeUnsignedLeb128P1(&stream); // name_idx DecodeUnsignedLeb128P1(&stream); // type_idx DecodeUnsignedLeb128P1(&stream); // sig_idx break; case DexFile::DBG_END_LOCAL: case DexFile::DBG_RESTART_LOCAL: DecodeUnsignedLeb128(&stream); // register_num break; case DexFile::DBG_SET_PROLOGUE_END: case DexFile::DBG_SET_EPILOGUE_BEGIN: break; case DexFile::DBG_SET_FILE: { DecodeUnsignedLeb128P1(&stream); // name_idx break; } default: { break; } } } } template class CollectionMap : public CollectionBase { public: CollectionMap() = default; ~CollectionMap() override { } template T* CreateAndAddItem(CollectionVector& vector, bool eagerly_assign_offsets, uint32_t offset, Args&&... args) { T* item = vector.CreateAndAddItem(std::forward(args)...); DCHECK(!GetExistingObject(offset)); DCHECK(!item->OffsetAssigned()); if (eagerly_assign_offsets) { item->SetOffset(offset); } AddItem(item, offset); return item; } // Returns the existing item if it is already inserted, null otherwise. T* GetExistingObject(uint32_t offset) { auto it = collection_.find(offset); return it != collection_.end() ? it->second : nullptr; } uint32_t Size() const override { return size(); } // Lower case for template interop with std::map. uint32_t size() const { return collection_.size(); } std::map& Collection() { return collection_; } private: std::map collection_; // CollectionMaps do not own the objects they contain, therefore AddItem is supported // rather than CreateAndAddItem. void AddItem(T* object, uint32_t offset) { auto it = collection_.emplace(offset, object); CHECK(it.second) << "CollectionMap already has an object with offset " << offset << " " << " and address " << it.first->second; } DISALLOW_COPY_AND_ASSIGN(CollectionMap); }; class BuilderMaps { public: BuilderMaps(Header* header, bool eagerly_assign_offsets) : header_(header), eagerly_assign_offsets_(eagerly_assign_offsets) { } void CreateStringId(const DexFile& dex_file, uint32_t i); void CreateTypeId(const DexFile& dex_file, uint32_t i); void CreateProtoId(const DexFile& dex_file, uint32_t i); void CreateFieldId(const DexFile& dex_file, uint32_t i); void CreateMethodId(const DexFile& dex_file, uint32_t i); void CreateClassDef(const DexFile& dex_file, uint32_t i); void CreateCallSiteId(const DexFile& dex_file, uint32_t i); void CreateMethodHandleItem(const DexFile& dex_file, uint32_t i); void CreateCallSitesAndMethodHandles(const DexFile& dex_file); TypeList* CreateTypeList(const dex::TypeList* type_list, uint32_t offset); EncodedArrayItem* CreateEncodedArrayItem(const DexFile& dex_file, const uint8_t* static_data, uint32_t offset); AnnotationItem* CreateAnnotationItem(const DexFile& dex_file, const dex::AnnotationItem* annotation); AnnotationSetItem* CreateAnnotationSetItem(const DexFile& dex_file, const dex::AnnotationSetItem* disk_annotations_item, uint32_t offset); AnnotationsDirectoryItem* CreateAnnotationsDirectoryItem(const DexFile& dex_file, const dex::AnnotationsDirectoryItem* disk_annotations_item, uint32_t offset); CodeItem* DedupeOrCreateCodeItem(const DexFile& dex_file, const dex::CodeItem* disk_code_item, uint32_t offset, uint32_t dex_method_index); ClassData* CreateClassData(const DexFile& dex_file, const dex::ClassDef& class_def); void AddAnnotationsFromMapListSection(const DexFile& dex_file, uint32_t start_offset, uint32_t count); void AddHiddenapiClassDataFromMapListSection(const DexFile& dex_file, uint32_t offset); void CheckAndSetRemainingOffsets(const DexFile& dex_file, const Options& options); // Sort the vectors buy map order (same order that was used in the input file). void SortVectorsByMapOrder(); private: bool GetIdsFromByteCode(const CodeItem* code, std::vector* type_ids, std::vector* string_ids, std::vector* method_ids, std::vector* field_ids); bool GetIdFromInstruction(const Instruction* dec_insn, std::vector* type_ids, std::vector* string_ids, std::vector* method_ids, std::vector* field_ids); EncodedValue* ReadEncodedValue(const DexFile& dex_file, const uint8_t** data); EncodedValue* ReadEncodedValue(const DexFile& dex_file, const uint8_t** data, uint8_t type, uint8_t length); void ReadEncodedValue(const DexFile& dex_file, const uint8_t** data, uint8_t type, uint8_t length, EncodedValue* item); MethodItem GenerateMethodItem(const DexFile& dex_file, const ClassAccessor::Method& method); ParameterAnnotation* GenerateParameterAnnotation( const DexFile& dex_file, MethodId* method_id, const dex::AnnotationSetRefList* annotation_set_ref_list, uint32_t offset); template Type* CreateAndAddIndexedItem(IndexedCollectionVector& vector, uint32_t offset, uint32_t index, Args&&... args) { Type* item = vector.CreateAndAddIndexedItem(index, std::forward(args)...); DCHECK(!item->OffsetAssigned()); if (eagerly_assign_offsets_) { item->SetOffset(offset); } return item; } Header* header_; // If we eagerly assign offsets during IR building or later after layout. Must be false if // changing the layout is enabled. bool eagerly_assign_offsets_; // Note: maps do not have ownership. CollectionMap string_datas_map_; CollectionMap type_lists_map_; CollectionMap encoded_array_items_map_; CollectionMap annotation_items_map_; CollectionMap annotation_set_items_map_; CollectionMap annotation_set_ref_lists_map_; CollectionMap annotations_directory_items_map_; CollectionMap debug_info_items_map_; // Code item maps need to check both the debug info offset and debug info offset, do not use // CollectionMap. // First offset is the code item offset, second is the debug info offset. std::map, CodeItem*> code_items_map_; CollectionMap class_datas_map_; DISALLOW_COPY_AND_ASSIGN(BuilderMaps); }; Header* DexIrBuilder(const DexFile& dex_file, bool eagerly_assign_offsets, const Options& options) { const DexFile::Header& disk_header = dex_file.GetHeader(); Header* header = new Header(disk_header.magic_, disk_header.checksum_, disk_header.signature_, disk_header.endian_tag_, disk_header.file_size_, disk_header.header_size_, disk_header.link_size_, disk_header.link_off_, disk_header.data_size_, disk_header.data_off_, dex_file.SupportsDefaultMethods(), dex_file.NumStringIds(), dex_file.NumTypeIds(), dex_file.NumProtoIds(), dex_file.NumFieldIds(), dex_file.NumMethodIds(), dex_file.NumClassDefs()); BuilderMaps builder_maps(header, eagerly_assign_offsets); // Walk the rest of the header fields. // StringId table. header->StringIds().SetOffset(disk_header.string_ids_off_); for (uint32_t i = 0; i < dex_file.NumStringIds(); ++i) { builder_maps.CreateStringId(dex_file, i); } // TypeId table. header->TypeIds().SetOffset(disk_header.type_ids_off_); for (uint32_t i = 0; i < dex_file.NumTypeIds(); ++i) { builder_maps.CreateTypeId(dex_file, i); } // ProtoId table. header->ProtoIds().SetOffset(disk_header.proto_ids_off_); for (uint32_t i = 0; i < dex_file.NumProtoIds(); ++i) { builder_maps.CreateProtoId(dex_file, i); } // FieldId table. header->FieldIds().SetOffset(disk_header.field_ids_off_); for (uint32_t i = 0; i < dex_file.NumFieldIds(); ++i) { builder_maps.CreateFieldId(dex_file, i); } // MethodId table. header->MethodIds().SetOffset(disk_header.method_ids_off_); for (uint32_t i = 0; i < dex_file.NumMethodIds(); ++i) { builder_maps.CreateMethodId(dex_file, i); } // ClassDef table. header->ClassDefs().SetOffset(disk_header.class_defs_off_); for (uint32_t i = 0; i < dex_file.NumClassDefs(); ++i) { if (!options.class_filter_.empty()) { // If the filter is enabled (not empty), filter out classes that don't have a matching // descriptor. const dex::ClassDef& class_def = dex_file.GetClassDef(i); const char* descriptor = dex_file.GetClassDescriptor(class_def); if (options.class_filter_.find(descriptor) == options.class_filter_.end()) { continue; } } builder_maps.CreateClassDef(dex_file, i); } // MapItem. header->SetMapListOffset(disk_header.map_off_); // CallSiteIds and MethodHandleItems. builder_maps.CreateCallSitesAndMethodHandles(dex_file); builder_maps.CheckAndSetRemainingOffsets(dex_file, options); // Sort the vectors by the map order (same order as the file). builder_maps.SortVectorsByMapOrder(); // Load the link data if it exists. header->SetLinkData(std::vector( dex_file.DataBegin() + dex_file.GetHeader().link_off_, dex_file.DataBegin() + dex_file.GetHeader().link_off_ + dex_file.GetHeader().link_size_)); return header; } /* * Get all the types, strings, methods, and fields referred to from bytecode. */ void BuilderMaps::CheckAndSetRemainingOffsets(const DexFile& dex_file, const Options& options) { const DexFile::Header& disk_header = dex_file.GetHeader(); // Read MapItems and validate/set remaining offsets. const dex::MapList* map = dex_file.GetMapList(); const uint32_t count = map->size_; for (uint32_t i = 0; i < count; ++i) { const dex::MapItem* item = map->list_ + i; switch (item->type_) { case DexFile::kDexTypeHeaderItem: CHECK_EQ(item->size_, 1u); CHECK_EQ(item->offset_, 0u); break; case DexFile::kDexTypeStringIdItem: CHECK_EQ(item->size_, header_->StringIds().Size()); CHECK_EQ(item->offset_, header_->StringIds().GetOffset()); break; case DexFile::kDexTypeTypeIdItem: CHECK_EQ(item->size_, header_->TypeIds().Size()); CHECK_EQ(item->offset_, header_->TypeIds().GetOffset()); break; case DexFile::kDexTypeProtoIdItem: CHECK_EQ(item->size_, header_->ProtoIds().Size()); CHECK_EQ(item->offset_, header_->ProtoIds().GetOffset()); break; case DexFile::kDexTypeFieldIdItem: CHECK_EQ(item->size_, header_->FieldIds().Size()); CHECK_EQ(item->offset_, header_->FieldIds().GetOffset()); break; case DexFile::kDexTypeMethodIdItem: CHECK_EQ(item->size_, header_->MethodIds().Size()); CHECK_EQ(item->offset_, header_->MethodIds().GetOffset()); break; case DexFile::kDexTypeClassDefItem: if (options.class_filter_.empty()) { // The filter may have removed some classes, this will get fixed up during writing. CHECK_EQ(item->size_, header_->ClassDefs().Size()); } CHECK_EQ(item->offset_, header_->ClassDefs().GetOffset()); break; case DexFile::kDexTypeCallSiteIdItem: CHECK_EQ(item->size_, header_->CallSiteIds().Size()); CHECK_EQ(item->offset_, header_->CallSiteIds().GetOffset()); break; case DexFile::kDexTypeMethodHandleItem: CHECK_EQ(item->size_, header_->MethodHandleItems().Size()); CHECK_EQ(item->offset_, header_->MethodHandleItems().GetOffset()); break; case DexFile::kDexTypeMapList: CHECK_EQ(item->size_, 1u); CHECK_EQ(item->offset_, disk_header.map_off_); break; case DexFile::kDexTypeTypeList: header_->TypeLists().SetOffset(item->offset_); break; case DexFile::kDexTypeAnnotationSetRefList: header_->AnnotationSetRefLists().SetOffset(item->offset_); break; case DexFile::kDexTypeAnnotationSetItem: header_->AnnotationSetItems().SetOffset(item->offset_); break; case DexFile::kDexTypeClassDataItem: header_->ClassDatas().SetOffset(item->offset_); break; case DexFile::kDexTypeCodeItem: header_->CodeItems().SetOffset(item->offset_); break; case DexFile::kDexTypeStringDataItem: header_->StringDatas().SetOffset(item->offset_); break; case DexFile::kDexTypeDebugInfoItem: header_->DebugInfoItems().SetOffset(item->offset_); break; case DexFile::kDexTypeAnnotationItem: header_->AnnotationItems().SetOffset(item->offset_); AddAnnotationsFromMapListSection(dex_file, item->offset_, item->size_); break; case DexFile::kDexTypeEncodedArrayItem: header_->EncodedArrayItems().SetOffset(item->offset_); break; case DexFile::kDexTypeAnnotationsDirectoryItem: header_->AnnotationsDirectoryItems().SetOffset(item->offset_); break; case DexFile::kDexTypeHiddenapiClassData: header_->HiddenapiClassDatas().SetOffset(item->offset_); AddHiddenapiClassDataFromMapListSection(dex_file, item->offset_); break; default: LOG(ERROR) << "Unknown map list item type."; } } } void BuilderMaps::CreateStringId(const DexFile& dex_file, uint32_t i) { const dex::StringId& disk_string_id = dex_file.GetStringId(dex::StringIndex(i)); StringData* string_data = string_datas_map_.CreateAndAddItem(header_->StringDatas(), eagerly_assign_offsets_, disk_string_id.string_data_off_, dex_file.GetStringData(disk_string_id)); CreateAndAddIndexedItem(header_->StringIds(), header_->StringIds().GetOffset() + i * StringId::ItemSize(), i, string_data); } void BuilderMaps::CreateTypeId(const DexFile& dex_file, uint32_t i) { const dex::TypeId& disk_type_id = dex_file.GetTypeId(dex::TypeIndex(i)); CreateAndAddIndexedItem(header_->TypeIds(), header_->TypeIds().GetOffset() + i * TypeId::ItemSize(), i, header_->StringIds()[disk_type_id.descriptor_idx_.index_]); } void BuilderMaps::CreateProtoId(const DexFile& dex_file, uint32_t i) { const dex::ProtoId& disk_proto_id = dex_file.GetProtoId(dex::ProtoIndex(i)); const dex::TypeList* type_list = dex_file.GetProtoParameters(disk_proto_id); TypeList* parameter_type_list = CreateTypeList(type_list, disk_proto_id.parameters_off_); CreateAndAddIndexedItem(header_->ProtoIds(), header_->ProtoIds().GetOffset() + i * ProtoId::ItemSize(), i, header_->StringIds()[disk_proto_id.shorty_idx_.index_], header_->TypeIds()[disk_proto_id.return_type_idx_.index_], parameter_type_list); } void BuilderMaps::CreateFieldId(const DexFile& dex_file, uint32_t i) { const dex::FieldId& disk_field_id = dex_file.GetFieldId(i); CreateAndAddIndexedItem(header_->FieldIds(), header_->FieldIds().GetOffset() + i * FieldId::ItemSize(), i, header_->TypeIds()[disk_field_id.class_idx_.index_], header_->TypeIds()[disk_field_id.type_idx_.index_], header_->StringIds()[disk_field_id.name_idx_.index_]); } void BuilderMaps::CreateMethodId(const DexFile& dex_file, uint32_t i) { const dex::MethodId& disk_method_id = dex_file.GetMethodId(i); CreateAndAddIndexedItem(header_->MethodIds(), header_->MethodIds().GetOffset() + i * MethodId::ItemSize(), i, header_->TypeIds()[disk_method_id.class_idx_.index_], header_->ProtoIds()[disk_method_id.proto_idx_.index_], header_->StringIds()[disk_method_id.name_idx_.index_]); } void BuilderMaps::CreateClassDef(const DexFile& dex_file, uint32_t i) { const dex::ClassDef& disk_class_def = dex_file.GetClassDef(i); const TypeId* class_type = header_->TypeIds()[disk_class_def.class_idx_.index_]; uint32_t access_flags = disk_class_def.access_flags_; const TypeId* superclass = header_->GetTypeIdOrNullPtr(disk_class_def.superclass_idx_.index_); const dex::TypeList* type_list = dex_file.GetInterfacesList(disk_class_def); TypeList* interfaces_type_list = CreateTypeList(type_list, disk_class_def.interfaces_off_); const StringId* source_file = header_->GetStringIdOrNullPtr(disk_class_def.source_file_idx_.index_); // Annotations. AnnotationsDirectoryItem* annotations = nullptr; const dex::AnnotationsDirectoryItem* disk_annotations_directory_item = dex_file.GetAnnotationsDirectory(disk_class_def); if (disk_annotations_directory_item != nullptr) { annotations = CreateAnnotationsDirectoryItem( dex_file, disk_annotations_directory_item, disk_class_def.annotations_off_); } // Static field initializers. const uint8_t* static_data = dex_file.GetEncodedStaticFieldValuesArray(disk_class_def); EncodedArrayItem* static_values = CreateEncodedArrayItem(dex_file, static_data, disk_class_def.static_values_off_); ClassData* class_data = CreateClassData(dex_file, disk_class_def); CreateAndAddIndexedItem(header_->ClassDefs(), header_->ClassDefs().GetOffset() + i * ClassDef::ItemSize(), i, class_type, access_flags, superclass, interfaces_type_list, source_file, annotations, static_values, class_data); } void BuilderMaps::CreateCallSiteId(const DexFile& dex_file, uint32_t i) { const dex::CallSiteIdItem& disk_call_site_id = dex_file.GetCallSiteId(i); const uint8_t* disk_call_item_ptr = dex_file.DataBegin() + disk_call_site_id.data_off_; EncodedArrayItem* call_site_item = CreateEncodedArrayItem(dex_file, disk_call_item_ptr, disk_call_site_id.data_off_); CreateAndAddIndexedItem(header_->CallSiteIds(), header_->CallSiteIds().GetOffset() + i * CallSiteId::ItemSize(), i, call_site_item); } void BuilderMaps::CreateMethodHandleItem(const DexFile& dex_file, uint32_t i) { const dex::MethodHandleItem& disk_method_handle = dex_file.GetMethodHandle(i); uint16_t index = disk_method_handle.field_or_method_idx_; DexFile::MethodHandleType type = static_cast(disk_method_handle.method_handle_type_); bool is_invoke = type == DexFile::MethodHandleType::kInvokeStatic || type == DexFile::MethodHandleType::kInvokeInstance || type == DexFile::MethodHandleType::kInvokeConstructor || type == DexFile::MethodHandleType::kInvokeDirect || type == DexFile::MethodHandleType::kInvokeInterface; static_assert(DexFile::MethodHandleType::kLast == DexFile::MethodHandleType::kInvokeInterface, "Unexpected method handle types."); IndexedItem* field_or_method_id; if (is_invoke) { field_or_method_id = header_->MethodIds()[index]; } else { field_or_method_id = header_->FieldIds()[index]; } CreateAndAddIndexedItem(header_->MethodHandleItems(), header_->MethodHandleItems().GetOffset() + i * MethodHandleItem::ItemSize(), i, type, field_or_method_id); } void BuilderMaps::CreateCallSitesAndMethodHandles(const DexFile& dex_file) { // Iterate through the map list and set the offset of the CallSiteIds and MethodHandleItems. const dex::MapList* map = dex_file.GetMapList(); for (uint32_t i = 0; i < map->size_; ++i) { const dex::MapItem* item = map->list_ + i; switch (item->type_) { case DexFile::kDexTypeCallSiteIdItem: header_->CallSiteIds().SetOffset(item->offset_); break; case DexFile::kDexTypeMethodHandleItem: header_->MethodHandleItems().SetOffset(item->offset_); break; default: break; } } // Populate MethodHandleItems first (CallSiteIds may depend on them). for (uint32_t i = 0; i < dex_file.NumMethodHandles(); i++) { CreateMethodHandleItem(dex_file, i); } // Populate CallSiteIds. for (uint32_t i = 0; i < dex_file.NumCallSiteIds(); i++) { CreateCallSiteId(dex_file, i); } } TypeList* BuilderMaps::CreateTypeList(const dex::TypeList* dex_type_list, uint32_t offset) { if (dex_type_list == nullptr) { return nullptr; } TypeList* type_list = type_lists_map_.GetExistingObject(offset); if (type_list == nullptr) { TypeIdVector* type_vector = new TypeIdVector(); uint32_t size = dex_type_list->Size(); for (uint32_t index = 0; index < size; ++index) { type_vector->push_back(header_->TypeIds()[ dex_type_list->GetTypeItem(index).type_idx_.index_]); } type_list = type_lists_map_.CreateAndAddItem(header_->TypeLists(), eagerly_assign_offsets_, offset, type_vector); } return type_list; } EncodedArrayItem* BuilderMaps::CreateEncodedArrayItem(const DexFile& dex_file, const uint8_t* static_data, uint32_t offset) { if (static_data == nullptr) { return nullptr; } EncodedArrayItem* encoded_array_item = encoded_array_items_map_.GetExistingObject(offset); if (encoded_array_item == nullptr) { uint32_t size = DecodeUnsignedLeb128(&static_data); EncodedValueVector* values = new EncodedValueVector(); for (uint32_t i = 0; i < size; ++i) { values->push_back(std::unique_ptr(ReadEncodedValue(dex_file, &static_data))); } // TODO: Calculate the size of the encoded array. encoded_array_item = encoded_array_items_map_.CreateAndAddItem(header_->EncodedArrayItems(), eagerly_assign_offsets_, offset, values); } return encoded_array_item; } void BuilderMaps::AddAnnotationsFromMapListSection(const DexFile& dex_file, uint32_t start_offset, uint32_t count) { uint32_t current_offset = start_offset; for (size_t i = 0; i < count; ++i) { // Annotation that we didn't process already, add it to the set. const dex::AnnotationItem* annotation = dex_file.GetAnnotationItemAtOffset(current_offset); AnnotationItem* annotation_item = CreateAnnotationItem(dex_file, annotation); DCHECK(annotation_item != nullptr); current_offset += annotation_item->GetSize(); } } void BuilderMaps::AddHiddenapiClassDataFromMapListSection(const DexFile& dex_file, uint32_t offset) { const dex::HiddenapiClassData* hiddenapi_class_data = dex_file.GetHiddenapiClassDataAtOffset(offset); DCHECK(hiddenapi_class_data == dex_file.GetHiddenapiClassData()); for (auto& class_def : header_->ClassDefs()) { uint32_t index = class_def->GetIndex(); ClassData* class_data = class_def->GetClassData(); const uint8_t* ptr = hiddenapi_class_data->GetFlagsPointer(index); std::unique_ptr flags = nullptr; if (ptr != nullptr) { DCHECK(class_data != nullptr); flags = std::make_unique(); for (const dex_ir::FieldItem& field : *class_data->StaticFields()) { flags->emplace(&field, DecodeUnsignedLeb128(&ptr)); } for (const dex_ir::FieldItem& field : *class_data->InstanceFields()) { flags->emplace(&field, DecodeUnsignedLeb128(&ptr)); } for (const dex_ir::MethodItem& method : *class_data->DirectMethods()) { flags->emplace(&method, DecodeUnsignedLeb128(&ptr)); } for (const dex_ir::MethodItem& method : *class_data->VirtualMethods()) { flags->emplace(&method, DecodeUnsignedLeb128(&ptr)); } } CreateAndAddIndexedItem(header_->HiddenapiClassDatas(), header_->HiddenapiClassDatas().GetOffset() + hiddenapi_class_data->flags_offset_[index], index, class_def.get(), std::move(flags)); } } AnnotationItem* BuilderMaps::CreateAnnotationItem(const DexFile& dex_file, const dex::AnnotationItem* annotation) { const uint8_t* const start_data = reinterpret_cast(annotation); const uint32_t offset = start_data - dex_file.DataBegin(); AnnotationItem* annotation_item = annotation_items_map_.GetExistingObject(offset); if (annotation_item == nullptr) { uint8_t visibility = annotation->visibility_; const uint8_t* annotation_data = annotation->annotation_; std::unique_ptr encoded_value( ReadEncodedValue(dex_file, &annotation_data, DexFile::kDexAnnotationAnnotation, 0)); annotation_item = annotation_items_map_.CreateAndAddItem(header_->AnnotationItems(), eagerly_assign_offsets_, offset, visibility, encoded_value->ReleaseEncodedAnnotation()); annotation_item->SetSize(annotation_data - start_data); } return annotation_item; } AnnotationSetItem* BuilderMaps::CreateAnnotationSetItem(const DexFile& dex_file, const dex::AnnotationSetItem* disk_annotations_item, uint32_t offset) { if (disk_annotations_item == nullptr || (disk_annotations_item->size_ == 0 && offset == 0)) { return nullptr; } AnnotationSetItem* annotation_set_item = annotation_set_items_map_.GetExistingObject(offset); if (annotation_set_item == nullptr) { std::vector* items = new std::vector(); for (uint32_t i = 0; i < disk_annotations_item->size_; ++i) { const dex::AnnotationItem* annotation = dex_file.GetAnnotationItem(disk_annotations_item, i); if (annotation == nullptr) { continue; } AnnotationItem* annotation_item = CreateAnnotationItem(dex_file, annotation); items->push_back(annotation_item); } annotation_set_item = annotation_set_items_map_.CreateAndAddItem(header_->AnnotationSetItems(), eagerly_assign_offsets_, offset, items); } return annotation_set_item; } AnnotationsDirectoryItem* BuilderMaps::CreateAnnotationsDirectoryItem(const DexFile& dex_file, const dex::AnnotationsDirectoryItem* disk_annotations_item, uint32_t offset) { AnnotationsDirectoryItem* annotations_directory_item = annotations_directory_items_map_.GetExistingObject(offset); if (annotations_directory_item != nullptr) { return annotations_directory_item; } const dex::AnnotationSetItem* class_set_item = dex_file.GetClassAnnotationSet(disk_annotations_item); AnnotationSetItem* class_annotation = nullptr; if (class_set_item != nullptr) { uint32_t item_offset = disk_annotations_item->class_annotations_off_; class_annotation = CreateAnnotationSetItem(dex_file, class_set_item, item_offset); } const dex::FieldAnnotationsItem* fields = dex_file.GetFieldAnnotations(disk_annotations_item); FieldAnnotationVector* field_annotations = nullptr; if (fields != nullptr) { field_annotations = new FieldAnnotationVector(); for (uint32_t i = 0; i < disk_annotations_item->fields_size_; ++i) { FieldId* field_id = header_->FieldIds()[fields[i].field_idx_]; const dex::AnnotationSetItem* field_set_item = dex_file.GetFieldAnnotationSetItem(fields[i]); uint32_t annotation_set_offset = fields[i].annotations_off_; AnnotationSetItem* annotation_set_item = CreateAnnotationSetItem(dex_file, field_set_item, annotation_set_offset); field_annotations->push_back(std::make_unique( field_id, annotation_set_item)); } } const dex::MethodAnnotationsItem* methods = dex_file.GetMethodAnnotations(disk_annotations_item); MethodAnnotationVector* method_annotations = nullptr; if (methods != nullptr) { method_annotations = new MethodAnnotationVector(); for (uint32_t i = 0; i < disk_annotations_item->methods_size_; ++i) { MethodId* method_id = header_->MethodIds()[methods[i].method_idx_]; const dex::AnnotationSetItem* method_set_item = dex_file.GetMethodAnnotationSetItem(methods[i]); uint32_t annotation_set_offset = methods[i].annotations_off_; AnnotationSetItem* annotation_set_item = CreateAnnotationSetItem(dex_file, method_set_item, annotation_set_offset); method_annotations->push_back(std::make_unique( method_id, annotation_set_item)); } } const dex::ParameterAnnotationsItem* parameters = dex_file.GetParameterAnnotations(disk_annotations_item); ParameterAnnotationVector* parameter_annotations = nullptr; if (parameters != nullptr) { parameter_annotations = new ParameterAnnotationVector(); for (uint32_t i = 0; i < disk_annotations_item->parameters_size_; ++i) { MethodId* method_id = header_->MethodIds()[parameters[i].method_idx_]; const dex::AnnotationSetRefList* list = dex_file.GetParameterAnnotationSetRefList(¶meters[i]); parameter_annotations->push_back(std::unique_ptr( GenerateParameterAnnotation(dex_file, method_id, list, parameters[i].annotations_off_))); } } // TODO: Calculate the size of the annotations directory. return annotations_directory_items_map_.CreateAndAddItem(header_->AnnotationsDirectoryItems(), eagerly_assign_offsets_, offset, class_annotation, field_annotations, method_annotations, parameter_annotations); } CodeItem* BuilderMaps::DedupeOrCreateCodeItem(const DexFile& dex_file, const dex::CodeItem* disk_code_item, uint32_t offset, uint32_t dex_method_index) { if (disk_code_item == nullptr) { return nullptr; } CodeItemDebugInfoAccessor accessor(dex_file, disk_code_item, dex_method_index); const uint32_t debug_info_offset = accessor.DebugInfoOffset(); // Create the offsets pair and dedupe based on it. std::pair offsets_pair(offset, debug_info_offset); auto existing = code_items_map_.find(offsets_pair); if (existing != code_items_map_.end()) { return existing->second; } const uint8_t* debug_info_stream = dex_file.GetDebugInfoStream(debug_info_offset); DebugInfoItem* debug_info = nullptr; if (debug_info_stream != nullptr) { debug_info = debug_info_items_map_.GetExistingObject(debug_info_offset); if (debug_info == nullptr) { uint32_t debug_info_size = GetDebugInfoStreamSize(debug_info_stream); uint8_t* debug_info_buffer = new uint8_t[debug_info_size]; memcpy(debug_info_buffer, debug_info_stream, debug_info_size); debug_info = debug_info_items_map_.CreateAndAddItem(header_->DebugInfoItems(), eagerly_assign_offsets_, debug_info_offset, debug_info_size, debug_info_buffer); } } uint32_t insns_size = accessor.InsnsSizeInCodeUnits(); uint16_t* insns = new uint16_t[insns_size]; memcpy(insns, accessor.Insns(), insns_size * sizeof(uint16_t)); TryItemVector* tries = nullptr; CatchHandlerVector* handler_list = nullptr; if (accessor.TriesSize() > 0) { tries = new TryItemVector(); handler_list = new CatchHandlerVector(); for (const dex::TryItem& disk_try_item : accessor.TryItems()) { uint32_t start_addr = disk_try_item.start_addr_; uint16_t insn_count = disk_try_item.insn_count_; uint16_t handler_off = disk_try_item.handler_off_; const CatchHandler* handlers = nullptr; for (std::unique_ptr& existing_handlers : *handler_list) { if (handler_off == existing_handlers->GetListOffset()) { handlers = existing_handlers.get(); break; } } if (handlers == nullptr) { bool catch_all = false; TypeAddrPairVector* addr_pairs = new TypeAddrPairVector(); for (CatchHandlerIterator it(accessor, disk_try_item); it.HasNext(); it.Next()) { const dex::TypeIndex type_index = it.GetHandlerTypeIndex(); const TypeId* type_id = header_->GetTypeIdOrNullPtr(type_index.index_); catch_all |= type_id == nullptr; addr_pairs->push_back(std::unique_ptr( new TypeAddrPair(type_id, it.GetHandlerAddress()))); } handlers = new CatchHandler(catch_all, handler_off, addr_pairs); handler_list->push_back(std::unique_ptr(handlers)); } TryItem* try_item = new TryItem(start_addr, insn_count, handlers); tries->push_back(std::unique_ptr(try_item)); } // Manually walk catch handlers list and add any missing handlers unreferenced by try items. const uint8_t* handlers_base = accessor.GetCatchHandlerData(); const uint8_t* handlers_data = handlers_base; uint32_t handlers_size = DecodeUnsignedLeb128(&handlers_data); while (handlers_size > handler_list->size()) { bool already_added = false; uint16_t handler_off = handlers_data - handlers_base; for (std::unique_ptr& existing_handlers : *handler_list) { if (handler_off == existing_handlers->GetListOffset()) { already_added = true; break; } } int32_t size = DecodeSignedLeb128(&handlers_data); bool has_catch_all = size <= 0; if (has_catch_all) { size = -size; } if (already_added) { for (int32_t i = 0; i < size; i++) { DecodeUnsignedLeb128(&handlers_data); DecodeUnsignedLeb128(&handlers_data); } if (has_catch_all) { DecodeUnsignedLeb128(&handlers_data); } continue; } TypeAddrPairVector* addr_pairs = new TypeAddrPairVector(); for (int32_t i = 0; i < size; i++) { const TypeId* type_id = header_->GetTypeIdOrNullPtr(DecodeUnsignedLeb128(&handlers_data)); uint32_t addr = DecodeUnsignedLeb128(&handlers_data); addr_pairs->push_back( std::unique_ptr(new TypeAddrPair(type_id, addr))); } if (has_catch_all) { uint32_t addr = DecodeUnsignedLeb128(&handlers_data); addr_pairs->push_back( std::unique_ptr(new TypeAddrPair(nullptr, addr))); } const CatchHandler* handler = new CatchHandler(has_catch_all, handler_off, addr_pairs); handler_list->push_back(std::unique_ptr(handler)); } } uint32_t size = dex_file.GetCodeItemSize(*disk_code_item); CodeItem* code_item = header_->CodeItems().CreateAndAddItem(accessor.RegistersSize(), accessor.InsSize(), accessor.OutsSize(), debug_info, insns_size, insns, tries, handler_list); code_item->SetSize(size); // Add the code item to the map. DCHECK(!code_item->OffsetAssigned()); if (eagerly_assign_offsets_) { code_item->SetOffset(offset); } code_items_map_.emplace(offsets_pair, code_item); // Add "fixup" references to types, strings, methods, and fields. // This is temporary, as we will probably want more detailed parsing of the // instructions here. std::vector type_ids; std::vector string_ids; std::vector method_ids; std::vector field_ids; if (GetIdsFromByteCode(code_item, /*out*/ &type_ids, /*out*/ &string_ids, /*out*/ &method_ids, /*out*/ &field_ids)) { CodeFixups* fixups = new CodeFixups(std::move(type_ids), std::move(string_ids), std::move(method_ids), std::move(field_ids)); code_item->SetCodeFixups(fixups); } return code_item; } ClassData* BuilderMaps::CreateClassData(const DexFile& dex_file, const dex::ClassDef& class_def) { // Read the fields and methods defined by the class, resolving the circular reference from those // to classes by setting class at the same time. const uint32_t offset = class_def.class_data_off_; ClassData* class_data = class_datas_map_.GetExistingObject(offset); if (class_data == nullptr && offset != 0u) { ClassAccessor accessor(dex_file, class_def); // Static fields. FieldItemVector* static_fields = new FieldItemVector(); for (const ClassAccessor::Field& field : accessor.GetStaticFields()) { FieldId* field_item = header_->FieldIds()[field.GetIndex()]; uint32_t access_flags = field.GetAccessFlags(); static_fields->emplace_back(access_flags, field_item); } FieldItemVector* instance_fields = new FieldItemVector(); for (const ClassAccessor::Field& field : accessor.GetInstanceFields()) { FieldId* field_item = header_->FieldIds()[field.GetIndex()]; uint32_t access_flags = field.GetAccessFlags(); instance_fields->emplace_back(access_flags, field_item); } // Direct methods. MethodItemVector* direct_methods = new MethodItemVector(); auto direct_methods_it = accessor.GetDirectMethods(); for (auto it = direct_methods_it.begin(); it != direct_methods_it.end(); ++it) { direct_methods->push_back(GenerateMethodItem(dex_file, *it)); } // Virtual methods. MethodItemVector* virtual_methods = new MethodItemVector(); auto virtual_methods_it = accessor.GetVirtualMethods(); const uint8_t* last_data_ptr; for (auto it = virtual_methods_it.begin(); ; ++it) { if (it == virtual_methods_it.end()) { last_data_ptr = it->GetDataPointer(); break; } virtual_methods->push_back(GenerateMethodItem(dex_file, *it)); } class_data = class_datas_map_.CreateAndAddItem(header_->ClassDatas(), eagerly_assign_offsets_, offset, static_fields, instance_fields, direct_methods, virtual_methods); class_data->SetSize(last_data_ptr - dex_file.GetClassData(class_def)); } return class_data; } void BuilderMaps::SortVectorsByMapOrder() { header_->StringDatas().SortByMapOrder(string_datas_map_.Collection()); header_->TypeLists().SortByMapOrder(type_lists_map_.Collection()); header_->EncodedArrayItems().SortByMapOrder(encoded_array_items_map_.Collection()); header_->AnnotationItems().SortByMapOrder(annotation_items_map_.Collection()); header_->AnnotationSetItems().SortByMapOrder(annotation_set_items_map_.Collection()); header_->AnnotationSetRefLists().SortByMapOrder(annotation_set_ref_lists_map_.Collection()); header_->AnnotationsDirectoryItems().SortByMapOrder( annotations_directory_items_map_.Collection()); header_->DebugInfoItems().SortByMapOrder(debug_info_items_map_.Collection()); header_->CodeItems().SortByMapOrder(code_items_map_); header_->ClassDatas().SortByMapOrder(class_datas_map_.Collection()); } bool BuilderMaps::GetIdsFromByteCode(const CodeItem* code, std::vector* type_ids, std::vector* string_ids, std::vector* method_ids, std::vector* field_ids) { bool has_id = false; IterationRange instructions = code->Instructions(); SafeDexInstructionIterator it(instructions.begin(), instructions.end()); for (; !it.IsErrorState() && it < instructions.end(); ++it) { // In case the instruction goes past the end of the code item, make sure to not process it. SafeDexInstructionIterator next = it; ++next; if (next.IsErrorState()) { break; } has_id |= GetIdFromInstruction(&it.Inst(), type_ids, string_ids, method_ids, field_ids); } // for return has_id; } bool BuilderMaps::GetIdFromInstruction(const Instruction* dec_insn, std::vector* type_ids, std::vector* string_ids, std::vector* method_ids, std::vector* field_ids) { // Determine index and width of the string. uint32_t index = 0; switch (Instruction::FormatOf(dec_insn->Opcode())) { // SOME NOT SUPPORTED: // case Instruction::k20bc: case Instruction::k21c: case Instruction::k35c: // case Instruction::k35ms: case Instruction::k3rc: // case Instruction::k3rms: // case Instruction::k35mi: // case Instruction::k3rmi: case Instruction::k45cc: case Instruction::k4rcc: index = dec_insn->VRegB(); break; case Instruction::k31c: index = dec_insn->VRegB(); break; case Instruction::k22c: // case Instruction::k22cs: index = dec_insn->VRegC(); break; default: break; } // switch // Determine index type, and add reference to the appropriate collection. switch (Instruction::IndexTypeOf(dec_insn->Opcode())) { case Instruction::kIndexTypeRef: if (index < header_->TypeIds().Size()) { type_ids->push_back(header_->TypeIds()[index]); return true; } break; case Instruction::kIndexStringRef: if (index < header_->StringIds().Size()) { string_ids->push_back(header_->StringIds()[index]); return true; } break; case Instruction::kIndexMethodRef: case Instruction::kIndexMethodAndProtoRef: if (index < header_->MethodIds().Size()) { method_ids->push_back(header_->MethodIds()[index]); return true; } break; case Instruction::kIndexFieldRef: if (index < header_->FieldIds().Size()) { field_ids->push_back(header_->FieldIds()[index]); return true; } break; case Instruction::kIndexUnknown: case Instruction::kIndexNone: case Instruction::kIndexVtableOffset: case Instruction::kIndexFieldOffset: default: break; } // switch return false; } EncodedValue* BuilderMaps::ReadEncodedValue(const DexFile& dex_file, const uint8_t** data) { const uint8_t encoded_value = *(*data)++; const uint8_t type = encoded_value & 0x1f; EncodedValue* item = new EncodedValue(type); ReadEncodedValue(dex_file, data, type, encoded_value >> 5, item); return item; } EncodedValue* BuilderMaps::ReadEncodedValue(const DexFile& dex_file, const uint8_t** data, uint8_t type, uint8_t length) { EncodedValue* item = new EncodedValue(type); ReadEncodedValue(dex_file, data, type, length, item); return item; } void BuilderMaps::ReadEncodedValue(const DexFile& dex_file, const uint8_t** data, uint8_t type, uint8_t length, EncodedValue* item) { switch (type) { case DexFile::kDexAnnotationByte: item->SetByte(static_cast(ReadVarWidth(data, length, false))); break; case DexFile::kDexAnnotationShort: item->SetShort(static_cast(ReadVarWidth(data, length, true))); break; case DexFile::kDexAnnotationChar: item->SetChar(static_cast(ReadVarWidth(data, length, false))); break; case DexFile::kDexAnnotationInt: item->SetInt(static_cast(ReadVarWidth(data, length, true))); break; case DexFile::kDexAnnotationLong: item->SetLong(static_cast(ReadVarWidth(data, length, true))); break; case DexFile::kDexAnnotationFloat: { // Fill on right. union { float f; uint32_t data; } conv; conv.data = static_cast(ReadVarWidth(data, length, false)) << (3 - length) * 8; item->SetFloat(conv.f); break; } case DexFile::kDexAnnotationDouble: { // Fill on right. union { double d; uint64_t data; } conv; conv.data = ReadVarWidth(data, length, false) << (7 - length) * 8; item->SetDouble(conv.d); break; } case DexFile::kDexAnnotationMethodType: { const uint32_t proto_index = static_cast(ReadVarWidth(data, length, false)); item->SetProtoId(header_->ProtoIds()[proto_index]); break; } case DexFile::kDexAnnotationMethodHandle: { const uint32_t method_handle_index = static_cast(ReadVarWidth(data, length, false)); item->SetMethodHandle(header_->MethodHandleItems()[method_handle_index]); break; } case DexFile::kDexAnnotationString: { const uint32_t string_index = static_cast(ReadVarWidth(data, length, false)); item->SetStringId(header_->StringIds()[string_index]); break; } case DexFile::kDexAnnotationType: { const uint32_t string_index = static_cast(ReadVarWidth(data, length, false)); item->SetTypeId(header_->TypeIds()[string_index]); break; } case DexFile::kDexAnnotationField: case DexFile::kDexAnnotationEnum: { const uint32_t field_index = static_cast(ReadVarWidth(data, length, false)); item->SetFieldId(header_->FieldIds()[field_index]); break; } case DexFile::kDexAnnotationMethod: { const uint32_t method_index = static_cast(ReadVarWidth(data, length, false)); item->SetMethodId(header_->MethodIds()[method_index]); break; } case DexFile::kDexAnnotationArray: { EncodedValueVector* values = new EncodedValueVector(); const uint32_t offset = *data - dex_file.DataBegin(); const uint32_t size = DecodeUnsignedLeb128(data); // Decode all elements. for (uint32_t i = 0; i < size; i++) { values->push_back(std::unique_ptr(ReadEncodedValue(dex_file, data))); } EncodedArrayItem* array_item = new EncodedArrayItem(values); if (eagerly_assign_offsets_) { array_item->SetOffset(offset); } item->SetEncodedArray(array_item); break; } case DexFile::kDexAnnotationAnnotation: { AnnotationElementVector* elements = new AnnotationElementVector(); const uint32_t type_idx = DecodeUnsignedLeb128(data); const uint32_t size = DecodeUnsignedLeb128(data); // Decode all name=value pairs. for (uint32_t i = 0; i < size; i++) { const uint32_t name_index = DecodeUnsignedLeb128(data); elements->push_back(std::make_unique( header_->StringIds()[name_index], ReadEncodedValue(dex_file, data))); } item->SetEncodedAnnotation(new EncodedAnnotation(header_->TypeIds()[type_idx], elements)); break; } case DexFile::kDexAnnotationNull: break; case DexFile::kDexAnnotationBoolean: item->SetBoolean(length != 0); break; default: break; } } MethodItem BuilderMaps::GenerateMethodItem(const DexFile& dex_file, const ClassAccessor::Method& method) { MethodId* method_id = header_->MethodIds()[method.GetIndex()]; uint32_t access_flags = method.GetAccessFlags(); const dex::CodeItem* disk_code_item = method.GetCodeItem(); // Temporary hack to prevent incorrectly deduping code items if they have the same offset since // they may have different debug info streams. CodeItem* code_item = DedupeOrCreateCodeItem(dex_file, disk_code_item, method.GetCodeItemOffset(), method.GetIndex()); return MethodItem(access_flags, method_id, code_item); } ParameterAnnotation* BuilderMaps::GenerateParameterAnnotation( const DexFile& dex_file, MethodId* method_id, const dex::AnnotationSetRefList* annotation_set_ref_list, uint32_t offset) { AnnotationSetRefList* set_ref_list = annotation_set_ref_lists_map_.GetExistingObject(offset); if (set_ref_list == nullptr) { std::vector* annotations = new std::vector(); for (uint32_t i = 0; i < annotation_set_ref_list->size_; ++i) { const dex::AnnotationSetItem* annotation_set_item = dex_file.GetSetRefItemItem(&annotation_set_ref_list->list_[i]); uint32_t set_offset = annotation_set_ref_list->list_[i].annotations_off_; annotations->push_back(CreateAnnotationSetItem(dex_file, annotation_set_item, set_offset)); } set_ref_list = annotation_set_ref_lists_map_.CreateAndAddItem(header_->AnnotationSetRefLists(), eagerly_assign_offsets_, offset, annotations); } return new ParameterAnnotation(method_id, set_ref_list); } } // namespace dex_ir } // namespace art