| /* |
| * Copyright (C) 2015 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 "linker/relative_patcher.h" |
| |
| #include "debug/method_debug_info.h" |
| #ifdef ART_ENABLE_CODEGEN_arm |
| #include "linker/arm/relative_patcher_thumb2.h" |
| #endif |
| #ifdef ART_ENABLE_CODEGEN_arm64 |
| #include "linker/arm64/relative_patcher_arm64.h" |
| #endif |
| #ifdef ART_ENABLE_CODEGEN_mips |
| #include "linker/mips/relative_patcher_mips.h" |
| #endif |
| #ifdef ART_ENABLE_CODEGEN_mips64 |
| #include "linker/mips64/relative_patcher_mips64.h" |
| #endif |
| #ifdef ART_ENABLE_CODEGEN_x86 |
| #include "linker/x86/relative_patcher_x86.h" |
| #endif |
| #ifdef ART_ENABLE_CODEGEN_x86_64 |
| #include "linker/x86_64/relative_patcher_x86_64.h" |
| #endif |
| #include "stream/output_stream.h" |
| |
| namespace art { |
| namespace linker { |
| |
| std::unique_ptr<RelativePatcher> RelativePatcher::Create( |
| InstructionSet instruction_set, |
| const InstructionSetFeatures* features, |
| RelativePatcherThunkProvider* thunk_provider, |
| RelativePatcherTargetProvider* target_provider) { |
| class RelativePatcherNone final : public RelativePatcher { |
| public: |
| RelativePatcherNone() { } |
| |
| uint32_t ReserveSpace(uint32_t offset, |
| const CompiledMethod* compiled_method ATTRIBUTE_UNUSED, |
| MethodReference method_ref ATTRIBUTE_UNUSED) override { |
| return offset; // No space reserved; no patches expected. |
| } |
| |
| uint32_t ReserveSpaceEnd(uint32_t offset) override { |
| return offset; // No space reserved; no patches expected. |
| } |
| |
| uint32_t WriteThunks(OutputStream* out ATTRIBUTE_UNUSED, uint32_t offset) override { |
| return offset; // No thunks added; no patches expected. |
| } |
| |
| void PatchCall(std::vector<uint8_t>* code ATTRIBUTE_UNUSED, |
| uint32_t literal_offset ATTRIBUTE_UNUSED, |
| uint32_t patch_offset ATTRIBUTE_UNUSED, |
| uint32_t target_offset ATTRIBUTE_UNUSED) override { |
| LOG(FATAL) << "Unexpected relative call patch."; |
| } |
| |
| void PatchPcRelativeReference(std::vector<uint8_t>* code ATTRIBUTE_UNUSED, |
| const LinkerPatch& patch ATTRIBUTE_UNUSED, |
| uint32_t patch_offset ATTRIBUTE_UNUSED, |
| uint32_t target_offset ATTRIBUTE_UNUSED) override { |
| LOG(FATAL) << "Unexpected relative dex cache array patch."; |
| } |
| |
| void PatchEntrypointCall(std::vector<uint8_t>* code ATTRIBUTE_UNUSED, |
| const LinkerPatch& patch ATTRIBUTE_UNUSED, |
| uint32_t patch_offset ATTRIBUTE_UNUSED) override { |
| LOG(FATAL) << "Unexpected entrypoint call patch."; |
| } |
| |
| void PatchBakerReadBarrierBranch(std::vector<uint8_t>* code ATTRIBUTE_UNUSED, |
| const LinkerPatch& patch ATTRIBUTE_UNUSED, |
| uint32_t patch_offset ATTRIBUTE_UNUSED) override { |
| LOG(FATAL) << "Unexpected baker read barrier branch patch."; |
| } |
| |
| std::vector<debug::MethodDebugInfo> GenerateThunkDebugInfo( |
| uint32_t executable_offset ATTRIBUTE_UNUSED) override { |
| return std::vector<debug::MethodDebugInfo>(); // No thunks added. |
| } |
| |
| private: |
| DISALLOW_COPY_AND_ASSIGN(RelativePatcherNone); |
| }; |
| |
| UNUSED(features); |
| UNUSED(thunk_provider); |
| UNUSED(target_provider); |
| switch (instruction_set) { |
| #ifdef ART_ENABLE_CODEGEN_x86 |
| case InstructionSet::kX86: |
| return std::unique_ptr<RelativePatcher>(new X86RelativePatcher()); |
| #endif |
| #ifdef ART_ENABLE_CODEGEN_x86_64 |
| case InstructionSet::kX86_64: |
| return std::unique_ptr<RelativePatcher>(new X86_64RelativePatcher()); |
| #endif |
| #ifdef ART_ENABLE_CODEGEN_arm |
| case InstructionSet::kArm: |
| // Fall through: we generate Thumb2 code for "arm". |
| case InstructionSet::kThumb2: |
| return std::unique_ptr<RelativePatcher>( |
| new Thumb2RelativePatcher(thunk_provider, target_provider)); |
| #endif |
| #ifdef ART_ENABLE_CODEGEN_arm64 |
| case InstructionSet::kArm64: |
| return std::unique_ptr<RelativePatcher>( |
| new Arm64RelativePatcher(thunk_provider, |
| target_provider, |
| features->AsArm64InstructionSetFeatures())); |
| #endif |
| #ifdef ART_ENABLE_CODEGEN_mips |
| case InstructionSet::kMips: |
| return std::unique_ptr<RelativePatcher>( |
| new MipsRelativePatcher(features->AsMipsInstructionSetFeatures())); |
| #endif |
| #ifdef ART_ENABLE_CODEGEN_mips64 |
| case InstructionSet::kMips64: |
| return std::unique_ptr<RelativePatcher>(new Mips64RelativePatcher()); |
| #endif |
| default: |
| return std::unique_ptr<RelativePatcher>(new RelativePatcherNone); |
| } |
| } |
| |
| bool RelativePatcher::WriteCodeAlignment(OutputStream* out, uint32_t aligned_code_delta) { |
| static const uint8_t kPadding[] = { |
| 0u, 0u, 0u, 0u, 0u, 0u, 0u, 0u, 0u, 0u, 0u, 0u, 0u, 0u, 0u, 0u |
| }; |
| DCHECK_LE(aligned_code_delta, sizeof(kPadding)); |
| if (UNLIKELY(!out->WriteFully(kPadding, aligned_code_delta))) { |
| return false; |
| } |
| size_code_alignment_ += aligned_code_delta; |
| return true; |
| } |
| |
| bool RelativePatcher::WriteThunk(OutputStream* out, const ArrayRef<const uint8_t>& thunk) { |
| if (UNLIKELY(!out->WriteFully(thunk.data(), thunk.size()))) { |
| return false; |
| } |
| size_relative_call_thunks_ += thunk.size(); |
| return true; |
| } |
| |
| bool RelativePatcher::WriteMiscThunk(OutputStream* out, const ArrayRef<const uint8_t>& thunk) { |
| if (UNLIKELY(!out->WriteFully(thunk.data(), thunk.size()))) { |
| return false; |
| } |
| size_misc_thunks_ += thunk.size(); |
| return true; |
| } |
| |
| } // namespace linker |
| } // namespace art |