| /* |
| * 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. |
| */ |
| |
| #include "dex_writer.h" |
| |
| #include <stdint.h> |
| |
| #include <queue> |
| #include <vector> |
| |
| #include "cdex/compact_dex_file.h" |
| #include "compact_dex_writer.h" |
| #include "dex_file_types.h" |
| #include "standard_dex_file.h" |
| #include "utf.h" |
| |
| namespace art { |
| |
| size_t EncodeIntValue(int32_t value, uint8_t* buffer) { |
| size_t length = 0; |
| if (value >= 0) { |
| while (value > 0x7f) { |
| buffer[length++] = static_cast<uint8_t>(value); |
| value >>= 8; |
| } |
| } else { |
| while (value < -0x80) { |
| buffer[length++] = static_cast<uint8_t>(value); |
| value >>= 8; |
| } |
| } |
| buffer[length++] = static_cast<uint8_t>(value); |
| return length; |
| } |
| |
| size_t EncodeUIntValue(uint32_t value, uint8_t* buffer) { |
| size_t length = 0; |
| do { |
| buffer[length++] = static_cast<uint8_t>(value); |
| value >>= 8; |
| } while (value != 0); |
| return length; |
| } |
| |
| size_t EncodeLongValue(int64_t value, uint8_t* buffer) { |
| size_t length = 0; |
| if (value >= 0) { |
| while (value > 0x7f) { |
| buffer[length++] = static_cast<uint8_t>(value); |
| value >>= 8; |
| } |
| } else { |
| while (value < -0x80) { |
| buffer[length++] = static_cast<uint8_t>(value); |
| value >>= 8; |
| } |
| } |
| buffer[length++] = static_cast<uint8_t>(value); |
| return length; |
| } |
| |
| union FloatUnion { |
| float f_; |
| uint32_t i_; |
| }; |
| |
| size_t EncodeFloatValue(float value, uint8_t* buffer) { |
| FloatUnion float_union; |
| float_union.f_ = value; |
| uint32_t int_value = float_union.i_; |
| size_t index = 3; |
| do { |
| buffer[index--] = int_value >> 24; |
| int_value <<= 8; |
| } while (int_value != 0); |
| return 3 - index; |
| } |
| |
| union DoubleUnion { |
| double d_; |
| uint64_t l_; |
| }; |
| |
| size_t EncodeDoubleValue(double value, uint8_t* buffer) { |
| DoubleUnion double_union; |
| double_union.d_ = value; |
| uint64_t long_value = double_union.l_; |
| size_t index = 7; |
| do { |
| buffer[index--] = long_value >> 56; |
| long_value <<= 8; |
| } while (long_value != 0); |
| return 7 - index; |
| } |
| |
| size_t DexWriter::Write(const void* buffer, size_t length, size_t offset) { |
| DCHECK_LE(offset + length, mem_map_->Size()); |
| memcpy(mem_map_->Begin() + offset, buffer, length); |
| return length; |
| } |
| |
| size_t DexWriter::WriteSleb128(uint32_t value, size_t offset) { |
| uint8_t buffer[8]; |
| EncodeSignedLeb128(buffer, value); |
| return Write(buffer, SignedLeb128Size(value), offset); |
| } |
| |
| size_t DexWriter::WriteUleb128(uint32_t value, size_t offset) { |
| uint8_t buffer[8]; |
| EncodeUnsignedLeb128(buffer, value); |
| return Write(buffer, UnsignedLeb128Size(value), offset); |
| } |
| |
| size_t DexWriter::WriteEncodedValue(dex_ir::EncodedValue* encoded_value, size_t offset) { |
| size_t original_offset = offset; |
| size_t start = 0; |
| size_t length; |
| uint8_t buffer[8]; |
| int8_t type = encoded_value->Type(); |
| switch (type) { |
| case DexFile::kDexAnnotationByte: |
| length = EncodeIntValue(encoded_value->GetByte(), buffer); |
| break; |
| case DexFile::kDexAnnotationShort: |
| length = EncodeIntValue(encoded_value->GetShort(), buffer); |
| break; |
| case DexFile::kDexAnnotationChar: |
| length = EncodeUIntValue(encoded_value->GetChar(), buffer); |
| break; |
| case DexFile::kDexAnnotationInt: |
| length = EncodeIntValue(encoded_value->GetInt(), buffer); |
| break; |
| case DexFile::kDexAnnotationLong: |
| length = EncodeLongValue(encoded_value->GetLong(), buffer); |
| break; |
| case DexFile::kDexAnnotationFloat: |
| length = EncodeFloatValue(encoded_value->GetFloat(), buffer); |
| start = 4 - length; |
| break; |
| case DexFile::kDexAnnotationDouble: |
| length = EncodeDoubleValue(encoded_value->GetDouble(), buffer); |
| start = 8 - length; |
| break; |
| case DexFile::kDexAnnotationMethodType: |
| length = EncodeUIntValue(encoded_value->GetProtoId()->GetIndex(), buffer); |
| break; |
| case DexFile::kDexAnnotationMethodHandle: |
| length = EncodeUIntValue(encoded_value->GetMethodHandle()->GetIndex(), buffer); |
| break; |
| case DexFile::kDexAnnotationString: |
| length = EncodeUIntValue(encoded_value->GetStringId()->GetIndex(), buffer); |
| break; |
| case DexFile::kDexAnnotationType: |
| length = EncodeUIntValue(encoded_value->GetTypeId()->GetIndex(), buffer); |
| break; |
| case DexFile::kDexAnnotationField: |
| case DexFile::kDexAnnotationEnum: |
| length = EncodeUIntValue(encoded_value->GetFieldId()->GetIndex(), buffer); |
| break; |
| case DexFile::kDexAnnotationMethod: |
| length = EncodeUIntValue(encoded_value->GetMethodId()->GetIndex(), buffer); |
| break; |
| case DexFile::kDexAnnotationArray: |
| offset += WriteEncodedValueHeader(type, 0, offset); |
| offset += WriteEncodedArray(encoded_value->GetEncodedArray()->GetEncodedValues(), offset); |
| return offset - original_offset; |
| case DexFile::kDexAnnotationAnnotation: |
| offset += WriteEncodedValueHeader(type, 0, offset); |
| offset += WriteEncodedAnnotation(encoded_value->GetEncodedAnnotation(), offset); |
| return offset - original_offset; |
| case DexFile::kDexAnnotationNull: |
| return WriteEncodedValueHeader(type, 0, offset); |
| case DexFile::kDexAnnotationBoolean: |
| return WriteEncodedValueHeader(type, encoded_value->GetBoolean() ? 1 : 0, offset); |
| default: |
| return 0; |
| } |
| offset += WriteEncodedValueHeader(type, length - 1, offset); |
| offset += Write(buffer + start, length, offset); |
| return offset - original_offset; |
| } |
| |
| size_t DexWriter::WriteEncodedValueHeader(int8_t value_type, size_t value_arg, size_t offset) { |
| uint8_t buffer[1] = { static_cast<uint8_t>((value_arg << 5) | value_type) }; |
| return Write(buffer, sizeof(uint8_t), offset); |
| } |
| |
| size_t DexWriter::WriteEncodedArray(dex_ir::EncodedValueVector* values, size_t offset) { |
| size_t original_offset = offset; |
| offset += WriteUleb128(values->size(), offset); |
| for (std::unique_ptr<dex_ir::EncodedValue>& value : *values) { |
| offset += WriteEncodedValue(value.get(), offset); |
| } |
| return offset - original_offset; |
| } |
| |
| size_t DexWriter::WriteEncodedAnnotation(dex_ir::EncodedAnnotation* annotation, size_t offset) { |
| size_t original_offset = offset; |
| offset += WriteUleb128(annotation->GetType()->GetIndex(), offset); |
| offset += WriteUleb128(annotation->GetAnnotationElements()->size(), offset); |
| for (std::unique_ptr<dex_ir::AnnotationElement>& annotation_element : |
| *annotation->GetAnnotationElements()) { |
| offset += WriteUleb128(annotation_element->GetName()->GetIndex(), offset); |
| offset += WriteEncodedValue(annotation_element->GetValue(), offset); |
| } |
| return offset - original_offset; |
| } |
| |
| size_t DexWriter::WriteEncodedFields(dex_ir::FieldItemVector* fields, size_t offset) { |
| size_t original_offset = offset; |
| uint32_t prev_index = 0; |
| for (std::unique_ptr<dex_ir::FieldItem>& field : *fields) { |
| uint32_t index = field->GetFieldId()->GetIndex(); |
| offset += WriteUleb128(index - prev_index, offset); |
| offset += WriteUleb128(field->GetAccessFlags(), offset); |
| prev_index = index; |
| } |
| return offset - original_offset; |
| } |
| |
| size_t DexWriter::WriteEncodedMethods(dex_ir::MethodItemVector* methods, size_t offset) { |
| size_t original_offset = offset; |
| uint32_t prev_index = 0; |
| for (std::unique_ptr<dex_ir::MethodItem>& method : *methods) { |
| uint32_t index = method->GetMethodId()->GetIndex(); |
| uint32_t code_off = method->GetCodeItem() == nullptr ? 0 : method->GetCodeItem()->GetOffset(); |
| offset += WriteUleb128(index - prev_index, offset); |
| offset += WriteUleb128(method->GetAccessFlags(), offset); |
| offset += WriteUleb128(code_off, offset); |
| prev_index = index; |
| } |
| return offset - original_offset; |
| } |
| |
| void DexWriter::WriteStrings() { |
| uint32_t string_data_off[1]; |
| for (std::unique_ptr<dex_ir::StringId>& string_id : header_->GetCollections().StringIds()) { |
| string_data_off[0] = string_id->DataItem()->GetOffset(); |
| Write(string_data_off, string_id->GetSize(), string_id->GetOffset()); |
| } |
| |
| for (auto& string_data_pair : header_->GetCollections().StringDatas()) { |
| std::unique_ptr<dex_ir::StringData>& string_data = string_data_pair.second; |
| uint32_t offset = string_data->GetOffset(); |
| offset += WriteUleb128(CountModifiedUtf8Chars(string_data->Data()), offset); |
| Write(string_data->Data(), strlen(string_data->Data()), offset); |
| } |
| } |
| |
| void DexWriter::WriteTypes() { |
| uint32_t descriptor_idx[1]; |
| for (std::unique_ptr<dex_ir::TypeId>& type_id : header_->GetCollections().TypeIds()) { |
| descriptor_idx[0] = type_id->GetStringId()->GetIndex(); |
| Write(descriptor_idx, type_id->GetSize(), type_id->GetOffset()); |
| } |
| } |
| |
| void DexWriter::WriteTypeLists() { |
| uint32_t size[1]; |
| uint16_t list[1]; |
| for (auto& type_list_pair : header_->GetCollections().TypeLists()) { |
| std::unique_ptr<dex_ir::TypeList>& type_list = type_list_pair.second; |
| size[0] = type_list->GetTypeList()->size(); |
| uint32_t offset = type_list->GetOffset(); |
| offset += Write(size, sizeof(uint32_t), offset); |
| for (const dex_ir::TypeId* type_id : *type_list->GetTypeList()) { |
| list[0] = type_id->GetIndex(); |
| offset += Write(list, sizeof(uint16_t), offset); |
| } |
| } |
| } |
| |
| void DexWriter::WriteProtos() { |
| uint32_t buffer[3]; |
| for (std::unique_ptr<dex_ir::ProtoId>& proto_id : header_->GetCollections().ProtoIds()) { |
| buffer[0] = proto_id->Shorty()->GetIndex(); |
| buffer[1] = proto_id->ReturnType()->GetIndex(); |
| buffer[2] = proto_id->Parameters() == nullptr ? 0 : proto_id->Parameters()->GetOffset(); |
| Write(buffer, proto_id->GetSize(), proto_id->GetOffset()); |
| } |
| } |
| |
| void DexWriter::WriteFields() { |
| uint16_t buffer[4]; |
| for (std::unique_ptr<dex_ir::FieldId>& field_id : header_->GetCollections().FieldIds()) { |
| buffer[0] = field_id->Class()->GetIndex(); |
| buffer[1] = field_id->Type()->GetIndex(); |
| buffer[2] = field_id->Name()->GetIndex(); |
| buffer[3] = field_id->Name()->GetIndex() >> 16; |
| Write(buffer, field_id->GetSize(), field_id->GetOffset()); |
| } |
| } |
| |
| void DexWriter::WriteMethods() { |
| uint16_t buffer[4]; |
| for (std::unique_ptr<dex_ir::MethodId>& method_id : header_->GetCollections().MethodIds()) { |
| buffer[0] = method_id->Class()->GetIndex(); |
| buffer[1] = method_id->Proto()->GetIndex(); |
| buffer[2] = method_id->Name()->GetIndex(); |
| buffer[3] = method_id->Name()->GetIndex() >> 16; |
| Write(buffer, method_id->GetSize(), method_id->GetOffset()); |
| } |
| } |
| |
| void DexWriter::WriteEncodedArrays() { |
| for (auto& encoded_array_pair : header_->GetCollections().EncodedArrayItems()) { |
| std::unique_ptr<dex_ir::EncodedArrayItem>& encoded_array = encoded_array_pair.second; |
| WriteEncodedArray(encoded_array->GetEncodedValues(), encoded_array->GetOffset()); |
| } |
| } |
| |
| void DexWriter::WriteAnnotations() { |
| uint8_t visibility[1]; |
| for (auto& annotation_pair : header_->GetCollections().AnnotationItems()) { |
| std::unique_ptr<dex_ir::AnnotationItem>& annotation = annotation_pair.second; |
| visibility[0] = annotation->GetVisibility(); |
| size_t offset = annotation->GetOffset(); |
| offset += Write(visibility, sizeof(uint8_t), offset); |
| WriteEncodedAnnotation(annotation->GetAnnotation(), offset); |
| } |
| } |
| |
| void DexWriter::WriteAnnotationSets() { |
| uint32_t size[1]; |
| uint32_t annotation_off[1]; |
| for (auto& annotation_set_pair : header_->GetCollections().AnnotationSetItems()) { |
| std::unique_ptr<dex_ir::AnnotationSetItem>& annotation_set = annotation_set_pair.second; |
| size[0] = annotation_set->GetItems()->size(); |
| size_t offset = annotation_set->GetOffset(); |
| offset += Write(size, sizeof(uint32_t), offset); |
| for (dex_ir::AnnotationItem* annotation : *annotation_set->GetItems()) { |
| annotation_off[0] = annotation->GetOffset(); |
| offset += Write(annotation_off, sizeof(uint32_t), offset); |
| } |
| } |
| } |
| |
| void DexWriter::WriteAnnotationSetRefs() { |
| uint32_t size[1]; |
| uint32_t annotations_off[1]; |
| for (auto& anno_set_ref_pair : header_->GetCollections().AnnotationSetRefLists()) { |
| std::unique_ptr<dex_ir::AnnotationSetRefList>& annotation_set_ref = anno_set_ref_pair.second; |
| size[0] = annotation_set_ref->GetItems()->size(); |
| size_t offset = annotation_set_ref->GetOffset(); |
| offset += Write(size, sizeof(uint32_t), offset); |
| for (dex_ir::AnnotationSetItem* annotation_set : *annotation_set_ref->GetItems()) { |
| annotations_off[0] = annotation_set == nullptr ? 0 : annotation_set->GetOffset(); |
| offset += Write(annotations_off, sizeof(uint32_t), offset); |
| } |
| } |
| } |
| |
| void DexWriter::WriteAnnotationsDirectories() { |
| uint32_t directory_buffer[4]; |
| uint32_t annotation_buffer[2]; |
| for (auto& annotations_directory_pair : header_->GetCollections().AnnotationsDirectoryItems()) { |
| std::unique_ptr<dex_ir::AnnotationsDirectoryItem>& annotations_directory = |
| annotations_directory_pair.second; |
| directory_buffer[0] = annotations_directory->GetClassAnnotation() == nullptr ? 0 : |
| annotations_directory->GetClassAnnotation()->GetOffset(); |
| directory_buffer[1] = annotations_directory->GetFieldAnnotations() == nullptr ? 0 : |
| annotations_directory->GetFieldAnnotations()->size(); |
| directory_buffer[2] = annotations_directory->GetMethodAnnotations() == nullptr ? 0 : |
| annotations_directory->GetMethodAnnotations()->size(); |
| directory_buffer[3] = annotations_directory->GetParameterAnnotations() == nullptr ? 0 : |
| annotations_directory->GetParameterAnnotations()->size(); |
| uint32_t offset = annotations_directory->GetOffset(); |
| offset += Write(directory_buffer, 4 * sizeof(uint32_t), offset); |
| if (annotations_directory->GetFieldAnnotations() != nullptr) { |
| for (std::unique_ptr<dex_ir::FieldAnnotation>& field : |
| *annotations_directory->GetFieldAnnotations()) { |
| annotation_buffer[0] = field->GetFieldId()->GetIndex(); |
| annotation_buffer[1] = field->GetAnnotationSetItem()->GetOffset(); |
| offset += Write(annotation_buffer, 2 * sizeof(uint32_t), offset); |
| } |
| } |
| if (annotations_directory->GetMethodAnnotations() != nullptr) { |
| for (std::unique_ptr<dex_ir::MethodAnnotation>& method : |
| *annotations_directory->GetMethodAnnotations()) { |
| annotation_buffer[0] = method->GetMethodId()->GetIndex(); |
| annotation_buffer[1] = method->GetAnnotationSetItem()->GetOffset(); |
| offset += Write(annotation_buffer, 2 * sizeof(uint32_t), offset); |
| } |
| } |
| if (annotations_directory->GetParameterAnnotations() != nullptr) { |
| for (std::unique_ptr<dex_ir::ParameterAnnotation>& parameter : |
| *annotations_directory->GetParameterAnnotations()) { |
| annotation_buffer[0] = parameter->GetMethodId()->GetIndex(); |
| annotation_buffer[1] = parameter->GetAnnotations()->GetOffset(); |
| offset += Write(annotation_buffer, 2 * sizeof(uint32_t), offset); |
| } |
| } |
| } |
| } |
| |
| void DexWriter::WriteDebugInfoItems() { |
| for (auto& debug_info_pair : header_->GetCollections().DebugInfoItems()) { |
| std::unique_ptr<dex_ir::DebugInfoItem>& debug_info = debug_info_pair.second; |
| Write(debug_info->GetDebugInfo(), debug_info->GetDebugInfoSize(), debug_info->GetOffset()); |
| } |
| } |
| |
| void DexWriter::WriteCodeItems() { |
| uint16_t uint16_buffer[4]; |
| uint32_t uint32_buffer[2]; |
| for (auto& code_item_pair : header_->GetCollections().CodeItems()) { |
| std::unique_ptr<dex_ir::CodeItem>& code_item = code_item_pair.second; |
| uint16_buffer[0] = code_item->RegistersSize(); |
| uint16_buffer[1] = code_item->InsSize(); |
| uint16_buffer[2] = code_item->OutsSize(); |
| uint16_buffer[3] = code_item->TriesSize(); |
| uint32_buffer[0] = code_item->DebugInfo() == nullptr ? 0 : code_item->DebugInfo()->GetOffset(); |
| uint32_buffer[1] = code_item->InsnsSize(); |
| size_t offset = code_item->GetOffset(); |
| offset += Write(uint16_buffer, 4 * sizeof(uint16_t), offset); |
| offset += Write(uint32_buffer, 2 * sizeof(uint32_t), offset); |
| offset += Write(code_item->Insns(), code_item->InsnsSize() * sizeof(uint16_t), offset); |
| if (code_item->TriesSize() != 0) { |
| if (code_item->InsnsSize() % 2 != 0) { |
| uint16_t padding[1] = { 0 }; |
| offset += Write(padding, sizeof(uint16_t), offset); |
| } |
| uint32_t start_addr[1]; |
| uint16_t insn_count_and_handler_off[2]; |
| for (std::unique_ptr<const dex_ir::TryItem>& try_item : *code_item->Tries()) { |
| start_addr[0] = try_item->StartAddr(); |
| insn_count_and_handler_off[0] = try_item->InsnCount(); |
| insn_count_and_handler_off[1] = try_item->GetHandlers()->GetListOffset(); |
| offset += Write(start_addr, sizeof(uint32_t), offset); |
| offset += Write(insn_count_and_handler_off, 2 * sizeof(uint16_t), offset); |
| } |
| // Leave offset pointing to the end of the try items. |
| WriteUleb128(code_item->Handlers()->size(), offset); |
| for (std::unique_ptr<const dex_ir::CatchHandler>& handlers : *code_item->Handlers()) { |
| size_t list_offset = offset + handlers->GetListOffset(); |
| uint32_t size = handlers->HasCatchAll() ? (handlers->GetHandlers()->size() - 1) * -1 : |
| handlers->GetHandlers()->size(); |
| list_offset += WriteSleb128(size, list_offset); |
| for (std::unique_ptr<const dex_ir::TypeAddrPair>& handler : *handlers->GetHandlers()) { |
| if (handler->GetTypeId() != nullptr) { |
| list_offset += WriteUleb128(handler->GetTypeId()->GetIndex(), list_offset); |
| } |
| list_offset += WriteUleb128(handler->GetAddress(), list_offset); |
| } |
| } |
| } |
| } |
| } |
| |
| void DexWriter::WriteClasses() { |
| uint32_t class_def_buffer[8]; |
| for (std::unique_ptr<dex_ir::ClassDef>& class_def : header_->GetCollections().ClassDefs()) { |
| class_def_buffer[0] = class_def->ClassType()->GetIndex(); |
| class_def_buffer[1] = class_def->GetAccessFlags(); |
| class_def_buffer[2] = class_def->Superclass() == nullptr ? dex::kDexNoIndex : |
| class_def->Superclass()->GetIndex(); |
| class_def_buffer[3] = class_def->InterfacesOffset(); |
| class_def_buffer[4] = class_def->SourceFile() == nullptr ? dex::kDexNoIndex : |
| class_def->SourceFile()->GetIndex(); |
| class_def_buffer[5] = class_def->Annotations() == nullptr ? 0 : |
| class_def->Annotations()->GetOffset(); |
| class_def_buffer[6] = class_def->GetClassData() == nullptr ? 0 : |
| class_def->GetClassData()->GetOffset(); |
| class_def_buffer[7] = class_def->StaticValues() == nullptr ? 0 : |
| class_def->StaticValues()->GetOffset(); |
| size_t offset = class_def->GetOffset(); |
| Write(class_def_buffer, class_def->GetSize(), offset); |
| } |
| |
| for (auto& class_data_pair : header_->GetCollections().ClassDatas()) { |
| std::unique_ptr<dex_ir::ClassData>& class_data = class_data_pair.second; |
| size_t offset = class_data->GetOffset(); |
| offset += WriteUleb128(class_data->StaticFields()->size(), offset); |
| offset += WriteUleb128(class_data->InstanceFields()->size(), offset); |
| offset += WriteUleb128(class_data->DirectMethods()->size(), offset); |
| offset += WriteUleb128(class_data->VirtualMethods()->size(), offset); |
| offset += WriteEncodedFields(class_data->StaticFields(), offset); |
| offset += WriteEncodedFields(class_data->InstanceFields(), offset); |
| offset += WriteEncodedMethods(class_data->DirectMethods(), offset); |
| offset += WriteEncodedMethods(class_data->VirtualMethods(), offset); |
| } |
| } |
| |
| void DexWriter::WriteCallSites() { |
| uint32_t call_site_off[1]; |
| for (std::unique_ptr<dex_ir::CallSiteId>& call_site_id : |
| header_->GetCollections().CallSiteIds()) { |
| call_site_off[0] = call_site_id->CallSiteItem()->GetOffset(); |
| Write(call_site_off, call_site_id->GetSize(), call_site_id->GetOffset()); |
| } |
| } |
| |
| void DexWriter::WriteMethodHandles() { |
| uint16_t method_handle_buff[4]; |
| for (std::unique_ptr<dex_ir::MethodHandleItem>& method_handle : |
| header_->GetCollections().MethodHandleItems()) { |
| method_handle_buff[0] = static_cast<uint16_t>(method_handle->GetMethodHandleType()); |
| method_handle_buff[1] = 0; // unused. |
| method_handle_buff[2] = method_handle->GetFieldOrMethodId()->GetIndex(); |
| method_handle_buff[3] = 0; // unused. |
| Write(method_handle_buff, method_handle->GetSize(), method_handle->GetOffset()); |
| } |
| } |
| |
| struct MapItemContainer { |
| MapItemContainer(uint32_t type, uint32_t size, uint32_t offset) |
| : type_(type), size_(size), offset_(offset) { } |
| |
| bool operator<(const MapItemContainer& other) const { |
| return offset_ > other.offset_; |
| } |
| |
| uint32_t type_; |
| uint32_t size_; |
| uint32_t offset_; |
| }; |
| |
| void DexWriter::WriteMapItem() { |
| dex_ir::Collections& collection = header_->GetCollections(); |
| std::priority_queue<MapItemContainer> queue; |
| |
| // Header and index section. |
| queue.push(MapItemContainer(DexFile::kDexTypeHeaderItem, 1, 0)); |
| if (collection.StringIdsSize() != 0) { |
| queue.push(MapItemContainer(DexFile::kDexTypeStringIdItem, collection.StringIdsSize(), |
| collection.StringIdsOffset())); |
| } |
| if (collection.TypeIdsSize() != 0) { |
| queue.push(MapItemContainer(DexFile::kDexTypeTypeIdItem, collection.TypeIdsSize(), |
| collection.TypeIdsOffset())); |
| } |
| if (collection.ProtoIdsSize() != 0) { |
| queue.push(MapItemContainer(DexFile::kDexTypeProtoIdItem, collection.ProtoIdsSize(), |
| collection.ProtoIdsOffset())); |
| } |
| if (collection.FieldIdsSize() != 0) { |
| queue.push(MapItemContainer(DexFile::kDexTypeFieldIdItem, collection.FieldIdsSize(), |
| collection.FieldIdsOffset())); |
| } |
| if (collection.MethodIdsSize() != 0) { |
| queue.push(MapItemContainer(DexFile::kDexTypeMethodIdItem, collection.MethodIdsSize(), |
| collection.MethodIdsOffset())); |
| } |
| if (collection.ClassDefsSize() != 0) { |
| queue.push(MapItemContainer(DexFile::kDexTypeClassDefItem, collection.ClassDefsSize(), |
| collection.ClassDefsOffset())); |
| } |
| if (collection.CallSiteIdsSize() != 0) { |
| queue.push(MapItemContainer(DexFile::kDexTypeCallSiteIdItem, collection.CallSiteIdsSize(), |
| collection.CallSiteIdsOffset())); |
| } |
| if (collection.MethodHandleItemsSize() != 0) { |
| queue.push(MapItemContainer(DexFile::kDexTypeMethodHandleItem, |
| collection.MethodHandleItemsSize(), collection.MethodHandleItemsOffset())); |
| } |
| |
| // Data section. |
| queue.push(MapItemContainer(DexFile::kDexTypeMapList, 1, collection.MapListOffset())); |
| if (collection.TypeListsSize() != 0) { |
| queue.push(MapItemContainer(DexFile::kDexTypeTypeList, collection.TypeListsSize(), |
| collection.TypeListsOffset())); |
| } |
| if (collection.AnnotationSetRefListsSize() != 0) { |
| queue.push(MapItemContainer(DexFile::kDexTypeAnnotationSetRefList, |
| collection.AnnotationSetRefListsSize(), collection.AnnotationSetRefListsOffset())); |
| } |
| if (collection.AnnotationSetItemsSize() != 0) { |
| queue.push(MapItemContainer(DexFile::kDexTypeAnnotationSetItem, |
| collection.AnnotationSetItemsSize(), collection.AnnotationSetItemsOffset())); |
| } |
| if (collection.ClassDatasSize() != 0) { |
| queue.push(MapItemContainer(DexFile::kDexTypeClassDataItem, collection.ClassDatasSize(), |
| collection.ClassDatasOffset())); |
| } |
| if (collection.CodeItemsSize() != 0) { |
| queue.push(MapItemContainer(DexFile::kDexTypeCodeItem, collection.CodeItemsSize(), |
| collection.CodeItemsOffset())); |
| } |
| if (collection.StringDatasSize() != 0) { |
| queue.push(MapItemContainer(DexFile::kDexTypeStringDataItem, collection.StringDatasSize(), |
| collection.StringDatasOffset())); |
| } |
| if (collection.DebugInfoItemsSize() != 0) { |
| queue.push(MapItemContainer(DexFile::kDexTypeDebugInfoItem, collection.DebugInfoItemsSize(), |
| collection.DebugInfoItemsOffset())); |
| } |
| if (collection.AnnotationItemsSize() != 0) { |
| queue.push(MapItemContainer(DexFile::kDexTypeAnnotationItem, collection.AnnotationItemsSize(), |
| collection.AnnotationItemsOffset())); |
| } |
| if (collection.EncodedArrayItemsSize() != 0) { |
| queue.push(MapItemContainer(DexFile::kDexTypeEncodedArrayItem, |
| collection.EncodedArrayItemsSize(), collection.EncodedArrayItemsOffset())); |
| } |
| if (collection.AnnotationsDirectoryItemsSize() != 0) { |
| queue.push(MapItemContainer(DexFile::kDexTypeAnnotationsDirectoryItem, |
| collection.AnnotationsDirectoryItemsSize(), collection.AnnotationsDirectoryItemsOffset())); |
| } |
| |
| uint32_t offset = collection.MapListOffset(); |
| uint16_t uint16_buffer[2]; |
| uint32_t uint32_buffer[2]; |
| uint16_buffer[1] = 0; |
| uint32_buffer[0] = queue.size(); |
| offset += Write(uint32_buffer, sizeof(uint32_t), offset); |
| while (!queue.empty()) { |
| const MapItemContainer& map_item = queue.top(); |
| uint16_buffer[0] = map_item.type_; |
| uint32_buffer[0] = map_item.size_; |
| uint32_buffer[1] = map_item.offset_; |
| offset += Write(uint16_buffer, 2 * sizeof(uint16_t), offset); |
| offset += Write(uint32_buffer, 2 * sizeof(uint32_t), offset); |
| queue.pop(); |
| } |
| } |
| |
| void DexWriter::WriteHeader() { |
| StandardDexFile::Header header; |
| static constexpr size_t kMagicAndVersionLen = |
| StandardDexFile::kDexMagicSize + StandardDexFile::kDexVersionLen; |
| std::copy_n(header_->Magic(), kMagicAndVersionLen, header.magic_); |
| header.checksum_ = header_->Checksum(); |
| std::copy_n(header_->Signature(), DexFile::kSha1DigestSize, header.signature_); |
| header.file_size_ = header_->FileSize(); |
| header.header_size_ = header_->GetSize(); |
| header.endian_tag_ = header_->EndianTag(); |
| header.link_size_ = header_->LinkSize(); |
| header.link_off_ = header_->LinkOffset(); |
| const dex_ir::Collections& collections = header_->GetCollections(); |
| header.map_off_ = collections.MapListOffset(); |
| header.string_ids_size_ = collections.StringIdsSize(); |
| header.string_ids_off_ = collections.StringIdsOffset(); |
| header.type_ids_size_ = collections.TypeIdsSize(); |
| header.type_ids_off_ = collections.TypeIdsOffset(); |
| header.proto_ids_size_ = collections.ProtoIdsSize(); |
| header.proto_ids_off_ = collections.ProtoIdsOffset(); |
| header.field_ids_size_ = collections.FieldIdsSize(); |
| header.field_ids_off_ = collections.FieldIdsOffset(); |
| header.method_ids_size_ = collections.MethodIdsSize(); |
| header.method_ids_off_ = collections.MethodIdsOffset(); |
| header.class_defs_size_ = collections.ClassDefsSize(); |
| header.class_defs_off_ = collections.ClassDefsOffset(); |
| header.data_size_ = header_->DataSize(); |
| header.data_off_ = header_->DataOffset(); |
| |
| static_assert(sizeof(header) == 0x70, "Size doesn't match dex spec"); |
| Write(reinterpret_cast<uint8_t*>(&header), sizeof(header), 0u); |
| } |
| |
| void DexWriter::WriteMemMap() { |
| WriteStrings(); |
| WriteTypes(); |
| WriteTypeLists(); |
| WriteProtos(); |
| WriteFields(); |
| WriteMethods(); |
| WriteEncodedArrays(); |
| WriteAnnotations(); |
| WriteAnnotationSets(); |
| WriteAnnotationSetRefs(); |
| WriteAnnotationsDirectories(); |
| WriteDebugInfoItems(); |
| WriteCodeItems(); |
| WriteClasses(); |
| WriteCallSites(); |
| WriteMethodHandles(); |
| WriteMapItem(); |
| WriteHeader(); |
| } |
| |
| void DexWriter::Output(dex_ir::Header* header, MemMap* mem_map, CompactDexLevel compact_dex_level) { |
| std::unique_ptr<DexWriter> writer; |
| if (compact_dex_level != CompactDexLevel::kCompactDexLevelNone) { |
| writer.reset(new CompactDexWriter(header, mem_map, compact_dex_level)); |
| } else { |
| writer.reset(new DexWriter(header, mem_map)); |
| } |
| writer->WriteMemMap(); |
| } |
| |
| } // namespace art |