1 /*
2 * Copyright (C) 2017 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 "interpreter/interpreter_intrinsics.h"
18
19 #include "dex/dex_instruction.h"
20 #include "intrinsics_enum.h"
21 #include "interpreter/interpreter_common.h"
22
23 namespace art {
24 namespace interpreter {
25
26
27 #define BINARY_INTRINSIC(name, op, get1, get2, set) \
28 static ALWAYS_INLINE bool name(ShadowFrame* shadow_frame, \
29 const Instruction* inst, \
30 uint16_t inst_data, \
31 JValue* result_register) \
32 REQUIRES_SHARED(Locks::mutator_lock_) { \
33 uint32_t arg[Instruction::kMaxVarArgRegs] = {}; \
34 inst->GetVarArgs(arg, inst_data); \
35 result_register->set(op(shadow_frame->get1, shadow_frame->get2)); \
36 return true; \
37 }
38
39 #define BINARY_II_INTRINSIC(name, op, set) \
40 BINARY_INTRINSIC(name, op, GetVReg(arg[0]), GetVReg(arg[1]), set)
41
42 #define BINARY_JJ_INTRINSIC(name, op, set) \
43 BINARY_INTRINSIC(name, op, GetVRegLong(arg[0]), GetVRegLong(arg[2]), set)
44
45 #define BINARY_JI_INTRINSIC(name, op, set) \
46 BINARY_INTRINSIC(name, op, GetVRegLong(arg[0]), GetVReg(arg[2]), set)
47
48 #define UNARY_INTRINSIC(name, op, get, set) \
49 static ALWAYS_INLINE bool name(ShadowFrame* shadow_frame, \
50 const Instruction* inst, \
51 uint16_t inst_data, \
52 JValue* result_register) \
53 REQUIRES_SHARED(Locks::mutator_lock_) { \
54 uint32_t arg[Instruction::kMaxVarArgRegs] = {}; \
55 inst->GetVarArgs(arg, inst_data); \
56 result_register->set(op(shadow_frame->get(arg[0]))); \
57 return true; \
58 }
59
60
61 // java.lang.Integer.reverse(I)I
62 UNARY_INTRINSIC(MterpIntegerReverse, ReverseBits32, GetVReg, SetI);
63
64 // java.lang.Integer.reverseBytes(I)I
65 UNARY_INTRINSIC(MterpIntegerReverseBytes, BSWAP, GetVReg, SetI);
66
67 // java.lang.Integer.bitCount(I)I
68 UNARY_INTRINSIC(MterpIntegerBitCount, POPCOUNT, GetVReg, SetI);
69
70 // java.lang.Integer.compare(II)I
71 BINARY_II_INTRINSIC(MterpIntegerCompare, Compare, SetI);
72
73 // java.lang.Integer.highestOneBit(I)I
74 UNARY_INTRINSIC(MterpIntegerHighestOneBit, HighestOneBitValue, GetVReg, SetI);
75
76 // java.lang.Integer.LowestOneBit(I)I
77 UNARY_INTRINSIC(MterpIntegerLowestOneBit, LowestOneBitValue, GetVReg, SetI);
78
79 // java.lang.Integer.numberOfLeadingZeros(I)I
80 UNARY_INTRINSIC(MterpIntegerNumberOfLeadingZeros, JAVASTYLE_CLZ, GetVReg, SetI);
81
82 // java.lang.Integer.numberOfTrailingZeros(I)I
83 UNARY_INTRINSIC(MterpIntegerNumberOfTrailingZeros, JAVASTYLE_CTZ, GetVReg, SetI);
84
85 // java.lang.Integer.rotateRight(II)I
86 BINARY_II_INTRINSIC(MterpIntegerRotateRight, (Rot<int32_t, false>), SetI);
87
88 // java.lang.Integer.rotateLeft(II)I
89 BINARY_II_INTRINSIC(MterpIntegerRotateLeft, (Rot<int32_t, true>), SetI);
90
91 // java.lang.Integer.signum(I)I
92 UNARY_INTRINSIC(MterpIntegerSignum, Signum, GetVReg, SetI);
93
94 // java.lang.Long.reverse(I)I
95 UNARY_INTRINSIC(MterpLongReverse, ReverseBits64, GetVRegLong, SetJ);
96
97 // java.lang.Long.reverseBytes(J)J
98 UNARY_INTRINSIC(MterpLongReverseBytes, BSWAP, GetVRegLong, SetJ);
99
100 // java.lang.Long.bitCount(J)I
101 UNARY_INTRINSIC(MterpLongBitCount, POPCOUNT, GetVRegLong, SetI);
102
103 // java.lang.Long.compare(JJ)I
104 BINARY_JJ_INTRINSIC(MterpLongCompare, Compare, SetI);
105
106 // java.lang.Long.highestOneBit(J)J
107 UNARY_INTRINSIC(MterpLongHighestOneBit, HighestOneBitValue, GetVRegLong, SetJ);
108
109 // java.lang.Long.lowestOneBit(J)J
110 UNARY_INTRINSIC(MterpLongLowestOneBit, LowestOneBitValue, GetVRegLong, SetJ);
111
112 // java.lang.Long.numberOfLeadingZeros(J)I
113 UNARY_INTRINSIC(MterpLongNumberOfLeadingZeros, JAVASTYLE_CLZ, GetVRegLong, SetJ);
114
115 // java.lang.Long.numberOfTrailingZeros(J)I
116 UNARY_INTRINSIC(MterpLongNumberOfTrailingZeros, JAVASTYLE_CTZ, GetVRegLong, SetJ);
117
118 // java.lang.Long.rotateRight(JI)J
119 BINARY_JJ_INTRINSIC(MterpLongRotateRight, (Rot<int64_t, false>), SetJ);
120
121 // java.lang.Long.rotateLeft(JI)J
122 BINARY_JJ_INTRINSIC(MterpLongRotateLeft, (Rot<int64_t, true>), SetJ);
123
124 // java.lang.Long.signum(J)I
125 UNARY_INTRINSIC(MterpLongSignum, Signum, GetVRegLong, SetI);
126
127 // java.lang.Short.reverseBytes(S)S
128 UNARY_INTRINSIC(MterpShortReverseBytes, BSWAP, GetVRegShort, SetS);
129
130 // java.lang.Math.min(II)I
131 BINARY_II_INTRINSIC(MterpMathMinIntInt, std::min, SetI);
132
133 // java.lang.Math.min(JJ)J
134 BINARY_JJ_INTRINSIC(MterpMathMinLongLong, std::min, SetJ);
135
136 // java.lang.Math.max(II)I
137 BINARY_II_INTRINSIC(MterpMathMaxIntInt, std::max, SetI);
138
139 // java.lang.Math.max(JJ)J
140 BINARY_JJ_INTRINSIC(MterpMathMaxLongLong, std::max, SetJ);
141
142 // java.lang.Math.abs(I)I
143 UNARY_INTRINSIC(MterpMathAbsInt, std::abs, GetVReg, SetI);
144
145 // java.lang.Math.abs(J)J
146 UNARY_INTRINSIC(MterpMathAbsLong, std::abs, GetVRegLong, SetJ);
147
148 // java.lang.Math.abs(F)F
149 UNARY_INTRINSIC(MterpMathAbsFloat, 0x7fffffff&, GetVReg, SetI);
150
151 // java.lang.Math.abs(D)D
152 UNARY_INTRINSIC(MterpMathAbsDouble, INT64_C(0x7fffffffffffffff)&, GetVRegLong, SetJ);
153
154 // java.lang.Math.sqrt(D)D
155 UNARY_INTRINSIC(MterpMathSqrt, std::sqrt, GetVRegDouble, SetD);
156
157 // java.lang.Math.ceil(D)D
158 UNARY_INTRINSIC(MterpMathCeil, std::ceil, GetVRegDouble, SetD);
159
160 // java.lang.Math.floor(D)D
161 UNARY_INTRINSIC(MterpMathFloor, std::floor, GetVRegDouble, SetD);
162
163 // java.lang.Math.sin(D)D
164 UNARY_INTRINSIC(MterpMathSin, std::sin, GetVRegDouble, SetD);
165
166 // java.lang.Math.cos(D)D
167 UNARY_INTRINSIC(MterpMathCos, std::cos, GetVRegDouble, SetD);
168
169 // java.lang.Math.tan(D)D
170 UNARY_INTRINSIC(MterpMathTan, std::tan, GetVRegDouble, SetD);
171
172 // java.lang.Math.asin(D)D
173 UNARY_INTRINSIC(MterpMathAsin, std::asin, GetVRegDouble, SetD);
174
175 // java.lang.Math.acos(D)D
176 UNARY_INTRINSIC(MterpMathAcos, std::acos, GetVRegDouble, SetD);
177
178 // java.lang.Math.atan(D)D
179 UNARY_INTRINSIC(MterpMathAtan, std::atan, GetVRegDouble, SetD);
180
181 // java.lang.String.charAt(I)C
MterpStringCharAt(ShadowFrame * shadow_frame,const Instruction * inst,uint16_t inst_data,JValue * result_register)182 static ALWAYS_INLINE bool MterpStringCharAt(ShadowFrame* shadow_frame,
183 const Instruction* inst,
184 uint16_t inst_data,
185 JValue* result_register)
186 REQUIRES_SHARED(Locks::mutator_lock_) {
187 uint32_t arg[Instruction::kMaxVarArgRegs] = {};
188 inst->GetVarArgs(arg, inst_data);
189 mirror::String* str = shadow_frame->GetVRegReference(arg[0])->AsString();
190 int length = str->GetLength();
191 int index = shadow_frame->GetVReg(arg[1]);
192 uint16_t res;
193 if (UNLIKELY(index < 0) || (index >= length)) {
194 return false; // Punt and let non-intrinsic version deal with the throw.
195 }
196 if (str->IsCompressed()) {
197 res = str->GetValueCompressed()[index];
198 } else {
199 res = str->GetValue()[index];
200 }
201 result_register->SetC(res);
202 return true;
203 }
204
205 // java.lang.String.compareTo(Ljava/lang/string)I
MterpStringCompareTo(ShadowFrame * shadow_frame,const Instruction * inst,uint16_t inst_data,JValue * result_register)206 static ALWAYS_INLINE bool MterpStringCompareTo(ShadowFrame* shadow_frame,
207 const Instruction* inst,
208 uint16_t inst_data,
209 JValue* result_register)
210 REQUIRES_SHARED(Locks::mutator_lock_) {
211 uint32_t arg[Instruction::kMaxVarArgRegs] = {};
212 inst->GetVarArgs(arg, inst_data);
213 mirror::String* str = shadow_frame->GetVRegReference(arg[0])->AsString();
214 mirror::Object* arg1 = shadow_frame->GetVRegReference(arg[1]);
215 if (arg1 == nullptr) {
216 return false;
217 }
218 result_register->SetI(str->CompareTo(arg1->AsString()));
219 return true;
220 }
221
222 #define STRING_INDEXOF_INTRINSIC(name, starting_pos) \
223 static ALWAYS_INLINE bool Mterp##name(ShadowFrame* shadow_frame, \
224 const Instruction* inst, \
225 uint16_t inst_data, \
226 JValue* result_register) \
227 REQUIRES_SHARED(Locks::mutator_lock_) { \
228 uint32_t arg[Instruction::kMaxVarArgRegs] = {}; \
229 inst->GetVarArgs(arg, inst_data); \
230 mirror::String* str = shadow_frame->GetVRegReference(arg[0])->AsString(); \
231 int ch = shadow_frame->GetVReg(arg[1]); \
232 if (ch >= 0x10000) { \
233 /* Punt if supplementary char. */ \
234 return false; \
235 } \
236 result_register->SetI(str->FastIndexOf(ch, starting_pos)); \
237 return true; \
238 }
239
240 // java.lang.String.indexOf(I)I
241 STRING_INDEXOF_INTRINSIC(StringIndexOf, 0);
242
243 // java.lang.String.indexOf(II)I
244 STRING_INDEXOF_INTRINSIC(StringIndexOfAfter, shadow_frame->GetVReg(arg[2]));
245
246 #define SIMPLE_STRING_INTRINSIC(name, operation) \
247 static ALWAYS_INLINE bool Mterp##name(ShadowFrame* shadow_frame, \
248 const Instruction* inst, \
249 uint16_t inst_data, \
250 JValue* result_register) \
251 REQUIRES_SHARED(Locks::mutator_lock_) { \
252 uint32_t arg[Instruction::kMaxVarArgRegs] = {}; \
253 inst->GetVarArgs(arg, inst_data); \
254 mirror::String* str = shadow_frame->GetVRegReference(arg[0])->AsString(); \
255 result_register->operation; \
256 return true; \
257 }
258
259 // java.lang.String.isEmpty()Z
260 SIMPLE_STRING_INTRINSIC(StringIsEmpty, SetZ(str->GetLength() == 0))
261
262 // java.lang.String.length()I
263 SIMPLE_STRING_INTRINSIC(StringLength, SetI(str->GetLength()))
264
265 // java.lang.String.getCharsNoCheck(II[CI)V
MterpStringGetCharsNoCheck(ShadowFrame * shadow_frame,const Instruction * inst,uint16_t inst_data,JValue * result_register ATTRIBUTE_UNUSED)266 static ALWAYS_INLINE bool MterpStringGetCharsNoCheck(ShadowFrame* shadow_frame,
267 const Instruction* inst,
268 uint16_t inst_data,
269 JValue* result_register ATTRIBUTE_UNUSED)
270 REQUIRES_SHARED(Locks::mutator_lock_) {
271 // Start, end & index already checked by caller - won't throw. Destination is uncompressed.
272 uint32_t arg[Instruction::kMaxVarArgRegs] = {};
273 inst->GetVarArgs(arg, inst_data);
274 mirror::String* str = shadow_frame->GetVRegReference(arg[0])->AsString();
275 int32_t start = shadow_frame->GetVReg(arg[1]);
276 int32_t end = shadow_frame->GetVReg(arg[2]);
277 int32_t index = shadow_frame->GetVReg(arg[4]);
278 mirror::CharArray* array = shadow_frame->GetVRegReference(arg[3])->AsCharArray();
279 uint16_t* dst = array->GetData() + index;
280 int32_t len = (end - start);
281 if (str->IsCompressed()) {
282 const uint8_t* src_8 = str->GetValueCompressed() + start;
283 for (int i = 0; i < len; i++) {
284 dst[i] = src_8[i];
285 }
286 } else {
287 uint16_t* src_16 = str->GetValue() + start;
288 memcpy(dst, src_16, len * sizeof(uint16_t));
289 }
290 return true;
291 }
292
293 // java.lang.String.equalsLjava/lang/Object;)Z
MterpStringEquals(ShadowFrame * shadow_frame,const Instruction * inst,uint16_t inst_data,JValue * result_register)294 static ALWAYS_INLINE bool MterpStringEquals(ShadowFrame* shadow_frame,
295 const Instruction* inst,
296 uint16_t inst_data,
297 JValue* result_register)
298 REQUIRES_SHARED(Locks::mutator_lock_) {
299 uint32_t arg[Instruction::kMaxVarArgRegs] = {};
300 inst->GetVarArgs(arg, inst_data);
301 mirror::String* str = shadow_frame->GetVRegReference(arg[0])->AsString();
302 mirror::Object* obj = shadow_frame->GetVRegReference(arg[1]);
303 bool res = false; // Assume not equal.
304 if ((obj != nullptr) && obj->IsString()) {
305 mirror::String* str2 = obj->AsString();
306 if (str->GetCount() == str2->GetCount()) {
307 // Length & compression status are same. Can use block compare.
308 void* bytes1;
309 void* bytes2;
310 int len = str->GetLength();
311 if (str->IsCompressed()) {
312 bytes1 = str->GetValueCompressed();
313 bytes2 = str2->GetValueCompressed();
314 } else {
315 len *= sizeof(uint16_t);
316 bytes1 = str->GetValue();
317 bytes2 = str2->GetValue();
318 }
319 res = (memcmp(bytes1, bytes2, len) == 0);
320 }
321 }
322 result_register->SetZ(res);
323 return true;
324 }
325
326 #define VARHANDLE_FENCE_INTRINSIC(name, std_memory_operation) \
327 static ALWAYS_INLINE bool name(ShadowFrame* shadow_frame ATTRIBUTE_UNUSED, \
328 const Instruction* inst ATTRIBUTE_UNUSED, \
329 uint16_t inst_data ATTRIBUTE_UNUSED, \
330 JValue* result_register ATTRIBUTE_UNUSED) \
331 REQUIRES_SHARED(Locks::mutator_lock_) { \
332 std::atomic_thread_fence(std_memory_operation); \
333 return true; \
334 }
335
336 // The VarHandle fence methods are static (unlike sun.misc.Unsafe versions).
337 // The fences for the LoadLoadFence and StoreStoreFence are stronger
338 // than strictly required, but the impact should be marginal.
339 VARHANDLE_FENCE_INTRINSIC(MterpVarHandleFullFence, std::memory_order_seq_cst)
340 VARHANDLE_FENCE_INTRINSIC(MterpVarHandleAcquireFence, std::memory_order_acquire)
341 VARHANDLE_FENCE_INTRINSIC(MterpVarHandleReleaseFence, std::memory_order_release)
342 VARHANDLE_FENCE_INTRINSIC(MterpVarHandleLoadLoadFence, std::memory_order_acquire)
343 VARHANDLE_FENCE_INTRINSIC(MterpVarHandleStoreStoreFence, std::memory_order_release)
344
345 #define METHOD_HANDLE_INVOKE_INTRINSIC(name) \
346 static ALWAYS_INLINE bool Mterp##name(ShadowFrame* shadow_frame, \
347 const Instruction* inst, \
348 uint16_t inst_data, \
349 JValue* result) \
350 REQUIRES_SHARED(Locks::mutator_lock_) { \
351 if (inst->Opcode() == Instruction::INVOKE_POLYMORPHIC) { \
352 return DoInvokePolymorphic<false>(Thread::Current(), *shadow_frame, inst, inst_data, result); \
353 } else { \
354 return DoInvokePolymorphic<true>(Thread::Current(), *shadow_frame, inst, inst_data, result); \
355 } \
356 }
357
358 METHOD_HANDLE_INVOKE_INTRINSIC(MethodHandleInvokeExact)
359 METHOD_HANDLE_INVOKE_INTRINSIC(MethodHandleInvoke)
360
361 #define VAR_HANDLE_ACCESSOR_INTRINSIC(name) \
362 static ALWAYS_INLINE bool Mterp##name(ShadowFrame* shadow_frame, \
363 const Instruction* inst, \
364 uint16_t inst_data, \
365 JValue* result) \
366 REQUIRES_SHARED(Locks::mutator_lock_) { \
367 return Do##name(Thread::Current(), *shadow_frame, inst, inst_data, result); \
368 }
369
370 VAR_HANDLE_ACCESSOR_INTRINSIC(VarHandleCompareAndExchange)
371 VAR_HANDLE_ACCESSOR_INTRINSIC(VarHandleCompareAndExchangeAcquire)
372 VAR_HANDLE_ACCESSOR_INTRINSIC(VarHandleCompareAndExchangeRelease)
373 VAR_HANDLE_ACCESSOR_INTRINSIC(VarHandleCompareAndSet)
374 VAR_HANDLE_ACCESSOR_INTRINSIC(VarHandleGet);
375 VAR_HANDLE_ACCESSOR_INTRINSIC(VarHandleGetAcquire)
VAR_HANDLE_ACCESSOR_INTRINSIC(VarHandleGetAndAdd)376 VAR_HANDLE_ACCESSOR_INTRINSIC(VarHandleGetAndAdd)
377 VAR_HANDLE_ACCESSOR_INTRINSIC(VarHandleGetAndAddAcquire)
378 VAR_HANDLE_ACCESSOR_INTRINSIC(VarHandleGetAndAddRelease)
379 VAR_HANDLE_ACCESSOR_INTRINSIC(VarHandleGetAndBitwiseAnd)
380 VAR_HANDLE_ACCESSOR_INTRINSIC(VarHandleGetAndBitwiseAndAcquire)
381 VAR_HANDLE_ACCESSOR_INTRINSIC(VarHandleGetAndBitwiseAndRelease)
382 VAR_HANDLE_ACCESSOR_INTRINSIC(VarHandleGetAndBitwiseOr)
383 VAR_HANDLE_ACCESSOR_INTRINSIC(VarHandleGetAndBitwiseOrAcquire)
384 VAR_HANDLE_ACCESSOR_INTRINSIC(VarHandleGetAndBitwiseOrRelease)
385 VAR_HANDLE_ACCESSOR_INTRINSIC(VarHandleGetAndBitwiseXor)
386 VAR_HANDLE_ACCESSOR_INTRINSIC(VarHandleGetAndBitwiseXorAcquire)
387 VAR_HANDLE_ACCESSOR_INTRINSIC(VarHandleGetAndBitwiseXorRelease)
388 VAR_HANDLE_ACCESSOR_INTRINSIC(VarHandleGetAndSet)
389 VAR_HANDLE_ACCESSOR_INTRINSIC(VarHandleGetAndSetAcquire)
390 VAR_HANDLE_ACCESSOR_INTRINSIC(VarHandleGetAndSetRelease)
391 VAR_HANDLE_ACCESSOR_INTRINSIC(VarHandleGetOpaque)
392 VAR_HANDLE_ACCESSOR_INTRINSIC(VarHandleGetVolatile)
393 VAR_HANDLE_ACCESSOR_INTRINSIC(VarHandleSet)
394 VAR_HANDLE_ACCESSOR_INTRINSIC(VarHandleSetOpaque)
395 VAR_HANDLE_ACCESSOR_INTRINSIC(VarHandleSetRelease)
396 VAR_HANDLE_ACCESSOR_INTRINSIC(VarHandleSetVolatile)
397 VAR_HANDLE_ACCESSOR_INTRINSIC(VarHandleWeakCompareAndSet)
398 VAR_HANDLE_ACCESSOR_INTRINSIC(VarHandleWeakCompareAndSetAcquire)
399 VAR_HANDLE_ACCESSOR_INTRINSIC(VarHandleWeakCompareAndSetPlain)
400 VAR_HANDLE_ACCESSOR_INTRINSIC(VarHandleWeakCompareAndSetRelease)
401
402 static ALWAYS_INLINE bool MterpReachabilityFence(ShadowFrame* shadow_frame ATTRIBUTE_UNUSED,
403 const Instruction* inst ATTRIBUTE_UNUSED,
404 uint16_t inst_data ATTRIBUTE_UNUSED,
405 JValue* result_register ATTRIBUTE_UNUSED)
406 REQUIRES_SHARED(Locks::mutator_lock_) {
407 // Do nothing; Its only purpose is to keep the argument reference live
408 // at preceding suspend points. That's automatic in the interpreter.
409 return true;
410 }
411
412 // Macro to help keep track of what's left to implement.
413 #define UNIMPLEMENTED_CASE(name) \
414 case Intrinsics::k##name: \
415 res = false; \
416 break;
417
418 #define INTRINSIC_CASE(name) \
419 case Intrinsics::k##name: \
420 res = Mterp##name(shadow_frame, inst, inst_data, result_register); \
421 break;
422
MterpHandleIntrinsic(ShadowFrame * shadow_frame,ArtMethod * const called_method,const Instruction * inst,uint16_t inst_data,JValue * result_register)423 bool MterpHandleIntrinsic(ShadowFrame* shadow_frame,
424 ArtMethod* const called_method,
425 const Instruction* inst,
426 uint16_t inst_data,
427 JValue* result_register)
428 REQUIRES_SHARED(Locks::mutator_lock_) {
429 Intrinsics intrinsic = static_cast<Intrinsics>(called_method->GetIntrinsic());
430 bool res = false; // Assume failure
431 switch (intrinsic) {
432 UNIMPLEMENTED_CASE(DoubleDoubleToRawLongBits /* (D)J */)
433 UNIMPLEMENTED_CASE(DoubleDoubleToLongBits /* (D)J */)
434 UNIMPLEMENTED_CASE(DoubleIsInfinite /* (D)Z */)
435 UNIMPLEMENTED_CASE(DoubleIsNaN /* (D)Z */)
436 UNIMPLEMENTED_CASE(DoubleLongBitsToDouble /* (J)D */)
437 UNIMPLEMENTED_CASE(FloatFloatToRawIntBits /* (F)I */)
438 UNIMPLEMENTED_CASE(FloatFloatToIntBits /* (F)I */)
439 UNIMPLEMENTED_CASE(FloatIsInfinite /* (F)Z */)
440 UNIMPLEMENTED_CASE(FloatIsNaN /* (F)Z */)
441 UNIMPLEMENTED_CASE(FloatIntBitsToFloat /* (I)F */)
442 INTRINSIC_CASE(IntegerReverse)
443 INTRINSIC_CASE(IntegerReverseBytes)
444 INTRINSIC_CASE(IntegerBitCount)
445 INTRINSIC_CASE(IntegerCompare)
446 INTRINSIC_CASE(IntegerHighestOneBit)
447 INTRINSIC_CASE(IntegerLowestOneBit)
448 INTRINSIC_CASE(IntegerNumberOfLeadingZeros)
449 INTRINSIC_CASE(IntegerNumberOfTrailingZeros)
450 INTRINSIC_CASE(IntegerRotateRight)
451 INTRINSIC_CASE(IntegerRotateLeft)
452 INTRINSIC_CASE(IntegerSignum)
453 INTRINSIC_CASE(LongReverse)
454 INTRINSIC_CASE(LongReverseBytes)
455 INTRINSIC_CASE(LongBitCount)
456 INTRINSIC_CASE(LongCompare)
457 INTRINSIC_CASE(LongHighestOneBit)
458 INTRINSIC_CASE(LongLowestOneBit)
459 INTRINSIC_CASE(LongNumberOfLeadingZeros)
460 INTRINSIC_CASE(LongNumberOfTrailingZeros)
461 INTRINSIC_CASE(LongRotateRight)
462 INTRINSIC_CASE(LongRotateLeft)
463 INTRINSIC_CASE(LongSignum)
464 INTRINSIC_CASE(ShortReverseBytes)
465 INTRINSIC_CASE(MathAbsDouble)
466 INTRINSIC_CASE(MathAbsFloat)
467 INTRINSIC_CASE(MathAbsLong)
468 INTRINSIC_CASE(MathAbsInt)
469 UNIMPLEMENTED_CASE(MathMinDoubleDouble /* (DD)D */)
470 UNIMPLEMENTED_CASE(MathMinFloatFloat /* (FF)F */)
471 INTRINSIC_CASE(MathMinLongLong)
472 INTRINSIC_CASE(MathMinIntInt)
473 UNIMPLEMENTED_CASE(MathMaxDoubleDouble /* (DD)D */)
474 UNIMPLEMENTED_CASE(MathMaxFloatFloat /* (FF)F */)
475 INTRINSIC_CASE(MathMaxLongLong)
476 INTRINSIC_CASE(MathMaxIntInt)
477 INTRINSIC_CASE(MathCos)
478 INTRINSIC_CASE(MathSin)
479 INTRINSIC_CASE(MathAcos)
480 INTRINSIC_CASE(MathAsin)
481 INTRINSIC_CASE(MathAtan)
482 UNIMPLEMENTED_CASE(MathAtan2 /* (DD)D */)
483 UNIMPLEMENTED_CASE(MathCbrt /* (D)D */)
484 UNIMPLEMENTED_CASE(MathCosh /* (D)D */)
485 UNIMPLEMENTED_CASE(MathExp /* (D)D */)
486 UNIMPLEMENTED_CASE(MathExpm1 /* (D)D */)
487 UNIMPLEMENTED_CASE(MathHypot /* (DD)D */)
488 UNIMPLEMENTED_CASE(MathLog /* (D)D */)
489 UNIMPLEMENTED_CASE(MathLog10 /* (D)D */)
490 UNIMPLEMENTED_CASE(MathNextAfter /* (DD)D */)
491 UNIMPLEMENTED_CASE(MathPow /* (DD)D */)
492 UNIMPLEMENTED_CASE(MathSinh /* (D)D */)
493 INTRINSIC_CASE(MathTan)
494 UNIMPLEMENTED_CASE(MathTanh /* (D)D */)
495 INTRINSIC_CASE(MathSqrt)
496 INTRINSIC_CASE(MathCeil)
497 INTRINSIC_CASE(MathFloor)
498 UNIMPLEMENTED_CASE(MathRint /* (D)D */)
499 UNIMPLEMENTED_CASE(MathRoundDouble /* (D)J */)
500 UNIMPLEMENTED_CASE(MathRoundFloat /* (F)I */)
501 UNIMPLEMENTED_CASE(SystemArrayCopyChar /* ([CI[CII)V */)
502 UNIMPLEMENTED_CASE(SystemArrayCopy /* (Ljava/lang/Object;ILjava/lang/Object;II)V */)
503 UNIMPLEMENTED_CASE(ThreadCurrentThread /* ()Ljava/lang/Thread; */)
504 UNIMPLEMENTED_CASE(MemoryPeekByte /* (J)B */)
505 UNIMPLEMENTED_CASE(MemoryPeekIntNative /* (J)I */)
506 UNIMPLEMENTED_CASE(MemoryPeekLongNative /* (J)J */)
507 UNIMPLEMENTED_CASE(MemoryPeekShortNative /* (J)S */)
508 UNIMPLEMENTED_CASE(MemoryPokeByte /* (JB)V */)
509 UNIMPLEMENTED_CASE(MemoryPokeIntNative /* (JI)V */)
510 UNIMPLEMENTED_CASE(MemoryPokeLongNative /* (JJ)V */)
511 UNIMPLEMENTED_CASE(MemoryPokeShortNative /* (JS)V */)
512 INTRINSIC_CASE(ReachabilityFence /* (Ljava/lang/Object;)V */)
513 INTRINSIC_CASE(StringCharAt)
514 INTRINSIC_CASE(StringCompareTo)
515 INTRINSIC_CASE(StringEquals)
516 INTRINSIC_CASE(StringGetCharsNoCheck)
517 INTRINSIC_CASE(StringIndexOf)
518 INTRINSIC_CASE(StringIndexOfAfter)
519 UNIMPLEMENTED_CASE(StringStringIndexOf /* (Ljava/lang/String;)I */)
520 UNIMPLEMENTED_CASE(StringStringIndexOfAfter /* (Ljava/lang/String;I)I */)
521 INTRINSIC_CASE(StringIsEmpty)
522 INTRINSIC_CASE(StringLength)
523 UNIMPLEMENTED_CASE(StringNewStringFromBytes /* ([BIII)Ljava/lang/String; */)
524 UNIMPLEMENTED_CASE(StringNewStringFromChars /* (II[C)Ljava/lang/String; */)
525 UNIMPLEMENTED_CASE(StringNewStringFromString /* (Ljava/lang/String;)Ljava/lang/String; */)
526 UNIMPLEMENTED_CASE(StringBufferAppend /* (Ljava/lang/String;)Ljava/lang/StringBuffer; */)
527 UNIMPLEMENTED_CASE(StringBufferLength /* ()I */)
528 UNIMPLEMENTED_CASE(StringBufferToString /* ()Ljava/lang/String; */)
529 UNIMPLEMENTED_CASE(StringBuilderAppend /* (Ljava/lang/String;)Ljava/lang/StringBuilder; */)
530 UNIMPLEMENTED_CASE(StringBuilderLength /* ()I */)
531 UNIMPLEMENTED_CASE(StringBuilderToString /* ()Ljava/lang/String; */)
532 UNIMPLEMENTED_CASE(UnsafeCASInt /* (Ljava/lang/Object;JII)Z */)
533 UNIMPLEMENTED_CASE(UnsafeCASLong /* (Ljava/lang/Object;JJJ)Z */)
534 UNIMPLEMENTED_CASE(UnsafeCASObject /* (Ljava/lang/Object;JLjava/lang/Object;Ljava/lang/Object;)Z */)
535 UNIMPLEMENTED_CASE(UnsafeGet /* (Ljava/lang/Object;J)I */)
536 UNIMPLEMENTED_CASE(UnsafeGetVolatile /* (Ljava/lang/Object;J)I */)
537 UNIMPLEMENTED_CASE(UnsafeGetObject /* (Ljava/lang/Object;J)Ljava/lang/Object; */)
538 UNIMPLEMENTED_CASE(UnsafeGetObjectVolatile /* (Ljava/lang/Object;J)Ljava/lang/Object; */)
539 UNIMPLEMENTED_CASE(UnsafeGetLong /* (Ljava/lang/Object;J)J */)
540 UNIMPLEMENTED_CASE(UnsafeGetLongVolatile /* (Ljava/lang/Object;J)J */)
541 UNIMPLEMENTED_CASE(UnsafePut /* (Ljava/lang/Object;JI)V */)
542 UNIMPLEMENTED_CASE(UnsafePutOrdered /* (Ljava/lang/Object;JI)V */)
543 UNIMPLEMENTED_CASE(UnsafePutVolatile /* (Ljava/lang/Object;JI)V */)
544 UNIMPLEMENTED_CASE(UnsafePutObject /* (Ljava/lang/Object;JLjava/lang/Object;)V */)
545 UNIMPLEMENTED_CASE(UnsafePutObjectOrdered /* (Ljava/lang/Object;JLjava/lang/Object;)V */)
546 UNIMPLEMENTED_CASE(UnsafePutObjectVolatile /* (Ljava/lang/Object;JLjava/lang/Object;)V */)
547 UNIMPLEMENTED_CASE(UnsafePutLong /* (Ljava/lang/Object;JJ)V */)
548 UNIMPLEMENTED_CASE(UnsafePutLongOrdered /* (Ljava/lang/Object;JJ)V */)
549 UNIMPLEMENTED_CASE(UnsafePutLongVolatile /* (Ljava/lang/Object;JJ)V */)
550 UNIMPLEMENTED_CASE(UnsafeGetAndAddInt /* (Ljava/lang/Object;JI)I */)
551 UNIMPLEMENTED_CASE(UnsafeGetAndAddLong /* (Ljava/lang/Object;JJ)J */)
552 UNIMPLEMENTED_CASE(UnsafeGetAndSetInt /* (Ljava/lang/Object;JI)I */)
553 UNIMPLEMENTED_CASE(UnsafeGetAndSetLong /* (Ljava/lang/Object;JJ)J */)
554 UNIMPLEMENTED_CASE(UnsafeGetAndSetObject /* (Ljava/lang/Object;JLjava/lang/Object;)Ljava/lang/Object; */)
555 UNIMPLEMENTED_CASE(UnsafeLoadFence /* ()V */)
556 UNIMPLEMENTED_CASE(UnsafeStoreFence /* ()V */)
557 UNIMPLEMENTED_CASE(UnsafeFullFence /* ()V */)
558 UNIMPLEMENTED_CASE(ReferenceGetReferent /* ()Ljava/lang/Object; */)
559 UNIMPLEMENTED_CASE(IntegerValueOf /* (I)Ljava/lang/Integer; */)
560 UNIMPLEMENTED_CASE(ThreadInterrupted /* ()Z */)
561 INTRINSIC_CASE(VarHandleFullFence)
562 INTRINSIC_CASE(VarHandleAcquireFence)
563 INTRINSIC_CASE(VarHandleReleaseFence)
564 INTRINSIC_CASE(VarHandleLoadLoadFence)
565 INTRINSIC_CASE(VarHandleStoreStoreFence)
566 INTRINSIC_CASE(MethodHandleInvokeExact)
567 INTRINSIC_CASE(MethodHandleInvoke)
568 INTRINSIC_CASE(VarHandleCompareAndExchange)
569 INTRINSIC_CASE(VarHandleCompareAndExchangeAcquire)
570 INTRINSIC_CASE(VarHandleCompareAndExchangeRelease)
571 INTRINSIC_CASE(VarHandleCompareAndSet)
572 INTRINSIC_CASE(VarHandleGet)
573 INTRINSIC_CASE(VarHandleGetAcquire)
574 INTRINSIC_CASE(VarHandleGetAndAdd)
575 INTRINSIC_CASE(VarHandleGetAndAddAcquire)
576 INTRINSIC_CASE(VarHandleGetAndAddRelease)
577 INTRINSIC_CASE(VarHandleGetAndBitwiseAnd)
578 INTRINSIC_CASE(VarHandleGetAndBitwiseAndAcquire)
579 INTRINSIC_CASE(VarHandleGetAndBitwiseAndRelease)
580 INTRINSIC_CASE(VarHandleGetAndBitwiseOr)
581 INTRINSIC_CASE(VarHandleGetAndBitwiseOrAcquire)
582 INTRINSIC_CASE(VarHandleGetAndBitwiseOrRelease)
583 INTRINSIC_CASE(VarHandleGetAndBitwiseXor)
584 INTRINSIC_CASE(VarHandleGetAndBitwiseXorAcquire)
585 INTRINSIC_CASE(VarHandleGetAndBitwiseXorRelease)
586 INTRINSIC_CASE(VarHandleGetAndSet)
587 INTRINSIC_CASE(VarHandleGetAndSetAcquire)
588 INTRINSIC_CASE(VarHandleGetAndSetRelease)
589 INTRINSIC_CASE(VarHandleGetOpaque)
590 INTRINSIC_CASE(VarHandleGetVolatile)
591 INTRINSIC_CASE(VarHandleSet)
592 INTRINSIC_CASE(VarHandleSetOpaque)
593 INTRINSIC_CASE(VarHandleSetRelease)
594 INTRINSIC_CASE(VarHandleSetVolatile)
595 INTRINSIC_CASE(VarHandleWeakCompareAndSet)
596 INTRINSIC_CASE(VarHandleWeakCompareAndSetAcquire)
597 INTRINSIC_CASE(VarHandleWeakCompareAndSetPlain)
598 INTRINSIC_CASE(VarHandleWeakCompareAndSetRelease)
599 case Intrinsics::kNone:
600 res = false;
601 break;
602 // Note: no default case to ensure we catch any newly added intrinsics.
603 }
604 return res;
605 }
606
607 } // namespace interpreter
608 } // namespace art
609