/*
 * Copyright (C) 2007 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.
 */

// An error class.
class BadError extends Error {
    public BadError(String s) {
        super("This is bad by convention: " + s);
    }
}

// A class that throws BadError during static initialization.
class BadInit {
    static int dummy;
    static {
        System.out.println("Static Init");
        if (true) {
            throw new BadError("BadInit");
        }
    }
}

// An error that doesn't have a <init>(String) method.
class BadErrorNoStringInit extends Error {
    public BadErrorNoStringInit() {
        super("This is bad by convention");
    }
}

// A class that throws BadErrorNoStringInit during static initialization.
class BadInitNoStringInit {
    static int dummy;
    static {
        System.out.println("Static BadInitNoStringInit");
        if (true) {
            throw new BadErrorNoStringInit();
        }
    }
}

// A class that throws BadError during static initialization, serving as a super class.
class BadSuperClass {
 static int dummy;
 static {
     System.out.println("BadSuperClass Static Init");
     if (true) {
         throw new BadError("BadInit");
     }
 }
}

// A class that derives from BadSuperClass.
class DerivedFromBadSuperClass extends BadSuperClass {
}

/**
 * Exceptions across method calls
 */
public class Main {
    public static void exceptions_007() {
        try {
            catchAndRethrow();
        } catch (NullPointerException npe) {
            System.out.print("Got an NPE: ");
            System.out.println(npe.getMessage());
            npe.printStackTrace(System.out);
        }
    }
    public static void main(String args[]) {
        exceptions_007();
        exceptionsRethrowClassInitFailure();
        exceptionsRethrowClassInitFailureNoStringInit();
        exceptionsForSuperClassInitFailure();
        exceptionsInMultiDex();
    }

    private static void catchAndRethrow() {
        try {
            throwNullPointerException();
        } catch (NullPointerException npe) {
            NullPointerException npe2;
            npe2 = new NullPointerException("second throw");
            npe2.initCause(npe);
            throw npe2;
        }
    }

    private static void throwNullPointerException() {
        throw new NullPointerException("first throw");
    }

    private static void exceptionsRethrowClassInitFailure() {
        try {
            try {
                BadInit.dummy = 1;
                throw new IllegalStateException("Should not reach here.");
            } catch (BadError e) {
                System.out.println(e);
            }

            // Check if it works a second time.

            try {
                BadInit.dummy = 1;
                throw new IllegalStateException("Should not reach here.");
            } catch (NoClassDefFoundError e) {
                System.out.println(e);
                System.out.println(e.getCause());
            }
        } catch (Exception error) {
            error.printStackTrace(System.out);
        }
    }

    private static void exceptionsRethrowClassInitFailureNoStringInit() {
        try {
            try {
                BadInitNoStringInit.dummy = 1;
                throw new IllegalStateException("Should not reach here.");
            } catch (BadErrorNoStringInit e) {
                System.out.println(e);
            }

            // Check if it works a second time.

            try {
                BadInitNoStringInit.dummy = 1;
                throw new IllegalStateException("Should not reach here.");
            } catch (NoClassDefFoundError e) {
                System.out.println(e);
                System.out.println(e.getCause());
            }
        } catch (Exception error) {
            error.printStackTrace(System.out);
        }
    }

    private static void exceptionsForSuperClassInitFailure() {
        try {
            // Resolve DerivedFromBadSuperClass.
            BadSuperClass.dummy = 1;
            throw new IllegalStateException("Should not reach here.");
        } catch (BadError e) {
            System.out.println(e);
        } catch (Throwable t) {
            t.printStackTrace(System.out);
        }
        try {
            // Before splitting ClassStatus::kError into
            // ClassStatus::kErrorUnresolved and ClassStatus::kErrorResolved,
            // this would trigger a
            //     CHECK(super_class->IsResolved())
            // failure in
            //     ClassLinker::LoadSuperAndInterfaces().
            // After the change we're getting either VerifyError
            // (for Optimizing) or NoClassDefFoundError wrapping
            // BadError (for interpreter or JIT).
            new DerivedFromBadSuperClass();
            throw new IllegalStateException("Should not reach here.");
        } catch (NoClassDefFoundError ncdfe) {
            if (!(ncdfe.getCause() instanceof BadError)) {
                ncdfe.getCause().printStackTrace(System.out);
            }
        } catch (VerifyError e) {
        } catch (Throwable t) {
            t.printStackTrace(System.out);
        }
    }

    private static void exceptionsInMultiDex() {
        try {
            MultiDexBadInit.dummy = 1;
            throw new IllegalStateException("Should not reach here.");
        } catch (Error e) {
            System.out.println(e);
        } catch (Throwable t) {
            t.printStackTrace(System.out);
        }
        // Before splitting ClassStatus::kError into
        // ClassStatus::kErrorUnresolved and ClassStatus::kErrorResolved,
        // the exception from wrapper 1 would have been
        // wrapped in NoClassDefFoundError but the exception
        // from wrapper 2 would have been unwrapped.
        try {
            MultiDexBadInitWrapper1.setDummy(1);
            throw new IllegalStateException("Should not reach here.");
        } catch (NoClassDefFoundError ncdfe) {
            System.out.println(ncdfe);
            System.out.println("  cause: " + ncdfe.getCause());
        } catch (Throwable t) {
            t.printStackTrace(System.out);
        }
        try {
            MultiDexBadInitWrapper2.setDummy(1);
            throw new IllegalStateException("Should not reach here.");
        } catch (NoClassDefFoundError ncdfe) {
            System.out.println(ncdfe);
            System.out.println("  cause: " + ncdfe.getCause());
        } catch (Throwable t) {
            t.printStackTrace(System.out);
        }
    }
}