1 /*
2  * Copyright (C) 2016 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 "jni_macro_assembler_arm64.h"
18 
19 #include "entrypoints/quick/quick_entrypoints.h"
20 #include "managed_register_arm64.h"
21 #include "offsets.h"
22 #include "thread.h"
23 
24 using namespace vixl::aarch64;  // NOLINT(build/namespaces)
25 
26 namespace art {
27 namespace arm64 {
28 
29 #ifdef ___
30 #error "ARM64 Assembler macro already defined."
31 #else
32 #define ___   asm_.GetVIXLAssembler()->
33 #endif
34 
35 #define reg_x(X) Arm64Assembler::reg_x(X)
36 #define reg_w(W) Arm64Assembler::reg_w(W)
37 #define reg_d(D) Arm64Assembler::reg_d(D)
38 #define reg_s(S) Arm64Assembler::reg_s(S)
39 
~Arm64JNIMacroAssembler()40 Arm64JNIMacroAssembler::~Arm64JNIMacroAssembler() {
41 }
42 
FinalizeCode()43 void Arm64JNIMacroAssembler::FinalizeCode() {
44   for (const std::unique_ptr<Arm64Exception>& exception : exception_blocks_) {
45     EmitExceptionPoll(exception.get());
46   }
47   ___ FinalizeCode();
48 }
49 
GetCurrentThread(ManagedRegister tr)50 void Arm64JNIMacroAssembler::GetCurrentThread(ManagedRegister tr) {
51   ___ Mov(reg_x(tr.AsArm64().AsXRegister()), reg_x(TR));
52 }
53 
GetCurrentThread(FrameOffset offset,ManagedRegister)54 void Arm64JNIMacroAssembler::GetCurrentThread(FrameOffset offset, ManagedRegister /* scratch */) {
55   StoreToOffset(TR, SP, offset.Int32Value());
56 }
57 
58 // See Arm64 PCS Section 5.2.2.1.
IncreaseFrameSize(size_t adjust)59 void Arm64JNIMacroAssembler::IncreaseFrameSize(size_t adjust) {
60   CHECK_ALIGNED(adjust, kStackAlignment);
61   AddConstant(SP, -adjust);
62   cfi().AdjustCFAOffset(adjust);
63 }
64 
65 // See Arm64 PCS Section 5.2.2.1.
DecreaseFrameSize(size_t adjust)66 void Arm64JNIMacroAssembler::DecreaseFrameSize(size_t adjust) {
67   CHECK_ALIGNED(adjust, kStackAlignment);
68   AddConstant(SP, adjust);
69   cfi().AdjustCFAOffset(-adjust);
70 }
71 
AddConstant(XRegister rd,int32_t value,Condition cond)72 void Arm64JNIMacroAssembler::AddConstant(XRegister rd, int32_t value, Condition cond) {
73   AddConstant(rd, rd, value, cond);
74 }
75 
AddConstant(XRegister rd,XRegister rn,int32_t value,Condition cond)76 void Arm64JNIMacroAssembler::AddConstant(XRegister rd,
77                                          XRegister rn,
78                                          int32_t value,
79                                          Condition cond) {
80   if ((cond == al) || (cond == nv)) {
81     // VIXL macro-assembler handles all variants.
82     ___ Add(reg_x(rd), reg_x(rn), value);
83   } else {
84     // temp = rd + value
85     // rd = cond ? temp : rn
86     UseScratchRegisterScope temps(asm_.GetVIXLAssembler());
87     temps.Exclude(reg_x(rd), reg_x(rn));
88     Register temp = temps.AcquireX();
89     ___ Add(temp, reg_x(rn), value);
90     ___ Csel(reg_x(rd), temp, reg_x(rd), cond);
91   }
92 }
93 
StoreWToOffset(StoreOperandType type,WRegister source,XRegister base,int32_t offset)94 void Arm64JNIMacroAssembler::StoreWToOffset(StoreOperandType type,
95                                             WRegister source,
96                                             XRegister base,
97                                             int32_t offset) {
98   switch (type) {
99     case kStoreByte:
100       ___ Strb(reg_w(source), MEM_OP(reg_x(base), offset));
101       break;
102     case kStoreHalfword:
103       ___ Strh(reg_w(source), MEM_OP(reg_x(base), offset));
104       break;
105     case kStoreWord:
106       ___ Str(reg_w(source), MEM_OP(reg_x(base), offset));
107       break;
108     default:
109       LOG(FATAL) << "UNREACHABLE";
110   }
111 }
112 
StoreToOffset(XRegister source,XRegister base,int32_t offset)113 void Arm64JNIMacroAssembler::StoreToOffset(XRegister source, XRegister base, int32_t offset) {
114   CHECK_NE(source, SP);
115   ___ Str(reg_x(source), MEM_OP(reg_x(base), offset));
116 }
117 
StoreSToOffset(SRegister source,XRegister base,int32_t offset)118 void Arm64JNIMacroAssembler::StoreSToOffset(SRegister source, XRegister base, int32_t offset) {
119   ___ Str(reg_s(source), MEM_OP(reg_x(base), offset));
120 }
121 
StoreDToOffset(DRegister source,XRegister base,int32_t offset)122 void Arm64JNIMacroAssembler::StoreDToOffset(DRegister source, XRegister base, int32_t offset) {
123   ___ Str(reg_d(source), MEM_OP(reg_x(base), offset));
124 }
125 
Store(FrameOffset offs,ManagedRegister m_src,size_t size)126 void Arm64JNIMacroAssembler::Store(FrameOffset offs, ManagedRegister m_src, size_t size) {
127   Arm64ManagedRegister src = m_src.AsArm64();
128   if (src.IsNoRegister()) {
129     CHECK_EQ(0u, size);
130   } else if (src.IsWRegister()) {
131     CHECK_EQ(4u, size);
132     StoreWToOffset(kStoreWord, src.AsWRegister(), SP, offs.Int32Value());
133   } else if (src.IsXRegister()) {
134     CHECK_EQ(8u, size);
135     StoreToOffset(src.AsXRegister(), SP, offs.Int32Value());
136   } else if (src.IsSRegister()) {
137     StoreSToOffset(src.AsSRegister(), SP, offs.Int32Value());
138   } else {
139     CHECK(src.IsDRegister()) << src;
140     StoreDToOffset(src.AsDRegister(), SP, offs.Int32Value());
141   }
142 }
143 
StoreRef(FrameOffset offs,ManagedRegister m_src)144 void Arm64JNIMacroAssembler::StoreRef(FrameOffset offs, ManagedRegister m_src) {
145   Arm64ManagedRegister src = m_src.AsArm64();
146   CHECK(src.IsXRegister()) << src;
147   StoreWToOffset(kStoreWord, src.AsOverlappingWRegister(), SP,
148                  offs.Int32Value());
149 }
150 
StoreRawPtr(FrameOffset offs,ManagedRegister m_src)151 void Arm64JNIMacroAssembler::StoreRawPtr(FrameOffset offs, ManagedRegister m_src) {
152   Arm64ManagedRegister src = m_src.AsArm64();
153   CHECK(src.IsXRegister()) << src;
154   StoreToOffset(src.AsXRegister(), SP, offs.Int32Value());
155 }
156 
StoreImmediateToFrame(FrameOffset offs,uint32_t imm,ManagedRegister m_scratch)157 void Arm64JNIMacroAssembler::StoreImmediateToFrame(FrameOffset offs,
158                                                    uint32_t imm,
159                                                    ManagedRegister m_scratch) {
160   Arm64ManagedRegister scratch = m_scratch.AsArm64();
161   CHECK(scratch.IsXRegister()) << scratch;
162   LoadImmediate(scratch.AsXRegister(), imm);
163   StoreWToOffset(kStoreWord, scratch.AsOverlappingWRegister(), SP,
164                  offs.Int32Value());
165 }
166 
StoreStackOffsetToThread(ThreadOffset64 tr_offs,FrameOffset fr_offs,ManagedRegister m_scratch)167 void Arm64JNIMacroAssembler::StoreStackOffsetToThread(ThreadOffset64 tr_offs,
168                                                       FrameOffset fr_offs,
169                                                       ManagedRegister m_scratch) {
170   Arm64ManagedRegister scratch = m_scratch.AsArm64();
171   CHECK(scratch.IsXRegister()) << scratch;
172   AddConstant(scratch.AsXRegister(), SP, fr_offs.Int32Value());
173   StoreToOffset(scratch.AsXRegister(), TR, tr_offs.Int32Value());
174 }
175 
StoreStackPointerToThread(ThreadOffset64 tr_offs)176 void Arm64JNIMacroAssembler::StoreStackPointerToThread(ThreadOffset64 tr_offs) {
177   UseScratchRegisterScope temps(asm_.GetVIXLAssembler());
178   Register temp = temps.AcquireX();
179   ___ Mov(temp, reg_x(SP));
180   ___ Str(temp, MEM_OP(reg_x(TR), tr_offs.Int32Value()));
181 }
182 
StoreSpanning(FrameOffset dest_off,ManagedRegister m_source,FrameOffset in_off,ManagedRegister m_scratch)183 void Arm64JNIMacroAssembler::StoreSpanning(FrameOffset dest_off,
184                                            ManagedRegister m_source,
185                                            FrameOffset in_off,
186                                            ManagedRegister m_scratch) {
187   Arm64ManagedRegister source = m_source.AsArm64();
188   Arm64ManagedRegister scratch = m_scratch.AsArm64();
189   StoreToOffset(source.AsXRegister(), SP, dest_off.Int32Value());
190   LoadFromOffset(scratch.AsXRegister(), SP, in_off.Int32Value());
191   StoreToOffset(scratch.AsXRegister(), SP, dest_off.Int32Value() + 8);
192 }
193 
194 // Load routines.
LoadImmediate(XRegister dest,int32_t value,Condition cond)195 void Arm64JNIMacroAssembler::LoadImmediate(XRegister dest, int32_t value, Condition cond) {
196   if ((cond == al) || (cond == nv)) {
197     ___ Mov(reg_x(dest), value);
198   } else {
199     // temp = value
200     // rd = cond ? temp : rd
201     if (value != 0) {
202       UseScratchRegisterScope temps(asm_.GetVIXLAssembler());
203       temps.Exclude(reg_x(dest));
204       Register temp = temps.AcquireX();
205       ___ Mov(temp, value);
206       ___ Csel(reg_x(dest), temp, reg_x(dest), cond);
207     } else {
208       ___ Csel(reg_x(dest), reg_x(XZR), reg_x(dest), cond);
209     }
210   }
211 }
212 
LoadWFromOffset(LoadOperandType type,WRegister dest,XRegister base,int32_t offset)213 void Arm64JNIMacroAssembler::LoadWFromOffset(LoadOperandType type,
214                                              WRegister dest,
215                                              XRegister base,
216                                              int32_t offset) {
217   switch (type) {
218     case kLoadSignedByte:
219       ___ Ldrsb(reg_w(dest), MEM_OP(reg_x(base), offset));
220       break;
221     case kLoadSignedHalfword:
222       ___ Ldrsh(reg_w(dest), MEM_OP(reg_x(base), offset));
223       break;
224     case kLoadUnsignedByte:
225       ___ Ldrb(reg_w(dest), MEM_OP(reg_x(base), offset));
226       break;
227     case kLoadUnsignedHalfword:
228       ___ Ldrh(reg_w(dest), MEM_OP(reg_x(base), offset));
229       break;
230     case kLoadWord:
231       ___ Ldr(reg_w(dest), MEM_OP(reg_x(base), offset));
232       break;
233     default:
234         LOG(FATAL) << "UNREACHABLE";
235   }
236 }
237 
238 // Note: We can extend this member by adding load type info - see
239 // sign extended A64 load variants.
LoadFromOffset(XRegister dest,XRegister base,int32_t offset)240 void Arm64JNIMacroAssembler::LoadFromOffset(XRegister dest, XRegister base, int32_t offset) {
241   CHECK_NE(dest, SP);
242   ___ Ldr(reg_x(dest), MEM_OP(reg_x(base), offset));
243 }
244 
LoadSFromOffset(SRegister dest,XRegister base,int32_t offset)245 void Arm64JNIMacroAssembler::LoadSFromOffset(SRegister dest, XRegister base, int32_t offset) {
246   ___ Ldr(reg_s(dest), MEM_OP(reg_x(base), offset));
247 }
248 
LoadDFromOffset(DRegister dest,XRegister base,int32_t offset)249 void Arm64JNIMacroAssembler::LoadDFromOffset(DRegister dest, XRegister base, int32_t offset) {
250   ___ Ldr(reg_d(dest), MEM_OP(reg_x(base), offset));
251 }
252 
Load(Arm64ManagedRegister dest,XRegister base,int32_t offset,size_t size)253 void Arm64JNIMacroAssembler::Load(Arm64ManagedRegister dest,
254                                   XRegister base,
255                                   int32_t offset,
256                                   size_t size) {
257   if (dest.IsNoRegister()) {
258     CHECK_EQ(0u, size) << dest;
259   } else if (dest.IsWRegister()) {
260     CHECK_EQ(4u, size) << dest;
261     ___ Ldr(reg_w(dest.AsWRegister()), MEM_OP(reg_x(base), offset));
262   } else if (dest.IsXRegister()) {
263     CHECK_NE(dest.AsXRegister(), SP) << dest;
264 
265     if (size == 1u) {
266       ___ Ldrb(reg_w(dest.AsOverlappingWRegister()), MEM_OP(reg_x(base), offset));
267     } else if (size == 4u) {
268       ___ Ldr(reg_w(dest.AsOverlappingWRegister()), MEM_OP(reg_x(base), offset));
269     }  else {
270       CHECK_EQ(8u, size) << dest;
271       ___ Ldr(reg_x(dest.AsXRegister()), MEM_OP(reg_x(base), offset));
272     }
273   } else if (dest.IsSRegister()) {
274     ___ Ldr(reg_s(dest.AsSRegister()), MEM_OP(reg_x(base), offset));
275   } else {
276     CHECK(dest.IsDRegister()) << dest;
277     ___ Ldr(reg_d(dest.AsDRegister()), MEM_OP(reg_x(base), offset));
278   }
279 }
280 
Load(ManagedRegister m_dst,FrameOffset src,size_t size)281 void Arm64JNIMacroAssembler::Load(ManagedRegister m_dst, FrameOffset src, size_t size) {
282   return Load(m_dst.AsArm64(), SP, src.Int32Value(), size);
283 }
284 
LoadFromThread(ManagedRegister m_dst,ThreadOffset64 src,size_t size)285 void Arm64JNIMacroAssembler::LoadFromThread(ManagedRegister m_dst,
286                                             ThreadOffset64 src,
287                                             size_t size) {
288   return Load(m_dst.AsArm64(), TR, src.Int32Value(), size);
289 }
290 
LoadRef(ManagedRegister m_dst,FrameOffset offs)291 void Arm64JNIMacroAssembler::LoadRef(ManagedRegister m_dst, FrameOffset offs) {
292   Arm64ManagedRegister dst = m_dst.AsArm64();
293   CHECK(dst.IsXRegister()) << dst;
294   LoadWFromOffset(kLoadWord, dst.AsOverlappingWRegister(), SP, offs.Int32Value());
295 }
296 
LoadRef(ManagedRegister m_dst,ManagedRegister m_base,MemberOffset offs,bool unpoison_reference)297 void Arm64JNIMacroAssembler::LoadRef(ManagedRegister m_dst,
298                                      ManagedRegister m_base,
299                                      MemberOffset offs,
300                                      bool unpoison_reference) {
301   Arm64ManagedRegister dst = m_dst.AsArm64();
302   Arm64ManagedRegister base = m_base.AsArm64();
303   CHECK(dst.IsXRegister() && base.IsXRegister());
304   LoadWFromOffset(kLoadWord, dst.AsOverlappingWRegister(), base.AsXRegister(),
305                   offs.Int32Value());
306   if (unpoison_reference) {
307     WRegister ref_reg = dst.AsOverlappingWRegister();
308     asm_.MaybeUnpoisonHeapReference(reg_w(ref_reg));
309   }
310 }
311 
LoadRawPtr(ManagedRegister m_dst,ManagedRegister m_base,Offset offs)312 void Arm64JNIMacroAssembler::LoadRawPtr(ManagedRegister m_dst,
313                                         ManagedRegister m_base,
314                                         Offset offs) {
315   Arm64ManagedRegister dst = m_dst.AsArm64();
316   Arm64ManagedRegister base = m_base.AsArm64();
317   CHECK(dst.IsXRegister() && base.IsXRegister());
318   // Remove dst and base form the temp list - higher level API uses IP1, IP0.
319   UseScratchRegisterScope temps(asm_.GetVIXLAssembler());
320   temps.Exclude(reg_x(dst.AsXRegister()), reg_x(base.AsXRegister()));
321   ___ Ldr(reg_x(dst.AsXRegister()), MEM_OP(reg_x(base.AsXRegister()), offs.Int32Value()));
322 }
323 
LoadRawPtrFromThread(ManagedRegister m_dst,ThreadOffset64 offs)324 void Arm64JNIMacroAssembler::LoadRawPtrFromThread(ManagedRegister m_dst, ThreadOffset64 offs) {
325   Arm64ManagedRegister dst = m_dst.AsArm64();
326   CHECK(dst.IsXRegister()) << dst;
327   LoadFromOffset(dst.AsXRegister(), TR, offs.Int32Value());
328 }
329 
330 // Copying routines.
Move(ManagedRegister m_dst,ManagedRegister m_src,size_t size)331 void Arm64JNIMacroAssembler::Move(ManagedRegister m_dst, ManagedRegister m_src, size_t size) {
332   Arm64ManagedRegister dst = m_dst.AsArm64();
333   Arm64ManagedRegister src = m_src.AsArm64();
334   if (!dst.Equals(src)) {
335     if (dst.IsXRegister()) {
336       if (size == 4) {
337         CHECK(src.IsWRegister());
338         ___ Mov(reg_w(dst.AsOverlappingWRegister()), reg_w(src.AsWRegister()));
339       } else {
340         if (src.IsXRegister()) {
341           ___ Mov(reg_x(dst.AsXRegister()), reg_x(src.AsXRegister()));
342         } else {
343           ___ Mov(reg_x(dst.AsXRegister()), reg_x(src.AsOverlappingXRegister()));
344         }
345       }
346     } else if (dst.IsWRegister()) {
347       CHECK(src.IsWRegister()) << src;
348       ___ Mov(reg_w(dst.AsWRegister()), reg_w(src.AsWRegister()));
349     } else if (dst.IsSRegister()) {
350       CHECK(src.IsSRegister()) << src;
351       ___ Fmov(reg_s(dst.AsSRegister()), reg_s(src.AsSRegister()));
352     } else {
353       CHECK(dst.IsDRegister()) << dst;
354       CHECK(src.IsDRegister()) << src;
355       ___ Fmov(reg_d(dst.AsDRegister()), reg_d(src.AsDRegister()));
356     }
357   }
358 }
359 
CopyRawPtrFromThread(FrameOffset fr_offs,ThreadOffset64 tr_offs,ManagedRegister m_scratch)360 void Arm64JNIMacroAssembler::CopyRawPtrFromThread(FrameOffset fr_offs,
361                                                   ThreadOffset64 tr_offs,
362                                                   ManagedRegister m_scratch) {
363   Arm64ManagedRegister scratch = m_scratch.AsArm64();
364   CHECK(scratch.IsXRegister()) << scratch;
365   LoadFromOffset(scratch.AsXRegister(), TR, tr_offs.Int32Value());
366   StoreToOffset(scratch.AsXRegister(), SP, fr_offs.Int32Value());
367 }
368 
CopyRawPtrToThread(ThreadOffset64 tr_offs,FrameOffset fr_offs,ManagedRegister m_scratch)369 void Arm64JNIMacroAssembler::CopyRawPtrToThread(ThreadOffset64 tr_offs,
370                                                 FrameOffset fr_offs,
371                                                 ManagedRegister m_scratch) {
372   Arm64ManagedRegister scratch = m_scratch.AsArm64();
373   CHECK(scratch.IsXRegister()) << scratch;
374   LoadFromOffset(scratch.AsXRegister(), SP, fr_offs.Int32Value());
375   StoreToOffset(scratch.AsXRegister(), TR, tr_offs.Int32Value());
376 }
377 
CopyRef(FrameOffset dest,FrameOffset src,ManagedRegister m_scratch)378 void Arm64JNIMacroAssembler::CopyRef(FrameOffset dest, FrameOffset src, ManagedRegister m_scratch) {
379   Arm64ManagedRegister scratch = m_scratch.AsArm64();
380   CHECK(scratch.IsXRegister()) << scratch;
381   LoadWFromOffset(kLoadWord, scratch.AsOverlappingWRegister(),
382                   SP, src.Int32Value());
383   StoreWToOffset(kStoreWord, scratch.AsOverlappingWRegister(),
384                  SP, dest.Int32Value());
385 }
386 
Copy(FrameOffset dest,FrameOffset src,ManagedRegister m_scratch,size_t size)387 void Arm64JNIMacroAssembler::Copy(FrameOffset dest,
388                                   FrameOffset src,
389                                   ManagedRegister m_scratch,
390                                   size_t size) {
391   Arm64ManagedRegister scratch = m_scratch.AsArm64();
392   CHECK(scratch.IsXRegister()) << scratch;
393   CHECK(size == 4 || size == 8) << size;
394   if (size == 4) {
395     LoadWFromOffset(kLoadWord, scratch.AsOverlappingWRegister(), SP, src.Int32Value());
396     StoreWToOffset(kStoreWord, scratch.AsOverlappingWRegister(), SP, dest.Int32Value());
397   } else if (size == 8) {
398     LoadFromOffset(scratch.AsXRegister(), SP, src.Int32Value());
399     StoreToOffset(scratch.AsXRegister(), SP, dest.Int32Value());
400   } else {
401     UNIMPLEMENTED(FATAL) << "We only support Copy() of size 4 and 8";
402   }
403 }
404 
Copy(FrameOffset dest,ManagedRegister src_base,Offset src_offset,ManagedRegister m_scratch,size_t size)405 void Arm64JNIMacroAssembler::Copy(FrameOffset dest,
406                                   ManagedRegister src_base,
407                                   Offset src_offset,
408                                   ManagedRegister m_scratch,
409                                   size_t size) {
410   Arm64ManagedRegister scratch = m_scratch.AsArm64();
411   Arm64ManagedRegister base = src_base.AsArm64();
412   CHECK(base.IsXRegister()) << base;
413   CHECK(scratch.IsXRegister() || scratch.IsWRegister()) << scratch;
414   CHECK(size == 4 || size == 8) << size;
415   if (size == 4) {
416     LoadWFromOffset(kLoadWord, scratch.AsWRegister(), base.AsXRegister(),
417                    src_offset.Int32Value());
418     StoreWToOffset(kStoreWord, scratch.AsWRegister(), SP, dest.Int32Value());
419   } else if (size == 8) {
420     LoadFromOffset(scratch.AsXRegister(), base.AsXRegister(), src_offset.Int32Value());
421     StoreToOffset(scratch.AsXRegister(), SP, dest.Int32Value());
422   } else {
423     UNIMPLEMENTED(FATAL) << "We only support Copy() of size 4 and 8";
424   }
425 }
426 
Copy(ManagedRegister m_dest_base,Offset dest_offs,FrameOffset src,ManagedRegister m_scratch,size_t size)427 void Arm64JNIMacroAssembler::Copy(ManagedRegister m_dest_base,
428                                   Offset dest_offs,
429                                   FrameOffset src,
430                                   ManagedRegister m_scratch,
431                                   size_t size) {
432   Arm64ManagedRegister scratch = m_scratch.AsArm64();
433   Arm64ManagedRegister base = m_dest_base.AsArm64();
434   CHECK(base.IsXRegister()) << base;
435   CHECK(scratch.IsXRegister() || scratch.IsWRegister()) << scratch;
436   CHECK(size == 4 || size == 8) << size;
437   if (size == 4) {
438     LoadWFromOffset(kLoadWord, scratch.AsWRegister(), SP, src.Int32Value());
439     StoreWToOffset(kStoreWord, scratch.AsWRegister(), base.AsXRegister(),
440                    dest_offs.Int32Value());
441   } else if (size == 8) {
442     LoadFromOffset(scratch.AsXRegister(), SP, src.Int32Value());
443     StoreToOffset(scratch.AsXRegister(), base.AsXRegister(), dest_offs.Int32Value());
444   } else {
445     UNIMPLEMENTED(FATAL) << "We only support Copy() of size 4 and 8";
446   }
447 }
448 
Copy(FrameOffset,FrameOffset,Offset,ManagedRegister,size_t)449 void Arm64JNIMacroAssembler::Copy(FrameOffset /*dst*/,
450                                   FrameOffset /*src_base*/,
451                                   Offset /*src_offset*/,
452                                   ManagedRegister /*mscratch*/,
453                                   size_t /*size*/) {
454   UNIMPLEMENTED(FATAL) << "Unimplemented Copy() variant";
455 }
456 
Copy(ManagedRegister m_dest,Offset dest_offset,ManagedRegister m_src,Offset src_offset,ManagedRegister m_scratch,size_t size)457 void Arm64JNIMacroAssembler::Copy(ManagedRegister m_dest,
458                                   Offset dest_offset,
459                                   ManagedRegister m_src,
460                                   Offset src_offset,
461                                   ManagedRegister m_scratch,
462                                   size_t size) {
463   Arm64ManagedRegister scratch = m_scratch.AsArm64();
464   Arm64ManagedRegister src = m_src.AsArm64();
465   Arm64ManagedRegister dest = m_dest.AsArm64();
466   CHECK(dest.IsXRegister()) << dest;
467   CHECK(src.IsXRegister()) << src;
468   CHECK(scratch.IsXRegister() || scratch.IsWRegister()) << scratch;
469   CHECK(size == 4 || size == 8) << size;
470   if (size == 4) {
471     if (scratch.IsWRegister()) {
472       LoadWFromOffset(kLoadWord, scratch.AsWRegister(), src.AsXRegister(),
473                     src_offset.Int32Value());
474       StoreWToOffset(kStoreWord, scratch.AsWRegister(), dest.AsXRegister(),
475                    dest_offset.Int32Value());
476     } else {
477       LoadWFromOffset(kLoadWord, scratch.AsOverlappingWRegister(), src.AsXRegister(),
478                     src_offset.Int32Value());
479       StoreWToOffset(kStoreWord, scratch.AsOverlappingWRegister(), dest.AsXRegister(),
480                    dest_offset.Int32Value());
481     }
482   } else if (size == 8) {
483     LoadFromOffset(scratch.AsXRegister(), src.AsXRegister(), src_offset.Int32Value());
484     StoreToOffset(scratch.AsXRegister(), dest.AsXRegister(), dest_offset.Int32Value());
485   } else {
486     UNIMPLEMENTED(FATAL) << "We only support Copy() of size 4 and 8";
487   }
488 }
489 
Copy(FrameOffset,Offset,FrameOffset,Offset,ManagedRegister,size_t)490 void Arm64JNIMacroAssembler::Copy(FrameOffset /*dst*/,
491                                   Offset /*dest_offset*/,
492                                   FrameOffset /*src*/,
493                                   Offset /*src_offset*/,
494                                   ManagedRegister /*scratch*/,
495                                   size_t /*size*/) {
496   UNIMPLEMENTED(FATAL) << "Unimplemented Copy() variant";
497 }
498 
MemoryBarrier(ManagedRegister m_scratch ATTRIBUTE_UNUSED)499 void Arm64JNIMacroAssembler::MemoryBarrier(ManagedRegister m_scratch ATTRIBUTE_UNUSED) {
500   // TODO: Should we check that m_scratch is IP? - see arm.
501   ___ Dmb(InnerShareable, BarrierAll);
502 }
503 
SignExtend(ManagedRegister mreg,size_t size)504 void Arm64JNIMacroAssembler::SignExtend(ManagedRegister mreg, size_t size) {
505   Arm64ManagedRegister reg = mreg.AsArm64();
506   CHECK(size == 1 || size == 2) << size;
507   CHECK(reg.IsWRegister()) << reg;
508   if (size == 1) {
509     ___ Sxtb(reg_w(reg.AsWRegister()), reg_w(reg.AsWRegister()));
510   } else {
511     ___ Sxth(reg_w(reg.AsWRegister()), reg_w(reg.AsWRegister()));
512   }
513 }
514 
ZeroExtend(ManagedRegister mreg,size_t size)515 void Arm64JNIMacroAssembler::ZeroExtend(ManagedRegister mreg, size_t size) {
516   Arm64ManagedRegister reg = mreg.AsArm64();
517   CHECK(size == 1 || size == 2) << size;
518   CHECK(reg.IsWRegister()) << reg;
519   if (size == 1) {
520     ___ Uxtb(reg_w(reg.AsWRegister()), reg_w(reg.AsWRegister()));
521   } else {
522     ___ Uxth(reg_w(reg.AsWRegister()), reg_w(reg.AsWRegister()));
523   }
524 }
525 
VerifyObject(ManagedRegister,bool)526 void Arm64JNIMacroAssembler::VerifyObject(ManagedRegister /*src*/, bool /*could_be_null*/) {
527   // TODO: not validating references.
528 }
529 
VerifyObject(FrameOffset,bool)530 void Arm64JNIMacroAssembler::VerifyObject(FrameOffset /*src*/, bool /*could_be_null*/) {
531   // TODO: not validating references.
532 }
533 
Call(ManagedRegister m_base,Offset offs,ManagedRegister m_scratch)534 void Arm64JNIMacroAssembler::Call(ManagedRegister m_base, Offset offs, ManagedRegister m_scratch) {
535   Arm64ManagedRegister base = m_base.AsArm64();
536   Arm64ManagedRegister scratch = m_scratch.AsArm64();
537   CHECK(base.IsXRegister()) << base;
538   CHECK(scratch.IsXRegister()) << scratch;
539   LoadFromOffset(scratch.AsXRegister(), base.AsXRegister(), offs.Int32Value());
540   ___ Blr(reg_x(scratch.AsXRegister()));
541 }
542 
Call(FrameOffset base,Offset offs,ManagedRegister m_scratch)543 void Arm64JNIMacroAssembler::Call(FrameOffset base, Offset offs, ManagedRegister m_scratch) {
544   Arm64ManagedRegister scratch = m_scratch.AsArm64();
545   CHECK(scratch.IsXRegister()) << scratch;
546   // Call *(*(SP + base) + offset)
547   LoadFromOffset(scratch.AsXRegister(), SP, base.Int32Value());
548   LoadFromOffset(scratch.AsXRegister(), scratch.AsXRegister(), offs.Int32Value());
549   ___ Blr(reg_x(scratch.AsXRegister()));
550 }
551 
CallFromThread(ThreadOffset64 offset ATTRIBUTE_UNUSED,ManagedRegister scratch ATTRIBUTE_UNUSED)552 void Arm64JNIMacroAssembler::CallFromThread(ThreadOffset64 offset ATTRIBUTE_UNUSED,
553                                             ManagedRegister scratch ATTRIBUTE_UNUSED) {
554   UNIMPLEMENTED(FATAL) << "Unimplemented Call() variant";
555 }
556 
CreateHandleScopeEntry(ManagedRegister m_out_reg,FrameOffset handle_scope_offs,ManagedRegister m_in_reg,bool null_allowed)557 void Arm64JNIMacroAssembler::CreateHandleScopeEntry(ManagedRegister m_out_reg,
558                                                     FrameOffset handle_scope_offs,
559                                                     ManagedRegister m_in_reg,
560                                                     bool null_allowed) {
561   Arm64ManagedRegister out_reg = m_out_reg.AsArm64();
562   Arm64ManagedRegister in_reg = m_in_reg.AsArm64();
563   // For now we only hold stale handle scope entries in x registers.
564   CHECK(in_reg.IsNoRegister() || in_reg.IsXRegister()) << in_reg;
565   CHECK(out_reg.IsXRegister()) << out_reg;
566   if (null_allowed) {
567     // Null values get a handle scope entry value of 0.  Otherwise, the handle scope entry is
568     // the address in the handle scope holding the reference.
569     // e.g. out_reg = (handle == 0) ? 0 : (SP+handle_offset)
570     if (in_reg.IsNoRegister()) {
571       LoadWFromOffset(kLoadWord, out_reg.AsOverlappingWRegister(), SP,
572                       handle_scope_offs.Int32Value());
573       in_reg = out_reg;
574     }
575     ___ Cmp(reg_w(in_reg.AsOverlappingWRegister()), 0);
576     if (!out_reg.Equals(in_reg)) {
577       LoadImmediate(out_reg.AsXRegister(), 0, eq);
578     }
579     AddConstant(out_reg.AsXRegister(), SP, handle_scope_offs.Int32Value(), ne);
580   } else {
581     AddConstant(out_reg.AsXRegister(), SP, handle_scope_offs.Int32Value(), al);
582   }
583 }
584 
CreateHandleScopeEntry(FrameOffset out_off,FrameOffset handle_scope_offset,ManagedRegister m_scratch,bool null_allowed)585 void Arm64JNIMacroAssembler::CreateHandleScopeEntry(FrameOffset out_off,
586                                                     FrameOffset handle_scope_offset,
587                                                     ManagedRegister m_scratch,
588                                                     bool null_allowed) {
589   Arm64ManagedRegister scratch = m_scratch.AsArm64();
590   CHECK(scratch.IsXRegister()) << scratch;
591   if (null_allowed) {
592     LoadWFromOffset(kLoadWord, scratch.AsOverlappingWRegister(), SP,
593                     handle_scope_offset.Int32Value());
594     // Null values get a handle scope entry value of 0.  Otherwise, the handle scope entry is
595     // the address in the handle scope holding the reference.
596     // e.g. scratch = (scratch == 0) ? 0 : (SP+handle_scope_offset)
597     ___ Cmp(reg_w(scratch.AsOverlappingWRegister()), 0);
598     // Move this logic in add constants with flags.
599     AddConstant(scratch.AsXRegister(), SP, handle_scope_offset.Int32Value(), ne);
600   } else {
601     AddConstant(scratch.AsXRegister(), SP, handle_scope_offset.Int32Value(), al);
602   }
603   StoreToOffset(scratch.AsXRegister(), SP, out_off.Int32Value());
604 }
605 
LoadReferenceFromHandleScope(ManagedRegister m_out_reg,ManagedRegister m_in_reg)606 void Arm64JNIMacroAssembler::LoadReferenceFromHandleScope(ManagedRegister m_out_reg,
607                                                           ManagedRegister m_in_reg) {
608   Arm64ManagedRegister out_reg = m_out_reg.AsArm64();
609   Arm64ManagedRegister in_reg = m_in_reg.AsArm64();
610   CHECK(out_reg.IsXRegister()) << out_reg;
611   CHECK(in_reg.IsXRegister()) << in_reg;
612   vixl::aarch64::Label exit;
613   if (!out_reg.Equals(in_reg)) {
614     // FIXME: Who sets the flags here?
615     LoadImmediate(out_reg.AsXRegister(), 0, eq);
616   }
617   ___ Cbz(reg_x(in_reg.AsXRegister()), &exit);
618   LoadFromOffset(out_reg.AsXRegister(), in_reg.AsXRegister(), 0);
619   ___ Bind(&exit);
620 }
621 
ExceptionPoll(ManagedRegister m_scratch,size_t stack_adjust)622 void Arm64JNIMacroAssembler::ExceptionPoll(ManagedRegister m_scratch, size_t stack_adjust) {
623   CHECK_ALIGNED(stack_adjust, kStackAlignment);
624   Arm64ManagedRegister scratch = m_scratch.AsArm64();
625   exception_blocks_.emplace_back(new Arm64Exception(scratch, stack_adjust));
626   LoadFromOffset(scratch.AsXRegister(),
627                  TR,
628                  Thread::ExceptionOffset<kArm64PointerSize>().Int32Value());
629   ___ Cbnz(reg_x(scratch.AsXRegister()), exception_blocks_.back()->Entry());
630 }
631 
CreateLabel()632 std::unique_ptr<JNIMacroLabel> Arm64JNIMacroAssembler::CreateLabel() {
633   return std::unique_ptr<JNIMacroLabel>(new Arm64JNIMacroLabel());
634 }
635 
Jump(JNIMacroLabel * label)636 void Arm64JNIMacroAssembler::Jump(JNIMacroLabel* label) {
637   CHECK(label != nullptr);
638   ___ B(Arm64JNIMacroLabel::Cast(label)->AsArm64());
639 }
640 
Jump(JNIMacroLabel * label,JNIMacroUnaryCondition condition,ManagedRegister test)641 void Arm64JNIMacroAssembler::Jump(JNIMacroLabel* label,
642                                   JNIMacroUnaryCondition condition,
643                                   ManagedRegister test) {
644   CHECK(label != nullptr);
645 
646   switch (condition) {
647     case JNIMacroUnaryCondition::kZero:
648       ___ Cbz(reg_x(test.AsArm64().AsXRegister()), Arm64JNIMacroLabel::Cast(label)->AsArm64());
649       break;
650     case JNIMacroUnaryCondition::kNotZero:
651       ___ Cbnz(reg_x(test.AsArm64().AsXRegister()), Arm64JNIMacroLabel::Cast(label)->AsArm64());
652       break;
653     default:
654       LOG(FATAL) << "Not implemented unary condition: " << static_cast<int>(condition);
655       UNREACHABLE();
656   }
657 }
658 
Bind(JNIMacroLabel * label)659 void Arm64JNIMacroAssembler::Bind(JNIMacroLabel* label) {
660   CHECK(label != nullptr);
661   ___ Bind(Arm64JNIMacroLabel::Cast(label)->AsArm64());
662 }
663 
EmitExceptionPoll(Arm64Exception * exception)664 void Arm64JNIMacroAssembler::EmitExceptionPoll(Arm64Exception* exception) {
665   UseScratchRegisterScope temps(asm_.GetVIXLAssembler());
666   temps.Exclude(reg_x(exception->scratch_.AsXRegister()));
667   Register temp = temps.AcquireX();
668 
669   // Bind exception poll entry.
670   ___ Bind(exception->Entry());
671   if (exception->stack_adjust_ != 0) {  // Fix up the frame.
672     DecreaseFrameSize(exception->stack_adjust_);
673   }
674   // Pass exception object as argument.
675   // Don't care about preserving X0 as this won't return.
676   ___ Mov(reg_x(X0), reg_x(exception->scratch_.AsXRegister()));
677   ___ Ldr(temp,
678           MEM_OP(reg_x(TR),
679                  QUICK_ENTRYPOINT_OFFSET(kArm64PointerSize, pDeliverException).Int32Value()));
680 
681   ___ Blr(temp);
682   // Call should never return.
683   ___ Brk();
684 }
685 
BuildFrame(size_t frame_size,ManagedRegister method_reg,ArrayRef<const ManagedRegister> callee_save_regs,const ManagedRegisterEntrySpills & entry_spills)686 void Arm64JNIMacroAssembler::BuildFrame(size_t frame_size,
687                                         ManagedRegister method_reg,
688                                         ArrayRef<const ManagedRegister> callee_save_regs,
689                                         const ManagedRegisterEntrySpills& entry_spills) {
690   // Setup VIXL CPURegList for callee-saves.
691   CPURegList core_reg_list(CPURegister::kRegister, kXRegSize, 0);
692   CPURegList fp_reg_list(CPURegister::kFPRegister, kDRegSize, 0);
693   for (auto r : callee_save_regs) {
694     Arm64ManagedRegister reg = r.AsArm64();
695     if (reg.IsXRegister()) {
696       core_reg_list.Combine(reg_x(reg.AsXRegister()).GetCode());
697     } else {
698       DCHECK(reg.IsDRegister());
699       fp_reg_list.Combine(reg_d(reg.AsDRegister()).GetCode());
700     }
701   }
702   size_t core_reg_size = core_reg_list.GetTotalSizeInBytes();
703   size_t fp_reg_size = fp_reg_list.GetTotalSizeInBytes();
704 
705   // Increase frame to required size.
706   DCHECK_ALIGNED(frame_size, kStackAlignment);
707   DCHECK_GE(frame_size, core_reg_size + fp_reg_size + static_cast<size_t>(kArm64PointerSize));
708   IncreaseFrameSize(frame_size);
709 
710   // Save callee-saves.
711   asm_.SpillRegisters(core_reg_list, frame_size - core_reg_size);
712   asm_.SpillRegisters(fp_reg_list, frame_size - core_reg_size - fp_reg_size);
713 
714   DCHECK(core_reg_list.IncludesAliasOf(reg_x(TR)));
715 
716   // Write ArtMethod*
717   DCHECK(X0 == method_reg.AsArm64().AsXRegister());
718   StoreToOffset(X0, SP, 0);
719 
720   // Write out entry spills
721   int32_t offset = frame_size + static_cast<size_t>(kArm64PointerSize);
722   for (size_t i = 0; i < entry_spills.size(); ++i) {
723     Arm64ManagedRegister reg = entry_spills.at(i).AsArm64();
724     if (reg.IsNoRegister()) {
725       // only increment stack offset.
726       ManagedRegisterSpill spill = entry_spills.at(i);
727       offset += spill.getSize();
728     } else if (reg.IsXRegister()) {
729       StoreToOffset(reg.AsXRegister(), SP, offset);
730       offset += 8;
731     } else if (reg.IsWRegister()) {
732       StoreWToOffset(kStoreWord, reg.AsWRegister(), SP, offset);
733       offset += 4;
734     } else if (reg.IsDRegister()) {
735       StoreDToOffset(reg.AsDRegister(), SP, offset);
736       offset += 8;
737     } else if (reg.IsSRegister()) {
738       StoreSToOffset(reg.AsSRegister(), SP, offset);
739       offset += 4;
740     }
741   }
742 }
743 
RemoveFrame(size_t frame_size,ArrayRef<const ManagedRegister> callee_save_regs,bool may_suspend)744 void Arm64JNIMacroAssembler::RemoveFrame(size_t frame_size,
745                                          ArrayRef<const ManagedRegister> callee_save_regs,
746                                          bool may_suspend) {
747   // Setup VIXL CPURegList for callee-saves.
748   CPURegList core_reg_list(CPURegister::kRegister, kXRegSize, 0);
749   CPURegList fp_reg_list(CPURegister::kFPRegister, kDRegSize, 0);
750   for (auto r : callee_save_regs) {
751     Arm64ManagedRegister reg = r.AsArm64();
752     if (reg.IsXRegister()) {
753       core_reg_list.Combine(reg_x(reg.AsXRegister()).GetCode());
754     } else {
755       DCHECK(reg.IsDRegister());
756       fp_reg_list.Combine(reg_d(reg.AsDRegister()).GetCode());
757     }
758   }
759   size_t core_reg_size = core_reg_list.GetTotalSizeInBytes();
760   size_t fp_reg_size = fp_reg_list.GetTotalSizeInBytes();
761 
762   // For now we only check that the size of the frame is large enough to hold spills and method
763   // reference.
764   DCHECK_GE(frame_size, core_reg_size + fp_reg_size + static_cast<size_t>(kArm64PointerSize));
765   DCHECK_ALIGNED(frame_size, kStackAlignment);
766 
767   DCHECK(core_reg_list.IncludesAliasOf(reg_x(TR)));
768 
769   cfi().RememberState();
770 
771   // Restore callee-saves.
772   asm_.UnspillRegisters(core_reg_list, frame_size - core_reg_size);
773   asm_.UnspillRegisters(fp_reg_list, frame_size - core_reg_size - fp_reg_size);
774 
775   if (kEmitCompilerReadBarrier && kUseBakerReadBarrier) {
776     vixl::aarch64::Register mr = reg_x(MR);  // Marking Register.
777     vixl::aarch64::Register tr = reg_x(TR);  // Thread Register.
778 
779     if (may_suspend) {
780       // The method may be suspended; refresh the Marking Register.
781       ___ Ldr(mr.W(), MemOperand(tr, Thread::IsGcMarkingOffset<kArm64PointerSize>().Int32Value()));
782     } else {
783       // The method shall not be suspended; no need to refresh the Marking Register.
784 
785       // Check that the Marking Register is a callee-save register,
786       // and thus has been preserved by native code following the
787       // AAPCS64 calling convention.
788       DCHECK(core_reg_list.IncludesAliasOf(mr))
789           << "core_reg_list should contain Marking Register X" << mr.GetCode();
790 
791       // The following condition is a compile-time one, so it does not have a run-time cost.
792       if (kIsDebugBuild) {
793         // The following condition is a run-time one; it is executed after the
794         // previous compile-time test, to avoid penalizing non-debug builds.
795         if (emit_run_time_checks_in_debug_mode_) {
796           // Emit a run-time check verifying that the Marking Register is up-to-date.
797           UseScratchRegisterScope temps(asm_.GetVIXLAssembler());
798           Register temp = temps.AcquireW();
799           // Ensure we are not clobbering a callee-save register that was restored before.
800           DCHECK(!core_reg_list.IncludesAliasOf(temp.X()))
801               << "core_reg_list should not contain scratch register X" << temp.GetCode();
802           asm_.GenerateMarkingRegisterCheck(temp);
803         }
804       }
805     }
806   }
807 
808   // Decrease frame size to start of callee saved regs.
809   DecreaseFrameSize(frame_size);
810 
811   // Return to LR.
812   ___ Ret();
813 
814   // The CFI should be restored for any code that follows the exit block.
815   cfi().RestoreState();
816   cfi().DefCFAOffset(frame_size);
817 }
818 
819 #undef ___
820 
821 }  // namespace arm64
822 }  // namespace art
823