aboutsummaryrefslogtreecommitdiffstats
path: root/tests/java5/generics
diff options
context:
space:
mode:
authoracolyer <acolyer>2005-08-05 10:19:23 +0000
committeracolyer <acolyer>2005-08-05 10:19:23 +0000
commit8f4da67cd5008837d59c6a182558b8403cbff58a (patch)
tree39c01bf55124a4ff26345b4a085759a502a75e36 /tests/java5/generics
parentc22c80074c9ef8876d758692815b9c100e51f7ed (diff)
downloadaspectj-8f4da67cd5008837d59c6a182558b8403cbff58a.tar.gz
aspectj-8f4da67cd5008837d59c6a182558b8403cbff58a.zip
tests and implementation for calls to bridge methods. This completes the implementation of the call pointcut for generic and parameterized types.
Diffstat (limited to 'tests/java5/generics')
-rw-r--r--tests/java5/generics/pointcuts/CallWithBridgeMethods.aj77
1 files changed, 77 insertions, 0 deletions
diff --git a/tests/java5/generics/pointcuts/CallWithBridgeMethods.aj b/tests/java5/generics/pointcuts/CallWithBridgeMethods.aj
new file mode 100644
index 000000000..7701a840b
--- /dev/null
+++ b/tests/java5/generics/pointcuts/CallWithBridgeMethods.aj
@@ -0,0 +1,77 @@
+/**
+ *
+ * Each method T m(T1,...Tn) throws S1,...Sm is translated to a method with the same name
+ * whose return type, argument types, and thrown types are the erasures of the corresponding
+ * types in the original method. In addition, if a method m of a class or interface C is
+ * inherited in a subclass D, a bridge method might need to be generated in D. The rules are
+ * as follows:
+ *
+ * If C.m is directly overridden by a method D.m in D, and the erasure of the return type or
+ * argument types of D.m differs from the erasure of the corresponding types in C.m, a bridge
+ * method needs to be generated.
+ *
+ */
+
+
+public aspect CallWithBridgeMethods {
+
+ void bar() {
+ SubGeneric rawType = new SubGeneric();
+ Generic rawVariableOfSuperType = rawType;
+ rawVariableOfSuperType.foo("hi"); // this call we go to the bridge method..., but
+ // appears in the bytecode as Generic.foo(Object)
+ rawType.foo("hi"); // this call will go to the bridge method, and
+ // appears in the bytecode as SubGeneric.foo(Object)
+ }
+
+ declare warning : call(* SubGeneric.foo(..)) && within(SubGeneric)
+ : "should not match call in bridge method";
+
+ declare warning : call(* SubGeneric.foo(Object))
+ : "should match call to bridge method on L23, this is a real call!";
+
+ declare warning : execution(* SubGeneric.foo(Object)) && within(SubGeneric)
+ : "but whilst you can call it, it doesn't execute!";
+}
+
+
+class Generic<T> {
+
+ public T foo(T someObject) {
+ return someObject;
+ }
+
+}
+
+class SubGeneric<N extends Number> extends Generic<N> {
+
+ public N foo(N someNumber) {
+ return someNumber;
+ }
+
+ // "bridge" method is:
+// public Object foo(Object someObject) {
+// Number n = (Number)someObject;
+// return foo(n);
+// }
+
+}
+
+class SubSubGeneric<N extends Number> extends SubGeneric<N> {
+
+ // inherits bridge method from super
+
+ // NO bridge method generated (this is in contrast to the statements in the Bracha paper).
+ // "Adding Generics to the Java Programming Language, Public Draft Specification v2.0",
+ // Bracha et. al, June 23, 2003.
+
+}
+
+
+class SubGenericWithSameErasure<T> extends Generic<T> {
+
+ // NO bridge method generated (this is in contrast to the statements in the Bracha paper).
+ // "Adding Generics to the Java Programming Language, Public Draft Specification v2.0",
+ // Bracha et. al, June 23, 2003.
+
+} \ No newline at end of file