xref: /aosp_15_r20/art/test/602-deoptimizeable/src/Main.java (revision 795d594fd825385562da6b089ea9b2033f3abf5a)
1*795d594fSAndroid Build Coastguard Worker /*
2*795d594fSAndroid Build Coastguard Worker  * Copyright (C) 2016 The Android Open Source Project
3*795d594fSAndroid Build Coastguard Worker  *
4*795d594fSAndroid Build Coastguard Worker  * Licensed under the Apache License, Version 2.0 (the "License");
5*795d594fSAndroid Build Coastguard Worker  * you may not use this file except in compliance with the License.
6*795d594fSAndroid Build Coastguard Worker  * You may obtain a copy of the License at
7*795d594fSAndroid Build Coastguard Worker  *
8*795d594fSAndroid Build Coastguard Worker  *      http://www.apache.org/licenses/LICENSE-2.0
9*795d594fSAndroid Build Coastguard Worker  *
10*795d594fSAndroid Build Coastguard Worker  * Unless required by applicable law or agreed to in writing, software
11*795d594fSAndroid Build Coastguard Worker  * distributed under the License is distributed on an "AS IS" BASIS,
12*795d594fSAndroid Build Coastguard Worker  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*795d594fSAndroid Build Coastguard Worker  * See the License for the specific language governing permissions and
14*795d594fSAndroid Build Coastguard Worker  * limitations under the License.
15*795d594fSAndroid Build Coastguard Worker  */
16*795d594fSAndroid Build Coastguard Worker 
17*795d594fSAndroid Build Coastguard Worker import java.lang.reflect.Method;
18*795d594fSAndroid Build Coastguard Worker import java.util.Arrays;
19*795d594fSAndroid Build Coastguard Worker import java.util.Comparator;
20*795d594fSAndroid Build Coastguard Worker import java.util.HashMap;
21*795d594fSAndroid Build Coastguard Worker 
22*795d594fSAndroid Build Coastguard Worker class TestObject {
23*795d594fSAndroid Build Coastguard Worker     public static boolean sHashCodeInvoked = false;
24*795d594fSAndroid Build Coastguard Worker     private int i;
25*795d594fSAndroid Build Coastguard Worker 
TestObject(int i)26*795d594fSAndroid Build Coastguard Worker     public TestObject(int i) {
27*795d594fSAndroid Build Coastguard Worker         this.i = i;
28*795d594fSAndroid Build Coastguard Worker     }
29*795d594fSAndroid Build Coastguard Worker 
equals(Object obj)30*795d594fSAndroid Build Coastguard Worker     public boolean equals(Object obj) {
31*795d594fSAndroid Build Coastguard Worker         return (obj instanceof TestObject) && (i == ((TestObject)obj).i);
32*795d594fSAndroid Build Coastguard Worker     }
33*795d594fSAndroid Build Coastguard Worker 
hashCode()34*795d594fSAndroid Build Coastguard Worker     public int hashCode() {
35*795d594fSAndroid Build Coastguard Worker         sHashCodeInvoked = true;
36*795d594fSAndroid Build Coastguard Worker         Main.deoptimizeAll();
37*795d594fSAndroid Build Coastguard Worker         return i % 64;
38*795d594fSAndroid Build Coastguard Worker     }
39*795d594fSAndroid Build Coastguard Worker }
40*795d594fSAndroid Build Coastguard Worker 
41*795d594fSAndroid Build Coastguard Worker public class Main {
42*795d594fSAndroid Build Coastguard Worker     static boolean sFlag = false;
43*795d594fSAndroid Build Coastguard Worker 
deoptimizeAll()44*795d594fSAndroid Build Coastguard Worker     public static native void deoptimizeAll();
undeoptimizeAll()45*795d594fSAndroid Build Coastguard Worker     public static native void undeoptimizeAll();
46*795d594fSAndroid Build Coastguard Worker 
execute(Runnable runnable)47*795d594fSAndroid Build Coastguard Worker     public static void execute(Runnable runnable) throws Exception {
48*795d594fSAndroid Build Coastguard Worker       Thread t = new Thread(runnable);
49*795d594fSAndroid Build Coastguard Worker       t.start();
50*795d594fSAndroid Build Coastguard Worker       t.join();
51*795d594fSAndroid Build Coastguard Worker     }
52*795d594fSAndroid Build Coastguard Worker 
main(String[] args)53*795d594fSAndroid Build Coastguard Worker     public static void main(String[] args) throws Exception {
54*795d594fSAndroid Build Coastguard Worker         System.loadLibrary(args[0]);
55*795d594fSAndroid Build Coastguard Worker         final HashMap<TestObject, Long> map = new HashMap<TestObject, Long>();
56*795d594fSAndroid Build Coastguard Worker 
57*795d594fSAndroid Build Coastguard Worker         // Single-frame deoptimization that covers partial fragment.
58*795d594fSAndroid Build Coastguard Worker         execute(new Runnable() {
59*795d594fSAndroid Build Coastguard Worker             public void run() {
60*795d594fSAndroid Build Coastguard Worker                 int[] arr = new int[3];
61*795d594fSAndroid Build Coastguard Worker                 int res = $noinline$run1(arr);
62*795d594fSAndroid Build Coastguard Worker                 if (res != 79) {
63*795d594fSAndroid Build Coastguard Worker                     System.out.println("Failure 1!");
64*795d594fSAndroid Build Coastguard Worker                     System.exit(0);
65*795d594fSAndroid Build Coastguard Worker                 }
66*795d594fSAndroid Build Coastguard Worker             }
67*795d594fSAndroid Build Coastguard Worker         });
68*795d594fSAndroid Build Coastguard Worker 
69*795d594fSAndroid Build Coastguard Worker         // Single-frame deoptimization that covers a full fragment.
70*795d594fSAndroid Build Coastguard Worker         execute(new Runnable() {
71*795d594fSAndroid Build Coastguard Worker             public void run() {
72*795d594fSAndroid Build Coastguard Worker                 try {
73*795d594fSAndroid Build Coastguard Worker                     int[] arr = new int[3];
74*795d594fSAndroid Build Coastguard Worker                     // Use reflection to call $noinline$run2 so that it does
75*795d594fSAndroid Build Coastguard Worker                     // full-fragment deoptimization since that is an upcall.
76*795d594fSAndroid Build Coastguard Worker                     Class<?> cls = Class.forName("Main");
77*795d594fSAndroid Build Coastguard Worker                     Method method = cls.getDeclaredMethod("$noinline$run2", int[].class);
78*795d594fSAndroid Build Coastguard Worker                     double res = (double)method.invoke(Main.class, arr);
79*795d594fSAndroid Build Coastguard Worker                     if (res != 79.3d) {
80*795d594fSAndroid Build Coastguard Worker                         System.out.println("Failure 2!");
81*795d594fSAndroid Build Coastguard Worker                         System.exit(0);
82*795d594fSAndroid Build Coastguard Worker                     }
83*795d594fSAndroid Build Coastguard Worker                 } catch (Exception e) {
84*795d594fSAndroid Build Coastguard Worker                     e.printStackTrace(System.out);
85*795d594fSAndroid Build Coastguard Worker                 }
86*795d594fSAndroid Build Coastguard Worker             }
87*795d594fSAndroid Build Coastguard Worker         });
88*795d594fSAndroid Build Coastguard Worker 
89*795d594fSAndroid Build Coastguard Worker         // Full-fragment deoptimization.
90*795d594fSAndroid Build Coastguard Worker         execute(new Runnable() {
91*795d594fSAndroid Build Coastguard Worker             public void run() {
92*795d594fSAndroid Build Coastguard Worker                 float res = $noinline$run3B();
93*795d594fSAndroid Build Coastguard Worker                 if (res != 0.034f) {
94*795d594fSAndroid Build Coastguard Worker                     System.out.println("Failure 3!");
95*795d594fSAndroid Build Coastguard Worker                     System.exit(0);
96*795d594fSAndroid Build Coastguard Worker                 }
97*795d594fSAndroid Build Coastguard Worker             }
98*795d594fSAndroid Build Coastguard Worker         });
99*795d594fSAndroid Build Coastguard Worker 
100*795d594fSAndroid Build Coastguard Worker         undeoptimizeAll();  // Make compiled code useable again.
101*795d594fSAndroid Build Coastguard Worker 
102*795d594fSAndroid Build Coastguard Worker         // Partial-fragment deoptimization.
103*795d594fSAndroid Build Coastguard Worker         execute(new Runnable() {
104*795d594fSAndroid Build Coastguard Worker             public void run() {
105*795d594fSAndroid Build Coastguard Worker                 try {
106*795d594fSAndroid Build Coastguard Worker                     map.put(new TestObject(10), Long.valueOf(100));
107*795d594fSAndroid Build Coastguard Worker                     if (map.get(new TestObject(10)) == null) {
108*795d594fSAndroid Build Coastguard Worker                         System.out.println("Expected map to contain TestObject(10)");
109*795d594fSAndroid Build Coastguard Worker                     }
110*795d594fSAndroid Build Coastguard Worker                 } catch (Exception e) {
111*795d594fSAndroid Build Coastguard Worker                     e.printStackTrace(System.out);
112*795d594fSAndroid Build Coastguard Worker                 }
113*795d594fSAndroid Build Coastguard Worker             }
114*795d594fSAndroid Build Coastguard Worker         });
115*795d594fSAndroid Build Coastguard Worker 
116*795d594fSAndroid Build Coastguard Worker         undeoptimizeAll();  // Make compiled code useable again.
117*795d594fSAndroid Build Coastguard Worker 
118*795d594fSAndroid Build Coastguard Worker         if (!TestObject.sHashCodeInvoked) {
119*795d594fSAndroid Build Coastguard Worker             System.out.println("hashCode() method not invoked!");
120*795d594fSAndroid Build Coastguard Worker         }
121*795d594fSAndroid Build Coastguard Worker         if (map.get(new TestObject(10)) != 100) {
122*795d594fSAndroid Build Coastguard Worker             System.out.println("Wrong hashmap value!");
123*795d594fSAndroid Build Coastguard Worker         }
124*795d594fSAndroid Build Coastguard Worker         System.out.println("Finishing");
125*795d594fSAndroid Build Coastguard Worker     }
126*795d594fSAndroid Build Coastguard Worker 
$noinline$run1(int[] arr)127*795d594fSAndroid Build Coastguard Worker     public static int $noinline$run1(int[] arr) {
128*795d594fSAndroid Build Coastguard Worker         // Prevent inlining.
129*795d594fSAndroid Build Coastguard Worker         if (sFlag) {
130*795d594fSAndroid Build Coastguard Worker             throw new Error();
131*795d594fSAndroid Build Coastguard Worker         }
132*795d594fSAndroid Build Coastguard Worker         boolean caught = false;
133*795d594fSAndroid Build Coastguard Worker         // BCE will use deoptimization for the code below.
134*795d594fSAndroid Build Coastguard Worker         try {
135*795d594fSAndroid Build Coastguard Worker             arr[0] = 1;
136*795d594fSAndroid Build Coastguard Worker             arr[1] = 1;
137*795d594fSAndroid Build Coastguard Worker             arr[2] = 1;
138*795d594fSAndroid Build Coastguard Worker             // This causes AIOOBE and triggers deoptimization from compiled code.
139*795d594fSAndroid Build Coastguard Worker             arr[3] = 1;
140*795d594fSAndroid Build Coastguard Worker         } catch (ArrayIndexOutOfBoundsException e) {
141*795d594fSAndroid Build Coastguard Worker             caught = true;
142*795d594fSAndroid Build Coastguard Worker         }
143*795d594fSAndroid Build Coastguard Worker         if (!caught) {
144*795d594fSAndroid Build Coastguard Worker             System.out.println("Expected exception");
145*795d594fSAndroid Build Coastguard Worker         }
146*795d594fSAndroid Build Coastguard Worker         return 79;
147*795d594fSAndroid Build Coastguard Worker     }
148*795d594fSAndroid Build Coastguard Worker 
$noinline$run2(int[] arr)149*795d594fSAndroid Build Coastguard Worker     public static double $noinline$run2(int[] arr) {
150*795d594fSAndroid Build Coastguard Worker         // Prevent inlining.
151*795d594fSAndroid Build Coastguard Worker         if (sFlag) {
152*795d594fSAndroid Build Coastguard Worker             throw new Error();
153*795d594fSAndroid Build Coastguard Worker         }
154*795d594fSAndroid Build Coastguard Worker         boolean caught = false;
155*795d594fSAndroid Build Coastguard Worker         // BCE will use deoptimization for the code below.
156*795d594fSAndroid Build Coastguard Worker         try {
157*795d594fSAndroid Build Coastguard Worker             arr[0] = 1;
158*795d594fSAndroid Build Coastguard Worker             arr[1] = 1;
159*795d594fSAndroid Build Coastguard Worker             arr[2] = 1;
160*795d594fSAndroid Build Coastguard Worker             // This causes AIOOBE and triggers deoptimization from compiled code.
161*795d594fSAndroid Build Coastguard Worker             arr[3] = 1;
162*795d594fSAndroid Build Coastguard Worker         } catch (ArrayIndexOutOfBoundsException e) {
163*795d594fSAndroid Build Coastguard Worker             caught = true;
164*795d594fSAndroid Build Coastguard Worker         }
165*795d594fSAndroid Build Coastguard Worker         if (!caught) {
166*795d594fSAndroid Build Coastguard Worker             System.out.println("Expected exception");
167*795d594fSAndroid Build Coastguard Worker         }
168*795d594fSAndroid Build Coastguard Worker         return 79.3d;
169*795d594fSAndroid Build Coastguard Worker     }
170*795d594fSAndroid Build Coastguard Worker 
$noinline$run3A()171*795d594fSAndroid Build Coastguard Worker     public static float $noinline$run3A() {
172*795d594fSAndroid Build Coastguard Worker         // Prevent inlining.
173*795d594fSAndroid Build Coastguard Worker         if (sFlag) {
174*795d594fSAndroid Build Coastguard Worker             throw new Error();
175*795d594fSAndroid Build Coastguard Worker         }
176*795d594fSAndroid Build Coastguard Worker         // Deoptimize callers.
177*795d594fSAndroid Build Coastguard Worker         deoptimizeAll();
178*795d594fSAndroid Build Coastguard Worker         return 0.034f;
179*795d594fSAndroid Build Coastguard Worker     }
180*795d594fSAndroid Build Coastguard Worker 
$noinline$run3B()181*795d594fSAndroid Build Coastguard Worker     public static float $noinline$run3B() {
182*795d594fSAndroid Build Coastguard Worker         // Prevent inlining.
183*795d594fSAndroid Build Coastguard Worker         if (sFlag) {
184*795d594fSAndroid Build Coastguard Worker             throw new Error();
185*795d594fSAndroid Build Coastguard Worker         }
186*795d594fSAndroid Build Coastguard Worker         float res = $noinline$run3A();
187*795d594fSAndroid Build Coastguard Worker         return res;
188*795d594fSAndroid Build Coastguard Worker     }
189*795d594fSAndroid Build Coastguard Worker }
190