1 /* 2 * Copyright (C) 2015 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 public class Main { $inline$classTypeTest(Object o)18 public static boolean $inline$classTypeTest(Object o) { 19 return o instanceof SubMain; 20 } 21 $inline$interfaceTypeTest(Object o)22 public static boolean $inline$interfaceTypeTest(Object o) { 23 return o instanceof Itf; 24 } 25 26 public static SubMain subMain; 27 public static Main mainField; 28 public static Unrelated unrelatedField; 29 public static FinalUnrelated finalUnrelatedField; 30 31 /// CHECK-START: boolean Main.classTypeTestNull() register (after) 32 /// CHECK-DAG: <<Const:i\d+>> IntConstant 0 33 /// CHECK-DAG: Return [<<Const>>] classTypeTestNull()34 public static boolean classTypeTestNull() { 35 return $inline$classTypeTest(null); 36 } 37 38 /// CHECK-START: boolean Main.classTypeTestExactMain() register (after) 39 /// CHECK-DAG: <<Const:i\d+>> IntConstant 0 40 /// CHECK-DAG: Return [<<Const>>] classTypeTestExactMain()41 public static boolean classTypeTestExactMain() { 42 return $inline$classTypeTest(new Main()); 43 } 44 45 /// CHECK-START: boolean Main.classTypeTestExactSubMain() register (after) 46 /// CHECK-DAG: <<Const:i\d+>> IntConstant 1 47 /// CHECK-DAG: Return [<<Const>>] classTypeTestExactSubMain()48 public static boolean classTypeTestExactSubMain() { 49 return $inline$classTypeTest(new SubMain()); 50 } 51 52 /// CHECK-START: boolean Main.classTypeTestSubMainOrNull() register (after) 53 /// CHECK-DAG: <<Value:z\d+>> NotEqual 54 /// CHECK-DAG: Return [<<Value>>] classTypeTestSubMainOrNull()55 public static boolean classTypeTestSubMainOrNull() { 56 return $inline$classTypeTest(subMain); 57 } 58 59 /// CHECK-START: boolean Main.classTypeTestMainOrNull() register (after) 60 /// CHECK-DAG: <<Value:z\d+>> InstanceOf 61 /// CHECK-DAG: Return [<<Value>>] classTypeTestMainOrNull()62 public static boolean classTypeTestMainOrNull() { 63 return $inline$classTypeTest(mainField); 64 } 65 66 /// CHECK-START: boolean Main.classTypeTestUnrelated() register (after) 67 /// CHECK-DAG: <<Const:i\d+>> IntConstant 0 68 /// CHECK-DAG: Return [<<Const>>] classTypeTestUnrelated()69 public static boolean classTypeTestUnrelated() { 70 return $inline$classTypeTest(unrelatedField); 71 } 72 73 /// CHECK-START: boolean Main.classTypeTestFinalUnrelated() register (after) 74 /// CHECK-DAG: <<Const:i\d+>> IntConstant 0 75 /// CHECK-DAG: Return [<<Const>>] classTypeTestFinalUnrelated()76 public static boolean classTypeTestFinalUnrelated() { 77 return $inline$classTypeTest(finalUnrelatedField); 78 } 79 80 /// CHECK-START: boolean Main.interfaceTypeTestNull() register (after) 81 /// CHECK-DAG: <<Const:i\d+>> IntConstant 0 82 /// CHECK-DAG: Return [<<Const>>] interfaceTypeTestNull()83 public static boolean interfaceTypeTestNull() { 84 return $inline$interfaceTypeTest(null); 85 } 86 87 /// CHECK-START: boolean Main.interfaceTypeTestExactMain() register (after) 88 /// CHECK-DAG: <<Const:i\d+>> IntConstant 0 89 /// CHECK-DAG: Return [<<Const>>] interfaceTypeTestExactMain()90 public static boolean interfaceTypeTestExactMain() { 91 return $inline$interfaceTypeTest(new Main()); 92 } 93 94 /// CHECK-START: boolean Main.interfaceTypeTestExactSubMain() register (after) 95 /// CHECK-DAG: <<Const:i\d+>> IntConstant 1 96 /// CHECK-DAG: Return [<<Const>>] interfaceTypeTestExactSubMain()97 public static boolean interfaceTypeTestExactSubMain() { 98 return $inline$interfaceTypeTest(new SubMain()); 99 } 100 101 /// CHECK-START: boolean Main.interfaceTypeTestSubMainOrNull() register (after) 102 /// CHECK-DAG: <<Value:z\d+>> NotEqual 103 /// CHECK-DAG: Return [<<Value>>] interfaceTypeTestSubMainOrNull()104 public static boolean interfaceTypeTestSubMainOrNull() { 105 return $inline$interfaceTypeTest(subMain); 106 } 107 108 /// CHECK-START: boolean Main.interfaceTypeTestMainOrNull() register (after) 109 /// CHECK-DAG: <<Value:z\d+>> InstanceOf 110 /// CHECK-DAG: Return [<<Value>>] interfaceTypeTestMainOrNull()111 public static boolean interfaceTypeTestMainOrNull() { 112 return $inline$interfaceTypeTest(mainField); 113 } 114 115 /// CHECK-START: boolean Main.interfaceTypeTestUnrelated() register (after) 116 /// CHECK-DAG: <<Value:z\d+>> InstanceOf 117 /// CHECK-DAG: Return [<<Value>>] interfaceTypeTestUnrelated()118 public static boolean interfaceTypeTestUnrelated() { 119 // This method is the main difference between doing an instanceof on an interface 120 // or a class. We have to keep the instanceof in case a subclass of Unrelated 121 // implements the interface. 122 return $inline$interfaceTypeTest(unrelatedField); 123 } 124 125 /// CHECK-START: boolean Main.interfaceTypeTestFinalUnrelated() register (after) 126 /// CHECK-DAG: <<Const:i\d+>> IntConstant 0 127 /// CHECK-DAG: Return [<<Const>>] interfaceTypeTestFinalUnrelated()128 public static boolean interfaceTypeTestFinalUnrelated() { 129 return $inline$interfaceTypeTest(finalUnrelatedField); 130 } 131 132 // Check that we remove the LoadClass instruction from the graph. 133 /// CHECK-START: boolean Main.knownTestWithLoadedClass() register (after) 134 /// CHECK-NOT: LoadClass knownTestWithLoadedClass()135 public static boolean knownTestWithLoadedClass() { 136 return new String() instanceof String; 137 } 138 139 // Check that we do not remove the LoadClass instruction from the graph. 140 /// CHECK-START: boolean Main.knownTestWithUnloadedClass() register (after) 141 /// CHECK: <<Const:i\d+>> IntConstant 0 142 /// CHECK: LoadClass 143 /// CHECK: Return [<<Const>>] knownTestWithUnloadedClass()144 public static boolean knownTestWithUnloadedClass() { 145 return $inline$returnMain() instanceof String; 146 } 147 $inline$returnMain()148 public static Object $inline$returnMain() { 149 return new Main(); 150 } 151 expect(boolean expected, boolean actual)152 public static void expect(boolean expected, boolean actual) { 153 if (expected != actual) { 154 throw new Error("Unexpected result"); 155 } 156 } 157 main(String[] args)158 public static void main(String[] args) { 159 expect(false, classTypeTestNull()); 160 expect(false, classTypeTestExactMain()); 161 expect(true, classTypeTestExactSubMain()); 162 163 subMain = null; 164 expect(false, classTypeTestSubMainOrNull()); 165 subMain = new SubMain(); 166 expect(true, classTypeTestSubMainOrNull()); 167 168 mainField = null; 169 expect(false, classTypeTestMainOrNull()); 170 mainField = new Main(); 171 expect(false, classTypeTestMainOrNull()); 172 mainField = new SubMain(); 173 expect(true, classTypeTestMainOrNull()); 174 175 unrelatedField = null; 176 expect(false, classTypeTestUnrelated()); 177 unrelatedField = new Unrelated(); 178 expect(false, classTypeTestUnrelated()); 179 180 finalUnrelatedField = null; 181 expect(false, classTypeTestFinalUnrelated()); 182 finalUnrelatedField = new FinalUnrelated(); 183 expect(false, classTypeTestFinalUnrelated()); 184 185 expect(false, interfaceTypeTestNull()); 186 expect(false, interfaceTypeTestExactMain()); 187 expect(true, interfaceTypeTestExactSubMain()); 188 189 subMain = null; 190 expect(false, interfaceTypeTestSubMainOrNull()); 191 subMain = new SubMain(); 192 expect(true, interfaceTypeTestSubMainOrNull()); 193 194 mainField = null; 195 expect(false, interfaceTypeTestMainOrNull()); 196 mainField = new Main(); 197 expect(false, interfaceTypeTestMainOrNull()); 198 mainField = new SubMain(); 199 expect(true, interfaceTypeTestMainOrNull()); 200 201 unrelatedField = null; 202 expect(false, interfaceTypeTestUnrelated()); 203 unrelatedField = new Unrelated(); 204 expect(false, interfaceTypeTestUnrelated()); 205 206 finalUnrelatedField = null; 207 expect(false, interfaceTypeTestFinalUnrelated()); 208 finalUnrelatedField = new FinalUnrelated(); 209 expect(false, interfaceTypeTestFinalUnrelated()); 210 } 211 } 212 213 interface Itf { 214 } 215 216 class SubMain extends Main implements Itf { 217 } 218 219 class Unrelated { 220 } 221 222 final class FinalUnrelated { 223 } 224