xref: /aosp_15_r20/art/test/530-checker-regression-reftyp-final/src/Main.java (revision 795d594fd825385562da6b089ea9b2033f3abf5a)
1*795d594fSAndroid Build Coastguard Worker /*
2*795d594fSAndroid Build Coastguard Worker  * Copyright (C) 2015 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 
19*795d594fSAndroid Build Coastguard Worker public class Main  {
20*795d594fSAndroid Build Coastguard Worker 
21*795d594fSAndroid Build Coastguard Worker   class MyClassA {}
22*795d594fSAndroid Build Coastguard Worker   class MyClassB extends MyClassA {}
23*795d594fSAndroid Build Coastguard Worker 
main(String[] args)24*795d594fSAndroid Build Coastguard Worker   public static void main(String[] args) throws Exception {
25*795d594fSAndroid Build Coastguard Worker     testReferenceTypePropagation();
26*795d594fSAndroid Build Coastguard Worker     invokeTestInliner();
27*795d594fSAndroid Build Coastguard Worker   }
28*795d594fSAndroid Build Coastguard Worker 
29*795d594fSAndroid Build Coastguard Worker   // Reference type propagation (RTP) used to assume that if a class is final,
30*795d594fSAndroid Build Coastguard Worker   // then the type must be exact. This does not hold for arrays which are always
31*795d594fSAndroid Build Coastguard Worker   // final, i.e. not extendable, but may be assigned to from values of the
32*795d594fSAndroid Build Coastguard Worker   // components type subclasses.
33*795d594fSAndroid Build Coastguard Worker 
testReferenceTypePropagation()34*795d594fSAndroid Build Coastguard Worker   public static void testReferenceTypePropagation() throws Exception {
35*795d594fSAndroid Build Coastguard Worker     boolean expectTrue;
36*795d594fSAndroid Build Coastguard Worker 
37*795d594fSAndroid Build Coastguard Worker     // Bug #1: RTP would set the type of `array` to exact Object[]. Instruction
38*795d594fSAndroid Build Coastguard Worker     // simplifier would then simplify the instanceof to `false`.
39*795d594fSAndroid Build Coastguard Worker     Object[] array = $noinline$getArray();
40*795d594fSAndroid Build Coastguard Worker     expectTrue = array instanceof MyClassA[];
41*795d594fSAndroid Build Coastguard Worker     if (!expectTrue) {
42*795d594fSAndroid Build Coastguard Worker       throw new Exception("Incorrect type check.");
43*795d594fSAndroid Build Coastguard Worker     }
44*795d594fSAndroid Build Coastguard Worker 
45*795d594fSAndroid Build Coastguard Worker     // Bug #2: This is the true-branch of the instanceof above. The bound type
46*795d594fSAndroid Build Coastguard Worker     // for `array` would be again set to exact MyClassA[] and incorrectly
47*795d594fSAndroid Build Coastguard Worker     // simplify the second instanceof to `false`.
48*795d594fSAndroid Build Coastguard Worker     expectTrue = array instanceof MyClassB[];
49*795d594fSAndroid Build Coastguard Worker     if (!expectTrue) {
50*795d594fSAndroid Build Coastguard Worker       throw new Exception("Incorrect type bound.");
51*795d594fSAndroid Build Coastguard Worker     }
52*795d594fSAndroid Build Coastguard Worker   }
53*795d594fSAndroid Build Coastguard Worker 
invokeTestInliner()54*795d594fSAndroid Build Coastguard Worker   public static void invokeTestInliner() throws Exception {
55*795d594fSAndroid Build Coastguard Worker     Class<?> c = Class.forName("TestCase");
56*795d594fSAndroid Build Coastguard Worker     Method m = c.getMethod("testInliner");
57*795d594fSAndroid Build Coastguard Worker     m.invoke(null);
58*795d594fSAndroid Build Coastguard Worker   }
59*795d594fSAndroid Build Coastguard Worker 
$noinline$getArray()60*795d594fSAndroid Build Coastguard Worker   public static Object[] $noinline$getArray() {
61*795d594fSAndroid Build Coastguard Worker     return new MyClassB[2];
62*795d594fSAndroid Build Coastguard Worker   }
63*795d594fSAndroid Build Coastguard Worker }
64