From 8819fad9f47fe572c45e601d7f321fe4ddf50b9e Mon Sep 17 00:00:00 2001 From: Andy Clement Date: Tue, 19 Feb 2019 12:25:01 -0800 Subject: 333274: more tests and fixes: nested @Around advice with proceed --- .../java/org/aspectj/lang/ProceedingJoinPoint.java | 13 +++++ .../aspectj/runtime/internal/AroundClosure.java | 21 +++++++ .../org/aspectj/runtime/reflect/JoinPointImpl.java | 68 +++++++++++++++------- 3 files changed, 82 insertions(+), 20 deletions(-) (limited to 'runtime') 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 @@ -29,6 +29,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 * 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 @@ -65,6 +65,21 @@ public abstract class AroundClosure { 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 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() @@ -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 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(); + } + 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); } } -- cgit v1.2.3