aboutsummaryrefslogtreecommitdiffstats
path: root/weaver
diff options
context:
space:
mode:
Diffstat (limited to 'weaver')
-rw-r--r--weaver/src/org/aspectj/weaver/bcel/BcelShadow.java268
1 files changed, 193 insertions, 75 deletions
diff --git a/weaver/src/org/aspectj/weaver/bcel/BcelShadow.java b/weaver/src/org/aspectj/weaver/bcel/BcelShadow.java
index 17f766f28..0442d8c68 100644
--- a/weaver/src/org/aspectj/weaver/bcel/BcelShadow.java
+++ b/weaver/src/org/aspectj/weaver/bcel/BcelShadow.java
@@ -1939,7 +1939,47 @@ public class BcelShadow extends Shadow {
weaveAroundClosure(munger, hasDynamicTest);
return;
}
-
+
+ // specific test for @AJ proceedInInners
+ if (munger.getConcreteAspect().isAnnotationStyleAspect()) {
+ //FIXME
+ // if we can't find one proceed()
+ // we suspect that the call is happening in an inner class
+ // so we don't inline it.
+ // Note: for code style, this is done at Aspect compilation time.
+ // Limitation: if there is two calls and one is done in an inner class, inlining will happen
+ // since we will see the non proceed in inner one as if it was the sole one.
+ // The one made in the inner class will never be madeXXX. Should be documented.
+ boolean canSeeProceedPassedToOther = false;
+ InstructionHandle curr = adviceMethod.getBody().getStart();
+ InstructionHandle end = adviceMethod.getBody().getEnd();
+ ConstantPoolGen cpg = adviceMethod.getEnclosingClass().getConstantPoolGen();
+ while (curr != end) {
+ InstructionHandle next = curr.getNext();
+ Instruction inst = curr.getInstruction();
+ /*if ((inst instanceof INVOKEINTERFACE)
+ && "proceed".equals(((INVOKEINTERFACE) inst).getMethodName(cpg))) {
+ canSeeProceed = true;
+ //continue since we may have a proceed in inner or pjp as arg
+ } else*/ if ((inst instanceof InvokeInstruction)
+ && ((InvokeInstruction)inst).getSignature(cpg).indexOf("Lorg/aspectj/lang/ProceedingJoinPoint;") > 0) {
+ // we may want to refine to exclude stuff returning jp ?
+ // does code style skip inline if i write dump(thisJoinPoint) ?
+ canSeeProceedPassedToOther = true;// we see one pjp passed around - dangerous
+ break;
+ }
+ curr = next;
+ }
+ if (canSeeProceedPassedToOther) {
+ // remember this decision to avoid re-analysis
+ adviceMethod.setCanInline(false);
+ weaveAroundClosure(munger, hasDynamicTest);
+ return;
+ }
+ }
+
+
+
// We can't inline around methods if they have around advice on them, this
// is because the weaving will extract the body and hence the proceed call.
//??? should consider optimizations to recognize simple cases that don't require body extraction
@@ -1954,8 +1994,9 @@ public class BcelShadow extends Shadow {
NameMangler.aroundCallbackMethodName(
getSignature(),
getEnclosingClass()),
- Modifier.PRIVATE,
- munger);
+ Modifier.PRIVATE,
+ munger
+ );
// now extract the advice into its own method
@@ -2164,18 +2205,22 @@ public class BcelShadow extends Shadow {
Instruction inst = curr.getInstruction();
if ((inst instanceof INVOKEINTERFACE)
&& "proceed".equals(((INVOKEINTERFACE) inst).getMethodName(cpg))) {
- //TODO proceed(...varargs) will need some stuff there
- InstructionList insteadProceedIl =
- getRedoneProceedCallForAnnotationStyle(
+ final boolean isProceedWithArgs;
+ if (((INVOKEINTERFACE) inst).getArgumentTypes(cpg).length == 1) {
+ // proceed with args as a boxed Object[]
+ isProceedWithArgs = true;
+ } else {
+ isProceedWithArgs = false;
+ }
+ InstructionList insteadProceedIl = getRedoneProceedCallForAnnotationStyle(
fact,
extractedMethod,
munger,
localAdviceMethod,
- proceedVarList);
- //TODO optimize, a POP is added somewhere, what for non void ?
- localAdviceMethod.getBody().append(
- curr, insteadProceedIl);
-
+ proceedVarList,
+ isProceedWithArgs
+ );
+ localAdviceMethod.getBody().append(curr, insteadProceedIl);
Utility.deleteInstruction(curr, localAdviceMethod);
}
curr = next;
@@ -2242,6 +2287,7 @@ public class BcelShadow extends Shadow {
* @param munger
* @param localAdviceMethod
* @param argVarList
+ * @param isProceedWithArgs
* @return
*/
private InstructionList getRedoneProceedCallForAnnotationStyle(
@@ -2249,57 +2295,129 @@ public class BcelShadow extends Shadow {
LazyMethodGen callbackMethod,
BcelAdvice munger,
LazyMethodGen localAdviceMethod,
- List argVarList)
+ List argVarList,
+ boolean isProceedWithArgs)
{
- //if (true) return new InstructionList();
+ // Notes:
+ // proceedingjp is on stack (since user was calling pjp.proceed(...)
+ // the boxed args to proceed() are on stack as well (Object[]) unless
+ // the call is to pjp.proceed(<noarg>)
- //TODO I think we just don't care about the proceed map
- // since that one must be the exact same of the advice sig.
- // perhaps with custom jp ?
+ // new Object[]{new Integer(argAdvice1-1)};// arg of proceed
+ // call to proceed(..) is NOT made
+ // instead we do
+ // itar callback args i
+ // get from array i, convert it to the callback arg i
+ // if ask for JP, push the one we got on the stack
+ // invoke callback, create conversion back to Object/Integer
- // we have on stack all the arguments for the ADVICE call.
- // we have in frame somewhere all the arguments for the non-advice call.
- BcelVar[] adviceVars = munger.getExposedStateAsBcelVars();
- IntMap proceedMap = makeProceedArgumentMap(adviceVars);
+ // rest of method -- (hence all those conversions)
+ // intValue() from original code
+ // int res = .. from original code
- //System.out.println(proceedMap + " for " + this);
- //System.out.println(argVarList);
-
- ResolvedTypeX[] proceedParamTypes =
- world.resolve(munger.getSignature().getParameterTypes());
- //System.out.println(proceedParamTypes);
+ //Note: we just don't care about the proceed map etc
InstructionList ret = new InstructionList();
- ret.append(new POP());//the joinpoint instance is there
- for (int i=0, len=callbackMethod.getArgumentTypes().length; i < len; i++) {
- Type stateType = callbackMethod.getArgumentTypes()[i];
- ResolvedTypeX stateTypeX = BcelWorld.fromBcel(stateType).resolve(world);
- if (proceedMap.hasKey(i)) {
- ret.append(new ALOAD(i));
- //throw new RuntimeException("unimplemented");
- //proceedVars[proceedMap.get(i)].appendLoadAndConvert(ret, fact, stateTypeX);
- } else {
- //FIXME Alex: odd code there
- //((BcelVar) argVarList.get(i)).appendLoad(ret, fact);
- //ret.append(new ALOAD(i));
+ // store the Object[] array on stack if proceed with args
+ if (isProceedWithArgs) {
+ Type objectArrayType = Type.getType("[Ljava/lang/Object;");
+ int localProceedArgArray = localAdviceMethod.allocateLocal(objectArrayType);
+ ret.append(InstructionFactory.createStore(objectArrayType, localProceedArgArray));
+
+ Type proceedingJpType = Type.getType("Lorg/aspectj/lang/ProceedingJoinPoint;");
+ int localJp = localAdviceMethod.allocateLocal(proceedingJpType);
+ ret.append(InstructionFactory.createStore(proceedingJpType, localJp));
+
+ // push on stack each element of the object array
+ // that is assumed to be consistent with the callback argument (ie munger args)
+ // TODO do we want to try catch ClassCast and AOOBE exception ?
+
+ // special logic when withincode is static or not
+ int startIndex = 0;
+ if (thisVar != null) {
+ startIndex = 1;
+ //TODO this logic is actually depending on target as well - test me
+ ret.append(new ALOAD(0));//thisVar
+ }
+ for (int i = startIndex, len=callbackMethod.getArgumentTypes().length; i < len; i++) {
+ Type stateType = callbackMethod.getArgumentTypes()[i];
+ ResolvedTypeX stateTypeX = BcelWorld.fromBcel(stateType).resolve(world);
if ("Lorg/aspectj/lang/JoinPoint;".equals(stateType.getSignature())) {
- ret.append(new ALOAD(i));
+ ret.append(new ALOAD(localJp));// from localAdvice signature
} else {
- ret.append(new ALOAD(i));
+ ret.append(InstructionFactory.createLoad(objectArrayType, localProceedArgArray));
+ ret.append(Utility.createConstant(fact, i-startIndex));
+ ret.append(InstructionFactory.createArrayLoad(Type.OBJECT));
+ ret.append(Utility.createConversion(
+ fact,
+ Type.OBJECT,
+ stateType
+ ));
+ }
+ }
+ } else {
+ Type proceedingJpType = Type.getType("Lorg/aspectj/lang/ProceedingJoinPoint;");
+ int localJp = localAdviceMethod.allocateLocal(proceedingJpType);
+ ret.append(InstructionFactory.createStore(proceedingJpType, localJp));
+
+ for (int i = 0, len=callbackMethod.getArgumentTypes().length; i < len; i++) {
+ Type stateType = callbackMethod.getArgumentTypes()[i];
+ ResolvedTypeX stateTypeX = BcelWorld.fromBcel(stateType).resolve(world);
+ if ("Lorg/aspectj/lang/JoinPoint;".equals(stateType.getSignature())) {
+ ret.append(new ALOAD(localJp));// from localAdvice signature
+ } else {
+ ret.append(InstructionFactory.createLoad(stateType, i));
}
}
}
+ // do the callback invoke
ret.append(Utility.createInvoke(fact, callbackMethod));
- ret.append(Utility.createConversion(fact, callbackMethod.getReturnType(),
- BcelWorld.makeBcelType(munger.getSignature().getReturnType())));
-
- ret.append(new ACONST_NULL());//will be POPed //FIXME Alex: if so clean up
- if (true) return ret;
+ // box it again. Handles cases where around advice does return something else than Object
+ if (!TypeX.OBJECT.equals(munger.getSignature().getReturnType())) {
+ ret.append(Utility.createConversion(
+ fact,
+ callbackMethod.getReturnType(),
+ Type.OBJECT
+ ));
+ }
+ ret.append(Utility.createConversion(
+ fact,
+ callbackMethod.getReturnType(),
+ BcelWorld.makeBcelType(munger.getSignature().getReturnType())
+ ));
+ return ret;
+//
+//
+//
+// if (proceedMap.hasKey(i)) {
+// ret.append(new ALOAD(i));
+// //throw new RuntimeException("unimplemented");
+// //proceedVars[proceedMap.get(i)].appendLoadAndConvert(ret, fact, stateTypeX);
+// } else {
+// //((BcelVar) argVarList.get(i)).appendLoad(ret, fact);
+// //ret.append(new ALOAD(i));
+// if ("Lorg/aspectj/lang/JoinPoint;".equals(stateType.getSignature())) {
+// ret.append(new ALOAD(i));
+// } else {
+// ret.append(new ALOAD(i));
+// }
+// }
+// }
+//
+// ret.append(Utility.createInvoke(fact, callbackMethod));
+// ret.append(Utility.createConversion(fact, callbackMethod.getReturnType(),
+// BcelWorld.makeBcelType(munger.getSignature().getReturnType())));
+//
+// //ret.append(new ACONST_NULL());//will be POPed
+// if (true) return ret;
+//
+//
+//
// // we have on stack all the arguments for the ADVICE call.
// // we have in frame somewhere all the arguments for the non-advice call.
//
@@ -2311,36 +2429,36 @@ public class BcelShadow extends Shadow {
//
// ResolvedTypeX[] proceedParamTypes =
// world.resolve(munger.getSignature().getParameterTypes());
- // remove this*JoinPoint* as arguments to proceed
- if (munger.getBaseParameterCount()+1 < proceedParamTypes.length) {
- int len = munger.getBaseParameterCount()+1;
- ResolvedTypeX[] newTypes = new ResolvedTypeX[len];
- System.arraycopy(proceedParamTypes, 0, newTypes, 0, len);
- proceedParamTypes = newTypes;
- }
-
- //System.out.println("stateTypes: " + Arrays.asList(stateTypes));
- BcelVar[] proceedVars =
- Utility.pushAndReturnArrayOfVars(proceedParamTypes, ret, fact, localAdviceMethod);
-
- Type[] stateTypes = callbackMethod.getArgumentTypes();
-// System.out.println("stateTypes: " + Arrays.asList(stateTypes));
-
- for (int i=0, len=stateTypes.length; i < len; i++) {
- Type stateType = stateTypes[i];
- ResolvedTypeX stateTypeX = BcelWorld.fromBcel(stateType).resolve(world);
- if (proceedMap.hasKey(i)) {
- //throw new RuntimeException("unimplemented");
- proceedVars[proceedMap.get(i)].appendLoadAndConvert(ret, fact, stateTypeX);
- } else {
- ((BcelVar) argVarList.get(i)).appendLoad(ret, fact);
- }
- }
-
- ret.append(Utility.createInvoke(fact, callbackMethod));
- ret.append(Utility.createConversion(fact, callbackMethod.getReturnType(),
- BcelWorld.makeBcelType(munger.getSignature().getReturnType())));
- return ret;
+// // remove this*JoinPoint* as arguments to proceed
+// if (munger.getBaseParameterCount()+1 < proceedParamTypes.length) {
+// int len = munger.getBaseParameterCount()+1;
+// ResolvedTypeX[] newTypes = new ResolvedTypeX[len];
+// System.arraycopy(proceedParamTypes, 0, newTypes, 0, len);
+// proceedParamTypes = newTypes;
+// }
+//
+// //System.out.println("stateTypes: " + Arrays.asList(stateTypes));
+// BcelVar[] proceedVars =
+// Utility.pushAndReturnArrayOfVars(proceedParamTypes, ret, fact, localAdviceMethod);
+//
+// Type[] stateTypes = callbackMethod.getArgumentTypes();
+//// System.out.println("stateTypes: " + Arrays.asList(stateTypes));
+//
+// for (int i=0, len=stateTypes.length; i < len; i++) {
+// Type stateType = stateTypes[i];
+// ResolvedTypeX stateTypeX = BcelWorld.fromBcel(stateType).resolve(world);
+// if (proceedMap.hasKey(i)) {
+// //throw new RuntimeException("unimplemented");
+// proceedVars[proceedMap.get(i)].appendLoadAndConvert(ret, fact, stateTypeX);
+// } else {
+// ((BcelVar) argVarList.get(i)).appendLoad(ret, fact);
+// }
+// }
+//
+// ret.append(Utility.createInvoke(fact, callbackMethod));
+// ret.append(Utility.createConversion(fact, callbackMethod.getReturnType(),
+// BcelWorld.makeBcelType(munger.getSignature().getReturnType())));
+// return ret;
}
public void weaveAroundClosure(