1 /*
2 * Copyright (C) 2013 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 #include "trampoline_compiler.h"
18
19 #include "base/arena_allocator.h"
20 #include "jni_env_ext.h"
21
22 #ifdef ART_ENABLE_CODEGEN_arm
23 #include "utils/arm/assembler_arm_vixl.h"
24 #endif
25
26 #ifdef ART_ENABLE_CODEGEN_arm64
27 #include "utils/arm64/assembler_arm64.h"
28 #endif
29
30 #ifdef ART_ENABLE_CODEGEN_mips
31 #include "utils/mips/assembler_mips.h"
32 #endif
33
34 #ifdef ART_ENABLE_CODEGEN_mips64
35 #include "utils/mips64/assembler_mips64.h"
36 #endif
37
38 #ifdef ART_ENABLE_CODEGEN_x86
39 #include "utils/x86/assembler_x86.h"
40 #endif
41
42 #ifdef ART_ENABLE_CODEGEN_x86_64
43 #include "utils/x86_64/assembler_x86_64.h"
44 #endif
45
46 #define __ assembler.
47
48 namespace art {
49
50 #ifdef ART_ENABLE_CODEGEN_arm
51 namespace arm {
52
53 #ifdef ___
54 #error "ARM Assembler macro already defined."
55 #else
56 #define ___ assembler.GetVIXLAssembler()->
57 #endif
58
CreateTrampoline(ArenaAllocator * allocator,EntryPointCallingConvention abi,ThreadOffset32 offset)59 static std::unique_ptr<const std::vector<uint8_t>> CreateTrampoline(
60 ArenaAllocator* allocator, EntryPointCallingConvention abi, ThreadOffset32 offset) {
61 using vixl::aarch32::MemOperand;
62 using vixl::aarch32::pc;
63 using vixl::aarch32::r0;
64 ArmVIXLAssembler assembler(allocator);
65
66 switch (abi) {
67 case kInterpreterAbi: // Thread* is first argument (R0) in interpreter ABI.
68 ___ Ldr(pc, MemOperand(r0, offset.Int32Value()));
69 break;
70 case kJniAbi: { // Load via Thread* held in JNIEnv* in first argument (R0).
71 vixl::aarch32::UseScratchRegisterScope temps(assembler.GetVIXLAssembler());
72 const vixl::aarch32::Register temp_reg = temps.Acquire();
73
74 // VIXL will use the destination as a scratch register if
75 // the offset is not encodable as an immediate operand.
76 ___ Ldr(temp_reg, MemOperand(r0, JNIEnvExt::SelfOffset(4).Int32Value()));
77 ___ Ldr(pc, MemOperand(temp_reg, offset.Int32Value()));
78 break;
79 }
80 case kQuickAbi: // TR holds Thread*.
81 ___ Ldr(pc, MemOperand(tr, offset.Int32Value()));
82 }
83
84 __ FinalizeCode();
85 size_t cs = __ CodeSize();
86 std::unique_ptr<std::vector<uint8_t>> entry_stub(new std::vector<uint8_t>(cs));
87 MemoryRegion code(entry_stub->data(), entry_stub->size());
88 __ FinalizeInstructions(code);
89
90 return std::move(entry_stub);
91 }
92
93 #undef ___
94
95 } // namespace arm
96 #endif // ART_ENABLE_CODEGEN_arm
97
98 #ifdef ART_ENABLE_CODEGEN_arm64
99 namespace arm64 {
CreateTrampoline(ArenaAllocator * allocator,EntryPointCallingConvention abi,ThreadOffset64 offset)100 static std::unique_ptr<const std::vector<uint8_t>> CreateTrampoline(
101 ArenaAllocator* allocator, EntryPointCallingConvention abi, ThreadOffset64 offset) {
102 Arm64Assembler assembler(allocator);
103
104 switch (abi) {
105 case kInterpreterAbi: // Thread* is first argument (X0) in interpreter ABI.
106 __ JumpTo(Arm64ManagedRegister::FromXRegister(X0), Offset(offset.Int32Value()),
107 Arm64ManagedRegister::FromXRegister(IP1));
108
109 break;
110 case kJniAbi: // Load via Thread* held in JNIEnv* in first argument (X0).
111 __ LoadRawPtr(Arm64ManagedRegister::FromXRegister(IP1),
112 Arm64ManagedRegister::FromXRegister(X0),
113 Offset(JNIEnvExt::SelfOffset(8).Int32Value()));
114
115 __ JumpTo(Arm64ManagedRegister::FromXRegister(IP1), Offset(offset.Int32Value()),
116 Arm64ManagedRegister::FromXRegister(IP0));
117
118 break;
119 case kQuickAbi: // X18 holds Thread*.
120 __ JumpTo(Arm64ManagedRegister::FromXRegister(TR), Offset(offset.Int32Value()),
121 Arm64ManagedRegister::FromXRegister(IP0));
122
123 break;
124 }
125
126 __ FinalizeCode();
127 size_t cs = __ CodeSize();
128 std::unique_ptr<std::vector<uint8_t>> entry_stub(new std::vector<uint8_t>(cs));
129 MemoryRegion code(entry_stub->data(), entry_stub->size());
130 __ FinalizeInstructions(code);
131
132 return std::move(entry_stub);
133 }
134 } // namespace arm64
135 #endif // ART_ENABLE_CODEGEN_arm64
136
137 #ifdef ART_ENABLE_CODEGEN_mips
138 namespace mips {
CreateTrampoline(ArenaAllocator * allocator,EntryPointCallingConvention abi,ThreadOffset32 offset)139 static std::unique_ptr<const std::vector<uint8_t>> CreateTrampoline(
140 ArenaAllocator* allocator, EntryPointCallingConvention abi, ThreadOffset32 offset) {
141 MipsAssembler assembler(allocator);
142
143 switch (abi) {
144 case kInterpreterAbi: // Thread* is first argument (A0) in interpreter ABI.
145 __ LoadFromOffset(kLoadWord, T9, A0, offset.Int32Value());
146 break;
147 case kJniAbi: // Load via Thread* held in JNIEnv* in first argument (A0).
148 __ LoadFromOffset(kLoadWord, T9, A0, JNIEnvExt::SelfOffset(4).Int32Value());
149 __ LoadFromOffset(kLoadWord, T9, T9, offset.Int32Value());
150 break;
151 case kQuickAbi: // S1 holds Thread*.
152 __ LoadFromOffset(kLoadWord, T9, S1, offset.Int32Value());
153 }
154 __ Jr(T9);
155 __ NopIfNoReordering();
156 __ Break();
157
158 __ FinalizeCode();
159 size_t cs = __ CodeSize();
160 std::unique_ptr<std::vector<uint8_t>> entry_stub(new std::vector<uint8_t>(cs));
161 MemoryRegion code(entry_stub->data(), entry_stub->size());
162 __ FinalizeInstructions(code);
163
164 return std::move(entry_stub);
165 }
166 } // namespace mips
167 #endif // ART_ENABLE_CODEGEN_mips
168
169 #ifdef ART_ENABLE_CODEGEN_mips64
170 namespace mips64 {
CreateTrampoline(ArenaAllocator * allocator,EntryPointCallingConvention abi,ThreadOffset64 offset)171 static std::unique_ptr<const std::vector<uint8_t>> CreateTrampoline(
172 ArenaAllocator* allocator, EntryPointCallingConvention abi, ThreadOffset64 offset) {
173 Mips64Assembler assembler(allocator);
174
175 switch (abi) {
176 case kInterpreterAbi: // Thread* is first argument (A0) in interpreter ABI.
177 __ LoadFromOffset(kLoadDoubleword, T9, A0, offset.Int32Value());
178 break;
179 case kJniAbi: // Load via Thread* held in JNIEnv* in first argument (A0).
180 __ LoadFromOffset(kLoadDoubleword, T9, A0, JNIEnvExt::SelfOffset(8).Int32Value());
181 __ LoadFromOffset(kLoadDoubleword, T9, T9, offset.Int32Value());
182 break;
183 case kQuickAbi: // Fall-through.
184 __ LoadFromOffset(kLoadDoubleword, T9, S1, offset.Int32Value());
185 }
186 __ Jr(T9);
187 __ Nop();
188 __ Break();
189
190 __ FinalizeCode();
191 size_t cs = __ CodeSize();
192 std::unique_ptr<std::vector<uint8_t>> entry_stub(new std::vector<uint8_t>(cs));
193 MemoryRegion code(entry_stub->data(), entry_stub->size());
194 __ FinalizeInstructions(code);
195
196 return std::move(entry_stub);
197 }
198 } // namespace mips64
199 #endif // ART_ENABLE_CODEGEN_mips
200
201 #ifdef ART_ENABLE_CODEGEN_x86
202 namespace x86 {
CreateTrampoline(ArenaAllocator * allocator,ThreadOffset32 offset)203 static std::unique_ptr<const std::vector<uint8_t>> CreateTrampoline(ArenaAllocator* allocator,
204 ThreadOffset32 offset) {
205 X86Assembler assembler(allocator);
206
207 // All x86 trampolines call via the Thread* held in fs.
208 __ fs()->jmp(Address::Absolute(offset));
209 __ int3();
210
211 __ FinalizeCode();
212 size_t cs = __ CodeSize();
213 std::unique_ptr<std::vector<uint8_t>> entry_stub(new std::vector<uint8_t>(cs));
214 MemoryRegion code(entry_stub->data(), entry_stub->size());
215 __ FinalizeInstructions(code);
216
217 return std::move(entry_stub);
218 }
219 } // namespace x86
220 #endif // ART_ENABLE_CODEGEN_x86
221
222 #ifdef ART_ENABLE_CODEGEN_x86_64
223 namespace x86_64 {
CreateTrampoline(ArenaAllocator * allocator,ThreadOffset64 offset)224 static std::unique_ptr<const std::vector<uint8_t>> CreateTrampoline(ArenaAllocator* allocator,
225 ThreadOffset64 offset) {
226 x86_64::X86_64Assembler assembler(allocator);
227
228 // All x86 trampolines call via the Thread* held in gs.
229 __ gs()->jmp(x86_64::Address::Absolute(offset, true));
230 __ int3();
231
232 __ FinalizeCode();
233 size_t cs = __ CodeSize();
234 std::unique_ptr<std::vector<uint8_t>> entry_stub(new std::vector<uint8_t>(cs));
235 MemoryRegion code(entry_stub->data(), entry_stub->size());
236 __ FinalizeInstructions(code);
237
238 return std::move(entry_stub);
239 }
240 } // namespace x86_64
241 #endif // ART_ENABLE_CODEGEN_x86_64
242
CreateTrampoline64(InstructionSet isa,EntryPointCallingConvention abi,ThreadOffset64 offset)243 std::unique_ptr<const std::vector<uint8_t>> CreateTrampoline64(InstructionSet isa,
244 EntryPointCallingConvention abi,
245 ThreadOffset64 offset) {
246 ArenaPool pool;
247 ArenaAllocator allocator(&pool);
248 switch (isa) {
249 #ifdef ART_ENABLE_CODEGEN_arm64
250 case InstructionSet::kArm64:
251 return arm64::CreateTrampoline(&allocator, abi, offset);
252 #endif
253 #ifdef ART_ENABLE_CODEGEN_mips64
254 case InstructionSet::kMips64:
255 return mips64::CreateTrampoline(&allocator, abi, offset);
256 #endif
257 #ifdef ART_ENABLE_CODEGEN_x86_64
258 case InstructionSet::kX86_64:
259 return x86_64::CreateTrampoline(&allocator, offset);
260 #endif
261 default:
262 UNUSED(abi);
263 UNUSED(offset);
264 LOG(FATAL) << "Unexpected InstructionSet: " << isa;
265 UNREACHABLE();
266 }
267 }
268
CreateTrampoline32(InstructionSet isa,EntryPointCallingConvention abi,ThreadOffset32 offset)269 std::unique_ptr<const std::vector<uint8_t>> CreateTrampoline32(InstructionSet isa,
270 EntryPointCallingConvention abi,
271 ThreadOffset32 offset) {
272 ArenaPool pool;
273 ArenaAllocator allocator(&pool);
274 switch (isa) {
275 #ifdef ART_ENABLE_CODEGEN_arm
276 case InstructionSet::kArm:
277 case InstructionSet::kThumb2:
278 return arm::CreateTrampoline(&allocator, abi, offset);
279 #endif
280 #ifdef ART_ENABLE_CODEGEN_mips
281 case InstructionSet::kMips:
282 return mips::CreateTrampoline(&allocator, abi, offset);
283 #endif
284 #ifdef ART_ENABLE_CODEGEN_x86
285 case InstructionSet::kX86:
286 UNUSED(abi);
287 return x86::CreateTrampoline(&allocator, offset);
288 #endif
289 default:
290 LOG(FATAL) << "Unexpected InstructionSet: " << isa;
291 UNREACHABLE();
292 }
293 }
294
295 } // namespace art
296