• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2008 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 /*
18  * Inlined native functions.  These definitions replace interpreted or
19  * native implementations at runtime; "intrinsic" might be a better word.
20  */
21 #include "Dalvik.h"
22 
23 #include <math.h>
24 
25 #ifdef HAVE__MEMCMP16
26 /* hand-coded assembly implementation, available on some platforms */
27 //#warning "trying memcmp16"
28 //#define CHECK_MEMCMP16
29 /* "count" is in 16-bit units */
30 extern "C" u4 __memcmp16(const u2* s0, const u2* s1, size_t count);
31 #endif
32 
33 /*
34  * Some notes on "inline" functions.
35  *
36  * These are NOT simply native implementations.  A full method definition
37  * must still be provided.  Depending on the flags passed into the VM
38  * at runtime, the original or inline version may be selected by the
39  * DEX optimizer.
40  *
41  * PLEASE DO NOT use this as the default location for native methods.
42  * The difference between this and an "internal native" static method
43  * call on a 200MHz ARM 9 is roughly 370ns vs. 700ns.  The code here
44  * "secretly replaces" the other method, so you can't avoid having two
45  * implementations.  Since the DEX optimizer mode can't be known ahead
46  * of time, both implementations must be correct and complete.
47  *
48  * The only stuff that really needs to be here are methods that
49  * are high-volume or must be low-overhead, e.g. certain String/Math
50  * methods and some java.util.concurrent.atomic operations.
51  *
52  * Normally, a class is loaded and initialized the first time a static
53  * method is invoked.  This property is NOT preserved here.  If you need
54  * to access a static field in a class, you must ensure initialization
55  * yourself (cheap/easy way is to check the resolved-methods table, and
56  * resolve the method if it hasn't been).
57  *
58  * DO NOT replace "synchronized" methods.  We do not support method
59  * synchronization here.
60  *
61  * DO NOT perform any allocations or do anything that could cause a
62  * garbage collection.  The method arguments are not visible to the GC
63  * and will not be pinned or updated when memory blocks move.  You are
64  * allowed to allocate and throw an exception so long as you only do so
65  * immediately before returning.
66  *
67  * Remember that these functions are executing while the thread is in
68  * the "RUNNING" state, not the "NATIVE" state.  If you perform a blocking
69  * operation you can stall the entire VM if the GC or debugger wants to
70  * suspend the thread.  Since these are arguably native implementations
71  * rather than VM internals, prefer NATIVE to VMWAIT if you want to change
72  * the thread state.
73  *
74  * Always write results to 32-bit or 64-bit fields in "pResult", e.g. do
75  * not write boolean results to pResult->z.  The interpreter expects
76  * 32 or 64 bits to be set.
77  *
78  * Inline op methods return "false" if an exception was thrown, "true" if
79  * everything went well.
80  *
81  * DO NOT provide implementations of methods that can be overridden by a
82  * subclass, as polymorphism does not work correctly.  For safety you should
83  * only provide inline functions for classes/methods declared "final".
84  *
85  * It's best to avoid inlining the overridden version of a method.  For
86  * example, String.hashCode() is inherited from Object.hashCode().  Code
87  * calling String.hashCode() through an Object reference will run the
88  * "slow" version, while calling it through a String reference gets
89  * the inlined version.  It's best to have just one version unless there
90  * are clear performance gains.
91  *
92  * Because the actual method is not called, debugger breakpoints on these
93  * methods will not happen.  (TODO: have the code here find the original
94  * method and call it when the debugger is active.)  Additional steps have
95  * been taken to allow method profiling to produce correct results.
96  */
97 
98 
99 /*
100  * ===========================================================================
101  *      org.apache.harmony.dalvik.NativeTestTarget
102  * ===========================================================================
103  */
104 
105 /*
106  * public static void emptyInlineMethod
107  *
108  * This exists only for benchmarks.
109  */
org_apache_harmony_dalvik_NativeTestTarget_emptyInlineMethod(u4 arg0,u4 arg1,u4 arg2,u4 arg3,JValue * pResult)110 static bool org_apache_harmony_dalvik_NativeTestTarget_emptyInlineMethod(
111     u4 arg0, u4 arg1, u4 arg2, u4 arg3, JValue* pResult)
112 {
113     // do nothing
114     return true;
115 }
116 
117 
118 /*
119  * ===========================================================================
120  *      java.lang.String
121  * ===========================================================================
122  */
123 
124 /*
125  * public char charAt(int index)
126  */
javaLangString_charAt(u4 arg0,u4 arg1,u4 arg2,u4 arg3,JValue * pResult)127 bool javaLangString_charAt(u4 arg0, u4 arg1, u4 arg2, u4 arg3,
128     JValue* pResult)
129 {
130     int count, offset;
131     ArrayObject* chars;
132 
133     /* null reference check on "this" */
134     if ((Object*) arg0 == NULL) {
135         dvmThrowNullPointerException(NULL);
136         return false;
137     }
138 
139     //ALOGI("String.charAt this=0x%08x index=%d", arg0, arg1);
140     count = dvmGetFieldInt((Object*) arg0, STRING_FIELDOFF_COUNT);
141     if ((s4) arg1 < 0 || (s4) arg1 >= count) {
142         dvmThrowStringIndexOutOfBoundsExceptionWithIndex(count, arg1);
143         return false;
144     } else {
145         offset = dvmGetFieldInt((Object*) arg0, STRING_FIELDOFF_OFFSET);
146         chars = (ArrayObject*)
147             dvmGetFieldObject((Object*) arg0, STRING_FIELDOFF_VALUE);
148 
149         pResult->i = ((const u2*)(void*)chars->contents)[arg1 + offset];
150         return true;
151     }
152 }
153 
154 #ifdef CHECK_MEMCMP16
155 /*
156  * Utility function when we're evaluating alternative implementations.
157  */
badMatch(StringObject * thisStrObj,StringObject * compStrObj,int expectResult,int newResult,const char * compareType)158 static void badMatch(StringObject* thisStrObj, StringObject* compStrObj,
159     int expectResult, int newResult, const char* compareType)
160 {
161     ArrayObject* thisArray;
162     ArrayObject* compArray;
163     const char* thisStr;
164     const char* compStr;
165     int thisOffset, compOffset, thisCount, compCount;
166 
167     thisCount =
168         dvmGetFieldInt((Object*) thisStrObj, STRING_FIELDOFF_COUNT);
169     compCount =
170         dvmGetFieldInt((Object*) compStrObj, STRING_FIELDOFF_COUNT);
171     thisOffset =
172         dvmGetFieldInt((Object*) thisStrObj, STRING_FIELDOFF_OFFSET);
173     compOffset =
174         dvmGetFieldInt((Object*) compStrObj, STRING_FIELDOFF_OFFSET);
175     thisArray = (ArrayObject*)
176         dvmGetFieldObject((Object*) thisStrObj, STRING_FIELDOFF_VALUE);
177     compArray = (ArrayObject*)
178         dvmGetFieldObject((Object*) compStrObj, STRING_FIELDOFF_VALUE);
179 
180     thisStr = dvmCreateCstrFromString(thisStrObj);
181     compStr = dvmCreateCstrFromString(compStrObj);
182 
183     ALOGE("%s expected %d got %d", compareType, expectResult, newResult);
184     ALOGE(" this (o=%d l=%d) '%s'", thisOffset, thisCount, thisStr);
185     ALOGE(" comp (o=%d l=%d) '%s'", compOffset, compCount, compStr);
186     dvmPrintHexDumpEx(ANDROID_LOG_INFO, LOG_TAG,
187         ((const u2*) thisArray->contents) + thisOffset, thisCount*2,
188         kHexDumpLocal);
189     dvmPrintHexDumpEx(ANDROID_LOG_INFO, LOG_TAG,
190         ((const u2*) compArray->contents) + compOffset, compCount*2,
191         kHexDumpLocal);
192     dvmAbort();
193 }
194 #endif
195 
196 /*
197  * public int compareTo(String s)
198  */
javaLangString_compareTo(u4 arg0,u4 arg1,u4 arg2,u4 arg3,JValue * pResult)199 bool javaLangString_compareTo(u4 arg0, u4 arg1, u4 arg2, u4 arg3,
200     JValue* pResult)
201 {
202     /*
203      * Null reference check on "this".  Normally this is performed during
204      * the setup of the virtual method call.  We need to do it before
205      * anything else.  While we're at it, check out the other string,
206      * which must also be non-null.
207      */
208     if ((Object*) arg0 == NULL || (Object*) arg1 == NULL) {
209         dvmThrowNullPointerException(NULL);
210         return false;
211     }
212 
213     /* quick test for comparison with itself */
214     if (arg0 == arg1) {
215         pResult->i = 0;
216         return true;
217     }
218 
219     /*
220      * This would be simpler and faster if we promoted StringObject to
221      * a full representation, lining up the C structure fields with the
222      * actual object fields.
223      */
224     int thisCount, thisOffset, compCount, compOffset;
225     ArrayObject* thisArray;
226     ArrayObject* compArray;
227     const u2* thisChars;
228     const u2* compChars;
229     int minCount, countDiff;
230 
231     thisCount = dvmGetFieldInt((Object*) arg0, STRING_FIELDOFF_COUNT);
232     compCount = dvmGetFieldInt((Object*) arg1, STRING_FIELDOFF_COUNT);
233     countDiff = thisCount - compCount;
234     minCount = (countDiff < 0) ? thisCount : compCount;
235     thisOffset = dvmGetFieldInt((Object*) arg0, STRING_FIELDOFF_OFFSET);
236     compOffset = dvmGetFieldInt((Object*) arg1, STRING_FIELDOFF_OFFSET);
237     thisArray = (ArrayObject*)
238         dvmGetFieldObject((Object*) arg0, STRING_FIELDOFF_VALUE);
239     compArray = (ArrayObject*)
240         dvmGetFieldObject((Object*) arg1, STRING_FIELDOFF_VALUE);
241     thisChars = ((const u2*)(void*)thisArray->contents) + thisOffset;
242     compChars = ((const u2*)(void*)compArray->contents) + compOffset;
243 
244 #ifdef HAVE__MEMCMP16
245     /*
246      * Use assembly version, which returns the difference between the
247      * characters.  The annoying part here is that 0x00e9 - 0xffff != 0x00ea,
248      * because the interpreter converts the characters to 32-bit integers
249      * *without* sign extension before it subtracts them (which makes some
250      * sense since "char" is unsigned).  So what we get is the result of
251      * 0x000000e9 - 0x0000ffff, which is 0xffff00ea.
252      */
253     int otherRes = __memcmp16(thisChars, compChars, minCount);
254 # ifdef CHECK_MEMCMP16
255     int i;
256     for (i = 0; i < minCount; i++) {
257         if (thisChars[i] != compChars[i]) {
258             pResult->i = (s4) thisChars[i] - (s4) compChars[i];
259             if (pResult->i != otherRes) {
260                 badMatch((StringObject*) arg0, (StringObject*) arg1,
261                     pResult->i, otherRes, "compareTo");
262             }
263             return true;
264         }
265     }
266 # endif
267     if (otherRes != 0) {
268         pResult->i = otherRes;
269         return true;
270     }
271 
272 #else
273     /*
274      * Straightforward implementation, examining 16 bits at a time.  Compare
275      * the characters that overlap, and if they're all the same then return
276      * the difference in lengths.
277      */
278     int i;
279     for (i = 0; i < minCount; i++) {
280         if (thisChars[i] != compChars[i]) {
281             pResult->i = (s4) thisChars[i] - (s4) compChars[i];
282             return true;
283         }
284     }
285 #endif
286 
287     pResult->i = countDiff;
288     return true;
289 }
290 
291 /*
292  * public boolean equals(Object anObject)
293  */
javaLangString_equals(u4 arg0,u4 arg1,u4 arg2,u4 arg3,JValue * pResult)294 bool javaLangString_equals(u4 arg0, u4 arg1, u4 arg2, u4 arg3,
295     JValue* pResult)
296 {
297     /*
298      * Null reference check on "this".
299      */
300     if ((Object*) arg0 == NULL) {
301         dvmThrowNullPointerException(NULL);
302         return false;
303     }
304 
305     /* quick test for comparison with itself */
306     if (arg0 == arg1) {
307         pResult->i = true;
308         return true;
309     }
310 
311     /*
312      * See if the other object is also a String.
313      *
314      * str.equals(null) is expected to return false, presumably based on
315      * the results of the instanceof test.
316      */
317     if (arg1 == 0 || ((Object*) arg0)->clazz != ((Object*) arg1)->clazz) {
318         pResult->i = false;
319         return true;
320     }
321 
322     /*
323      * This would be simpler and faster if we promoted StringObject to
324      * a full representation, lining up the C structure fields with the
325      * actual object fields.
326      */
327     int thisCount, thisOffset, compCount, compOffset;
328     ArrayObject* thisArray;
329     ArrayObject* compArray;
330     const u2* thisChars;
331     const u2* compChars;
332 
333     /* quick length check */
334     thisCount = dvmGetFieldInt((Object*) arg0, STRING_FIELDOFF_COUNT);
335     compCount = dvmGetFieldInt((Object*) arg1, STRING_FIELDOFF_COUNT);
336     if (thisCount != compCount) {
337         pResult->i = false;
338         return true;
339     }
340 
341     /*
342      * You may, at this point, be tempted to pull out the hashCode fields
343      * and compare them.  If both fields have been initialized, and they
344      * are not equal, we can return false immediately.
345      *
346      * However, the hashCode field is often not set.  If it is set,
347      * there's an excellent chance that the String is being used as a key
348      * in a hashed data structure (e.g. HashMap).  That data structure has
349      * already made the comparison and determined that the hashes are equal,
350      * making a check here redundant.
351      *
352      * It's not clear that checking the hashes will be a win in "typical"
353      * use cases.  We err on the side of simplicity and ignore them.
354      */
355 
356     thisOffset = dvmGetFieldInt((Object*) arg0, STRING_FIELDOFF_OFFSET);
357     compOffset = dvmGetFieldInt((Object*) arg1, STRING_FIELDOFF_OFFSET);
358     thisArray = (ArrayObject*)
359         dvmGetFieldObject((Object*) arg0, STRING_FIELDOFF_VALUE);
360     compArray = (ArrayObject*)
361         dvmGetFieldObject((Object*) arg1, STRING_FIELDOFF_VALUE);
362     thisChars = ((const u2*)(void*)thisArray->contents) + thisOffset;
363     compChars = ((const u2*)(void*)compArray->contents) + compOffset;
364 
365 #ifdef HAVE__MEMCMP16
366     pResult->i = (__memcmp16(thisChars, compChars, thisCount) == 0);
367 # ifdef CHECK_MEMCMP16
368     int otherRes = (memcmp(thisChars, compChars, thisCount * 2) == 0);
369     if (pResult->i != otherRes) {
370         badMatch((StringObject*) arg0, (StringObject*) arg1,
371             otherRes, pResult->i, "equals-1");
372     }
373 # endif
374 #else
375     /*
376      * Straightforward implementation, examining 16 bits at a time.  The
377      * direction of the loop doesn't matter, and starting at the end may
378      * give us an advantage when comparing certain types of strings (e.g.
379      * class names).
380      *
381      * We want to go forward for benchmarks against __memcmp16 so we get a
382      * meaningful comparison when the strings don't match (could also test
383      * with palindromes).
384      */
385     int i;
386     //for (i = 0; i < thisCount; i++)
387     for (i = thisCount-1; i >= 0; --i)
388     {
389         if (thisChars[i] != compChars[i]) {
390             pResult->i = false;
391             return true;
392         }
393     }
394     pResult->i = true;
395 #endif
396 
397     return true;
398 }
399 
400 /*
401  * public int length()
402  */
javaLangString_length(u4 arg0,u4 arg1,u4 arg2,u4 arg3,JValue * pResult)403 bool javaLangString_length(u4 arg0, u4 arg1, u4 arg2, u4 arg3,
404     JValue* pResult)
405 {
406     //ALOGI("String.length this=0x%08x pResult=%p", arg0, pResult);
407 
408     /* null reference check on "this" */
409     if ((Object*) arg0 == NULL) {
410         dvmThrowNullPointerException(NULL);
411         return false;
412     }
413 
414     pResult->i = dvmGetFieldInt((Object*) arg0, STRING_FIELDOFF_COUNT);
415     return true;
416 }
417 
418 /*
419  * public boolean isEmpty()
420  */
javaLangString_isEmpty(u4 arg0,u4 arg1,u4 arg2,u4 arg3,JValue * pResult)421 bool javaLangString_isEmpty(u4 arg0, u4 arg1, u4 arg2, u4 arg3,
422     JValue* pResult)
423 {
424     //ALOGI("String.isEmpty this=0x%08x pResult=%p", arg0, pResult);
425 
426     /* null reference check on "this" */
427     if ((Object*) arg0 == NULL) {
428         dvmThrowNullPointerException(NULL);
429         return false;
430     }
431 
432     pResult->i = (dvmGetFieldInt((Object*) arg0, STRING_FIELDOFF_COUNT) == 0);
433     return true;
434 }
435 
436 /*
437  * Determine the index of the first character matching "ch".  The string
438  * to search is described by "chars", "offset", and "count".
439  *
440  * The character must be <= 0xffff. Supplementary characters are handled in
441  * Java.
442  *
443  * The "start" parameter must be clamped to [0..count].
444  *
445  * Returns -1 if no match is found.
446  */
indexOfCommon(Object * strObj,int ch,int start)447 static inline int indexOfCommon(Object* strObj, int ch, int start)
448 {
449     //if ((ch & 0xffff) != ch)        /* 32-bit code point */
450     //    return -1;
451 
452     /* pull out the basic elements */
453     ArrayObject* charArray =
454         (ArrayObject*) dvmGetFieldObject(strObj, STRING_FIELDOFF_VALUE);
455     const u2* chars = (const u2*)(void*)charArray->contents;
456     int offset = dvmGetFieldInt(strObj, STRING_FIELDOFF_OFFSET);
457     int count = dvmGetFieldInt(strObj, STRING_FIELDOFF_COUNT);
458     //ALOGI("String.indexOf(0x%08x, 0x%04x, %d) off=%d count=%d",
459     //    (u4) strObj, ch, start, offset, count);
460 
461     /* factor out the offset */
462     chars += offset;
463 
464     if (start < 0)
465         start = 0;
466     else if (start > count)
467         start = count;
468 
469 #if 0
470     /* 16-bit loop, simple */
471     while (start < count) {
472         if (chars[start] == ch)
473             return start;
474         start++;
475     }
476 #else
477     /* 16-bit loop, slightly better on ARM */
478     const u2* ptr = chars + start;
479     const u2* endPtr = chars + count;
480     while (ptr < endPtr) {
481         if (*ptr++ == ch)
482             return (ptr-1) - chars;
483     }
484 #endif
485 
486     return -1;
487 }
488 
489 /*
490  * public int indexOf(int c, int start)
491  *
492  * Scan forward through the string for a matching character.
493  * The character must be <= 0xffff; this method does not handle supplementary
494  * characters.
495  */
javaLangString_fastIndexOf_II(u4 arg0,u4 arg1,u4 arg2,u4 arg3,JValue * pResult)496 bool javaLangString_fastIndexOf_II(u4 arg0, u4 arg1, u4 arg2, u4 arg3,
497     JValue* pResult)
498 {
499     /* null reference check on "this" */
500     if ((Object*) arg0 == NULL) {
501         dvmThrowNullPointerException(NULL);
502         return false;
503     }
504 
505     pResult->i = indexOfCommon((Object*) arg0, arg1, arg2);
506     return true;
507 }
508 
509 
510 /*
511  * ===========================================================================
512  *      java.lang.Math
513  * ===========================================================================
514  */
515 
516 union Convert32 {
517     u4 arg;
518     float ff;
519 };
520 
521 union Convert64 {
522     u4 arg[2];
523     s8 ll;
524     double dd;
525 };
526 
527 /*
528  * public static int abs(int)
529  */
javaLangMath_abs_int(u4 arg0,u4 arg1,u4 arg2,u4 arg3,JValue * pResult)530 bool javaLangMath_abs_int(u4 arg0, u4 arg1, u4 arg2, u4 arg3,
531     JValue* pResult)
532 {
533     s4 val = (s4) arg0;
534     pResult->i = (val >= 0) ? val : -val;
535     return true;
536 }
537 
538 /*
539  * public static long abs(long)
540  */
javaLangMath_abs_long(u4 arg0,u4 arg1,u4 arg2,u4 arg3,JValue * pResult)541 bool javaLangMath_abs_long(u4 arg0, u4 arg1, u4 arg2, u4 arg3,
542     JValue* pResult)
543 {
544     Convert64 convert;
545     convert.arg[0] = arg0;
546     convert.arg[1] = arg1;
547     s8 val = convert.ll;
548     pResult->j = (val >= 0) ? val : -val;
549     return true;
550 }
551 
552 /*
553  * public static float abs(float)
554  */
javaLangMath_abs_float(u4 arg0,u4 arg1,u4 arg2,u4 arg3,JValue * pResult)555 bool javaLangMath_abs_float(u4 arg0, u4 arg1, u4 arg2, u4 arg3,
556     JValue* pResult)
557 {
558     Convert32 convert;
559     /* clear the sign bit; assumes a fairly common fp representation */
560     convert.arg = arg0 & 0x7fffffff;
561     pResult->f = convert.ff;
562     return true;
563 }
564 
565 /*
566  * public static double abs(double)
567  */
javaLangMath_abs_double(u4 arg0,u4 arg1,u4 arg2,u4 arg3,JValue * pResult)568 bool javaLangMath_abs_double(u4 arg0, u4 arg1, u4 arg2, u4 arg3,
569     JValue* pResult)
570 {
571     Convert64 convert;
572     convert.arg[0] = arg0;
573     convert.arg[1] = arg1;
574     /* clear the sign bit in the (endian-dependent) high word */
575     convert.ll &= 0x7fffffffffffffffULL;
576     pResult->d = convert.dd;
577     return true;
578 }
579 
580 /*
581  * public static int min(int)
582  */
javaLangMath_min_int(u4 arg0,u4 arg1,u4 arg2,u4 arg3,JValue * pResult)583 bool javaLangMath_min_int(u4 arg0, u4 arg1, u4 arg2, u4 arg3,
584     JValue* pResult)
585 {
586     pResult->i = ((s4) arg0 < (s4) arg1) ? arg0 : arg1;
587     return true;
588 }
589 
590 /*
591  * public static int max(int)
592  */
javaLangMath_max_int(u4 arg0,u4 arg1,u4 arg2,u4 arg3,JValue * pResult)593 bool javaLangMath_max_int(u4 arg0, u4 arg1, u4 arg2, u4 arg3,
594     JValue* pResult)
595 {
596     pResult->i = ((s4) arg0 > (s4) arg1) ? arg0 : arg1;
597     return true;
598 }
599 
600 /*
601  * public static double sqrt(double)
602  *
603  * With ARM VFP enabled, gcc turns this into an fsqrtd instruction, followed
604  * by an fcmpd of the result against itself.  If it doesn't match (i.e.
605  * it's NaN), the libm sqrt() is invoked.
606  */
javaLangMath_sqrt(u4 arg0,u4 arg1,u4 arg2,u4 arg3,JValue * pResult)607 bool javaLangMath_sqrt(u4 arg0, u4 arg1, u4 arg2, u4 arg3,
608     JValue* pResult)
609 {
610     Convert64 convert;
611     convert.arg[0] = arg0;
612     convert.arg[1] = arg1;
613     pResult->d = sqrt(convert.dd);
614     return true;
615 }
616 
617 /*
618  * public static double cos(double)
619  */
javaLangMath_cos(u4 arg0,u4 arg1,u4 arg2,u4 arg3,JValue * pResult)620 bool javaLangMath_cos(u4 arg0, u4 arg1, u4 arg2, u4 arg3,
621     JValue* pResult)
622 {
623     Convert64 convert;
624     convert.arg[0] = arg0;
625     convert.arg[1] = arg1;
626     pResult->d = cos(convert.dd);
627     return true;
628 }
629 
630 /*
631  * public static double sin(double)
632  */
javaLangMath_sin(u4 arg0,u4 arg1,u4 arg2,u4 arg3,JValue * pResult)633 bool javaLangMath_sin(u4 arg0, u4 arg1, u4 arg2, u4 arg3,
634     JValue* pResult)
635 {
636     Convert64 convert;
637     convert.arg[0] = arg0;
638     convert.arg[1] = arg1;
639     pResult->d = sin(convert.dd);
640     return true;
641 }
642 
643 /*
644  * ===========================================================================
645  *      java.lang.Float
646  * ===========================================================================
647  */
648 
javaLangFloat_floatToIntBits(u4 arg0,u4 arg1,u4 arg2,u4 arg,JValue * pResult)649 bool javaLangFloat_floatToIntBits(u4 arg0, u4 arg1, u4 arg2, u4 arg,
650     JValue* pResult)
651 {
652     Convert32 convert;
653     convert.arg = arg0;
654     pResult->i = isnanf(convert.ff) ? 0x7fc00000 : arg0;
655     return true;
656 }
657 
javaLangFloat_floatToRawIntBits(u4 arg0,u4 arg1,u4 arg2,u4 arg,JValue * pResult)658 bool javaLangFloat_floatToRawIntBits(u4 arg0, u4 arg1, u4 arg2, u4 arg,
659     JValue* pResult)
660 {
661     pResult->i = arg0;
662     return true;
663 }
664 
javaLangFloat_intBitsToFloat(u4 arg0,u4 arg1,u4 arg2,u4 arg,JValue * pResult)665 bool javaLangFloat_intBitsToFloat(u4 arg0, u4 arg1, u4 arg2, u4 arg,
666     JValue* pResult)
667 {
668     Convert32 convert;
669     convert.arg = arg0;
670     pResult->f = convert.ff;
671     return true;
672 }
673 
674 /*
675  * ===========================================================================
676  *      java.lang.Double
677  * ===========================================================================
678  */
679 
javaLangDouble_doubleToLongBits(u4 arg0,u4 arg1,u4 arg2,u4 arg,JValue * pResult)680 bool javaLangDouble_doubleToLongBits(u4 arg0, u4 arg1, u4 arg2, u4 arg,
681     JValue* pResult)
682 {
683     Convert64 convert;
684     convert.arg[0] = arg0;
685     convert.arg[1] = arg1;
686     pResult->j = isnan(convert.dd) ? 0x7ff8000000000000LL : convert.ll;
687     return true;
688 }
689 
javaLangDouble_doubleToRawLongBits(u4 arg0,u4 arg1,u4 arg2,u4 arg,JValue * pResult)690 bool javaLangDouble_doubleToRawLongBits(u4 arg0, u4 arg1, u4 arg2,
691     u4 arg, JValue* pResult)
692 {
693     Convert64 convert;
694     convert.arg[0] = arg0;
695     convert.arg[1] = arg1;
696     pResult->j = convert.ll;
697     return true;
698 }
699 
javaLangDouble_longBitsToDouble(u4 arg0,u4 arg1,u4 arg2,u4 arg,JValue * pResult)700 bool javaLangDouble_longBitsToDouble(u4 arg0, u4 arg1, u4 arg2, u4 arg,
701     JValue* pResult)
702 {
703     Convert64 convert;
704     convert.arg[0] = arg0;
705     convert.arg[1] = arg1;
706     pResult->d = convert.dd;
707     return true;
708 }
709 
710 /*
711  * ===========================================================================
712  *      Infrastructure
713  * ===========================================================================
714  */
715 
716 /*
717  * Table of methods.
718  *
719  * The DEX optimizer uses the class/method/signature string fields to decide
720  * which calls it can trample.  The interpreter just uses the function
721  * pointer field.
722  *
723  * IMPORTANT: you must update DALVIK_VM_BUILD in DalvikVersion.h if you make
724  * changes to this table.
725  *
726  * NOTE: If present, the JIT will also need to know about changes
727  * to this table.  Update the NativeInlineOps enum in InlineNative.h and
728  * the dispatch code in compiler/codegen/<target>/Codegen.c.
729  */
730 const InlineOperation gDvmInlineOpsTable[] = {
731     { org_apache_harmony_dalvik_NativeTestTarget_emptyInlineMethod,
732         "Lorg/apache/harmony/dalvik/NativeTestTarget;",
733         "emptyInlineMethod", "()V" },
734 
735     { javaLangString_charAt,
736         "Ljava/lang/String;", "charAt", "(I)C" },
737     { javaLangString_compareTo,
738         "Ljava/lang/String;", "compareTo", "(Ljava/lang/String;)I" },
739     { javaLangString_equals,
740         "Ljava/lang/String;", "equals", "(Ljava/lang/Object;)Z" },
741     { javaLangString_fastIndexOf_II,
742         "Ljava/lang/String;", "fastIndexOf", "(II)I" },
743     { javaLangString_isEmpty,
744         "Ljava/lang/String;", "isEmpty", "()Z" },
745     { javaLangString_length,
746         "Ljava/lang/String;", "length", "()I" },
747 
748     { javaLangMath_abs_int,
749         "Ljava/lang/Math;", "abs", "(I)I" },
750     { javaLangMath_abs_long,
751         "Ljava/lang/Math;", "abs", "(J)J" },
752     { javaLangMath_abs_float,
753         "Ljava/lang/Math;", "abs", "(F)F" },
754     { javaLangMath_abs_double,
755         "Ljava/lang/Math;", "abs", "(D)D" },
756     { javaLangMath_min_int,
757         "Ljava/lang/Math;", "min", "(II)I" },
758     { javaLangMath_max_int,
759         "Ljava/lang/Math;", "max", "(II)I" },
760     { javaLangMath_sqrt,
761         "Ljava/lang/Math;", "sqrt", "(D)D" },
762     { javaLangMath_cos,
763         "Ljava/lang/Math;", "cos", "(D)D" },
764     { javaLangMath_sin,
765         "Ljava/lang/Math;", "sin", "(D)D" },
766 
767     { javaLangFloat_floatToIntBits,
768         "Ljava/lang/Float;", "floatToIntBits", "(F)I" },
769     { javaLangFloat_floatToRawIntBits,
770         "Ljava/lang/Float;", "floatToRawIntBits", "(F)I" },
771     { javaLangFloat_intBitsToFloat,
772         "Ljava/lang/Float;", "intBitsToFloat", "(I)F" },
773 
774     { javaLangDouble_doubleToLongBits,
775         "Ljava/lang/Double;", "doubleToLongBits", "(D)J" },
776     { javaLangDouble_doubleToRawLongBits,
777         "Ljava/lang/Double;", "doubleToRawLongBits", "(D)J" },
778     { javaLangDouble_longBitsToDouble,
779         "Ljava/lang/Double;", "longBitsToDouble", "(J)D" },
780 };
781 
782 /*
783  * Allocate some tables.
784  */
dvmInlineNativeStartup()785 bool dvmInlineNativeStartup()
786 {
787     gDvm.inlinedMethods =
788         (Method**) calloc(NELEM(gDvmInlineOpsTable), sizeof(Method*));
789     if (gDvm.inlinedMethods == NULL)
790         return false;
791 
792     return true;
793 }
794 
795 /*
796  * Free generated tables.
797  */
dvmInlineNativeShutdown()798 void dvmInlineNativeShutdown()
799 {
800     free(gDvm.inlinedMethods);
801 }
802 
803 
804 /*
805  * Get a pointer to the inlineops table.
806  */
dvmGetInlineOpsTable()807 const InlineOperation* dvmGetInlineOpsTable()
808 {
809     return gDvmInlineOpsTable;
810 }
811 
812 /*
813  * Get the number of entries in the inlineops table.
814  */
dvmGetInlineOpsTableLength()815 int dvmGetInlineOpsTableLength()
816 {
817     return NELEM(gDvmInlineOpsTable);
818 }
819 
dvmFindInlinableMethod(const char * classDescriptor,const char * methodName,const char * methodSignature)820 Method* dvmFindInlinableMethod(const char* classDescriptor,
821     const char* methodName, const char* methodSignature)
822 {
823     /*
824      * Find the class.
825      */
826     ClassObject* clazz = dvmFindClassNoInit(classDescriptor, NULL);
827     if (clazz == NULL) {
828         ALOGE("dvmFindInlinableMethod: can't find class '%s'",
829             classDescriptor);
830         dvmClearException(dvmThreadSelf());
831         return NULL;
832     }
833 
834     /*
835      * Method could be virtual or direct.  Try both.  Don't use
836      * the "hier" versions.
837      */
838     Method* method = dvmFindDirectMethodByDescriptor(clazz, methodName,
839         methodSignature);
840     if (method == NULL) {
841         method = dvmFindVirtualMethodByDescriptor(clazz, methodName,
842             methodSignature);
843     }
844     if (method == NULL) {
845         ALOGE("dvmFindInlinableMethod: can't find method %s.%s %s",
846             clazz->descriptor, methodName, methodSignature);
847         return NULL;
848     }
849 
850     /*
851      * Check that the method is appropriate for inlining.
852      */
853     if (!dvmIsFinalClass(clazz) && !dvmIsFinalMethod(method)) {
854         ALOGE("dvmFindInlinableMethod: can't inline non-final method %s.%s",
855             clazz->descriptor, method->name);
856         return NULL;
857     }
858     if (dvmIsSynchronizedMethod(method) ||
859             dvmIsDeclaredSynchronizedMethod(method)) {
860         ALOGE("dvmFindInlinableMethod: can't inline synchronized method %s.%s",
861             clazz->descriptor, method->name);
862         return NULL;
863     }
864 
865     return method;
866 }
867 
868 /*
869  * Populate the methods table on first use.  It's possible the class
870  * hasn't been resolved yet, so we need to do the full "calling the
871  * method for the first time" routine.  (It's probably okay to skip
872  * the access checks.)
873  *
874  * Currently assuming that we're only inlining stuff loaded by the
875  * bootstrap class loader.  This is a safe assumption for many reasons.
876  */
dvmResolveInlineNative(int opIndex)877 Method* dvmResolveInlineNative(int opIndex)
878 {
879     assert(opIndex >= 0 && opIndex < NELEM(gDvmInlineOpsTable));
880     Method* method = gDvm.inlinedMethods[opIndex];
881     if (method != NULL) {
882         return method;
883     }
884 
885     method = dvmFindInlinableMethod(
886         gDvmInlineOpsTable[opIndex].classDescriptor,
887         gDvmInlineOpsTable[opIndex].methodName,
888         gDvmInlineOpsTable[opIndex].methodSignature);
889 
890     if (method == NULL) {
891         /* We already reported the error. */
892         return NULL;
893     }
894 
895     gDvm.inlinedMethods[opIndex] = method;
896     IF_ALOGV() {
897         char* desc = dexProtoCopyMethodDescriptor(&method->prototype);
898         ALOGV("Registered for profile: %s.%s %s",
899             method->clazz->descriptor, method->name, desc);
900         free(desc);
901     }
902 
903     return method;
904 }
905 
906 /*
907  * Make an inline call for the "debug" interpreter, used when the debugger
908  * or profiler is active.
909  */
dvmPerformInlineOp4Dbg(u4 arg0,u4 arg1,u4 arg2,u4 arg3,JValue * pResult,int opIndex)910 bool dvmPerformInlineOp4Dbg(u4 arg0, u4 arg1, u4 arg2, u4 arg3,
911     JValue* pResult, int opIndex)
912 {
913     Method* method = dvmResolveInlineNative(opIndex);
914     if (method == NULL) {
915         return (*gDvmInlineOpsTable[opIndex].func)(arg0, arg1, arg2, arg3,
916             pResult);
917     }
918 
919     Thread* self = dvmThreadSelf();
920     TRACE_METHOD_ENTER(self, method);
921     bool result = (*gDvmInlineOpsTable[opIndex].func)(arg0, arg1, arg2, arg3,
922         pResult);
923     TRACE_METHOD_EXIT(self, method);
924     return result;
925 }
926