/* * Copyright (C) 2018 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 void main(String args[]) { simpleTest(); hierarchyTest(); } public static void simpleTest() { // Partial initialization of Bad; ignoring the error. Error badClinit = null; try { new Bad(11); } catch (Error e) { badClinit = e; } // Call foo() on the escaped instance of Bad. try { bad.foo(); } catch (NoClassDefFoundError ncdfe) { // On RI, the NCDFE has no cause. On ART, the badClinit is the cause. if (ncdfe.getCause() == badClinit || ncdfe.getCause() == null) { System.out.println("Caught NoClassDefFoundError."); } else { ncdfe.printStackTrace(); } } // Call bar() on the escaped instance of Bad. try { bad.bar(); } catch (NoClassDefFoundError ncdfe) { // On RI, the NCDFE has no cause. On ART, the badClinit is the cause. if (ncdfe.getCause() == badClinit || ncdfe.getCause() == null) { System.out.println("Caught NoClassDefFoundError."); } else { ncdfe.printStackTrace(); } } // Test that we handle bad instance correctly in the resolution trampoline. bad.$noinline$testResolutionTrampoline(); } public static void hierarchyTest() { // Partial initialization of BadSuper; ignoring the error. Fully initializes BadSub. Error badClinit = null; try { new BadSuper(0); } catch (Error e) { badClinit = e; } // Call BadSuper.foo() on the escaped instance of BadSuper. try { badSuper.foo(); } catch (NoClassDefFoundError ncdfe) { // On RI, the NCDFE has no cause. On ART, the badClinit is the cause. if (ncdfe.getCause() == badClinit || ncdfe.getCause() == null) { System.out.println("Caught NoClassDefFoundError."); } else { ncdfe.printStackTrace(); } } // Call BadSub.bar() on the escaped instance of BadSub. try { badSub.bar(); } catch (NoClassDefFoundError ncdfe) { // On RI, the NCDFE has no cause. On ART, the badClinit is the cause. if (ncdfe.getCause() == badClinit || ncdfe.getCause() == null) { System.out.println("Caught NoClassDefFoundError."); } else { ncdfe.printStackTrace(); } } // Test that we can even create instances of BadSub with erroneous superclass BadSuper. try { new BadSub(-1, -2).bar(); } catch (NoClassDefFoundError ncdfe) { // On RI, the NCDFE has no cause. On ART, the badClinit is the cause. if (ncdfe.getCause() == badClinit || ncdfe.getCause() == null) { System.out.println("Caught NoClassDefFoundError."); } else { ncdfe.printStackTrace(); } } // Test that we cannot create instances of BadSuper from BadSub. try { badSub.allocSuper(11111); // Should throw. System.out.println("Allocated BadSuper!"); } catch (NoClassDefFoundError ncdfe) { // On RI, the NCDFE has no cause. On ART, the badClinit is the cause. if (ncdfe.getCause() == badClinit || ncdfe.getCause() == null) { System.out.println("Caught NoClassDefFoundError."); } else { ncdfe.printStackTrace(); } } } public static Bad bad; public static BadSuper badSuper; public static BadSub badSub; } class Bad { static { // Create an instance of Bad and let it escape in Main.bad. Main.bad = new Bad(33); staticValue = 42; if (true) { throw new Error("Bad "); } } public void foo() { System.out.println("Bad.foo()"); System.out.println("Bad.instanceValue = " + instanceValue); System.out.println("Bad.staticValue = " + staticValue); } public void bar() { System.out.println("Bad.bar()"); System.out.println("Bad.staticValue [indirect] = " + Helper.$inline$getBadStaticValue()); } public Bad(int iv) { instanceValue = iv; } public int instanceValue; public static int staticValue; public static class Helper { public static int $inline$getBadStaticValue() { return Bad.staticValue; } } public void $noinline$testResolutionTrampoline() { // The first call to private method uses the resolution trampoline when AOT-compiled. $noinline$testResolutionTrampolineCallee(); } private void $noinline$testResolutionTrampolineCallee() { System.out.println("Bad.$noinline$testResolutionTrampolineCallee()"); } } class BadSuper { static { Main.badSuper = new BadSuper(1); Main.badSub = new BadSub(11, 111); // Fully initializes BadSub. BadSuper.superStaticValue = 42; BadSub.subStaticValue = 4242; if (true) { throw new Error("Bad "); } } public void foo() { System.out.println("BadSuper.foo()"); System.out.println("BadSuper.superInstanceValue = " + superInstanceValue); System.out.println("BadSuper.superStaticValue = " + superStaticValue); } public BadSuper(int superiv) { superInstanceValue = superiv; } public int superInstanceValue; public static int superStaticValue; } // Note: If we tried to initialize BadSub before BadSuper, it would end up erroneous // because the superclass fails initialization. However, since we start initializing the // BadSuper first, BadSub is initialized successfully while BadSuper is "initializing" // and remains initialized after the BadSuper's class initializer throws. class BadSub extends BadSuper { public void bar() { System.out.println("BadSub.bar()"); System.out.println("BadSub.subInstanceValue = " + subInstanceValue); System.out.println("BadSub.subStaticValue = " + subStaticValue); System.out.println("BadSuper.superInstanceValue = " + superInstanceValue); System.out.println("BadSuper.superStaticValue = " + superStaticValue); } public BadSuper allocSuper(int superiv) { System.out.println("BadSub.allocSuper(.)"); return new BadSuper(superiv); } public BadSub(int subiv, int superiv) { super(superiv); subInstanceValue = subiv; } public int subInstanceValue; public static int subStaticValue; }