diff options
Diffstat (limited to 'weaver')
-rw-r--r-- | weaver/src/main/java/org/aspectj/weaver/bcel/BcelAdvice.java | 13 | ||||
-rw-r--r-- | weaver/src/main/java/org/aspectj/weaver/bcel/BcelShadow.java | 62 |
2 files changed, 70 insertions, 5 deletions
diff --git a/weaver/src/main/java/org/aspectj/weaver/bcel/BcelAdvice.java b/weaver/src/main/java/org/aspectj/weaver/bcel/BcelAdvice.java index d22b17d12..924e6e10c 100644 --- a/weaver/src/main/java/org/aspectj/weaver/bcel/BcelAdvice.java +++ b/weaver/src/main/java/org/aspectj/weaver/bcel/BcelAdvice.java @@ -76,7 +76,17 @@ class BcelAdvice extends Advice { super(attribute, pointcut, simplify(attribute.getKind(), adviceSignature)); this.concreteAspect = concreteAspect; } - + + public boolean bindsProceedingJoinPoint() { + UnresolvedType[] parameterTypes = signature.getParameterTypes(); + for (int i=0;i<parameterTypes.length;i++) { + if (parameterTypes[i].equals(UnresolvedType.PROCEEDING_JOINPOINT)) { + return true; + } + } + return false; + } + /** * A heavyweight BcelMethod object is only required for around advice that will be inlined. For other kinds of advice it is * possible to save some space. @@ -603,6 +613,7 @@ class BcelAdvice extends Advice { } else { previousIsClosure = true; il.append(closureInstantiation.copy()); + shadow.closureVarInitialized = true; } } } else if ("Lorg/aspectj/lang/JoinPoint$StaticPart;".equals(getSignature().getParameterTypes()[i] diff --git a/weaver/src/main/java/org/aspectj/weaver/bcel/BcelShadow.java b/weaver/src/main/java/org/aspectj/weaver/bcel/BcelShadow.java index c93a0f26e..6d0cb4f60 100644 --- a/weaver/src/main/java/org/aspectj/weaver/bcel/BcelShadow.java +++ b/weaver/src/main/java/org/aspectj/weaver/bcel/BcelShadow.java @@ -70,6 +70,7 @@ import org.aspectj.weaver.patterns.NotPointcut; import org.aspectj.weaver.patterns.OrPointcut; import org.aspectj.weaver.patterns.ThisOrTargetPointcut; + /* * Some fun implementation stuff: * @@ -881,6 +882,12 @@ public class BcelShadow extends Shadow { private Map<ResolvedType, AnnotationAccessVar> withinAnnotationVars = null; private Map<ResolvedType, AnnotationAccessVar> withincodeAnnotationVars = null; private boolean allArgVarsInitialized = false; + + // If in annotation style and the relevant advice is using PJP then this will + // be set to true when the closure variable is initialized - if it gets set + // (which means link() has been called) then we will need to call unlink() + // after the code has been run. + boolean closureVarInitialized = false; @Override public Var getThisVar() { @@ -2836,6 +2843,8 @@ public class BcelShadow extends Shadow { } } + BcelVar aroundClosureInstance = null; + public void weaveAroundClosure(BcelAdvice munger, boolean hasDynamicTest) { InstructionFactory fact = getFactory(); @@ -2934,21 +2943,66 @@ public class BcelShadow extends Shadow { bitflags |= 0x000001; } + closureVarInitialized = false; + // ATAJ for @AJ aspect we need to link the closure with the joinpoint instance if (munger.getConcreteAspect() != null && munger.getConcreteAspect().isAnnotationStyleAspect() && munger.getDeclaringAspect() != null && munger.getDeclaringAspect().resolve(world).isAnnotationStyleAspect()) { + + aroundClosureInstance = genTempVar(AjcMemberMaker.AROUND_CLOSURE_TYPE); + closureInstantiation.append(fact.createDup(1)); + aroundClosureInstance.appendStore(closureInstantiation, fact); + // stick the bitflags on the stack and call the variant of linkClosureAndJoinPoint that takes an int closureInstantiation.append(fact.createConstant(Integer.valueOf(bitflags))); - closureInstantiation.append(Utility.createInvoke(getFactory(), getWorld(), - new MemberImpl(Member.METHOD, UnresolvedType.forName("org.aspectj.runtime.internal.AroundClosure"), - Modifier.PUBLIC, "linkClosureAndJoinPoint", String.format("%s%s", "(I)", "Lorg/aspectj/lang/ProceedingJoinPoint;")))); + if (needAroundClosureStacking) { + closureInstantiation.append(Utility.createInvoke(getFactory(), getWorld(), + new MemberImpl(Member.METHOD, UnresolvedType.forName("org.aspectj.runtime.internal.AroundClosure"), + Modifier.PUBLIC, "linkStackClosureAndJoinPoint", String.format("%s%s", "(I)", "Lorg/aspectj/lang/ProceedingJoinPoint;")))); + + } else { + closureInstantiation.append(Utility.createInvoke(getFactory(), getWorld(), + new MemberImpl(Member.METHOD, UnresolvedType.forName("org.aspectj.runtime.internal.AroundClosure"), + Modifier.PUBLIC, "linkClosureAndJoinPoint", String.format("%s%s", "(I)", "Lorg/aspectj/lang/ProceedingJoinPoint;")))); + } + } InstructionList advice = new InstructionList(); advice.append(munger.getAdviceArgSetup(this, null, closureInstantiation)); // invoke the advice - advice.append(munger.getNonTestAdviceInstructions(this)); + InstructionHandle tryUnlinkPosition = advice.append(munger.getNonTestAdviceInstructions(this)); + + if (needAroundClosureStacking) { + // Call AroundClosure.unlink() in a 'finally' block + if (munger.getConcreteAspect() != null && munger.getConcreteAspect().isAnnotationStyleAspect() + && munger.getDeclaringAspect() != null + && munger.getDeclaringAspect().resolve(world).isAnnotationStyleAspect() + && closureVarInitialized) { + + // Call unlink when 'normal' flow occurring + aroundClosureInstance.appendLoad(advice, fact); + InstructionHandle unlinkInsn = advice.append(Utility.createInvoke(getFactory(), getWorld(), new MemberImpl(Member.METHOD, UnresolvedType + .forName("org.aspectj.runtime.internal.AroundClosure"), Modifier.PUBLIC, "unlink", + "()V"))); + + InstructionHandle jumpOverHandler = advice.append(InstructionConstants.NOP); + + // Call unlink in finally block + InstructionHandle handlerStart = advice.append(InstructionConstants.POP); + aroundClosureInstance.appendLoad(advice, fact); + advice.append(Utility.createInvoke(getFactory(), getWorld(), new MemberImpl(Member.METHOD, UnresolvedType + .forName("org.aspectj.runtime.internal.AroundClosure"), Modifier.PUBLIC, "unlink", + "()V"))); + advice.append(InstructionConstants.ACONST_NULL); + advice.append(InstructionConstants.ATHROW); + InstructionHandle jumpTarget = advice.append(InstructionConstants.NOP); + jumpOverHandler.setInstruction(InstructionFactory.createBranchInstruction(Constants.GOTO, jumpTarget)); + enclosingMethod.addExceptionHandler(tryUnlinkPosition, unlinkInsn, handlerStart, null/* ==finally */, false); + } + } + advice.append(returnConversionCode); if (getKind() == Shadow.MethodExecution && linenumber > 0) { advice.getStart().addTargeter(new LineNumberTag(linenumber)); |