aboutsummaryrefslogtreecommitdiffstats
path: root/runtime
diff options
context:
space:
mode:
authorAndy Clement <aclement@pivotal.io>2019-02-19 12:25:01 -0800
committerAndy Clement <aclement@pivotal.io>2019-02-19 12:25:01 -0800
commit8819fad9f47fe572c45e601d7f321fe4ddf50b9e (patch)
tree6f3b7a4cfdd6e2d33796d50bd5dc96ff92c92eda /runtime
parentb858c788515524c2224165fc615638b75266954f (diff)
downloadaspectj-8819fad9f47fe572c45e601d7f321fe4ddf50b9e.tar.gz
aspectj-8819fad9f47fe572c45e601d7f321fe4ddf50b9e.zip
333274: more tests and fixes: nested @Around advice with proceed
Diffstat (limited to 'runtime')
-rw-r--r--runtime/src/main/java/org/aspectj/lang/ProceedingJoinPoint.java13
-rw-r--r--runtime/src/main/java/org/aspectj/runtime/internal/AroundClosure.java21
-rw-r--r--runtime/src/main/java/org/aspectj/runtime/reflect/JoinPointImpl.java68
3 files changed, 82 insertions, 20 deletions
diff --git a/runtime/src/main/java/org/aspectj/lang/ProceedingJoinPoint.java b/runtime/src/main/java/org/aspectj/lang/ProceedingJoinPoint.java
index d75aa4df4..ac85681ef 100644
--- a/runtime/src/main/java/org/aspectj/lang/ProceedingJoinPoint.java
+++ b/runtime/src/main/java/org/aspectj/lang/ProceedingJoinPoint.java
@@ -30,6 +30,19 @@ public interface ProceedingJoinPoint extends JoinPoint {
void set$AroundClosure(AroundClosure arc);
/**
+ * The joinpoint needs to know about its closure so that proceed can delegate to closure.run().
+ * This internal method should not be called directly, and won't be visible to the end-user when
+ * packed in a jar (synthetic method). This should maintain a stack of closures as multiple around
+ * advice with proceed are targeting a joinpoint and the stack will need to be unwound when
+ * exiting nested advice. Passing a non null arc indicates a push, passing null indicates a pop.
+ *
+ * @param arc the around closure to associate with this joinpoint
+ */
+ default void stack$AroundClosure(AroundClosure arc) {
+ throw new UnsupportedOperationException();
+ }
+
+ /**
* Proceed with the next advice or target method invocation
*
* @return the result of proceeding
diff --git a/runtime/src/main/java/org/aspectj/runtime/internal/AroundClosure.java b/runtime/src/main/java/org/aspectj/runtime/internal/AroundClosure.java
index 41c165149..c7a9acc1e 100644
--- a/runtime/src/main/java/org/aspectj/runtime/internal/AroundClosure.java
+++ b/runtime/src/main/java/org/aspectj/runtime/internal/AroundClosure.java
@@ -72,6 +72,21 @@ public abstract class AroundClosure {
* @param flags indicating whether this/target found at joinpoint and bound
* @return the associated ProceedingJoinPoint
*/
+ public ProceedingJoinPoint linkStackClosureAndJoinPoint(int flags) {
+ //TODO is this cast safe ?
+ ProceedingJoinPoint jp = (ProceedingJoinPoint)state[state.length-1];
+ jp.stack$AroundClosure(this);
+ this.bitflags = flags;
+ return jp;
+ }
+
+ /**
+ * This method is called to implicitly associate the closure with the joinpoint
+ * as required for @AJ aspect proceed()
+ *
+ * @param flags indicating whether this/target found at joinpoint and bound
+ * @return the associated ProceedingJoinPoint
+ */
public ProceedingJoinPoint linkClosureAndJoinPoint(int flags) {
//TODO is this cast safe ?
ProceedingJoinPoint jp = (ProceedingJoinPoint)state[state.length-1];
@@ -79,4 +94,10 @@ public abstract class AroundClosure {
this.bitflags = flags;
return jp;
}
+
+ public void unlink() {
+ ProceedingJoinPoint jp = (ProceedingJoinPoint)state[state.length-1];
+ jp.stack$AroundClosure(null);
+ }
+
}
diff --git a/runtime/src/main/java/org/aspectj/runtime/reflect/JoinPointImpl.java b/runtime/src/main/java/org/aspectj/runtime/reflect/JoinPointImpl.java
index 69bff1e3c..0263ee841 100644
--- a/runtime/src/main/java/org/aspectj/runtime/reflect/JoinPointImpl.java
+++ b/runtime/src/main/java/org/aspectj/runtime/reflect/JoinPointImpl.java
@@ -13,6 +13,8 @@
package org.aspectj.runtime.reflect;
+import java.util.Stack;
+
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.Signature;
@@ -134,47 +136,73 @@ class JoinPointImpl implements ProceedingJoinPoint {
return staticPart.toLongString();
}
- // To proceed we need a closure to proceed on
- private AroundClosure arc;
+ // To proceed we need a closure to proceed on. Generated code
+ // will either be using arc or arcs but not both. arcs being non-null
+ // indicates it is in use (even if an empty stack)
+ private AroundClosure arc = null;
+ private Stack<AroundClosure> arcs = null;
public void set$AroundClosure(AroundClosure arc) {
this.arc = arc;
}
+ public void stack$AroundClosure(AroundClosure arc) {
+ // If input parameter arc is null this is the 'unlink' call from AroundClosure
+ if (arcs == null) {
+ arcs = new Stack<AroundClosure>();
+ }
+ if (arc==null) {
+ this.arcs.pop();
+ } else {
+ this.arcs.push(arc);
+ }
+ }
+
public Object proceed() throws Throwable {
// when called from a before advice, but be a no-op
- if (arc == null)
- return null;
- else
- return arc.run(arc.getState());
+ if (arcs == null) {
+ if (arc == null) {
+ return null;
+ } else {
+ return arc.run(arc.getState());
+ }
+ } else {
+ return arcs.peek().run(arcs.peek().getState());
+ }
}
public Object proceed(Object[] adviceBindings) throws Throwable {
// when called from a before advice, but be a no-op
- if (arc == null)
+ AroundClosure ac = null;
+ if (arcs == null) {
+ ac = arc;
+ } else {
+ ac = arcs.peek();
+ }
+
+ if (ac == null) {
return null;
- else {
-
+ } else {
// Based on the bit flags in the AroundClosure we can determine what to
// expect in the adviceBindings array. We may or may not be expecting
// the first value to be a new this or a new target... (see pr126167)
- int flags = arc.getFlags();
+ int flags = ac.getFlags();
boolean unset = (flags & 0x100000) != 0;
boolean thisTargetTheSame = (flags & 0x010000) != 0;
boolean hasThis = (flags & 0x001000) != 0;
boolean bindsThis = (flags & 0x000100) != 0;
boolean hasTarget = (flags & 0x000010) != 0;
boolean bindsTarget = (flags & 0x000001) != 0;
-
+
// state is always consistent with caller?,callee?,formals...,jp
- Object[] state = arc.getState();
-
+ Object[] state = ac.getState();
+
// these next two numbers can differ because some join points have a this and
// target that are the same (eg. call) - and yet you can bind this and target
// separately.
-
+
// In the state array, [0] may be this, [1] may be target
-
+
int firstArgumentIndexIntoAdviceBindings = 0;
int firstArgumentIndexIntoState = 0;
firstArgumentIndexIntoState += (hasThis ? 1 : 0);
@@ -202,8 +230,8 @@ class JoinPointImpl implements ProceedingJoinPoint {
// This previous variant doesn't seem to cope with only binding target at a joinpoint
// which has both this and target. It forces you to supply this even if you didn't bind
// it.
-// firstArgumentIndexIntoAdviceBindings = (hasThis ? 1 : 0) + 1;
-// state[hasThis ? 1 : 0] = adviceBindings[hasThis ? 1 : 0];
+ // firstArgumentIndexIntoAdviceBindings = (hasThis ? 1 : 0) + 1;
+ // state[hasThis ? 1 : 0] = adviceBindings[hasThis ? 1 : 0];
int targetPositionInAdviceBindings = (hasThis && bindsThis) ? 1 : 0;
firstArgumentIndexIntoAdviceBindings = ((hasThis&&bindsThis)?1:0)+((hasTarget&&bindsTarget&&!thisTargetTheSame)?1:0);
@@ -213,12 +241,12 @@ class JoinPointImpl implements ProceedingJoinPoint {
// leave state[0]/state[1] alone, they are OK
}
}
-
+
// copy the rest across
for (int i = firstArgumentIndexIntoAdviceBindings; i < adviceBindings.length; i++) {
state[firstArgumentIndexIntoState + (i - firstArgumentIndexIntoAdviceBindings)] = adviceBindings[i];
}
-
+
// old code that did this, didnt allow this/target overriding
// for (int i = state.length-2; i >= 0; i--) {
// int formalIndex = (adviceBindings.length - 1) - (state.length-2) + i;
@@ -226,7 +254,7 @@ class JoinPointImpl implements ProceedingJoinPoint {
// state[i] = adviceBindings[formalIndex];
// }
// }
- return arc.run(state);
+ return ac.run(state);
}
}