/* * Copyright (C) 2015 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ public class Main { public static boolean $inline$classTypeTest(Object o) { return o instanceof SubMain; } public static boolean $inline$interfaceTypeTest(Object o) { return o instanceof Itf; } public static SubMain subMain; public static Main mainField; public static Unrelated unrelatedField; public static FinalUnrelated finalUnrelatedField; /// CHECK-START: boolean Main.classTypeTestNull() register (after) /// CHECK-DAG: <<Const:i\d+>> IntConstant 0 /// CHECK-DAG: Return [<<Const>>] public static boolean classTypeTestNull() { return $inline$classTypeTest(null); } /// CHECK-START: boolean Main.classTypeTestExactMain() register (after) /// CHECK-DAG: <<Const:i\d+>> IntConstant 0 /// CHECK-DAG: Return [<<Const>>] public static boolean classTypeTestExactMain() { return $inline$classTypeTest(new Main()); } /// CHECK-START: boolean Main.classTypeTestExactSubMain() register (after) /// CHECK-DAG: <<Const:i\d+>> IntConstant 1 /// CHECK-DAG: Return [<<Const>>] public static boolean classTypeTestExactSubMain() { return $inline$classTypeTest(new SubMain()); } /// CHECK-START: boolean Main.classTypeTestSubMainOrNull() register (after) /// CHECK-DAG: <<Value:z\d+>> NotEqual /// CHECK-DAG: Return [<<Value>>] public static boolean classTypeTestSubMainOrNull() { return $inline$classTypeTest(subMain); } /// CHECK-START: boolean Main.classTypeTestMainOrNull() register (after) /// CHECK-DAG: <<Value:z\d+>> InstanceOf /// CHECK-DAG: Return [<<Value>>] public static boolean classTypeTestMainOrNull() { return $inline$classTypeTest(mainField); } /// CHECK-START: boolean Main.classTypeTestUnrelated() register (after) /// CHECK-DAG: <<Const:i\d+>> IntConstant 0 /// CHECK-DAG: Return [<<Const>>] public static boolean classTypeTestUnrelated() { return $inline$classTypeTest(unrelatedField); } /// CHECK-START: boolean Main.classTypeTestFinalUnrelated() register (after) /// CHECK-DAG: <<Const:i\d+>> IntConstant 0 /// CHECK-DAG: Return [<<Const>>] public static boolean classTypeTestFinalUnrelated() { return $inline$classTypeTest(finalUnrelatedField); } /// CHECK-START: boolean Main.interfaceTypeTestNull() register (after) /// CHECK-DAG: <<Const:i\d+>> IntConstant 0 /// CHECK-DAG: Return [<<Const>>] public static boolean interfaceTypeTestNull() { return $inline$interfaceTypeTest(null); } /// CHECK-START: boolean Main.interfaceTypeTestExactMain() register (after) /// CHECK-DAG: <<Const:i\d+>> IntConstant 0 /// CHECK-DAG: Return [<<Const>>] public static boolean interfaceTypeTestExactMain() { return $inline$interfaceTypeTest(new Main()); } /// CHECK-START: boolean Main.interfaceTypeTestExactSubMain() register (after) /// CHECK-DAG: <<Const:i\d+>> IntConstant 1 /// CHECK-DAG: Return [<<Const>>] public static boolean interfaceTypeTestExactSubMain() { return $inline$interfaceTypeTest(new SubMain()); } /// CHECK-START: boolean Main.interfaceTypeTestSubMainOrNull() register (after) /// CHECK-DAG: <<Value:z\d+>> NotEqual /// CHECK-DAG: Return [<<Value>>] public static boolean interfaceTypeTestSubMainOrNull() { return $inline$interfaceTypeTest(subMain); } /// CHECK-START: boolean Main.interfaceTypeTestMainOrNull() register (after) /// CHECK-DAG: <<Value:z\d+>> InstanceOf /// CHECK-DAG: Return [<<Value>>] public static boolean interfaceTypeTestMainOrNull() { return $inline$interfaceTypeTest(mainField); } /// CHECK-START: boolean Main.interfaceTypeTestUnrelated() register (after) /// CHECK-DAG: <<Value:z\d+>> InstanceOf /// CHECK-DAG: Return [<<Value>>] public static boolean interfaceTypeTestUnrelated() { // This method is the main difference between doing an instanceof on an interface // or a class. We have to keep the instanceof in case a subclass of Unrelated // implements the interface. return $inline$interfaceTypeTest(unrelatedField); } /// CHECK-START: boolean Main.interfaceTypeTestFinalUnrelated() register (after) /// CHECK-DAG: <<Const:i\d+>> IntConstant 0 /// CHECK-DAG: Return [<<Const>>] public static boolean interfaceTypeTestFinalUnrelated() { return $inline$interfaceTypeTest(finalUnrelatedField); } // Check that we remove the LoadClass instruction from the graph. /// CHECK-START: boolean Main.knownTestWithLoadedClass() register (after) /// CHECK-NOT: LoadClass public static boolean knownTestWithLoadedClass() { return new String() instanceof String; } // Check that we do not remove the LoadClass instruction from the graph. /// CHECK-START: boolean Main.knownTestWithUnloadedClass() register (after) /// CHECK: <<Const:i\d+>> IntConstant 0 /// CHECK: LoadClass /// CHECK: Return [<<Const>>] public static boolean knownTestWithUnloadedClass() { return $inline$returnUnrelated() instanceof String; } public static Object $inline$returnUnrelated() { return new Unrelated(); } public static void expect(boolean expected, boolean actual) { if (expected != actual) { throw new Error("Unexpected result"); } } public static void main(String[] args) { expect(false, classTypeTestNull()); expect(false, classTypeTestExactMain()); expect(true, classTypeTestExactSubMain()); subMain = null; expect(false, classTypeTestSubMainOrNull()); subMain = new SubMain(); expect(true, classTypeTestSubMainOrNull()); mainField = null; expect(false, classTypeTestMainOrNull()); mainField = new Main(); expect(false, classTypeTestMainOrNull()); mainField = new SubMain(); expect(true, classTypeTestMainOrNull()); unrelatedField = null; expect(false, classTypeTestUnrelated()); unrelatedField = new Unrelated(); expect(false, classTypeTestUnrelated()); finalUnrelatedField = null; expect(false, classTypeTestFinalUnrelated()); finalUnrelatedField = new FinalUnrelated(); expect(false, classTypeTestFinalUnrelated()); expect(false, interfaceTypeTestNull()); expect(false, interfaceTypeTestExactMain()); expect(true, interfaceTypeTestExactSubMain()); subMain = null; expect(false, interfaceTypeTestSubMainOrNull()); subMain = new SubMain(); expect(true, interfaceTypeTestSubMainOrNull()); mainField = null; expect(false, interfaceTypeTestMainOrNull()); mainField = new Main(); expect(false, interfaceTypeTestMainOrNull()); mainField = new SubMain(); expect(true, interfaceTypeTestMainOrNull()); unrelatedField = null; expect(false, interfaceTypeTestUnrelated()); unrelatedField = new Unrelated(); expect(false, interfaceTypeTestUnrelated()); finalUnrelatedField = null; expect(false, interfaceTypeTestFinalUnrelated()); finalUnrelatedField = new FinalUnrelated(); expect(false, interfaceTypeTestFinalUnrelated()); } } interface Itf { } class SubMain extends Main implements Itf { } class Unrelated { } final class FinalUnrelated { }