/*
|
* 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 {
|
}
|