/*
* Copyright (C) 2016 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.
*/
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
//
// Two classes A and B with method foo().
//
class A {
A() { System.out.println("new A"); }
public void foo() { System.out.println("I am A's foo"); }
// We previously used to invoke this method with a Y instance, due
// to invoke-super underspecified behavior.
public void bar() { System.out.println("I am A's bar"); }
}
class B {
B() { System.out.println("new B"); }
public void foo() { System.out.println("I am B's foo"); }
}
//
// Two subclasses X and Y that call foo() on super.
//
class X extends A {
public void foo() { super.foo(); }
}
class Y extends B {
public void foo() { super.foo(); }
}
//
// Driver class.
//
public class Main {
public static void main(String[] args) throws Exception {
// The normal stuff, X's super goes to A, Y's super goes to B.
new X().foo();
new Y().foo();
// And now it gets interesting.
// In bytecode, we define a class Z that is a subclass of A, and we call
// invoke-super on an instance of Y.
Class<?> z = Class.forName("Z");
Method m = z.getMethod("foo");
try {
m.invoke(z.newInstance());
throw new Error("Expected InvocationTargetException");
} catch (InvocationTargetException e) {
if (!(e.getCause() instanceof NoSuchMethodError)) {
throw new Error("Expected NoSuchMethodError");
}
}
System.out.println("passed");
}
}