diff options
author | wisberg <wisberg> | 2002-12-16 17:58:19 +0000 |
---|---|---|
committer | wisberg <wisberg> | 2002-12-16 17:58:19 +0000 |
commit | d842c4f1139629c1f062b74ba818d233b2c31043 (patch) | |
tree | 842d3871620bc0eb60edcd95e55804d67e0f61fa /weaver/src/org | |
parent | 3ce247199704eae6b2c92c6e38c69584e3250c52 (diff) | |
download | aspectj-d842c4f1139629c1f062b74ba818d233b2c31043.tar.gz aspectj-d842c4f1139629c1f062b74ba818d233b2c31043.zip |
initial version
Diffstat (limited to 'weaver/src/org')
131 files changed, 22227 insertions, 0 deletions
diff --git a/weaver/src/org/aspectj/weaver/Advice.java b/weaver/src/org/aspectj/weaver/Advice.java new file mode 100644 index 000000000..e20627ab0 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/Advice.java @@ -0,0 +1,243 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver; + +import java.util.*; +import java.util.List; + +import org.aspectj.bridge.MessageUtil; +import org.aspectj.weaver.patterns.*; + +public abstract class Advice extends ShadowMunger { + + protected AdviceKind kind; + protected Member signature; + protected int extraParameterFlags; + protected int lexicalPosition; + + // not necessarily declaring aspect, this is a semantics change from 1.0 + protected ResolvedTypeX concreteAspect; // null until after concretize + + protected List innerCflowEntries = Collections.EMPTY_LIST; // just for cflow*Entry kinds + protected int nFreeVars; // just for cflow*Entry kinds + + protected TypePattern exceptionType; // just for Softener kind + + public static Advice makeCflowEntry(World world, Pointcut entry, boolean isBelow, Member stackField, int nFreeVars, List innerCflowEntries) { + Advice ret = world.concreteAdvice(isBelow ? AdviceKind.CflowBelowEntry : AdviceKind.CflowEntry, + entry, stackField, 0, entry); + //0); + ret.innerCflowEntries = innerCflowEntries; + ret.nFreeVars = nFreeVars; + return ret; + } + + public static Advice makePerCflowEntry(World world, Pointcut entry, boolean isBelow, + Member stackField, ResolvedTypeX inAspect, List innerCflowEntries) + { + Advice ret = world.concreteAdvice(isBelow ? AdviceKind.PerCflowBelowEntry : AdviceKind.PerCflowEntry, + entry, stackField, 0, entry); + ret.innerCflowEntries = innerCflowEntries; + ret.concreteAspect = inAspect; + return ret; + } + + public static Advice makePerObjectEntry(World world, Pointcut entry, boolean isThis, + ResolvedTypeX inAspect) + { + Advice ret = world.concreteAdvice(isThis ? AdviceKind.PerThisEntry : AdviceKind.PerTargetEntry, + entry, null, 0, entry); + + ret.concreteAspect = inAspect; + return ret; + } + + public static Advice makeSoftener(World world, Pointcut entry, TypePattern exceptionType) { + Advice ret = world.concreteAdvice(AdviceKind.Softener, + entry, null, 0, entry); + + ret.exceptionType = exceptionType; + //System.out.println("made ret: " + ret + " with " + exceptionType); + return ret; + } + + + public Advice(AdviceKind kind, Pointcut pointcut, Member signature, + int extraParameterFlags, int start, int end, ISourceContext sourceContext) + { + super(pointcut, start, end, sourceContext); + this.kind = kind; + this.signature = signature; + this.extraParameterFlags = extraParameterFlags; + this.lexicalPosition = start; //XXX should go away + } + + + public boolean match(Shadow shadow, World world) { + if (super.match(shadow, world)) { + if (hasExtraParameter() && kind == AdviceKind.AfterReturning) { + return getExtraParameterType().isConvertableFrom(shadow.getReturnType(), world); + } else if (kind == AdviceKind.PerTargetEntry) { + return shadow.hasTarget(); + } else if (kind == AdviceKind.PerThisEntry) { + return shadow.hasThis(); + } else if (kind == AdviceKind.Around) { + if (shadow.getKind() == Shadow.PreInitialization) { + world.getMessageHandler().handleMessage( + MessageUtil.error("Around on pre-initialization not supported in 1.1", + getSourceLocation())); + world.getMessageHandler().handleMessage( + MessageUtil.error("Around on pre-initialization not supported in 1.1", + shadow.getSourceLocation())); + return false; + } else { + if (!getSignature().getReturnType().isConvertableFrom(shadow.getReturnType(), world)) { + //System.err.println(this + ", " + sourceContext + ", " + start); + + world.getMessageHandler().handleMessage( + MessageUtil.error("incompatible return type applying to " + shadow, + getSourceLocation())); + //XXX need a crosscutting error message here + return false; + } + } + } + return true; + } else { + return false; + } + } + + + // ---- + + public AdviceKind getKind() { + return kind; + } + + public Member getSignature() { + return signature; + } + + public boolean hasExtraParameter() { + return (extraParameterFlags & ExtraArgument) != 0; + } + + protected int getExtraParameterCount() { + return countOnes(extraParameterFlags & ParameterMask); + } + + public static int countOnes(int bits) { + int ret = 0; + while (bits != 0) { + if ((bits & 1) != 0) ret += 1; + bits = bits >> 1; + } + return ret; + } + + public int getBaseParameterCount() { + return signature.getParameterTypes().length - getExtraParameterCount(); + } + + public TypeX getExtraParameterType() { + if (!hasExtraParameter()) return ResolvedTypeX.MISSING; + return signature.getParameterTypes()[getBaseParameterCount()]; + } + + public TypeX getDeclaringAspect() { + return signature.getDeclaringType(); + } + + protected String extraParametersToString() { + if (extraParameterFlags == 0) { + return ""; + } else { + return "(extraFlags: " + extraParameterFlags + ")"; + } + } + + public Pointcut getPointcut() { + return pointcut; + } + + // ---- + + /** @param fromType is guaranteed to be a non-abstract aspect + * @param perClause has been concretized at a higher level + */ + public ShadowMunger concretize(ResolvedTypeX fromType, World world, PerClause clause) { + // assert !fromType.isAbstract(); + Pointcut p = pointcut.concretize(fromType, signature.getArity(), this); + if (clause != null) { + p = new AndPointcut(clause, p); + p.state = Pointcut.CONCRETE; + } + + Advice munger = world.concreteAdvice(kind, p, signature, extraParameterFlags, start, end, sourceContext); + munger.concreteAspect = fromType; + //System.err.println("concretizing here " + p + " with clause " + clause); + return munger; + } + + // ---- from object + + public String toString() { + return "(" + + getKind() + + extraParametersToString() + + ": " + + pointcut + + "->" + + signature + + ")"; + } + public boolean equals(Object other) { + if (! (other instanceof Advice)) return false; + Advice o = (Advice) other; + return o.kind == kind && o.pointcut.equals(pointcut) && o.signature.equals(signature) && + o.extraParameterFlags == extraParameterFlags; + } + private volatile int hashCode = 0; + public int hashCode() { + if (hashCode == 0) { + int result = 17; + result = 37*result + kind.hashCode(); + result = 37*result + pointcut.hashCode(); + if (signature != null) result = 37*result + signature.hashCode(); + hashCode = result; + } + return hashCode; + } + + // ---- fields + + + public static final int ExtraArgument = 1; + public static final int ThisJoinPoint = 2; + public static final int ThisJoinPointStaticPart = 4; + public static final int ThisEnclosingJoinPointStaticPart = 8; + public static final int ParameterMask = 0xf; + + public static final int CanInline = 0x40; + + public void setLexicalPosition(int lexicalPosition) { + this.lexicalPosition = lexicalPosition; + } + + public ResolvedTypeX getConcreteAspect() { + return concreteAspect; + } + +} diff --git a/weaver/src/org/aspectj/weaver/AdviceKind.java b/weaver/src/org/aspectj/weaver/AdviceKind.java new file mode 100644 index 000000000..17e94328c --- /dev/null +++ b/weaver/src/org/aspectj/weaver/AdviceKind.java @@ -0,0 +1,110 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver; + +import java.io.*; + +import org.aspectj.util.TypeSafeEnum; + +/** + * The 5 kinds of advice in AspectJ. + * + * @author Erik Hilsdale + * @author Jim Hugunin + */ +public class AdviceKind extends TypeSafeEnum { + private int precedence; + private boolean isAfter; + private boolean isCflow; + public AdviceKind(String name, int key, int precedence, boolean isAfter, boolean isCflow) { + super(name, key); + this.precedence = precedence; + this.isAfter = isAfter; + this.isCflow = isCflow; + } + + public static AdviceKind read(DataInputStream s) throws IOException { + int key = s.readByte(); + switch(key) { + case 1: return Before; + case 2: return After; + case 3: return AfterThrowing; + case 4: return AfterReturning; + case 5: return Around; + case 6: return CflowEntry; + case 7: return CflowBelowEntry; + + case 8: return InterInitializer; + + case 9: return PerCflowEntry; + case 10: return PerCflowBelowEntry; + case 11: return PerThisEntry; + case 12: return PerTargetEntry; + + case 13: return Softener; + } + throw new RuntimeException("unimplemented kind: " + key); + } + + public static final AdviceKind Before = new AdviceKind("before", 1, 0, false, false); + public static final AdviceKind After = new AdviceKind("after", 2, 0, true, false); + public static final AdviceKind AfterThrowing = new AdviceKind("afterThrowing", 3, 0, true, false); + public static final AdviceKind AfterReturning = new AdviceKind("afterReturning", 4, 0, true, false); + public static final AdviceKind Around = new AdviceKind("around", 5, 0, false, false); + + // these kinds can't be declared, but are used by the weaver + public static final AdviceKind CflowEntry = new AdviceKind("cflowEntry", 6, 1, false, true); + public static final AdviceKind CflowBelowEntry = new AdviceKind("cflowBelowEntry", 7, -1, false, true); //XXX resolve precednece with the below + public static final AdviceKind InterInitializer = new AdviceKind("interInitializer", 8, -2, false, false); + + public static final AdviceKind PerCflowEntry = new AdviceKind("perCflowEntry", 9, 1, false, true); + public static final AdviceKind PerCflowBelowEntry = new AdviceKind("perCflowBelowEntry", 10, -1, false, true); + + public static final AdviceKind PerThisEntry = new AdviceKind("perThisEntry", 11, 1, false, false); + public static final AdviceKind PerTargetEntry = new AdviceKind("perTargetEntry", 12, 1, false, false); + + public static final AdviceKind Softener = new AdviceKind("softener", 13, 1, false, false); + + + public static AdviceKind stringToKind(String s) { + if (s.equals(Before.getName())) return Before; + if (s.equals(After.getName())) return After; + if (s.equals(AfterThrowing.getName())) return AfterThrowing; + if (s.equals(AfterReturning.getName())) return AfterReturning; + if (s.equals(Around.getName())) return Around; + throw new IllegalArgumentException("unknown kind: " + "\"" + s + "\""); + } + + public boolean isAfter() { + return this.isAfter; + } + + public boolean isCflow() { + return this.isCflow; + } + + public int getPrecedence() { + return precedence; + } + + public boolean isPerEntry() { + return this == PerCflowEntry || this == PerCflowBelowEntry || + this == PerThisEntry || this == PerTargetEntry; + } + + public boolean isPerObjectEntry() { + return this == PerThisEntry || this == PerTargetEntry; + } + +} diff --git a/weaver/src/org/aspectj/weaver/AjAttribute.java b/weaver/src/org/aspectj/weaver/AjAttribute.java new file mode 100644 index 000000000..864513371 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/AjAttribute.java @@ -0,0 +1,411 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver; + +import java.io.*; + +import org.aspectj.util.FileUtil; +import org.aspectj.weaver.patterns.*; + +/** + * These attributes are written to and read from .class files (see the JVM spec). + * + * <p>Each member or type can have a number of AjAttributes. Each + * such attribute is in 1-1 correspondence with an Unknown bcel attribute. + * Creating one of these does NOTHING to the underlying thing, so if you really + * want to add an attribute to a particular thing, well, you'd better actually do that. + * + * @author Erik Hilsdale + * @author Jim Hugunin + */ +public abstract class AjAttribute { + + public static final String AttributePrefix = "org.aspectj.weaver"; + + protected abstract void write(DataOutputStream s) throws IOException; + + public abstract String getNameString(); + public char[] getNameChars() { + return getNameString().toCharArray(); + } + + /** + * Just writes the contents + */ + public byte[] getBytes() { + try { + ByteArrayOutputStream b0 = new ByteArrayOutputStream(); + DataOutputStream s0 = new DataOutputStream(b0); + write(s0); + return b0.toByteArray(); + } catch (IOException e) { + // shouldn't happen with ByteArrayOutputStreams + throw new RuntimeException("sanity check"); + } + } + + /** + * Writes the full attribute, i.e. name_index, length, and contents + */ + public byte[] getAllBytes(short nameIndex) { + try { + byte[] bytes = getBytes(); + + ByteArrayOutputStream b0 = new ByteArrayOutputStream(); + DataOutputStream s0 = new DataOutputStream(b0); + + s0.writeShort(nameIndex); + s0.writeInt(bytes.length); + s0.write(bytes); + return b0.toByteArray(); + } catch (IOException e) { + // shouldn't happen with ByteArrayOutputStreams + throw new RuntimeException("sanity check"); + } + } + + public static AjAttribute read(String name, byte[] bytes, ISourceContext context) { + try { + if (bytes == null) bytes = new byte[0]; + DataInputStream s = new DataInputStream(new ByteArrayInputStream(bytes)); + if (name.equals(Aspect.AttributeName)) { + return new Aspect(PerClause.readPerClause(s, context)); + } else if (name.equals(WeaverState.AttributeName)) { + return new WeaverState(WeaverStateKind.read(s)); + } else if (name.equals(AdviceAttribute.AttributeName)) { + return AdviceAttribute.read(s, context); + } else if (name.equals(PointcutDeclarationAttribute.AttributeName)) { + return new PointcutDeclarationAttribute(ResolvedPointcutDefinition.read(s, context)); + } else if (name.equals(TypeMunger.AttributeName)) { + return new TypeMunger(ResolvedTypeMunger.read(s, context)); + } else if (name.equals(AjSynthetic.AttributeName)) { + return new AjSynthetic(); + } else if (name.equals(DeclareAttribute.AttributeName)) { + return new DeclareAttribute(Declare.read(s, context)); + } else if (name.equals(PrivilegedAttribute.AttributeName)) { + return PrivilegedAttribute.read(s, context); + } else if (name.equals(SourceContextAttribute.AttributeName)) { + return SourceContextAttribute.read(s); + } else if (name.equals(EffectiveSignatureAttribute.AttributeName)) { + return EffectiveSignatureAttribute.read(s, context); + } else { + throw new BCException("unknown attribute" + name); + } + } catch (IOException e) { + throw new BCException("malformed " + name + " attribute " + e); + } + } + + //---- + + /** Synthetic members should have NO advice put on them or on their contents. + * This attribute is currently unused as we consider all members starting + * with NameMangler.PREFIX to automatically be synthetic. As we use this we might + * find that we want multiple + * kinds of synthetic. In particular, if we want to treat the call to a synthetic getter + * (say, of an introduced field) as a field reference itself, then a method might want + * a particular kind of AjSynthetic attribute that also includes a signature of what + * it stands for. + */ + public static class AjSynthetic extends AjAttribute { + public static final String AttributeName = "org.aspectj.weaver.AjSynthetic"; + + public String getNameString() { + return AttributeName; + } + private ResolvedTypeMunger munger; + public AjSynthetic() {} + + public void write(DataOutputStream s) throws IOException {} + } + + public static class TypeMunger extends AjAttribute { + public static final String AttributeName = "org.aspectj.weaver.TypeMunger"; + + public String getNameString() { + return AttributeName; + } + private ResolvedTypeMunger munger; + public TypeMunger(ResolvedTypeMunger munger) { + this.munger = munger; + } + + public void write(DataOutputStream s) throws IOException { + munger.write(s); + } + + public ConcreteTypeMunger reify(World world, ResolvedTypeX aspectType) { + return world.concreteTypeMunger(munger, aspectType); + } + } + + public static class WeaverState extends AjAttribute { + public static final String AttributeName = "org.aspectj.weaver.WeaverState"; + + public String getNameString() { + return AttributeName; + } + private WeaverStateKind kind; + public WeaverState(WeaverStateKind kind) { + this.kind = kind; + } + public void write(DataOutputStream s) throws IOException { + kind.write(s); + } + + public WeaverStateKind reify() { + return kind; + } + } + + public static class SourceContextAttribute extends AjAttribute { + public static final String AttributeName = "org.aspectj.weaver.SourceContext"; + + public String getNameString() { + return AttributeName; + } + + private String sourceFileName; + private int[] lineBreaks; + + public SourceContextAttribute(String sourceFileName, int[] lineBreaks) { + this.sourceFileName = sourceFileName; + this.lineBreaks = lineBreaks; + } + public void write(DataOutputStream s) throws IOException { + s.writeUTF(sourceFileName); + FileUtil.writeIntArray(s, lineBreaks); + } + + public static SourceContextAttribute read(DataInputStream s) throws IOException { + return new SourceContextAttribute(s.readUTF(), FileUtil.readIntArray(s)); + } + public int[] getLineBreaks() { + return lineBreaks; + } + + public String getSourceFileName() { + return sourceFileName; + } + } + + public static class PointcutDeclarationAttribute extends AjAttribute { + public static final String AttributeName = "org.aspectj.weaver.PointcutDeclaration"; + + public String getNameString() { + return AttributeName; + } + + private ResolvedPointcutDefinition pointcutDef; + public PointcutDeclarationAttribute(ResolvedPointcutDefinition pointcutDef) { + this.pointcutDef = pointcutDef; + } + public void write(DataOutputStream s) throws IOException { + pointcutDef.write(s); + } + + public ResolvedPointcutDefinition reify() { + return pointcutDef; + } + } + + public static class DeclareAttribute extends AjAttribute { + public static final String AttributeName = "org.aspectj.weaver.Declare"; + + public String getNameString() { + return AttributeName; + } + + private Declare declare; + public DeclareAttribute(Declare declare) { + this.declare = declare; + } + public void write(DataOutputStream s) throws IOException { + declare.write(s); + } + + public Declare getDeclare() { + return declare; + } + } + + public static class AdviceAttribute extends AjAttribute { + public static final String AttributeName = "org.aspectj.weaver.Advice"; + + public String getNameString() { + return AttributeName; + } + + private AdviceKind kind; + private Pointcut pointcut; + private int extraArgumentFlags; + private int start; + private int end; + private ISourceContext sourceContext; + + /** + * @param lexicalPosition must be greater than the lexicalPosition + * of any advice declared before this one in an aspect, otherwise, + * it can be any value. + */ + public AdviceAttribute(AdviceKind kind, Pointcut pointcut, int extraArgumentFlags, int start, int end, ISourceContext sourceContext) { + this.kind = kind; + this.pointcut = pointcut; + this.extraArgumentFlags = extraArgumentFlags; + this.start = start; + this.end = end; + this.sourceContext = sourceContext; + } + + public static AdviceAttribute read(DataInputStream s, ISourceContext context) throws IOException { + return new AdviceAttribute( + AdviceKind.read(s), + Pointcut.read(s, context), + s.readByte(), + s.readInt(), s.readInt(), context); + } + + public void write(DataOutputStream s) throws IOException { + kind.write(s); + pointcut.write(s); + s.writeByte(extraArgumentFlags); + s.writeInt(start); + s.writeInt(end); + } + public Advice reify(Member signature, World world) { + return world.concreteAdvice(kind, pointcut, signature, extraArgumentFlags, start, end, sourceContext); + } + + public String toString() { + return "AdviceAttribute(" + kind + ", " + pointcut + ", " + + extraArgumentFlags + ", " + start+")"; + } + + public int getExtraArgumentFlags() { + return extraArgumentFlags; + } + + public AdviceKind getKind() { + return kind; + } + + public Pointcut getPointcut() { + return pointcut; + } + + } + + public static class Aspect extends AjAttribute { + public static final String AttributeName = "org.aspectj.weaver.Aspect"; + public String getNameString() { + return AttributeName; + } + private PerClause perClause; + + public Aspect(PerClause perClause) { + this.perClause = perClause; + } + + public PerClause reify(ResolvedTypeX inAspect) { + //XXXperClause.concretize(inAspect); + return perClause; + } + + public void write(DataOutputStream s) throws IOException { + perClause.write(s); + } + } + + public static class PrivilegedAttribute extends AjAttribute { + public static final String AttributeName = "org.aspectj.weaver.Privileged"; + + public String getNameString() { + return AttributeName; + } + + private ResolvedMember[] accessedMembers; + public PrivilegedAttribute(ResolvedMember[] accessedMembers) { + this.accessedMembers = accessedMembers; + } + public void write(DataOutputStream s) throws IOException { + s.writeInt(accessedMembers.length); + for (int i = 0, len = accessedMembers.length; i < len; i++) { + accessedMembers[i].write(s); + } + } + + public ResolvedMember[] getAccessedMembers() { + return accessedMembers; + } + + public static PrivilegedAttribute read(DataInputStream s, ISourceContext context) throws IOException { + int len = s.readInt(); + ResolvedMember[] members = new ResolvedMember[len]; + for (int i=0; i < len; i++) { + members[i] = ResolvedMember.readResolvedMember(s, context); + } + return new PrivilegedAttribute(members); + } + } + + + public static class EffectiveSignatureAttribute extends AjAttribute { + public static final String AttributeName = "org.aspectj.weaver.EffectiveSignature"; + + public String getNameString() { + return AttributeName; + } + + private ResolvedMember effectiveSignature; + private Shadow.Kind shadowKind; + private boolean weaveBody; + public EffectiveSignatureAttribute(ResolvedMember effectiveSignature, Shadow.Kind shadowKind, boolean weaveBody) { + this.effectiveSignature = effectiveSignature; + this.shadowKind = shadowKind; + this.weaveBody = weaveBody; + } + public void write(DataOutputStream s) throws IOException { + effectiveSignature.write(s); + shadowKind.write(s); + s.writeBoolean(weaveBody); + } + + public static EffectiveSignatureAttribute read(DataInputStream s, ISourceContext context) throws IOException { + return new EffectiveSignatureAttribute( + ResolvedMember.readResolvedMember(s, context), + Shadow.Kind.read(s), + s.readBoolean()); + } + + public ResolvedMember getEffectiveSignature() { + return effectiveSignature; + } + + public String toString() { + return "EffectiveSignatureAttribute(" + effectiveSignature + ", " + shadowKind + ")"; + } + + public Shadow.Kind getShadowKind() { + return shadowKind; + } + + public boolean isWeaveBody() { + return weaveBody; + } + + } + + + +} diff --git a/weaver/src/org/aspectj/weaver/AjcMemberMaker.java b/weaver/src/org/aspectj/weaver/AjcMemberMaker.java new file mode 100644 index 000000000..7996af37f --- /dev/null +++ b/weaver/src/org/aspectj/weaver/AjcMemberMaker.java @@ -0,0 +1,508 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver; + +import java.lang.reflect.Modifier; + + +public class AjcMemberMaker { + private static final int PUBLIC_STATIC_FINAL = + Modifier.PUBLIC | Modifier.STATIC | Modifier.FINAL; + + private static final int PRIVATE_STATIC = + Modifier.PRIVATE | Modifier.STATIC; + + private static final int PUBLIC_STATIC = + Modifier.PUBLIC | Modifier.STATIC; + + private static final int VISIBILITY = + Modifier.PUBLIC | Modifier.PRIVATE | Modifier.PROTECTED; + + public static final TypeX CFLOW_STACK_TYPE = + TypeX.forName(NameMangler.CFLOW_STACK_TYPE); + public static final TypeX AROUND_CLOSURE_TYPE = + TypeX.forName("org.aspectj.runtime.internal.AroundClosure"); + + public static final TypeX CONVERSIONS_TYPE = + TypeX.forName("org.aspectj.runtime.internal.Conversions"); + + public static final TypeX NO_ASPECT_BOUND_EXCEPTION = + TypeX.forName("org.aspectj.lang.NoAspectBoundException"); + + public static ResolvedMember ajcClinitMethod(TypeX declaringType) { + return new ResolvedMember( + Member.METHOD, + declaringType, + PRIVATE_STATIC, + NameMangler.AJC_CLINIT_NAME, + "()V"); + } + + public static Member noAspectBoundExceptionInit() { + return new ResolvedMember( + Member.METHOD, + NO_ASPECT_BOUND_EXCEPTION, + Modifier.PUBLIC, + "<init>", + "()V"); + } + + + public static ResolvedMember perCflowPush(TypeX declaringType) { + return new ResolvedMember( + Member.METHOD, + declaringType, + PUBLIC_STATIC, + NameMangler.PERCFLOW_PUSH_METHOD, + "()V"); + } + + public static ResolvedMember perCflowField(TypeX declaringType) { + return new ResolvedMember( + Member.FIELD, + declaringType, + PUBLIC_STATIC_FINAL, + NameMangler.PERCFLOW_FIELD_NAME, + CFLOW_STACK_TYPE.getSignature()); + } + + public static ResolvedMember perSingletonField(TypeX declaringType) { + return new ResolvedMember( + Member.FIELD, + declaringType, + PUBLIC_STATIC_FINAL, + NameMangler.PERSINGLETON_FIELD_NAME, + declaringType.getSignature()); + } + + + public static ResolvedMember perObjectField(TypeX declaringType, ResolvedTypeX aspectType) { + int modifiers = Modifier.PRIVATE; + if (!TypeX.SERIALIZABLE.isAssignableFrom(aspectType, aspectType.getWorld())) { + modifiers |= Modifier.TRANSIENT; + } + return new ResolvedMember( + Member.FIELD, + declaringType, + modifiers, + aspectType, + NameMangler.perObjectInterfaceField(aspectType), + TypeX.NONE); + } + + + public static ResolvedMember perObjectBind(TypeX declaringType) { + return new ResolvedMember( + Member.METHOD, + declaringType, + PUBLIC_STATIC, + NameMangler.PEROBJECT_BIND_METHOD, + "(Ljava/lang/Object;)V"); + } + + + public static TypeX perObjectInterfaceType(TypeX aspectType) { + return TypeX.forName(aspectType.getName()+"$ajcMightHaveAspect"); + } + + public static ResolvedMember perObjectInterfaceGet(TypeX aspectType) { + return new ResolvedMember( + Member.METHOD, + perObjectInterfaceType(aspectType), + Modifier.PUBLIC | Modifier.ABSTRACT, + NameMangler.perObjectInterfaceGet(aspectType), + "()" + aspectType.getSignature()); + } + + public static ResolvedMember perObjectInterfaceSet(TypeX aspectType) { + return new ResolvedMember( + Member.METHOD, + perObjectInterfaceType(aspectType), + Modifier.PUBLIC | Modifier.ABSTRACT, + NameMangler.perObjectInterfaceSet(aspectType), + "(" + aspectType.getSignature() + ")V"); + } + + + + + public static ResolvedMember perSingletonAspectOfMethod(TypeX declaringType) { + return new ResolvedMember(Member.METHOD, + declaringType, PUBLIC_STATIC, "aspectOf", + "()" + declaringType.getSignature()); + } + + public static ResolvedMember perSingletonHasAspectMethod(TypeX declaringType) { + return new ResolvedMember(Member.METHOD, + declaringType, PUBLIC_STATIC, "hasAspect", + "()Z"); + }; + + public static ResolvedMember perCflowAspectOfMethod(TypeX declaringType) { + return perSingletonAspectOfMethod(declaringType); + } + + public static ResolvedMember perCflowHasAspectMethod(TypeX declaringType) { + return perSingletonHasAspectMethod(declaringType); + }; + + public static ResolvedMember perObjectAspectOfMethod(TypeX declaringType) { + return new ResolvedMember(Member.METHOD, + declaringType, PUBLIC_STATIC, "aspectOf", + "(Ljava/lang/Object;)" + declaringType.getSignature()); + } + + public static ResolvedMember perObjectHasAspectMethod(TypeX declaringType) { + return new ResolvedMember(Member.METHOD, + declaringType, PUBLIC_STATIC, "hasAspect", + "(Ljava/lang/Object;)Z"); + }; + + // -- privileged accessors + + public static ResolvedMember privilegedAccessMethodForMethod(TypeX aspectType, ResolvedMember method) { + String sig; + sig = method.getSignature(); + return new ResolvedMember(Member.METHOD, + method.getDeclaringType(), + Modifier.PUBLIC | (method.isStatic() ? Modifier.STATIC : 0), + NameMangler.privilegedAccessMethodForMethod(method.getName(), + method.getDeclaringType(), aspectType), + sig); + //XXX needs thrown exceptions to be correct + } + + public static ResolvedMember privilegedAccessMethodForFieldGet(TypeX aspectType, Member field) { + String sig; + if (field.isStatic()) { + sig = "()" + field.getSignature(); + } else { + sig = "(" + field.getDeclaringType().getSignature() + ")" + field.getSignature(); + } + + return new ResolvedMember(Member.METHOD, + field.getDeclaringType(), + PUBLIC_STATIC, //Modifier.PUBLIC | (field.isStatic() ? Modifier.STATIC : 0), + NameMangler.privilegedAccessMethodForFieldGet(field.getName(), + field.getDeclaringType(), aspectType), + sig); + } + + public static ResolvedMember privilegedAccessMethodForFieldSet(TypeX aspectType, Member field) { + String sig; + if (field.isStatic()) { + sig = "(" + field.getSignature() + ")V"; + } else { + sig = "(" + field.getDeclaringType().getSignature() + field.getSignature() + ")V"; + } + + return new ResolvedMember(Member.METHOD, + field.getDeclaringType(), + PUBLIC_STATIC, //Modifier.PUBLIC | (field.isStatic() ? Modifier.STATIC : 0), + NameMangler.privilegedAccessMethodForFieldSet(field.getName(), + field.getDeclaringType(), aspectType), + sig); + } + + + + // --- runtimeLibrary api stuff + + public static Member cflowStackPeekInstance() { + return new Member( + Member.METHOD, + CFLOW_STACK_TYPE, + 0, + "peekInstance", + "()Ljava/lang/Object;"); + } + + public static Member cflowStackPushInstance() { + return new Member( + Member.METHOD, + CFLOW_STACK_TYPE, + 0, + "pushInstance", + "(Ljava/lang/Object;)V"); + } + + public static Member cflowStackIsValid() { + return new Member( + Member.METHOD, + CFLOW_STACK_TYPE, + 0, + "isValid", + "()Z"); + } + public static Member cflowStackInit() { + return new Member( + Member.CONSTRUCTOR, + CFLOW_STACK_TYPE, + 0, + "<init>", + "()V"); + } + public static Member aroundClosurePreInitializationField() { + return new Member( + Member.FIELD, + AROUND_CLOSURE_TYPE, + 0, + "preInitializationState", + "[Ljava/lang/Object;"); + } + public static Member aroundClosurePreInitializationGetter() { + return new Member( + Member.METHOD, + AROUND_CLOSURE_TYPE, + 0, + "getPreInitializationState", + "()[Ljava/lang/Object;"); + } + + + public static ResolvedMember preIntroducedConstructor( + TypeX aspectType, + TypeX targetType, + TypeX[] paramTypes) + { + return new ResolvedMember( + Member.METHOD, + aspectType, + PUBLIC_STATIC_FINAL, + TypeX.OBJECTARRAY, + NameMangler.preIntroducedConstructor(aspectType, targetType), + paramTypes); + } + + public static ResolvedMember postIntroducedConstructor( + TypeX aspectType, + TypeX targetType, + TypeX[] paramTypes) + { + return new ResolvedMember( + Member.METHOD, + aspectType, + PUBLIC_STATIC_FINAL, + ResolvedTypeX.VOID, + NameMangler.postIntroducedConstructor(aspectType, targetType), + TypeX.insert(targetType, paramTypes)); + } + + public static ResolvedMember interConstructor(ResolvedTypeX targetType, ResolvedMember constructor, TypeX aspectType) { +// +// ResolvedTypeX targetType, +// TypeX[] argTypes, +// int modifiers) +// { + ResolvedMember ret = + new ResolvedMember( + Member.CONSTRUCTOR, + targetType, + Modifier.PUBLIC, + ResolvedTypeX.VOID, + "<init>", + constructor.getParameterTypes()); + //System.out.println("ret: " + ret + " mods: " + Modifier.toString(modifiers)); + if (Modifier.isPublic(constructor.getModifiers())) + return ret; + int i = 0; + while (true) { + ret = addCookieTo(ret, aspectType); + if (targetType.lookupMemberNoSupers(ret) == null) + return ret; + } + } + + public static ResolvedMember interFieldInitializer(ResolvedMember field, TypeX aspectType) { + return new ResolvedMember(Member.METHOD, aspectType, PUBLIC_STATIC, + NameMangler.interFieldInitializer(aspectType, field.getDeclaringType(), field.getName()), + field.isStatic() ? "()V" : "(" + field.getDeclaringType().getSignature() + ")V" + ); + } + + + + private static int makePublic(int modifiers) { + return (modifiers & ~VISIBILITY) | Modifier.PUBLIC; + } + + + /** + * This static method goes on the aspect that declares the inter-type field + */ + public static ResolvedMember interFieldSetDispatcher(ResolvedMember field, TypeX aspectType) { + return new ResolvedMember(Member.METHOD, aspectType, PUBLIC_STATIC, + ResolvedTypeX.VOID, + NameMangler.interFieldSetDispatcher(aspectType, field.getDeclaringType(), field.getName()), + field.isStatic() ? new TypeX[] {field.getReturnType()} + : new TypeX[] {field.getDeclaringType(), field.getReturnType()} + ); + } + + /** + * This static method goes on the aspect that declares the inter-type field + */ + public static ResolvedMember interFieldGetDispatcher(ResolvedMember field, TypeX aspectType) { + return new ResolvedMember(Member.METHOD, aspectType, PUBLIC_STATIC, + field.getReturnType(), + NameMangler.interFieldGetDispatcher(aspectType, field.getDeclaringType(), field.getName()), + field.isStatic() ? TypeX.NONE : new TypeX[] {field.getDeclaringType()} + ); + } + + /** + * This field goes on the class the field + * is declared onto + */ + public static ResolvedMember interFieldClassField(ResolvedMember field, TypeX aspectType) { + return new ResolvedMember(Member.FIELD, field.getDeclaringType(), makePublic(field.getModifiers()), + field.getReturnType(), + NameMangler.interFieldClassField(field.getModifiers(), aspectType, field.getDeclaringType(), field.getName()), + TypeX.NONE + ); + } + + + /** + * This field goes on top-most implementers of the interface the field + * is declared onto + */ + public static ResolvedMember interFieldInterfaceField(ResolvedMember field, TypeX onClass, TypeX aspectType) { + return new ResolvedMember(Member.FIELD, onClass, Modifier.PUBLIC, + field.getReturnType(), + NameMangler.interFieldInterfaceField(aspectType, field.getDeclaringType(), field.getName()), + TypeX.NONE + ); + } + + /** + * This instance method goes on the interface the field is declared onto + * as well as its top-most implementors + */ + public static ResolvedMember interFieldInterfaceSetter(ResolvedMember field, ResolvedTypeX onType, TypeX aspectType) { + int modifiers = Modifier.PUBLIC; + if (onType.isInterface()) modifiers |= Modifier.ABSTRACT; + return new ResolvedMember(Member.METHOD, onType, modifiers, + ResolvedTypeX.VOID, + NameMangler.interFieldInterfaceSetter(aspectType, field.getDeclaringType(), field.getName()), + new TypeX[] {field.getReturnType()} + ); + } + + /** + * This instance method goes on the interface the field is declared onto + * as well as its top-most implementors + */ + public static ResolvedMember interFieldInterfaceGetter(ResolvedMember field, ResolvedTypeX onType, TypeX aspectType) { + int modifiers = Modifier.PUBLIC; + if (onType.isInterface()) modifiers |= Modifier.ABSTRACT; + return new ResolvedMember(Member.METHOD, onType, modifiers, + field.getReturnType(), + NameMangler.interFieldInterfaceGetter(aspectType, field.getDeclaringType(), field.getName()), + TypeX.NONE + ); + } + + + + + /** + * This method goes on the target type of the inter-type method. (and possibly the topmost-implemeters, + * if the target type is an interface) + */ + public static ResolvedMember interMethod(ResolvedMember meth, TypeX aspectType, boolean onInterface) + { + if (Modifier.isPublic(meth.getModifiers()) && !onInterface) return meth; + + int modifiers = makePublic(meth.getModifiers()); + if (onInterface) modifiers |= Modifier.ABSTRACT; + + return new ResolvedMember(Member.METHOD, meth.getDeclaringType(), + modifiers, + meth.getReturnType(), + NameMangler.interMethod(meth.getModifiers(), aspectType, meth.getDeclaringType(), meth.getName()), + meth.getParameterTypes()); + } + + /** + * This static method goes on the declaring aspect of the inter-type method. + */ + public static ResolvedMember interMethodDispatcher(ResolvedMember meth, TypeX aspectType) + { + TypeX[] paramTypes = meth.getParameterTypes(); + if (!meth.isStatic()) { + paramTypes = TypeX.insert(meth.getDeclaringType(), paramTypes); + } + + return new ResolvedMember(Member.METHOD, aspectType, PUBLIC_STATIC, + meth.getReturnType(), + NameMangler.interMethodDispatcher(aspectType, meth.getDeclaringType(), meth.getName()), + paramTypes); + } + + /** + * This static method goes on the declaring aspect of the inter-type method. + */ + public static ResolvedMember interMethodBody(ResolvedMember meth, TypeX aspectType) + { + TypeX[] paramTypes = meth.getParameterTypes(); + if (!meth.isStatic()) { + paramTypes = TypeX.insert(meth.getDeclaringType(), paramTypes); + } + + return new ResolvedMember(Member.METHOD, aspectType, PUBLIC_STATIC, + meth.getReturnType(), + NameMangler.interMethodBody(aspectType, meth.getDeclaringType(), meth.getName()), + paramTypes); + } + + + + + private static ResolvedMember addCookieTo(ResolvedMember ret, TypeX aspectType) { + TypeX[] params = ret.getParameterTypes(); + + TypeX[] freshParams = TypeX.add(params, aspectType); + return new ResolvedMember( + ret.getKind(), + ret.getDeclaringType(), + ret.getModifiers(), + ret.getReturnType(), + ret.getName(), + freshParams); + } + + public static ResolvedMember toObjectConversionMethod(TypeX fromType) { + if (fromType.isPrimitive()) { + String name = fromType.toString() + "Object"; + return new ResolvedMember( + Member.METHOD, + CONVERSIONS_TYPE, + PUBLIC_STATIC, + TypeX.OBJECT, + name, + new TypeX[] { fromType }); + } else { + return null; + } + } + public static Member interfaceConstructor(ResolvedTypeX resolvedTypeX) { + return new ResolvedMember( + Member.CONSTRUCTOR, + resolvedTypeX, + Modifier.PUBLIC, + "<init>", + "()V"); + } +} diff --git a/weaver/src/org/aspectj/weaver/AsmAdaptor.java b/weaver/src/org/aspectj/weaver/AsmAdaptor.java new file mode 100644 index 000000000..6830ab2bb --- /dev/null +++ b/weaver/src/org/aspectj/weaver/AsmAdaptor.java @@ -0,0 +1,157 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver; + +import java.util.*; +import java.util.Iterator; + +import org.aspectj.asm.*; +import org.aspectj.asm.StructureModel; +import org.aspectj.bridge.*; +import org.aspectj.bridge.SourceLocation; + +public class AsmAdaptor { + public static void noteMunger(StructureModel model, Shadow shadow, ShadowMunger munger) { + if (munger instanceof Advice) { + Advice a = (Advice)munger; + if (a.getKind().isPerEntry() || a.getKind().isCflow()) { + // ??? might want to show these in the future + return; + } + +// System.out.println("--------------------------"); + ProgramElementNode targetNode = getNode(model, shadow); + ProgramElementNode adviceNode = getNode(model, a); +// System.out.println("> target: " + targetNode + ", advice: " + adviceNode); + createAppropriateLinks(targetNode, adviceNode); + } + } + + private static void createAppropriateLinks( + ProgramElementNode target, + ProgramElementNode advice) + { + if (target == null || advice == null) return; + addLink(target, new LinkNode(advice), org.aspectj.asm.AdviceAssociation.METHOD_RELATION, true); + addLink(advice, new LinkNode(target), org.aspectj.asm.AdviceAssociation.METHOD_RELATION, false); +// System.out.println(">> added target: " + target + ", advice: " + advice); + } + + private static void addLink( + ProgramElementNode onNode, + LinkNode linkNode, + Relation relation, + boolean isBack) + { + RelationNode node = null; + String relationName = isBack ? relation.getBackNavigationName() : relation.getForwardNavigationName(); + + //System.err.println("on: " + onNode + " relationName: " + relationName + " existin: " + onNode.getRelations()); + + for (Iterator i = onNode.getRelations().iterator(); i.hasNext();) { + RelationNode relationNode = (RelationNode) i.next(); + if (relationName.equals(relationNode.getName())) { + node = relationNode; + break; + } + } + if (node == null) { + node = new RelationNode(relation, relationName, new ArrayList()); + onNode.getRelations().add(node); + } + node.getChildren().add(linkNode); + + } + + private static ProgramElementNode getNode(StructureModel model, Advice a) { + //ResolvedTypeX inAspect = a.getConcreteAspect(); + Member member = a.getSignature(); + return lookupMember(model, member); + } + + private static ProgramElementNode getNode(StructureModel model, Shadow shadow) { + Member enclosingMember = shadow.getEnclosingCodeSignature(); + + ProgramElementNode enclosingNode = lookupMember(model, enclosingMember); + + Member shadowSig = shadow.getSignature(); + if (!shadowSig.equals(enclosingMember)) { + ProgramElementNode bodyNode = findOrCreateBodyNode(enclosingNode, shadowSig, shadow); + return bodyNode; + } else { + return enclosingNode; + } + } + + private static ProgramElementNode findOrCreateBodyNode( + ProgramElementNode enclosingNode, + Member shadowSig, Shadow shadow) + { + for (Iterator it = enclosingNode.getChildren().iterator(); it.hasNext(); ) { + ProgramElementNode node = (ProgramElementNode)it.next(); + if (shadowSig.getName().equals(node.getBytecodeName()) && + shadowSig.getSignature().equals(node.getBytecodeSignature())) + { + return node; + } + } + + ISourceLocation sl = shadow.getSourceLocation(); + + ProgramElementNode peNode = new ProgramElementNode( + shadow.toString(), + ProgramElementNode.Kind.CODE, + new SourceLocation(enclosingNode.getSourceLocation().getSourceFile(), sl.getLine()), + 0, + "", + new ArrayList()); + + System.err.println(peNode.getSourceLocation()); + peNode.setBytecodeName(shadowSig.getName()); + peNode.setBytecodeSignature(shadowSig.getSignature()); + enclosingNode.addChild(peNode); + return peNode; + } + + + + + + public static ProgramElementNode lookupMember(StructureModel model, Member member) { + TypeX declaringType = member.getDeclaringType(); + ProgramElementNode classNode = + model.findNodeForClass(declaringType.getPackageName(), declaringType.getClassName()); + return findMemberInClass(classNode, member); + } + + private static ProgramElementNode findMemberInClass( + ProgramElementNode classNode, + Member member) + { + if (classNode == null) return null; // XXX remove this check + for (Iterator it = classNode.getChildren().iterator(); it.hasNext(); ) { + ProgramElementNode node = (ProgramElementNode)it.next(); + if (member.getName().equals(node.getBytecodeName()) && + member.getSignature().equals(node.getBytecodeSignature())) + { + return node; + } + } + return null; + } + + + + +} diff --git a/weaver/src/org/aspectj/weaver/BCException.java b/weaver/src/org/aspectj/weaver/BCException.java new file mode 100644 index 000000000..35ec52780 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/BCException.java @@ -0,0 +1,29 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver; + +/** + * Exception to use inside the bcweaver. + */ +public class BCException extends RuntimeException { + + public BCException() { + super(); + } + + public BCException(String s) { + super(s); + } + +} diff --git a/weaver/src/org/aspectj/weaver/BetaException.java b/weaver/src/org/aspectj/weaver/BetaException.java new file mode 100644 index 000000000..9cf85fceb --- /dev/null +++ b/weaver/src/org/aspectj/weaver/BetaException.java @@ -0,0 +1,29 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver; + +/** + * Exception to use inside the bcweaver. + */ +public class BetaException extends RuntimeException { + + public BetaException() { + super(); + } + + public BetaException(String s) { + super(s); + } + +} diff --git a/weaver/src/org/aspectj/weaver/Checker.java b/weaver/src/org/aspectj/weaver/Checker.java new file mode 100644 index 000000000..eec7d05af --- /dev/null +++ b/weaver/src/org/aspectj/weaver/Checker.java @@ -0,0 +1,61 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver; + +import org.aspectj.weaver.patterns.*; +import org.aspectj.weaver.patterns.PerClause; +import org.aspectj.bridge.*; + + +public class Checker extends ShadowMunger { + + private String msg; + private boolean isError; + + public Checker(DeclareErrorOrWarning deow) { + super(deow.getPointcut(), deow.getStart(), deow.getEnd(), deow.getSourceContext()); + this.msg = deow.getMessage(); + this.isError = deow.isError(); + } + + public ShadowMunger concretize(ResolvedTypeX fromType, World world, PerClause clause) { + pointcut = pointcut.concretize(fromType, 0); + return this; + } + + public void specializeOn(Shadow shadow) { + throw new RuntimeException("illegal state"); + } + + public void implementOn(Shadow shadow) { + throw new RuntimeException("illegal state"); + } + + public boolean match(Shadow shadow, World world) { + if (super.match(shadow, world)) { + world.getMessageHandler().handleMessage( + new Message(msg, + isError ? IMessage.ERROR : IMessage.WARNING, + null, + shadow.getSourceLocation())); + } + return false; + } + + + public int compareTo(Object other) { + return 0; + } + +} diff --git a/weaver/src/org/aspectj/weaver/ConcreteTypeMunger.java b/weaver/src/org/aspectj/weaver/ConcreteTypeMunger.java new file mode 100644 index 000000000..9f55b28e9 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/ConcreteTypeMunger.java @@ -0,0 +1,76 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver; + +import org.aspectj.bridge.ISourceLocation; +import org.aspectj.util.PartialOrder; + +public abstract class ConcreteTypeMunger implements PartialOrder.PartialComparable { + protected ResolvedTypeMunger munger; + protected ResolvedTypeX aspectType; + + public ConcreteTypeMunger(ResolvedTypeMunger munger, ResolvedTypeX aspectType) { + this.munger = munger; + this.aspectType = aspectType; + } + + //public abstract boolean munge(LazyClassGen gen); + + public ResolvedTypeMunger getMunger() { + return munger; + } + + public ResolvedTypeX getAspectType() { + return aspectType; + } + + public ResolvedMember getSignature() { + return munger.getSignature(); + } + + public ISourceLocation getSourceLocation() { + return null; //XXX + } + + public boolean matches(ResolvedTypeX onType) { + if (munger == null) throw new RuntimeException("huh: " + this); + return munger.matches(onType); + } + + public ResolvedMember getMatchingSyntheticMember(Member member) { + return munger.getMatchingSyntheticMember(member, aspectType); + } + + public int compareTo(Object other) { + ConcreteTypeMunger o = (ConcreteTypeMunger) other; + + ResolvedTypeX otherAspect = o.aspectType; + + if (aspectType.equals(otherAspect)) { + return getSignature().getStart() < o.getSignature().getStart() ? -1: +1; + } else if (aspectType.isAssignableFrom(o.aspectType)) { + return +1; + } else if (o.aspectType.isAssignableFrom(aspectType)) { + return -1; + } else { + return 0; + } + } + + public int fallbackCompareTo(Object other) { + ConcreteTypeMunger o = (ConcreteTypeMunger) other; + return 0; + } + +} diff --git a/weaver/src/org/aspectj/weaver/CrosscuttingMembers.java b/weaver/src/org/aspectj/weaver/CrosscuttingMembers.java new file mode 100644 index 000000000..8635cdaf6 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/CrosscuttingMembers.java @@ -0,0 +1,201 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver; + +import java.util.*; + +import org.aspectj.weaver.patterns.*; +import org.aspectj.weaver.patterns.PerClause; +import org.aspectj.bridge.*; + + +/** + * This holds on to all members that have an invasive effect outside of + * there own compilation unit. These members need to be all gathered up and in + * a world before any weaving can take place. + * + * They are also important in the compilation process and need to be gathered + * up before the inter-type declaration weaving stage (unsurprisingly). + * + * All members are concrete. + */ + +public class CrosscuttingMembers { + private ResolvedTypeX inAspect; + private World world; + + private PerClause perClause; + + private List shadowMungers = new ArrayList(4); + private List typeMungers = new ArrayList(4); + + private List declareParents = new ArrayList(4); + private List declareSofts = new ArrayList(0); + private List declareDominates = new ArrayList(4); + + + public CrosscuttingMembers(ResolvedTypeX inAspect) { + this.inAspect = inAspect; + this.world = inAspect.getWorld(); + } + +// public void addConcreteShadowMungers(Collection c) { +// shadowMungers.addAll(c); +// } + + public void addConcreteShadowMunger(ShadowMunger m) { + // assert m is concrete + shadowMungers.add(m); + } + + public void addShadowMungers(Collection c) { + for (Iterator i = c.iterator(); i.hasNext(); ) { + addShadowMunger( (ShadowMunger)i.next() ); + } + } + + private void addShadowMunger(ShadowMunger m) { + //if (inAspect.isAbstract()) return; // we don't do mungers for abstract aspects + addConcreteShadowMunger(m.concretize(inAspect, world, perClause)); + } + + public void addTypeMungers(Collection c) { + typeMungers.addAll(c); + } + + public void addTypeMunger(ConcreteTypeMunger m) { + if (m == null) return; //??? + typeMungers.add(m); + } + + public void addDeclares(Collection c) { + for (Iterator i = c.iterator(); i.hasNext(); ) { + addDeclare( (Declare)i.next() ); + } + } + + + + public void addDeclare(Declare declare) { + // this is not extensible, oh well + if (declare instanceof DeclareErrorOrWarning) { + ShadowMunger m = new Checker((DeclareErrorOrWarning)declare); + addShadowMunger(m); + } else if (declare instanceof DeclareDominates) { + declareDominates.add(declare); + } else if (declare instanceof DeclareParents) { + declareParents.add(declare); + } else if (declare instanceof DeclareSoft) { + DeclareSoft d = (DeclareSoft)declare; + Pointcut concretePointcut = d.getPointcut().concretize(inAspect, 0); + declareSofts.add(new DeclareSoft(d.getException(), concretePointcut)); + ShadowMunger m = Advice.makeSoftener(world, concretePointcut, d.getException()); + addConcreteShadowMunger(m); + } else { + throw new RuntimeException("unimplemented"); + } + } + + public void addPrivilegedAccesses(Collection accessedMembers) { + for (Iterator i = accessedMembers.iterator(); i.hasNext(); ) { + addPrivilegedAccess( (ResolvedMember)i.next() ); + } + } + + private void addPrivilegedAccess(ResolvedMember member) { + //System.err.println("add priv access: " + member); + addTypeMunger(world.concreteTypeMunger(new PrivilegedAccessMunger(member), inAspect)); + } + + + + public Collection getCflowEntries() { + ArrayList ret = new ArrayList(); + for (Iterator i = shadowMungers.iterator(); i.hasNext(); ) { + ShadowMunger m = (ShadowMunger)i.next(); + if (m instanceof Advice) { + Advice a = (Advice)m; + if (a.getKind().isCflow()) { + ret.add(a); + } + } + } + return ret; + } + + public boolean replaceWith(CrosscuttingMembers other) { + boolean changed = false; + if (!perClause.equals(other.perClause)) { + changed = true; + perClause = other.perClause; + } + + //XXX all of the below should be set equality rather than list equality + if (!shadowMungers.equals(other.shadowMungers)) { + changed = true; + shadowMungers = other.shadowMungers; + } + + if (!typeMungers.equals(other.typeMungers)) { + changed = true; + typeMungers = other.typeMungers; + } + + if (!declareDominates.equals(other.declareDominates)) { + changed = true; + declareDominates = other.declareDominates; + } + + if (!declareParents.equals(other.declareParents)) { + changed = true; + declareParents = other.declareParents; + } + + if (!declareSofts.equals(other.declareSofts)) { + changed = true; + declareSofts = other.declareSofts; + } + + return changed; + } + + public PerClause getPerClause() { + return perClause; + } + + public void setPerClause(PerClause perClause) { + this.perClause = perClause.concretize(inAspect); + } + + public List getDeclareDominates() { + return declareDominates; + } + + public List getDeclareParents() { + return declareParents; + } + + public List getDeclareSofts() { + return declareSofts; + } + + public List getShadowMungers() { + return shadowMungers; + } + + public List getTypeMungers() { + return typeMungers; + } + +} diff --git a/weaver/src/org/aspectj/weaver/CrosscuttingMembersSet.java b/weaver/src/org/aspectj/weaver/CrosscuttingMembersSet.java new file mode 100644 index 000000000..ceebbfeb6 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/CrosscuttingMembersSet.java @@ -0,0 +1,140 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver; + +import java.util.*; + +import org.aspectj.weaver.patterns.*; +import org.aspectj.weaver.patterns.PerClause; +import org.aspectj.bridge.*; + + +/** + * This holds on to all CrosscuttingMembers for a world. It handles + * management of change. + */ + +public class CrosscuttingMembersSet { + private World world; + private Map members = new HashMap(); + + private List shadowMungers = null; + private List typeMungers = null; + private List declareSofts = null; + private List declareParents = null; + private List declareDominates = null; + + public CrosscuttingMembersSet(World world) { + this.world = world; + } + + /** + * @return whether or not that was a change to the global signature + * XXX for efficiency we will need a richer representation than this + */ + public boolean addOrReplaceAspect(ResolvedTypeX aspectType) { + CrosscuttingMembers xcut = (CrosscuttingMembers)members.get(aspectType); + if (xcut == null) { + members.put(aspectType, aspectType.collectCrosscuttingMembers()); + clearCaches(); + return true; + } else { + if (xcut.replaceWith(aspectType.collectCrosscuttingMembers())) { + clearCaches(); + return true; + } else { + return false; + } + } + } + + public void deleteAspect(TypeX aspectType) { + members.remove(aspectType); + clearCaches(); + } + + //XXX only for testing + public void addFixedCrosscuttingMembers(ResolvedTypeX aspectType) { + members.put(aspectType, aspectType.crosscuttingMembers); + clearCaches(); + } + + + private void clearCaches() { + shadowMungers = null; + typeMungers = null; + declareSofts = null; + declareParents = null; + declareDominates = null; + } + + + public List getShadowMungers() { + if (shadowMungers == null) { + ArrayList ret = new ArrayList(); + for (Iterator i = members.values().iterator(); i.hasNext(); ) { + ret.addAll(((CrosscuttingMembers)i.next()).getShadowMungers()); + } + shadowMungers = ret; + } + return shadowMungers; + } + + public List getTypeMungers() { + if (typeMungers == null) { + ArrayList ret = new ArrayList(); + for (Iterator i = members.values().iterator(); i.hasNext(); ) { + ret.addAll(((CrosscuttingMembers)i.next()).getTypeMungers()); + } + typeMungers = ret; + } + return typeMungers; + } + + public List getDeclareSofts() { + if (declareSofts == null) { + ArrayList ret = new ArrayList(); + for (Iterator i = members.values().iterator(); i.hasNext(); ) { + ret.addAll(((CrosscuttingMembers)i.next()).getDeclareSofts()); + } + declareSofts = ret; + } + return declareSofts; + } + + public List getDeclareParents() { + if (declareParents == null) { + ArrayList ret = new ArrayList(); + for (Iterator i = members.values().iterator(); i.hasNext(); ) { + ret.addAll(((CrosscuttingMembers)i.next()).getDeclareParents()); + } + declareParents = ret; + } + return declareParents; + } + + public List getDeclareDominates() { + if (declareDominates == null) { + ArrayList ret = new ArrayList(); + for (Iterator i = members.values().iterator(); i.hasNext(); ) { + ret.addAll(((CrosscuttingMembers)i.next()).getDeclareDominates()); + } + declareDominates = ret; + } + return declareDominates; + } + + + +} diff --git a/weaver/src/org/aspectj/weaver/IClassWeaver.java b/weaver/src/org/aspectj/weaver/IClassWeaver.java new file mode 100644 index 000000000..3dd152e8d --- /dev/null +++ b/weaver/src/org/aspectj/weaver/IClassWeaver.java @@ -0,0 +1,29 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver; + +/** + * An IClassWeaver is initialized with a class (a type, really, but let's ignore that for now) + * and a world, and has one method that actually weaves the contents of the world into the class + * implementation. + */ + +public interface IClassWeaver { + + /** perform the weaving. + * + * @return <code>true</code> if the class is changed by the weaving, <code>false</code> otherwise. + */ + boolean weave(); +} diff --git a/weaver/src/org/aspectj/weaver/IHasPosition.java b/weaver/src/org/aspectj/weaver/IHasPosition.java new file mode 100644 index 000000000..8fb5688a7 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/IHasPosition.java @@ -0,0 +1,33 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver; + +public interface IHasPosition { + /** + * The starting index of this location in the character stream. + */ + int getStart(); + + /** + * The ending index of this location in the character stream + * + * This points to the last character in this token. + * + * If a location truly had no contents, then start == end + 1. We don't recommend this. + */ + int getEnd(); +// +// String getFileName(); + +} diff --git a/weaver/src/org/aspectj/weaver/IHasSourceLocation.java b/weaver/src/org/aspectj/weaver/IHasSourceLocation.java new file mode 100644 index 000000000..2ca7a01c8 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/IHasSourceLocation.java @@ -0,0 +1,21 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver; + +import org.aspectj.bridge.ISourceLocation; + +public interface IHasSourceLocation extends IHasPosition { + ISourceContext getSourceContext(); + ISourceLocation getSourceLocation(); +} diff --git a/weaver/src/org/aspectj/weaver/ISourceContext.java b/weaver/src/org/aspectj/weaver/ISourceContext.java new file mode 100644 index 000000000..c5cf2620f --- /dev/null +++ b/weaver/src/org/aspectj/weaver/ISourceContext.java @@ -0,0 +1,20 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver; + +import org.aspectj.bridge.ISourceLocation; + +public interface ISourceContext { + public ISourceLocation makeSourceLocation(IHasPosition position); +} diff --git a/weaver/src/org/aspectj/weaver/IWeaver.java b/weaver/src/org/aspectj/weaver/IWeaver.java new file mode 100644 index 000000000..4c38424ac --- /dev/null +++ b/weaver/src/org/aspectj/weaver/IWeaver.java @@ -0,0 +1,24 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver; + +/** + * A weaver is given all the aspects it will weave. It should create an appropriate kind of + * IWorld. It then should be given a bunch of classes (types with implementation), creates an + * appropriate IClassWeaver for each such class, and weaves. The IWeaver is responsible for + * IO. + */ +public interface IWeaver { + +} diff --git a/weaver/src/org/aspectj/weaver/IntMap.java b/weaver/src/org/aspectj/weaver/IntMap.java new file mode 100644 index 000000000..712a64637 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/IntMap.java @@ -0,0 +1,134 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver; + +import java.util.*; + +public class IntMap { + public static final IntMap EMPTY = new IntMap(0) { + public boolean directlyInAdvice() { return true; } + public Advice getEnclosingAdvice() { return null; } //XXX possible + }; + + + // XXX begin hack to avoid a signature refactoring in Pointcut + private Advice enclosingAdvice; + private List/*ResolvedPointcutDefinition*/ enclosingDefinition = new ArrayList(); + + public void pushEnclosingDefinition(ResolvedPointcutDefinition def) { + enclosingDefinition.add(def); + } + + public void popEnclosingDefinitition() { + enclosingDefinition.remove(enclosingDefinition.size()-1); + } + + + public ResolvedPointcutDefinition peekEnclosingDefinitition() { + return (ResolvedPointcutDefinition)enclosingDefinition.get(enclosingDefinition.size()-1); + } + + + public boolean directlyInAdvice() { + return enclosingDefinition.isEmpty(); + } + + public Advice getEnclosingAdvice() { + return enclosingAdvice; + } + + public void setEnclosingAdvice(Advice advice) { + this.enclosingAdvice = advice; + } + + public Member getAdviceSignature() { + return getEnclosingAdvice().signature; + } + + public void copyContext(IntMap bindings) { + this.enclosingAdvice = bindings.enclosingAdvice; + this.enclosingDefinition = bindings.enclosingDefinition; + } + + // XXX end hack to avoid a signature refactoring in Pointcut + + + private static final int MISSING = -1; + + private int[] map; + + private IntMap(int[] map) { + this.map = map; + } + public IntMap() { + map = new int[0]; + } + public IntMap(int initialCapacity) { + map = new int[initialCapacity]; + for (int i = 0; i < initialCapacity; i++) { + map[i] = MISSING; + } + } + + public void put(int key, int val) { + /* assert (val >= 0 && key >= 0) */ + if (key >= map.length) { + int[] tmp = new int[key * 2 + 1]; //??? better expansion function + System.arraycopy(map, 0, tmp, 0, map.length); + for (int i = map.length, len = tmp.length; i < len; i++) tmp[i] = MISSING; + map = tmp; + } + map[key] = val; + } + + public int get(int key) { + return map[key]; + } + + public boolean hasKey(int key) { + return (key < map.length && map[key] != MISSING); + } + + // ---- factory methods + + public static IntMap idMap(int size) { + int[] map = new int[size]; + for (int i = 0; i < size; i++) { + map[i] = i; + } + return new IntMap(map); + } + + // ---- from object + + public String toString() { + StringBuffer buf = new StringBuffer("["); + boolean seenFirst = false; + for (int i = 0, len = map.length; i < len; i++) { + if (map[i] != MISSING) { + if (seenFirst) { + buf.append(", "); + } + seenFirst = true; + buf.append(i); + buf.append(" -> "); + buf.append(map[i]); + } + } + buf.append("]"); + return buf.toString(); + } + + +} diff --git a/weaver/src/org/aspectj/weaver/Iterators.java b/weaver/src/org/aspectj/weaver/Iterators.java new file mode 100644 index 000000000..2685ba92f --- /dev/null +++ b/weaver/src/org/aspectj/weaver/Iterators.java @@ -0,0 +1,215 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver; + +import java.util.*; + +public final class Iterators { + + /** + * Private constructor, nobody should ever make one of these + */ + private Iterators() { + super(); + } + + /** + * A getter represents a mapping function from Object to Iterator + */ + public interface Getter { + Iterator get(Object target); + } + + /** + * A filter represents a mapping function from Iterator to Iterator + */ + public interface Filter { + Iterator filter(Iterator in); + } + + /** + * Create a new filter F that, when wrapped around another iterator I, + * creates a new iterator I' that will return only those values of I + * that have not yet been returned by I', discarding duplicates. + */ + public static Filter dupFilter() { + return new Filter() { + final Set seen = new HashSet(); // should have weak ptrs? + public Iterator filter(final Iterator in) { + return new Iterator() { + boolean fresh = false; + Object peek; + public boolean hasNext() { + if (fresh) return true; + while (true) { + if (! in.hasNext()) return false; + peek = in.next(); + if (! seen.contains(peek)) { + return fresh = true; + } else { + peek = null; // garbage collection + } + } + } + public Object next() { + if (! hasNext()) throw new NoSuchElementException(); + Object ret = peek; + peek = null; + fresh = false; + return ret; + } + public void remove() { + throw new UnsupportedOperationException(); + } + }; + } + }; + } + + /** + * Creates an iterator that will return the elements of a specified array, + * in order. Like Arrays.asList(o).iterator(), without all that pesky safety. + */ + + public static Iterator array(final Object[] o) { + return new Iterator() { + int i = 0; + int len = o.length; + public boolean hasNext() { + return i < len; + } + public Object next() { + if (i < len) { + return o[i++]; + } else { + throw new NoSuchElementException(); + } + } + public void remove() { + throw new UnsupportedOperationException(); + } + }; + } + + /** creates an iterator I based on a base iterator A and a getter G. + * I returns, in order, forall (i in I), G(i). + */ + public static Iterator mapOver(final Iterator a, final Getter g) { + return new Iterator() { + Iterator delegate = + new Iterator() { + public boolean hasNext() { + if (! a.hasNext()) return false; + Object o = a.next(); + delegate = append1(g.get(o), this); + return delegate.hasNext(); + } + public Object next() { + if (! hasNext()) throw new UnsupportedOperationException(); + return delegate.next(); + } + public void remove() { throw new UnsupportedOperationException(); } + }; + public boolean hasNext() { + return delegate.hasNext(); + } + public Object next() { + return delegate.next(); + } + public void remove() { throw new UnsupportedOperationException(); } + }; + } + + /** creates an iterator I based on a base iterator A and a getter G. + * I returns, in order, forall (i in I) i :: forall (i' in g(i)) recur(i', g) + */ + public static Iterator recur(final Object a, final Getter g) { + return new Iterator() { + Iterator delegate = one(a); + public boolean hasNext() { + return delegate.hasNext(); + } + public Object next() { + Object next = delegate.next(); + delegate = append(g.get(next), delegate); + return next; + } + public void remove() { throw new UnsupportedOperationException(); } + }; + } + /** creates an iterator I based on base iterators A and B. Returns + * the elements returned by A followed by those returned by B. If + * B is empty, simply returns A, and if A is empty, simply returns B. + * Do NOT USE if b.hasNext() is not idempotent. + */ + public static Iterator append(final Iterator a, final Iterator b) { + if (! b.hasNext()) return a; + return append1(a, b); + } + /** creates an iterator I based on base iterators A and B. Returns + * the elements returned by A followed by those returned by B. If A + * is empty, simply returns B. Guaranteed not to call B.hasNext() until + * A is empty. + */ + public static Iterator append1(final Iterator a, final Iterator b) { + if (! a.hasNext()) return b; + return new Iterator() { + public boolean hasNext() { return a.hasNext() || b.hasNext(); } + public Object next() { + if (a.hasNext()) return a.next(); + if (b.hasNext()) return b.next(); + throw new NoSuchElementException(); + } + public void remove() { throw new UnsupportedOperationException(); } + }; + } + /** creates an iterator I based on a base iterator A and an object O. Returns + * the elements returned by A, followed by O. + */ + public static Iterator snoc(final Iterator first, final Object last) { + return new Iterator() { + Object last1 = last; + public boolean hasNext() { return first.hasNext() || last1 != null; } + public Object next() { + if (first.hasNext()) return first.next(); + else if (last1 == null) throw new NoSuchElementException(); + Object ret = last1; + last1 = null; + return ret; + } + public void remove() { throw new UnsupportedOperationException(); } + }; + } + /** creates an iterator I based on an object O. Returns O, once. + */ + public static Iterator one(final Object it) { + return new Iterator() { + boolean avail = true; + public boolean hasNext() { return avail; } + public Object next() { + if (! avail) throw new NoSuchElementException(); + avail = false; + return it; + } + public void remove() { throw new UnsupportedOperationException(); } + }; + } + /** creates an empty iterator. + */ + public static final Iterator EMPTY = new Iterator() { + public boolean hasNext() { return false; } + public Object next() { throw new NoSuchElementException(); } + public void remove() { throw new UnsupportedOperationException(); } + }; +} diff --git a/weaver/src/org/aspectj/weaver/Lint.java b/weaver/src/org/aspectj/weaver/Lint.java new file mode 100644 index 000000000..82135a1df --- /dev/null +++ b/weaver/src/org/aspectj/weaver/Lint.java @@ -0,0 +1,141 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver; + +import java.io.*; +import java.io.File; +import java.text.MessageFormat; +import java.util.*; + +import org.aspectj.bridge.*; +import org.aspectj.bridge.IMessage.Kind; + +public class Lint { + private Map kinds = new HashMap(); + private World world; + + public final Kind invalidAbsoluteTypeName = + new Kind("invalidAbsoluteTypeName", "no match for this type name: {0}"); + + public final Kind invalidWildcardTypeName = + new Kind("invalidWildcardTypeName", "no match for this type pattern: {0}"); + + public final Kind unresolvableMember = + new Kind("unresolvableMember", "can not resolve this member: {0}"); + + + public Lint(World world) { + this.world = world; + } + + + public void setAll(String messageKind) { + setAll(getMessageKind(messageKind)); + } + + private void setAll(IMessage.Kind messageKind) { + for (Iterator i = kinds.values().iterator(); i.hasNext(); ) { + Kind kind = (Kind)i.next(); + kind.setKind(messageKind); + } + } + + public void setFromProperties(File file) { + try { + InputStream s = new FileInputStream(file); + setFromProperties(s); + } catch (IOException ioe) { + MessageUtil.error(world.getMessageHandler(), "problem loading Xlint properties file: " + + file.getPath() + ", " + ioe.getMessage()); + } + } + + public void loadDefaultProperties() { + InputStream s = getClass().getResourceAsStream("XlintDefault.properties"); + if (s == null) { + MessageUtil.warn(world.getMessageHandler(), "couldn't load XlintDefault.properties"); + return; + } + try { + setFromProperties(s); + } catch (IOException ioe) { + MessageUtil.error(world.getMessageHandler(), "problem loading XlintDefault.properties, " + + ioe.getMessage()); + } + + } + + + private void setFromProperties(InputStream s) throws IOException { + Properties p = new Properties(); + p.load(s); + setFromProperties(p); + } + + + public void setFromProperties(Properties properties) { + for (Iterator i = properties.entrySet().iterator(); i.hasNext(); ) { + Map.Entry entry = (Map.Entry)i.next(); + Kind kind = (Kind)kinds.get(entry.getKey()); + if (kind == null) { + MessageUtil.error(world.getMessageHandler(), "invalid Xlint key: " + entry.getKey()); + } else { + kind.setKind(getMessageKind((String)entry.getValue())); + } + } + } + + private IMessage.Kind getMessageKind(String v) { + if (v.equals("ignore")) return null; + else if (v.equals("warning")) return IMessage.WARNING; + else if (v.equals("error")) return IMessage.ERROR; + + MessageUtil.error(world.getMessageHandler(), + "invalid Xlint message kind (must be one of ignore, warning, error): " + v); + return null; + } + + + + public class Kind { + private String name; + private String message; + private IMessage.Kind kind = IMessage.WARNING; + public Kind(String name, String message) { + this.name = name; + this.message = message; + kinds.put(this.name, this); + } + + public boolean isEnabled() { + return kind != null; + } + + public IMessage.Kind getKind() { + return kind; + } + + public void setKind(IMessage.Kind kind) { + this.kind = kind; + } + + public void signal(String info, ISourceLocation location) { + if (kind == null) return; + + String text = MessageFormat.format(message, new Object[] {info} ); + text += " [Xlint:" + name + "]"; + world.getMessageHandler().handleMessage(new Message(text, kind, null, location)); + } + } +} diff --git a/weaver/src/org/aspectj/weaver/Member.java b/weaver/src/org/aspectj/weaver/Member.java new file mode 100644 index 000000000..284454fdf --- /dev/null +++ b/weaver/src/org/aspectj/weaver/Member.java @@ -0,0 +1,745 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver; + +import java.io.*; +import java.lang.reflect.Modifier; +import java.util.*; + +import org.aspectj.util.TypeSafeEnum; + +public class Member implements Comparable { + + private final Kind kind; + private final TypeX declaringType; + protected final int modifiers; // protected because ResolvedMember uses it + private final TypeX returnType; + private final String name; + private final TypeX[] parameterTypes; + private final String signature; + + public Member( + Kind kind, + TypeX declaringType, + int modifiers, + String name, + String signature) + { + this.kind = kind; + this.declaringType = declaringType; + this.modifiers = modifiers; + this.name = name; + this.signature = signature; + if (kind == FIELD) { + this.returnType = TypeX.forSignature(signature); + this.parameterTypes = TypeX.NONE; + } else { + Object[] returnAndParams = signatureToTypes(signature); + this.returnType = (TypeX) returnAndParams[0]; + this.parameterTypes = (TypeX[]) returnAndParams[1]; + } + } + + public Member( + Kind kind, + TypeX declaringType, + int modifiers, + TypeX returnType, + String name, + TypeX[] parameterTypes) + { + super(); + this.kind = kind; + this.declaringType = declaringType; + this.modifiers = modifiers; + this.returnType = returnType; + this.name = name; + this.parameterTypes = parameterTypes; + if (kind == FIELD) { + this.signature = returnType.getSignature(); + } else { + this.signature = typesToSignature(returnType, parameterTypes); + } + } + + public ResolvedMember resolve(World world) { + return world.resolve(this); + } + + // ---- utility methods + + /** returns an Object[] pair of TypeX, TypeX[] representing return type, + * argument types parsed from the JVM bytecode signature of a method. Yes, + * this should actually return a nice statically-typed pair object, but we + * don't have one of those. + * + * <blockquote><pre> + * TypeX.signatureToTypes("()[Z")[0].equals(Type.forSignature("[Z")) + * TypeX.signatureToTypes("(JJ)I")[1] + * .equals(TypeX.forSignatures(new String[] {"J", "J"})) + * </pre></blockquote> + * + * @param signature the JVM bytecode method signature string we want to break apart + * @return a pair of TypeX, TypeX[] representing the return types and parameter types. + */ + public static String typesToSignature(TypeX returnType, TypeX[] paramTypes) { + StringBuffer buf = new StringBuffer(); + buf.append("("); + for (int i = 0, len = paramTypes.length; i < len; i++) { + buf.append(paramTypes[i].getSignature()); + } + buf.append(")"); + buf.append(returnType.getSignature()); + return buf.toString(); + } + + /** returns an Object[] pair of TypeX, TypeX[] representing return type, + * argument types parsed from the JVM bytecode signature of a method. Yes, + * this should actually return a nice statically-typed pair object, but we + * don't have one of those. + * + * <blockquote><pre> + * TypeX.signatureToTypes("()[Z")[0].equals(Type.forSignature("[Z")) + * TypeX.signatureToTypes("(JJ)I")[1] + * .equals(TypeX.forSignatures(new String[] {"J", "J"})) + * </pre></blockquote> + * + * @param signature the JVM bytecode method signature string we want to break apart + * @return a pair of TypeX, TypeX[] representing the return types and parameter types. + */ + private static Object[] signatureToTypes(String sig) { + List l = new ArrayList(); + int i = 1; + while (true) { + char c = sig.charAt(i); + if (c == ')') break; + int start = i; + while (c == '[') c = sig.charAt(++i); + if (c == 'L') { + i = sig.indexOf(';', start) + 1; + l.add(TypeX.forSignature(sig.substring(start, i))); + } else { + l.add(TypeX.forSignature(sig.substring(start, ++i))); + } + } + TypeX[] paramTypes = (TypeX[]) l.toArray(new TypeX[l.size()]); + TypeX returnType = TypeX.forSignature(sig.substring(i+1, sig.length())); + return new Object[] { returnType, paramTypes }; + } + + // ---- factory methods + public static Member field(String declaring, int mods, String name, String signature) { + return field(declaring, mods, TypeX.forSignature(signature), name); + } + public static Member field(TypeX declaring, int mods, String name, TypeX type) { + return new Member(FIELD, declaring, mods, type, name, TypeX.NONE); + } + public static Member method(TypeX declaring, int mods, String name, String signature) { + Object[] pair = signatureToTypes(signature); + return method(declaring, mods, (TypeX) pair[0], name, (TypeX[]) pair[1]); + } + public static Member pointcut(TypeX declaring, String name, String signature) { + Object[] pair = signatureToTypes(signature); + return pointcut(declaring, 0, (TypeX) pair[0], name, (TypeX[]) pair[1]); + } + + + private static Member field(String declaring, int mods, TypeX ty, String name) { + return new Member( + FIELD, + TypeX.forName(declaring), + mods, + ty, + name, + TypeX.NONE); + } + + public static Member method(TypeX declTy, int mods, TypeX rTy, String name, TypeX[] paramTys) { + return new Member( + //??? this calls <clinit> a method + name.equals("<init>") ? CONSTRUCTOR : METHOD, + declTy, + mods, + rTy, + name, + paramTys); + } + private static Member pointcut(TypeX declTy, int mods, TypeX rTy, String name, TypeX[] paramTys) { + return new Member( + POINTCUT, + declTy, + mods, + rTy, + name, + paramTys); + } + + public static Member makeExceptionHandlerSignature(TypeX inType, TypeX catchType) { + return new Member( + HANDLER, + inType, + Modifier.STATIC, + "<catch>", + "(" + catchType.getSignature() + ")V"); + } + + // ---- parsing methods + + /** Takes a string in this form: + * + * <blockquote><pre> + * static? TypeName TypeName.Id + * </pre></blockquote> + * Pretty much just for testing, and as such should perhaps be moved. + */ + + public static Member fieldFromString(String str) { + str = str.trim(); + final int len = str.length(); + int i = 0; + int mods = 0; + if (str.startsWith("static", i)) { + mods = Modifier.STATIC; + i += 6; + while (Character.isWhitespace(str.charAt(i))) i++; + } + int start = i; + while (! Character.isWhitespace(str.charAt(i))) i++; + TypeX retTy = TypeX.forName(str.substring(start, i)); + + start = i; + i = str.lastIndexOf('.'); + TypeX declaringTy = TypeX.forName(str.substring(start, i).trim()); + start = ++i; + String name = str.substring(start, len).trim(); + return new Member( + FIELD, + declaringTy, + mods, + retTy, + name, + TypeX.NONE); + } + + /** Takes a string in this form: + * + * <blockquote><pre> + * (static|interface|private)? TypeName TypeName . Id ( TypeName , ...) + * </pre></blockquote> + * Pretty much just for testing, and as such should perhaps be moved. + */ + + public static Member methodFromString(String str) { + str = str.trim(); + final int len = str.length(); + int i = 0; + + int mods = 0; + if (str.startsWith("static", i)) { + mods = Modifier.STATIC; + i += 6; + } else if (str.startsWith("interface", i)) { + mods = Modifier.INTERFACE; + i += 9; + } else if (str.startsWith("private", i)) { + mods = Modifier.PRIVATE; + i += 7; + } + while (Character.isWhitespace(str.charAt(i))) i++; + + int start = i; + while (! Character.isWhitespace(str.charAt(i))) i++; + TypeX returnTy = TypeX.forName(str.substring(start, i)); + + start = i; + i = str.indexOf('(', i); + i = str.lastIndexOf('.', i); + TypeX declaringTy = TypeX.forName(str.substring(start, i).trim()); + + start = ++i; + i = str.indexOf('(', i); + String name = str.substring(start, i).trim(); + start = ++i; + i = str.indexOf(')', i); + + String[] paramTypeNames = parseIds(str.substring(start, i).trim()); + + return method(declaringTy, mods, returnTy, name, TypeX.forNames(paramTypeNames)); + } + + private static String[] parseIds(String str) { + if (str.length() == 0) return ZERO_STRINGS; + List l = new ArrayList(); + int start = 0; + while (true) { + int i = str.indexOf(',', start); + if (i == -1) { + l.add(str.substring(start).trim()); + break; + } + l.add(str.substring(start, i).trim()); + start = i+1; + } + return (String[]) l.toArray(new String[l.size()]); + } + + private static final String[] ZERO_STRINGS = new String[0]; + + // ---- things we know without resolution + + public boolean equals(Object other) { + if (! (other instanceof Member)) return false; + Member o = (Member) other; + + return (kind == o.kind + && name.equals(o.name) + && signature.equals(o.signature) + && declaringType.equals(o.declaringType)); + } + + public int compareTo(Object other) { + Member o = (Member) other; + + int i = getName().compareTo(o.getName()); + if (i != 0) return i; + return getSignature().compareTo(o.getSignature()); + } + + /** + * Equality is checked based on the underlying signature, so the hash code + * of a member is based on its kind, name, signature, and declaring type. The + * algorithm for this was taken from page 38 of effective java. + */ + private volatile int hashCode = 0; + public int hashCode() { + if (hashCode == 0) { + int result = 17; + result = 37*result + kind.hashCode(); + result = 37*result + name.hashCode(); + result = 37*result + signature.hashCode(); + result = 37*result + declaringType.hashCode(); + hashCode = result; + } + return hashCode; + } + + public String toString() { + StringBuffer buf = new StringBuffer(); + buf.append(returnType); + buf.append(' '); + buf.append(declaringType); + buf.append('.'); + buf.append(name); + if (kind != FIELD) { + buf.append("("); + if (parameterTypes.length != 0) { + buf.append(parameterTypes[0]); + for (int i=1, len = parameterTypes.length; i < len; i++) { + buf.append(", "); + buf.append(parameterTypes[i]); + } + } + buf.append(")"); + } + return buf.toString(); + } + + public String toLongString() { + StringBuffer buf = new StringBuffer(); + buf.append(kind); + buf.append(' '); + if (modifiers != 0) { + buf.append(Modifier.toString(modifiers)); + buf.append(' '); + } + buf.append(toString()); + buf.append(" <"); + buf.append(signature); + buf.append(" >"); + return buf.toString(); + } + + public Kind getKind() { return kind; } + public TypeX getDeclaringType() { return declaringType; } + public TypeX getReturnType() { return returnType; } + public TypeX getType() { return returnType; } + public String getName() { return name; } + public TypeX[] getParameterTypes() { return parameterTypes; } + public String getSignature() { return signature; } + public int getArity() { return parameterTypes.length; } + + public boolean isCompatibleWith(Member am) { + if (kind != METHOD || am.getKind() != METHOD) return true; + if (! name.equals(am.getName())) return true; + if (! equalTypes(getParameterTypes(), am.getParameterTypes())) return true; + return getReturnType().equals(am.getReturnType()); + } + + private static boolean equalTypes(TypeX[] a, TypeX[] b) { + int len = a.length; + if (len != b.length) return false; + for (int i = 0; i < len; i++) { + if (!a[i].equals(b[i])) return false; + } + return true; + } + + // ---- things we know only with resolution + + public int getModifiers(World world) { + return world.getModifiers(this); + } + + public TypeX[] getExceptions(World world) { + return world.getExceptions(this); + } + + public final boolean isProtected(World world) { + return Modifier.isProtected(world.getModifiers(this)); + } + public final boolean isStatic(World world) { + return Modifier.isStatic(world.getModifiers(this)); + } + public final boolean isStrict(World world) { + return Modifier.isStrict(world.getModifiers(this)); + } + + public final boolean isStatic() { + return Modifier.isStatic(modifiers); + } + + public final boolean isInterface() { + return Modifier.isInterface(modifiers); // this is kinda weird + } + + public final boolean isPrivate() { + return Modifier.isPrivate(modifiers); + } + + public final int getCallsiteModifiers() { + return modifiers & ~ Modifier.INTERFACE; + } + + public final String getExtractableName() { + if (name.equals("<init>")) return "init$"; + else if (name.equals("<clinit>")) return "clinit$"; + else return name; + } + + // ---- fields 'n' stuff + + public static final Member[] NONE = new Member[0]; + + public static class Kind extends TypeSafeEnum { + public Kind(String name, int key) { super(name, key); } + + public static Kind read(DataInputStream s) throws IOException { + int key = s.readByte(); + switch(key) { + case 1: return METHOD; + case 2: return FIELD; + case 3: return CONSTRUCTOR; + case 4: return STATIC_INITIALIZATION; + case 5: return POINTCUT; + case 6: return ADVICE; + case 7: return HANDLER; + } + throw new BCException("weird kind " + key); + } + } + + public static final Kind METHOD = new Kind("METHOD", 1); + public static final Kind FIELD = new Kind("FIELD", 2); + public static final Kind CONSTRUCTOR = new Kind("CONSTRUCTOR", 3); + public static final Kind STATIC_INITIALIZATION = new Kind("STATIC_INITIALIZATION", 4); + public static final Kind POINTCUT = new Kind("POINTCUT", 5); + public static final Kind ADVICE = new Kind("ADVICE", 6); + public static final Kind HANDLER = new Kind("HANDLER", 7); + + + + + public Collection/*ResolvedTypeX*/ getDeclaringTypes(World world) { + ResolvedTypeX myType = getDeclaringType().resolve(world); + Collection ret = new HashSet(); + if (kind == CONSTRUCTOR) { + // this is wrong if the member doesn't exist, but that doesn't matter + ret.add(myType); + } else if (isStatic() || kind == FIELD) { + walkUpStatic(ret, myType); + } else { + walkUp(ret, myType); + } + + return ret; + } + + private boolean walkUp(Collection acc, ResolvedTypeX curr) { + if (acc.contains(curr)) return true; + + boolean b = false; + for (Iterator i = curr.getDirectSupertypes(); i.hasNext(); ) { + b |= walkUp(acc, (ResolvedTypeX)i.next()); + } + + if (!b) { + b = curr.lookupMemberNoSupers(this) != null; + } + if (b) acc.add(curr); + return b; + } + + private boolean walkUpStatic(Collection acc, ResolvedTypeX curr) { + if (curr.lookupMemberNoSupers(this) != null) { + acc.add(curr); + return true; + } else { + boolean b = false; + for (Iterator i = curr.getDirectSupertypes(); i.hasNext(); ) { + b |= walkUp(acc, (ResolvedTypeX)i.next()); + } + if (b) acc.add(curr); + return b; + } + } + + // ---- reflective thisJoinPoint stuff + public String getSignatureMakerName() { + Kind kind = getKind(); + if (kind == METHOD) { + return "makeMethodSig"; + } else if (kind == CONSTRUCTOR) { + return "makeConstructorSig"; + } else if (kind == FIELD) { + return "makeFieldSig"; + } else if (kind == HANDLER) { + return "makeCatchClauseSig"; + } else if (kind == STATIC_INITIALIZATION) { + return "makeInitializerSig"; + } else if (kind == ADVICE) { + return "makeAdviceSig"; + } else { + throw new RuntimeException("unimplemented"); + } + } + + + + + public String getSignatureType() { + Kind kind = getKind(); + if (kind == METHOD) { + return "org.aspectj.lang.reflect.MethodSignature"; + } else if (kind == CONSTRUCTOR) { + return "org.aspectj.lang.reflect.ConstructorSignature"; + } else if (kind == FIELD) { + return "org.aspectj.lang.reflect.FieldSignature"; + } else if (kind == HANDLER) { + return "org.aspectj.lang.reflect.CatchClauseSignature"; + } else if (kind == STATIC_INITIALIZATION) { + return "org.aspectj.lang.reflect.InitializerSignature"; + } else if (kind == ADVICE) { + return "org.aspectj.lang.reflect.AdviceSignature"; + } else { + throw new RuntimeException("unimplemented"); + } + } + + public String getSignatureString(World world) { + Kind kind = getKind(); + if (kind == METHOD) { + return getMethodSignatureString(world); + } else if (kind == CONSTRUCTOR) { + return getConstructorSignatureString(world); + } else if (kind == FIELD) { + return getFieldSignatureString(world); + } else if (kind == HANDLER) { + return getHandlerSignatureString(world); + } else if (kind == STATIC_INITIALIZATION) { + return getStaticInitializationSignatureString(world); + } else if (kind == ADVICE) { + return getAdviceSignatureString(world); + } else { + throw new RuntimeException("unimplemented"); + } + } + + private String getHandlerSignatureString(World world) { + StringBuffer buf = new StringBuffer(); + buf.append(makeString(0)); + buf.append('-'); + //buf.append(getName()); + buf.append('-'); + buf.append(makeString(getDeclaringType())); + buf.append('-'); + buf.append(makeString(getParameterTypes()[0])); + buf.append('-'); + //XXX we don't actually try to find the handler parameter name + //XXX it probably wouldn't be too hard + String pName = "<missing>"; + //String[] pNames = getParameterNames(world); + //if (pNames != null) pName = pNames[0]; + buf.append(pName); + buf.append('-'); + return buf.toString(); + } + + private String getStaticInitializationSignatureString(World world) { + StringBuffer buf = new StringBuffer(); + buf.append(makeString(getModifiers(world))); + buf.append('-'); + //buf.append(getName()); + buf.append('-'); + buf.append(makeString(getDeclaringType())); + buf.append('-'); + return buf.toString(); + } + + + + protected String getAdviceSignatureString(World world) { + StringBuffer buf = new StringBuffer(); + buf.append(makeString(getModifiers(world))); + buf.append('-'); + buf.append(getName()); + buf.append('-'); + buf.append(makeString(getDeclaringType())); + buf.append('-'); + buf.append(makeString(getParameterTypes())); + buf.append('-'); + buf.append(makeString(getParameterNames(world))); + buf.append('-'); + buf.append(makeString(getExceptions(world))); + buf.append('-'); + buf.append(makeString(getReturnType())); + buf.append('-'); + return buf.toString(); + } + + + protected String getMethodSignatureString(World world) { + StringBuffer buf = new StringBuffer(); + buf.append(makeString(getModifiers(world))); + buf.append('-'); + buf.append(getName()); + buf.append('-'); + buf.append(makeString(getDeclaringType())); + buf.append('-'); + buf.append(makeString(getParameterTypes())); + buf.append('-'); + buf.append(makeString(getParameterNames(world))); + buf.append('-'); + buf.append(makeString(getExceptions(world))); + buf.append('-'); + buf.append(makeString(getReturnType())); + buf.append('-'); + return buf.toString(); + } + + + + protected String getConstructorSignatureString(World world) { + StringBuffer buf = new StringBuffer(); + buf.append(makeString(getModifiers(world))); + buf.append('-'); + buf.append('-'); + buf.append(makeString(getDeclaringType())); + buf.append('-'); + buf.append(makeString(getParameterTypes())); + buf.append('-'); + buf.append(makeString(getParameterNames(world))); + buf.append('-'); + buf.append(makeString(getExceptions(world))); + buf.append('-'); + return buf.toString(); + } + + + + + protected String getFieldSignatureString(World world) { + StringBuffer buf = new StringBuffer(); + buf.append(makeString(getModifiers(world))); + buf.append('-'); + buf.append(getName()); + buf.append('-'); + buf.append(makeString(getDeclaringType())); + buf.append('-'); + buf.append(makeString(getReturnType())); + buf.append('-'); + return buf.toString(); + } + + protected String makeString(int i) { + return Integer.toString(i, 16); //??? expensive + } + + + + + protected String makeString(TypeX t) { + // this is the inverse of the odd behavior for Class.forName w/ arrays + if (t.isArray()) { + return t.getSignature(); + } else { + return t.getName(); + } + } + + + + protected String makeString(TypeX[] types) { + if (types == null) return ""; + StringBuffer buf = new StringBuffer(); + for (int i = 0, len=types.length; i < len; i++) { + buf.append(makeString(types[i])); + buf.append(':'); + } + return buf.toString(); + } + + + + protected String makeString(String[] names) { + if (names == null) return ""; + StringBuffer buf = new StringBuffer(); + for (int i = 0, len=names.length; i < len; i++) { + buf.append(names[i]); + buf.append(':'); + } + return buf.toString(); + } + + public String[] getParameterNames(World world) { + return world.getParameterNames(this); + } + + + + + + + // ---- + + + + + + + + + + +} + diff --git a/weaver/src/org/aspectj/weaver/NameMangler.java b/weaver/src/org/aspectj/weaver/NameMangler.java new file mode 100644 index 000000000..69a3271fa --- /dev/null +++ b/weaver/src/org/aspectj/weaver/NameMangler.java @@ -0,0 +1,300 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver; + +import java.lang.reflect.Modifier; + +import org.aspectj.util.FuzzyBoolean; +import org.aspectj.weaver.bcel.*; +import org.aspectj.weaver.bcel.BcelObjectType; + +public class NameMangler { + private NameMangler() { + throw new RuntimeException("static"); + } + + public static final String PREFIX = "ajc$"; + + + public static final String CFLOW_STACK_TYPE = "org.aspectj.runtime.internal.CFlowStack"; + public static final String SOFT_EXCEPTION_TYPE = "org.aspectj.lang.SoftException"; + + public static final String PERSINGLETON_FIELD_NAME = PREFIX + "perSingletonInstance"; + public static final String PERCFLOW_FIELD_NAME = PREFIX + "perCflowStack"; + //public static final String PERTHIS_FIELD_NAME = PREFIX + "perSingletonInstance"; + + // ----- + public static final String PERCFLOW_PUSH_METHOD = PREFIX + "perCflowPush"; + + public static final String PEROBJECT_BIND_METHOD = PREFIX + "perObjectBind"; + + public static final String AJC_CLINIT_NAME = PREFIX + "clinit"; + + + + public static String perObjectInterfaceGet(TypeX aspectType) { + return makeName(aspectType.getNameAsIdentifier(), "perObjectGet"); + } + + public static String perObjectInterfaceSet(TypeX aspectType) { + return makeName(aspectType.getNameAsIdentifier(), "perObjectSet"); + } + + public static String perObjectInterfaceField(TypeX aspectType) { + return makeName(aspectType.getNameAsIdentifier(), "perObjectField"); + } + + + + public static String privilegedAccessMethodForMethod(String name, TypeX objectType, TypeX aspectType) { + return makeName("privMethod", aspectType.getNameAsIdentifier(), + objectType.getNameAsIdentifier(), name); + } + + public static String privilegedAccessMethodForFieldGet(String name, TypeX objectType, TypeX aspectType) { + return makeName("privFieldGet", aspectType.getNameAsIdentifier(), + objectType.getNameAsIdentifier(), name); + } + + public static String privilegedAccessMethodForFieldSet(String name, TypeX objectType, TypeX aspectType) { + return makeName("privFieldSet", aspectType.getNameAsIdentifier(), + objectType.getNameAsIdentifier(), name); + } + + + + /** + * The name of methods corresponding to advice declarations + */ + public static String adviceName(TypeX aspectType, AdviceKind kind, int position) { + return makeName(kind.getName(), aspectType.getNameAsIdentifier(), + Integer.toHexString(position)); + } + + /** + * This field goes on top-most implementers of the interface the field + * is declared onto + */ + public static String interFieldInterfaceField(TypeX aspectType, TypeX interfaceType, String name) { + return makeName("interField", aspectType.getNameAsIdentifier(), + interfaceType.getNameAsIdentifier(), name); + } + + /** + * This instance method goes on the interface the field is declared onto + * as well as its top-most implementors + */ + public static String interFieldInterfaceSetter(TypeX aspectType, TypeX interfaceType, String name) { + return makeName("interFieldSet", aspectType.getNameAsIdentifier(), + interfaceType.getNameAsIdentifier(), name); + } + + + /** + * This instance method goes on the interface the field is declared onto + * as well as its top-most implementors + */ + public static String interFieldInterfaceGetter(TypeX aspectType, TypeX interfaceType, String name) { + return makeName("interFieldGet", aspectType.getNameAsIdentifier(), + interfaceType.getNameAsIdentifier(), name); + } + + + /** + * This static method goes on the aspect that declares the inter-type field + */ + public static String interFieldSetDispatcher(TypeX aspectType, TypeX onType, String name) { + return makeName("interFieldSetDispatch", aspectType.getNameAsIdentifier(), + onType.getNameAsIdentifier(), name); + } + + /** + * This static method goes on the aspect that declares the inter-type field + */ + public static String interFieldGetDispatcher(TypeX aspectType, TypeX onType, String name) { + return makeName("interFieldGetDispatch", aspectType.getNameAsIdentifier(), + onType.getNameAsIdentifier(), name); + } + + + /** + * This field goes on the class the field + * is declared onto + */ + public static String interFieldClassField(int modifiers, TypeX aspectType, TypeX classType, String name) { + if (Modifier.isPublic(modifiers)) return name; + //??? might want to handle case where aspect and class are in same package similar to public + return makeName("interField", makeVisibilityName(modifiers, aspectType), name); + } + +// /** +// * This static method goes on the aspect that declares the inter-type field +// */ +// public static String classFieldSetDispatcher(TypeX aspectType, TypeX classType, String name) { +// return makeName("interFieldSetDispatch", aspectType.getNameAsIdentifier(), +// classType.getNameAsIdentifier(), name); +// } +// +// /** +// * This static method goes on the aspect that declares the inter-type field +// */ +// public static String classFieldGetDispatcher(TypeX aspectType, TypeX classType, String name) +// { +// return makeName( +// "interFieldGetDispatch", +// aspectType.getNameAsIdentifier(), +// classType.getNameAsIdentifier(), +// name); +// } + + /** + * This static void method goes on the aspect that declares the inter-type field and is called + * from the appropriate place (target's initializer, or clinit, or topmost implementer's inits), + * to initialize the field; + */ + + public static String interFieldInitializer(TypeX aspectType, TypeX classType, String name) + { + return makeName( + "interFieldInit", + aspectType.getNameAsIdentifier(), + classType.getNameAsIdentifier(), + name); + } + + + // ---- + + /** + * This method goes on the target type of the inter-type method. (and possibly the topmost-implemeters, + * if the target type is an interface) + */ + public static String interMethod(int modifiers, TypeX aspectType, TypeX classType, String name) + { + if (Modifier.isPublic(modifiers)) return name; + //??? might want to handle case where aspect and class are in same package similar to public + return makeName("interMethodDispatch2", makeVisibilityName(modifiers, aspectType), name); + } + + /** + * This static method goes on the declaring aspect of the inter-type method. + */ + public static String interMethodDispatcher(TypeX aspectType, TypeX classType, String name) + { + return makeName("interMethodDispatch1", aspectType.getNameAsIdentifier(), + classType.getNameAsIdentifier(), name); + } + + /** + * This static method goes on the declaring aspect of the inter-type method. + */ + public static String interMethodBody(TypeX aspectType, TypeX classType, String name) + { + return makeName("interMethod", aspectType.getNameAsIdentifier(), + classType.getNameAsIdentifier(), name); + } + + // ---- + + /** + * This static method goes on the declaring aspect of the inter-type constructor. + */ + public static String preIntroducedConstructor( + TypeX aspectType, + TypeX targetType) + { + return makeName("preInterConstructor", aspectType.getNameAsIdentifier(), + targetType.getNameAsIdentifier()); + } + + /** + * This static method goes on the declaring aspect of the inter-type constructor. + */ + public static String postIntroducedConstructor( + TypeX aspectType, + TypeX targetType) + { + return makeName("postInterConstructor", aspectType.getNameAsIdentifier(), + targetType.getNameAsIdentifier()); + } + // ---- + + /** + * This static method goes on the declaring aspect of the inter-type method. + */ + public static String superDispatcher(TypeX classType, String name) + { + return makeName("superDispatch", + classType.getNameAsIdentifier(), name); + } + + // ---- + + private static TypeX getOutermostType(TypeX type) { + TypeX outerType = type.getDeclaringType(); + if (outerType == null) return type; + return getOutermostType(outerType); + } + + + private static String makeVisibilityName(int modifiers, TypeX aspectType) { + if (Modifier.isPrivate(modifiers)) { + return getOutermostType(aspectType).getNameAsIdentifier(); + } else if (Modifier.isProtected(modifiers)) { + throw new RuntimeException("protected inter-types not allowed"); + } else if (Modifier.isPublic(modifiers)) { + return ""; + } else { + return aspectType.getPackageNameAsIdentifier(); + } + } + + private static String makeName(String s1, String s2) { + return "ajc$" + s1 + "$" + s2; + } + public static String makeName(String s1, String s2, String s3) { + return "ajc$" + s1 + "$" + s2 + "$" + s3; + } + public static String makeName(String s1, String s2, String s3, String s4) { + return "ajc$" + s1 + "$" + s2 + "$" + s3 + "$" + s4; + } + public static String cflowStack(CrosscuttingMembers xcut) { + return makeName("cflowStack", Integer.toHexString(xcut.getCflowEntries().size())); + } + + + + public static String makeClosureClassName( + BcelObjectType enclosingType, + int index) + { + return enclosingType.getName() + "$AjcClosure" + index; + } + + public static String aroundCallbackMethodName( + Member shadowSig, + LazyClassGen enclosingType) + { + String ret = + shadowSig.getExtractableName() + + "_aroundBody" + + enclosingType.getNewGeneratedNameTag(); + return ret; + } + + public static String proceedMethodName(String adviceMethodName) { + return adviceMethodName + "proceed"; + } + +} diff --git a/weaver/src/org/aspectj/weaver/NewConstructorTypeMunger.java b/weaver/src/org/aspectj/weaver/NewConstructorTypeMunger.java new file mode 100644 index 000000000..5940d4789 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/NewConstructorTypeMunger.java @@ -0,0 +1,78 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver; + +import java.io.*; +import java.util.Set; + +import org.aspectj.weaver.ResolvedTypeMunger.Kind; + +public class NewConstructorTypeMunger extends ResolvedTypeMunger { + private ResolvedMember syntheticConstructor; + private ResolvedMember explicitConstructor; + + + public NewConstructorTypeMunger( + ResolvedMember signature, + ResolvedMember syntheticConstructor, + ResolvedMember explicitConstructor, + Set superMethodsCalled) + { + super(Constructor, signature); + this.syntheticConstructor = syntheticConstructor; + this.explicitConstructor = explicitConstructor; + this.setSuperMethodsCalled(superMethodsCalled); + + } + + //XXX horrible name clash here + public ResolvedMember getDispatchMethod(TypeX aspectType) { + return AjcMemberMaker.interMethodBody(signature, aspectType); + } + + public void write(DataOutputStream s) throws IOException { + kind.write(s); + signature.write(s); + syntheticConstructor.write(s); + explicitConstructor.write(s); + writeSuperMethodsCalled(s); + } + + public static ResolvedTypeMunger readConstructor(DataInputStream s, ISourceContext context) throws IOException { + return new NewConstructorTypeMunger( + ResolvedMember.readResolvedMember(s, context), + ResolvedMember.readResolvedMember(s, context), + ResolvedMember.readResolvedMember(s, context), + readSuperMethodsCalled(s)); + } + + public ResolvedMember getExplicitConstructor() { + return explicitConstructor; + } + + public ResolvedMember getSyntheticConstructor() { + return syntheticConstructor; + } + + public void setExplicitConstructor(ResolvedMember explicitConstructor) { + this.explicitConstructor = explicitConstructor; + } + + public ResolvedMember getMatchingSyntheticMember(Member member, ResolvedTypeX aspectType) { + ResolvedMember ret = getSyntheticConstructor(); + if (ResolvedTypeX.matches(ret, member)) return getSignature(); + return null; + } + +} diff --git a/weaver/src/org/aspectj/weaver/NewFieldTypeMunger.java b/weaver/src/org/aspectj/weaver/NewFieldTypeMunger.java new file mode 100644 index 000000000..f30db0970 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/NewFieldTypeMunger.java @@ -0,0 +1,43 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver; + +import java.io.*; +import java.lang.reflect.Modifier; +import java.util.Set; + +import org.aspectj.weaver.ResolvedTypeMunger.Kind; + +public class NewFieldTypeMunger extends ResolvedTypeMunger { + public NewFieldTypeMunger(ResolvedMember signature, Set superMethodsCalled) { + super(Field, signature); + this.setSuperMethodsCalled(superMethodsCalled); + } + + public ResolvedMember getInitMethod(TypeX aspectType) { + return AjcMemberMaker.interFieldInitializer(signature, aspectType); + } + + public void write(DataOutputStream s) throws IOException { + kind.write(s); + signature.write(s); + writeSuperMethodsCalled(s); + } + + public static ResolvedTypeMunger readField(DataInputStream s, ISourceContext context) throws IOException { + return new NewFieldTypeMunger( + ResolvedMember.readResolvedMember(s, context), + readSuperMethodsCalled(s)); + } +} diff --git a/weaver/src/org/aspectj/weaver/NewMethodTypeMunger.java b/weaver/src/org/aspectj/weaver/NewMethodTypeMunger.java new file mode 100644 index 000000000..32d77aea6 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/NewMethodTypeMunger.java @@ -0,0 +1,53 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver; + +import java.io.*; +import java.util.Set; + +import org.aspectj.weaver.ResolvedTypeMunger.Kind; + +public class NewMethodTypeMunger extends ResolvedTypeMunger { + public NewMethodTypeMunger( + ResolvedMember signature, + Set superMethodsCalled) + { + super(Method, signature); + this.setSuperMethodsCalled(superMethodsCalled); + + } + + //XXX horrible name clash here + public ResolvedMember getDispatchMethod(TypeX aspectType) { + return AjcMemberMaker.interMethodBody(signature, aspectType); + } + + public void write(DataOutputStream s) throws IOException { + kind.write(s); + signature.write(s); + writeSuperMethodsCalled(s); + } + + public static ResolvedTypeMunger readMethod(DataInputStream s, ISourceContext context) throws IOException { + return new NewMethodTypeMunger( + ResolvedMember.readResolvedMember(s, context), + readSuperMethodsCalled(s)); + } + + public ResolvedMember getMatchingSyntheticMember(Member member, ResolvedTypeX aspectType) { + ResolvedMember ret = AjcMemberMaker.interMethodDispatcher(getSignature(), aspectType); + if (ResolvedTypeX.matches(ret, member)) return getSignature(); + return null; + } +} diff --git a/weaver/src/org/aspectj/weaver/PerObjectInterfaceTypeMunger.java b/weaver/src/org/aspectj/weaver/PerObjectInterfaceTypeMunger.java new file mode 100644 index 000000000..7f5096882 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/PerObjectInterfaceTypeMunger.java @@ -0,0 +1,68 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver; + +import java.io.*; +import java.util.Set; + +import org.aspectj.weaver.ResolvedTypeMunger.Kind; +import org.aspectj.weaver.patterns.Pointcut; + +public class PerObjectInterfaceTypeMunger extends ResolvedTypeMunger { + private ResolvedMember getMethod; + private ResolvedMember setMethod; + private TypeX aspectType; + private TypeX interfaceType; + private Pointcut testPointcut; + + + public PerObjectInterfaceTypeMunger(TypeX aspectType, Pointcut testPointcut) { + super(PerObjectInterface, null); + this.aspectType = aspectType; + this.testPointcut = testPointcut; + this.interfaceType = AjcMemberMaker.perObjectInterfaceType(aspectType); + this.getMethod = AjcMemberMaker.perObjectInterfaceGet(aspectType); + this.setMethod = AjcMemberMaker.perObjectInterfaceSet(aspectType); + } + + + public void write(DataOutputStream s) throws IOException { + throw new RuntimeException("shouldn't be serialized"); + } + public TypeX getAspectType() { + return aspectType; + } + + public ResolvedMember getGetMethod() { + return getMethod; + } + + public TypeX getInterfaceType() { + return interfaceType; + } + + public ResolvedMember getSetMethod() { + return setMethod; + } + + public Pointcut getTestPointcut() { + return testPointcut; + } + + public boolean matches(ResolvedTypeX matchType) { + //??? this matches many more types than are needed + return !matchType.isInterface(); + } + +} diff --git a/weaver/src/org/aspectj/weaver/PrivilegedAccessMunger.java b/weaver/src/org/aspectj/weaver/PrivilegedAccessMunger.java new file mode 100644 index 000000000..c957a5656 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/PrivilegedAccessMunger.java @@ -0,0 +1,36 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver; + +import java.io.*; +import java.util.Set; + +import org.aspectj.weaver.ResolvedTypeMunger.Kind; +import org.aspectj.weaver.patterns.Pointcut; + +public class PrivilegedAccessMunger extends ResolvedTypeMunger { + public PrivilegedAccessMunger(ResolvedMember member) { + super(PrivilegedAccess, member); + } + + + public void write(DataOutputStream s) throws IOException { + throw new RuntimeException("shouldn't be serialized"); + } + + public ResolvedMember getMember() { + return getSignature(); + } + +} diff --git a/weaver/src/org/aspectj/weaver/ResolvedMember.java b/weaver/src/org/aspectj/weaver/ResolvedMember.java new file mode 100644 index 000000000..e7d007cde --- /dev/null +++ b/weaver/src/org/aspectj/weaver/ResolvedMember.java @@ -0,0 +1,156 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver; + +import java.io.*; + +import org.aspectj.bridge.ISourceLocation; + +/** + * This is the declared member, i.e. it will always correspond to an + * actual method/... declaration + */ +public class ResolvedMember extends Member implements IHasPosition { + + protected String[] parameterNames = null; + protected TypeX[] checkedExceptions = TypeX.NONE; + + + // these three fields hold the source location of this member + protected int start, end; + protected ISourceContext sourceContext = null; + + // ---- + + public ResolvedMember( + Kind kind, + TypeX declaringType, + int modifiers, + TypeX returnType, + String name, + TypeX[] parameterTypes) + { + super(kind, declaringType, modifiers, returnType, name, parameterTypes); + } + + public ResolvedMember( + Kind kind, + TypeX declaringType, + int modifiers, + String name, + String signature) + { + super(kind, declaringType, modifiers, name, signature); + } + + public static final ResolvedMember[] NONE = new ResolvedMember[0]; + + // ---- + + public final int getModifiers(World world) { + return modifiers; + } + public final int getModifiers() { + return modifiers; + } + + // ---- + + + public final TypeX[] getExceptions(World world) { + return getExceptions(); + } + + public TypeX[] getExceptions() { + return checkedExceptions; + } + + public ShadowMunger getAssociatedShadowMunger() { + return null; + } + + // ??? true or false? + public boolean isAjSynthetic() { + return true; + } + + public void write(DataOutputStream s) throws IOException { + getKind().write(s); + getDeclaringType().write(s); + s.writeInt(modifiers); + s.writeUTF(getName()); + s.writeUTF(getSignature()); + TypeX.write(getExceptions(), s); + + s.writeInt(getStart()); + s.writeInt(getEnd()); + + } + + public static ResolvedMember readResolvedMember(DataInputStream s, ISourceContext sourceContext) throws IOException { + ResolvedMember m = new ResolvedMember(Kind.read(s), TypeX.read(s), s.readInt(), s.readUTF(), s.readUTF()); + m.checkedExceptions = TypeX.readArray(s); + m.start = s.readInt(); + m.end = s.readInt(); + m.sourceContext = sourceContext; + return m; + } + + public ResolvedMember resolve(World world) { + return this; + } + + public ISourceContext getSourceContext(World world) { + return getDeclaringType().resolve(world).getSourceContext(); + } + + public final String[] getParameterNames() { + return parameterNames; + } + public final String[] getParameterNames(World world) { + return getParameterNames(); + } + + public AjAttribute.EffectiveSignatureAttribute getEffectiveSignature() { + return null; + } + + public ISourceLocation getSourceLocation() { + //System.out.println("get context: " + this + " is " + sourceContext); + if (sourceContext == null) { + //System.err.println("no context: " + this); + return null; + } + return sourceContext.makeSourceLocation(this); + } + + public int getEnd() { + return end; + } + + public ISourceContext getSourceContext() { + return sourceContext; + } + + public int getStart() { + return start; + } + + public void setPosition(int sourceStart, int sourceEnd) { + this.start = sourceStart; + this.end = sourceEnd; + } + +} + diff --git a/weaver/src/org/aspectj/weaver/ResolvedPointcutDefinition.java b/weaver/src/org/aspectj/weaver/ResolvedPointcutDefinition.java new file mode 100644 index 000000000..8c38d6a99 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/ResolvedPointcutDefinition.java @@ -0,0 +1,91 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver; + +import java.io.*; + +import org.aspectj.weaver.patterns.Pointcut; + + +public class ResolvedPointcutDefinition extends ResolvedMember { + private Pointcut pointcut; + + public ResolvedPointcutDefinition( + TypeX declaringType, + int modifiers, + String name, + TypeX[] parameterTypes, + Pointcut pointcut) + { + super( + POINTCUT, + declaringType, + modifiers, + ResolvedTypeX.VOID, + name, + parameterTypes); + this.pointcut = pointcut; + //XXXpointcut.assertState(Pointcut.RESOLVED); + checkedExceptions = TypeX.NONE; + } + + // ---- + + public void write(DataOutputStream s) throws IOException { + getDeclaringType().write(s); + s.writeInt(getModifiers()); + s.writeUTF(getName()); + TypeX.write(getParameterTypes(), s); + pointcut.write(s); + } + + public static ResolvedPointcutDefinition read(DataInputStream s, ISourceContext context) throws IOException { + return new ResolvedPointcutDefinition( + TypeX.read(s), + s.readInt(), + s.readUTF(), + TypeX.readArray(s), + Pointcut.read(s, context)); + } + + public String toString() { + StringBuffer buf = new StringBuffer(); + buf.append("poincut "); + buf.append(getName()); + buf.append("("); + for (int i=0; i < getParameterTypes().length; i++) { + if (i > 0) buf.append(", "); + buf.append(getParameterTypes()[i].toString()); + } + buf.append("): "); + buf.append(pointcut); + + return buf.toString(); + } + + public Pointcut getPointcut() { + return pointcut; + } + + public boolean isAjSynthetic() { + return true; + } + + // for testing + public static final ResolvedPointcutDefinition DUMMY = + new ResolvedPointcutDefinition(TypeX.OBJECT, 0, "missing", + TypeX.NONE, Pointcut.makeMatchesNothing(Pointcut.RESOLVED)); + + +} diff --git a/weaver/src/org/aspectj/weaver/ResolvedTypeMunger.java b/weaver/src/org/aspectj/weaver/ResolvedTypeMunger.java new file mode 100644 index 000000000..27c0435b7 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/ResolvedTypeMunger.java @@ -0,0 +1,162 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver; + +import java.io.*; +import java.util.*; +import java.util.Set; + +import org.aspectj.util.TypeSafeEnum; + +/** This is an abstraction over method/field introduction. It might not have the chops + * to handle other inter-type declarations. This is the thing that is used on the + * eclipse side and serialized into a ConcreteTypeMunger. + */ +public abstract class ResolvedTypeMunger { + protected Kind kind; + protected ResolvedMember signature; + + private Set /* resolvedMembers */ superMethodsCalled = Collections.EMPTY_SET; + + public ResolvedTypeMunger(Kind kind, ResolvedMember signature) { + this.kind = kind; + this.signature = signature; + } + + // ---- + + // fromType is guaranteed to be a non-abstract aspect + public ConcreteTypeMunger concretize(World world, ResolvedTypeX aspectType) { + ConcreteTypeMunger munger = world.concreteTypeMunger(this, aspectType); + return munger; + } + + + public boolean matches(ResolvedTypeX matchType) { + ResolvedTypeX onType = matchType.getWorld().resolve(signature.getDeclaringType()); + if (matchType.equals(onType)) return true; + + if (onType.isInterface()) { + return matchType.isTopmostImplementor(onType); + } else { + return false; + } + } + + // ---- + + public String toString() { + return "ResolvedTypeMunger(" + getKind() + ", " + getSignature() +")"; + //.superMethodsCalled + ")"; + } + + // ---- + + public static ResolvedTypeMunger read(DataInputStream s, ISourceContext context) throws IOException { + Kind kind = Kind.read(s); + if (kind == Field) { + return NewFieldTypeMunger.readField(s, context); + } else if (kind == Method) { + return NewMethodTypeMunger.readMethod(s, context); + + } else if (kind == Constructor) { + return NewConstructorTypeMunger.readConstructor(s, context); + } else { + throw new RuntimeException("unimplemented"); + } + } + + protected static Set readSuperMethodsCalled(DataInputStream s) throws IOException { + Set ret = new HashSet(); + int n = s.readInt(); + for (int i=0; i < n; i++) { + ret.add(ResolvedMember.readResolvedMember(s, null)); + } + return ret; + } + + protected void writeSuperMethodsCalled(DataOutputStream s) throws IOException { + if (superMethodsCalled == null) { + s.writeInt(0); + return; + } + + List ret = new ArrayList(superMethodsCalled); + Collections.sort(ret); + int n = ret.size(); + s.writeInt(n); + for (Iterator i = ret.iterator(); i.hasNext(); ) { + ResolvedMember m = (ResolvedMember)i.next(); + m.write(s); + } + } + + + public abstract void write(DataOutputStream s) throws IOException; + + public Kind getKind() { + return kind; + } + + + + public static class Kind extends TypeSafeEnum { + private Kind(String name, int key) { + super(name, key); + } + + public static Kind read(DataInputStream s) throws IOException { + int key = s.readByte(); + switch(key) { + case 1: return Field; + case 2: return Method; + case 5: return Constructor; + } + throw new BCException("bad kind: " + key); + } + } + + // ---- fields + + public static final Kind Field = new Kind("Field", 1); + public static final Kind Method = new Kind("Method", 2); + public static final Kind Constructor = new Kind("Constructor", 5); + + // not serialized, only created during concretization of aspects + public static final Kind PerObjectInterface = new Kind("PerObjectInterface", 3); + public static final Kind PrivilegedAccess = new Kind("PrivilegedAccess", 4); + + public static final String SUPER_DISPATCH_NAME = "superDispatch"; + + + public void setSuperMethodsCalled(Set c) { + this.superMethodsCalled = c; + } + + public Set getSuperMethodsCalled() { + return superMethodsCalled; + } + + + public ResolvedMember getSignature() { + return signature; + } + + // ---- + + public ResolvedMember getMatchingSyntheticMember(Member member, ResolvedTypeX aspectType) { + return null; + } + +} diff --git a/weaver/src/org/aspectj/weaver/ResolvedTypeX.java b/weaver/src/org/aspectj/weaver/ResolvedTypeX.java new file mode 100644 index 000000000..b23dbe58f --- /dev/null +++ b/weaver/src/org/aspectj/weaver/ResolvedTypeX.java @@ -0,0 +1,975 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver; + +import java.lang.reflect.Modifier; +import java.util.*; + +import org.aspectj.bridge.MessageUtil; +import org.aspectj.weaver.bcel.BcelObjectType; +import org.aspectj.weaver.patterns.*; +import org.aspectj.weaver.patterns.PerClause; + +public abstract class ResolvedTypeX extends TypeX { + + protected World world; + + ResolvedTypeX(String signature, World world) { + super(signature); + this.world = world; + } + + // ---- things that don't require a world + + + /** returns Iterator<ResolvedTypeX> + */ + public final Iterator getDirectSupertypes() { + Iterator ifacesIterator = Iterators.array(getDeclaredInterfaces()); + ResolvedTypeX superclass = getSuperclass(); + if (superclass == null) { + return ifacesIterator; + } else { + return Iterators.snoc(ifacesIterator, superclass); + } + } + + public abstract ResolvedMember[] getDeclaredFields(); + public abstract ResolvedMember[] getDeclaredMethods(); + public abstract ResolvedTypeX[] getDeclaredInterfaces(); + public abstract ResolvedMember[] getDeclaredPointcuts(); + public abstract ResolvedTypeX getSuperclass(); + public abstract int getModifiers(); + + + public abstract boolean needsNoConversionFrom(TypeX other); + public abstract boolean isCoerceableFrom(TypeX other); + public abstract boolean isAssignableFrom(TypeX other); + + // ---- things that would require a world if I weren't resolved + public final Iterator getDirectSupertypes(World world) { + return getDirectSupertypes(); + } + + public final ResolvedMember[] getDeclaredFields(World world) { + return getDeclaredFields(); + } + public final ResolvedMember[] getDeclaredMethods(World world) { + return getDeclaredMethods(); + } + public final TypeX[] getDeclaredInterfaces(World world) { + return getDeclaredInterfaces(); + } + public final ResolvedMember[] getDeclaredPointcuts(World world) { + return getDeclaredPointcuts(); + } + + public final int getModifiers(World world) { + return getModifiers(); + } + public final TypeX getSuperclass(World world) { + return getSuperclass(); + } + + // conversions + public final boolean isAssignableFrom(TypeX other, World world) { + return isAssignableFrom(other); + } + public final boolean isCoerceableFrom(TypeX other, World world) { + return isCoerceableFrom(other); + } + public boolean needsNoConversionFrom(TypeX other, World world) { + return needsNoConversionFrom(other); + } + public final boolean isConvertableFrom(TypeX other) { + if (this.equals(OBJECT) || other.equals(OBJECT)) return true; + return this.isCoerceableFrom(other); + } + + // utilities + public ResolvedTypeX getResolvedComponentType() { + return null; + } + public ResolvedTypeX resolve(World world) { + return this; + } + public World getWorld() { + return world; + } + + // ---- things from object + + public final boolean equals(Object other) { + if (other instanceof ResolvedTypeX) { + return this == other; + } else { + return super.equals(other); + } + } + + // ---- difficult things + + /** + * returns an iterator through all of the fields of this type, in order + * for checking from JVM spec 2ed 5.4.3.2. This means that the order is + * + * <ul><li> fields from current class </li> + * <li> recur into direct superinterfaces </li> + * <li> recur into superclass </li> + * </ul> + * + * We keep a hashSet of interfaces that we've visited so we don't spiral + * out into 2^n land. + */ + public Iterator getFields() { + final Iterators.Filter dupFilter = Iterators.dupFilter(); + Iterators.Getter typeGetter = new Iterators.Getter() { + public Iterator get(Object o) { + return + dupFilter.filter( + ((ResolvedTypeX)o).getDirectSupertypes()); + } + }; + Iterators.Getter fieldGetter = new Iterators.Getter() { + public Iterator get(Object o) { + return Iterators.array(((ResolvedTypeX)o).getDeclaredFields()); + } + }; + return + Iterators.mapOver( + Iterators.recur(this, typeGetter), + fieldGetter); + } + + /** + * returns an iterator through all of the methods of this type, in order + * for checking from JVM spec 2ed 5.4.3.3. This means that the order is + * + * <ul><li> methods from current class </li> + * <li> recur into superclass, all the way up, not touching interfaces </li> + * <li> recur into all superinterfaces, in some unspecified order </li> + * </ul> + * + * We keep a hashSet of interfaces that we've visited so we don't spiral + * out into 2^n land. + */ + public Iterator getMethods() { + final Iterators.Filter dupFilter = Iterators.dupFilter(); + Iterators.Getter ifaceGetter = new Iterators.Getter() { + public Iterator get(Object o) { + return + dupFilter.filter( + Iterators.array(((ResolvedTypeX)o).getDeclaredInterfaces()) + ); + } + }; + Iterators.Getter methodGetter = new Iterators.Getter() { + public Iterator get(Object o) { + return Iterators.array(((ResolvedTypeX)o).getDeclaredMethods()); + } + }; + return + Iterators.mapOver( + Iterators.append( + new Iterator() { + ResolvedTypeX curr = ResolvedTypeX.this; + public boolean hasNext() { + return curr != null; + } + public Object next() { + ResolvedTypeX ret = curr; + curr = curr.getSuperclass(); + return ret; + } + public void remove() { + throw new UnsupportedOperationException(); + } + }, + Iterators.recur(this, ifaceGetter)), + methodGetter); + } + + /** + * described in JVM spec 2ed 5.4.3.2 + */ + public ResolvedMember lookupField(Member m) { + return lookupMember(m, getFields()); + } + + /** + * described in JVM spec 2ed 5.4.3.3 + */ + public ResolvedMember lookupMethod(Member m) { + return lookupMember(m, getMethods()); + } + + /** return null if not found */ + private ResolvedMember lookupMember(Member m, Iterator i) { + while (i.hasNext()) { + ResolvedMember f = (ResolvedMember) i.next(); + if (matches(f, m)) return f; + } + return null; //ResolvedMember.Missing; + //throw new BCException("can't find " + m); + } + + /** return null if not found */ + private ResolvedMember lookupMember(Member m, ResolvedMember[] a) { + for (int i = 0; i < a.length; i++) { + ResolvedMember f = a[i]; + if (matches(f, m)) return f; + } + return null; + } + + + public static boolean matches(Member m1, Member m2) { + return m1.getName().equals(m2.getName()) && m1.getSignature().equals(m2.getSignature()); + } + + + public static boolean conflictingSignature(Member m1, Member m2) { + if (m1 == null || m2 == null) return false; + + if (!m1.getName().equals(m2.getName())) { return false; } + if (m1.getKind() != m2.getKind()) { return false; } + + if (m1.getKind() == Member.FIELD) { + return m1.getDeclaringType().equals(m2.getDeclaringType()); + } + + TypeX[] p1 = m1.getParameterTypes(); + TypeX[] p2 = m2.getParameterTypes(); + int n = p1.length; + if (n != p2.length) return false; + + for (int i=0; i < n; i++) { + if (!p1[i].equals(p2[i])) return false; + } + return true; + } + + + /** + * returns an iterator through all of the pointcuts of this type, in order + * for checking from JVM spec 2ed 5.4.3.2 (as for fields). This means that the order is + * + * <ul><li> pointcuts from current class </li> + * <li> recur into direct superinterfaces </li> + * <li> recur into superclass </li> + * </ul> + * + * We keep a hashSet of interfaces that we've visited so we don't spiral + * out into 2^n land. + */ + private Iterator getPointcuts() { + final Iterators.Filter dupFilter = Iterators.dupFilter(); + // same order as fields + Iterators.Getter typeGetter = new Iterators.Getter() { + public Iterator get(Object o) { + return + dupFilter.filter( + ((ResolvedTypeX)o).getDirectSupertypes()); + } + }; + Iterators.Getter pointcutGetter = new Iterators.Getter() { + public Iterator get(Object o) { + //System.err.println("getting for " + o); + return Iterators.array(((ResolvedTypeX)o).getDeclaredPointcuts()); + } + }; + return + Iterators.mapOver( + Iterators.recur(this, typeGetter), + pointcutGetter); + } + + public ResolvedPointcutDefinition findPointcut(String name) { + //System.err.println("looking for pointcuts " + this); + for (Iterator i = getPointcuts(); i.hasNext(); ) { + ResolvedPointcutDefinition f = (ResolvedPointcutDefinition) i.next(); + //System.err.println(f); + if (name.equals(f.getName())) { + return f; + } + } + return null; // should we throw an exception here? + } + + + // all about collecting CrosscuttingMembers + + //??? collecting data-structure, shouldn't really be a field + public CrosscuttingMembers crosscuttingMembers; + +// +// private List extraConcreteShadowMungers = new ArrayList(); //XXX makes testing easier... +// public void addExtraConcreteShadowMunger(ShadowMunger munger) { +// munger.pointcut.assertState(Pointcut.CONCRETE); +// extraConcreteShadowMungers.add(munger); +// } +// public List getExtraConcreteShadowMungers() { +// return extraConcreteShadowMungers; +// } +// +// private List extraConcreteTypeMungers = new ArrayList(); //XXX makes testing easier...Collections.EMPTY_LIST; +// public void addExtraConcreteTypeMunger(ConcreteTypeMunger munger) { +// //munger.pointcut.assertState(Pointcut.CONCRETE); +// extraConcreteTypeMungers.add(munger); +// } +// public List getExtraConcreteTypeMungers() { +// return extraConcreteTypeMungers; +// } + + + public CrosscuttingMembers collectCrosscuttingMembers() { + crosscuttingMembers = new CrosscuttingMembers(this); + crosscuttingMembers.setPerClause(getPerClause()); + crosscuttingMembers.addShadowMungers(collectShadowMungers()); + crosscuttingMembers.addTypeMungers(getTypeMungers()); + crosscuttingMembers.addDeclares(collectDeclares()); + crosscuttingMembers.addPrivilegedAccesses(getPrivilegedAccesses()); + + //System.err.println("collected cc members: " + this + ", " + collectDeclares()); + return crosscuttingMembers; + } + + private final Collection collectDeclares() { + if (! this.isAspect() ) return Collections.EMPTY_LIST; + + ArrayList ret = new ArrayList(); + //if (this.isAbstract()) { + for (Iterator i = getDeclares().iterator(); i.hasNext();) { + Declare dec = (Declare) i.next(); + if (!dec.isAdviceLike()) ret.add(dec); + } + if (!this.isAbstract()) { + //ret.addAll(getDeclares()); + final Iterators.Filter dupFilter = Iterators.dupFilter(); + Iterators.Getter typeGetter = new Iterators.Getter() { + public Iterator get(Object o) { + return + dupFilter.filter( + ((ResolvedTypeX)o).getDirectSupertypes()); + } + }; + Iterator typeIterator = Iterators.recur(this, typeGetter); + + while (typeIterator.hasNext()) { + ResolvedTypeX ty = (ResolvedTypeX) typeIterator.next(); + //System.out.println("super: " + ty + ", " + ); + for (Iterator i = ty.getDeclares().iterator(); i.hasNext();) { + Declare dec = (Declare) i.next(); + if (dec.isAdviceLike()) ret.add(dec); + } + } + } + + return ret; + } + + + + + private final Collection collectShadowMungers() { + if (! this.isAspect() || this.isAbstract()) return Collections.EMPTY_LIST; + + ArrayList acc = new ArrayList(); + final Iterators.Filter dupFilter = Iterators.dupFilter(); + Iterators.Getter typeGetter = new Iterators.Getter() { + public Iterator get(Object o) { + return + dupFilter.filter( + ((ResolvedTypeX)o).getDirectSupertypes()); + } + }; + Iterator typeIterator = Iterators.recur(this, typeGetter); + + while (typeIterator.hasNext()) { + ResolvedTypeX ty = (ResolvedTypeX) typeIterator.next(); + acc.addAll(ty.getDeclaredShadowMungers()); + } + + return acc; + } + + public PerClause getPerClause() { return null; } + protected Collection getDeclares() { + return Collections.EMPTY_LIST; + } + protected Collection getTypeMungers() { return Collections.EMPTY_LIST; } + + protected Collection getPrivilegedAccesses() { return Collections.EMPTY_LIST; } + + + + // ---- useful things + + public final boolean isInterface() { + return Modifier.isInterface(getModifiers()); + } + + public final boolean isAbstract() { + return Modifier.isAbstract(getModifiers()); + } + + public boolean isClass() { + return false; + } + + public boolean isAspect() { + return false; + } + + + public final boolean isFinal() { + return Modifier.isFinal(getModifiers()); + } + + public Collection getDeclaredAdvice() { + List l = new ArrayList(); + ResolvedMember[] methods = getDeclaredMethods(); + for (int i=0, len = methods.length; i < len; i++) { + ShadowMunger munger = methods[i].getAssociatedShadowMunger(); + if (munger != null) l.add(munger); + } + return l; + } + + private List shadowMungers = new ArrayList(0); + + public Collection getDeclaredShadowMungers() { + Collection c = getDeclaredAdvice(); + c.addAll(shadowMungers); + return c; + } + + + public void addShadowMunger(ShadowMunger munger) { + shadowMungers.add(munger); + } + + + // ---- only for testing! + + + public ResolvedMember[] getDeclaredJavaFields() { + return filterInJavaVisible(getDeclaredFields()); + } + public ResolvedMember[] getDeclaredJavaMethods() { + return filterInJavaVisible(getDeclaredMethods()); + } + public ShadowMunger[] getDeclaredShadowMungersArray() { + List l = (List) getDeclaredShadowMungers(); + return (ShadowMunger[]) l.toArray(new ShadowMunger[l.size()]); + } + private ResolvedMember[] filterInJavaVisible(ResolvedMember[] ms) { + List l = new ArrayList(); + for (int i=0, len = ms.length; i < len; i++) { + if (! ms[i].isAjSynthetic() && ms[i].getAssociatedShadowMunger() == null) { + l.add(ms[i]); + } + } + return (ResolvedMember[]) l.toArray(new ResolvedMember[l.size()]); + } + + public abstract ISourceContext getSourceContext(); + + + // ---- fields + + public static final ResolvedTypeX[] NONE = new ResolvedTypeX[0]; + + public static final Primitive BYTE = new Primitive("B", 1, 0); + public static final Primitive CHAR = new Primitive("C", 1, 1); + public static final Primitive DOUBLE = new Primitive("D", 2, 2); + public static final Primitive FLOAT = new Primitive("F", 1, 3); + public static final Primitive INT = new Primitive("I", 1, 4); + public static final Primitive LONG = new Primitive("J", 2, 5); + public static final Primitive SHORT = new Primitive("S", 1, 6); + public static final Primitive VOID = new Primitive("V", 0, 8); + public static final Primitive BOOLEAN = new Primitive("Z", 1, 7); + public static final Missing MISSING = new Missing(); + + // ---- types + + public static abstract class Name extends ResolvedTypeX { + protected ISourceContext sourceContext; + + public Name(String signature, World world) { + super(signature, world); + } + + public final boolean isClass() { + return !isAspect() && !isInterface(); + } + + public abstract boolean isAspect(); + public final boolean isAssignableFrom(TypeX o) { + if (o.isPrimitive()) return false; + ResolvedTypeX other = o.resolve(world); + + return isAssignableFrom(other); + } + private boolean isAssignableFrom(ResolvedTypeX other) { + if (this == other) return true; + for(Iterator i = other.getDirectSupertypes(); i.hasNext(); ) { + if (this.isAssignableFrom((ResolvedTypeX) i.next())) return true; + } + return false; + } + public final boolean isCoerceableFrom(TypeX o) { + ResolvedTypeX other = o.resolve(world); + + if (this.isAssignableFrom(other) || other.isAssignableFrom(this)) { + return true; + } + if (!this.isInterface() && !other.isInterface()) { + return false; + } + if (this.isFinal() || other.isFinal()) { + return false; + } + // ??? needs to be Methods, not just declared methods? JLS 5.5 unclear + ResolvedMember[] a = getDeclaredMethods(); + ResolvedMember[] b = ((Name)other).getDeclaredMethods(); + for (int ai = 0, alen = a.length; ai < alen; ai++) { + for (int bi = 0, blen = b.length; bi < blen; bi++) { + if (! b[bi].isCompatibleWith(a[ai])) return false; + } + } + return true; + } + public final boolean needsNoConversionFrom(TypeX o) { + return isAssignableFrom(o); + } + + public ResolvedMember addPerSingletonField() { + throw new RuntimeException("unimplemented"); + } + + public ISourceContext getSourceContext() { + return sourceContext; + } + + } + + static class Array extends ResolvedTypeX { + ResolvedTypeX componentType; + Array(String s, World world, ResolvedTypeX componentType) { + super(s, world); + this.componentType = componentType; + } + public final ResolvedMember[] getDeclaredFields() { + return ResolvedMember.NONE; + } + public final ResolvedMember[] getDeclaredMethods() { + // ??? should this return clone? Probably not... + return ResolvedMember.NONE; + } + public final ResolvedTypeX[] getDeclaredInterfaces() { + return + new ResolvedTypeX[] { + world.resolve(CLONEABLE), + world.resolve(SERIALIZABLE) + }; + } + public final ResolvedMember[] getDeclaredPointcuts() { + return ResolvedMember.NONE; + } + + public final ResolvedTypeX getSuperclass() { + return world.resolve(OBJECT); + } + public final boolean isAssignableFrom(TypeX o) { + if (! o.isArray()) return false; + if (o.getComponentType().isPrimitive()) { + return o.equals(this); + } else { + return getComponentType().isAssignableFrom(o.getComponentType(), world); + } + } + public final boolean isCoerceableFrom(TypeX o) { + if (o.equals(TypeX.OBJECT) || + o.equals(TypeX.SERIALIZABLE) || + o.equals(TypeX.CLONEABLE)) { + return true; + } + if (! o.isArray()) return false; + if (o.getComponentType().isPrimitive()) { + return o.equals(this); + } else { + return getComponentType().isCoerceableFrom(o.getComponentType(), world); + } + } + public final boolean needsNoConversionFrom(TypeX o) { + return isAssignableFrom(o); + } + public final int getModifiers() { + int mask = Modifier.PUBLIC | Modifier.PRIVATE | Modifier.PROTECTED; + return (componentType.getModifiers() & mask) | Modifier.FINAL; + } + public TypeX getComponentType() { + return componentType; + } + public ResolvedTypeX getResolvedComponentType() { + return componentType; + } + public ISourceContext getSourceContext() { + return getResolvedComponentType().getSourceContext(); + } + } + + static class Primitive extends ResolvedTypeX { + private int size; + private int index; + Primitive(String signature, int size, int index) { + super(signature, null); + this.size = size; + this.index = index; + } + public final int getSize() { + return size; + } + public final int getModifiers() { + return Modifier.PUBLIC | Modifier.FINAL; + } + public final boolean isPrimitive() { + return true; + } + public final boolean isAssignableFrom(TypeX other) { + if (! other.isPrimitive()) return false; + return assignTable[((Primitive)other).index][index]; + } + public final boolean isCoerceableFrom(TypeX other) { + if (this == other) return true; + if (! other.isPrimitive()) return false; + if (index > 6 || ((Primitive)other).index > 6) return false; + return true; + } + public final boolean needsNoConversionFrom(TypeX other) { + if (! other.isPrimitive()) return false; + return noConvertTable[((Primitive)other).index][index]; + } + private static final boolean[][] assignTable = + {// to: B C D F I J S V Z from + { true , true , true , true , true , true , true , false, false }, // B + { false, true , true , true , true , true , false, false, false }, // C + { false, false, true , false, false, false, false, false, false }, // D + { false, false, true , true , false, false, false, false, false }, // F + { false, false, true , true , true , true , false, false, false }, // I + { false, false, true , true , false, true , false, false, false }, // J + { false, false, true , true , true , true , true , false, false }, // S + { false, false, false, false, false, false, false, true , false }, // V + { false, false, false, false, false, false, false, false, true }, // Z + }; + private static final boolean[][] noConvertTable = + {// to: B C D F I J S V Z from + { true , true , false, false, true , false, true , false, false }, // B + { false, true , false, false, true , false, false, false, false }, // C + { false, false, true , false, false, false, false, false, false }, // D + { false, false, false, true , false, false, false, false, false }, // F + { false, false, false, false, true , false, false, false, false }, // I + { false, false, false, false, false, true , false, false, false }, // J + { false, false, false, false, true , false, true , false, false }, // S + { false, false, false, false, false, false, false, true , false }, // V + { false, false, false, false, false, false, false, false, true }, // Z + }; + + // ---- + + public final ResolvedMember[] getDeclaredFields() { + return ResolvedMember.NONE; + } + public final ResolvedMember[] getDeclaredMethods() { + return ResolvedMember.NONE; + } + public final ResolvedTypeX[] getDeclaredInterfaces() { + return ResolvedTypeX.NONE; + } + public final ResolvedMember[] getDeclaredPointcuts() { + return ResolvedMember.NONE; + } + + public final ResolvedTypeX getSuperclass() { + return null; + } + + public ISourceContext getSourceContext() { + return null; + } + + } + + static class Missing extends ResolvedTypeX { + Missing() { + super(MISSING_NAME, null); + } +// public final String toString() { +// return "<missing>"; +// } + public final String getName() { + return MISSING_NAME; + } + public final ResolvedMember[] getDeclaredFields() { + return ResolvedMember.NONE; + } + public final ResolvedMember[] getDeclaredMethods() { + return ResolvedMember.NONE; + } + public final ResolvedTypeX[] getDeclaredInterfaces() { + return ResolvedTypeX.NONE; + } + + public final ResolvedMember[] getDeclaredPointcuts() { + return ResolvedMember.NONE; + } + public final ResolvedTypeX getSuperclass() { + return null; + } + public final int getModifiers() { + return 0; + } + public final boolean isAssignableFrom(TypeX other) { + return false; + } + public final boolean isCoerceableFrom(TypeX other) { + return false; + } + public boolean needsNoConversionFrom(TypeX other) { + return false; + } + public ISourceContext getSourceContext() { + return null; + } + + } + + /** return null if not found */ + public ResolvedMember lookupMemberNoSupers(Member member) { + if (member.getKind() == Member.FIELD) { + return lookupMember(member, getDeclaredFields()); + } else { + // assert member.getKind() == Member.METHOD || member.getKind() == Member.CONSTRUCTOR + return lookupMember(member, getDeclaredMethods()); + } + } + + protected List interTypeMungers = new ArrayList(0); + + public List getInterTypeMungers() { + return interTypeMungers; + } + + private static ResolvedTypeX getOutermostType(ResolvedTypeX t) { + TypeX dec = t.getDeclaringType(); + if (dec == null) return t; + return getOutermostType(dec.resolve(t.getWorld())); + } + + public static boolean isVisible(int modifiers, ResolvedTypeX targetType, ResolvedTypeX fromType) { + //System.err.println("mod: " + modifiers + ", " + targetType + " and " + fromType); + + if (Modifier.isPublic(modifiers)) { + return true; + } else if (Modifier.isPrivate(modifiers)) { + return getOutermostType(targetType).equals(getOutermostType(fromType)); + } else if (Modifier.isProtected(modifiers)) { + return samePackage(targetType, fromType) || targetType.isAssignableFrom(fromType); + } else { // package-visible + return samePackage(targetType, fromType); + } + } + + private static boolean samePackage( + ResolvedTypeX targetType, + ResolvedTypeX fromType) + { + String p1 = targetType.getPackageName(); + String p2 = fromType.getPackageName(); + if (p1 == null) return p2 == null; + if (p2 == null) return false; + return p1.equals(p2); + } + + public void addInterTypeMunger(ConcreteTypeMunger munger) { + ResolvedMember sig = munger.getSignature(); + if (sig == null) { + interTypeMungers.add(munger); + return; + } + + //System.err.println("add: " + munger + " to " + this.getClassName() + " with " + interTypeMungers); + if (sig.getKind() == Member.METHOD) { + if (!compareToExistingMembers(munger, getMethods())) return; + if (this.isInterface()) { + if (!compareToExistingMembers(munger, + Arrays.asList(world.resolve(OBJECT).getDeclaredMethods()).iterator())) return; + } + } else if (sig.getKind() == Member.FIELD) { + if (!compareToExistingMembers(munger, Arrays.asList(getDeclaredFields()).iterator())) return; + } else { + if (!compareToExistingMembers(munger, Arrays.asList(getDeclaredMethods()).iterator())) return; + } + + + // now compare to existingMungers + for (Iterator i = interTypeMungers.iterator(); i.hasNext(); ) { + ConcreteTypeMunger existingMunger = (ConcreteTypeMunger)i.next(); + if (conflictingSignature(existingMunger.getSignature(), munger.getSignature())) { + //System.err.println("match " + munger + " with " + existingMunger); + if (isVisible(munger.getSignature().getModifiers(), + munger.getAspectType(), existingMunger.getAspectType())) + { + //System.err.println(" is visible"); + int c = compareMemberPrecedence(sig, existingMunger.getSignature()); + if (c == 0) { + c = getWorld().comparePrecedence(munger.getAspectType(), existingMunger.getAspectType()); + } + //System.err.println(" compare: " + c); + if (c < 0) { + // the existing munger dominates the new munger + return; + } else if (c > 0) { + // the new munger dominates the existing one + i.remove(); + break; + } else { + interTypeConflictError(munger, existingMunger); + interTypeConflictError(existingMunger, munger); + return; + } + } + } + } + //System.err.println("adding: " + munger + " to " + this); + interTypeMungers.add(munger); + } + + + //??? returning too soon + private boolean compareToExistingMembers(ConcreteTypeMunger munger, Iterator existingMembers) { + ResolvedMember sig = munger.getSignature(); + while (existingMembers.hasNext()) { + ResolvedMember existingMember = (ResolvedMember)existingMembers.next(); + + if (conflictingSignature(existingMember, munger.getSignature())) { + //System.err.println("conflict: " + existingMember + " with " + munger); + if (isVisible(existingMember.getModifiers(), this, munger.getAspectType())) { + int c = compareMemberPrecedence(sig, existingMember); + //System.err.println(" c: " + c); + if (c < 0) return false; + else if (c > 0) { + //interTypeMungers.add(munger); + //??? might need list of these overridden abstracts + continue; + } else { + if (this instanceof BcelObjectType) return false; //XXX ignores separate comp + getWorld().getMessageHandler().handleMessage( + MessageUtil.error("inter-type declaration from " + munger.getAspectType() + + " conflicts with existing member: " + existingMember, + munger.getSourceLocation()) + ); + } + } else { + //interTypeMungers.add(munger); + } + //return; + } + } + return true; + } + + + + + private int compareMemberPrecedence(ResolvedMember m1, ResolvedMember m2) { + if (!m1.getReturnType().equals(m2.getReturnType())) return 0; + + if (Modifier.isAbstract(m1.getModifiers())) return -1; + if (Modifier.isAbstract(m2.getModifiers())) return +1; + + if (m1.getDeclaringType().equals(m2.getDeclaringType())) return 0; + + ResolvedTypeX t1 = m1.getDeclaringType().resolve(world); + ResolvedTypeX t2 = m2.getDeclaringType().resolve(world); + if (t1.isAssignableFrom(t2)) { + checkVisibility(m1.getModifiers(), m2.getModifiers()); //XXX needs to be before abstract + return -1; + } + if (t2.isAssignableFrom(t1)) { + checkVisibility(m2.getModifiers(), m1.getModifiers()); + return +1; + } + return 0; + } + + + private void checkVisibility(int parentMods, int childMods) { + //childMods must be equal to or greater than parentMods visibility + + } + + + + private void interTypeConflictError( + ConcreteTypeMunger m1, + ConcreteTypeMunger m2) + { + //XXX this works only if we ignore separate compilation issues + if (this instanceof BcelObjectType) return; + + //System.err.println("conflict at " + m2.getSourceLocation()); + getWorld().getMessageHandler().handleMessage(MessageUtil.error( + "intertype declaration from " + + m1.getAspectType().getClassName() + + " conflicts with intertype declaration: " + + m2.getSignature() + + " from " + + m2.getAspectType().getClassName(), + m2.getSourceLocation())); + } + + + public ResolvedMember lookupSyntheticMember(Member member) { + //??? horribly inefficient + //for (Iterator i = + //System.err.println("lookup " + member + " in " + interTypeMungers); + for (Iterator i = interTypeMungers.iterator(); i.hasNext(); ) { + ConcreteTypeMunger m = (ConcreteTypeMunger)i.next(); + ResolvedMember ret = m.getMatchingSyntheticMember(member); + if (ret != null) { + //System.err.println(" found: " + ret); + return ret; + } + } + return null; + } + + public void clearInterTypeMungers() { + interTypeMungers = new ArrayList(); + } + + + public boolean isTopmostImplementor(ResolvedTypeX interfaceType) { + if (isInterface()) return false; + if (!interfaceType.isAssignableFrom(this)) return false; + // check that I'm truly the topmost implementor + if (interfaceType.isAssignableFrom(this.getSuperclass())) { + return false; + } + return true; + } + +} diff --git a/weaver/src/org/aspectj/weaver/Shadow.java b/weaver/src/org/aspectj/weaver/Shadow.java new file mode 100644 index 000000000..22f86deb9 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/Shadow.java @@ -0,0 +1,233 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver; + +import java.io.*; +import java.lang.reflect.Modifier; +import java.util.*; + +import org.apache.bcel.Constants; +import org.apache.bcel.classfile.Field; +import org.aspectj.weaver.ast.Var; +import org.aspectj.weaver.patterns.IScope; +import org.aspectj.bridge.*; +import org.aspectj.bridge.SourceLocation; +import org.aspectj.lang.JoinPoint; +import org.aspectj.util.*; + +/* + * The superclass of anything representing a the shadow of a join point. A shadow represents + * some bit of code, and encompasses both entry and exit from that code. All shadows have a kind + * and a signature. + */ + +public abstract class Shadow { + private final Kind kind; + private final Member signature; + protected List mungers = new ArrayList(1); + + // ---- + + protected Shadow(Kind kind, Member signature) { + this.kind = kind; + this.signature = signature; + } + + // ---- + + public abstract World getIWorld(); + + public final boolean hasTarget() { + return !(getSignature().isStatic() || (getKind() == ConstructorCall) + || (getKind() == PreInitialization)); + } + + public final TypeX getTargetType() { + if (!hasTarget()) return ResolvedTypeX.MISSING; + return getSignature().getDeclaringType(); + } + public TypeX getArgType(int arg) { + if (getKind() == FieldSet) return getSignature().getReturnType(); + return getSignature().getParameterTypes()[arg]; + } + + public int getArgCount() { + if (getKind() == FieldSet) return 1; + return getSignature() + .getParameterTypes().length; + } + + public abstract boolean hasThis(); + public abstract TypeX getThisType(); + public abstract TypeX getEnclosingType(); + public abstract Var getThisVar(); + public abstract Var getTargetVar(); + public abstract Var getArgVar(int i); + + public abstract Var getThisJoinPointVar(); + public abstract Var getThisJoinPointStaticPartVar(); + public abstract Var getThisEnclosingJoinPointStaticPartVar(); + + public abstract Member getEnclosingCodeSignature(); + + + /** returns the kind of shadow this is, representing what happens under this shadow + */ + public Kind getKind() { + return kind; + } + + /** returns the signature of the thing under this shadow + */ + public Member getSignature() { + return signature; + } + + + public TypeX getReturnType() { + if (kind == ConstructorCall) return getSignature().getDeclaringType(); + return getSignature().getReturnType(); + } + + + /** + * These names are the ones that will be returned by thisJoinPoint.getKind() + * Those need to be documented somewhere + */ + public static final Kind MethodCall = new Kind(JoinPoint.METHOD_CALL, 1, true); + public static final Kind ConstructorCall = new Kind(JoinPoint.CONSTRUCTOR_CALL, 2, true); + public static final Kind MethodExecution = new Kind(JoinPoint.METHOD_EXECUTION, 3, false); + public static final Kind ConstructorExecution = new Kind(JoinPoint.CONSTRUCTOR_EXECUTION, 4, false); + public static final Kind FieldGet = new Kind(JoinPoint.FIELD_GET, 5, true); + public static final Kind FieldSet = new Kind(JoinPoint.FIELD_SET, 6, true); + public static final Kind StaticInitialization = new Kind(JoinPoint.STATICINITIALIZATION, 7, false); + public static final Kind PreInitialization = new Kind(JoinPoint.PREINTIALIZATION, 8, false); + public static final Kind AdviceExecution = new Kind(JoinPoint.ADVICE_EXECUTION, 9, false); + public static final Kind Initialization = new Kind(JoinPoint.INITIALIZATION, 10, false); + public static final Kind ExceptionHandler = new Kind(JoinPoint.EXCEPTION_HANDLER, 11, true); + + + /** A type-safe enum representing the kind of shadows + */ + public static final class Kind extends TypeSafeEnum { + private boolean argsOnStack; + + public Kind(String name, int key, boolean argsOnStack) { + super(name, key); + this.argsOnStack = argsOnStack; + } + + public String toLegalJavaIdentifier() { + return getName().replace('-', '_'); + } + + public boolean argsOnStack() { + return argsOnStack; + } + + // !!! this is false for handlers! + public boolean allowsExtraction() { + return true; + } + + public String getSimpleName() { + int dash = getName().lastIndexOf('-'); + if (dash == -1) return getName(); + else return getName().substring(dash+1); + } + + public static Kind read(DataInputStream s) throws IOException { + int key = s.readByte(); + switch(key) { + case 1: return MethodCall; + case 2: return ConstructorCall; + case 3: return MethodExecution; + case 4: return ConstructorExecution; + case 5: return FieldGet; + case 6: return FieldSet; + case 7: return StaticInitialization; + case 8: return PreInitialization; + case 9: return AdviceExecution; + case 10: return Initialization; + case 11: return ExceptionHandler; + } + throw new BCException("unknown kind: " + key); + } + } + + public void addMunger(ShadowMunger munger) { + this.mungers.add(munger); + } + + public final void implement() { + sortMungers(); + if (mungers == null) return; + prepareForMungers(); + implementMungers(); + } + + private void sortMungers() { + List sorted = PartialOrder.sort(mungers); + if (sorted == null) { + // this means that we have circular dependencies + for (Iterator i = mungers.iterator(); i.hasNext(); ) { + ShadowMunger m = (ShadowMunger)i.next(); + getIWorld().getMessageHandler().handleMessage( + MessageUtil.error("circular dependency at " + this, m.getSourceLocation())); + } + } + mungers = sorted; + } + + /** Prepare the shadow for implementation. After this is done, the shadow + * should be in such a position that each munger simply needs to be implemented. + */ + protected void prepareForMungers() { + throw new RuntimeException("Generic shadows cannot be prepared"); + } + + /** Actually implement the (non-empty) mungers associated with this shadow */ + private void implementMungers() { + World world = getIWorld(); + for (Iterator iter = mungers.iterator(); iter.hasNext();) { + ShadowMunger munger = (ShadowMunger) iter.next(); + munger.implementOn(this); + if (world.getModel() != null) { + System.err.println("munger: " + munger + " on " + this); + AsmAdaptor.noteMunger(world.getModel(), this, munger); + } + } + } + + public String makeReflectiveFactoryString() { + return null; //XXX + } + + public abstract SourceLocation getSourceLocation(); + + // ---- utility + + public String toString() { + return getKind() + "(" + getSignature() + ")"; // + getSourceLines(); + } + + + + + + // ---- type access methods + + + +} diff --git a/weaver/src/org/aspectj/weaver/ShadowMunger.java b/weaver/src/org/aspectj/weaver/ShadowMunger.java new file mode 100644 index 000000000..4cce6743b --- /dev/null +++ b/weaver/src/org/aspectj/weaver/ShadowMunger.java @@ -0,0 +1,87 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver; + +import org.aspectj.bridge.ISourceLocation; +import org.aspectj.util.PartialOrder; +import org.aspectj.weaver.patterns.*; +import org.aspectj.weaver.patterns.PerClause; + +/** + * For every shadow munger, nothing can be done with it until it is concretized. Then... + * + * (Then we call fast match.) + * + * For every shadow munger, for every shadow, + * first match is called, + * then (if match returned true) the shadow munger is specialized for the shadow, + * which may modify state. + * Then implement is called. + */ + +public abstract class ShadowMunger implements PartialOrder.PartialComparable, IHasPosition { + protected Pointcut pointcut; + + // these three fields hold the source location of this munger + protected int start, end; + protected ISourceContext sourceContext; + + + public ShadowMunger(Pointcut pointcut, int start, int end, ISourceContext sourceContext) { + this.pointcut = pointcut; + this.start = start; + this.end = end; + this.sourceContext = sourceContext; + } + + public abstract ShadowMunger concretize(ResolvedTypeX fromType, World world, PerClause clause); + + public abstract void specializeOn(Shadow shadow); + public abstract void implementOn(Shadow shadow); + + /** + * All overriding methods should call super + */ + public boolean match(Shadow shadow, World world) { + return pointcut.match(shadow).maybeTrue(); + } + + public int fallbackCompareTo(Object other) { + return toString().compareTo(toString()); + } + + public int getEnd() { + return end; + } + + public int getStart() { + return start; + } + + public ISourceLocation getSourceLocation() { + //System.out.println("get context: " + this + " is " + sourceContext); + if (sourceContext == null) { + //System.err.println("no context: " + this); + return null; + } + return sourceContext.makeSourceLocation(this); + } + + // ---- fields + + public static final ShadowMunger[] NONE = new ShadowMunger[0]; + + + +} diff --git a/weaver/src/org/aspectj/weaver/StaticJoinPointFactory.java b/weaver/src/org/aspectj/weaver/StaticJoinPointFactory.java new file mode 100644 index 000000000..d0f5ab4f1 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/StaticJoinPointFactory.java @@ -0,0 +1,58 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver; + +import java.io.StringReader; +import java.util.*; + +public class StaticJoinPointFactory { +// int usedKeys; +// +// List/*String*/ strings = new ArrayList(); +// Map/*String,Integer*/ keysForStrings = new HashMap(); +// +// public StaticJoinPointFactory() { +// super(); +// } +// +// static char[] encoding = new char[] { +// '0', '1', '2', '3', '4', '5', '6', '7', '8', '9',//10 +// 'a', 'b', 'z', //36 +// 'A', 'B', 'Z', //62 +// '%', '$', //64 +// }; +// +// static int TWO_WORDS = 64*64-1; +// static int WORD_MASK = 63; +// +// public void write(String s, StringBuffer result) { +// int i = getIndex(s); +// encode(i, result); +// } +// +// void encode(int i, StringBuffer result) { +// if (i > TWO_WORDS) { +// throw new RuntimeException("unimplemented"); +// } else { +// result.append( encoding[(i >> 6) & WORD_MASK] ); +// result.append( encoding[i & WORD_MASK] ); +// } +// } +// +// public String read(StringReader reader) { +// int i = reader.read(); +// +// } + +} diff --git a/weaver/src/org/aspectj/weaver/TypeX.java b/weaver/src/org/aspectj/weaver/TypeX.java new file mode 100644 index 000000000..94c86201c --- /dev/null +++ b/weaver/src/org/aspectj/weaver/TypeX.java @@ -0,0 +1,683 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver; + +import java.io.*; +import java.lang.reflect.Modifier; +import java.util.Iterator; + +import org.aspectj.weaver.patterns.PerClause; + +public class TypeX { + /** + * This is the bytecode string representation of this Type + */ + protected String signature; + + /** + * @param signature the bytecode string representation of this Type + */ + protected TypeX(String signature) { + super(); + this.signature = signature; + } + + // ---- Things we can do without a world + + /** + * This is the size of this type as used in JVM. + */ + public int getSize() { + return 1; + } + + /** + * Equality is checked based on the underlying signature. + * {@link ResolvedType} objects' equals is by reference. + */ + public boolean equals(Object other) { + if (! (other instanceof TypeX)) return false; + return signature.equals(((TypeX) other).signature); + } + + /** + * Equality is checked based on the underlying signature, so the hash code + * of a particular type is the hash code of its signature string. + */ + public final int hashCode() { + return signature.hashCode(); + } + + /** + * Constructs a TypeX for a java language type name. For example: + * + * <blockquote><pre> + * TypeX.forName("java.lang.Thread[]") + * TypeX.forName("int") + * </pre></blockquote> + * + * Types may equivalently be produced by this or by {@link #forSignature(String)}. + * + * <blockquote><pre> + * TypeX.forName("java.lang.Thread[]").equals(Type.forSignature("[Ljava/lang/Thread;") + * TypeX.forName("int").equals(Type.forSignature("I")) + * </pre></blockquote> + * + * @param name the java language type name in question. + * @return a type object representing that java language type. + */ + public static TypeX forName(String name) { + return forSignature(nameToSignature(name)); + } + + /** Constructs a TypeX for each java language type name in an incoming array. + * + * @param names an array of java language type names. + * @return an array of TypeX objects. + * @see #forName(String) + */ + public static TypeX[] forNames(String[] names) { + TypeX[] ret = new TypeX[names.length]; + for (int i = 0, len = names.length; i < len; i++) { + ret[i] = TypeX.forName(names[i]); + } + return ret; + } + + /** + * Creates a new type array with a fresh type appended to the end. + * + * @param types the left hand side of the new array + * @param end the right hand side of the new array + */ + public static TypeX[] add(TypeX[] types, TypeX end) { + int len = types.length; + TypeX[] ret = new TypeX[len + 1]; + System.arraycopy(types, 0, ret, 0, len); + ret[len] = end; + return ret; + } + + /** + * Creates a new type array with a fresh type inserted at the beginning. + * + * + * @param start the left hand side of the new array + * @param types the right hand side of the new array + */ + public static TypeX[] insert(TypeX start, TypeX[] types) { + int len = types.length; + TypeX[] ret = new TypeX[len + 1]; + ret[0] = start; + System.arraycopy(types, 0, ret, 1, len); + return ret; + } + + /** + * Constructs a Type for a JVM bytecode signature string. For example: + * + * <blockquote><pre> + * TypeX.forSignature("[Ljava/lang/Thread;") + * TypeX.forSignature("I"); + * </pre></blockquote> + * + * Types may equivalently be produced by this or by {@link #forName(String)}. + * + * <blockquote><pre> + * TypeX.forName("java.lang.Thread[]").equals(Type.forSignature("[Ljava/lang/Thread;") + * TypeX.forName("int").equals(Type.forSignature("I")) + * </pre></blockquote> + * + * @param signature the JVM bytecode signature string for the desired type. + * @return a type object represnting that JVM bytecode signature. + */ + public static TypeX forSignature(String signature) { + switch (signature.charAt(0)) { + case 'B': return ResolvedTypeX.BYTE; + case 'C': return ResolvedTypeX.CHAR; + case 'D': return ResolvedTypeX.DOUBLE; + case 'F': return ResolvedTypeX.FLOAT; + case 'I': return ResolvedTypeX.INT; + case 'J': return ResolvedTypeX.LONG; + case 'L': + return new TypeX(signature); + case 'S': return ResolvedTypeX.SHORT; + case 'V': return ResolvedTypeX.VOID; + case 'Z': return ResolvedTypeX.BOOLEAN; + case '[': + return new TypeX(signature); + default: + throw new BCException("Bad type signature " + signature); + } + } + + /** Constructs a TypeX for each JVM bytecode type signature in an incoming array. + * + * @param names an array of JVM bytecode type signatures + * @return an array of TypeX objects. + * @see #forSignature(String) + */ + public static TypeX[] forSignatures(String[] sigs) { + TypeX[] ret = new TypeX[sigs.length]; + for (int i = 0, len = sigs.length; i < len; i++) { + ret[i] = TypeX.forSignature(sigs[i]); + } + return ret; + } + + /** + * Returns the name of this type in java language form. For all + * TypeX t: + * + * <blockquote><pre> + * TypeX.forName(t.getName()).equals(t) + * </pre></blockquote> + * + * and for all String s where s is a lexically valid java language typename: + * + * <blockquote><pre> + * TypeX.forName(s).getName().equals(s) + * </pre></blockquote> + * + * This produces a more esthetically pleasing string than + * {@link java.lang.Class#getName()}. + * + * @return the java language name of this type. + */ + public String getName() { + return signatureToName(signature); + } + + /** + * Returns an array of strings representing the java langauge names of + * an array of types. + * + * @param types an array of TypeX objects + * @return an array of Strings fo the java language names of types. + * @see #getName() + */ + public static String[] getNames(TypeX[] types) { + String[] ret = new String[types.length]; + for (int i = 0, len = types.length; i < len; i++) { + ret[i] = types[i].getName(); + } + return ret; + } + + /** + * Returns the name of this type in JVM signature form. For all + * TypeX t: + * + * <blockquote><pre> + * TypeX.forSignature(t.getSignature()).equals(t) + * </pre></blockquote> + * + * and for all String s where s is a lexically valid JVM type signature string: + * + * <blockquote><pre> + * TypeX.forSignature(s).getSignature().equals(s) + * </pre></blockquote> + * + * @return the java JVM signature string for this type. + */ + public String getSignature() { + return signature; + } + + /** + * Determins if this represents an array type. + * + * @return true iff this represents an array type. + */ + public final boolean isArray() { + return signature.startsWith("["); + } + + /** + * Returns a TypeX object representing the declaring type of this type, or + * null if this type does not represent a non-package-level-type. + * + * <strong>Warning</strong>: This is guaranteed to work for all member types. + * For anonymous/local types, the only guarantee is given in JLS 13.1, where + * it guarantees that if you call getDeclaringType() repeatedly, you will eventually + * get the top-level class, but it does not say anything about classes in between. + * + * @return the declaring TypeX object, or null. + */ + public TypeX getDeclaringType() { + if (isArray()) return null; + String name = getName(); + int lastDollar = name.lastIndexOf('$'); + if (lastDollar != -1) { + return TypeX.forName(name.substring(0, lastDollar)); + } else { + return null; + } + } + + /** + * Returns a TypeX object representing the component type of this array, or + * null if this type does not represent an array type. + * + * @return the component TypeX object, or null. + */ + public TypeX getComponentType() { + if (isArray()) { + return forSignature(signature.substring(1)); + } else { + return null; + } + } + + /** + * Determines if this represents a primitive type. A primitive type + * is one of nine predefined resolved types. + * + * @return true iff this type represents a primitive type + * + * @see ResolvedTypeX#Boolean + * @see ResolvedTypeX#Character + * @see ResolvedTypeX#Byte + * @see ResolvedTypeX#Short + * @see ResolvedTypeX#Integer + * @see ResolvedTypeX#Long + * @see ResolvedTypeX#Float + * @see ResolvedTypeX#Double + * @see ResolvedTypeX#Void + */ + public boolean isPrimitive() { + return false; + } + + + /** + * Returns a java language string representation of this type. + */ + public String toString() { + return getName(); + } + + // ---- requires worlds + + /** + * Types may have pointcuts just as they have methods and fields. + */ + public ResolvedPointcutDefinition findPointcut(String name, World world) { + return world.findPointcut(this, name); + } + + /** + * Determines if variables of this type could be assigned values of another + * with lots of help. This is the same as isCoercableFrom, but java.lang.Object + * is convertable from and to all types. + * + * @param other the other type + * @param world the {@link World} in which the possible assignment should be checked. + * @return true iff variables of this type could be assigned values of other with possible conversion + */ + public final boolean isConvertableFrom(TypeX other, World world) { + if (this.equals(OBJECT) || other.equals(OBJECT)) return true; + return this.isCoerceableFrom(other, world); + } + + /** + * Determines if variables of this type could be assigned values of another + * without any conversion computation of any kind. For primitive types + * this means equality, and for reference types it means assignability. + * + * @param other the other type + * @param world the {@link World} in which the possible assignment should be checked. + * @return true iff variables of this type could be assigned values of other without any conversion computation + */ + public boolean needsNoConversionFrom(TypeX other, World world) { + // primitives override this method, so we know we're not primitive. + // So if the other is primitive, don't bother asking the world anything. + if (other.isPrimitive()) return false; + return world.needsNoConversionFrom(this, other); + } + + /** + * Determines if the variables of this type could be assigned values + * of another type without casting. This still allows for assignment conversion + * as per JLS 2ed 5.2. + * + * @param other the other type + * @param world the {@link World} in which the possible assignment should be checked. + * @return true iff variables of this type could be assigned values of other without casting + * @exception NullPointerException if other is null + */ + public boolean isAssignableFrom(TypeX other, World world) { + // primitives override this method, so we know we're not primitive. + // So if the other is primitive, don't bother asking the world anything. + if (other.isPrimitive()) return false; + return world.isAssignableFrom(this, other); + } + + /** + * Determines if values of another type could possibly be cast to + * this type. The rules followed are from JLS 2ed 5.5, "Casting Conversion". + * + * <p> This method should be commutative, i.e., for all TypeX a, b and all World w: + * + * <blockquote><pre> + * a.isCoerceableFrom(b, w) == b.isCoerceableFrom(a, w) + * </pre></blockquote> + * + * @param other the other type + * @param world the {@link World} in which the possible coersion should be checked. + * @return true iff values of other could possibly be cast to this type. + * @exception NullPointerException if other is null. + */ + public boolean isCoerceableFrom(TypeX other, World world) { + // primitives override this method, so we know we're not primitive. + // So if the other is primitive, don't bother asking the world anything. + if (other.isPrimitive()) return false; + return world.isCoerceableFrom(this, other); + } + + /** + * Determines if this represents an interface type. + * + * @param world the {@link World} in which we should check. + * @return true iff this represents an interface type. + */ + public final boolean isInterface(World world) { + return world.resolve(this).isInterface(); + } + + /** + * Determines if this represents a class type. + * + * @param world the {@link World} in which we should check. + * @return true iff this represents a class type. + */ + public final boolean isClass(World world) { + return world.resolve(this).isClass(); + } + + + /** + * Determines if this represents an aspect type. + * + * @param world the {@link World} in which we should check. + * @return true iff this represents an aspect type. + */ + public final boolean isAspect(World world) { + return world.resolve(this).isAspect(); + } + + + /** + * Returns a TypeX object representing the superclass of this type, or null. + * If this represents a java.lang.Object, a primitive type, or void, this + * method returns null. + * + * <p> + * This differs from {@link java.lang.class#getSuperclass()} in that + * this returns a TypeX object representing java.lang.Object for interfaces instead + * of returning null. + * + * @param world the {@link World} in which the lookup should be made. + * @return this type's superclass, or null if none exists. + */ + public TypeX getSuperclass(World world) { + return world.getSuperclass(this); + } + + /** + * Returns an array of TypeX objects representing the declared interfaces + * of this type. + * + * <p> + * If this object represents a class, the declared interfaces are those it + * implements. If this object represents an interface, the declared interfaces + * are those it extends. If this object represents a primitive, an empty + * array is returned. If this object represents an array, an array + * containing types for java.lang.Cloneable and java.io.Serializable is returned. + * + * @param world the {@link World} in which the lookup should be made. + * @return an iterator through the declared interfaces of this type. + */ + public TypeX[] getDeclaredInterfaces(World world) { + return world.getDeclaredInterfaces(this); + } + + /** + * Returns an iterator through TypeX objects representing all the direct + * supertypes of this type. That is, through the superclass, if any, and + * all declared interfaces. + * + * @param world the {@link World} in which the lookup should be made. + * @return an iterator through the direct supertypes of this type. + */ + public Iterator getDirectSupertypes(World world) { + return world.resolve(this).getDirectSupertypes(); + } + + /** + * Returns the modifiers for this type. + * + * See {@link java.lang.Class#getModifiers()} for a description + * of the weirdness of this methods on primitives and arrays. + * + * @param world the {@link World} in which the lookup is made. + * @return an int representing the modifiers for this type + * @see java.lang.reflect.Modifier + */ + public int getModifiers(World world) { + return world.getModifiers(this); + } + + /** + * Returns an array representing the declared fields of this object. This may include + * non-user-visible fields. + * This method returns an + * empty array if it represents an array type or a primitive type, so + * the implicit length field of arrays is just that, implicit. + * + * @param world the {@link World} in which the lookup is done. + * @return the array representing the declared fields of this type + */ + public ResolvedMember[] getDeclaredFields(World world) { + return world.getDeclaredFields(this); + } + + /** + * Returns an array representing the declared methods of this object. This includes + * constructors and the static initialzation method. This also includes all + * shadowMungers in an aspect. So it may include more than the user-visible methods. + * This method returns an + * empty array if it represents an array type or a primitive type. + * + * @param world the {@link World} in which the lookup is done. + * @return the array representing the declared methods of this type + */ + public ResolvedMember[] getDeclaredMethods(World world) { + return world.getDeclaredMethods(this); + } + + + /** + * Returns an array representing the declared pointcuts of this object. + * This method returns an + * empty array if it represents an array type or a primitive type. + * + * @param world the {@link World} in which the lookup is done. + * @return the array representing the declared pointcuts of this type + */ + public ResolvedMember[] getDeclaredPointcuts(World world) { + return world.getDeclaredPointcuts(this); + } + + /** + * Returns a resolved version of this type according to a particular world. + * + * @param world thie {@link World} within which to resolve. + * @return a resolved type representing this type in the appropriate world. + */ + public ResolvedTypeX resolve(World world) { + return world.resolve(this); + } + + // ---- fields + + public static final TypeX[] NONE = new TypeX[0]; + public static final TypeX OBJECT = forSignature("Ljava/lang/Object;"); + public static final TypeX OBJECTARRAY = forSignature("[Ljava/lang/Object;"); + public static final TypeX CLONEABLE = forSignature("Ljava/lang/Cloneable;"); + public static final TypeX SERIALIZABLE = forSignature("Ljava/io/Serializable;"); + public static final TypeX THROWABLE = forSignature("Ljava/lang/Throwable;"); + + // ---- helpers + + private static String signatureToName(String signature) { + switch (signature.charAt(0)) { + case 'B': return "byte"; + case 'C': return "char"; + case 'D': return "double"; + case 'F': return "float"; + case 'I': return "int"; + case 'J': return "long"; + case 'L': + return signature.substring(1, signature.length() - 1).replace('/', '.'); + case 'S': return "short"; + case 'V': return "void"; + case 'Z': return "boolean"; + case '[': + return signatureToName(signature.substring(1, signature.length())) + "[]"; + default: + throw new BCException("Bad type signature: " + signature); + } + } + + private static String nameToSignature(String name) { + if (name.equals("byte")) return "B"; + if (name.equals("char")) return "C"; + if (name.equals("double")) return "D"; + if (name.equals("float")) return "F"; + if (name.equals("int")) return "I"; + if (name.equals("long")) return "J"; + if (name.equals("short")) return "S"; + if (name.equals("boolean")) return "Z"; + if (name.equals("void")) return "V"; + if (name.endsWith("[]")) + return "[" + nameToSignature(name.substring(0, name.length() - 2)); + if (name.length() != 0) // lots more tests could be made here... + return "L" + name.replace('.', '/') + ";"; + else + throw new BCException("Bad type name: " + name); + } + + public void write(DataOutputStream s) throws IOException { + s.writeUTF(signature); + } + + public static TypeX read(DataInputStream s) throws IOException { + String sig = s.readUTF(); + if (sig.equals(MISSING_NAME)) { + return ResolvedTypeX.MISSING; + } else { + return TypeX.forSignature(sig); + } + } + + public static void write(TypeX[] types, DataOutputStream s) throws IOException { + int len = types.length; + s.writeShort(len); + for (int i=0; i < len; i++) { + types[i].write(s); + } + } + + public static TypeX[] readArray(DataInputStream s) throws IOException { + int len = s.readShort(); + TypeX[] types = new TypeX[len]; + for (int i=0; i < len; i++) { + types[i] = TypeX.read(s); + } + return types; + } + + + /** + * For debugging purposes + */ + public void dump(World world) { + if (isAspect(world)) System.out.print("aspect "); + else if (isInterface(world)) System.out.print("interface "); + else if (isClass(world)) System.out.print("class "); + + System.out.println(toString()); + dumpResolvedMembers("fields", getDeclaredFields(world)); + dumpResolvedMembers("methods", getDeclaredMethods(world)); + dumpResolvedMembers("pointcuts", getDeclaredPointcuts(world)); + } + + private void dumpResolvedMembers(String label, ResolvedMember[] l) { + final String indent = " "; + System.out.println(label); + if (l == null) { + System.out.println(indent + "null"); + return; + } + + for (int i=0, len=l.length; i < len; i++) { + System.out.println(indent + l[i]); + } + } + + // ---- + + public String getNameAsIdentifier() { + return getName().replace('.', '_'); + } + + public String getPackageNameAsIdentifier() { + String name = getName(); + int index = name.lastIndexOf('.'); + if (index == -1) { + return ""; + } else { + return name.substring(0, index).replace('.', '_'); + } + } + + public String getPackageName() { + String name = getName(); + int index = name.lastIndexOf('.'); + if (index == -1) { + return null; + } else { + return name.substring(0, index); + } + } + + public String getClassName() { + String name = getName(); + int index = name.lastIndexOf('.'); + if (index == -1) { + return name; + } else { + return name.substring(index+1); + } + } + + + + + + public static final String MISSING_NAME = "<missing>"; + +} + diff --git a/weaver/src/org/aspectj/weaver/WeaverStateKind.java b/weaver/src/org/aspectj/weaver/WeaverStateKind.java new file mode 100644 index 000000000..da3fd35f3 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/WeaverStateKind.java @@ -0,0 +1,47 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver; + +import java.io.*; + +import org.aspectj.util.TypeSafeEnum; + +public class WeaverStateKind extends TypeSafeEnum { + private WeaverStateKind(String name, int key) { + super(name, key); + } + + public static final WeaverStateKind read(DataInputStream s) throws IOException { + byte b = s.readByte(); + switch(b) { + case 0: return Untouched; + case 2: return Woven; + } + throw new RuntimeException("bad WeaverState.Kind: " + b); + } + + + public static final WeaverStateKind Untouched = new WeaverStateKind("Untouched", 0); + public static final WeaverStateKind Woven = new WeaverStateKind("Woven", 2); + + + public byte[] getBytes() { + return new byte[] { getKey(), }; + } + + public boolean isWoven() { + return this == Woven; + } + +} diff --git a/weaver/src/org/aspectj/weaver/World.java b/weaver/src/org/aspectj/weaver/World.java new file mode 100644 index 000000000..c12acf985 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/World.java @@ -0,0 +1,320 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver; + +import java.util.*; + +import org.aspectj.weaver.patterns.*; +import org.aspectj.weaver.patterns.Pointcut; +import org.aspectj.asm.StructureModel; +import org.aspectj.bridge.*; +import org.aspectj.bridge.IMessageHandler; + +public abstract class World { + protected IMessageHandler messageHandler = IMessageHandler.SYSTEM_ERR; + + protected Map typeMap = new HashMap(); // Signature to ResolvedType + + protected CrosscuttingMembersSet crosscuttingMembersSet = new CrosscuttingMembersSet(this); + + protected StructureModel model = null; + + protected Lint lint = new Lint(this); + + protected World() { + super(); + typeMap.put("B", ResolvedTypeX.BYTE); + typeMap.put("S", ResolvedTypeX.SHORT); + typeMap.put("I", ResolvedTypeX.INT); + typeMap.put("J", ResolvedTypeX.LONG); + typeMap.put("F", ResolvedTypeX.FLOAT); + typeMap.put("D", ResolvedTypeX.DOUBLE); + typeMap.put("C", ResolvedTypeX.CHAR); + typeMap.put("Z", ResolvedTypeX.BOOLEAN); + typeMap.put("V", ResolvedTypeX.VOID); + } + + public ResolvedTypeX[] resolve(TypeX[] types) { + int len = types.length; + ResolvedTypeX[] ret = new ResolvedTypeX[len]; + for (int i=0; i<len; i++) { + ret[i] = resolve(types[i]); + } + return ret; + } + + public ResolvedTypeX resolve(TypeX ty) { + return resolve(ty, false); + } + + public ResolvedTypeX resolve(TypeX ty, boolean allowMissing) { + //System.out.println("resolve: " + ty + " world " + typeMap.keySet()); + String signature = ty.getSignature(); + ResolvedTypeX ret = (ResolvedTypeX)typeMap.get(signature); + if (ret != null) return ret; + + if (ty.isArray()) { + ret = new ResolvedTypeX.Array(signature, this, resolve(ty.getComponentType(), allowMissing)); + } else { + ret = resolveObjectType(ty); + if (!allowMissing && ret == ResolvedTypeX.MISSING) { + //Thread.currentThread().dumpStack(); + MessageUtil.error(messageHandler, "can't find type " + ty.getName()); + // + " on classpath " + classPath); + } + } + //System.out.println("ret: " + ret); + typeMap.put(signature, ret); + return ret; + } + + //XXX helper method might be bad + public ResolvedTypeX resolve(String name) { + return resolve(TypeX.forName(name)); + } + protected abstract ResolvedTypeX resolveObjectType(TypeX ty); + + protected final boolean isCoerceableFrom(TypeX type, TypeX other) { + return resolve(type).isCoerceableFrom(other); + } + + protected final boolean isAssignableFrom(TypeX type, TypeX other) { + return resolve(type).isAssignableFrom(other); + } + + public boolean needsNoConversionFrom(TypeX type, TypeX other) { + return resolve(type).needsNoConversionFrom(other); + } + + protected final boolean isInterface(TypeX type) { + return resolve(type).isInterface(); + } + + protected final ResolvedTypeX getSuperclass(TypeX type) { + return resolve(type).getSuperclass(); + } + + protected final TypeX[] getDeclaredInterfaces(TypeX type) { + return resolve(type).getDeclaredInterfaces(); + } + + protected final int getModifiers(TypeX type) { + return resolve(type).getModifiers(); + } + + protected final ResolvedMember[] getDeclaredFields(TypeX type) { + return resolve(type).getDeclaredFields(); + } + + protected final ResolvedMember[] getDeclaredMethods(TypeX type) { + return resolve(type).getDeclaredMethods(); + } + + protected final ResolvedMember[] getDeclaredPointcuts(TypeX type) { + return resolve(type).getDeclaredPointcuts(); + } + + // ---- members + + + // XXX should we worry about dealing with context and looking up access? + public ResolvedMember resolve(Member member) { + ResolvedTypeX declaring = member.getDeclaringType().resolve(this); + ResolvedMember ret; + if (member.getKind() == Member.FIELD) { + ret = declaring.lookupField(member); + } else { + ret = declaring.lookupMethod(member); + } + + if (ret != null) return ret; + + return declaring.lookupSyntheticMember(member); + } + + protected int getModifiers(Member member) { + return resolve(member).getModifiers(); + } + + protected String[] getParameterNames(Member member) { + return resolve(member).getParameterNames(); + } + + protected TypeX[] getExceptions(Member member) { + return resolve(member).getExceptions(); + } + + // ---- pointcuts + + public ResolvedPointcutDefinition findPointcut(TypeX typeX, String name) { + throw new RuntimeException("not implemented yet"); + } + + /** + * Get the shadow mungers of this world. + * + * @return a list of {@link IShadowMunger}s appropriate for this world. + */ + //public abstract List getShadowMungers(); + + // ---- empty world + + public static final World EMPTY = new World() { + public List getShadowMungers() { return Collections.EMPTY_LIST; } + public ResolvedTypeX resolveObjectType(TypeX ty) { + return ResolvedTypeX.MISSING; + } + public Advice concreteAdvice(AdviceKind kind, Pointcut p, Member m, int extraMods, + int start, int end, ISourceContext context) { + throw new RuntimeException("unimplemented"); + } + public ConcreteTypeMunger concreteTypeMunger(ResolvedTypeMunger munger, ResolvedTypeX aspectType) { + throw new RuntimeException("unimplemented"); + } + }; + + + public abstract Advice concreteAdvice( + AdviceKind kind, + Pointcut p, + Member signature, + int extraParameterFlags, + int start, int end, ISourceContext context); + + public final Advice concreteAdvice( + AdviceKind kind, + Pointcut p, + Member signature, + int extraParameterFlags, + IHasSourceLocation loc) + { + return concreteAdvice(kind, p, signature, extraParameterFlags, loc.getStart(), loc.getEnd(), loc.getSourceContext()); + } + + + + public ConcreteTypeMunger makeCflowStackFieldAdder(ResolvedMember cflowField) { + throw new RuntimeException("unimplemented"); + } + + public abstract ConcreteTypeMunger concreteTypeMunger(ResolvedTypeMunger munger, ResolvedTypeX aspectType); + + /** + * Nobody should hold onto a copy of this message handler, or setMessageHandler won't + * work right. + */ + public IMessageHandler getMessageHandler() { + return messageHandler; + } + + public void setMessageHandler(IMessageHandler messageHandler) { + this.messageHandler = messageHandler; + } + +// public void addDeclare(ResolvedTypeX onType, Declare declare, boolean forWeaving) { +// // this is not extensible, oh well +// if (declare instanceof DeclareErrorOrWarning) { +// ShadowMunger m = new Checker((DeclareErrorOrWarning)declare); +// onType.addShadowMunger(m); +// } else if (declare instanceof DeclareDominates) { +// declareDominates.add(declare); +// } else if (declare instanceof DeclareParents) { +// declareParents.add(declare); +// } else if (declare instanceof DeclareSoft) { +// DeclareSoft d = (DeclareSoft)declare; +// declareSoft.add(d); +// if (forWeaving) { +// ShadowMunger m = Advice.makeSoftener(this, d.getPointcut().concretize(onType, 0), d.getException()); +// onType.addShadowMunger(m); +// } +// } else { +// throw new RuntimeException("unimplemented"); +// } +// } + + + /** + * Same signature as org.aspectj.util.PartialOrder.PartialComparable.compareTo + */ + public int compareByDominates(ResolvedTypeX aspect1, ResolvedTypeX aspect2) { + //System.out.println("dom compare: " + aspect1 + " with " + aspect2); + //System.out.println(crosscuttingMembersSet.getDeclareDominates()); + + //??? We probably want to cache this result. This is order N where N is the + //??? number of dominates declares in the whole system. + //??? This method can be called a large number of times. + int order = 0; + for (Iterator i = crosscuttingMembersSet.getDeclareDominates().iterator(); i.hasNext(); ) { + DeclareDominates d = (DeclareDominates)i.next(); + int thisOrder = d.compare(aspect1, aspect2); + //System.out.println("comparing: " + thisOrder + ": " + d); + if (thisOrder != 0) { + if (order != 0 && order != thisOrder) { + throw new BCException("conflicting dominates orders"); + } else { + order = thisOrder; + } + } + } + + + return order; + } + + + public int comparePrecedence(ResolvedTypeX aspect1, ResolvedTypeX aspect2) { + //System.err.println("compare precedence " + aspect1 + ", " + aspect2); + if (aspect1.equals(aspect2)) return 0; + + int ret = compareByDominates(aspect1, aspect2); + if (ret != 0) return ret; + + if (aspect1.isAssignableFrom(aspect2)) return -1; + else if (aspect2.isAssignableFrom(aspect1)) return +1; + + return 0; + } + + + + + public List getDeclareParents() { + return crosscuttingMembersSet.getDeclareParents(); + } + + public List getDeclareSoft() { + return crosscuttingMembersSet.getDeclareSofts(); + } + + public CrosscuttingMembersSet getCrosscuttingMembersSet() { + return crosscuttingMembersSet; + } + + public StructureModel getModel() { + return model; + } + + public void setModel(StructureModel model) { + this.model = model; + } + + public Lint getLint() { + return lint; + } + + public void setLint(Lint lint) { + this.lint = lint; + } + +} diff --git a/weaver/src/org/aspectj/weaver/XlintDefault.properties b/weaver/src/org/aspectj/weaver/XlintDefault.properties new file mode 100644 index 000000000..eb5944dfa --- /dev/null +++ b/weaver/src/org/aspectj/weaver/XlintDefault.properties @@ -0,0 +1,4 @@ +invalidAbsoluteTypeName = warning +invalidWildcardTypeName = ignore + +unresolvableMember = warning
\ No newline at end of file diff --git a/weaver/src/org/aspectj/weaver/ast/ASTNode.java b/weaver/src/org/aspectj/weaver/ast/ASTNode.java new file mode 100644 index 000000000..261605b10 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/ast/ASTNode.java @@ -0,0 +1,22 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.ast; + +public abstract class ASTNode { + + public ASTNode() { + super(); + } + +} diff --git a/weaver/src/org/aspectj/weaver/ast/And.java b/weaver/src/org/aspectj/weaver/ast/And.java new file mode 100644 index 000000000..1dc00be7f --- /dev/null +++ b/weaver/src/org/aspectj/weaver/ast/And.java @@ -0,0 +1,51 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.ast; + + +public class And extends Test { + Test left, right; + + public And(Test left, Test right) { + super(); + this.left = left; + this.right = right; + } + + public void accept(ITestVisitor v) { + v.visit(this); + } + + public String toString() { + return "(" + left + " && " + right + ")"; + } + + public boolean equals(Object other) { + if (other instanceof And) { + And o = (And) other; + return o.left.equals(left) && o.right.equals(right); + } else { + return false; + } + } + + public Test getLeft() { + return left; + } + + public Test getRight() { + return right; + } + +} diff --git a/weaver/src/org/aspectj/weaver/ast/Call.java b/weaver/src/org/aspectj/weaver/ast/Call.java new file mode 100644 index 000000000..b521c1199 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/ast/Call.java @@ -0,0 +1,41 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.ast; + +import org.aspectj.weaver.Member; + +public class Call extends Test { + // assert m.return value is boolean + private final Member method; + private final Expr[] args; + + public Call(Member m, Expr[] args) { + super(); + this.method = m; + this.args = args; + } + + public void accept(ITestVisitor v) { + v.visit(this); + } + + public Expr[] getArgs() { + return args; + } + + public Member getMethod() { + return method; + } + +} diff --git a/weaver/src/org/aspectj/weaver/ast/CallExpr.java b/weaver/src/org/aspectj/weaver/ast/CallExpr.java new file mode 100644 index 000000000..19fe73b6b --- /dev/null +++ b/weaver/src/org/aspectj/weaver/ast/CallExpr.java @@ -0,0 +1,48 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.ast; + +import org.aspectj.weaver.*; +import org.aspectj.weaver.Member; + +public class CallExpr extends Expr { + // assert m.return value is boolean + private final Member method; + private final Expr[] args; + private final ResolvedTypeX returnType; // yes, stored in method as well, but that one isn't resolved + + public CallExpr(Member m, Expr[] args, ResolvedTypeX returnType) { + super(); + this.method = m; + this.args = args; + this.returnType = returnType; + } + + public void accept(IExprVisitor v) { + v.visit(this); + } + + public Expr[] getArgs() { + return args; + } + + public Member getMethod() { + return method; + } + + public ResolvedTypeX getType() { + return returnType; + } + +} diff --git a/weaver/src/org/aspectj/weaver/ast/Expr.java b/weaver/src/org/aspectj/weaver/ast/Expr.java new file mode 100644 index 000000000..372c88114 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/ast/Expr.java @@ -0,0 +1,36 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.ast; + +import org.aspectj.weaver.*; + +public abstract class Expr extends ASTNode { + + public Expr() { + super(); + } + + public static final Expr[] NONE = new Expr[0]; + + public abstract void accept(IExprVisitor v); + public abstract ResolvedTypeX getType(); + + public static FieldGet makeFieldGet(Member myField, ResolvedTypeX inAspect) { + return new FieldGet(myField, inAspect); + } + public static Expr makeCallExpr(Member member, Expr[] exprs, ResolvedTypeX returnType) { + return new CallExpr(member, exprs, returnType); + } + +} diff --git a/weaver/src/org/aspectj/weaver/ast/FieldGet.java b/weaver/src/org/aspectj/weaver/ast/FieldGet.java new file mode 100644 index 000000000..89253b103 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/ast/FieldGet.java @@ -0,0 +1,48 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.ast; + +import org.aspectj.weaver.*; + + +public class FieldGet extends Expr { + Member field; + ResolvedTypeX resolvedType; + + public FieldGet(Member field, ResolvedTypeX resolvedType) { + super(); + this.field = field; + this.resolvedType = resolvedType; + } + + public ResolvedTypeX getType() { + return resolvedType; + } + + public String toString() { + return "(FieldGet " + field + ")"; + } + + public void accept(IExprVisitor v) { + v.visit(this); + } + public Member getField() { + return field; + } + + public ResolvedTypeX getResolvedType() { + return resolvedType; + } + +} diff --git a/weaver/src/org/aspectj/weaver/ast/FieldGetCall.java b/weaver/src/org/aspectj/weaver/ast/FieldGetCall.java new file mode 100644 index 000000000..e3d19938b --- /dev/null +++ b/weaver/src/org/aspectj/weaver/ast/FieldGetCall.java @@ -0,0 +1,47 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.ast; + +import org.aspectj.weaver.Member; + +public class FieldGetCall extends Test { + // assert m.return value is boolean + private final Member field; + private final Member method; + private final Expr[] args; + + public FieldGetCall(Member f, Member m, Expr[] args) { + super(); + this.field = f; + this.method = m; + this.args = args; + } + + public void accept(ITestVisitor v) { + v.visit(this); + } + + public Expr[] getArgs() { + return args; + } + + public Member getMethod() { + return method; + } + + public Member getField() { + return field; + } + +} diff --git a/weaver/src/org/aspectj/weaver/ast/IExprVisitor.java b/weaver/src/org/aspectj/weaver/ast/IExprVisitor.java new file mode 100644 index 000000000..38b53ffb3 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/ast/IExprVisitor.java @@ -0,0 +1,23 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.ast; + + +public interface IExprVisitor { + + void visit(Var i); + void visit(FieldGet fieldGet); + void visit(CallExpr callExpr); + +} diff --git a/weaver/src/org/aspectj/weaver/ast/ITestVisitor.java b/weaver/src/org/aspectj/weaver/ast/ITestVisitor.java new file mode 100644 index 000000000..1787ea62e --- /dev/null +++ b/weaver/src/org/aspectj/weaver/ast/ITestVisitor.java @@ -0,0 +1,27 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.ast; + + +public interface ITestVisitor { + + void visit(And e); + void visit(Instanceof i); + void visit(Not not); + void visit(Or or); + void visit(Literal literal); + void visit(Call call); + void visit(FieldGetCall fieldGetCall); + +} diff --git a/weaver/src/org/aspectj/weaver/ast/Instanceof.java b/weaver/src/org/aspectj/weaver/ast/Instanceof.java new file mode 100644 index 000000000..ad2d85ecb --- /dev/null +++ b/weaver/src/org/aspectj/weaver/ast/Instanceof.java @@ -0,0 +1,52 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.ast; + +import org.aspectj.weaver.TypeX; + +public class Instanceof extends Test { + Var var; + TypeX type; + + public Instanceof(Var left, TypeX right) { + super(); + this.var = left; + this.type = right; + } + + public void accept(ITestVisitor v) { + v.visit(this); + } + + public String toString() { + return "(" + var + " instanceof " + type + ")"; + } + + public boolean equals(Object other) { + if (other instanceof Instanceof) { + Instanceof o = (Instanceof) other; + return o.var.equals(var) && o.type.equals(type); + } else { + return false; + } + } + + public Var getVar() { + return var; + } + + public TypeX getType() { + return type; + } +} diff --git a/weaver/src/org/aspectj/weaver/ast/Literal.java b/weaver/src/org/aspectj/weaver/ast/Literal.java new file mode 100644 index 000000000..ed7388fa2 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/ast/Literal.java @@ -0,0 +1,39 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.ast; + + +public class Literal extends Test { + + boolean noTest; + boolean val; + + private Literal(boolean val, boolean noTest) { + super(); + this.val = val; + this.noTest = noTest; + } + + public void accept(ITestVisitor v) { + v.visit(this); + } + + public static final Literal TRUE = new Literal(true, false); + public static final Literal FALSE = new Literal(false, false); +// public static final Literal NO_TEST = new Literal(false, true); + + public String toString() { + return noTest ? "NO_TEST" : val ? "TRUE" : "FALSE"; + } +} diff --git a/weaver/src/org/aspectj/weaver/ast/Not.java b/weaver/src/org/aspectj/weaver/ast/Not.java new file mode 100644 index 000000000..7333e8a87 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/ast/Not.java @@ -0,0 +1,45 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.ast; + + +public class Not extends Test { + Test body; + + public Not(Test left) { + super(); + this.body = left; + } + + public void accept(ITestVisitor v) { + v.visit(this); + } + + public Test getBody() { + return body; + } + + public String toString() { + return "!" + body; + } + + public boolean equals(Object other) { + if (other instanceof Not) { + Not o = (Not) other; + return o.body.equals(body); + } else { + return false; + } + } +} diff --git a/weaver/src/org/aspectj/weaver/ast/Or.java b/weaver/src/org/aspectj/weaver/ast/Or.java new file mode 100644 index 000000000..0f9575dbb --- /dev/null +++ b/weaver/src/org/aspectj/weaver/ast/Or.java @@ -0,0 +1,49 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.ast; + +public class Or extends Test { + Test left, right; + + public Or(Test left, Test right) { + super(); + this.left = left; + this.right = right; + } + + public void accept(ITestVisitor v) { + v.visit(this); + } + + public String toString() { + return "(" + left + " || " + right + ")"; + } + + public boolean equals(Object other) { + if (other instanceof Or) { + Or o = (Or) other; + return o.left.equals(left) && o.right.equals(right); + } else { + return false; + } + } + + public Test getLeft() { + return left; + } + + public Test getRight() { + return right; + } +} diff --git a/weaver/src/org/aspectj/weaver/ast/Test.java b/weaver/src/org/aspectj/weaver/ast/Test.java new file mode 100644 index 000000000..d7385a547 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/ast/Test.java @@ -0,0 +1,91 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.ast; + +import org.aspectj.weaver.*; +import org.aspectj.weaver.ResolvedTypeX; + +public abstract class Test extends ASTNode { + + public Test() { + super(); + } + + public abstract void accept(ITestVisitor v); + + public static Test makeAnd(Test a, Test b) { +// if (a == Literal.NO_TEST) return b; +// if (b == Literal.NO_TEST) return a; + if (a == Literal.TRUE) { + if (b == Literal.TRUE) { + return a; + } else { + return b; + } + } else if (b == Literal.TRUE) { + return a; + } else if (a == Literal.FALSE || b == Literal.FALSE) { + return Literal.FALSE; + } else { + return new And(a, b); + } + } + + public static Test makeOr(Test a, Test b) { +// if (a == Literal.NO_TEST) return a; +// if (b == Literal.NO_TEST) return b; + if (a == Literal.FALSE) { + return b; + } else if (b == Literal.FALSE) { + return a; + } else if (a == Literal.TRUE || b == Literal.TRUE) { + return Literal.TRUE; + } else { + return new Or(a, b); + } + } + + public static Test makeNot(Test a) { + if (a instanceof Not) { + return ((Not) a).getBody(); + } else if (a == Literal.TRUE) { + return Literal.FALSE; + } else if (a == Literal.FALSE) { + return Literal.TRUE; +// } else if (a == Literal.NO_TEST) { +// return a; + } else { + return new Not(a); + } + } + + // uses our special rules that anything matches object + public static Test makeInstanceof(Var v, ResolvedTypeX ty) { + if (ty.equals(ResolvedTypeX.OBJECT)) return Literal.TRUE; + + Test e; + if (ty.isAssignableFrom(v.getType())) e = Literal.TRUE; + else if (! ty.isCoerceableFrom(v.getType())) e = Literal.FALSE; + else e = new Instanceof(v, ty); + return e; + } + + public static Test makeCall(Member m, Expr[] args) { + return new Call(m, args); + } + public static Test makeFieldGetCall(Member f, Member m, Expr[] args) { + return new FieldGetCall(f, m, args); + } + +} diff --git a/weaver/src/org/aspectj/weaver/ast/Var.java b/weaver/src/org/aspectj/weaver/ast/Var.java new file mode 100644 index 000000000..52246f3c6 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/ast/Var.java @@ -0,0 +1,38 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.ast; + +import org.aspectj.weaver.*; + + +public class Var extends Expr { + ResolvedTypeX type; + + public Var(ResolvedTypeX type) { + super(); + this.type = type; + } + + public ResolvedTypeX getType() { + return type; + } + + public String toString() { + return "(Var " + type + ")"; + } + + public void accept(IExprVisitor v) { + v.visit(this); + } +} diff --git a/weaver/src/org/aspectj/weaver/bcel/BcelAdvice.java b/weaver/src/org/aspectj/weaver/bcel/BcelAdvice.java new file mode 100644 index 000000000..b3a42edf3 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/bcel/BcelAdvice.java @@ -0,0 +1,312 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.bcel; + +import org.apache.bcel.generic.*; +import org.aspectj.weaver.*; +import org.aspectj.weaver.ast.*; +import org.aspectj.weaver.patterns.*; + +/** + * Advice implemented for bcel. + * + * @author Erik Hilsdale + * @author Jim Hugunin + */ +public class BcelAdvice extends Advice { + private Test pointcutTest; + private ExposedState exposedState; + + public BcelAdvice(AdviceKind kind, Pointcut pointcut, Member signature, + int extraArgumentFlags, + int start, int end, ISourceContext sourceContext, ResolvedTypeX concreteAspect) + { + super(kind, pointcut, signature, extraArgumentFlags, start, end, sourceContext); + this.concreteAspect = concreteAspect; + } + + // ---- implementations of ShadowMunger's methods + + public void specializeOn(Shadow shadow) { + if (getKind() == AdviceKind.Around) { + ((BcelShadow)shadow).initializeForAroundClosure(); + } + + //XXX this case is just here for supporting lazy test code + if (getKind() == null) { + exposedState = new ExposedState(0); + return; + } + if (getKind().isPerEntry()) { + exposedState = new ExposedState(0); + } else if (getKind().isCflow()) { + exposedState = new ExposedState(nFreeVars); + } else if (getSignature() != null) { + exposedState = new ExposedState(getSignature()); + } else { + exposedState = new ExposedState(0); + return; //XXX this case is just here for supporting lazy test code + } + pointcutTest = getPointcut().findResidue(shadow, exposedState); + + // make sure thisJoinPoint parameters are initialized + if ((extraParameterFlags & ThisJoinPoint) != 0) { + ((BcelShadow)shadow).getThisJoinPointVar(); + } + + if ((extraParameterFlags & ThisJoinPointStaticPart) != 0) { + ((BcelShadow)shadow).getThisJoinPointStaticPartVar(); + } + + if ((extraParameterFlags & ThisEnclosingJoinPointStaticPart) != 0) { + ((BcelShadow)shadow).getThisEnclosingJoinPointStaticPartVar(); + } + } + + public void implementOn(Shadow s) { + BcelShadow shadow = (BcelShadow) s; + if (getKind() == AdviceKind.Before) { + shadow.weaveBefore(this); + } else if (getKind() == AdviceKind.AfterReturning) { + shadow.weaveAfterReturning(this); + } else if (getKind() == AdviceKind.AfterThrowing) { + TypeX catchType = + hasExtraParameter() + ? getExtraParameterType() + : TypeX.THROWABLE; + shadow.weaveAfterThrowing(this, catchType); + } else if (getKind() == AdviceKind.After) { + shadow.weaveAfter(this); + } else if (getKind() == AdviceKind.Around) { + shadow.weaveAroundClosure(this, hasDynamicTests()); + } else if (getKind() == AdviceKind.InterInitializer) { + shadow.weaveAfterReturning(this); + } else if (getKind().isCflow()) { + shadow.weaveCflowEntry(this, getSignature()); + } else if (getKind() == AdviceKind.PerThisEntry) { + shadow.weavePerObjectEntry(this, (BcelVar)shadow.getThisVar()); + } else if (getKind() == AdviceKind.PerTargetEntry) { + shadow.weavePerObjectEntry(this, (BcelVar)shadow.getTargetVar()); + } else if (getKind() == AdviceKind.Softener) { + shadow.weaveSoftener(this, ((ExactTypePattern)exceptionType).getType()); + } else { + throw new BCException("unimplemented kind: " + getKind()); + } + } + + // ---- implementations + + // only call me after prepare has been called + public boolean hasDynamicTests() { +// if (hasExtraParameter() && getKind() == AdviceKind.AfterReturning) { +// TypeX extraParameterType = getExtraParameterType(); +// if (! extraParameterType.equals(TypeX.OBJECT) +// && ! extraParameterType.isPrimitive()) +// return true; +// } + + + return pointcutTest != null && + !(pointcutTest == Literal.TRUE);// || pointcutTest == Literal.NO_TEST); + } + + + /** + * get the instruction list for the really simple version of this advice. + * Is broken apart + * for other advice, but if you want it in one block, this is the method to call. + * + * @param s The shadow around which these instructions will eventually live. + * @param extraArgVar The var that will hold the return value or thrown exception + * for afterX advice + * @param ifNoAdvice The instructionHandle to jump to if the dynamic + * tests for this munger fails. + */ + InstructionList getAdviceInstructions( + BcelShadow s, + BcelVar extraArgVar, + InstructionHandle ifNoAdvice) + { + BcelShadow shadow = (BcelShadow) s; + InstructionFactory fact = shadow.getFactory(); + BcelWorld world = shadow.getWorld(); + + InstructionList il = new InstructionList(); + + // we test to see if we have the right kind of thing... + // after throwing does this just by the exception mechanism. + if (hasExtraParameter() && getKind() == AdviceKind.AfterReturning) { + TypeX extraParameterType = getExtraParameterType(); + if (! extraParameterType.equals(TypeX.OBJECT) + && ! extraParameterType.isPrimitive()) { + il.append( + BcelRenderer.renderTest( + fact, + world, + Test.makeInstanceof( + extraArgVar, getExtraParameterType().resolve(world)), + null, + ifNoAdvice, + null)); + } + } + il.append(getAdviceArgSetup(shadow, extraArgVar, null)); + il.append(getNonTestAdviceInstructions(shadow)); + + InstructionHandle ifYesAdvice = il.getStart(); + il.insert(getTestInstructions(shadow, ifYesAdvice, ifNoAdvice, ifYesAdvice)); + return il; + } + + public InstructionList getAdviceArgSetup( + BcelShadow shadow, + BcelVar extraVar, + InstructionList closureInstantiation) + { + InstructionFactory fact = shadow.getFactory(); + BcelWorld world = shadow.getWorld(); + InstructionList il = new InstructionList(); + +// if (targetAspectField != null) { +// il.append(fact.createFieldAccess( +// targetAspectField.getDeclaringType().getName(), +// targetAspectField.getName(), +// BcelWorld.makeBcelType(targetAspectField.getType()), +// Constants.GETSTATIC)); +// } +// + //System.err.println("BcelAdvice: " + exposedState); + + + if (exposedState.getAspectInstance() != null) { + il.append( + BcelRenderer.renderExpr(fact, world, exposedState.getAspectInstance())); + } + for (int i = 0, len = exposedState.size(); i < len; i++) { + BcelVar v = (BcelVar) exposedState.get(i); + if (v == null) continue; + TypeX desiredTy = getSignature().getParameterTypes()[i]; + v.appendLoadAndConvert(il, fact, desiredTy.resolve(world)); + } + + + if (getKind() == AdviceKind.Around) { + il.append(closureInstantiation); + } else if (hasExtraParameter()) { + extraVar.appendLoadAndConvert( + il, + fact, + getExtraParameterType().resolve(world)); + } + + // handle thisJoinPoint parameters + if ((extraParameterFlags & ThisJoinPoint) != 0) { + shadow.getThisJoinPointBcelVar().appendLoad(il, fact); + } + + if ((extraParameterFlags & ThisJoinPointStaticPart) != 0) { + shadow.getThisJoinPointStaticPartBcelVar().appendLoad(il, fact); + } + + if ((extraParameterFlags & ThisEnclosingJoinPointStaticPart) != 0) { + shadow.getThisEnclosingJoinPointStaticPartBcelVar().appendLoad(il, fact); + } + + + return il; + } + + public InstructionList getNonTestAdviceInstructions(BcelShadow shadow) { + return new InstructionList( + Utility.createInvoke(shadow.getFactory(), shadow.getWorld(), getSignature())); + } + + public InstructionList getTestInstructions( + BcelShadow shadow, + InstructionHandle sk, + InstructionHandle fk, + InstructionHandle next) + { + //System.err.println("test: " + pointcutTest); + return BcelRenderer.renderTest( + shadow.getFactory(), + shadow.getWorld(), + pointcutTest, + sk, + fk, + next); + } + + public int compareTo(Object other) { + if (!(other instanceof BcelAdvice)) return 0; + BcelAdvice o = (BcelAdvice)other; + + //System.err.println("compareTo: " + this + ", " + o); + if (kind.getPrecedence() != o.kind.getPrecedence()) { + if (kind.getPrecedence() > o.kind.getPrecedence()) return +1; + else return -1; + } + + if (kind.isCflow()) { + if (this.innerCflowEntries.contains(o)) return -1; + else if (o.innerCflowEntries.contains(this)) return +1; + else return 0; + } + + + if (kind.isPerEntry() || kind == AdviceKind.Softener) { + return 0; + } + + //System.out.println("compare: " + this + " with " + other); + World world = concreteAspect.getWorld(); + + int ret = + concreteAspect.getWorld().compareByDominates( + concreteAspect, + o.concreteAspect); + if (ret != 0) return ret; + + + ResolvedTypeX declaringAspect = getDeclaringAspect().resolve(world); + ResolvedTypeX o_declaringAspect = o.getDeclaringAspect().resolve(world); + + + if (declaringAspect == o_declaringAspect) { + if (kind.isAfter() || o.kind.isAfter()) { + return this.lexicalPosition < o.lexicalPosition ? -1: +1; + } else { + return this.lexicalPosition < o.lexicalPosition ? +1: -1; + } + } else if (declaringAspect.isAssignableFrom(o_declaringAspect)) { + return -1; + } else if (o_declaringAspect.isAssignableFrom(declaringAspect)) { + return +1; + } else { + return 0; + } + } + + public BcelVar[] getExposedStateAsBcelVars() { + //System.out.println("vars: " + Arrays.asList(exposedState.vars)); + if (exposedState == null) return BcelVar.NONE; + int len = exposedState.vars.length; + BcelVar[] ret = new BcelVar[len]; + for (int i=0; i < len; i++) { + ret[i] = (BcelVar)exposedState.vars[i]; + } + return ret; //(BcelVar[]) exposedState.vars; + } + +} diff --git a/weaver/src/org/aspectj/weaver/bcel/BcelAttributes.java b/weaver/src/org/aspectj/weaver/bcel/BcelAttributes.java new file mode 100644 index 000000000..840d7b04b --- /dev/null +++ b/weaver/src/org/aspectj/weaver/bcel/BcelAttributes.java @@ -0,0 +1,52 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.bcel; + +import java.util.*; + +import org.apache.bcel.classfile.*; +import org.apache.bcel.generic.ConstantPoolGen; +import org.aspectj.weaver.*; +import org.aspectj.weaver.AjAttribute; + + +// this is a class o' static methods for reading attributes. It's pretty much a bridge from +// bcel to AjAttribute. +class BcelAttributes { + + public static List readAjAttributes(Attribute[] as, ISourceContext context) { + List l = new ArrayList(); + for (int i = as.length - 1; i >= 0; i--) { + Attribute a = as[i]; + if (a instanceof Unknown) { + Unknown u = (Unknown) a; + String name = u.getName(); + if (name.startsWith(AjAttribute.AttributePrefix)) { + l.add(AjAttribute.read(name, u.getBytes(), context)); + } + } + } + return l; + } + + public static Attribute bcelAttribute(AjAttribute a, ConstantPoolGen pool) { + int nameIndex = pool.addUtf8(a.getNameString()); + byte[] bytes = a.getBytes(); + int length = bytes.length; + + return new Unknown(nameIndex, length, bytes, pool.getConstantPool()); + + } + +} diff --git a/weaver/src/org/aspectj/weaver/bcel/BcelCflowAccessVar.java b/weaver/src/org/aspectj/weaver/bcel/BcelCflowAccessVar.java new file mode 100644 index 000000000..a04bcc292 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/bcel/BcelCflowAccessVar.java @@ -0,0 +1,92 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.bcel; + +import org.apache.bcel.Constants; +import org.apache.bcel.generic.*; +import org.aspectj.weaver.*; +import org.aspectj.weaver.ResolvedTypeX; +import org.aspectj.weaver.ast.Var; + +/** + * XXX Erik and I need to discuss this hierarchy. Having FieldRef + * extend Var is convenient, but hopefully there's a better design. + * + * This is always a static reference. + */ +public class BcelCflowAccessVar extends BcelVar { + + + private Member stackField; + private int index; + + /** + * @param type The type to convert to from Object + * @param stackField the member containing the CFLOW_STACK_TYPE + * @param index yeah yeah + */ + public BcelCflowAccessVar(ResolvedTypeX type, Member stackField, int index) { + super(type, 0); + this.stackField = stackField; + this.index = index; + } + + public String toString() { + return "BcelCflowAccessVar(" + getType() + " " + stackField + "." + index + ")"; + } + + public Instruction createLoad(InstructionFactory fact) { + throw new RuntimeException("unimplemented"); + } + public Instruction createStore(InstructionFactory fact) { + throw new RuntimeException("unimplemented"); + } + + public InstructionList createCopyFrom(InstructionFactory fact, int oldSlot) { + throw new RuntimeException("unimplemented"); + } + + public void appendLoad(InstructionList il, InstructionFactory fact) { + il.append(createLoadInstructions(getType(), fact)); + } + + public InstructionList createLoadInstructions(ResolvedTypeX toType, InstructionFactory fact) { + InstructionList il = new InstructionList(); + + il.append(Utility.createGet(fact, stackField)); + il.append(Utility.createConstant(fact, index)); + il.append( + fact.createInvoke( + NameMangler.CFLOW_STACK_TYPE, "get", + Type.OBJECT, new Type[] { Type.INT }, + Constants.INVOKEVIRTUAL)); + il.append(Utility.createConversion(fact, Type.OBJECT, BcelWorld.makeBcelType(toType))); + + return il; + + } + + public void appendLoadAndConvert( + InstructionList il, + InstructionFactory fact, + ResolvedTypeX toType) { + il.append(createLoadInstructions(toType, fact)); + + } + + public void insertLoad(InstructionList il, InstructionFactory fact) { + il.insert(createLoadInstructions(getType(), fact)); + } + +} diff --git a/weaver/src/org/aspectj/weaver/bcel/BcelCflowStackFieldAdder.java b/weaver/src/org/aspectj/weaver/bcel/BcelCflowStackFieldAdder.java new file mode 100644 index 000000000..c398cec63 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/bcel/BcelCflowStackFieldAdder.java @@ -0,0 +1,72 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.bcel; + +import org.apache.bcel.Constants; +import org.apache.bcel.classfile.Field; +import org.apache.bcel.generic.*; +import org.apache.bcel.generic.FieldGen; +import org.aspectj.weaver.*; + +public class BcelCflowStackFieldAdder extends BcelTypeMunger { + private ResolvedMember cflowStackField; + public BcelCflowStackFieldAdder(ResolvedMember cflowStackField) { + super(null, null); + this.cflowStackField = cflowStackField; + } + + public boolean munge(BcelClassWeaver weaver) { + LazyClassGen gen = weaver.getLazyClassGen(); + if (!gen.getType().equals(cflowStackField.getDeclaringType())) return false; + + Field f = new FieldGen(cflowStackField.getModifiers(), + BcelWorld.makeBcelType(cflowStackField.getReturnType()), + cflowStackField.getName(), + gen.getConstantPoolGen()).getField(); + gen.addField(f); + + LazyMethodGen clinit = gen.getStaticInitializer(); + InstructionList setup = new InstructionList(); + InstructionFactory fact = gen.getFactory(); + + setup.append(fact.createNew(new ObjectType(NameMangler.CFLOW_STACK_TYPE))); + setup.append(fact.createDup(1)); + setup.append(fact.createInvoke( + NameMangler.CFLOW_STACK_TYPE, + "<init>", + Type.VOID, + new Type[0], + Constants.INVOKESPECIAL)); + + + setup.append(Utility.createSet(fact, cflowStackField)); + clinit.getBody().insert(setup); + + return true; + } + + + public ResolvedMember getMatchingSyntheticMember(Member member) { + return null; + } + + public ResolvedMember getSignature() { + return cflowStackField; + } + + public boolean matches(ResolvedTypeX onType) { + return onType.equals(cflowStackField.getDeclaringType()); + } + +} diff --git a/weaver/src/org/aspectj/weaver/bcel/BcelClassWeaver.java b/weaver/src/org/aspectj/weaver/bcel/BcelClassWeaver.java new file mode 100644 index 000000000..74d2ace0f --- /dev/null +++ b/weaver/src/org/aspectj/weaver/bcel/BcelClassWeaver.java @@ -0,0 +1,889 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.bcel; + +import java.util.*; + +import org.apache.bcel.Constants; +import org.apache.bcel.generic.*; +import org.aspectj.util.PartialOrder; +import org.aspectj.weaver.*; + +class BcelClassWeaver implements IClassWeaver { + + /** + * This is called from {@link BcelWeaver} to perform the per-class weaving process. + */ + public static boolean weave( + BcelWorld world, + LazyClassGen clazz, + List shadowMungers, + List typeMungers) + { + boolean b = new BcelClassWeaver(world, clazz, shadowMungers, typeMungers).weave(); + //clazz.print(); + return b; + } + + // -------------------------------------------- + + private final LazyClassGen clazz; + private final List shadowMungers; + private final List typeMungers; + + private final BcelObjectType ty; // alias of clazz.getType() + private final BcelWorld world; // alias of ty.getWorld() + private final ConstantPoolGen cpg; // alias of clazz.getConstantPoolGen() + private final InstructionFactory fact; // alias of clazz.getFactory(); + + + private final List addedLazyMethodGens = new ArrayList(); + private final Set addedDispatchTargets = new HashSet(); + + + private List addedSuperInitializersAsList = null; // List<IfaceInitList> + private final Map addedSuperInitializers = new HashMap(); // Interface -> IfaceInitList + private List addedThisInitializers = new ArrayList(); // List<NewFieldMunger> + private List addedClassInitializers = new ArrayList(); // List<NewFieldMunger> + + private BcelShadow clinitShadow = null; + + + + /** + * This holds the initialization and pre-initialization shadows for this class + * that were actually matched by mungers (if no match, then we don't even create the + * shadows really). + */ + private final List initializationShadows = new ArrayList(1); + + private BcelClassWeaver( + BcelWorld world, + LazyClassGen clazz, + List shadowMungers, + List typeMungers) + { + super(); + // assert world == clazz.getType().getWorld() + this.world = world; + this.clazz = clazz; + this.shadowMungers = shadowMungers; + this.typeMungers = typeMungers; + this.ty = clazz.getType(); + this.cpg = clazz.getConstantPoolGen(); + this.fact = clazz.getFactory(); + initializeSuperInitializerMap(ty); + } + + // -------------------------------------------- + + private void initializeSuperInitializerMap(ResolvedTypeX child) { + ResolvedTypeX[] superInterfaces = child.getDeclaredInterfaces(); + for (int i=0, len=superInterfaces.length; i < len; i++) { + if (ty.isTopmostImplementor(superInterfaces[i])) { + if (addSuperInitializer(superInterfaces[i])) { + initializeSuperInitializerMap(superInterfaces[i]); + } + } + } + } + + private boolean addSuperInitializer(ResolvedTypeX onType) { + IfaceInitList l = (IfaceInitList) addedSuperInitializers.get(onType); + if (l != null) return false; + l = new IfaceInitList(onType); + addedSuperInitializers.put(onType, l); + return true; + } + + public void addInitializer(ConcreteTypeMunger cm) { + NewFieldTypeMunger m = (NewFieldTypeMunger) cm.getMunger(); + ResolvedTypeX onType = m.getSignature().getDeclaringType().resolve(world); + if (m.getSignature().isStatic()) { + addedClassInitializers.add(cm); + } else { + if (onType == ty) { + addedThisInitializers.add(cm); + } else { + IfaceInitList l = (IfaceInitList) addedSuperInitializers.get(onType); + l.list.add(cm); + } + } + } + + private static class IfaceInitList implements PartialOrder.PartialComparable { + final ResolvedTypeX onType; + List list = new ArrayList(); + IfaceInitList(ResolvedTypeX onType) { + this.onType = onType; + } + + public int compareTo(Object other) { + IfaceInitList o = (IfaceInitList)other; + if (onType.isAssignableFrom(o.onType)) return +1; + else if (o.onType.isAssignableFrom(onType)) return -1; + else return 0; + } + + public int fallbackCompareTo(Object other) { + return 0; + } + } + + // XXX this is being called, but the result doesn't seem to be being used + public boolean addDispatchTarget(ResolvedMember m) { + return addedDispatchTargets.add(m); + } + + public void addLazyMethodGen(LazyMethodGen gen) { + addedLazyMethodGens.add(gen); + } + + public void addOrReplaceLazyMethodGen(LazyMethodGen mg) { + if (alreadyDefined(clazz, mg)) return; + + for (Iterator i = addedLazyMethodGens.iterator(); i.hasNext(); ) { + LazyMethodGen existing = (LazyMethodGen)i.next(); + if (signaturesMatch(mg, existing)) { + if (existing.definingType == null) { + // this means existing was introduced on the class itself + return; + } else if (mg.definingType.isAssignableFrom(existing.definingType)) { + // existing is mg's subtype and dominates mg + return; + } else if (existing.definingType.isAssignableFrom(mg.definingType)) { + // mg is existing's subtype and dominates existing + i.remove(); + addedLazyMethodGens.add(mg); + return; + } else { + throw new BCException("conflict between: " + mg + " and " + existing); + } + } + } + addedLazyMethodGens.add(mg); + } + + private boolean alreadyDefined(LazyClassGen clazz, LazyMethodGen mg) { + for (Iterator i = clazz.getMethodGens().iterator(); i.hasNext(); ) { + if (signaturesMatch(mg, (LazyMethodGen)i.next())) { + return true; + } + } + return false; + } + + private boolean signaturesMatch(LazyMethodGen mg, LazyMethodGen existing) { + return mg.getName().equals(existing.getName()) && + mg.getSignature().equals(existing.getSignature()); + } + + // ---- + + public boolean weave() { + if (clazz.getWeaverState().isWoven()) { + throw new RuntimeException("already woven: " + clazz); + } + + boolean isChanged = false; + + // we want to "touch" all aspects + if (clazz.getType().isAspect()) isChanged = true; + + + // start by munging all typeMungers + for (Iterator i = typeMungers.iterator(); i.hasNext(); ) { + BcelTypeMunger munger = (BcelTypeMunger)i.next(); + isChanged |= munger.munge(this); + } + + // XXX do major sort of stuff + // sort according to: Major: type hierarchy + // within each list: dominates + // don't forget to sort addedThisInitialiers according to dominates + addedSuperInitializersAsList = new ArrayList(addedSuperInitializers.values()); + addedSuperInitializersAsList = PartialOrder.sort(addedSuperInitializersAsList); + if (addedSuperInitializersAsList == null) { + throw new BCException("circularity in inter-types"); + } + + // this will create a static initializer if there isn't one + // this is in just as bad taste as NOPs + LazyMethodGen staticInit = clazz.getStaticInitializer(); + staticInit.getBody().insert(genInitInstructions(addedClassInitializers, true)); + + + // now go through each method, and match against each method. This + // sets up each method's {@link LazyMethodGen#matchedShadows} field, + // and it also possibly adds to {@link #initializationShadows}. + List methodGens = new ArrayList(clazz.getMethodGens()); + for (Iterator i = methodGens.iterator(); i.hasNext();) { + LazyMethodGen mg = (LazyMethodGen)i.next(); + if (! mg.hasBody()) continue; + isChanged |= match(mg); + } + if (! isChanged) return false; + + // now we weave all but the initialization shadows + for (Iterator i = methodGens.iterator(); i.hasNext();) { + LazyMethodGen mg = (LazyMethodGen)i.next(); + if (! mg.hasBody()) continue; + implement(mg); + } + + + // if we matched any initialization shadows, we inline and weave + if (! initializationShadows.isEmpty()) { + inlineSelfConstructors(methodGens); + positionAndImplement(initializationShadows); + } + + + // finally, if we changed, we add in the introduced methods. + if (isChanged) { + clazz.setWeaverState(WeaverStateKind.Woven); + weaveInAddedMethods(); + } + + return isChanged; + } + + + private void inlineSelfConstructors(List methodGens) { + for (Iterator i = methodGens.iterator(); i.hasNext();) { + LazyMethodGen mg = (LazyMethodGen) i.next(); + if (! mg.getName().equals("<init>")) continue; + InstructionHandle ih = findSuperOrThisCall(mg); + if (ih != null && isThisCall(ih)) { + LazyMethodGen donor = getCalledMethod(ih); + inlineMethod(donor, mg, ih); + } + } + } + + private void positionAndImplement(List initializationShadows) { + for (Iterator i = initializationShadows.iterator(); i.hasNext(); ) { + BcelShadow s = (BcelShadow) i.next(); + positionInitializationShadow(s); + //s.getEnclosingMethod().print(); + s.implement(); + } + } + + private void positionInitializationShadow(BcelShadow s) { + LazyMethodGen mg = s.getEnclosingMethod(); + InstructionHandle call = findSuperOrThisCall(mg); + InstructionList body = mg.getBody(); + ShadowRange r = new ShadowRange(body); + r.associateWithShadow((BcelShadow) s); + if (s.getKind() == Shadow.PreInitialization) { + // XXX assert first instruction is an ALOAD_0. + // a pre shadow goes from AFTER the first instruction (which we believe to + // be an ALOAD_0) to just before the call to super + r.associateWithTargets( + Range.genStart(body, body.getStart().getNext()), + Range.genEnd(body, call.getPrev())); + } else { + // assert s.getKind() == Shadow.Initialization + r.associateWithTargets( + Range.genStart(body, call.getNext()), + Range.genEnd(body)); + } + } + + private boolean isThisCall(InstructionHandle ih) { + INVOKESPECIAL inst = (INVOKESPECIAL) ih.getInstruction(); + return inst.getClassName(cpg).equals(clazz.getName()); + } + + + /** inline a particular call in bytecode. + * + * @param donor the method we want to inline + * @param recipient the method containing the call we want to inline + * @param call the instructionHandle in recipient's body holding the call we want to + * inline. + */ + public static void inlineMethod( + LazyMethodGen donor, + LazyMethodGen recipient, + InstructionHandle call) + { + // assert recipient.contains(call) + + /* Implementation notes: + * + * We allocate two slots for every tempvar so we don't screw up + * longs and doubles which may share space. This could be conservatively avoided + * (no reference to a long/double instruction, don't do it) or packed later. + * Right now we don't bother to pack. + * + * Allocate a new var for each formal param of the inlined. Fill with stack + * contents. Then copy the inlined instructions in with the appropriate remap + * table. Any framelocs used by locals in inlined are reallocated to top of + * frame, + */ + final InstructionFactory fact = recipient.getEnclosingClass().getFactory(); + + IntMap frameEnv = new IntMap(); + + // this also sets up the initial environment + InstructionList argumentStores = + genArgumentStores(donor, recipient, frameEnv, fact); + + InstructionList inlineInstructions = + genInlineInstructions(donor, recipient, frameEnv, fact); + + inlineInstructions.insert(argumentStores); + + recipient.getBody().append(call, inlineInstructions); + Utility.deleteInstruction(call, recipient); + } + + /** generate the instructions to be inlined. + * + * @param donor the method from which we will copy (and adjust frame and jumps) + * instructions. + * @param recipient the method the instructions will go into. Used to get the frame + * size so we can allocate new frame locations for locals in donor. + * @param frameEnv an environment to map from donor frame to recipient frame, + * initially populated with argument locations. + * @param fact an instruction factory for recipient + */ + private static InstructionList genInlineInstructions( + LazyMethodGen donor, + LazyMethodGen recipient, + IntMap frameEnv, + InstructionFactory fact) + { + InstructionList footer = new InstructionList(); + InstructionHandle end = footer.append(fact.NOP); + + InstructionList ret = new InstructionList(); + InstructionList sourceList = donor.getBody(); + + Map srcToDest = new HashMap(); + ConstantPoolGen donorCpg = donor.getEnclosingClass().getConstantPoolGen(); + ConstantPoolGen recipientCpg = recipient.getEnclosingClass().getConstantPoolGen(); + + boolean isAcrossClass = donorCpg != recipientCpg; + + // first pass: copy the instructions directly, populate the srcToDest map, + // fix frame instructions + for (InstructionHandle src = sourceList.getStart(); + src != null; + src = src.getNext()) + { + Instruction fresh = src.getInstruction().copy(); + InstructionHandle dest; + if (fresh instanceof CPInstruction) { + // need to reset index to go to new constant pool. This is totally + // a computation leak... we're testing this LOTS of times. Sigh. + if (isAcrossClass) { + CPInstruction cpi = (CPInstruction) fresh; + cpi.setIndex(recipientCpg.addConstant(donorCpg.getConstant(cpi.getIndex()), donorCpg)); + } + } + if (src.getInstruction() == Range.RANGEINSTRUCTION) { + dest = ret.append(Range.RANGEINSTRUCTION); + } else if (fresh instanceof ReturnInstruction) { + dest = ret.append(fact.createBranchInstruction(Constants.GOTO, end)); + } else if (fresh instanceof BranchInstruction) { + dest = ret.append((BranchInstruction) fresh); + } else if ( + fresh instanceof LocalVariableInstruction || fresh instanceof RET) { + IndexedInstruction indexed = (IndexedInstruction) fresh; + int oldIndex = indexed.getIndex(); + int freshIndex; + if (!frameEnv.hasKey(oldIndex)) { + freshIndex = recipient.allocateLocal(2); + frameEnv.put(oldIndex, freshIndex); + } else { + freshIndex = frameEnv.get(oldIndex); + } + indexed.setIndex(freshIndex); + dest = ret.append(fresh); + } else { + dest = ret.append(fresh); + } + srcToDest.put(src, dest); + } + + // second pass: retarget branch instructions, copy ranges and tags + Map tagMap = new HashMap(); + Map shadowMap = new HashMap(); + for (InstructionHandle dest = ret.getStart(), src = sourceList.getStart(); + dest != null; + dest = dest.getNext(), src = src.getNext()) { + Instruction inst = dest.getInstruction(); + + // retarget branches + if (inst instanceof BranchInstruction) { + BranchInstruction branch = (BranchInstruction) inst; + InstructionHandle oldTarget = branch.getTarget(); + InstructionHandle newTarget = + (InstructionHandle) srcToDest.get(oldTarget); + if (newTarget == null) { + // assert this is a GOTO + // this was a return instruction we previously replaced + } else { + branch.setTarget(newTarget); + if (branch instanceof Select) { + Select select = (Select) branch; + InstructionHandle[] oldTargets = select.getTargets(); + for (int k = oldTargets.length - 1; k >= 0; k--) { + select.setTarget( + k, + (InstructionHandle) srcToDest.get(oldTargets[k])); + } + } + } + } + + //copy over tags and range attributes + InstructionTargeter[] srcTargeters = src.getTargeters(); + if (srcTargeters != null) { + for (int j = srcTargeters.length - 1; j >= 0; j--) { + InstructionTargeter old = srcTargeters[j]; + if (old instanceof Tag) { + Tag oldTag = (Tag) old; + Tag fresh = (Tag) tagMap.get(oldTag); + if (fresh == null) { + fresh = oldTag.copy(); + tagMap.put(oldTag, fresh); + } + dest.addTargeter(fresh); + } else if (old instanceof ExceptionRange) { + ExceptionRange er = (ExceptionRange) old; + if (er.getStart() == src) { + ExceptionRange freshEr = + new ExceptionRange( + recipient.getBody(), + er.getCatchType(), + er.getPriority()); + freshEr.associateWithTargets( + dest, + (InstructionHandle)srcToDest.get(er.getEnd()), + (InstructionHandle)srcToDest.get(er.getHandler())); + } + } else if (old instanceof ShadowRange) { + ShadowRange oldRange = (ShadowRange) old; + if (oldRange.getStart() == src) { + BcelShadow oldShadow = oldRange.getShadow(); + BcelShadow freshEnclosing = + oldShadow.getEnclosingShadow() == null + ? null + : (BcelShadow) shadowMap.get(oldShadow.getEnclosingShadow()); + BcelShadow freshShadow = + oldShadow.copyInto(recipient, freshEnclosing); + ShadowRange freshRange = new ShadowRange(recipient.getBody()); + freshRange.associateWithShadow(freshShadow); + freshRange.associateWithTargets( + dest, + (InstructionHandle) srcToDest.get(oldRange.getEnd())); + shadowMap.put(oldRange, freshRange); + //recipient.matchedShadows.add(freshShadow); + } + } + } + } + } + ret.append(footer); + return ret; + } + + /** generate the argument stores in preparation for inlining. + * + * @param donor the method we will inline from. Used to get the signature. + * @param recipient the method we will inline into. Used to get the frame size + * so we can allocate fresh locations. + * @param frameEnv an empty environment we populate with a map from donor frame to + * recipient frame. + * @param fact an instruction factory for recipient + */ + private static InstructionList genArgumentStores( + LazyMethodGen donor, + LazyMethodGen recipient, + IntMap frameEnv, + InstructionFactory fact) + { + InstructionList ret = new InstructionList(); + + int donorFramePos = 0; + + // writing ret back to front because we're popping. + if (! donor.isStatic()) { + int targetSlot = recipient.allocateLocal(Type.OBJECT); + ret.insert(fact.createStore(Type.OBJECT, targetSlot)); + frameEnv.put(donorFramePos, targetSlot); + donorFramePos += 1; + } + Type[] argTypes = donor.getArgumentTypes(); + for (int i = 0, len = argTypes.length; i < len; i++) { + Type argType = argTypes[i]; + int argSlot = recipient.allocateLocal(argType); + ret.insert(fact.createStore(argType, argSlot)); + frameEnv.put(donorFramePos, argSlot); + donorFramePos += argType.getSize(); + } + return ret; + } + + /** get a called method: Assumes the called method is in this class, + * and the reference to it is exact (a la INVOKESPECIAL). + * + * @param ih The InvokeInstruction instructionHandle pointing to the called method. + */ + private LazyMethodGen getCalledMethod( + InstructionHandle ih) + { + InvokeInstruction inst = (InvokeInstruction) ih.getInstruction(); + + String methodName = inst.getName(cpg); + String signature = inst.getSignature(cpg); + + return clazz.getLazyMethodGen(methodName, signature); + } + + private void weaveInAddedMethods() { + Collections.sort(addedLazyMethodGens, + new Comparator() { + public int compare(Object a, Object b) { + LazyMethodGen aa = (LazyMethodGen) a; + LazyMethodGen bb = (LazyMethodGen) b; + int i = aa.getName().compareTo(bb.getName()); + if (i != 0) return i; + return aa.getSignature().compareTo(bb.getSignature()); + } + } + ); + + for (Iterator i = addedLazyMethodGens.iterator(); i.hasNext(); ) { + clazz.addMethodGen((LazyMethodGen)i.next()); + } + } + + void addPerSingletonField(Member field) { + ObjectType aspectType = (ObjectType) BcelWorld.makeBcelType(field.getReturnType()); + String aspectName = field.getReturnType().getName(); + + LazyMethodGen clinit = clazz.getStaticInitializer(); + InstructionList setup = new InstructionList(); + InstructionFactory fact = clazz.getFactory(); + + setup.append(fact.createNew(aspectType)); + setup.append(fact.createDup(1)); + setup.append(fact.createInvoke( + aspectName, + "<init>", + Type.VOID, + new Type[0], + Constants.INVOKESPECIAL)); + setup.append(fact.createFieldAccess(aspectName, field.getName(), aspectType, Constants.PUTSTATIC)); + clinit.getBody().insert(setup); + } + + /** + * Returns null if this is not a Java constructor, and then we won't + * weave into it at all + */ + private InstructionHandle findSuperOrThisCall(LazyMethodGen mg) { + int depth = 1; + InstructionHandle start = mg.getBody().getStart(); + while (true) { + if (start == null) return null; + + Instruction inst = start.getInstruction(); + if (inst instanceof INVOKESPECIAL + && ((INVOKESPECIAL) inst).getName(cpg).equals("<init>")) { + depth--; + if (depth == 0) return start; + } else if (inst instanceof NEW) { + depth++; + } + start = start.getNext(); + } + } + + // ---- + + private boolean match(LazyMethodGen mg) { + BcelShadow enclosingShadow; + + List shadowAccumulator = new ArrayList(); + // we want to match ajsynthetic constructors... + if (mg.getName().equals("<init>")) { + // XXX the enclosing join point is wrong for things before ignoreMe. + InstructionHandle superOrThisCall = findSuperOrThisCall(mg); + + // we don't walk bodies of things where it's a wrong constructor thingie + if (superOrThisCall == null) return false; + + enclosingShadow = BcelShadow.makeConstructorExecution(world, mg, superOrThisCall); + + // walk the body + if (shouldWeaveBody(mg)) { //!mg.isAjSynthetic()) { + for (InstructionHandle h = mg.getBody().getStart(); + h != null; + h = h.getNext()) { + if (h == superOrThisCall) + continue; + match(mg, h, enclosingShadow, shadowAccumulator); + } + match(enclosingShadow, shadowAccumulator); + } + + // XXX we don't do pre-inits of interfaces + + // now add interface inits and cexecs + if (superOrThisCall != null && ! isThisCall(superOrThisCall)) { + InstructionHandle curr = enclosingShadow.getRange().getStart(); + for (Iterator i = addedSuperInitializersAsList.iterator(); i.hasNext(); ) { + IfaceInitList l = (IfaceInitList) i.next(); + // generate the cexec jp + Member ifaceInitSig = AjcMemberMaker.interfaceConstructor(l.onType); + BcelShadow cexecShadow = + BcelShadow.makeIfaceConstructorExecution( + world, + mg, + curr, + ifaceInitSig); + if (match(cexecShadow, shadowAccumulator)) { + cexecShadow.getRange().getBody().append(cexecShadow.getRange().getStart(), fact.NOP); + } + // generate the init jp around it + BcelShadow initShadow = + BcelShadow.makeIfaceInitialization( + world, + mg, + cexecShadow, + ifaceInitSig); + match(initShadow, shadowAccumulator); + // insert code in place + InstructionList inits = genInitInstructions(l.list, false); + initShadow.getRange().insert(inits, Range.InsideAfter); + } + + // now we add our initialization code + InstructionList inits = genInitInstructions(addedThisInitializers, false); + enclosingShadow.getRange().insert(inits, Range.OutsideBefore); + } + + // actually, you only need to inline the self constructors that are + // in a particular group (partition the constructors into groups where members + // call or are called only by those in the group). Then only inline + // constructors + // in groups where at least one initialization jp matched. Future work. + boolean addedInitialization = + match( + BcelShadow.makeUnfinishedInitialization(world, mg), + initializationShadows); + addedInitialization |= + match( + BcelShadow.makeUnfinishedPreinitialization(world, mg), + initializationShadows); + mg.matchedShadows = shadowAccumulator; + return addedInitialization || !shadowAccumulator.isEmpty(); + } else if (!shouldWeaveBody(mg)) { //.isAjSynthetic()) { + return false; + } else { + if (mg.getName().equals("<clinit>")) { + clinitShadow = enclosingShadow = BcelShadow.makeStaticInitialization(world, mg); + //System.err.println(enclosingShadow); + } else if (mg.isAdviceMethod()) { + enclosingShadow = BcelShadow.makeAdviceExecution(world, mg); + } else { + AjAttribute.EffectiveSignatureAttribute effective = mg.getEffectiveSignature(); + if (effective == null) { + enclosingShadow = BcelShadow.makeMethodExecution(world, mg); + } else if (effective.isWeaveBody()) { + enclosingShadow = BcelShadow.makeShadowForMethod(world, mg, effective.getShadowKind(), effective.getEffectiveSignature()); + } else { + return false; + } + } + + for (InstructionHandle h = mg.getBody().getStart(); + h != null; + h = h.getNext()) { + match(mg, h, enclosingShadow, shadowAccumulator); + } + match(enclosingShadow, shadowAccumulator); + mg.matchedShadows = shadowAccumulator; + return !shadowAccumulator.isEmpty(); + } + } + + private boolean shouldWeaveBody(LazyMethodGen mg) { + if (mg.isAjSynthetic()) return mg.getName().equals("<clinit>"); + AjAttribute.EffectiveSignatureAttribute a = mg.getEffectiveSignature(); + if (a != null) return a.isWeaveBody(); + return true; + } + + + /** + * first sorts the mungers, then gens the initializers in the right order + */ + private InstructionList genInitInstructions(List list, boolean isStatic) { + list = PartialOrder.sort(list); + if (list == null) { + throw new BCException("circularity in inter-types"); + } + + InstructionList ret = new InstructionList(); + + for (Iterator i = list.iterator(); i.hasNext();) { + ConcreteTypeMunger cmunger = (ConcreteTypeMunger) i.next(); + NewFieldTypeMunger munger = (NewFieldTypeMunger) cmunger.getMunger(); + ResolvedMember initMethod = munger.getInitMethod(cmunger.getAspectType()); + if (!isStatic) ret.append(fact.ALOAD_0); + ret.append(Utility.createInvoke(fact, world, initMethod)); + } + return ret; + } + + + private void match( + LazyMethodGen mg, + InstructionHandle ih, + BcelShadow enclosingShadow, + List shadowAccumulator) + { + Instruction i = ih.getInstruction(); + if (i instanceof FieldInstruction) { + FieldInstruction fi = (FieldInstruction) i; + if (i instanceof PUTFIELD || i instanceof PUTSTATIC) { + match( + BcelShadow.makeFieldSet(world, mg, ih, enclosingShadow), + shadowAccumulator); + } else { + match( + BcelShadow.makeFieldGet(world, mg, ih, enclosingShadow), + shadowAccumulator); + } + } else if (i instanceof InvokeInstruction) { + InvokeInstruction ii = (InvokeInstruction) i; + if (ii.getMethodName(clazz.getConstantPoolGen()).equals("<init>")) { + match( + BcelShadow.makeConstructorCall(world, mg, ih, enclosingShadow), + shadowAccumulator); + } else if (ii instanceof INVOKESPECIAL) { + String onTypeName = ii.getClassName(cpg); + if (onTypeName.equals(mg.getEnclosingClass().getName())) { + // we are private + matchInvokeInstruction(mg, ih, ii, enclosingShadow, shadowAccumulator); + } else { + // we are a super call, and this is not a join point in AspectJ-1.{0,1} + } + } else { + matchInvokeInstruction(mg, ih, ii, enclosingShadow, shadowAccumulator); + } + } + // performance optimization... we only actually care about ASTORE instructions, + // since that's what every javac type thing ever uses to start a handler, but for + // now we'll do this for everybody. + if (Range.isRangeHandle(ih)) return; + InstructionTargeter[] targeters = ih.getTargeters(); + if (targeters != null) { + for (int j = 0; j < targeters.length; j++) { + InstructionTargeter t = targeters[j]; + if (t instanceof ExceptionRange) { + // assert t.getHandler() == ih + ExceptionRange er = (ExceptionRange) t; + if (er.getCatchType() == null) continue; + match( + BcelShadow.makeExceptionHandler( + world, + er, + mg, ih, enclosingShadow), + shadowAccumulator); + } + } + } + } + + private void matchInvokeInstruction(LazyMethodGen mg, + InstructionHandle ih, + InvokeInstruction invoke, + BcelShadow enclosingShadow, + List shadowAccumulator) + { + String methodName = invoke.getName(cpg); + if (methodName.startsWith(NameMangler.PREFIX)) { + Member method = + world.makeMethodSignature(clazz, invoke); + ResolvedMember declaredSig = method.resolve(world); + if (declaredSig == null) return; + AjAttribute.EffectiveSignatureAttribute effectiveSig = declaredSig.getEffectiveSignature(); + if (effectiveSig == null) return; + //System.err.println("call to inter-type member: " + effectiveSig); + if (effectiveSig.isWeaveBody()) return; + + match(BcelShadow.makeShadowForMethodCall(world, mg, ih, enclosingShadow, + effectiveSig.getShadowKind(), effectiveSig.getEffectiveSignature()), + shadowAccumulator); + } else { + match( + BcelShadow.makeMethodCall(world, mg, ih, enclosingShadow), + shadowAccumulator); + } + } + + private boolean match(BcelShadow shadow, List shadowAccumulator) { + //System.err.println("match: " + shadow); + boolean isMatched = false; + for (Iterator i = shadowMungers.iterator(); i.hasNext(); ) { + ShadowMunger munger = (ShadowMunger)i.next(); + if (munger.match(shadow, world)) { + shadow.addMunger(munger); + isMatched = true; + } + } + if (isMatched) shadowAccumulator.add(shadow); + return isMatched; + } + + // ---- + + private void implement(LazyMethodGen mg) { + List shadows = mg.matchedShadows; + if (shadows == null) return; + // We depend on a partial order such that inner shadows are earlier on the list + // than outer shadows. That's fine. This order is preserved if: + + // A preceeds B iff B.getStart() is LATER THAN A.getStart(). + + for (Iterator i = shadows.iterator(); i.hasNext(); ) { + BcelShadow shadow = (BcelShadow)i.next(); + shadow.implement(); + } + mg.matchedShadows = null; + } + + // ---- + + public LazyClassGen getLazyClassGen() { + return clazz; + } + + public List getShadowMungers() { + return shadowMungers; + } + + public BcelWorld getWorld() { + return world; + } + +} diff --git a/weaver/src/org/aspectj/weaver/bcel/BcelField.java b/weaver/src/org/aspectj/weaver/bcel/BcelField.java new file mode 100644 index 000000000..aa9d4cec6 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/bcel/BcelField.java @@ -0,0 +1,60 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.bcel; + +import java.util.*; + +import org.apache.bcel.classfile.Field; +import org.aspectj.weaver.*; + +final class BcelField extends ResolvedMember { + + private Field field; + private boolean isAjSynthetic; + + BcelField(BcelObjectType declaringType, Field field) { + super( + FIELD, + declaringType, + field.getAccessFlags(), + field.getName(), + field.getSignature()); + this.field = field; + unpackAttributes(declaringType.getWorld()); + checkedExceptions = TypeX.NONE; + } + + // ---- + + BcelObjectType getBcelDeclaringType() { + return (BcelObjectType) getDeclaringType(); // I want covariant return types. + } + + private void unpackAttributes(World world) { + List as = BcelAttributes.readAjAttributes(field.getAttributes(), getSourceContext(world)); + for (Iterator iter = as.iterator(); iter.hasNext();) { + AjAttribute a = (AjAttribute) iter.next(); + if (a instanceof AjAttribute.AjSynthetic) { + isAjSynthetic = true; + } else { + throw new BCException("weird field attribute " + a); + } + } + isAjSynthetic = false; + } + + public boolean isAjSynthetic() { + return isAjSynthetic; // || getName().startsWith(NameMangler.PREFIX); + } +} diff --git a/weaver/src/org/aspectj/weaver/bcel/BcelFieldRef.java b/weaver/src/org/aspectj/weaver/bcel/BcelFieldRef.java new file mode 100644 index 000000000..6896f0ba4 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/bcel/BcelFieldRef.java @@ -0,0 +1,103 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.bcel; + +import org.apache.bcel.Constants; +import org.apache.bcel.generic.*; +import org.aspectj.weaver.*; +import org.aspectj.weaver.ResolvedTypeX; +import org.aspectj.weaver.ast.Var; + +/** + * XXX Erik and I need to discuss this hierarchy. Having FieldRef + * extend Var is convenient, but hopefully there's a better design. + * + * This is always a static reference. + */ +public class BcelFieldRef extends BcelVar { + + private String className, fieldName; + + public BcelFieldRef(ResolvedTypeX type, String className, String fieldName) { + super(type, 0); + this.className = className; + this.fieldName = fieldName; + } + + public String toString() { + return "BcelFieldRef(" + getType() + " " + className + "." + fieldName + ")"; + } + + //public int getSlot() { return slot; } + + public Instruction createLoad(InstructionFactory fact) { + return fact.createFieldAccess(className, fieldName, + BcelWorld.makeBcelType(getType()), Constants.GETSTATIC); + } + public Instruction createStore(InstructionFactory fact) { + return fact.createFieldAccess(className, fieldName, + BcelWorld.makeBcelType(getType()), Constants.PUTSTATIC); + } + + public InstructionList createCopyFrom(InstructionFactory fact, int oldSlot) { + throw new RuntimeException("unimplemented"); + } + + // this is an array var +// void appendConvertableArrayLoad( +// InstructionList il, +// InstructionFactory fact, +// int index, +// ResolvedTypeX convertTo) +// { +// ResolvedTypeX convertFromType = getType().getResolvedComponentType(); +// appendLoad(il, fact); +// il.append(Utility.createConstant(fact, index)); +// il.append(fact.createArrayLoad(BcelWorld.makeBcelType(convertFromType))); +// Utility.appendConversion(il, fact, convertFromType, convertTo); +// } +// +// void appendConvertableArrayStore( +// InstructionList il, +// InstructionFactory fact, +// int index, +// BcelFieldRef storee) +// { +// ResolvedTypeX convertToType = getType().getResolvedComponentType(); +// appendLoad(il, fact); +// il.append(Utility.createConstant(fact, index)); +// storee.appendLoad(il, fact); +// Utility.appendConversion(il, fact, storee.getType(), convertToType); +// il.append(fact.createArrayStore(BcelWorld.makeBcelType(convertToType))); +// } +// +// InstructionList createConvertableArrayStore( +// InstructionFactory fact, +// int index, +// BcelFieldRef storee) +// { +// InstructionList il = new InstructionList(); +// appendConvertableArrayStore(il, fact, index, storee); +// return il; +// } +// InstructionList createConvertableArrayLoad( +// InstructionFactory fact, +// int index, +// ResolvedTypeX convertTo) +// { +// InstructionList il = new InstructionList(); +// appendConvertableArrayLoad(il, fact, index, convertTo); +// return il; +// } +} diff --git a/weaver/src/org/aspectj/weaver/bcel/BcelMethod.java b/weaver/src/org/aspectj/weaver/bcel/BcelMethod.java new file mode 100644 index 000000000..c43ae2241 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/bcel/BcelMethod.java @@ -0,0 +1,107 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.bcel; + +import java.lang.reflect.Modifier; +import java.util.*; + +import org.apache.bcel.classfile.*; +import org.aspectj.weaver.*; + +final class BcelMethod extends ResolvedMember { + + private Method method; + private boolean isAjSynthetic; + private ShadowMunger associatedShadowMunger; + private AjAttribute.EffectiveSignatureAttribute effectiveSignature; + + BcelMethod(BcelObjectType declaringType, Method method) { + super( + method.getName().equals("<init>") ? CONSTRUCTOR : METHOD, + declaringType, + declaringType.isInterface() + ? method.getAccessFlags() | Modifier.INTERFACE + : method.getAccessFlags(), + method.getName(), + method.getSignature()); + this.method = method; + unpackAjAttributes(declaringType.getWorld()); + unpackJavaAttributes(); + } + + // ---- + + BcelObjectType getBcelDeclaringType() { + return (BcelObjectType) getDeclaringType(); // I want covariant return types. + } + + private void unpackJavaAttributes() { + ExceptionTable exnTable = method.getExceptionTable(); + checkedExceptions = (exnTable == null) + ? TypeX.NONE + : TypeX.forNames(exnTable.getExceptionNames()); + + LocalVariableTable varTable = method.getLocalVariableTable(); + int len = getArity(); + if (varTable == null) { + this.parameterNames = Utility.makeArgNames(len); + } else { + TypeX[] paramTypes = getParameterTypes(); + String[] paramNames = new String[len]; + int index = isStatic() ? 0 : 1; + for (int i = 0; i < len; i++) { + LocalVariable lv = varTable.getLocalVariable(index); + if (lv == null) { + paramNames[i] = "arg" + i; + } else { + paramNames[i] = lv.getName(); + } + index += paramTypes[i].getSize(); + } + this.parameterNames = paramNames; + } + } + + private void unpackAjAttributes(World world) { + List as = BcelAttributes.readAjAttributes(method.getAttributes(), getSourceContext(world)); + //System.out.println("unpack: " + this + ", " + as); + for (Iterator iter = as.iterator(); iter.hasNext();) { + AjAttribute a = (AjAttribute) iter.next(); + if (a instanceof AjAttribute.AdviceAttribute) { + associatedShadowMunger = ((AjAttribute.AdviceAttribute)a).reify(this, world); + return; + } else if (a instanceof AjAttribute.AjSynthetic) { + isAjSynthetic = true; + } else if (a instanceof AjAttribute.EffectiveSignatureAttribute) { + //System.out.println("found effective: " + this); + effectiveSignature = (AjAttribute.EffectiveSignatureAttribute)a; + } else { + throw new BCException("weird method attribute " + a); + } + } + associatedShadowMunger = null; + } + + public boolean isAjSynthetic() { + return isAjSynthetic; // || getName().startsWith(NameMangler.PREFIX); + } + + public ShadowMunger getAssociatedShadowMunger() { + return associatedShadowMunger; + } + + public AjAttribute.EffectiveSignatureAttribute getEffectiveSignature() { + return effectiveSignature; + } +} diff --git a/weaver/src/org/aspectj/weaver/bcel/BcelObjectType.java b/weaver/src/org/aspectj/weaver/bcel/BcelObjectType.java new file mode 100644 index 000000000..303e12561 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/bcel/BcelObjectType.java @@ -0,0 +1,236 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.bcel; + +import java.io.PrintStream; +import java.util.*; + +import org.apache.bcel.classfile.*; +import org.aspectj.weaver.*; +import org.aspectj.weaver.patterns.*; +import org.aspectj.weaver.patterns.PerClause; + +// ??? exposed for testing +public class BcelObjectType extends ResolvedTypeX.Name { + private JavaClass javaClass; + private boolean isObject = false; // set upon construction + private LazyClassGen lazyClassGen = null; // set lazily if it's an aspect + + // lazy, for no particular reason I can discern + private ResolvedTypeX[] interfaces = null; + private ResolvedTypeX superClass = null; + private ResolvedMember[] fields = null; + private ResolvedMember[] methods = null; + + // strangely non-lazy + private ResolvedPointcutDefinition[] pointcuts = null; + private PerClause perClause = null; + private WeaverStateKind weaverState = null; + private List typeMungers = Collections.EMPTY_LIST; + private List declares = Collections.EMPTY_LIST; + private ResolvedMember[] privilegedAccess = null; + + + public Collection getTypeMungers() { + return typeMungers; + } + + + public Collection getDeclares() { + return declares; + } + + public Collection getPrivilegedAccesses() { + if (privilegedAccess == null) return Collections.EMPTY_LIST; + return Arrays.asList(privilegedAccess); + } + + + // IMPORTANT! THIS DOESN'T do real work on the java class, just stores it away. + BcelObjectType(String signature, World world, JavaClass javaClass) { + super(signature, world); + this.javaClass = javaClass; + + sourceContext = new BcelSourceContext(this); + + // this should only ever be java.lang.Object which is + // the only class in Java-1.4 with no superclasses + isObject = (javaClass.getSuperclassNameIndex() == 0); + unpackAspectAttributes(); + } + + public int getModifiers() { + return javaClass.getAccessFlags(); + } + + public ResolvedTypeX getSuperclass() { + if (isObject) return null; + if (superClass == null) { + superClass = world.resolve(TypeX.forName(javaClass.getSuperclassName())); + } + return superClass; + } + + public ResolvedTypeX[] getDeclaredInterfaces() { + if (interfaces == null) { + String[] ifaceNames = javaClass.getInterfaceNames(); + interfaces = new ResolvedTypeX[ifaceNames.length]; + for (int i = 0, len = ifaceNames.length; i < len; i++) { + interfaces[i] = world.resolve(TypeX.forName(ifaceNames[i])); + } + } + return interfaces; + } + + public ResolvedMember[] getDeclaredMethods() { + if (methods == null) { + Method[] ms = javaClass.getMethods(); + ResolvedMember[] ret = new ResolvedMember[ms.length]; + for (int i = ms.length - 1; i >= 0; i--) { + ret[i] = new BcelMethod(this, ms[i]); + } + methods = ret; + } + return methods; + } + + public ResolvedMember[] getDeclaredFields() { + if (fields == null) { + Field[] fs = javaClass.getFields(); + ResolvedMember[] ret = new ResolvedMember[fs.length]; + for (int i = 0, len = fs.length; i < len; i++) { + ret[i] = new BcelField(this, fs[i]); + } + fields = ret; + } + return fields; + } + + // ---- + // fun based on the aj attributes + + public ResolvedMember[] getDeclaredPointcuts() { + return pointcuts; + } + + //??? method only used for testing + public void addPointcutDefinition(ResolvedPointcutDefinition d) { + int len = pointcuts.length; + ResolvedPointcutDefinition[] ret = new ResolvedPointcutDefinition[len+1]; + System.arraycopy(pointcuts, 0, ret, 0, len); + ret[len] = d; + pointcuts = ret; + } + + + + public boolean isAspect() { + return perClause != null; + } + + private void unpackAspectAttributes() { + List pointcuts = new ArrayList(); + typeMungers = new ArrayList(); + declares = new ArrayList(); + List l = BcelAttributes.readAjAttributes(javaClass.getAttributes(), getSourceContext()); + for (Iterator iter = l.iterator(); iter.hasNext();) { + AjAttribute a = (AjAttribute) iter.next(); + if (a instanceof AjAttribute.Aspect) { + perClause = ((AjAttribute.Aspect)a).reify(this); + } else if (a instanceof AjAttribute.PointcutDeclarationAttribute) { + pointcuts.add(((AjAttribute.PointcutDeclarationAttribute)a).reify()); + } else if (a instanceof AjAttribute.WeaverState) { + weaverState = ((AjAttribute.WeaverState)a).reify(); + } else if (a instanceof AjAttribute.TypeMunger) { + typeMungers.add(((AjAttribute.TypeMunger)a).reify(getWorld(), this)); + } else if (a instanceof AjAttribute.DeclareAttribute) { + declares.add(((AjAttribute.DeclareAttribute)a).getDeclare()); + } else if (a instanceof AjAttribute.PrivilegedAttribute) { + privilegedAccess = ((AjAttribute.PrivilegedAttribute)a).getAccessedMembers(); + } else if (a instanceof AjAttribute.SourceContextAttribute) { + ((BcelSourceContext)sourceContext).addAttributeInfo((AjAttribute.SourceContextAttribute)a); + } else { + throw new BCException("bad attribute " + a); + } + } + this.pointcuts = (ResolvedPointcutDefinition[]) + pointcuts.toArray(new ResolvedPointcutDefinition[pointcuts.size()]); +// this.typeMungers = (BcelTypeMunger[]) +// typeMungers.toArray(new BcelTypeMunger[typeMungers.size()]); +// this.declares = (Declare[]) +// declares.toArray(new Declare[declares.size()]); + } + + public PerClause getPerClause() { + return perClause; + } + + + JavaClass getJavaClass() { + return javaClass; + } + + /** + * Switch to a new JavaClass and clear all caches + */ + void replaceJavaClass(JavaClass jc) { + if (this.javaClass == jc) return; + + this.javaClass = jc; + this.interfaces = null; + this.superClass = null; + this.fields = null; + this.methods = null; + this.pointcuts = null; + this.perClause = null; + this.weaverState = null; + this.lazyClassGen = null; + //XXX is clearing these caches sufficient + } + + public WeaverStateKind getWeaverState() { + return weaverState; + } + + public void setWeaverState(WeaverStateKind weaverState) { + this.weaverState = weaverState; + } + + public void printWackyStuff(PrintStream out) { + if (typeMungers.size() > 0) { + out.println(" TypeMungers: " + typeMungers); + } + if (declares.size() > 0) { + out.println(" declares: " + declares); + } + } + + /** + * Return the lazyClassGen associated with this type. For aspect types, this + * value will be cached, since it is used to inline advice. For non-aspect + * types, this lazyClassGen is always newly constructed. + */ + public LazyClassGen getLazyClassGen() { + LazyClassGen ret = lazyClassGen; + if (ret == null) { + ret = new LazyClassGen(this); + if (isAspect()) { + lazyClassGen = ret; + } + } + return ret; + } +} + + diff --git a/weaver/src/org/aspectj/weaver/bcel/BcelRenderer.java b/weaver/src/org/aspectj/weaver/bcel/BcelRenderer.java new file mode 100644 index 000000000..ab5595916 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/bcel/BcelRenderer.java @@ -0,0 +1,232 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.bcel; + +import org.apache.bcel.Constants; +import org.apache.bcel.generic.*; +import org.aspectj.weaver.*; +import org.aspectj.weaver.BCException; +import org.aspectj.weaver.ast.*; + +// we generate right to left, btw. +public class BcelRenderer implements ITestVisitor, IExprVisitor { + + private InstructionList instructions; + private InstructionFactory fact; + private BcelWorld world; + + InstructionHandle sk, fk, next = null; + + private BcelRenderer(InstructionFactory fact, BcelWorld world) { + super(); + this.fact = fact; + this.world = world; + this.instructions = new InstructionList(); + } + + // ---- renderers + + public static InstructionList renderExpr( + InstructionFactory fact, + BcelWorld world, + Expr e) + { + BcelRenderer renderer = new BcelRenderer(fact, world); + e.accept(renderer); + return renderer.instructions; + } + public static InstructionList renderExpr( + InstructionFactory fact, + BcelWorld world, + Expr e, + Type desiredType) + { + BcelRenderer renderer = new BcelRenderer(fact, world); + e.accept(renderer); + InstructionList il = renderer.instructions; + il.append(Utility.createConversion(fact, world.makeBcelType(e.getType()), desiredType)); + return il; + } + + public static InstructionList renderExprs( + InstructionFactory fact, + BcelWorld world, + Expr[] es) + { + BcelRenderer renderer = new BcelRenderer(fact, world); + for (int i = es.length - 1; i >= 0; i--) { + es[i].accept(renderer); + } + return renderer.instructions; + } + + /* + * Get the instructions representing this test. + * + * @param e test to render + * @param sk instructionHandle to jump to if our rendered check succeeds (typically start of advice) + * @param fk instructionHandle to jump to if our rendered check fails (typically after end of advice) + * @param next instructionHandle that will follow this generated code. Passing in null will generate + * one unnecessary GOTO instruction. + * + * @returns the instruction list representing this expression + */ + public static InstructionList renderTest( + InstructionFactory fact, + BcelWorld world, + Test e, + InstructionHandle sk, + InstructionHandle fk, + InstructionHandle next) + { + BcelRenderer renderer = new BcelRenderer(fact, world); + renderer.recur(e, sk, fk, next); + return renderer.instructions; + } + + /* + * Get the instructions representing this test. + * + * @param e test to render + * @param sk instructionHandle to jump to if our rendered check succeeds (typically start of advice) + * @param fk instructionHandle to jump to if our rendered check fails (typically after end of advice) + * + * @returns the instruction list representing this expression + */ + public static InstructionList renderTest( + InstructionFactory fact, + BcelWorld world, + Test e, + InstructionHandle sk, + InstructionHandle fk) + { + return renderTest(fact, world, e, sk, fk, null); + } + + // ---- recurrers + + private void recur( + Test e, + InstructionHandle sk, + InstructionHandle fk, + InstructionHandle next) + { + this.sk = sk; + this.fk = fk; + this.next = next; + e.accept(this); + } + + // ---- test visitors + + public void visit(And e) { + InstructionHandle savedFk = fk; + recur(e.getRight(), sk, fk, next); + InstructionHandle ning = instructions.getStart(); + recur(e.getLeft(), ning, savedFk, ning); + } + + public void visit(Or e) { + InstructionHandle savedSk = sk; + recur(e.getRight(), sk, fk, next); + recur(e.getLeft(), savedSk, instructions.getStart(), instructions.getStart()); + } + + public void visit(Not e) { + recur(e.getBody(), fk, sk, next); + } + + public void visit(Instanceof i) { + instructions.insert(createJumpBasedOnBooleanOnStack()); + instructions.insert( + Utility.createInstanceof(fact, (ObjectType) world.makeBcelType(i.getType()))); + i.getVar().accept(this); + } + + private InstructionList createJumpBasedOnBooleanOnStack() { + InstructionList il = new InstructionList(); + if (sk == fk) { + // don't bother generating if it doesn't matter + if (sk != next) { + il.insert(fact.createBranchInstruction(Constants.GOTO, sk)); + } + return il; + } + + if (fk == next) { + il.insert(fact.createBranchInstruction(Constants.IFNE, sk)); + } else if (sk == next) { + il.insert(fact.createBranchInstruction(Constants.IFEQ, fk)); + } else { + il.insert(fact.createBranchInstruction(Constants.GOTO, sk)); + il.insert(fact.createBranchInstruction(Constants.IFEQ, fk)); + } + return il; + } + + + public void visit(Literal literal) { + if (literal == Literal.FALSE) + throw new BCException("bad"); + } + + public void visit(Call call) { + Member method = call.getMethod(); + // assert method.isStatic() + Expr[] args = call.getArgs(); + //System.out.println("args: " + Arrays.asList(args)); + InstructionList callIl = renderExprs(fact, world, args); + //System.out.println("rendered args: " + callIl); + callIl.append(Utility.createInvoke(fact, world, method)); + callIl.append(createJumpBasedOnBooleanOnStack()); + instructions.insert(callIl); + } + + public void visit(FieldGetCall fieldGetCall) { + Member field = fieldGetCall.getField(); + Member method = fieldGetCall.getMethod(); + InstructionList il = new InstructionList(); + il.append(Utility.createGet(fact, field)); + // assert !method.isStatic() + Expr[] args = fieldGetCall.getArgs(); + //System.out.println("args: " + Arrays.asList(args)); + il.append(renderExprs(fact, world, args)); + //System.out.println("rendered args: " + callIl); + il.append(Utility.createInvoke(fact, world, method)); + il.append(createJumpBasedOnBooleanOnStack()); + instructions.insert(il); + } + + // ---- expr visitors + + public void visit(Var var) { + BcelVar bvar = (BcelVar) var; + bvar.insertLoad(instructions, fact); + } + + public void visit(FieldGet fieldGet) { + Member field = fieldGet.getField(); + // assert field.isStatic() + instructions.insert(Utility.createGet(fact, field)); + } + + public void visit(CallExpr call) { + Member method = call.getMethod(); + // assert method.isStatic() + Expr[] args = call.getArgs(); + InstructionList callIl = renderExprs(fact, world, args); + callIl.append(Utility.createInvoke(fact, world, method)); + instructions.insert(callIl); + } +} diff --git a/weaver/src/org/aspectj/weaver/bcel/BcelShadow.java b/weaver/src/org/aspectj/weaver/bcel/BcelShadow.java new file mode 100644 index 000000000..432acd17b --- /dev/null +++ b/weaver/src/org/aspectj/weaver/bcel/BcelShadow.java @@ -0,0 +1,1731 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.bcel; + +import java.io.File; +import java.lang.reflect.Modifier; +import java.util.*; + +import org.apache.bcel.Constants; +import org.apache.bcel.classfile.Field; +import org.apache.bcel.generic.*; +import org.aspectj.bridge.SourceLocation; +import org.aspectj.weaver.*; +import org.aspectj.weaver.Shadow.Kind; +import org.aspectj.weaver.ast.Var; + + +/* + * Some fun implementation stuff: + * + * * expressionKind advice is non-execution advice + * * may have a target. + * * if the body is extracted, it will be extracted into + * a static method. The first argument to the static + * method is the target + * * advice may expose a this object, but that's the advice's + * consideration, not ours. This object will NOT be cached in another + * local, but will always come from frame zero. + * + * * non-expressionKind advice is execution advice + * * may have a this. + * * target is same as this, and is exposed that way to advice + * (i.e., target will not be cached, will always come from frame zero) + * * if the body is extracted, it will be extracted into a method + * with same static/dynamic modifier as enclosing method. If non-static, + * target of callback call will be this. + * + * * because of these two facts, the setup of the actual arguments (including + * possible target) callback method is the same for both kinds of advice: + * push the targetVar, if it exists (it will not exist for advice on static + * things), then push all the argVars. + * + * Protected things: + * + * * the above is sufficient for non-expressionKind advice for protected things, + * since the target will always be this. + * + * * For expressionKind things, we have to modify the signature of the callback + * method slightly. For non-static expressionKind things, we modify + * the first argument of the callback method NOT to be the type specified + * by the method/field signature (the owner), but rather we type it to + * the currentlyEnclosing type. We are guaranteed this will be fine, + * since the verifier verifies that the target is a subtype of the currently + * enclosingType. + * + * Worries: + * + * * ConstructorCalls will be weirder than all of these, since they + * supposedly don't have a target (according to AspectJ), but they clearly + * do have a target of sorts, just one that needs to be pushed on the stack, + * dupped, and not touched otherwise until the constructor runs. + * + */ + +public class BcelShadow extends Shadow { + + private ShadowRange range; + private final BcelWorld world; + private final LazyMethodGen enclosingMethod; + private final BcelShadow enclosingShadow; + + private boolean fallsThrough; + + // ---- initialization + + /** + * This generates an unassociated shadow, rooted in a particular method but not rooted + * to any particular point in the code. It should be given to a rooted ShadowRange + * in the {@link ShadowRange#associateWithShadow(BcelShadow)} method. + */ + public BcelShadow( + BcelWorld world, + Kind kind, + Member signature, + LazyMethodGen enclosingMethod, + BcelShadow enclosingShadow) + { + super(kind, signature); + this.world = world; + this.enclosingMethod = enclosingMethod; + this.enclosingShadow = enclosingShadow; + fallsThrough = kind.argsOnStack(); + } + + // ---- copies all state, including Shadow's mungers... + + public BcelShadow copyInto(LazyMethodGen recipient, BcelShadow enclosing) { + BcelShadow s = new BcelShadow(world, getKind(), getSignature(), recipient, enclosing); + List src = mungers; + List dest = s.mungers; + + for (Iterator i = src.iterator(); i.hasNext(); ) { + dest.add(i.next()); + } + return s; + } + + // ---- overridden behaviour + + public World getIWorld() { + return world; + } + + + + private void deleteNewAndDup() { + final ConstantPoolGen cpg = getEnclosingClass().getConstantPoolGen(); + int depth = 1; + InstructionHandle ih = range.getStart(); + + while (true) { + Instruction inst = ih.getInstruction(); + if (inst instanceof INVOKESPECIAL + && ((INVOKESPECIAL) inst).getName(cpg).equals("<init>")) { + depth++; + } else if (inst instanceof NEW) { + depth--; + if (depth == 0) break; + } + ih = ih.getPrev(); + } + // now IH points to the NEW. We're followed by the DUP, and that is followed + // by the actual instruciton we care about. + InstructionHandle newHandle = ih; + InstructionHandle endHandle = newHandle.getNext(); + InstructionHandle nextHandle; + if (endHandle.getInstruction() instanceof DUP) { + nextHandle = endHandle.getNext(); + retargetFrom(newHandle, nextHandle); + retargetFrom(endHandle, nextHandle); + } else if (endHandle.getInstruction() instanceof DUP_X1) { + InstructionHandle dupHandle = endHandle; + endHandle = endHandle.getNext(); + nextHandle = endHandle.getNext(); + if (endHandle.getInstruction() instanceof SWAP) {} + else { + // XXX see next XXX comment + throw new RuntimeException("Unhandled kind of new " + endHandle); + } + retargetFrom(newHandle, nextHandle); + retargetFrom(dupHandle, nextHandle); + retargetFrom(endHandle, nextHandle); + } else { + // XXX we want to fail gracefully here. This should not be picked out as a join point, + // probably. So change BcelClassWeaver.match appropriately. + throw new RuntimeException("Unhandled kind of new"); + } + // assert (dupHandle.getInstruction() instanceof DUP); + + try { + range.getBody().delete(newHandle, endHandle); + } catch (TargetLostException e) { + throw new BCException("shouldn't happen"); + } + } + private void retargetFrom(InstructionHandle old, InstructionHandle fresh) { + InstructionTargeter[] sources = old.getTargeters(); + if (sources != null) { + for (int i = sources.length - 1; i >= 0; i--) { + sources[i].updateTarget(old, fresh); + } + } + } + + protected void prepareForMungers() { + // if we're a constructor call, we need to remove the new:dup or the new:dup_x1:swap, + // and store all our + // arguments on the frame. + + // ??? This is a bit of a hack (for the Java langauge). We do this because + // we sometime add code "outsideBefore" when dealing with weaving join points. We only + // do this for exposing state that is on the stack. It turns out to just work for + // everything except for constructor calls and exception handlers. If we were to clean + // this up, every ShadowRange would have three instructionHandle points, the start of + // the arg-setup code, the start of the running code, and the end of the running code. + if (getKind() == ConstructorCall) { + deleteNewAndDup(); + initializeArgVars(); + } else if (getKind() == ExceptionHandler) { + ShadowRange range = getRange(); + InstructionList body = range.getBody(); + InstructionHandle start = range.getStart(); + InstructionHandle freshIh = body.insert(start, getFactory().NOP); + InstructionTargeter[] targeters = start.getTargeters(); + for (int i = 0; i < targeters.length; i++) { + InstructionTargeter t = targeters[i]; + if (t instanceof ExceptionRange) { + ExceptionRange er = (ExceptionRange) t; + er.updateTarget(start, freshIh, body); + } + } + } + + // now we ask each munger to request our state + for (Iterator iter = mungers.iterator(); iter.hasNext();) { + ShadowMunger munger = (ShadowMunger) iter.next(); + munger.specializeOn(this); + } + + // If we are an expression kind, we require our target/arguments on the stack + // before we do our actual thing. However, they may have been removed + // from the stack as the shadowMungers have requested state. + // if any of our shadowMungers requested either the arguments or target, + // the munger will have added code + // to pop the target/arguments into temporary variables, represented by + // targetVar and argVars. In such a case, we must make sure to re-push the + // values. + + // If we are nonExpressionKind, we don't expect arguments on the stack + // so this is moot. If our argVars happen to be null, then we know that + // no ShadowMunger has squirrelled away our arguments, so they're still + // on the stack. + InstructionFactory fact = getFactory(); + if (getKind().argsOnStack() && argVars != null) { + range.insert( + BcelRenderer.renderExprs(fact, world, argVars), + Range.InsideBefore); + if (targetVar != null) { + range.insert( + BcelRenderer.renderExpr(fact, world, targetVar), + Range.InsideBefore); + } + if (getKind() == ConstructorCall) { + range.insert((Instruction) fact.createDup(1), Range.InsideBefore); + range.insert( + fact.createNew( + (ObjectType) BcelWorld.makeBcelType( + getSignature().getDeclaringType())), + Range.InsideBefore); + } + } + } + + // ---- getters + + public ShadowRange getRange() { + return range; + } + public void setRange(ShadowRange range) { + this.range = range; + } + + public int getSourceLine() { + if (range == null) return 0; + int ret = Utility.getSourceLine(range.getStart()); + if (ret < 0) return 0; + return ret; + } + + // overrides + public TypeX getEnclosingType() { + return world.resolve(getEnclosingClass().getClassName()); + } + + public LazyClassGen getEnclosingClass() { + return enclosingMethod.getEnclosingClass(); + } + + public BcelWorld getWorld() { + return world; + } + + // ---- factory methods + + public static BcelShadow makeConstructorExecution( + BcelWorld world, + LazyMethodGen enclosingMethod, + InstructionHandle justBeforeStart) + { + final InstructionList body = enclosingMethod.getBody(); + BcelShadow s = + new BcelShadow( + world, + ConstructorExecution, + world.makeMethodSignature(enclosingMethod), + enclosingMethod, + null); + ShadowRange r = new ShadowRange(body); + r.associateWithShadow(s); + r.associateWithTargets( + Range.genStart(body, justBeforeStart.getNext()), + Range.genEnd(body)); + return s; + } + + public static BcelShadow makeStaticInitialization( + BcelWorld world, + LazyMethodGen enclosingMethod) + { + InstructionList body = enclosingMethod.getBody(); + BcelShadow s = + new BcelShadow( + world, + StaticInitialization, + world.makeMethodSignature(enclosingMethod), + enclosingMethod, + null); + ShadowRange r = new ShadowRange(body); + r.associateWithShadow(s); + r.associateWithTargets( + Range.genStart(body), + Range.genEnd(body)); + return s; + } + + /** Make the shadow for an exception handler. Currently makes an empty shadow that + * only allows before advice to be woven into it. + */ + + + public static BcelShadow makeExceptionHandler( + BcelWorld world, + ExceptionRange exceptionRange, + LazyMethodGen enclosingMethod, + InstructionHandle startOfHandler, + BcelShadow enclosingShadow) + { + InstructionList body = enclosingMethod.getBody(); + TypeX catchType = exceptionRange.getCatchType(); + TypeX inType = enclosingMethod.getEnclosingClass().getType(); + BcelShadow s = + new BcelShadow( + world, + ExceptionHandler, + Member.makeExceptionHandlerSignature(inType, catchType), + enclosingMethod, + enclosingShadow); + ShadowRange r = new ShadowRange(body); + r.associateWithShadow(s); + InstructionHandle start = Range.genStart(body, startOfHandler); + InstructionHandle end = Range.genEnd(body, start); + + r.associateWithTargets(start, end); + exceptionRange.updateTarget(startOfHandler, start, body); + return s; + } + + /** create an init join point associated w/ an interface in the body of a constructor */ + + public static BcelShadow makeIfaceInitialization( + BcelWorld world, + LazyMethodGen constructor, + BcelShadow ifaceCExecShadow, + Member interfaceConstructorSignature) + { + InstructionList body = constructor.getBody(); + TypeX inType = constructor.getEnclosingClass().getType(); + BcelShadow s = + new BcelShadow( + world, + Initialization, + interfaceConstructorSignature, + constructor, + null); + s.fallsThrough = true; + ShadowRange r = new ShadowRange(body); + r.associateWithShadow(s); + InstructionHandle start = Range.genStart(body, ifaceCExecShadow.getRange().getStart()); + InstructionHandle end = Range.genEnd(body, ifaceCExecShadow.getRange().getEnd()); + + r.associateWithTargets(start, end); + return s; + } + + public static BcelShadow makeIfaceConstructorExecution( + BcelWorld world, + LazyMethodGen constructor, + InstructionHandle next, + Member interfaceConstructorSignature) + { + final InstructionFactory fact = constructor.getEnclosingClass().getFactory(); + InstructionList body = constructor.getBody(); + TypeX inType = constructor.getEnclosingClass().getType(); + BcelShadow s = + new BcelShadow( + world, + ConstructorExecution, + interfaceConstructorSignature, + constructor, + null); + s.fallsThrough = true; + ShadowRange r = new ShadowRange(body); + r.associateWithShadow(s); + // ??? this may or may not work + InstructionHandle start = Range.genStart(body, next); + //InstructionHandle end = Range.genEnd(body, body.append(start, fact.NOP)); + InstructionHandle end = Range.genStart(body, next); + //body.append(start, fact.NOP); + + r.associateWithTargets(start, end); + return s; + } + + + /** Create an initialization join point associated with a constructor, but not + * with any body of code yet. If this is actually matched, it's range will be set + * when we inline self constructors. + * + * @param constructor The constructor starting this initialization. + */ + public static BcelShadow makeUnfinishedInitialization( + BcelWorld world, + LazyMethodGen constructor) + { + return new BcelShadow( + world, + Initialization, + world.makeMethodSignature(constructor), + constructor, + null); + } + + public static BcelShadow makeUnfinishedPreinitialization( + BcelWorld world, + LazyMethodGen constructor) + { + BcelShadow ret = new BcelShadow( + world, + PreInitialization, + world.makeMethodSignature(constructor), + constructor, + null); + ret.fallsThrough = true; + return ret; + } + + + public static BcelShadow makeMethodExecution( + BcelWorld world, + LazyMethodGen enclosingMethod) + { + return makeShadowForMethod(world, enclosingMethod, MethodExecution, + world.makeMethodSignature(enclosingMethod)); + } + + + public static BcelShadow makeShadowForMethod(BcelWorld world, + LazyMethodGen enclosingMethod, Shadow.Kind kind, Member sig) + { + final InstructionList body = enclosingMethod.getBody(); + BcelShadow s = + new BcelShadow( + world, + kind, + sig, + enclosingMethod, + null); + ShadowRange r = new ShadowRange(body); + r.associateWithShadow(s); + r.associateWithTargets( + Range.genStart(body), + Range.genEnd(body)); + return s; + } + + + + public static BcelShadow makeAdviceExecution( + BcelWorld world, + LazyMethodGen enclosingMethod) + { + final InstructionList body = enclosingMethod.getBody(); + BcelShadow s = + new BcelShadow( + world, + AdviceExecution, + world.makeMethodSignature(enclosingMethod), + enclosingMethod, + null); + ShadowRange r = new ShadowRange(body); + r.associateWithShadow(s); + r.associateWithTargets(Range.genStart(body), Range.genEnd(body)); + return s; + } + + + // constructor call shadows are <em>initially</em> just around the + // call to the constructor. If ANY advice gets put on it, we move + // the NEW instruction inside the join point, which involves putting + // all the arguments in temps. + public static BcelShadow makeConstructorCall( + BcelWorld world, + LazyMethodGen enclosingMethod, + InstructionHandle callHandle, + BcelShadow enclosingShadow) + { + final InstructionList body = enclosingMethod.getBody(); + + Member sig = world.makeMethodSignature( + enclosingMethod.getEnclosingClass(), + (InvokeInstruction) callHandle.getInstruction()); + + BcelShadow s = + new BcelShadow( + world, + ConstructorCall, + sig, + enclosingMethod, + enclosingShadow); + ShadowRange r = new ShadowRange(body); + r.associateWithShadow(s); + r.associateWithTargets( + Range.genStart(body, callHandle), + Range.genEnd(body, callHandle)); + retargetAllBranches(callHandle, r.getStart()); + return s; + } + + public static BcelShadow makeMethodCall( + BcelWorld world, + LazyMethodGen enclosingMethod, + InstructionHandle callHandle, + BcelShadow enclosingShadow) + { + final InstructionList body = enclosingMethod.getBody(); + BcelShadow s = + new BcelShadow( + world, + MethodCall, + world.makeMethodSignature( + enclosingMethod.getEnclosingClass(), + (InvokeInstruction) callHandle.getInstruction()), + enclosingMethod, + enclosingShadow); + ShadowRange r = new ShadowRange(body); + r.associateWithShadow(s); + r.associateWithTargets( + Range.genStart(body, callHandle), + Range.genEnd(body, callHandle)); + retargetAllBranches(callHandle, r.getStart()); + return s; + } + + + public static BcelShadow makeShadowForMethodCall( + BcelWorld world, + LazyMethodGen enclosingMethod, + InstructionHandle callHandle, + BcelShadow enclosingShadow, + Kind kind, + ResolvedMember sig) + { + final InstructionList body = enclosingMethod.getBody(); + BcelShadow s = + new BcelShadow( + world, + kind, + sig, + enclosingMethod, + enclosingShadow); + ShadowRange r = new ShadowRange(body); + r.associateWithShadow(s); + r.associateWithTargets( + Range.genStart(body, callHandle), + Range.genEnd(body, callHandle)); + retargetAllBranches(callHandle, r.getStart()); + return s; + } + + + public static BcelShadow makeFieldGet( + BcelWorld world, + LazyMethodGen enclosingMethod, + InstructionHandle getHandle, + BcelShadow enclosingShadow) + { + final InstructionList body = enclosingMethod.getBody(); + BcelShadow s = + new BcelShadow( + world, + FieldGet, + world.makeFieldSignature( + enclosingMethod.getEnclosingClass(), + (FieldInstruction) getHandle.getInstruction()), + enclosingMethod, + enclosingShadow); + ShadowRange r = new ShadowRange(body); + r.associateWithShadow(s); + r.associateWithTargets( + Range.genStart(body, getHandle), + Range.genEnd(body, getHandle)); + retargetAllBranches(getHandle, r.getStart()); + return s; + } + + public static BcelShadow makeFieldSet( + BcelWorld world, + LazyMethodGen enclosingMethod, + InstructionHandle setHandle, + BcelShadow enclosingShadow) + { + final InstructionList body = enclosingMethod.getBody(); + BcelShadow s = + new BcelShadow( + world, + FieldSet, + world.makeFieldSignature( + enclosingMethod.getEnclosingClass(), + (FieldInstruction) setHandle.getInstruction()), + enclosingMethod, + enclosingShadow); + ShadowRange r = new ShadowRange(body); + r.associateWithShadow(s); + r.associateWithTargets( + Range.genStart(body, setHandle), + Range.genEnd(body, setHandle)); + retargetAllBranches(setHandle, r.getStart()); + return s; + } + + public static void retargetAllBranches(InstructionHandle from, InstructionHandle to) { + InstructionTargeter[] sources = from.getTargeters(); + if (sources != null) { + for (int i = sources.length - 1; i >= 0; i--) { + InstructionTargeter source = sources[i]; + if (source instanceof BranchInstruction) { + source.updateTarget(from, to); + } + } + } + } + + // ---- type access methods + + public boolean hasThis() { + if (getKind() == PreInitialization) return false; + return !getEnclosingCodeSignature().isStatic(); + //???return !enclosingMethod.isStatic(); + } + public TypeX getThisType() { + if (!hasThis()) return ResolvedTypeX.MISSING; + return getEnclosingCodeSignature().getDeclaringType(); + //???return TypeX.forName(getEnclosingClass().getClassName()); + } + + public boolean hasRealTarget() { + return hasTarget() && isExpressionKind(); + } + + + private ObjectType getTargetBcelType() { + return (ObjectType) world.makeBcelType(getTargetType()); + } + private Type getArgBcelType(int arg) { + return world.makeBcelType(getArgType(arg)); + } + + // ---- kinding + + public boolean isExpressionKind() { + if (getKind() == PreInitialization) return true; + return getKind().argsOnStack(); + } + + // ---- argument getting methods + + private BcelVar thisVar = null; + private BcelVar targetVar = null; + private BcelVar[] argVars = null; + + public Var getThisVar() { + if (!hasThis()) { + throw new IllegalStateException("no this"); + } + initializeThisVar(); + return thisVar; + } + public Var getTargetVar() { + if (!hasTarget()) { + throw new IllegalStateException("no target"); + } + initializeTargetVar(); + return targetVar; + } + public Var getArgVar(int i) { + initializeArgVars(); + return argVars[i]; + } + + // reflective thisJoinPoint support + private BcelVar thisJoinPointVar = null; + private BcelVar thisJoinPointStaticPartVar = null; //XXX should be field + private BcelVar thisEnclosingJoinPointStaticPartVar = null; //XXX should be field + + public final Var getThisJoinPointVar() { + return getThisJoinPointBcelVar(); + } + public final Var getThisJoinPointStaticPartVar() { + return getThisJoinPointStaticPartBcelVar(); + } + public final Var getThisEnclosingJoinPointStaticPartVar() { + return getThisEnclosingJoinPointStaticPartBcelVar(); + } + + public BcelVar getThisJoinPointBcelVar() { + if (thisJoinPointVar == null) { + thisJoinPointVar = genTempVar(TypeX.forName("org.aspectj.lang.JoinPoint")); + InstructionFactory fact = getFactory(); + InstructionList il = new InstructionList(); + BcelVar staticPart = getThisJoinPointStaticPartBcelVar(); + staticPart.appendLoad(il, fact); + if (hasThis()) { + ((BcelVar)getThisVar()).appendLoad(il, fact); + } else { + il.append(new ACONST_NULL()); + } + if (hasTarget()) { + ((BcelVar)getTargetVar()).appendLoad(il, fact); + } else { + il.append(new ACONST_NULL()); + } + il.append(makeArgsObjectArray()); + + il.append(fact.createInvoke("org.aspectj.runtime.reflect.Factory", + "makeJP", LazyClassGen.tjpType, + new Type[] { LazyClassGen.staticTjpType, + Type.OBJECT, Type.OBJECT, new ArrayType(Type.OBJECT, 1)}, + Constants.INVOKESTATIC)); + il.append(thisJoinPointVar.createStore(fact)); + range.insert(il, Range.OutsideBefore); + } + return thisJoinPointVar; + } + + public BcelVar getThisJoinPointStaticPartBcelVar() { + if (thisJoinPointStaticPartVar == null) { + Field field = getEnclosingClass().getTjpField(this); + thisJoinPointStaticPartVar = + new BcelFieldRef( + world.resolve(TypeX.forName("org.aspectj.lang.JoinPoint$StaticPart")), + getEnclosingClass().getClassName(), + field.getName()); + } + return thisJoinPointStaticPartVar; + } + + public BcelVar getThisEnclosingJoinPointStaticPartBcelVar() { + if (enclosingShadow == null) { + // the enclosing of an execution is itself + return getThisJoinPointStaticPartBcelVar(); + } else { + return enclosingShadow.getThisJoinPointStaticPartBcelVar(); + } + } + + public Member getEnclosingCodeSignature() { + if (enclosingShadow == null) { + return getSignature(); + } else { + return enclosingShadow.getSignature(); + } + } + + + private InstructionList makeArgsObjectArray() { + InstructionFactory fact = getFactory(); + BcelVar arrayVar = genTempVar(TypeX.OBJECTARRAY); + final InstructionList il = new InstructionList(); + int alen = getArgCount() ; + il.append(Utility.createConstant(fact, alen)); + il.append((Instruction)fact.createNewArray(Type.OBJECT, (short)1)); + arrayVar.appendStore(il, fact); + + int stateIndex = 0; + for (int i = 0, len = getArgCount(); i<len; i++) { + arrayVar.appendConvertableArrayStore(il, fact, stateIndex, (BcelVar)getArgVar(i)); + stateIndex++; + } + arrayVar.appendLoad(il, fact); + return il; + } + + // ---- initializing var tables + + /* initializing this is doesn't do anything, because this + * is protected from side-effects, so we don't need to copy its location + */ + + private void initializeThisVar() { + if (thisVar != null) return; + thisVar = new BcelVar(getThisType().resolve(world), 0); + } + public void initializeTargetVar() { + InstructionFactory fact = getFactory(); + if (targetVar != null) return; + if (! isExpressionKind()) { + initializeThisVar(); + targetVar = thisVar; + } else { + initializeArgVars(); // gotta pop off the args before we find the target + TypeX type = getTargetType(); + targetVar = genTempVar(type, "ajc$target"); + range.insert(targetVar.createStore(fact), Range.OutsideBefore); + } + targetVar.setPositionInAroundState(0); + } + public void initializeArgVars() { + if (argVars != null) return; + InstructionFactory fact = getFactory(); + int len = getArgCount(); + argVars = new BcelVar[len]; + + if (getKind().argsOnStack()) { + // we move backwards because we're popping off the stack + for (int i = len - 1; i >= 0; i--) { + TypeX type = getArgType(i); + BcelVar tmp = genTempVar(type, "ajc$arg" + i); + range.insert(tmp.createStore(getFactory()), Range.OutsideBefore); + int position = i; + if (hasTarget()) position += 1; + tmp.setPositionInAroundState(position); + argVars[i] = tmp; + } + } else { + int index = 0; + if (hasThis()) index++; + for (int i = 0; i < len; i++) { + TypeX type = getArgType(i); + BcelVar tmp = genTempVar(type, "ajc$arg" + i); + range.insert(tmp.createCopyFrom(fact, index), Range.OutsideBefore); + argVars[i] = tmp; + int position = i; + if (hasTarget()) position += 1; + tmp.setPositionInAroundState(position); + index += type.getSize(); + } + } + } + public void initializeForAroundClosure() { + initializeArgVars(); + if (hasTarget()) initializeTargetVar(); + } + + + // ---- weave methods + + void weaveBefore(BcelAdvice munger) { + range.insert( + munger.getAdviceInstructions(this, null, range.getRealStart()), + Range.InsideBefore); + } + + public void weaveAfter(BcelAdvice munger) { + weaveAfterThrowing(munger, TypeX.THROWABLE); + weaveAfterReturning(munger); + } + + /** + * We guarantee that the return value is on the top of the stack when + * munger.getAdviceInstructions() will be run + * (Unless we have a void return type in which case there's nothing) + */ + public void weaveAfterReturning(BcelAdvice munger) { + InstructionFactory fact = getFactory(); + List returns = new ArrayList(); + Instruction ret = null; + for (InstructionHandle ih = range.getStart(); ih != range.getEnd(); ih = ih.getNext()) { + if (ih.getInstruction() instanceof ReturnInstruction) { + returns.add(ih); + ret = ih.getInstruction().copy(); + } + } + InstructionList retList; + InstructionHandle afterAdvice; + if (ret != null) { + retList = new InstructionList(ret); + afterAdvice = retList.getStart(); + } else /* if (munger.hasDynamicTests()) */ { + retList = new InstructionList(fact.NOP); + afterAdvice = retList.getStart(); +// } else { +// retList = new InstructionList(); +// afterAdvice = null; + } + + InstructionList advice = new InstructionList(); + BcelVar tempVar = null; + if (munger.hasExtraParameter()) { + TypeX tempVarType = getReturnType(); + if (tempVarType.equals(ResolvedTypeX.VOID)) { + tempVar = genTempVar(TypeX.OBJECT); + advice.append(getFactory().ACONST_NULL); + tempVar.appendStore(advice, getFactory()); + } else { + tempVar = genTempVar(tempVarType); + advice.append(getFactory().createDup(tempVarType.getSize())); + tempVar.appendStore(advice, getFactory()); + } + } + advice.append(munger.getAdviceInstructions(this, tempVar, afterAdvice)); + + if (ret != null) { + InstructionHandle gotoTarget = advice.getStart(); + for (Iterator i = returns.iterator(); i.hasNext(); ) { + InstructionHandle ih = (InstructionHandle) i.next(); + Utility.replaceInstruction(ih, fact.createBranchInstruction(Constants.GOTO, gotoTarget), enclosingMethod); + } + range.append(advice); + range.append(retList); + } else { + range.append(advice); + range.append(retList); + } + } + + public void weaveAfterThrowing(BcelAdvice munger, TypeX catchType) { + InstructionFactory fact = getFactory(); + InstructionList handler = new InstructionList(); + BcelVar exceptionVar = genTempVar(catchType); + exceptionVar.appendStore(handler, fact); + + + InstructionList endHandler = new InstructionList( + exceptionVar.createLoad(fact)); + handler.append(munger.getAdviceInstructions(this, exceptionVar, endHandler.getStart())); + handler.append(endHandler); + handler.append(fact.ATHROW); + InstructionHandle handlerStart = handler.getStart(); + + if (isFallsThrough()) { + InstructionHandle jumpTarget = handler.append(fact.NOP); + handler.insert(fact.createBranchInstruction(Constants.GOTO, jumpTarget)); + } + InstructionHandle protectedEnd = handler.getStart(); + range.insert(handler, Range.InsideAfter); + + enclosingMethod.addExceptionHandler(range.getStart().getNext(), protectedEnd.getPrev(), + handlerStart, (ObjectType)BcelWorld.makeBcelType(catchType), //???Type.THROWABLE, + // high priority if our args are on the stack + isExpressionKind()); + } + + + public void weaveSoftener(BcelAdvice munger, TypeX catchType) { + InstructionFactory fact = getFactory(); + InstructionList handler = new InstructionList(); + BcelVar exceptionVar = genTempVar(catchType); + exceptionVar.appendStore(handler, fact); + + handler.append(fact.createNew(NameMangler.SOFT_EXCEPTION_TYPE)); + handler.append(fact.createDup(1)); + handler.append(exceptionVar.createLoad(fact)); + handler.append(fact.createInvoke(NameMangler.SOFT_EXCEPTION_TYPE, "<init>", + Type.VOID, new Type[] { Type.THROWABLE }, Constants.INVOKESPECIAL)); //??? special + handler.append(fact.ATHROW); + InstructionHandle handlerStart = handler.getStart(); + + if (isFallsThrough()) { + InstructionHandle jumpTarget = range.getEnd();//handler.append(fact.NOP); + handler.insert(fact.createBranchInstruction(Constants.GOTO, jumpTarget)); + } + InstructionHandle protectedEnd = handler.getStart(); + range.insert(handler, Range.InsideAfter); + + enclosingMethod.addExceptionHandler(range.getStart().getNext(), protectedEnd.getPrev(), + handlerStart, (ObjectType)BcelWorld.makeBcelType(catchType), + // high priority if our args are on the stack + isExpressionKind()); + } + + + public void weavePerObjectEntry(final BcelAdvice munger, final BcelVar onVar) { + final InstructionFactory fact = getFactory(); + + + InstructionList entryInstructions = new InstructionList(); + InstructionList entrySuccessInstructions = new InstructionList(); + onVar.appendLoad(entrySuccessInstructions, fact); + + entrySuccessInstructions.append( + Utility.createInvoke(fact, world, + AjcMemberMaker.perObjectBind(munger.getConcreteAspect()))); + + InstructionList testInstructions = + munger.getTestInstructions(this, entrySuccessInstructions.getStart(), + range.getRealStart(), + entrySuccessInstructions.getStart()); + + entryInstructions.append(testInstructions); + entryInstructions.append(entrySuccessInstructions); + + range.insert(entryInstructions, Range.InsideBefore); + } + + + public void weaveCflowEntry(final BcelAdvice munger, final Member cflowStackField) { + final boolean isPer = munger.getKind() == AdviceKind.PerCflowBelowEntry || + munger.getKind() == AdviceKind.PerCflowEntry; + + final Type objectArrayType = new ArrayType(Type.OBJECT, 1); + final InstructionFactory fact = getFactory(); + + final BcelVar testResult = genTempVar(ResolvedTypeX.BOOLEAN); + + InstructionList entryInstructions = new InstructionList(); + { + InstructionList entrySuccessInstructions = new InstructionList(); + + if (munger.hasDynamicTests()) { + entryInstructions.append(Utility.createConstant(fact, 0)); + testResult.appendStore(entryInstructions, fact); + + entrySuccessInstructions.append(Utility.createConstant(fact, 1)); + testResult.appendStore(entrySuccessInstructions, fact); + } + + if (isPer) { + entrySuccessInstructions.append( + fact.createInvoke(munger.getConcreteAspect().getName(), + NameMangler.PERCFLOW_PUSH_METHOD, + Type.VOID, + new Type[] { }, + Constants.INVOKESTATIC)); + } else { + BcelVar[] cflowStateVars = munger.getExposedStateAsBcelVars(); + + BcelVar arrayVar = genTempVar(TypeX.OBJECTARRAY); + + int alen = cflowStateVars.length; + entrySuccessInstructions.append(Utility.createConstant(fact, alen)); + entrySuccessInstructions.append((Instruction)fact.createNewArray(Type.OBJECT, (short)1)); + arrayVar.appendStore(entrySuccessInstructions, fact); + + for (int i = 0; i < alen; i++) { + arrayVar.appendConvertableArrayStore(entrySuccessInstructions, fact, i, cflowStateVars[i]); + } + + entrySuccessInstructions.append( + Utility.createGet(fact, cflowStackField)); + arrayVar.appendLoad(entrySuccessInstructions, fact); + + entrySuccessInstructions.append( + fact.createInvoke(NameMangler.CFLOW_STACK_TYPE, "push", Type.VOID, + new Type[] { objectArrayType }, + Constants.INVOKEVIRTUAL)); + } + + + InstructionList testInstructions = + munger.getTestInstructions(this, entrySuccessInstructions.getStart(), + range.getRealStart(), + entrySuccessInstructions.getStart()); + + entryInstructions.append(testInstructions); + entryInstructions.append(entrySuccessInstructions); + } + + // this is the same for both per and non-per + weaveAfter(new BcelAdvice(null, null, null, 0, 0, 0, null, null) { + public InstructionList getAdviceInstructions(BcelShadow s, BcelVar extraArgVar, InstructionHandle ifNoAdvice) { + InstructionList exitInstructions = new InstructionList(); + if (munger.hasDynamicTests()) { + testResult.appendLoad(exitInstructions, fact); + exitInstructions.append(fact.createBranchInstruction(Constants.IFEQ, ifNoAdvice)); + } + exitInstructions.append( + Utility.createGet(fact, cflowStackField)); + exitInstructions.append( + fact.createInvoke(NameMangler.CFLOW_STACK_TYPE, "pop", Type.VOID, new Type[] {}, Constants.INVOKEVIRTUAL)); + return exitInstructions; + }}); + + + range.insert(entryInstructions, Range.InsideBefore); + } + + public void weaveAroundInline( + BcelAdvice munger, + boolean hasDynamicTest) + { + /* Implementation notes: + * + * AroundInline still extracts the instructions of the original shadow into + * an extracted method. This allows inlining of even that advice that doesn't + * call proceed or calls proceed more than once. + * + * It extracts the instructions of the original shadow into a method. + * + * Then it inlines the instructions of the advice in its place, taking care + * to treat the closure argument specially (it doesn't exist). + * + * Then it searches in the instructions of the advice for any call to the + * proceed method. + * + * At such a call, there is stuff on the stack representing the arguments to + * proceed. Pop these into the frame. + * + * Now build the stack for the call to the extracted method, taking values + * either from the join point state or from the new frame locs from proceed. + * Now call the extracted method. The right return value should be on the + * stack, so no cast is necessary. + * + * If only one call to proceed is made, we can re-inline the original shadow. + * We are not doing that presently. + */ + + // start by exposing various useful things into the frame + final InstructionFactory fact = getFactory(); + + // now generate the aroundBody method + LazyMethodGen extractedMethod = + extractMethod( + NameMangler.aroundCallbackMethodName( + getSignature(), + getEnclosingClass())); + + // the shadow is now empty. First, create a correct call + // to the around advice. This includes both the call (which may involve + // value conversion of the advice arguments) and the return + // (which may involve value conversion of the return value). Right now + // we push a null for the unused closure. It's sad, but there it is. + + InstructionList advice = new InstructionList(); + InstructionHandle adviceMethodInvocation; + { + // ??? we don't actually need to push NULL for the closure if we take care + advice.append(munger.getAdviceArgSetup(this, null, new InstructionList(fact.ACONST_NULL))); + adviceMethodInvocation = + advice.append( + Utility.createInvoke(fact, getWorld(), munger.getSignature())); + advice.append( + Utility.createConversion( + getFactory(), + world.makeBcelType(munger.getSignature().getReturnType()), + extractedMethod.getReturnType())); + if (! isFallsThrough()) { + advice.append(fact.createReturn(extractedMethod.getReturnType())); + } + } + + // now, situate the call inside the possible dynamic tests, + // and actually add the whole mess to the shadow + if (! hasDynamicTest) { + range.append(advice); + } else { + InstructionList callback = makeCallToCallback(extractedMethod); + if (! isExpressionKind()) { + callback.append(fact.createReturn(extractedMethod.getReturnType())); + } else { + advice.append(fact.createBranchInstruction(Constants.GOTO, range.getEnd())); + } + range.append(munger.getTestInstructions(this, advice.getStart(), callback.getStart(), advice.getStart())); + range.append(advice); + range.append(callback); + } + + // now the range contains everything we need. We now inline the advice method. + LazyMethodGen adviceMethod = + ((BcelObjectType) munger.getConcreteAspect()) + .getLazyClassGen() + .getLazyMethodGen(munger.getSignature()); + + BcelClassWeaver.inlineMethod(adviceMethod, enclosingMethod, adviceMethodInvocation); + + // now search through the advice, looking for a call to PROCEED. + // Then we replace the call to proceed with some argument setup, and a + // call to the extracted method. + String proceedName = + NameMangler.proceedMethodName(munger.getSignature().getName()); + + InstructionHandle curr = getRange().getStart(); + InstructionHandle end = getRange().getEnd(); + ConstantPoolGen cpg = extractedMethod.getEnclosingClass().getConstantPoolGen(); + while (curr != end) { + InstructionHandle next = curr.getNext(); + Instruction inst = curr.getInstruction(); + if ((inst instanceof INVOKESTATIC) + && proceedName.equals(((INVOKESTATIC) inst).getMethodName(cpg))) { + + + enclosingMethod.getBody().append(curr, getRedoneProceedCall(fact, extractedMethod, munger)); + Utility.deleteInstruction(curr, enclosingMethod); + } + curr = next; + } + // and that's it. + } + + private InstructionList getRedoneProceedCall( + InstructionFactory fact, + LazyMethodGen callbackMethod, + BcelAdvice munger) + { + InstructionList ret = new InstructionList(); + // we have on stack all the arguments for the ADVICE call. + // we have in frame somewhere all the arguments for the non-advice call. + + List argVarList = new ArrayList(); + + // start w/ stuff + if (targetVar != null) argVarList.add(targetVar); + for (int i = 0, len = getArgCount(); i < len; i++) { + argVarList.add(argVars[i]); + } + + BcelVar[] adviceVars = munger.getExposedStateAsBcelVars(); + //??? this is too easy +// for (int i=0; i < adviceVars.length; i++) { +// if (adviceVars[i] != null) +// adviceVars[i].setPositionInAroundState(i); +// } + + + IntMap proceedMap = makeProceedArgumentMap(adviceVars); + +// System.out.println(proceedMap); +// System.out.println(Arrays.asList(adviceVars)); + + ResolvedTypeX[] proceedParamTypes = world.resolve(munger.getSignature().getParameterTypes()); + + BcelVar[] proceedVars = + Utility.pushAndReturnArrayOfVars(proceedParamTypes, ret, fact, enclosingMethod); + + + Type[] stateTypes = callbackMethod.getArgumentTypes(); + + 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( + BcelAdvice munger, + boolean hasDynamicTest) + { + InstructionFactory fact = getFactory(); + + // MOVE OUT ALL THE INSTRUCTIONS IN MY SHADOW INTO ANOTHER METHOD! + LazyMethodGen callbackMethod = + extractMethod( + NameMangler.aroundCallbackMethodName( + getSignature(), + getEnclosingClass())); + + BcelVar[] adviceVars = munger.getExposedStateAsBcelVars(); + + String closureClassName = + NameMangler.makeClosureClassName( + getEnclosingClass().getType(), + getEnclosingClass().getNewGeneratedNameTag()); + + Member constructorSig = new Member(Member.CONSTRUCTOR, + TypeX.forName(closureClassName), 0, "<init>", + "([Ljava/lang/Object;)V"); + + BcelVar closureHolder = null; + + // This is not being used currently since getKind() == preinitializaiton + // cannot happen in around advice + if (getKind() == PreInitialization) { + closureHolder = genTempVar(AjcMemberMaker.AROUND_CLOSURE_TYPE); + } + + InstructionList closureInstantiation = + makeClosureInstantiation(constructorSig, closureHolder); + + LazyMethodGen constructor = + makeClosureClassAndReturnConstructor( + closureClassName, + callbackMethod, + makeProceedArgumentMap(adviceVars) + ); + + InstructionList returnConversionCode; + if (getKind() == PreInitialization) { + returnConversionCode = new InstructionList(); + + BcelVar stateTempVar = genTempVar(TypeX.OBJECTARRAY); + closureHolder.appendLoad(returnConversionCode, fact); + + returnConversionCode.append( + Utility.createInvoke( + fact, + world, + AjcMemberMaker.aroundClosurePreInitializationGetter())); + stateTempVar.appendStore(returnConversionCode, fact); + + Type[] stateTypes = getSuperConstructorParameterTypes(); + + returnConversionCode.append(fact.ALOAD_0); // put "this" back on the stack + for (int i = 0, len = stateTypes.length; i < len; i++) { + stateTempVar.appendConvertableArrayLoad( + returnConversionCode, + fact, + i, + world.resolve(BcelWorld.fromBcel(stateTypes[i]))); + } + } else { + returnConversionCode = + Utility.createConversion( + getFactory(), + world.makeBcelType(munger.getSignature().getReturnType()), + callbackMethod.getReturnType()); + if (! isFallsThrough()) { + returnConversionCode.append(fact.createReturn(callbackMethod.getReturnType())); + } + } + + InstructionList advice = new InstructionList(); + advice.append(munger.getAdviceArgSetup(this, null, closureInstantiation)); +// advice.append(closureInstantiation); + advice.append(munger.getNonTestAdviceInstructions(this)); + advice.append(returnConversionCode); + + if (! hasDynamicTest) { + range.append(advice); + } else { + InstructionList callback = makeCallToCallback(callbackMethod); + InstructionList postCallback = new InstructionList(); + if (! isExpressionKind()) { + callback.append(fact.createReturn(callbackMethod.getReturnType())); + } else { + advice.append(fact.createBranchInstruction(Constants.GOTO, postCallback.append(fact.NOP))); + } + range.append(munger.getTestInstructions(this, advice.getStart(), callback.getStart(), advice.getStart())); + range.append(advice); + range.append(callback); + range.append(postCallback); + } + } + + // XXX only used for testing + InstructionList makeCallToCallback(LazyMethodGen callbackMethod) { + InstructionFactory fact = getFactory(); + InstructionList callback = new InstructionList(); + if (targetVar != null) { + callback.append(BcelRenderer.renderExpr(fact, world, targetVar)); + } + callback.append(BcelRenderer.renderExprs(fact, world, argVars)); + callback.append(Utility.createInvoke(fact, callbackMethod)); + return callback; + } + + /** side-effect-free */ + private InstructionList makeClosureInstantiation(Member constructor, BcelVar holder) { + +// LazyMethodGen constructor) { + InstructionFactory fact = getFactory(); + BcelVar arrayVar = genTempVar(TypeX.OBJECTARRAY); + //final Type objectArrayType = new ArrayType(Type.OBJECT, 1); + final InstructionList il = new InstructionList(); + int alen = getArgCount() + (targetVar == null ? 0 : 1) + (thisJoinPointVar == null ? 0 : 1); + il.append(Utility.createConstant(fact, alen)); + il.append((Instruction)fact.createNewArray(Type.OBJECT, (short)1)); + arrayVar.appendStore(il, fact); + + int stateIndex = 0; + if (targetVar != null) { + arrayVar.appendConvertableArrayStore(il, fact, stateIndex, targetVar); + targetVar.setPositionInAroundState(stateIndex); + stateIndex++; + } + for (int i = 0, len = getArgCount(); i<len; i++) { + arrayVar.appendConvertableArrayStore(il, fact, stateIndex, argVars[i]); + argVars[i].setPositionInAroundState(stateIndex); + stateIndex++; + } + if (thisJoinPointVar != null) { + arrayVar.appendConvertableArrayStore(il, fact, stateIndex, thisJoinPointVar); + thisJoinPointVar.setPositionInAroundState(stateIndex); + stateIndex++; + } + il.append(fact.createNew(new ObjectType(constructor.getDeclaringType().getName()))); + il.append(new DUP()); + arrayVar.appendLoad(il, fact); + il.append(Utility.createInvoke(fact, world, constructor)); + if (getKind() == PreInitialization) { + il.append(fact.DUP); + holder.appendStore(il, fact); + } + return il; + } + + + private IntMap makeProceedArgumentMap(BcelVar[] adviceArgs) { + //System.err.println("coming in with " + Arrays.asList(adviceArgs)); + + IntMap ret = new IntMap(); + for(int i = 0, len = adviceArgs.length; i < len; i++) { + BcelVar v = (BcelVar) adviceArgs[i]; + if (v == null) continue; // XXX we don't know why this is required + int pos = v.getPositionInAroundState(); + if (pos >= 0) { // need this test to avoid args bound via cflow + ret.put(pos, i); + } + } + //System.err.println("returning " + ret); + + return ret; + } + + /** + * + * + * @param callbackMethod the method we will call back to when our run method gets called. + * + * @param proceedMap A map from state position to proceed argument position. May be + * non covering on state position. + */ + + private LazyMethodGen makeClosureClassAndReturnConstructor( + String closureClassName, + LazyMethodGen callbackMethod, + IntMap proceedMap) + { + String superClassName = "org.aspectj.runtime.internal.AroundClosure"; + Type objectArrayType = new ArrayType(Type.OBJECT, 1); + + LazyClassGen closureClass = new LazyClassGen(closureClassName, + superClassName, + getEnclosingClass().getFileName(), + Modifier.PUBLIC, + new String[] {}); + InstructionFactory fact = new InstructionFactory(closureClass.getConstantPoolGen()); + + // constructor + LazyMethodGen constructor = new LazyMethodGen(Modifier.PUBLIC, + Type.VOID, + "<init>", + new Type[] {objectArrayType}, + new String[] {}, + closureClass); + InstructionList cbody = constructor.getBody(); + cbody.append(fact.createLoad(Type.OBJECT, 0)); + cbody.append(fact.createLoad(objectArrayType, 1)); + cbody.append(fact.createInvoke(superClassName, "<init>", Type.VOID, + new Type[] {objectArrayType}, Constants.INVOKESPECIAL)); + cbody.append(fact.createReturn(Type.VOID)); + + closureClass.addMethodGen(constructor); + + // method + LazyMethodGen runMethod = new LazyMethodGen(Modifier.PUBLIC, + Type.OBJECT, + "run", + new Type[] {objectArrayType}, + new String[] {}, + closureClass); + InstructionList mbody = runMethod.getBody(); + BcelVar proceedVar = new BcelVar(TypeX.OBJECTARRAY.resolve(world), 1); + // int proceedVarIndex = 1; + BcelVar stateVar = new BcelVar(TypeX.OBJECTARRAY.resolve(world), runMethod.allocateLocal(1)); + // int stateVarIndex = runMethod.allocateLocal(1); + mbody.append(fact.createThis()); + mbody.append(fact.createGetField(superClassName, "state", objectArrayType)); + mbody.append(stateVar.createStore(fact)); + // mbody.append(fact.createStore(objectArrayType, stateVarIndex)); + + Type[] stateTypes = callbackMethod.getArgumentTypes(); + + 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)) { + mbody.append( + proceedVar.createConvertableArrayLoad(fact, proceedMap.get(i), + stateTypeX)); + } else { + mbody.append( + stateVar.createConvertableArrayLoad(fact, i, + stateTypeX)); + } + } + + + mbody.append(Utility.createInvoke(fact, callbackMethod)); + + if (getKind() == PreInitialization) { + mbody.append(Utility.createSet( + fact, + AjcMemberMaker.aroundClosurePreInitializationField())); + mbody.append(fact.ACONST_NULL); + } else { + mbody.append( + Utility.createConversion( + fact, + callbackMethod.getReturnType(), + Type.OBJECT)); + } + mbody.append(fact.createReturn(Type.OBJECT)); + + closureClass.addMethodGen(runMethod); + + // class + getEnclosingClass().addGeneratedInner(closureClass); + + return constructor; + } + + // ---- extraction methods + + + public LazyMethodGen extractMethod(String newMethodName) { + LazyMethodGen.assertGoodBody(range.getBody(), newMethodName); + if (!getKind().allowsExtraction()) throw new BCException(); + LazyMethodGen freshMethod = createMethodGen(newMethodName); + +// System.err.println("******"); +// System.err.println("ABOUT TO EXTRACT METHOD for" + this); +// enclosingMethod.print(System.err); +// System.err.println("INTO"); +// freshMethod.print(System.err); +// System.err.println("WITH REMAP"); +// System.err.println(makeRemap()); + + range.extractInstructionsInto(freshMethod, makeRemap(), + (getKind() != PreInitialization) && + isFallsThrough()); + if (getKind() == PreInitialization) { + addPreInitializationReturnCode( + freshMethod, + getSuperConstructorParameterTypes()); + } + getEnclosingClass().addMethodGen(freshMethod); + + return freshMethod; + } + + private void addPreInitializationReturnCode( + LazyMethodGen extractedMethod, + Type[] superConstructorTypes) + { + InstructionList body = extractedMethod.getBody(); + final InstructionFactory fact = getFactory(); + + BcelVar arrayVar = new BcelVar( + world.resolve(TypeX.OBJECTARRAY), + extractedMethod.allocateLocal(1)); + + int len = superConstructorTypes.length; + + body.append(Utility.createConstant(fact, len)); + + body.append((Instruction)fact.createNewArray(Type.OBJECT, (short)1)); + arrayVar.appendStore(body, fact); + + for (int i = len - 1; i >= 0; i++) { + // convert thing on top of stack to object + body.append( + Utility.createConversion(fact, superConstructorTypes[i], Type.OBJECT)); + // push object array + arrayVar.appendLoad(body, fact); + // swap + body.append(fact.SWAP); + // do object array store. + body.append(Utility.createConstant(fact, i)); + body.append(fact.SWAP); + body.append(fact.createArrayStore(Type.OBJECT)); + } + arrayVar.appendLoad(body, fact); + body.append(fact.ARETURN); + } + + private Type[] getSuperConstructorParameterTypes() { + // assert getKind() == PreInitialization + InstructionHandle superCallHandle = getRange().getEnd().getNext(); + InvokeInstruction superCallInstruction = + (InvokeInstruction) superCallHandle.getInstruction(); + return superCallInstruction.getArgumentTypes( + getEnclosingClass().getConstantPoolGen()); + } + + + /** make a map from old frame location to new frame location. Any unkeyed frame + * location picks out a copied local */ + private IntMap makeRemap() { + IntMap ret = new IntMap(5); + int reti = 0; + if (targetVar != null) { + ret.put(targetVar.getSlot(), reti++); + } + for (int i = 0, len = argVars.length; i < len; i++) { + ret.put(argVars[i].getSlot(), reti); + reti += argVars[i].getType().getSize(); + } + if (thisJoinPointVar != null) { + ret.put(thisJoinPointVar.getSlot(), reti++); + } + // we not only need to put the arguments, we also need to remap their + // aliases, which we so helpfully put into temps at the beginning of this join + // point. + if (! getKind().argsOnStack()) { + int index = 0; + if (hasThis()) { ret.put(0, 0); index++; } + for (int i = 0; i < getArgCount(); i++) { + TypeX type = getArgType(i); + ret.put(index, index); + index += type.getSize(); + } + } + return ret; + } + + /** + * The new method is nonStatic iff we're from nonExpression advice + * with an enclosing non-static method. + * Otherwise, it's static + */ + private LazyMethodGen createMethodGen(String newMethodName) { + Type[] parameterTypes = world.makeBcelTypes(getSignature().getParameterTypes()); + int modifiers = Modifier.FINAL; + + // XXX some bug +// if (! isExpressionKind() && getSignature().isStrict(world)) { +// modifiers |= Modifier.STRICT; +// } + modifiers |= Modifier.STATIC; + if (hasTarget()) { + TypeX targetType = getTargetType(); + parameterTypes = addType(world.makeBcelType(targetType), parameterTypes); + } + + // We always want to pass down thisJoinPoint in case we have already woven + // some advice in here. If we only have a single piece of around advice on a + // join point, it is unnecessary to accept (and pass) tjp. + if (thisJoinPointVar != null) { + parameterTypes = addTypeToEnd(LazyClassGen.tjpType, parameterTypes); + } + + TypeX returnType; + if (getKind() == ConstructorCall) { + returnType = getSignature().getDeclaringType(); + } else if (getKind() == PreInitialization) { + returnType = TypeX.OBJECTARRAY; + } else { + returnType = getSignature().getReturnType(); + } + return + new LazyMethodGen( + modifiers, + world.makeBcelType(returnType), + newMethodName, + parameterTypes, + new String[0], + // XXX again, we need to look up methods! +// TypeX.getNames(getSignature().getExceptions(world)), + getEnclosingClass()); + } + + private Type[] addType(Type type, Type[] types) { + int len = types.length; + Type[] ret = new Type[len+1]; + ret[0] = type; + System.arraycopy(types, 0, ret, 1, len); + return ret; + } + + private Type[] addTypeToEnd(Type type, Type[] types) { + int len = types.length; + Type[] ret = new Type[len+1]; + ret[len] = type; + System.arraycopy(types, 0, ret, 0, len); + return ret; + } + + public BcelVar genTempVar(TypeX typeX) { + return new BcelVar(typeX.resolve(world), genTempVarIndex(typeX.getSize())); + } + +// public static final boolean CREATE_TEMP_NAMES = true; + + public BcelVar genTempVar(TypeX typeX, String localName) { + BcelVar tv = genTempVar(typeX); + +// if (CREATE_TEMP_NAMES) { +// for (InstructionHandle ih = range.getStart(); ih != range.getEnd(); ih = ih.getNext()) { +// if (Range.isRangeHandle(ih)) continue; +// ih.addTargeter(new LocalVariableTag(typeX, localName, tv.getSlot())); +// } +// } + return tv; + } + + // eh doesn't think we need to garbage collect these (64K is a big number...) + private int genTempVarIndex(int size) { + return enclosingMethod.allocateLocal(size); + } + + public InstructionFactory getFactory() { + return getEnclosingClass().getFactory(); + } + + public SourceLocation getSourceLocation() { + return new SourceLocation(new File(getEnclosingClass().getFileName()), getSourceLine()); + } + + public BcelShadow getEnclosingShadow() { + return enclosingShadow; + } + + public LazyMethodGen getEnclosingMethod() { + return enclosingMethod; + } + + public boolean isFallsThrough() { + return fallsThrough; + } +} diff --git a/weaver/src/org/aspectj/weaver/bcel/BcelSourceContext.java b/weaver/src/org/aspectj/weaver/bcel/BcelSourceContext.java new file mode 100644 index 000000000..aedfe074b --- /dev/null +++ b/weaver/src/org/aspectj/weaver/bcel/BcelSourceContext.java @@ -0,0 +1,55 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.bcel; + +import java.io.File; +import java.util.Arrays; + +import org.aspectj.bridge.*; +import org.aspectj.weaver.*; +import org.aspectj.weaver.ISourceContext; +import org.aspectj.weaver.AjAttribute.SourceContextAttribute; + +public class BcelSourceContext implements ISourceContext { + private BcelObjectType inObject; + private String sourceFileName; + private int[] lineBreaks; + + public BcelSourceContext(BcelObjectType inObject) { + this.inObject = inObject; + } + + + public ISourceLocation makeSourceLocation(IHasPosition position) { + String fileName = sourceFileName; + if (fileName == null) inObject.getJavaClass().getFileName(); + if (fileName == null) fileName = inObject.getName() + ".class"; + + if (lineBreaks != null) { + int line = Arrays.binarySearch(lineBreaks, position.getStart()); + if (line < 0) line = -line; + return new SourceLocation(new File(fileName), line); //??? have more info + } else { + return new SourceLocation(new File(fileName), 0); + } + } + + + + public void addAttributeInfo(SourceContextAttribute sourceContextAttribute) { + this.sourceFileName = sourceContextAttribute.getSourceFileName(); + this.lineBreaks = sourceContextAttribute.getLineBreaks(); + } + +} diff --git a/weaver/src/org/aspectj/weaver/bcel/BcelTypeMunger.java b/weaver/src/org/aspectj/weaver/bcel/BcelTypeMunger.java new file mode 100644 index 000000000..a14f0e0b3 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/bcel/BcelTypeMunger.java @@ -0,0 +1,566 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.bcel; + +import java.lang.reflect.Modifier; +import java.util.*; + +import org.apache.bcel.Constants; +import org.apache.bcel.generic.*; +import org.aspectj.weaver.*; +import org.aspectj.weaver.patterns.*; + + +//XXX addLazyMethodGen is probably bad everywhere +public class BcelTypeMunger extends ConcreteTypeMunger { + + public BcelTypeMunger(ResolvedTypeMunger munger, ResolvedTypeX aspectType) { + super(munger, aspectType); + } + + public String toString() { + return "(BcelTypeMunger " + getMunger() + ")"; + } + + public boolean munge(BcelClassWeaver weaver) { + if (munger.getKind() == ResolvedTypeMunger.Field) { + return mungeNewField(weaver, (NewFieldTypeMunger)munger); + } else if (munger.getKind() == ResolvedTypeMunger.Method) { + return mungeNewMethod(weaver, (NewMethodTypeMunger)munger); + } else if (munger.getKind() == ResolvedTypeMunger.PerObjectInterface) { + return mungePerObjectInterface(weaver, (PerObjectInterfaceTypeMunger)munger); + } else if (munger.getKind() == ResolvedTypeMunger.PrivilegedAccess) { + return mungePrivilegedAccess(weaver, (PrivilegedAccessMunger)munger); + } else if (munger.getKind() == ResolvedTypeMunger.Constructor) { + return mungeNewConstructor(weaver, (NewConstructorTypeMunger)munger); + } else { + throw new RuntimeException("unimplemented"); + } + } + + + + private boolean mungePrivilegedAccess( + BcelClassWeaver weaver, + PrivilegedAccessMunger munger) + { + LazyClassGen gen = weaver.getLazyClassGen(); + ResolvedMember member = munger.getMember(); + + ResolvedTypeX onType = weaver.getWorld().resolve(member.getDeclaringType()); + //System.out.println("munging: " + gen + " with " + member); + if (onType.equals(gen.getType())) { + if (member.getKind() == Member.FIELD) { + //System.out.println("matched: " + gen); + addFieldGetter(gen, member, + AjcMemberMaker.privilegedAccessMethodForFieldGet(aspectType, member)); + addFieldSetter(gen, member, + AjcMemberMaker.privilegedAccessMethodForFieldSet(aspectType, member)); + return true; + } else if (member.getKind() == Member.METHOD) { + addMethodDispatch(gen, member, + AjcMemberMaker.privilegedAccessMethodForMethod(aspectType, member)); + } else { + throw new RuntimeException("unimplemented"); + } + } + return false; + } + + private void addFieldGetter( + LazyClassGen gen, + ResolvedMember field, + ResolvedMember accessMethod) + { + LazyMethodGen mg = makeMethodGen(gen, accessMethod); + InstructionList il = new InstructionList(); + InstructionFactory fact = gen.getFactory(); + if (field.isStatic()) { + il.append(fact.createFieldAccess( + gen.getClassName(), + field.getName(), + BcelWorld.makeBcelType(field.getType()), Constants.GETSTATIC)); + } else { + il.append(fact.ALOAD_0); + il.append(fact.createFieldAccess( + gen.getClassName(), + field.getName(), + BcelWorld.makeBcelType(field.getType()), Constants.GETFIELD)); + } + il.append(fact.createReturn(BcelWorld.makeBcelType(field.getType()))); + mg.getBody().insert(il); + + gen.addMethodGen(mg); + } + + private void addFieldSetter( + LazyClassGen gen, + ResolvedMember field, + ResolvedMember accessMethod) + { + LazyMethodGen mg = makeMethodGen(gen, accessMethod); + InstructionList il = new InstructionList(); + InstructionFactory fact = gen.getFactory(); + Type fieldType = BcelWorld.makeBcelType(field.getType()); + + if (field.isStatic()) { + il.append(fact.createLoad(fieldType, 0)); + il.append(fact.createFieldAccess( + gen.getClassName(), + field.getName(), + fieldType, Constants.PUTSTATIC)); + } else { + il.append(fact.ALOAD_0); + il.append(fact.createLoad(fieldType, 1)); + il.append(fact.createFieldAccess( + gen.getClassName(), + field.getName(), + fieldType, Constants.PUTFIELD)); + } + il.append(fact.createReturn(Type.VOID)); + mg.getBody().insert(il); + + gen.addMethodGen(mg); + } + + private void addMethodDispatch( + LazyClassGen gen, + ResolvedMember method, + ResolvedMember accessMethod) + { + LazyMethodGen mg = makeMethodGen(gen, accessMethod); + InstructionList il = new InstructionList(); + InstructionFactory fact = gen.getFactory(); + //Type fieldType = BcelWorld.makeBcelType(field.getType()); + Type[] paramTypes = BcelWorld.makeBcelTypes(method.getParameterTypes()); + + int pos = 0; + + if (!method.isStatic()) { + il.append(fact.ALOAD_0); + pos++; + } + for (int i = 0, len = paramTypes.length; i < len; i++) { + Type paramType = paramTypes[i]; + il.append(fact.createLoad(paramType, pos)); + pos+=paramType.getSize(); + } + il.append(Utility.createInvoke(fact, (BcelWorld)aspectType.getWorld(), + method)); + il.append(fact.createReturn(BcelWorld.makeBcelType(method.getReturnType()))); + + mg.getBody().insert(il); + + gen.addMethodGen(mg); + } + + + + private LazyMethodGen makeMethodGen(LazyClassGen gen, ResolvedMember member) { + return new LazyMethodGen( + member.getModifiers(), + BcelWorld.makeBcelType(member.getReturnType()), + member.getName(), + BcelWorld.makeBcelTypes(member.getParameterTypes()), + TypeX.getNames(member.getExceptions()), + gen); + } + + + private FieldGen makeFieldGen(LazyClassGen gen, ResolvedMember member) { + return new FieldGen( + member.getModifiers(), + BcelWorld.makeBcelType(member.getReturnType()), + member.getName(), + gen.getConstantPoolGen()); + } + + + + + private boolean mungePerObjectInterface( + BcelClassWeaver weaver, + PerObjectInterfaceTypeMunger munger) + { + LazyClassGen gen = weaver.getLazyClassGen(); + + if (couldMatch(gen.getType(), munger.getTestPointcut())) { + FieldGen fg = makeFieldGen(gen, + AjcMemberMaker.perObjectField(gen.getType(), aspectType)); + + gen.addField(fg.getField()); + + + Type fieldType = BcelWorld.makeBcelType(aspectType); + LazyMethodGen mg = new LazyMethodGen( + Modifier.PUBLIC, + fieldType, + NameMangler.perObjectInterfaceGet(aspectType), + new Type[0], new String[0], + gen); + InstructionList il = new InstructionList(); + InstructionFactory fact = gen.getFactory(); + il.append(fact.ALOAD_0); + il.append(fact.createFieldAccess( + gen.getClassName(), + fg.getName(), + fieldType, Constants.GETFIELD)); + il.append(fact.createReturn(fieldType)); + mg.getBody().insert(il); + + gen.addMethodGen(mg); + + LazyMethodGen mg1 = new LazyMethodGen( + Modifier.PUBLIC, + Type.VOID, + NameMangler.perObjectInterfaceSet(aspectType), + + new Type[]{fieldType,}, new String[0], + gen); + InstructionList il1 = new InstructionList(); + il1.append(fact.ALOAD_0); + il1.append(fact.createLoad(fieldType, 1)); + il1.append(fact.createFieldAccess( + gen.getClassName(), + fg.getName(), + fieldType, Constants.PUTFIELD)); + il1.append(fact.createReturn(Type.VOID)); + mg1.getBody().insert(il1); + + gen.addMethodGen(mg1); + + gen.addInterface(munger.getInterfaceType()); + + return true; + } else { + return false; + } + } + + private boolean couldMatch( + BcelObjectType bcelObjectType, + Pointcut pointcut) { + return !bcelObjectType.isInterface(); + } + + private boolean mungeNewMethod(BcelClassWeaver weaver, NewMethodTypeMunger munger) { + ResolvedMember signature = munger.getSignature(); + ResolvedMember dispatchMethod = munger.getDispatchMethod(aspectType); + + LazyClassGen gen = weaver.getLazyClassGen(); + + ResolvedTypeX onType = weaver.getWorld().resolve(signature.getDeclaringType()); + boolean onInterface = onType.isInterface(); + + if (onType.equals(gen.getType())) { + ResolvedMember introMethod = + AjcMemberMaker.interMethod(signature, aspectType, onInterface); + + LazyMethodGen mg = makeMethodGen(gen, introMethod); + + if (!onInterface && !Modifier.isAbstract(introMethod.getModifiers())) { + InstructionList body = mg.getBody(); + InstructionFactory fact = gen.getFactory(); + int pos = 0; + + if (!signature.isStatic()) { + body.append(fact.createThis()); + pos++; + } + Type[] paramTypes = BcelWorld.makeBcelTypes(introMethod.getParameterTypes()); + for (int i = 0, len = paramTypes.length; i < len; i++) { + Type paramType = paramTypes[i]; + body.append(fact.createLoad(paramType, pos)); + pos+=paramType.getSize(); + } + body.append(Utility.createInvoke(fact, weaver.getWorld(), dispatchMethod)); + body.append(fact.createReturn(BcelWorld.makeBcelType(introMethod.getReturnType()))); + } else { + //??? this is okay + //if (!(mg.getBody() == null)) throw new RuntimeException("bas"); + } + + + // XXX make sure to check that we set exceptions properly on this guy. + weaver.addLazyMethodGen(mg); + + Set neededSuperCalls = munger.getSuperMethodsCalled(); + + for (Iterator iter = neededSuperCalls.iterator(); iter.hasNext(); ) { + ResolvedMember superMethod = (ResolvedMember) iter.next(); + if (weaver.addDispatchTarget(superMethod)) { + String dispatchName = genSuperDispatchName(onType, superMethod); + LazyMethodGen dispatcher = makeDispatcher(gen, dispatchName, superMethod, weaver.getWorld()); + + weaver.addLazyMethodGen(dispatcher); + } + } + + return true; + } else if (onInterface && gen.getType().isTopmostImplementor(onType) && + !Modifier.isAbstract(signature.getModifiers())) + { + ResolvedMember introMethod = + AjcMemberMaker.interMethod(signature, aspectType, false); + + LazyMethodGen mg = makeMethodGen(gen, introMethod); + + // + + Type[] paramTypes = BcelWorld.makeBcelTypes(introMethod.getParameterTypes()); + Type returnType = BcelWorld.makeBcelType(introMethod.getReturnType()); + + InstructionList body = mg.getBody(); + InstructionFactory fact = gen.getFactory(); + int pos = 0; + + if (!introMethod.isStatic()) { + body.append(fact.createThis()); + pos++; + } + for (int i = 0, len = paramTypes.length; i < len; i++) { + Type paramType = paramTypes[i]; + body.append(fact.createLoad(paramType, pos)); + pos+=paramType.getSize(); + } + body.append(Utility.createInvoke(fact, weaver.getWorld(), dispatchMethod)); + body.append(fact.createReturn(returnType)); + mg.definingType = onType; + + weaver.addOrReplaceLazyMethodGen(mg); + + return true; + } else { + return false; + } + } + + + private boolean mungeNewConstructor( + BcelClassWeaver weaver, + NewConstructorTypeMunger newConstructorTypeMunger) + { + final LazyClassGen currentClass = weaver.getLazyClassGen(); + final InstructionFactory fact = currentClass.getFactory(); + + ResolvedMember newConstructorMember = newConstructorTypeMunger.getSyntheticConstructor(); + TypeX onType = newConstructorMember.getDeclaringType(); + + + if (! onType.equals(currentClass.getType())) return false; + + ResolvedMember explicitConstructor = newConstructorTypeMunger.getExplicitConstructor(); + //int declaredParameterCount = newConstructorTypeMunger.getDeclaredParameterCount(); + LazyMethodGen freshConstructor = + makeMethodGen(currentClass, newConstructorMember); + currentClass.addMethodGen(freshConstructor); + //weaver.addLazyMethodGen(freshConstructor); + + InstructionList body = freshConstructor.getBody(); + + // add to body: push arts for call to pre, from actual args starting at 1 (skipping this), going to + // declared argcount + 1 + TypeX[] declaredParams = newConstructorTypeMunger.getSignature().getParameterTypes(); + Type[] paramTypes = freshConstructor.getArgumentTypes(); + int frameIndex = 1; + for (int i = 0, len = declaredParams.length; i < len; i++) { + body.append(fact.createLoad(paramTypes[i], frameIndex)); + frameIndex += paramTypes[i].getSize(); + } + // do call to pre + Member preMethod = AjcMemberMaker.preIntroducedConstructor(aspectType, onType, declaredParams); + body.append(Utility.createInvoke(fact, null, preMethod)); + + // create a local, and store return pre stuff into it. + int arraySlot = freshConstructor.allocateLocal(1); + body.append(fact.createStore(Type.OBJECT, arraySlot)); + + // put this on the stack + body.append(fact.ALOAD_0); + + // unpack pre args onto stack + TypeX[] superParamTypes = explicitConstructor.getParameterTypes(); + + for (int i = 0, len = superParamTypes.length; i < len; i++) { + body.append(fact.createLoad(Type.OBJECT, arraySlot)); + body.append(Utility.createConstant(fact, i)); + body.append(fact.createArrayLoad(Type.OBJECT)); + body.append(Utility.createConversion(fact, Type.OBJECT, BcelWorld.makeBcelType(superParamTypes[i]))); + } + + // call super/this + + body.append(Utility.createInvoke(fact, null, explicitConstructor)); + + // put this back on the stack + + body.append(fact.ALOAD_0); + + // unpack params onto stack + Member postMethod = AjcMemberMaker.postIntroducedConstructor(aspectType, onType, declaredParams); + TypeX[] postParamTypes = postMethod.getParameterTypes(); + + for (int i = 1, len = postParamTypes.length; i < len; i++) { + body.append(fact.createLoad(Type.OBJECT, arraySlot)); + body.append(Utility.createConstant(fact, superParamTypes.length + i-1)); + body.append(fact.createArrayLoad(Type.OBJECT)); + body.append(Utility.createConversion(fact, Type.OBJECT, BcelWorld.makeBcelType(postParamTypes[i]))); + } + + // call post + body.append(Utility.createInvoke(fact, null, postMethod)); + + // don't forget to return!! + + body.append(fact.RETURN); + + return true; + } + + + private static LazyMethodGen makeDispatcher( + LazyClassGen onGen, + String dispatchName, + ResolvedMember superMethod, + BcelWorld world) + { + Type[] paramTypes = BcelWorld.makeBcelTypes(superMethod.getParameterTypes()); + Type returnType = BcelWorld.makeBcelType(superMethod.getReturnType()); + + LazyMethodGen mg = + new LazyMethodGen( + Modifier.PUBLIC, + returnType, + dispatchName, + paramTypes, + TypeX.getNames(superMethod.getExceptions()), + onGen); + InstructionList body = mg.getBody(); + + // assert (!superMethod.isStatic()) + InstructionFactory fact = onGen.getFactory(); + int pos = 0; + + body.append(fact.createThis()); + pos++; + for (int i = 0, len = paramTypes.length; i < len; i++) { + Type paramType = paramTypes[i]; + body.append(fact.createLoad(paramType, pos)); + pos+=paramType.getSize(); + } + body.append(Utility.createSuperInvoke(fact, world, superMethod)); + body.append(fact.createReturn(returnType)); + + return mg; + } + + + private static String genSuperDispatchName( + ResolvedTypeX onType, + ResolvedMember superMethod) + { + return "ajc$" + ResolvedTypeMunger.SUPER_DISPATCH_NAME + "$" + onType.getNameAsIdentifier() + "$" + superMethod.getName(); + } + + + + private boolean mungeNewField(BcelClassWeaver weaver, NewFieldTypeMunger munger) { + ResolvedMember initMethod = munger.getInitMethod(aspectType); + + LazyClassGen gen = weaver.getLazyClassGen(); + ResolvedMember field = munger.getSignature(); + + + ResolvedTypeX onType = weaver.getWorld().resolve(field.getDeclaringType()); + boolean onInterface = onType.isInterface(); + + if (onType.equals(gen.getType())) { + if (onInterface) { + LazyMethodGen mg = makeMethodGen(gen, + AjcMemberMaker.interFieldInterfaceGetter(field, onType, aspectType)); + gen.addMethodGen(mg); + + LazyMethodGen mg1 = makeMethodGen(gen, + AjcMemberMaker.interFieldInterfaceSetter(field, onType, aspectType)); + gen.addMethodGen(mg1); + } else { + weaver.addInitializer(this); + FieldGen fg = makeFieldGen(gen, + AjcMemberMaker.interFieldClassField(field, aspectType)); + gen.addField(fg.getField()); + } + return true; + } else if (onInterface && gen.getType().isTopmostImplementor(onType)) { + // wew know that we can't be static since we don't allow statics on interfaces + if (field.isStatic()) throw new RuntimeException("unimplemented"); + weaver.addInitializer(this); + //System.err.println("impl body on " + gen.getType() + " for " + munger); + Type fieldType = BcelWorld.makeBcelType(field.getType()); + + FieldGen fg = makeFieldGen(gen, + AjcMemberMaker.interFieldInterfaceField(field, onType, aspectType)); + gen.addField(fg.getField()); + + //this uses a shadow munger to add init method to constructors + //weaver.getShadowMungers().add(makeInitCallShadowMunger(initMethod)); + + LazyMethodGen mg = makeMethodGen(gen, + AjcMemberMaker.interFieldInterfaceGetter(field, gen.getType(), aspectType)); + InstructionList il = new InstructionList(); + InstructionFactory fact = gen.getFactory(); + if (field.isStatic()) { + il.append(fact.createFieldAccess( + gen.getClassName(), + fg.getName(), + fieldType, Constants.GETSTATIC)); + } else { + il.append(fact.ALOAD_0); + il.append(fact.createFieldAccess( + gen.getClassName(), + fg.getName(), + fieldType, Constants.GETFIELD)); + } + il.append(fact.createReturn(fieldType)); + mg.getBody().insert(il); + + gen.addMethodGen(mg); + + LazyMethodGen mg1 = makeMethodGen(gen, + AjcMemberMaker.interFieldInterfaceSetter(field, gen.getType(), aspectType)); + InstructionList il1 = new InstructionList(); + if (field.isStatic()) { + il1.append(fact.createLoad(fieldType, 0)); + il1.append(fact.createFieldAccess( + gen.getClassName(), + fg.getName(), + fieldType, Constants.PUTSTATIC)); + } else { + il1.append(fact.ALOAD_0); + il1.append(fact.createLoad(fieldType, 1)); + il1.append(fact.createFieldAccess( + gen.getClassName(), + fg.getName(), + fieldType, Constants.PUTFIELD)); + } + il1.append(fact.createReturn(Type.VOID)); + mg1.getBody().insert(il1); + + gen.addMethodGen(mg1); + + return true; + } else { + return false; + } + } +} diff --git a/weaver/src/org/aspectj/weaver/bcel/BcelVar.java b/weaver/src/org/aspectj/weaver/bcel/BcelVar.java new file mode 100644 index 000000000..8423e9c79 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/bcel/BcelVar.java @@ -0,0 +1,126 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.bcel; + +import org.apache.bcel.generic.*; +import org.aspectj.weaver.*; +import org.aspectj.weaver.ResolvedTypeX; +import org.aspectj.weaver.ast.Var; + +public class BcelVar extends Var { + + private int positionInAroundState = -1; + + private int slot; + + public BcelVar(ResolvedTypeX type, int slot) { + super(type); + this.slot = slot; + } + + public String toString() { + return "BcelVar(" + getType() + " " + slot + + ((positionInAroundState != -1) ? (" " + positionInAroundState) : "") + + + ")"; + } + + public int getSlot() { return slot; } + + public Instruction createLoad(InstructionFactory fact) { + return fact.createLoad(BcelWorld.makeBcelType(getType()), slot); + } + public Instruction createStore(InstructionFactory fact) { + return fact.createStore(BcelWorld.makeBcelType(getType()), slot); + } + + public void appendStore(InstructionList il, InstructionFactory fact) { + il.append(createStore(fact)); + } + public void appendLoad(InstructionList il, InstructionFactory fact) { + il.append(createLoad(fact)); + } + public void appendLoadAndConvert(InstructionList il, InstructionFactory fact, ResolvedTypeX toType) { + il.append(createLoad(fact)); + Utility.appendConversion(il, fact, getType(), toType); + } + public void insertLoad(InstructionList il, InstructionFactory fact) { + il.insert(createLoad(fact)); + } + public InstructionList createCopyFrom(InstructionFactory fact, int oldSlot) { + InstructionList il = new InstructionList(); + il.append(fact.createLoad(BcelWorld.makeBcelType(getType()), oldSlot)); + il.append(createStore(fact)); + return il; + } + + // this is an array var + void appendConvertableArrayLoad( + InstructionList il, + InstructionFactory fact, + int index, + ResolvedTypeX convertTo) + { + ResolvedTypeX convertFromType = getType().getResolvedComponentType(); + appendLoad(il, fact); + il.append(Utility.createConstant(fact, index)); + il.append(fact.createArrayLoad(BcelWorld.makeBcelType(convertFromType))); + Utility.appendConversion(il, fact, convertFromType, convertTo); + } + + void appendConvertableArrayStore( + InstructionList il, + InstructionFactory fact, + int index, + BcelVar storee) + { + ResolvedTypeX convertToType = getType().getResolvedComponentType(); + appendLoad(il, fact); + il.append(Utility.createConstant(fact, index)); + storee.appendLoad(il, fact); + Utility.appendConversion(il, fact, storee.getType(), convertToType); + il.append(fact.createArrayStore(BcelWorld.makeBcelType(convertToType))); + } + + InstructionList createConvertableArrayStore( + InstructionFactory fact, + int index, + BcelVar storee) + { + InstructionList il = new InstructionList(); + appendConvertableArrayStore(il, fact, index, storee); + return il; + } + InstructionList createConvertableArrayLoad( + InstructionFactory fact, + int index, + ResolvedTypeX convertTo) + { + InstructionList il = new InstructionList(); + appendConvertableArrayLoad(il, fact, index, convertTo); + return il; + } + public int getPositionInAroundState() { + return positionInAroundState; + } + + public void setPositionInAroundState(int positionInAroundState) { + this.positionInAroundState = positionInAroundState; + } + + // random useful fields + + public static final BcelVar[] NONE = new BcelVar[] {}; + +} diff --git a/weaver/src/org/aspectj/weaver/bcel/BcelWeaver.java b/weaver/src/org/aspectj/weaver/bcel/BcelWeaver.java new file mode 100644 index 000000000..63eb043d5 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/bcel/BcelWeaver.java @@ -0,0 +1,374 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.bcel; + +import java.io.*; +import java.util.*; +import java.util.zip.*; + +import org.apache.bcel.classfile.*; +import org.apache.bcel.classfile.JavaClass; +import org.aspectj.util.FileUtil; +import org.aspectj.weaver.*; + +public class BcelWeaver implements IWeaver { + private BcelWorld world; + private CrosscuttingMembersSet xcutSet; + + public BcelWeaver(BcelWorld world) { + super(); + this.world = world; + this.xcutSet = world.getCrosscuttingMembersSet(); + } + + public BcelWeaver() { + this(new BcelWorld()); + } + + // ---- fields + private Map sourceJavaClasses = new HashMap(); /* String -> UnwovenClassFile */ + private List addedClasses = new ArrayList(); /* List<UnovenClassFile> */ + private List deletedTypenames = new ArrayList(); /* List<String> */ + private boolean needToReweaveWorld = false; + + private List shadowMungerList = null; // setup by prepareForWeave + private List typeMungerList = null; // setup by prepareForWeave + + private ZipOutputStream zipOutputStream; + + // ---- + + // only called for testing + public void setShadowMungers(List l) { + shadowMungerList = l; + } + + + public void addLibraryAspect(String aspectName) { + ResolvedTypeX type = world.resolve(aspectName); + System.out.println("type: " + type + " for " + aspectName); + if (type.isAspect()) { + xcutSet.addOrReplaceAspect(type); + } else { + throw new RuntimeException("unimplemented"); + } + } + + + + public void addLibraryJarFile(File inFile) throws IOException { + ZipInputStream inStream = new ZipInputStream(new FileInputStream(inFile)); //??? buffered + + List addedAspects = new ArrayList(); + + while (true) { + ZipEntry entry = inStream.getNextEntry(); + if (entry == null) break; + + if (entry.isDirectory() || !entry.getName().endsWith(".class")) { + continue; + } + + ClassParser parser = new ClassParser(new ByteArrayInputStream(FileUtil.readAsByteArray(inStream)), entry.getName()); + JavaClass jc = parser.parse(); + inStream.closeEntry(); + + ResolvedTypeX type = world.addSourceObjectType(jc); + if (type.isAspect()) { + addedAspects.add(type); + } + + } + + inStream.close(); + + for (Iterator i = addedAspects.iterator(); i.hasNext();) { + ResolvedTypeX aspectX = (ResolvedTypeX) i.next(); + xcutSet.addOrReplaceAspect(aspectX); + } + } + + + /** Adds all class files in the jar + */ + public void addJarFile(File inFile, File outDir) throws IOException { + needToReweaveWorld = true; + //System.err.println("adding jar: " + inFile); + ZipInputStream inStream = new ZipInputStream(new FileInputStream(inFile)); //??? buffered + + while (true) { + ZipEntry entry = inStream.getNextEntry(); + if (entry == null) break; + + if (entry.isDirectory() || !entry.getName().endsWith(".class")) { + continue; //??? need to pass other things along untouched +// outStream.putNextEntry(entry); +// outStream.write(Utility.getByteArray(inStream)); +// outStream.closeEntry(); +// return; + } + //System.err.println("adding class: " + entry.getName()); + + byte[] bytes = FileUtil.readAsByteArray(inStream); + String filename = entry.getName(); + UnwovenClassFile classFile = new UnwovenClassFile(new File(outDir, filename).getAbsolutePath(), bytes); + inStream.closeEntry(); + this.addClassFile(classFile); + } + + inStream.close(); + } + + + /** Should be addOrReplace + */ + public void addClassFile(UnwovenClassFile classFile) { + addedClasses.add(classFile); + sourceJavaClasses.put(classFile.getClassName(), classFile); + world.addSourceObjectType(classFile.getJavaClass()); + } + + + public void deleteClassFile(String typename) { + deletedTypenames.add(typename); + sourceJavaClasses.remove(typename); + world.deleteSourceObjectType(TypeX.forName(typename)); + } + + // ---- weave preparation + + public void prepareForWeave() { + needToReweaveWorld = false; + + + // update mungers + for (Iterator i = addedClasses.iterator(); i.hasNext(); ) { + UnwovenClassFile jc = (UnwovenClassFile)i.next(); + String name = jc.getClassName(); + ResolvedTypeX type = world.resolve(name); + if (type.isAspect()) { + needToReweaveWorld |= xcutSet.addOrReplaceAspect(type); + } + } + + for (Iterator i = deletedTypenames.iterator(); i.hasNext(); ) { + String name = (String)i.next(); + xcutSet.deleteAspect(TypeX.forName(name)); + needToReweaveWorld = true; + } + + shadowMungerList = xcutSet.getShadowMungers(); + typeMungerList = xcutSet.getTypeMungers(); + + //XXX this gets us a stable (but completely meaningless) order + Collections.sort( + shadowMungerList, + new Comparator() { + public int compare(Object o1, Object o2) { + return o1.toString().compareTo(o2.toString()); + } + }); + } + + public void dumpUnwoven(File file) throws IOException { + BufferedOutputStream os = FileUtil.makeOutputStream(file); + this.zipOutputStream = new ZipOutputStream(os); + dumpUnwoven(); + zipOutputStream.close(); //this flushes and closes the acutal file + } + + + public void dumpUnwoven() throws IOException { + Collection filesToDump = new HashSet(sourceJavaClasses.values()); + for (Iterator i = filesToDump.iterator(); i.hasNext(); ) { + UnwovenClassFile classFile = (UnwovenClassFile)i.next(); + dumpUnchanged(classFile); + } + } + + + // ---- weaving + + public Collection weave(File file) throws IOException { + OutputStream os = FileUtil.makeOutputStream(file); + this.zipOutputStream = new ZipOutputStream(os); + Collection c = weave(); + zipOutputStream.close(); //this flushes and closes the acutal file + return c; + } + + public Collection weave() throws IOException { + prepareForWeave(); + Collection filesToWeave; + if (needToReweaveWorld) { + filesToWeave = sourceJavaClasses.values(); + } else { + filesToWeave = addedClasses; + } + + Collection wovenClassNames = new ArrayList(); + + //XXX this isn't quite the right place for this... + for (Iterator i = filesToWeave.iterator(); i.hasNext(); ) { + UnwovenClassFile classFile = (UnwovenClassFile)i.next(); + String className = classFile.getClassName(); + ResolvedTypeX onType = world.resolve(className); + weave(onType); + } + + // first weave into aspects + for (Iterator i = filesToWeave.iterator(); i.hasNext(); ) { + UnwovenClassFile classFile = (UnwovenClassFile)i.next(); + String className = classFile.getClassName(); + BcelObjectType classType = (BcelObjectType) world.resolve(className); + if (classType.isAspect()) { + weave(classFile, classType); + wovenClassNames.add(className); + } + } + + // then weave into non-aspects + for (Iterator i = filesToWeave.iterator(); i.hasNext(); ) { + UnwovenClassFile classFile = (UnwovenClassFile)i.next(); + String className = classFile.getClassName(); + BcelObjectType classType = (BcelObjectType) world.resolve(className); + if (! classType.isAspect()) { + weave(classFile, classType); + wovenClassNames.add(className); + } + } + + if (zipOutputStream != null && !needToReweaveWorld) { + Collection filesToDump = new HashSet(sourceJavaClasses.values()); + filesToDump.removeAll(filesToWeave); + for (Iterator i = filesToDump.iterator(); i.hasNext(); ) { + UnwovenClassFile classFile = (UnwovenClassFile)i.next(); + dumpUnchanged(classFile); + } + } + + addedClasses = new ArrayList(); + deletedTypenames = new ArrayList(); + + return wovenClassNames; + } + + private void weave(ResolvedTypeX onType) { + onType.clearInterTypeMungers(); + for (Iterator i = typeMungerList.iterator(); i.hasNext(); ) { + ConcreteTypeMunger m = (ConcreteTypeMunger)i.next(); + if (m.matches(onType)) { + onType.addInterTypeMunger(m); + } + } + } + + + // non-private for testing + LazyClassGen weave(UnwovenClassFile classFile, BcelObjectType classType) throws IOException { + JavaClass javaClass = classType.getJavaClass(); + List shadowMungers = fastMatch(shadowMungerList, javaClass); + List typeMungers = fastMatch(classType.getInterTypeMungers(), javaClass); + + LazyClassGen clazz = null; + + if (shadowMungers.size() > 0 || typeMungers.size() > 0) { + clazz = classType.getLazyClassGen(); + try { + boolean isChanged = BcelClassWeaver.weave(world, clazz, shadowMungers, typeMungers); + if (isChanged) { + dump(classFile, clazz); + return clazz; + } + } catch (RuntimeException re) { + System.err.println("trouble in: "); + clazz.print(System.err); + throw re; + } catch (Error re) { + System.err.println("trouble in: "); + clazz.print(System.err); + throw re; + } + } + + dumpUnchanged(classFile); + return clazz; + } + + // ---- writing + + private void dumpUnchanged(UnwovenClassFile classFile) throws IOException { + if (zipOutputStream != null) { + writeZipEntry(getEntryName(classFile.getJavaClass().getClassName()), classFile.getBytes()); + } else { + classFile.writeUnchangedBytes(); + } + } + + private String getEntryName(String className) { + //XXX what does bcel's getClassName do for inner names + return className.replace('.', '/') + ".class"; + } + + private void dump(UnwovenClassFile classFile, LazyClassGen clazz) throws IOException { + if (zipOutputStream != null) { + String mainClassName = classFile.getJavaClass().getClassName(); + + writeZipEntry(getEntryName(mainClassName), + clazz.getJavaClass().getBytes()); + if (!clazz.getChildClasses().isEmpty()) { + for (Iterator i = clazz.getChildClasses().iterator(); i.hasNext();) { + UnwovenClassFile.ChildClass c = (UnwovenClassFile.ChildClass) i.next(); + writeZipEntry(getEntryName(mainClassName + "$" + c.name), c.bytes); + } + } + } else { + classFile.writeWovenBytes( + clazz.getJavaClass().getBytes(), + clazz.getChildClasses() + ); + } + } + + private void writeZipEntry(String name, byte[] bytes) throws IOException { + ZipEntry newEntry = new ZipEntry(name); //??? get compression scheme right + + zipOutputStream.putNextEntry(newEntry); + zipOutputStream.write(bytes); + zipOutputStream.closeEntry(); + } + + // ---- fast matching + +// boolean fastMatch(JavaClass jc) { +// ConstantPool pool = jc.getConstantPool(); +// for (int i=0, len=pool.getLength(); i < len; i++) { +// Constant c = pool.getConstant(i); +// if (c instanceof ConstantNameAndType) { +// ConstantNameAndType nt = (ConstantNameAndType)c; +// if (nt.getName(pool).equals("toShortString")) { +// //System.out.println("found in " + jc); +// return true; +// } +// } +// } +// return false; +// } + + //XXX need to implement a real fast-match here + private List fastMatch(List list, JavaClass javaClass) { + if (list == null) return Collections.EMPTY_LIST; + return list; + } +} diff --git a/weaver/src/org/aspectj/weaver/bcel/BcelWorld.java b/weaver/src/org/aspectj/weaver/bcel/BcelWorld.java new file mode 100644 index 000000000..1ff0c4e34 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/bcel/BcelWorld.java @@ -0,0 +1,320 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.bcel; + +import java.io.*; +import java.io.IOException; +import java.lang.reflect.Modifier; +import java.util.*; +import java.util.zip.*; +import java.util.zip.ZipInputStream; + +import org.apache.bcel.classfile.*; +import org.apache.bcel.generic.*; +import org.apache.bcel.util.ClassPath; +import org.aspectj.util.FileUtil; +import org.aspectj.weaver.*; +import org.aspectj.weaver.patterns.*; +import org.aspectj.asm.StructureModel; +import org.aspectj.bridge.*; +import org.aspectj.bridge.IMessageHandler; + +public class BcelWorld extends World { + private ClassPathManager classPath; + + //private ClassPathManager aspectPath = null; + private List aspectPathEntries; + + // ---- constructors + + public BcelWorld() { + this(""); + } + + public BcelWorld(String cp) { + this(makeDefaultClasspath(cp), IMessageHandler.THROW); + } + + private static List makeDefaultClasspath(String cp) { + List classPath = new ArrayList(); + classPath.addAll(getPathEntries(cp)); + classPath.addAll(getPathEntries(ClassPath.getClassPath())); + //System.err.println("classpath: " + classPath); + return classPath; + + } + + private static List getPathEntries(String s) { + List ret = new ArrayList(); + StringTokenizer tok = new StringTokenizer(s, File.pathSeparator); + + while(tok.hasMoreTokens()) ret.add(tok.nextToken()); + + return ret; + } + + public BcelWorld(List classPath, IMessageHandler handler) { + //this.aspectPath = new ClassPathManager(aspectPath, handler); + this.classPath = new ClassPathManager(classPath, handler); + setMessageHandler(handler); + } + + /** + * Parse a string into advice. + * + * <blockquote><pre> + * Kind ( Id , ... ) : Pointcut -> MethodSignature + * </pre></blockquote> + */ + public Advice shadowMunger(String str, int extraFlag) { + str = str.trim(); + int start = 0; + int i = str.indexOf('('); + AdviceKind kind = + AdviceKind.stringToKind(str.substring(start, i)); + start = ++i; + i = str.indexOf(')', i); + String[] ids = parseIds(str.substring(start, i).trim()); + //start = ++i; + + + + i = str.indexOf(':', i); + start = ++i; + i = str.indexOf("->", i); + Pointcut pointcut = Pointcut.fromString(str.substring(start, i).trim()); + Member m = Member.methodFromString(str.substring(i+2, str.length()).trim()); + + // now, we resolve + TypeX[] types = m.getParameterTypes(); + FormalBinding[] bindings = new FormalBinding[ids.length]; + for (int j = 0, len = ids.length; j < len; j++) { + bindings[j] = new FormalBinding(types[j], ids[j], j, 0, 0, "fromString"); + } + + Pointcut p = + pointcut.resolve(new SimpleScope(this, bindings)); + + return concreteAdvice(kind, p, m, extraFlag, 0, 0, null); + } + + private String[] parseIds(String str) { + if (str.length() == 0) return ZERO_STRINGS; + List l = new ArrayList(); + int start = 0; + while (true) { + int i = str.indexOf(',', start); + if (i == -1) { + l.add(str.substring(start).trim()); + break; + } + l.add(str.substring(start, i).trim()); + start = i+1; + } + return (String[]) l.toArray(new String[l.size()]); + } + + // ---- various interactions with bcel + + public static Type makeBcelType(TypeX type) { + return Type.getType(type.getSignature()); + } + + static Type[] makeBcelTypes(TypeX[] types) { + Type[] ret = new Type[types.length]; + for (int i = 0, len = types.length; i < len; i++) { + ret[i] = makeBcelType(types[i]); + } + return ret; + } + + public static TypeX fromBcel(Type t) { + return TypeX.forSignature(t.getSignature()); + } + + static TypeX[] fromBcel(Type[] ts) { + TypeX[] ret = new TypeX[ts.length]; + for (int i = 0, len = ts.length; i < len; i++) { + ret[i] = fromBcel(ts[i]); + } + return ret; + } + + public ResolvedTypeX resolve(Type t) { + return resolve(fromBcel(t)); + } + + // ---- fluf + public ResolvedTypeX resolveObjectType(TypeX ty) { + String name = ty.getName(); + JavaClass jc = null; + //UnwovenClassFile classFile = (UnwovenClassFile)sourceJavaClasses.get(name); + //if (classFile != null) jc = classFile.getJavaClass(); +// if (jc == null) { +// jc = lookupJavaClass(aspectPath, name); +// } + if (jc == null) { + jc = lookupJavaClass(classPath, name); + } + if (jc == null) { + return ResolvedTypeX.MISSING; + } else { + return new BcelObjectType(ty.getSignature(), this, jc); + } + } + + private JavaClass lookupJavaClass(ClassPathManager classPath, String name) { + if (classPath == null) return null; + try { + ClassPathManager.ClassFile file = classPath.find(TypeX.forName(name)); + if (file == null) return null; + + ClassParser parser = new ClassParser(file.getInputStream(), file.getPath()); + + JavaClass jc = parser.parse(); + + return jc; + } catch (IOException ioe) { + return null; + } + } + + + BcelObjectType addSourceObjectType(JavaClass jc) { + String signature = TypeX.forName(jc.getClassName()).getSignature(); + BcelObjectType ret = (BcelObjectType)typeMap.get(signature); + if (ret == null) { + ret = new BcelObjectType(signature, this, jc); + typeMap.put(signature, ret); + } else { + ret.replaceJavaClass(jc); + } + return ret; + } + + void deleteSourceObjectType(TypeX ty) { + typeMap.remove(ty.getSignature()); + } + + public static Member makeFieldSignature(LazyClassGen cg, FieldInstruction fi) { + ConstantPoolGen cpg = cg.getConstantPoolGen(); + return + Member.field( + fi.getClassName(cpg), + (fi instanceof GETSTATIC || fi instanceof PUTSTATIC) + ? Modifier.STATIC + : 0, + fi.getName(cpg), + fi.getSignature(cpg)); + } + + public static Member makeFieldSetSignature(LazyClassGen cg, FieldInstruction fi) { + ConstantPoolGen cpg = cg.getConstantPoolGen(); + return + Member.field( + fi.getClassName(cpg), + (fi instanceof GETSTATIC || fi instanceof PUTSTATIC) + ? Modifier.STATIC + : 0, + fi.getName(cpg), + "(" + fi.getSignature(cpg) + ")" +fi.getSignature(cpg)); + } + + public Member makeMethodSignature(LazyMethodGen mg) { + ResolvedMember ret = mg.getMemberView(); + if (ret == null) { + int mods = mg.getAccessFlags(); + if (mg.getEnclosingClass().isInterface()) { + mods |= Modifier.INTERFACE; + } + return new ResolvedMember(mg.getName().equals("<init>") ? Member.CONSTRUCTOR : Member.METHOD, + TypeX.forName(mg.getClassName()), + mods, + fromBcel(mg.getReturnType()), + mg.getName(), + fromBcel(mg.getArgumentTypes()) + ); + } else { + return ret; + } + + } + + public static Member makeMethodSignature(LazyClassGen cg, InvokeInstruction ii) { + ConstantPoolGen cpg = cg.getConstantPoolGen(); + String declaring = ii.getClassName(cpg); + String name = ii.getName(cpg); + String signature = ii.getSignature(cpg); + + int modifier = + (ii instanceof INVOKEINTERFACE) + ? Modifier.INTERFACE + : (ii instanceof INVOKESTATIC) + ? Modifier.STATIC + : (ii instanceof INVOKESPECIAL && ! name.equals("<init>")) + ? Modifier.PRIVATE + : 0; + return Member.method(TypeX.forName(declaring), modifier, name, signature); + } + + public static Member makeMungerMethodSignature(JavaClass javaClass, Method method) { + int mods = 0; + if (method.isStatic()) mods = Modifier.STATIC; + else if (javaClass.isInterface()) mods = Modifier.INTERFACE; + else if (method.isPrivate()) mods = Modifier.PRIVATE; + return Member.method( + TypeX.forName(javaClass.getClassName()), mods, method.getName(), method.getSignature()); + } + + public JavaClass lookupJavaClass(String className) { + ResolvedTypeX t = resolve(TypeX.forName(className)); + if (t instanceof BcelObjectType) { + return ((BcelObjectType)t).getJavaClass(); + } else { + return null; + } + } + + private static final String[] ZERO_STRINGS = new String[0]; + + public String toString() { + StringBuffer buf = new StringBuffer(); + buf.append("BcelWorld("); + //buf.append(shadowMungerMap); + buf.append(")"); + return buf.toString(); + } + + public Advice concreteAdvice( + AdviceKind kind, + Pointcut p, + Member signature, + int extraParameterFlags, + int start, int end, ISourceContext sourceContext) + { + //System.err.println("concrete advice: " + signature + " context " + sourceContext); + return new BcelAdvice(kind, p, signature, extraParameterFlags, start, end, sourceContext, null); + } + + public ConcreteTypeMunger concreteTypeMunger( + ResolvedTypeMunger munger, ResolvedTypeX aspectType) + { + return new BcelTypeMunger(munger, aspectType); + } + + public ConcreteTypeMunger makeCflowStackFieldAdder(ResolvedMember cflowField) { + return new BcelCflowStackFieldAdder(cflowField); + } + +} diff --git a/weaver/src/org/aspectj/weaver/bcel/ClassPathManager.java b/weaver/src/org/aspectj/weaver/bcel/ClassPathManager.java new file mode 100644 index 000000000..6f8d6deba --- /dev/null +++ b/weaver/src/org/aspectj/weaver/bcel/ClassPathManager.java @@ -0,0 +1,191 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.bcel; + +import java.io.*; +import java.io.File; +import java.util.*; +import java.util.zip.*; +import java.util.zip.ZipFile; + +import org.aspectj.bridge.*; +import org.aspectj.bridge.IMessageHandler; +import org.aspectj.weaver.TypeX; + + +public class ClassPathManager { + + private List entries; + + public ClassPathManager(List classpath, IMessageHandler handler) { + entries = new ArrayList(); + for (Iterator i = classpath.iterator(); i.hasNext();) { + String name = (String) i.next(); + File f = new File(name); + String lc = name.toLowerCase(); + if (lc.endsWith(".jar") || lc.endsWith(".zip")) { + if (!f.isFile()) { + MessageUtil.info(handler, "zipfile classpath entry does not exist: " + name); + continue; + } + try { + entries.add(new ZipFileEntry(f)); + } catch (IOException ioe) { + MessageUtil.warn(handler, "zipfile classpath entry is invalid: " + name + "<" + ioe.getMessage() + ">"); + continue; + } + } else { + if (!f.isDirectory()) { + MessageUtil.info(handler, "directory classpath entry does not exist: " + name); + continue; + } + entries.add(new DirEntry(f)); + } + } + } + + public ClassFile find(TypeX type) { + String name = type.getName(); + for (Iterator i = entries.iterator(); i.hasNext(); ) { + Entry entry = (Entry)i.next(); + ClassFile ret = entry.find(name); + if (ret != null) return ret; + } + return null; + } + + public String toString() { + StringBuffer buf = new StringBuffer(); + boolean start = true; + for (Iterator i = entries.iterator(); i.hasNext(); ) { + if (start) { start = false; } + else {buf.append(File.pathSeparator); } + buf.append(i.next()); + } + return buf.toString(); + } + + /** + * This method is extremely expensive and should only be called rarely + */ + public List getAllClassFiles() { + List ret = new ArrayList(); + for (Iterator i = entries.iterator(); i.hasNext(); ) { + Entry entry = (Entry)i.next(); + ret.addAll(entry.getAllClassFiles()); + } + return ret; + } + + + + public abstract static class ClassFile { + public abstract InputStream getInputStream() throws IOException; + public abstract String getPath(); + } + + + public abstract static class Entry { + public abstract ClassFile find(String name); + public abstract List getAllClassFiles(); + } + + + private static class FileClassFile extends ClassFile { + private File file; + public FileClassFile(File file) { + this.file = file; + } + + public InputStream getInputStream() throws IOException { + return new FileInputStream(file); + } + + public String getPath() { return file.getPath(); } + } + + public class DirEntry extends Entry { + private String dirPath; + + public DirEntry(File dir) { this.dirPath = dir.getPath(); } + public DirEntry(String dirPath) { this.dirPath = dirPath; } + + public ClassFile find(String name) { + File f = new File(dirPath + File.separator + name.replace('.', File.separatorChar) + ".class"); + if (f.isFile()) return new FileClassFile(f); + else return null; + } + + public List getAllClassFiles() { + throw new RuntimeException("unimplemented"); + } + + public String toString() { return dirPath; } + } + + + private static class ZipEntryClassFile extends ClassFile { + private ZipEntry entry; + private ZipFile zipFile; + public ZipEntryClassFile(ZipFile zipFile, ZipEntry entry) { + this.zipFile = zipFile; + this.entry = entry; + } + + public InputStream getInputStream() throws IOException { + return zipFile.getInputStream(entry); + } + + public String getPath() { return entry.getName(); } + + } + + + public class ZipFileEntry extends Entry { + private ZipFile zipFile; + + public ZipFileEntry(File file) throws IOException { + this(new ZipFile(file)); + } + + public ZipFileEntry(ZipFile zipFile) { + this.zipFile = zipFile; + } + + public ClassFile find(String name) { + String key = name.replace('.', '/') + ".class"; + ZipEntry entry = zipFile.getEntry(key); + if (entry != null) return new ZipEntryClassFile(zipFile, entry); + else return null; + } + + public List getAllClassFiles() { + List ret = new ArrayList(); + for (Enumeration e = zipFile.entries(); e.hasMoreElements(); ) { + ZipEntry entry = (ZipEntry)e.nextElement(); + String name = entry.getName(); + if (hasClassExtension(name)) ret.add(new ZipEntryClassFile(zipFile, entry)); + } + return ret; + } + + + public String toString() { return zipFile.getName(); } + } + + + private static boolean hasClassExtension(String name) { + return name.toLowerCase().endsWith((".class")); + } +} diff --git a/weaver/src/org/aspectj/weaver/bcel/ExceptionRange.java b/weaver/src/org/aspectj/weaver/bcel/ExceptionRange.java new file mode 100644 index 000000000..2e6520361 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/bcel/ExceptionRange.java @@ -0,0 +1,146 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.bcel; + +import org.apache.bcel.generic.*; +import org.aspectj.weaver.TypeX; + +/** exceptionRanges are set initially to be low priority. The various setPriority methods + * should be used accordingly. The priority is used when we pack the exception table into + * a method... the exception table should be sorted from high to low priority. Exceptions we generate + * for advice is either high priority (higher than anything coming from the original method... + * most kinds of non-execution advice) or low priority (lower than anything coming from the + * original method, for execution advice). + * + * <p> ??? This does not account for handler, or any other + * "statement-level" advice. When such statement level advice happens, we may want to go to a float level, + * so we can set the priority of advice to be lower than anything it encloses, and higher than + * anything enclosing it. + */ + +/* we're actually using the fact that we're an instruction targeter, for the + * handler */ +public final class ExceptionRange extends Range { + + private InstructionHandle handler; + private final TypeX exceptionType; + private final int priority; + + // ---- initialization + + /** + * After this constructor is called, this range is not well situated unless + * {@link #associateWithTargets} is called + */ + public ExceptionRange(InstructionList body, TypeX exceptionType, int priority) { + super(body); + this.exceptionType = exceptionType; + this.priority = priority; + } + public ExceptionRange(InstructionList body, TypeX exceptionType, boolean highPriority) { + this(body, exceptionType, highPriority ? Integer.MAX_VALUE : -1); + } + public void associateWithTargets( + InstructionHandle start, + InstructionHandle end, + InstructionHandle handler) + { + // assert body.contains(start) && body.contains(end) && body.contains(handler) + this.start = start; + this.end = end; + this.handler = handler; + start.addTargeter(this); + end.addTargeter(this); + handler.addTargeter(this); + } + + // ---- + + public InstructionHandle getHandler() { + return handler; + } + public TypeX getCatchType() { + return exceptionType; + } + public int getPriority() { + return priority; + } + + // ---- from object + + public String toString() { + String str; + if (exceptionType == null) { + str = "finally"; + } else { + str = "catch " + exceptionType; + } +// if (priority >= 0 && priority < Integer.MAX_VALUE) { +// str += " (priority " + priority + ")"; +// } + return str; + } + public boolean equals(Object other) { + if (!(other instanceof ExceptionRange)) return false; + ExceptionRange o = (ExceptionRange) other; + return o.getStart() == getStart() + && o.getEnd() == getEnd() + && o.handler == handler + && ((o.exceptionType == null) + ? (exceptionType == null) : + o.exceptionType.equals(exceptionType)) + && o.priority == priority; + } + private volatile int hashCode = 0; + public int hashCode() { + if (hashCode == 0) { + int ret = 17; + ret = 37*ret + getStart().hashCode(); + ret = 37*ret + getEnd().hashCode(); + ret = 37*ret + handler.hashCode(); + ret = 37*ret + ((exceptionType == null) ? 0 : exceptionType.hashCode()); + ret = 37*ret + priority; + hashCode = ret; + } + return hashCode; + } + + public void updateTarget( + InstructionHandle oldIh, + InstructionHandle newIh, + InstructionList newBody) + { + super.updateTarget(oldIh, newIh, newBody); + // we're guaranteed that start, end, and handler are distinct instruction handles. + if (oldIh == handler) { + handler = newIh; + } + } + public static boolean isExceptionStart(InstructionHandle ih) { + if (! isRangeHandle(ih)) return false; + Range r = getRange(ih); + if (! (r instanceof ExceptionRange)) return false; + ExceptionRange er = (ExceptionRange) r; + return er.getStart() == ih; + } + public static boolean isExceptionEnd(InstructionHandle ih) { + if (! isRangeHandle(ih)) return false; + Range r = getRange(ih); + if (! (r instanceof ExceptionRange)) return false; + ExceptionRange er = (ExceptionRange) r; + return er.getEnd() == ih; + } + + +} diff --git a/weaver/src/org/aspectj/weaver/bcel/LazyClassGen.java b/weaver/src/org/aspectj/weaver/bcel/LazyClassGen.java new file mode 100644 index 000000000..0a1861b47 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/bcel/LazyClassGen.java @@ -0,0 +1,471 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.bcel; + +import java.io.*; +import java.lang.reflect.Modifier; +import java.util.*; + +import org.apache.bcel.Constants; +import org.apache.bcel.classfile.*; +import org.apache.bcel.generic.*; +import org.apache.bcel.util.ClassPath; +import org.aspectj.weaver.*; +import org.aspectj.util.CollectionUtil; + +public final class LazyClassGen { + + /** Emit disassembled class and newline to out */ + public static void disassemble(String path, String name, PrintStream out) + throws IOException { + if (null == out) { + return; + } + //out.println("classPath: " + classPath); + + BcelWorld world = new BcelWorld(path); + + LazyClassGen clazz = new LazyClassGen((BcelObjectType) world.resolve(name)); + clazz.print(out); + out.println(); + } + + private BcelObjectType myType; // XXX is not set for types we create + private ClassGen myGen; + private ConstantPoolGen constantPoolGen; + + private List /*LazyMethodGen*/ + methodGens = new ArrayList(); + private List /*LazyClassGen*/ + classGens = new ArrayList(); + + private int childCounter = 0; + + public int getNewGeneratedNameTag() { + return childCounter++; + } + + private InstructionFactory fact; + // ---- + + public LazyClassGen( + String class_name, + String super_class_name, + String file_name, + int access_flags, + String[] interfaces) + { + myGen = new ClassGen(class_name, super_class_name, file_name, access_flags, interfaces); + constantPoolGen = myGen.getConstantPool(); + fact = new InstructionFactory(constantPoolGen); + } + + //Non child type, so it comes from a real type in the world. + public LazyClassGen(BcelObjectType myType) { + myGen = new ClassGen(myType.getJavaClass()); + constantPoolGen = myGen.getConstantPool(); + fact = new InstructionFactory(constantPoolGen); + this.myType = myType; + + Method[] methods = myGen.getMethods(); + for (int i = 0; i < methods.length; i++) { + addMethodGen(new LazyMethodGen(methods[i], this)); + } + } + +// public void addAttribute(Attribute i) { +// myGen.addAttribute(i); +// } + + // ---- + + public String getInternalClassName() { + return getConstantPoolGen().getConstantPool().getConstantString( + myGen.getClassNameIndex(), + Constants.CONSTANT_Class); + + } + + public File getPackagePath(File root) { + String str = getInternalClassName(); + int index = str.lastIndexOf('/'); + if (index == -1) + return root; + return new File(root, str.substring(0, index)); + } + + public String getClassId() { + String str = getInternalClassName(); + int index = str.lastIndexOf('/'); + if (index == -1) + return str; + return str.substring(index + 1); + } + + + public void addMethodGen(LazyMethodGen gen) { + //assert gen.getClassName() == super.getClassName(); + methodGens.add(gen); + } + + public List getMethodGens() { + return Collections.unmodifiableList(methodGens); + + } + + private void writeBack() { + addAjcInitializers(); + + + int len = methodGens.size(); + myGen.setMethods(new Method[0]); + for (int i = 0; i < len; i++) { + LazyMethodGen gen = (LazyMethodGen) methodGens.get(i); + // we skip empty clinits + if (isEmptyClinit(gen)) continue; + myGen.addMethod(gen.getMethod()); + } + } + + public JavaClass getJavaClass() { + writeBack(); + return myGen.getJavaClass(); + } + + public void addGeneratedInner(LazyClassGen newClass) { + classGens.add(newClass); + } + + public void addInterface(TypeX typeX) { + myGen.addInterface(typeX.getName()); + } + + // non-recursive, may be a bug, ha ha. + private List getClassGens() { + List ret = new ArrayList(); + ret.add(this); + ret.addAll(classGens); + + return ret; + } + + + public List getChildClasses() { + if (classGens.isEmpty()) return Collections.EMPTY_LIST; + List ret = new ArrayList(); + for (Iterator i = classGens.iterator(); i.hasNext();) { + LazyClassGen clazz = (LazyClassGen) i.next(); + byte[] bytes = clazz.getJavaClass().getBytes(); + String name = clazz.getName(); + int index = name.lastIndexOf('$'); + // XXX this could be bad, check use of dollar signs. + name = name.substring(index+1); + ret.add(new UnwovenClassFile.ChildClass(name, bytes)); + } + return ret; + } + + public String toString() { + return toShortString(); + } + + public String toShortString() { + String s = + org.apache.bcel.classfile.Utility.accessToString(myGen.getAccessFlags(), true); + if (s != "") + s += " "; + s += org.apache.bcel.classfile.Utility.classOrInterface(myGen.getAccessFlags()); + s += " "; + s += myGen.getClassName(); + return s; + } + + public String toLongString() { + ByteArrayOutputStream s = new ByteArrayOutputStream(); + print(new PrintStream(s)); + return new String(s.toByteArray()); + } + + public void print() { print(System.out); } + + public void print(PrintStream out) { + List classGens = getClassGens(); + for (Iterator iter = classGens.iterator(); iter.hasNext();) { + LazyClassGen element = (LazyClassGen) iter.next(); + element.printOne(out); + if (iter.hasNext()) out.println(); + } + } + + private void printOne(PrintStream out) { + out.print(toShortString()); + out.print(" extends "); + out.print( + org.apache.bcel.classfile.Utility.compactClassName( + myGen.getSuperclassName(), + false)); + + int size = myGen.getInterfaces().length; + + if (size > 0) { + out.print(" implements "); + for (int i = 0; i < size; i++) { + out.print(myGen.getInterfaceNames()[i]); + if (i < size - 1) + out.print(", "); + } + } + out.print(":"); + out.println(); + // XXX make sure to pass types correctly around, so this doesn't happen. + if (myType != null) { + myType.printWackyStuff(out); + } + Field[] fields = myGen.getFields(); + for (int i = 0, len = fields.length; i < len; i++) { + out.print(" "); + out.println(fields[i]); + } + List methodGens = getMethodGens(); + for (Iterator iter = methodGens.iterator(); iter.hasNext();) { + LazyMethodGen gen = (LazyMethodGen) iter.next(); + // we skip empty clinits + if (isEmptyClinit(gen)) continue; + gen.print(out); + if (iter.hasNext()) out.println(); + } +// out.println(" ATTRIBS: " + Arrays.asList(myGen.getAttributes())); + + out.println("end " + toShortString()); + } + + private boolean isEmptyClinit(LazyMethodGen gen) { + if (!gen.getName().equals("<clinit>")) return false; + //System.err.println("checking clinig: " + gen); + InstructionHandle start = gen.getBody().getStart(); + while (start != null) { + if (Range.isRangeHandle(start) || (start.getInstruction() instanceof RETURN)) { + start = start.getNext(); + } else { + return false; + } + } + + return true; + } + + public ConstantPoolGen getConstantPoolGen() { + return constantPoolGen; + } + + public String getName() { + return myGen.getClassName(); + } + + public WeaverStateKind getWeaverState() { + WeaverStateKind kind = myType.getWeaverState(); + if (kind == null) return WeaverStateKind.Untouched; + return kind; + } + + public void setWeaverState(WeaverStateKind s) { + Attribute[] attributes = myGen.getAttributes(); + if (attributes != null) { + for (int i = attributes.length - 1; i >=0; i--) { + Attribute a = attributes[i]; + if (a instanceof Unknown) { + Unknown u = (Unknown) a; + if (u.getName().equals(AjAttribute.WeaverState.AttributeName)) { + myGen.removeAttribute(u); + } + } + } + } + myGen.addAttribute(BcelAttributes.bcelAttribute( + new AjAttribute.WeaverState(s), + getConstantPoolGen())); + } + + public InstructionFactory getFactory() { + return fact; + } + + public LazyMethodGen getStaticInitializer() { + for (Iterator i = methodGens.iterator(); i.hasNext();) { + LazyMethodGen gen = (LazyMethodGen) i.next(); + if (gen.getName().equals("<clinit>")) return gen; + } + LazyMethodGen clinit = new LazyMethodGen( + Modifier.STATIC, + Type.VOID, + "<clinit>", + new Type[0], + CollectionUtil.NO_STRINGS, + this); + clinit.getBody().insert(getFactory().RETURN); + methodGens.add(clinit); + return clinit; + } + + + // reflective thisJoinPoint support + Map/*BcelShadow, Field*/ tjpFields = new HashMap(); + public static final ObjectType tjpType = + new ObjectType("org.aspectj.lang.JoinPoint"); + public static final ObjectType staticTjpType = + new ObjectType("org.aspectj.lang.JoinPoint$StaticPart"); + private static final ObjectType sigType = + new ObjectType("org.aspectj.lang.Signature"); + private static final ObjectType slType = + new ObjectType("org.aspectj.lang.reflect.SourceLocation"); + private static final ObjectType factoryType = + new ObjectType("org.aspectj.runtime.reflect.Factory"); + private static final ObjectType classType = + new ObjectType("java.lang.Class"); + + public Field getTjpField(BcelShadow shadow) { + Field ret = (Field)tjpFields.get(shadow); + if (ret != null) return ret; + + ret = new FieldGen(Modifier.STATIC | Modifier.PUBLIC | Modifier.FINAL, + staticTjpType, + "ajc$tjp_" + tjpFields.size(), + getConstantPoolGen()).getField(); + addField(ret); + tjpFields.put(shadow, ret); + return ret; + } + + private void addAjcInitializers() { + if (tjpFields.size() == 0) return; + + InstructionList il = initializeAllTjps(); + getStaticInitializer().getBody().insert(il); + } + + + private InstructionList initializeAllTjps() { + InstructionList list = new InstructionList(); + InstructionFactory fact = getFactory(); + + // make a new factory + list.append(fact.createNew(factoryType)); + list.append(fact.createDup(1)); + + list.append(new PUSH(getConstantPoolGen(), getFileName())); + + // load the current Class object + //XXX check that this works correctly for inners/anonymous + list.append(new PUSH(getConstantPoolGen(), getClassName())); + //XXX do we need to worry about the fact the theorectically this could throw + //a ClassNotFoundException + list.append(fact.createInvoke("java.lang.Class", "forName", classType, + new Type[] {Type.STRING}, Constants.INVOKESTATIC)); + + list.append(fact.createInvoke(factoryType.getClassName(), "<init>", + Type.VOID, new Type[] {Type.STRING, classType}, + Constants.INVOKESPECIAL)); + + list.append(fact.createStore(factoryType, 0)); + + List entries = new ArrayList(tjpFields.entrySet()); + Collections.sort(entries, new Comparator() { + public int compare(Object a, Object b) { + Map.Entry ae = (Map.Entry) a; + Map.Entry be = (Map.Entry) b; + return ((Field) ae.getValue()) + .getName() + .compareTo(((Field)be.getValue()).getName()); + } + }); + + for (Iterator i = entries.iterator(); i.hasNext(); ) { + Map.Entry entry = (Map.Entry)i.next(); + initializeTjp(fact, list, (Field)entry.getValue(), (BcelShadow)entry.getKey()); + } + + return list; + } + + + private void initializeTjp(InstructionFactory fact, InstructionList list, + Field field, BcelShadow shadow) + { + Member sig = shadow.getSignature(); + //ResolvedMember mem = shadow.getSignature().resolve(shadow.getWorld()); + + // load the factory + list.append(fact.createLoad(factoryType, 0)); + + // load the kind + list.append(new PUSH(getConstantPoolGen(), shadow.getKind().getName())); + + // create the signature + list.append(fact.createLoad(factoryType, 0)); + list.append(new PUSH(getConstantPoolGen(), sig.getSignatureString(shadow.getWorld()))); + list.append(fact.createInvoke(factoryType.getClassName(), + sig.getSignatureMakerName(), + new ObjectType(sig.getSignatureType()), + new Type[] { Type.STRING }, + Constants.INVOKEVIRTUAL)); + + //XXX should load source location from shadow + list.append(Utility.createConstant(fact, shadow.getSourceLine())); + + + list.append(fact.createInvoke(factoryType.getClassName(), + "makeSJP", staticTjpType, + new Type[] { Type.STRING, sigType, Type.INT}, + Constants.INVOKEVIRTUAL)); + + // put it in the field + list.append(fact.createFieldAccess(getClassName(), field.getName(), + staticTjpType, Constants.PUTSTATIC)); + } + + + public BcelObjectType getType() { + return myType; + } + + public String getFileName() { + return myGen.getFileName(); + } + + public void addField(Field field) { + myGen.addField(field); + } + + public String getClassName() { + return myGen.getClassName(); + } + + public boolean isInterface() { + return myGen.isInterface(); + } + + public LazyMethodGen getLazyMethodGen(Member m) { + return getLazyMethodGen(m.getName(), m.getSignature()); + } + + public LazyMethodGen getLazyMethodGen(String name, String signature) { + for (Iterator i = methodGens.iterator(); i.hasNext();) { + LazyMethodGen gen = (LazyMethodGen) i.next(); + if (gen.getName().equals(name) && gen.getSignature().equals(signature)) + return gen; + } + throw new BCException("Class " + this.getName() + " does not have a method " + + name + " with signature " + signature); + } +} diff --git a/weaver/src/org/aspectj/weaver/bcel/LazyMethodGen.java b/weaver/src/org/aspectj/weaver/bcel/LazyMethodGen.java new file mode 100644 index 000000000..63d105e19 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/bcel/LazyMethodGen.java @@ -0,0 +1,1060 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.bcel; + +import java.io.*; +import java.lang.reflect.Modifier; +import java.util.*; + +import org.apache.bcel.Constants; +import org.apache.bcel.classfile.*; +import org.apache.bcel.generic.*; +import org.aspectj.weaver.*; + + +/** + * A LazyMethodGen should be treated as a MethodGen. It's our way of abstracting over the + * low-level Method objects. It converts through {@link MethodGen} to create + * and to serialize, but that's it. + * + * <p> At any rate, there are two ways to create LazyMethodGens. + * One is from a method, which + * does work through MethodGen to do the correct thing. + * The other is the creation of a completely empty + * LazyMethodGen, and it is used when we're constructing code from scratch. + * + * <p> We stay away from targeters for rangey things like Shadows and Exceptions. + */ + +public final class LazyMethodGen { + private final int accessFlags; + private final Type returnType; + private final String name; + private final Type[] argumentTypes; + //private final String[] argumentNames; + private final String[] declaredExceptions; + private final InstructionList body; // leaving null for abstracts + private final Attribute[] attributes; + private final LazyClassGen enclosingClass; + private final BcelMethod memberView; + + private int maxLocals; + + /** + * only used by {@link BcelClassWeaver} + */ + List /*ShadowMungers*/ matchedShadows; + + // Used for interface introduction + // this is the type of the interface the method is technically on + public ResolvedTypeX definingType = null; + + public LazyMethodGen( + int accessFlags, + Type returnType, + String name, + Type[] paramTypes, + String[] declaredExceptions, + LazyClassGen enclosingClass) + { + this.memberView = null; // ??? should be okay, since constructed ones aren't woven into + this.accessFlags = accessFlags; + this.returnType = returnType; + this.name = name; + this.argumentTypes = paramTypes; + //this.argumentNames = Utility.makeArgNames(paramTypes.length); + this.declaredExceptions = declaredExceptions; + if (!Modifier.isAbstract(accessFlags)) { + body = new InstructionList(); + setMaxLocals(calculateMaxLocals()); + } else { + body = null; + } + this.attributes = new Attribute[0]; + this.enclosingClass = enclosingClass; + assertGoodBody(); + } + + private int calculateMaxLocals() { + int ret = 0; + if (!Modifier.isStatic(accessFlags)) ret++; + for (int i = 0, len = argumentTypes.length; i < len; i++) { + ret += argumentTypes[i].getSize(); + } + return ret; + } + + // build from an existing method + public LazyMethodGen(Method m, LazyClassGen enclosingClass) { + this.enclosingClass = enclosingClass; + if (!m.isAbstract() && m.getCode() == null) { + throw new RuntimeException("bad non-abstract method with no code: " + m + " on " + enclosingClass); + } + MethodGen gen = new MethodGen(m, enclosingClass.getName(), enclosingClass.getConstantPoolGen()); + this.memberView = new BcelMethod(enclosingClass.getType(), m); + this.accessFlags = gen.getAccessFlags(); + this.returnType = gen.getReturnType(); + this.name = gen.getName(); + this.argumentTypes = gen.getArgumentTypes(); + //this.argumentNames = gen.getArgumentNames(); + this.declaredExceptions = gen.getExceptions(); + this.attributes = gen.getAttributes(); + this.maxLocals = gen.getMaxLocals(); + if (gen.isAbstract() || gen.isNative()) { + body = null; + } else { + body = gen.getInstructionList(); + unpackHandlers(gen); + unpackLineNumbers(gen); + unpackLocals(gen); + } + assertGoodBody(); + } + + // XXX we're relying on the javac promise I've just made up that we won't have an early exception + // in the list mask a later exception: That is, for two exceptions E and F, + // if E preceeds F, then either E \cup F = {}, or E \nonstrictsubset F. So when we add F, + // we add it on the _OUTSIDE_ of any handlers that share starts or ends with it. + + // with that in mind, we merrily go adding ranges for exceptions. + + private void unpackHandlers(MethodGen gen) { + CodeExceptionGen[] exns = gen.getExceptionHandlers(); + if (exns != null) { + int len = exns.length; + int priority = len - 1; + for (int i = 0; i < len; i++, priority--) { + CodeExceptionGen exn = exns[i]; + + InstructionHandle start = + Range.genStart( + body, + getOutermostExceptionStart(exn.getStartPC())); + InstructionHandle end = Range.genEnd(body, getOutermostExceptionEnd(exn.getEndPC())); + // this doesn't necessarily handle overlapping correctly!!! + ExceptionRange er = + new ExceptionRange( + body, + exn.getCatchType() == null + ? null + : BcelWorld.fromBcel(exn.getCatchType()), + priority); + er.associateWithTargets(start, end, exn.getHandlerPC()); + exn.setStartPC(null); // also removes from target + exn.setEndPC(null); // also removes from target + exn.setHandlerPC(null); // also removes from target + } + gen.removeExceptionHandlers(); + } + } + + private InstructionHandle getOutermostExceptionStart(InstructionHandle ih) { + while (true) { + if (ExceptionRange.isExceptionStart(ih.getPrev())) { + ih = ih.getPrev(); + } else { + return ih; + } + } + } + private InstructionHandle getOutermostExceptionEnd(InstructionHandle ih) { + while (true) { + if (ExceptionRange.isExceptionEnd(ih.getNext())) { + ih = ih.getNext(); + } else { + return ih; + } + } + } + + + private void unpackLineNumbers(MethodGen gen) { + LineNumberTag lr = null; + for (InstructionHandle ih = body.getStart(); ih != null; ih = ih.getNext()) { + InstructionTargeter[] targeters = ih.getTargeters(); + if (targeters != null) { + for (int i = targeters.length - 1; i >= 0; i--) { + InstructionTargeter targeter = targeters[i]; + if (targeter instanceof LineNumberGen) { + LineNumberGen lng = (LineNumberGen) targeter; + lng.updateTarget(ih, null); + lr = new LineNumberTag(lng.getSourceLine()); + } + } + } + if (lr != null) { + ih.addTargeter(lr); + } + } + gen.removeLineNumbers(); + } + + private void unpackLocals(MethodGen gen) { + Set locals = new HashSet(); + for (InstructionHandle ih = body.getStart(); ih != null; ih = ih.getNext()) { + InstructionTargeter[] targeters = ih.getTargeters(); + List ends = new ArrayList(0); + if (targeters != null) { + for (int i = targeters.length - 1; i >= 0; i--) { + InstructionTargeter targeter = targeters[i]; + if (targeter instanceof LocalVariableGen) { + LocalVariableGen lng = (LocalVariableGen) targeter; + LocalVariableTag lr = new LocalVariableTag(BcelWorld.fromBcel(lng.getType()), lng.getName(), lng.getIndex()); + if (lng.getStart() == ih) { + lng.setStart(null); + locals.add(lr); + } else { + lng.setEnd(null); + ends.add(lr); + } + } + } + } + for (Iterator i = locals.iterator(); i.hasNext(); ) { + ih.addTargeter((LocalVariableTag) i.next()); + } + locals.removeAll(ends); + } + gen.removeLocalVariables(); + } + + // =============== + + public int allocateLocal(Type type) { + return allocateLocal(type.getSize()); + } + + public int allocateLocal(int slots) { + int max = getMaxLocals(); + setMaxLocals(max + slots); + return max; + } + + public Method getMethod() { + MethodGen gen = pack(); + return gen.getMethod(); + } + + // ============================= + + public String toString() { + return toLongString(); + } + + public String toShortString() { + org.apache.bcel.classfile.Utility util = null; // EVIL! + String access = util.accessToString(getAccessFlags()); + + StringBuffer buf = new StringBuffer(); + + if (!access.equals("")) { + buf.append(access); + buf.append(" "); + } + buf.append(util.signatureToString(getReturnType().getSignature(), true)); + buf.append(" "); + buf.append(getName()); + buf.append("("); + { + int len = argumentTypes.length; + if (len > 0) { + buf.append(util.signatureToString(argumentTypes[0].getSignature(), true)); + for (int i = 1; i < argumentTypes.length; i++) { + buf.append(", "); + buf.append(util.signatureToString(argumentTypes[i].getSignature(), true)); + } + } + } + buf.append(")"); + + { + int len = declaredExceptions != null ? declaredExceptions.length : 0; + if (len > 0) { + buf.append(" throws "); + buf.append(declaredExceptions[0]); + for (int i = 1; i < declaredExceptions.length; i++) { + buf.append(", "); + buf.append(declaredExceptions[i]); + } + } + } + return buf.toString(); + } + + public String toLongString() { + ByteArrayOutputStream s = new ByteArrayOutputStream(); + print(new PrintStream(s)); + return new String(s.toByteArray()); + } + + public void print() { + print(System.out); + } + + public void print(PrintStream out) { + out.print(" " + toShortString()); + printAspectAttributes(out); + + InstructionList body = getBody(); + if (body == null) { + out.println(";"); + return; + } + out.println(":"); + new BodyPrinter(out).run(); + out.println(" end " + toShortString()); + } + + + private void printAspectAttributes(PrintStream out) { + ISourceContext context = null; + if (enclosingClass != null && enclosingClass.getType() != null) { + context = enclosingClass.getType().getSourceContext(); + } + List as = BcelAttributes.readAjAttributes(attributes, context); + if (! as.isEmpty()) { + out.println(" " + as.get(0)); // XXX assuming exactly one attribute, munger... + } + } + + + private class BodyPrinter { + Map prefixMap = new HashMap(); + Map suffixMap = new HashMap(); + Map labelMap = new HashMap(); + + InstructionList body; + PrintStream out; + ConstantPool pool; + List ranges; + + BodyPrinter(PrintStream out) { + this.pool = enclosingClass.getConstantPoolGen().getConstantPool(); + this.body = getBody(); + this.out = out; + } + + void run() { + assignLabels(); + print(); + } + + // label assignment + void assignLabels() { + LinkedList exnTable = new LinkedList(); + String pendingLabel = null; +// boolean hasPendingTargeters = false; + int lcounter = 0; + for (InstructionHandle ih = body.getStart(); ih != null; ih = ih.getNext()) { + InstructionTargeter[] targeters = ih.getTargeters(); + if (targeters != null) { + for (int i = targeters.length - 1; i >= 0; i--) { + InstructionTargeter t = targeters[i]; + if (t instanceof ExceptionRange) { + // assert isRangeHandle(h); + ExceptionRange r = (ExceptionRange) t; + if (r.getStart() == ih) { + insertHandler(r, exnTable); + } + } else if (t instanceof BranchInstruction) { + if (pendingLabel == null) { + pendingLabel = "L" + lcounter++; + } + } else { + // assert isRangeHandle(h) + } + } + } + if (pendingLabel != null) { + labelMap.put(ih, pendingLabel); + if (! Range.isRangeHandle(ih)) { + pendingLabel = null; + } + } + } + int ecounter = 0; + for (Iterator i = exnTable.iterator(); i.hasNext();) { + ExceptionRange er = (ExceptionRange) i.next(); + String exceptionLabel = "E" + ecounter++; + labelMap.put(Range.getRealStart(er.getHandler()), exceptionLabel); + labelMap.put(er.getHandler(), exceptionLabel); + } + } + + // printing + + void print() { + int depth = 0; + int currLine = -1; + bodyPrint: + for (InstructionHandle ih = body.getStart(); ih != null; ih = ih.getNext()) { + if (Range.isRangeHandle(ih)) { + Range r = Range.getRange(ih); + // don't print empty ranges, that is, ranges who contain no actual instructions + for (InstructionHandle xx = r.getStart(); Range.isRangeHandle(xx); xx = xx.getNext()) { + if (xx == r.getEnd()) continue bodyPrint; + } + // doesn't handle nested: if (r.getStart().getNext() == r.getEnd()) continue; + if (r.getStart() == ih) { + printRangeString(r, depth++); + } else { + if (r.getEnd() != ih) throw new RuntimeException("bad"); + printRangeString(r, --depth); + } + } else { + printInstruction(ih, depth); + int line = getLineNumber(ih, currLine); + if (line != currLine) { + currLine = line; + out.println(" (line " + line + ")"); + } else { + out.println(); + } + } + } + } + + void printRangeString(Range r, int depth) { + printDepth(depth); + out.println(getRangeString(r, labelMap)); + } + + + String getRangeString(Range r, Map labelMap) { + if (r instanceof ExceptionRange) { + ExceptionRange er = (ExceptionRange) r; + return er.toString() + " -> " + labelMap.get(er.getHandler()); + } else { + return r.toString(); + } + } + + void printDepth(int depth) { + pad(BODY_INDENT); + while (depth > 0) { + out.print("| "); + depth--; + } + } + + + void printLabel(String s, int depth) { + int space = Math.max(CODE_INDENT - depth * 2, 0); + if (s == null) { + pad(space); + } else { + space = Math.max(space - (s.length() + 2), 0); + pad(space); + out.print(s); + out.print(": "); + } + } + + void printInstruction(InstructionHandle h, int depth) { + printDepth(depth); + printLabel((String) labelMap.get(h), depth); + + Instruction inst = h.getInstruction(); + if (inst instanceof CPInstruction) { + CPInstruction cpinst = (CPInstruction) inst; + out.print(Constants.OPCODE_NAMES[cpinst.getOpcode()].toUpperCase()); + out.print(" "); + out.print(pool.constantToString(pool.getConstant(cpinst.getIndex()))); + } else if (inst instanceof Select) { + Select sinst = (Select) inst; + out.println(Constants.OPCODE_NAMES[sinst.getOpcode()].toUpperCase()); + int[] matches = sinst.getMatchs(); + InstructionHandle[] targets = sinst.getTargets(); + InstructionHandle defaultTarget = sinst.getTarget(); + for (int i = 0, len = matches.length; i < len; i++) { + printDepth(depth); + printLabel(null, depth); + out.print(" "); + out.print(matches[i]); + out.print(": \t"); + out.println(labelMap.get(targets[i])); + } + printDepth(depth); + printLabel(null, depth); + out.print(" "); + out.print("default: \t"); + out.print(labelMap.get(defaultTarget)); + } else if (inst instanceof BranchInstruction) { + BranchInstruction brinst = (BranchInstruction) inst; + out.print(Constants.OPCODE_NAMES[brinst.getOpcode()].toUpperCase()); + out.print(" "); + out.print(labelMap.get(brinst.getTarget())); + } else if (inst instanceof LocalVariableInstruction) { + LocalVariableInstruction lvinst = (LocalVariableInstruction) inst; + out.print(inst.toString(false).toUpperCase()); + int index = lvinst.getIndex(); + LocalVariableTag tag = getLocalVariableTag(h, index); + if (tag != null) { + out.print(" // "); + out.print(tag.getType()); + out.print(" "); + out.print(tag.getName()); + } + } else { + out.print(inst.toString(false).toUpperCase()); + } + } + + + + + static final int BODY_INDENT = 4; + static final int CODE_INDENT = 16; + + void pad(int size) { + for (int i = 0; i < size; i++) { + out.print(" "); + } + } + } + + + static LocalVariableTag getLocalVariableTag( + InstructionHandle ih, + int index) + { + InstructionTargeter[] targeters = ih.getTargeters(); + if (targeters == null) return null; + for (int i = targeters.length - 1; i >= 0; i--) { + InstructionTargeter t = targeters[i]; + if (t instanceof LocalVariableTag) { + LocalVariableTag lvt = (LocalVariableTag) t; + if (lvt.getSlot() == index) return lvt; + } + } + return null; + } + + static int getLineNumber( + InstructionHandle ih, + int prevLine) + { + InstructionTargeter[] targeters = ih.getTargeters(); + if (targeters == null) return prevLine; + for (int i = targeters.length - 1; i >= 0; i--) { + InstructionTargeter t = targeters[i]; + if (t instanceof LineNumberTag) { + return ((LineNumberTag)t).getLineNumber(); + } + } + return prevLine; + } + + public boolean isStatic() { + return Modifier.isStatic(getAccessFlags()); + } + + public void addExceptionHandler( + InstructionHandle start, + InstructionHandle end, + InstructionHandle handlerStart, + ObjectType catchType, + boolean highPriority) { + + InstructionHandle start1 = Range.genStart(body, start); + InstructionHandle end1 = Range.genEnd(body, end); + + ExceptionRange er = + new ExceptionRange(body, BcelWorld.fromBcel(catchType), highPriority); + er.associateWithTargets(start1, end1, handlerStart); + } + + public int getAccessFlags() { + return accessFlags; + } + + public Type[] getArgumentTypes() { + return argumentTypes; + } + +// public String[] getArgumentNames() { +// return argumentNames; +// } + + public LazyClassGen getEnclosingClass() { + return enclosingClass; + } + + public int getMaxLocals() { + return maxLocals; + } + + public String getName() { + return name; + } + + public Type getReturnType() { + return returnType; + } + + public void setMaxLocals(int maxLocals) { + this.maxLocals = maxLocals; + } + + public InstructionList getBody() { + return body; + } + + public boolean hasBody() { return body != null; } + + + public Attribute[] getAttributes() { + return attributes; + } + + public String[] getDeclaredExceptions() { + return declaredExceptions; + } + + public String getClassName() { + return enclosingClass.getName(); + } + + + // ---- packing! + + public MethodGen pack() { + MethodGen gen = + new MethodGen( + getAccessFlags(), + getReturnType(), + getArgumentTypes(), + null, //getArgumentNames(), + getName(), + getEnclosingClass().getName(), + new InstructionList(), + getEnclosingClass().getConstantPoolGen()); + for (int i = 0, len = declaredExceptions.length; i < len; i++) { + gen.addException(declaredExceptions[i]); + } + for (int i = 0, len = attributes.length; i < len; i++) { + gen.addAttribute(attributes[i]); + } + if (hasBody()) { + packBody(gen); + gen.setMaxLocals(); + gen.setMaxStack(); + } + return gen; + } + + /** fill the newly created method gen with our body, + * inspired by InstructionList.copy() + */ + public void packBody(MethodGen gen) { + HashMap map = new HashMap(); + InstructionList fresh = gen.getInstructionList(); + + /* Make copies of all instructions, append them to the new list + * and associate old instruction references with the new ones, i.e., + * a 1:1 mapping. + */ + for (InstructionHandle ih = getBody().getStart(); ih != null; ih = ih.getNext()) { + if (Range.isRangeHandle(ih)) { + continue; + } + Instruction i = ih.getInstruction(); + Instruction c = i.copy(); // Use clone for shallow copy + + if (c instanceof BranchInstruction) + map.put(ih, fresh.append((BranchInstruction) c)); + else + map.put(ih, fresh.append(c)); + } + // at this point, no rangeHandles are in fresh. Let's use that... + + /* Update branch targets and insert various attributes. + * Insert our exceptionHandlers + * into a sorted list, so they can be added in order later. + */ + InstructionHandle ih = getBody().getStart(); + InstructionHandle jh = fresh.getStart(); + + LinkedList exnList = new LinkedList(); + + Map localVariableStarts = new HashMap(); + Map localVariableEnds = new HashMap(); + + int currLine = -1; + + while (ih != null) { + if (map.get(ih) == null) { + // we're a range instruction + Range r = Range.getRange(ih); + if (r instanceof ExceptionRange) { + ExceptionRange er = (ExceptionRange) r; + if (er.getStart() == ih) { + // order is important, insert handlers in order of start + insertHandler(er, exnList); + } + } else { + // we must be a shadow range or something equally useless, + // so forget about doing anything + } + // just increment ih. + ih = ih.getNext(); + } else { + // assert map.get(ih) == jh + Instruction i = ih.getInstruction(); + Instruction j = jh.getInstruction(); + + if (i instanceof BranchInstruction) { + BranchInstruction bi = (BranchInstruction) i; + BranchInstruction bj = (BranchInstruction) j; + InstructionHandle itarget = bi.getTarget(); // old target + +// try { + // New target is in hash map + bj.setTarget(remap(itarget, map)); +// } catch (NullPointerException e) { +// print(); +// System.out.println("Was trying to remap " + bi); +// System.out.println("who's target was supposedly " + itarget); +// throw e; +// } + + if (bi instanceof Select) { + // Either LOOKUPSWITCH or TABLESWITCH + InstructionHandle[] itargets = ((Select) bi).getTargets(); + InstructionHandle[] jtargets = ((Select) bj).getTargets(); + + for (int k = itargets.length - 1; k >= 0; k--) { + // Update all targets + jtargets[k] = remap(itargets[k], map); + jtargets[k].addTargeter(bj); + } + } + } + + // now deal with line numbers + // and store up info for local variables + InstructionTargeter[] targeters = ih.getTargeters(); + if (targeters != null) { + for (int k = targeters.length - 1; k >= 0; k--) { + InstructionTargeter targeter = targeters[k]; + if (targeter instanceof LineNumberTag) { + int line = ((LineNumberTag)targeter).getLineNumber(); + if (line != currLine) { + gen.addLineNumber(jh, line); + currLine = line; + } + } else if (targeter instanceof LocalVariableTag) { + LocalVariableTag lvt = (LocalVariableTag) targeter; + if (i instanceof LocalVariableInstruction) { + int index = ((LocalVariableInstruction)i).getIndex(); + if (lvt.getSlot() == index) { + if (localVariableStarts.get(lvt) == null) { + localVariableStarts.put(lvt, jh); + } + localVariableEnds.put(lvt, jh); + } + } + } + } + } + // now continue + ih = ih.getNext(); + jh = jh.getNext(); + } + } + + // now add exception handlers + for (Iterator iter = exnList.iterator(); iter.hasNext();) { + ExceptionRange r = (ExceptionRange) iter.next(); + if (r.isEmpty()) continue; + gen.addExceptionHandler( + remap(r.getRealStart(), map), + remap(r.getRealEnd(), map), + remap(r.getHandler(), map), + (r.getCatchType() == null) + ? null + : (ObjectType) BcelWorld.makeBcelType(r.getCatchType())); + } + // now add local variables + gen.removeLocalVariables(); + for (Iterator iter = localVariableStarts.keySet().iterator(); iter.hasNext(); ) { + LocalVariableTag tag = (LocalVariableTag) iter.next(); + gen.addLocalVariable( + tag.getName(), + BcelWorld.makeBcelType(tag.getType()), + tag.getSlot(), + (InstructionHandle) localVariableStarts.get(tag), + (InstructionHandle) localVariableEnds.get(tag)); + } + } + + private static InstructionHandle remap(InstructionHandle ih, Map map) { + while (true) { + Object ret = map.get(ih); + if (ret == null) { + ih = ih.getNext(); + } else { + return (InstructionHandle) ret; + } + } + } + + // exception ordering + // An exception A preceeds an exception B in the exception table iff: + + // * A and B were in the original method, and A preceeded B in the original exception table + // * If A has a higher priority than B, than it preceeds B. + // * If A and B have the same priority, then the one whose START happens EARLIEST has LEAST priority. + // in short, the outermost exception has least priority. + // we implement this with a LinkedList. We could possibly implement this with a java.util.SortedSet, + // but I don't trust the only implementation, TreeSet, to do the right thing. + + private static void insertHandler(ExceptionRange fresh, LinkedList l) { + for (ListIterator iter = l.listIterator(); iter.hasNext();) { + ExceptionRange r = (ExceptionRange) iter.next(); + if (fresh.getPriority() >= r.getPriority()) { + iter.previous(); + iter.add(fresh); + return; + } + } + l.add(fresh); + } + + + + public boolean isPrivate() { + return Modifier.isPrivate(getAccessFlags()); + } + + + // ---- + + /** A good body is a body with the following properties: + * + * <ul> + * <li> For each branch instruction S in body, target T of S is in body. + * <li> For each branch instruction S in body, target T of S has S as a targeter. + * <li> For each instruction T in body, for each branch instruction S that is a + * targeter of T, S is in body. + * <li> For each non-range-handle instruction T in body, for each instruction S + * that is a targeter of T, S is + * either a branch instruction, an exception range or a tag + * <li> For each range-handle instruction T in body, there is exactly one targeter S + * that is a range. + * <li> For each range-handle instruction T in body, the range R targeting T is in body. + * <li> For each instruction T in body, for each exception range R targeting T, R is + * in body. + * <li> For each exception range R in body, let T := R.handler. T is in body, and R is one + * of T's targeters + * <li> All ranges are properly nested: For all ranges Q and R, if Q.start preceeds + * R.start, then R.end preceeds Q.end. + * </ul> + * + * Where the shorthand "R is in body" means "R.start is in body, R.end is in body, and + * any InstructionHandle stored in a field of R (such as an exception handle) is in body". + */ + + public void assertGoodBody() { + // XXX this is REALLY slow since we get the string first... + assertGoodBody(getBody(), toString()); + } + + // XXX we might not want to release with any calls to this incredibly inefficient method. + public static void assertGoodBody(InstructionList il, String from) { + if (true) return; + if (il == null) return; + Set body = new HashSet(); + Stack ranges = new Stack(); + for (InstructionHandle ih = il.getStart(); ih != null; ih = ih.getNext()) { + body.add(ih); + if (ih.getInstruction() instanceof BranchInstruction) { + body.add(ih.getInstruction()); + } + } + + for (InstructionHandle ih = il.getStart(); ih != null; ih = ih.getNext()) { + assertGoodHandle(ih, body, ranges, from); + InstructionTargeter[] ts = ih.getTargeters(); + if (ts != null) { + for (int i = ts.length - 1; i >= 0; i--) { + assertGoodTargeter(ts[i], ih, body, from); + } + } + } + } + + private static void assertGoodHandle(InstructionHandle ih, Set body, Stack ranges, String from) { + Instruction inst = ih.getInstruction(); + if ((inst instanceof BranchInstruction) ^ (ih instanceof BranchHandle)) { + throw new BCException("bad instruction/handle pair in " + from); + } + if (Range.isRangeHandle(ih)) { + assertGoodRangeHandle(ih, body, ranges, from); + } else if (inst instanceof BranchInstruction) { + assertGoodBranchInstruction((BranchHandle) ih, (BranchInstruction) inst, body, ranges, from); + } + } + + private static void assertGoodBranchInstruction( + BranchHandle ih, + BranchInstruction inst, + Set body, + Stack ranges, + String from) + { + if (ih.getTarget() != inst.getTarget()) { + throw new BCException("bad branch instruction/handle pair in " + from); + } + InstructionHandle target = ih.getTarget(); + assertInBody(target, body, from); + assertTargetedBy(target, inst, from); + if (inst instanceof Select) { + Select sel = (Select) inst; + InstructionHandle[] itargets = sel.getTargets(); + for (int k = itargets.length - 1; k >= 0; k--) { + assertInBody(itargets[k], body, from); + assertTargetedBy(itargets[k], inst, from); + } + } + } + + /** ih is an InstructionHandle or a BranchInstruction */ + private static void assertInBody(Object ih, Set body, String from) { + if (! body.contains(ih)) throw new BCException("thing not in body in " + from); + } + + private static void assertGoodRangeHandle(InstructionHandle ih, Set body, Stack ranges, String from) { + Range r = getRangeAndAssertExactlyOne(ih, from); + assertGoodRange(r, body, from); + if (r.getStart() == ih) { + ranges.push(r); + } else if (r.getEnd() == ih) { + if (ranges.peek() != r) throw new BCException("bad range inclusion in " + from); + ranges.pop(); + } + } + + private static void assertGoodRange(Range r, Set body, String from) { + assertInBody(r.getStart(), body, from); + assertRangeHandle(r.getStart(), from); + assertTargetedBy(r.getStart(), r, from); + + assertInBody(r.getEnd(), body, from); + assertRangeHandle(r.getEnd(), from); + assertTargetedBy(r.getEnd(), r, from); + + if (r instanceof ExceptionRange) { + ExceptionRange er = (ExceptionRange) r; + assertInBody(er.getHandler(), body, from); + assertTargetedBy(er.getHandler(), r, from); + } + } + + private static void assertRangeHandle(InstructionHandle ih, String from) { + if (! Range.isRangeHandle(ih)) throw new BCException("bad range handle " + ih + " in " + from); + } + + + private static void assertTargetedBy( + InstructionHandle target, + InstructionTargeter targeter, + String from) + { + InstructionTargeter[] ts = target.getTargeters(); + if (ts == null) throw new BCException("bad targeting relationship in " + from); + for (int i = ts.length - 1; i >= 0; i--) { + if (ts[i] == targeter) return; + } + throw new RuntimeException("bad targeting relationship in " + from); + } + + private static void assertTargets(InstructionTargeter targeter, InstructionHandle target, String from) { + if (targeter instanceof Range) { + Range r = (Range) targeter; + if (r.getStart() == target || r.getEnd() == target) return; + if (r instanceof ExceptionRange) { + if (((ExceptionRange)r).getHandler() == target) return; + } + } else if (targeter instanceof BranchInstruction) { + BranchInstruction bi = (BranchInstruction) targeter; + if (bi.getTarget() == target) return; + if (targeter instanceof Select) { + Select sel = (Select) targeter; + InstructionHandle[] itargets = sel.getTargets(); + for (int k = itargets.length - 1; k >= 0; k--) { + if (itargets[k] == target) return; + } + } + } else if (targeter instanceof Tag) { + return; + } + throw new BCException(targeter + " doesn't target " + target + " in " + from ); + } + + private static Range getRangeAndAssertExactlyOne(InstructionHandle ih, String from) { + Range ret = null; + InstructionTargeter[] ts = ih.getTargeters(); + if (ts == null) throw new BCException("range handle with no range in " + from); + for (int i = ts.length - 1; i >= 0; i--) { + if (ts[i] instanceof Range) { + if (ret != null) throw new BCException("range handle with multiple ranges in " + from); + ret = (Range) ts[i]; + } + } + if (ret == null) throw new BCException("range handle with no range in " + from); + return ret; + } + + private static void assertGoodTargeter( + InstructionTargeter t, + InstructionHandle ih, + Set body, + String from) + { + assertTargets(t, ih, from); + if (t instanceof Range) { + assertGoodRange((Range) t, body, from); + } else if (t instanceof BranchInstruction) { + assertInBody(t, body, from); + } + } + + + // ---- + + boolean isAdviceMethod() { + return memberView.getAssociatedShadowMunger() != null; + } + + boolean isAjSynthetic() { + if (memberView == null) return true; + return memberView.isAjSynthetic(); + } + + public AjAttribute.EffectiveSignatureAttribute getEffectiveSignature() { + //if (memberView == null) return null; + return memberView.getEffectiveSignature(); + } + + public String getSignature() { + return Member.typesToSignature(BcelWorld.fromBcel(getReturnType()), + BcelWorld.fromBcel(getArgumentTypes())); + } + + public BcelMethod getMemberView() { + return memberView; + } + +} diff --git a/weaver/src/org/aspectj/weaver/bcel/LineNumberTag.java b/weaver/src/org/aspectj/weaver/bcel/LineNumberTag.java new file mode 100644 index 000000000..ef5822f3a --- /dev/null +++ b/weaver/src/org/aspectj/weaver/bcel/LineNumberTag.java @@ -0,0 +1,39 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.bcel; + +/** we don't actually target instructions, but instructions target us. */ +public class LineNumberTag extends Tag { + + private final int lineNumber; + + public LineNumberTag(int lineNumber) { + this.lineNumber = lineNumber; + } + + public int getLineNumber() { return lineNumber; } + + // ---- from Object + + public String toString() { + return "line " + lineNumber; + } + public boolean equals(Object other) { + if (! (other instanceof LineNumberTag)) return false; + return lineNumber == ((LineNumberTag)other).lineNumber; + } + public int hashCode() { + return lineNumber; + } +} diff --git a/weaver/src/org/aspectj/weaver/bcel/LocalVariableTag.java b/weaver/src/org/aspectj/weaver/bcel/LocalVariableTag.java new file mode 100644 index 000000000..609dac005 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/bcel/LocalVariableTag.java @@ -0,0 +1,60 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.bcel; + +import org.aspectj.weaver.TypeX; + +public final class LocalVariableTag extends Tag { + private final TypeX type; + private final String name; + private final int slot; + + public LocalVariableTag(TypeX type, String name, int slot) { + this.type = type; + this.name = name; + this.slot = slot; + } + + public String getName() { + return name; + } + public int getSlot() { + return slot; + } + public TypeX getType() { + return type; + } + + // ---- from Object + + public String toString() { + return "local " + slot + ": " + type + " " + name; + } + public boolean equals(Object other) { + if (!(other instanceof LocalVariableTag)) return false; + LocalVariableTag o = (LocalVariableTag)other; + return o.type.equals(type) && o.name.equals(name) && o.slot == slot; + } + private volatile int hashCode = 0; + public int hashCode() { + if (hashCode == 0) { + int ret = 17; + ret = 37*ret + type.hashCode(); + ret = 37*ret + name.hashCode(); + ret = 37*ret + slot; + hashCode = ret; + } + return hashCode; + } +} diff --git a/weaver/src/org/aspectj/weaver/bcel/Range.java b/weaver/src/org/aspectj/weaver/bcel/Range.java new file mode 100644 index 000000000..0d6eb21b6 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/bcel/Range.java @@ -0,0 +1,225 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.bcel; + +import java.lang.reflect.Modifier; +import java.util.Map; + +import org.apache.bcel.Constants; +import org.apache.bcel.generic.*; +import org.aspectj.weaver.*; +import org.aspectj.weaver.IntMap; + +abstract class Range implements InstructionTargeter { + + protected InstructionList body; + protected InstructionHandle start; + protected InstructionHandle end; + + // ---- initialization + + protected Range(InstructionList il) { + this.body = il; + } + + + // ---- + + final InstructionList getBody() { + return body; + } + final InstructionHandle getStart() { + return start; + } + final InstructionHandle getEnd() { + return end; + } + + // ---- + + boolean isEmpty() { + InstructionHandle ih = start; + while (ih != end) { + if (! Range.isRangeHandle(ih)) return false; + ih = ih.getNext(); + } + return true; + } + + static InstructionHandle getRealStart(InstructionHandle ih) { + while (Range.isRangeHandle(ih)) { + ih = ih.getNext(); + } + return ih; + } + + InstructionHandle getRealStart() { + return getRealStart(start); + } + + static InstructionHandle getRealEnd(InstructionHandle ih) { + while (Range.isRangeHandle(ih)) { + ih = ih.getPrev(); + } + return ih; + } + InstructionHandle getRealEnd() { + return getRealEnd(end); + } + + static InstructionHandle getRealPrev(InstructionHandle ih) { + InstructionHandle ret = ih.getPrev(); + while (isRangeHandle(ret)) { + ret = ret.getPrev(); + } + return ret; + } + + // ---- + + InstructionHandle insert(Instruction i, Where where) { + InstructionList il = new InstructionList(); + InstructionHandle ret = il.insert(i); + insert(il, where); + return ret; + } + void insert(InstructionList freshIl, Where where) { + InstructionHandle h; + if (where == InsideBefore || where == OutsideBefore) { + h = getStart(); + } else { + h = getEnd(); + } + if (where == InsideBefore || where == OutsideAfter) { + body.append(h, freshIl); + } else { + InstructionHandle newStart = body.insert(h, freshIl); + if (where == OutsideBefore) { + // XXX this is slow. There's a better design than this. We should + // never have to retarget branches apart from the creation of ranges. + // basically, we should never weave OutsideBefore. + BcelShadow.retargetAllBranches(h, newStart); + } + } + + } + InstructionHandle append(Instruction i) { + return insert(i, InsideAfter); + } + void append(InstructionList i) { + insert(i, InsideAfter); + } + + static InstructionHandle remap(InstructionHandle h, Map m) { + return (InstructionHandle) m.get(h); + } + + private static void setLineNumberFromNext(InstructionHandle ih) { + int lineNumber = Utility.getSourceLine(ih.getNext()); + if (lineNumber != -1) { + Utility.setSourceLine(ih, lineNumber); + } + } + + static InstructionHandle genStart(InstructionList body) { + InstructionHandle ih = body.insert(Range.RANGEINSTRUCTION); + setLineNumberFromNext(ih); + return ih; + } + + static InstructionHandle genEnd(InstructionList body) { + return body.append(Range.RANGEINSTRUCTION); + } + static InstructionHandle genStart(InstructionList body, InstructionHandle ih) { + if (ih == null) return genStart(body); + InstructionHandle freshIh = body.insert(ih, Range.RANGEINSTRUCTION); + setLineNumberFromNext(freshIh); + return freshIh; + } + + static InstructionHandle genEnd(InstructionList body, InstructionHandle ih) { + if (ih == null) return genEnd(body); + return body.append(ih, Range.RANGEINSTRUCTION); + } + + // ----- + + public boolean containsTarget(InstructionHandle ih) { + return false; + } + + public final void updateTarget(InstructionHandle old_ih, InstructionHandle new_ih) { + throw new RuntimeException("Ranges must be updated with an enclosing instructionList"); + } + + protected void updateTarget( + InstructionHandle old_ih, + InstructionHandle new_ih, + InstructionList new_il) + { + old_ih.removeTargeter(this); + if (new_ih != null) + new_ih.addTargeter(this); + body = new_il; + + if (old_ih == start) { + start = new_ih; + } + if (old_ih == end) { + end = new_ih; + } + } + + public static final boolean isRangeHandle(InstructionHandle ih) { + if (ih == null) return false; + return ih.getInstruction() == Range.RANGEINSTRUCTION; + } + + protected static final Range getRange(InstructionHandle ih) { + // assert isRangeHandle(ih) + Range ret = null; + InstructionTargeter[] targeters = ih.getTargeters(); + for (int i = targeters.length - 1; i >= 0; i--) { + if (targeters[i] instanceof Range) { + Range r = (Range) targeters[i]; + if (r.getStart() != ih && r.getEnd() != ih) continue; + if (ret != null) throw new BCException("multiple ranges on same range handle: " + ret + ", " + targeters[i]); + ret = (Range) targeters[i]; + } + } + if (ret == null) throw new BCException("shouldn't happen"); + return ret; + } + + // ---- + + static final Where InsideBefore = new Where("insideBefore"); + static final Where InsideAfter = new Where("insideAfter"); + static final Where OutsideBefore = new Where("outsideBefore"); + static final Where OutsideAfter = new Where("outsideAfter"); + + // ---- constants + + // note that this is STUPIDLY copied by Instruction.copy(), so don't do that. + + public static final Instruction RANGEINSTRUCTION = new IMPDEP1(); + + // ---- + + static class Where { + private String name; + public Where(String name) { this.name = name; } + public String toString() { return name; } + } +} diff --git a/weaver/src/org/aspectj/weaver/bcel/ShadowRange.java b/weaver/src/org/aspectj/weaver/bcel/ShadowRange.java new file mode 100644 index 000000000..edd22db95 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/bcel/ShadowRange.java @@ -0,0 +1,199 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.bcel; + +import org.apache.bcel.generic.*; +import org.aspectj.weaver.*; + +final class ShadowRange extends Range { + + private BcelShadow shadow; + + // ---- initialization + + /** + * After this constructor is called, this range is not well situated unless both + * {@link #associateWithTargets} and {@link #associateWithShadow} are called. + */ + public ShadowRange(InstructionList body) { + super(body); + } + protected void associateWithTargets(InstructionHandle start, InstructionHandle end) { + // assert body.contains(start) && body.contains(end); + this.start = start; + this.end = end; + start.addTargeter(this); + end.addTargeter(this); + } + public void associateWithShadow(BcelShadow shadow) { + this.shadow = shadow; + shadow.setRange(this); + } + + // ---- + + public Shadow.Kind getKind() { + return shadow.getKind(); + } + + public String toString() { + return shadow.toString(); + } + + void extractInstructionsInto(LazyMethodGen freshMethod, IntMap remap, boolean addReturn) { + LazyMethodGen.assertGoodBody(getBody(), toString()); + freshMethod.assertGoodBody(); + InstructionList freshBody = freshMethod.getBody(); + + for (InstructionHandle oldIh = start.getNext(); oldIh != end; oldIh = oldIh.getNext()) { + // first we copy the instruction itself. + Instruction oldI = oldIh.getInstruction(); + Instruction freshI = (oldI == RANGEINSTRUCTION) ? oldI : oldI.copy(); + + // Now we add it to the new instruction list. + InstructionHandle freshIh; + if (freshI instanceof BranchInstruction) { + //If it's a targeting instruction, + // update the target(s) to point to the new copy instead of the old copy. + BranchInstruction oldBranch = (BranchInstruction) oldI; + BranchInstruction freshBranch = (BranchInstruction) freshI; + InstructionHandle oldTarget = oldBranch.getTarget(); + oldTarget.removeTargeter(oldBranch); + oldTarget.addTargeter(freshBranch); + if (freshBranch instanceof Select) { + Select oldSelect = (Select) oldI; + Select freshSelect = (Select) freshI; + InstructionHandle[] oldTargets = freshSelect.getTargets(); + for (int k = oldTargets.length - 1; k >= 0; k--) { + oldTargets[k].removeTargeter(oldSelect); + oldTargets[k].addTargeter(freshSelect); + } + } + freshIh = freshBody.append(freshBranch); + } else { + freshIh = freshBody.append(freshI); + } + + // if source comes before target: + // source <--> target + // --> [process: target.removeTargeter(source); target.addTargeter(sourcecopy)] + // source ---------\ + // v + // sourcecopy <--> target + // --> [ process: sourcecopy.updateTarget(target, targetcopy) ] + // source ----> target + // sourcecopy <--> targetcopy + + // if target comes before source + + // target <--> source + // --> [process: source.updateTarget(target, targetcopy) ] + // target + // targetcopy <--> source + // --> [process: targetcopy.removeTargeter(source); targetcopy.addTargeter(sourcecopy)] + // target source + // v + // targetcopy <--> sourcecopy + + // now deal with the old instruction's targeters. Update them all to point to us + // instead of the old instruction. We use updateTarget to do this. One goal is + // to make sure we remove all targeters from the old guy, so we can successfully + // delete it. + InstructionTargeter[] sources = oldIh.getTargeters(); + if (sources != null) { + for (int j = sources.length - 1; j >= 0; j--) { + InstructionTargeter source = sources[j]; + if (source instanceof LocalVariableTag) { + // XXX destroying local variable info + source.updateTarget(oldIh, null); + } else if (source instanceof Range) { + // exceptions and shadows are just moved + ((Range)source).updateTarget(oldIh, freshIh, freshBody); + } else { + // line numbers can be shared, + // branches will be copied along with us. + source.updateTarget(oldIh, freshIh); + } + } + } + // we're now done with the old instruction entirely, and will ignore them through + // the rest of this loop. The only time we'll see them again is a second pass to + // delete them. + + // now deal with local variable instructions. If this points to a remapped + // frame location, update the instruction's index. If this doesn't, + // do compaction/expansion: allocate a new local variable, and modify the remap + // to handle it. XXX We're doing the safe thing and allocating ALL these local variables + // as double-wides, in case the location is found to hold a double-wide later. + if (freshI instanceof LocalVariableInstruction || freshI instanceof RET) { + IndexedInstruction indexedI = (IndexedInstruction) freshI; + int oldIndex = indexedI.getIndex(); + int freshIndex; + if (! remap.hasKey(oldIndex)) { + freshIndex = freshMethod.allocateLocal(2); + remap.put(oldIndex, freshIndex); + } else { + freshIndex = remap.get(oldIndex); + } + indexedI.setIndex(freshIndex); + } +// System.err.println("JUST COPIED: " + oldIh.getInstruction().toString(freshMethod.getEnclosingClass().getConstantPoolGen().getConstantPool()) +// + " INTO " + freshIh.getInstruction().toString(freshMethod.getEnclosingClass().getConstantPoolGen().getConstantPool())); + } + + // we've now copied out all the instructions. + // now delete the instructions... we've already taken care of the damn + // targets, but since TargetLostException is checked, we have to do this stuff. + try { + for (InstructionHandle oldIh = start.getNext(); oldIh != end; ) { + InstructionHandle next = oldIh.getNext(); + body.delete(oldIh); + oldIh = next; + } + } catch (TargetLostException e) { + throw new BCException("shouldn't have gotten a target lost"); + } + + // now add the return, if one is warranted. + InstructionHandle ret = null; + if (addReturn) { + // we really should pull this out somewhere... + ret = freshBody.append( + new InstructionFactory(freshMethod.getEnclosingClass().getConstantPoolGen()) + .createReturn(freshMethod.getReturnType())); + } + // and remap all the old targeters of the end handle of the range to the return. + InstructionTargeter[] ts = end.getTargeters(); + if (ts != null) { // shouldn't be the case, but let's test for paranoia + for (int j = ts.length - 1; j >= 0; j--) { + InstructionTargeter t = ts[j]; + if (t == this) continue; + if (! addReturn) { + throw new BCException("range has target, but we aren't adding a return"); + } else { + t.updateTarget(end, ret); + } + } + } + + LazyMethodGen.assertGoodBody(getBody(), toString()); + freshMethod.assertGoodBody(); + } + + public BcelShadow getShadow() { + return shadow; + } + + +} diff --git a/weaver/src/org/aspectj/weaver/bcel/Tag.java b/weaver/src/org/aspectj/weaver/bcel/Tag.java new file mode 100644 index 000000000..cfa5eb41f --- /dev/null +++ b/weaver/src/org/aspectj/weaver/bcel/Tag.java @@ -0,0 +1,42 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.bcel; + +import org.apache.bcel.generic.*; + +/** A tag is an instruction-targeter that doesn't bother remembering its target(s) */ +abstract class Tag implements InstructionTargeter, Cloneable { + + public Tag() { + } + + // ---- from InstructionTargeter + public boolean containsTarget(InstructionHandle ih) { + return false; + } + + public void updateTarget(InstructionHandle old_ih, InstructionHandle new_ih) { + old_ih.removeTargeter(this); + if (new_ih != null) + new_ih.addTargeter(this); + } + + public Tag copy() { + try { + return (Tag)clone(); + } catch (CloneNotSupportedException e) { + throw new RuntimeException("Sanity check, can't clone me"); + } + } +} diff --git a/weaver/src/org/aspectj/weaver/bcel/UnwovenClassFile.java b/weaver/src/org/aspectj/weaver/bcel/UnwovenClassFile.java new file mode 100644 index 000000000..9ea49d53f --- /dev/null +++ b/weaver/src/org/aspectj/weaver/bcel/UnwovenClassFile.java @@ -0,0 +1,181 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.bcel; + +import java.io.*; +import java.util.*; + +import org.apache.bcel.classfile.JavaClass; +import org.aspectj.util.FileUtil; + +public class UnwovenClassFile { + protected String filename; + protected byte[] bytes; +// protected JavaClass javaClass = null; + protected byte[] writtenBytes = null; + protected List /* ChildClass */ writtenChildClasses = new ArrayList(0); + protected String className = null; + + public UnwovenClassFile(String filename, byte[] bytes) { + this.filename = filename; + this.bytes = bytes; + } + + public String getFilename() { + return filename; + } + + public String makeInnerFileName(String innerName) { + String prefix = filename.substring(0, filename.length()-6); // strip the .class + return prefix + "$" + innerName + ".class"; + } + + public byte[] getBytes() { +// if (bytes == null) bytes = javaClass.getBytes(); + return bytes; + } + + public JavaClass getJavaClass() { + //XXX need to know when to make a new class and when not to + //XXX this is an important optimization + if (getBytes() == null) { + System.out.println("no bytes for: " + getFilename()); + Thread.currentThread().dumpStack(); + } + return Utility.makeJavaClass(filename, getBytes()); +// if (javaClass == null) javaClass = Utility.makeJavaClass(filename, getBytes()); +// return javaClass; + } + + public boolean exists() { + return getBytes() != null; + } + + + public void writeUnchangedBytes() throws IOException { + writeWovenBytes(getBytes(), Collections.EMPTY_LIST); + } + + public void writeWovenBytes(byte[] bytes, List childClasses) throws IOException { + writeChildClasses(childClasses); + + //System.err.println("about to write: " + this + ", " + writtenBytes + ", "); +// + writtenBytes != null + " && " + unchanged(bytes, writtenBytes) ); + + if (writtenBytes != null && !unchanged(bytes, writtenBytes)) return; + + BufferedOutputStream os = FileUtil.makeOutputStream(new File(filename)); + os.write(bytes); + os.close(); + + writtenBytes = bytes; + } + + private void writeChildClasses(List childClasses) throws IOException { + //??? we only really need to delete writtenChildClasses whose + //??? names aren't in childClasses; however, it's unclear + //??? how much that will affect performance + deleteAllChildClasses(); + + childClasses.removeAll(writtenChildClasses); //XXX is this right + + for (Iterator iter = childClasses.iterator(); iter.hasNext();) { + ChildClass childClass = (ChildClass) iter.next(); + writeChildClassFile(childClass.name, childClass.bytes); + + } + + writtenChildClasses = childClasses; + + } + + private void writeChildClassFile(String innerName, byte[] bytes) throws IOException { + BufferedOutputStream os = + FileUtil.makeOutputStream(new File(makeInnerFileName(innerName))); + os.write(bytes); + os.close(); + } + + + protected void deleteAllChildClasses() { + for (Iterator iter = writtenChildClasses.iterator(); iter.hasNext();) { + ChildClass childClass = (ChildClass) iter.next(); + deleteChildClassFile(childClass.name); + } + } + + protected void deleteChildClassFile(String innerName) { + File childClassFile = new File(makeInnerFileName(innerName)); + childClassFile.delete(); + } + + + + private static boolean unchanged(byte[] b1, byte[] b2) { + int len = b1.length; + if (b2.length != len) return false; + for (int i=0; i < len; i++) { + if (b1[i] != b2[i]) return false; + } + return true; + } + + + public String getClassName() { + if (className == null) className = getJavaClass().getClassName(); + return className; + } + public byte[] getWrittenBytes() { + return writtenBytes; + } + + public String toString() { + return "UnwovenClassFile(" + filename + ", " + getClassName() + ")"; + } + + public void deleteRealFile() throws IOException { + new File(filename).delete(); + } + + + + + // record + public static class ChildClass { + public final String name; + public final byte[] bytes; + + ChildClass(String name, byte[] bytes) { + this.name = name; + this.bytes = bytes; + } + + public boolean equals(Object other) { + if (! (other instanceof ChildClass)) return false; + ChildClass o = (ChildClass) other; + return o.name.equals(name) && unchanged(o.bytes, bytes); + } + public int hashCode() { + return name.hashCode(); + } + + public String toString() { + return "(ChildClass " + name + ")"; + } + } +} + + + + diff --git a/weaver/src/org/aspectj/weaver/bcel/UnwovenZipClassFile.java b/weaver/src/org/aspectj/weaver/bcel/UnwovenZipClassFile.java new file mode 100644 index 000000000..2c4c760a9 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/bcel/UnwovenZipClassFile.java @@ -0,0 +1,45 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.bcel; + +import java.io.IOException; +import java.util.List; +import java.util.zip.*; + +//XXX we believe this is now unneeded +public class UnwovenZipClassFile extends UnwovenClassFile { + private ZipOutputStream zipOutputStream; + + public UnwovenZipClassFile(ZipOutputStream zipOutputStream, String filename, byte[] bytes) { + super(filename, bytes); + this.zipOutputStream = zipOutputStream; + } + + + public void writeWovenBytes(byte[] bytes, List childClasses) throws IOException { + //??? we rewrite this every time + if (!childClasses.isEmpty()) { + throw new RuntimeException("unimplemented"); + } + + ZipEntry newEntry = new ZipEntry(filename); //??? get compression scheme right + + zipOutputStream.putNextEntry(newEntry); + zipOutputStream.write(bytes); + zipOutputStream.closeEntry(); + + writtenBytes = bytes; + } + +} diff --git a/weaver/src/org/aspectj/weaver/bcel/Utility.java b/weaver/src/org/aspectj/weaver/bcel/Utility.java new file mode 100644 index 000000000..9da48c01f --- /dev/null +++ b/weaver/src/org/aspectj/weaver/bcel/Utility.java @@ -0,0 +1,442 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.bcel; + +import java.io.*; +import java.lang.reflect.Modifier; + +import org.apache.bcel.Constants; +import org.apache.bcel.classfile.*; +import org.apache.bcel.generic.*; +import org.aspectj.weaver.*; + +public class Utility { + + private Utility() { + super(); + } + + + public static Instruction createSuperInvoke( + InstructionFactory fact, + BcelWorld world, + Member signature) { + short kind; + if (signature.isInterface()) { + throw new RuntimeException("bad"); + } else if (signature.isPrivate() || signature.getName().equals("<init>")) { + throw new RuntimeException("unimplemented, possibly bad"); + } else if (signature.isStatic()) { + throw new RuntimeException("bad"); + } else { + kind = Constants.INVOKESPECIAL; + } + + return fact.createInvoke( + signature.getDeclaringType().getName(), + signature.getName(), + world.makeBcelType(signature.getReturnType()), + world.makeBcelTypes(signature.getParameterTypes()), + kind); + } + + // XXX don't need the world now + public static Instruction createInvoke( + InstructionFactory fact, + BcelWorld world, + Member signature) { + short kind; + if (signature.isInterface()) { + kind = Constants.INVOKEINTERFACE; + } else if (signature.isPrivate() || signature.getName().equals("<init>")) { + kind = Constants.INVOKESPECIAL; + } else if (signature.isStatic()) { + kind = Constants.INVOKESTATIC; + } else { + kind = Constants.INVOKEVIRTUAL; + } + + return fact.createInvoke( + signature.getDeclaringType().getName(), + signature.getName(), + BcelWorld.makeBcelType(signature.getReturnType()), + BcelWorld.makeBcelTypes(signature.getParameterTypes()), + kind); + } + + public static Instruction createGet(InstructionFactory fact, Member signature) { + short kind; + if (signature.isStatic()) { + kind = Constants.GETSTATIC; + } else { + kind = Constants.GETFIELD; + } + + return fact.createFieldAccess( + signature.getDeclaringType().getName(), + signature.getName(), + BcelWorld.makeBcelType(signature.getReturnType()), + kind); + } + + public static Instruction createSet(InstructionFactory fact, Member signature) { + short kind; + if (signature.isStatic()) { + kind = Constants.PUTSTATIC; + } else { + kind = Constants.PUTFIELD; + } + + return fact.createFieldAccess( + signature.getDeclaringType().getName(), + signature.getName(), + BcelWorld.makeBcelType(signature.getReturnType()), + kind); + } + + public static Instruction createInvoke( + InstructionFactory fact, + JavaClass declaringClass, + Method newMethod) { + short kind; + if (newMethod.isInterface()) { + kind = Constants.INVOKEINTERFACE; + } else if (newMethod.isPrivate() || newMethod.getName().equals("<init>")) { + kind = Constants.INVOKESPECIAL; + } else if (newMethod.isStatic()) { + kind = Constants.INVOKESTATIC; + } else { + kind = Constants.INVOKEVIRTUAL; + } + + return fact.createInvoke( + declaringClass.getClassName(), + newMethod.getName(), + Type.getReturnType(newMethod.getSignature()), + Type.getArgumentTypes(newMethod.getSignature()), + kind); + } + + public static Instruction createInstanceof(InstructionFactory fact, ObjectType t) { + return new INSTANCEOF(fact.getConstantPool().addClass(t)); + } + + + public static Instruction createInvoke( + InstructionFactory fact, + LazyMethodGen m) { + short kind; + if (m.getEnclosingClass().isInterface()) { + kind = Constants.INVOKEINTERFACE; + } else if (m.isPrivate() || m.getName().equals("<init>")) { + kind = Constants.INVOKESPECIAL; + } else if (m.isStatic()) { + kind = Constants.INVOKESTATIC; + } else { + kind = Constants.INVOKEVIRTUAL; + } + + return fact.createInvoke( + m.getClassName(), + m.getName(), + m.getReturnType(), + m.getArgumentTypes(), + kind); + } + + + // ??? these should perhaps be cached. Remember to profile this to see if it's a problem. + public static String[] makeArgNames(int n) { + String[] ret = new String[n]; + for (int i=0; i<n; i++) { + ret[i] = "arg" + i; + } + return ret; + } + + public static void appendConversion( + InstructionList il, + InstructionFactory fact, + ResolvedTypeX fromType, + ResolvedTypeX toType) + { + if (! toType.isConvertableFrom(fromType)) { + throw new BCException("can't convert from " + fromType + " to " + toType); + } + if (toType.needsNoConversionFrom(fromType)) return; + + if (toType.equals(ResolvedTypeX.VOID)) { + // assert fromType.equals(TypeX.OBJECT) + il.append(fact.createPop(fromType.getSize())); + } else if (fromType.equals(ResolvedTypeX.VOID)) { + // assert toType.equals(TypeX.OBJECT) + il.append(fact.createNull(Type.OBJECT)); + return; + } else if (fromType.equals(TypeX.OBJECT)) { + Type to = BcelWorld.makeBcelType(toType); + if (toType.isPrimitive()) { + String name = toType.toString() + "Value"; + il.append( + fact.createInvoke( + "org.aspectj.runtime.internal.Conversions", + name, + to, + new Type[] { Type.OBJECT }, + Constants.INVOKESTATIC)); + } else { + il.append(fact.createCheckCast((ReferenceType)to)); + } + } else if (toType.equals(TypeX.OBJECT)) { + // assert fromType.isPrimitive() + Type from = BcelWorld.makeBcelType(fromType); + String name = fromType.toString() + "Object"; + il.append( + fact.createInvoke( + "org.aspectj.runtime.internal.Conversions", + name, + Type.OBJECT, + new Type[] { from }, + Constants.INVOKESTATIC)); + } else if (fromType.isPrimitive()) { + // assert toType.isPrimitive() + Type from = BcelWorld.makeBcelType(fromType); + Type to = BcelWorld.makeBcelType(toType); + try { + il.append(fact.createCast(from, to)); + } catch (RuntimeException e) { + il.append(fact.createCast(from, Type.INT)); + il.append(fact.createCast(Type.INT, to)); + } + } else { + Type to = BcelWorld.makeBcelType(toType); + // assert ! fromType.isPrimitive() && ! toType.isPrimitive() + il.append(fact.createCheckCast((ReferenceType) to)); + } + } + + + public static InstructionList createConversion( + InstructionFactory fact, + Type fromType, + Type toType) { + //System.out.println("cast to: " + toType); + + InstructionList il = new InstructionList(); + if (fromType.equals(toType)) + return il; + if (toType.equals(Type.VOID)) { + il.append(fact.createPop(fromType.getSize())); + return il; + } + + if (fromType.equals(Type.VOID)) { + if (toType instanceof BasicType) + throw new BCException("attempting to cast from void to basic type"); + il.append(fact.createNull(Type.OBJECT)); + return il; + } + + if (fromType.equals(Type.OBJECT)) { + if (toType instanceof BasicType) { + String name = toType.toString() + "Value"; + il.append( + fact.createInvoke( + "org.aspectj.runtime.internal.Conversions", + name, + toType, + new Type[] { Type.OBJECT }, + Constants.INVOKESTATIC)); + return il; + } + } + + if (toType.equals(Type.OBJECT)) { + if (fromType instanceof BasicType) { + String name = fromType.toString() + "Object"; + il.append( + fact.createInvoke( + "org.aspectj.runtime.internal.Conversions", + name, + Type.OBJECT, + new Type[] { fromType }, + Constants.INVOKESTATIC)); + return il; + } else if (fromType instanceof ReferenceType) { + return il; + } else { + throw new RuntimeException(); + } + } + + if (fromType instanceof ReferenceType + && ((ReferenceType)fromType).isAssignmentCompatibleWith(toType)) { + return il; + } + + il.append(fact.createCast(fromType, toType)); + return il; + } + + public static Instruction createConstant( + InstructionFactory fact, + int i) { + Instruction inst; + switch(i) { + case -1: inst = fact.ICONST_M1; break; + case 0: inst = fact.ICONST_0; break; + case 1: inst = fact.ICONST_1; break; + case 2: inst = fact.ICONST_2; break; + case 3: inst = fact.ICONST_3; break; + case 4: inst = fact.ICONST_4; break; + case 5: inst = fact.ICONST_5; break; + } + if (i <= Byte.MAX_VALUE && i >= Byte.MIN_VALUE) { + inst = new BIPUSH((byte)i); + } else if (i <= Short.MAX_VALUE && i >= Short.MIN_VALUE) { + inst = new SIPUSH((short)i); + } else { + inst = new LDC(fact.getClassGen().getConstantPool().addInteger(i)); + } + return inst; + } + + public static JavaClass makeJavaClass(String filename, byte[] bytes) { + try { + ClassParser parser = new ClassParser(new ByteArrayInputStream(bytes), filename); + return parser.parse(); + } catch (IOException e) { + throw new BCException("malformed class file"); + } + } + + public static String arrayToString(int[] a) { + int len = a.length; + if (len == 0) return "[]"; + StringBuffer buf = new StringBuffer("["); + buf.append(a[0]); + for (int i = 1; i < len; i++) { + buf.append(", "); + buf.append(a[i]); + } + buf.append("]"); + return buf.toString(); + } + + /** + * replace an instruction handle with another instruction, in this case, a branch instruction. + * + * @param ih the instruction handle to replace. + * @param branchInstruction the branch instruction to replace ih with + * @param enclosingMethod where to find ih's instruction list. + */ + public static void replaceInstruction( + InstructionHandle ih, + BranchInstruction branchInstruction, + LazyMethodGen enclosingMethod) + { + + InstructionList il = enclosingMethod.getBody(); + InstructionHandle fresh = il.append(ih, branchInstruction); + deleteInstruction(ih, fresh, enclosingMethod); + } + + /** delete an instruction handle and retarget all targeters of the deleted instruction + * to the next instruction. Obviously, this should not be used to delete + * a control transfer instruction unless you know what you're doing. + * + * @param ih the instruction handle to delete. + * @param enclosingMethod where to find ih's instruction list. + */ + public static void deleteInstruction( + InstructionHandle ih, + LazyMethodGen enclosingMethod) + { + deleteInstruction(ih, ih.getNext(), enclosingMethod); + } + + + /** delete an instruction handle and retarget all targeters of the deleted instruction + * to the provided target. + * + * @param ih the instruction handle to delete + * @param retargetTo the instruction handle to retarget targeters of ih to. + * @param enclosingMethod where to find ih's instruction list. + */ + public static void deleteInstruction( + InstructionHandle ih, + InstructionHandle retargetTo, + LazyMethodGen enclosingMethod) + { + InstructionList il = enclosingMethod.getBody(); + InstructionTargeter[] targeters = ih.getTargeters(); + if (targeters != null) { + for (int i = targeters.length - 1; i >= 0; i--) { + InstructionTargeter targeter = targeters[i]; + targeter.updateTarget(ih, retargetTo); + } + ih.removeAllTargeters(); + } + try { + il.delete(ih); + } catch (TargetLostException e) { + throw new BCException("this really can't happen"); + } + } + + /** returns -1 if no source line attribute */ + public static int getSourceLine(InstructionHandle ih) { + InstructionTargeter[] ts = ih.getTargeters(); + if (ts != null) { + for (int j = ts.length - 1; j >= 0; j--) { + InstructionTargeter t = ts[j]; + if (t instanceof LineNumberTag) { + return ((LineNumberTag)t).getLineNumber(); + } + } + } + return -1; + } + + // assumes that there is no already extant source line tag. Otherwise we'll have to be better. + public static void setSourceLine(InstructionHandle ih, int lineNumber) { + ih.addTargeter(new LineNumberTag(lineNumber)); + } + + public static int makePublic(int i) { + return i & ~(Modifier.PROTECTED | Modifier.PRIVATE) | Modifier.PUBLIC; + } + public static int makePrivate(int i) { + return i & ~(Modifier.PROTECTED | Modifier.PUBLIC) | Modifier.PRIVATE; + } + public static BcelVar[] pushAndReturnArrayOfVars( + ResolvedTypeX[] proceedParamTypes, + InstructionList il, + InstructionFactory fact, + LazyMethodGen enclosingMethod) + { + int len = proceedParamTypes.length; + BcelVar[] ret = new BcelVar[len]; + + for (int i = len - 1; i >= 0; i--) { + ResolvedTypeX typeX = proceedParamTypes[i]; + Type type = BcelWorld.makeBcelType(typeX); + int local = enclosingMethod.allocateLocal(type); + + il.append(fact.createStore(type, local)); + ret[i] = new BcelVar(typeX, local); + } + return ret; + } + +} diff --git a/weaver/src/org/aspectj/weaver/bcel/ZipFileWeaver.java b/weaver/src/org/aspectj/weaver/bcel/ZipFileWeaver.java new file mode 100644 index 000000000..c243ec6bf --- /dev/null +++ b/weaver/src/org/aspectj/weaver/bcel/ZipFileWeaver.java @@ -0,0 +1,41 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.bcel; + +import java.io.*; +import java.util.zip.*; + +import org.aspectj.util.FileUtil; + + +//XXX delete very soon +public class ZipFileWeaver { + File inFile; + public ZipFileWeaver(File inFile) { + super(); + this.inFile = inFile; + } + + public void weave(BcelWeaver weaver, File outFile) throws IOException { + int count = 0; + long startTime = System.currentTimeMillis(); + weaver.addJarFile(inFile, new File(".")); + weaver.weave(outFile); + long stopTime = System.currentTimeMillis(); + + + System.out.println("handled " + count + " entries, in " + + (stopTime-startTime)/1000. + " seconds"); + } +} diff --git a/weaver/src/org/aspectj/weaver/patterns/AndPointcut.java b/weaver/src/org/aspectj/weaver/patterns/AndPointcut.java new file mode 100644 index 000000000..96a2b4ce2 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/patterns/AndPointcut.java @@ -0,0 +1,84 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.patterns; + +import java.io.*; +import java.util.Map; + +import org.aspectj.weaver.*; +import org.aspectj.weaver.ast.*; +import org.aspectj.util.*; + +public class AndPointcut extends Pointcut { + Pointcut left, right; // exposed for testing + + public AndPointcut(Pointcut left, Pointcut right) { + super(); + this.left = left; + this.right = right; + setLocation(left.getSourceContext(), left.getStart(), right.getEnd()); + } + + public FuzzyBoolean match(Shadow shadow) { + return left.match(shadow).and(right.match(shadow)); + } + + public String toString() { + return "(" + left.toString() + " && " + right.toString() + ")"; + } + + public boolean equals(Object other) { + if (!(other instanceof AndPointcut)) return false; + AndPointcut o = (AndPointcut)other; + return o.left.equals(left) && o.right.equals(right); + } + + public int hashCode() { + int result = 19; + result = 37*result + left.hashCode(); + result = 37*result + right.hashCode(); + return result; + } + + public void resolveBindings(IScope scope, Bindings bindings) { + left.resolveBindings(scope, bindings); + right.resolveBindings(scope, bindings); + } + + + + public void write(DataOutputStream s) throws IOException { + s.writeByte(Pointcut.AND); + left.write(s); + right.write(s); + writeLocation(s); + } + + public static Pointcut read(DataInputStream s, ISourceContext context) throws IOException { + AndPointcut ret = new AndPointcut(Pointcut.read(s, context), Pointcut.read(s, context)); + ret.readLocation(context, s); + return ret; + } + + + public Test findResidue(Shadow shadow, ExposedState state) { + return Test.makeAnd(left.findResidue(shadow, state), right.findResidue(shadow, state)); + } + + public Pointcut concretize1(ResolvedTypeX inAspect, IntMap bindings) { + return new AndPointcut(left.concretize1(inAspect, bindings), + right.concretize1(inAspect, bindings)); + } + +} diff --git a/weaver/src/org/aspectj/weaver/patterns/AndTypePattern.java b/weaver/src/org/aspectj/weaver/patterns/AndTypePattern.java new file mode 100644 index 000000000..86a63f971 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/patterns/AndTypePattern.java @@ -0,0 +1,79 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.patterns; + +import java.io.*; + +import org.aspectj.util.FuzzyBoolean; +import org.aspectj.weaver.*; +import org.aspectj.weaver.ResolvedTypeX; + +/** + * left && right + * + * <p>any binding to formals is explicitly forbidden for any composite by the language + * + * @author Erik Hilsdale + * @author Jim Hugunin + */ +public class AndTypePattern extends TypePattern { + private TypePattern left, right; + + public AndTypePattern(TypePattern left, TypePattern right) { + super(false); //??? we override all methods that care about includeSubtypes + this.left = left; + this.right = right; + setLocation(left.getSourceContext(), left.getStart(), right.getEnd()); + } + + public FuzzyBoolean matchesInstanceof(ResolvedTypeX type) { + return left.matchesInstanceof(type).and(right.matchesInstanceof(type)); + } + + protected boolean matchesExactly(ResolvedTypeX type) { + //??? if these had side-effects, this sort-circuit could be a mistake + return left.matchesExactly(type) && right.matchesExactly(type); + } + + public boolean matchesStatically(ResolvedTypeX type) { + return left.matchesStatically(type) && right.matchesStatically(type); + } + + public void write(DataOutputStream s) throws IOException { + s.writeByte(TypePattern.AND); + left.write(s); + right.write(s); + writeLocation(s); + } + + public static TypePattern read(DataInputStream s, ISourceContext context) throws IOException { + TypePattern ret = new AndTypePattern(TypePattern.read(s, context), TypePattern.read(s, context)); + ret.readLocation(context, s); + return ret; + } + + public TypePattern resolveBindings( + IScope scope, + Bindings bindings, + boolean allowBinding) { + left = left.resolveBindings(scope, bindings, false); + right = right.resolveBindings(scope, bindings, false); + return this; + } + + public String toString() { + return "(" + left.toString() + " && " + right.toString() + ")"; + } + +} diff --git a/weaver/src/org/aspectj/weaver/patterns/ArgsPointcut.java b/weaver/src/org/aspectj/weaver/patterns/ArgsPointcut.java new file mode 100644 index 000000000..4c3d22449 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/patterns/ArgsPointcut.java @@ -0,0 +1,140 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.patterns; + +import java.io.*; + +import org.aspectj.util.FuzzyBoolean; +import org.aspectj.weaver.*; +import org.aspectj.weaver.ast.*; + +/** + * args(arguments) + * + * @author Erik Hilsdale + * @author Jim Hugunin + */ +public class ArgsPointcut extends NameBindingPointcut { + TypePatternList arguments; + + public ArgsPointcut(TypePatternList arguments) { + this.arguments = arguments; + } + + public FuzzyBoolean match(Shadow shadow) { + int n = shadow.getArgCount(); + TypeX[] argTypes = new TypeX[n]; + for (int i=0; i < n; i++) { + argTypes[i] = shadow.getArgType(i); + } + FuzzyBoolean ret = + arguments.matches(shadow.getIWorld().resolve(argTypes), TypePattern.DYNAMIC); + return ret; + } + + public void write(DataOutputStream s) throws IOException { + s.writeByte(Pointcut.ARGS); + arguments.write(s); + writeLocation(s); + } + + public static Pointcut read(DataInputStream s, ISourceContext context) throws IOException { + ArgsPointcut ret = new ArgsPointcut(TypePatternList.read(s, context)); + ret.readLocation(context, s); + return ret; + } + + + public boolean equals(Object other) { + if (!(other instanceof ArgsPointcut)) return false; + ArgsPointcut o = (ArgsPointcut)other; + return o.arguments.equals(this.arguments); + } + + public int hashCode() { + return arguments.hashCode(); + } + + public void resolveBindings(IScope scope, Bindings bindings) { + arguments.resolveBindings(scope, bindings, true); + } + + public void postRead(ResolvedTypeX enclosingType) { + arguments.postRead(enclosingType); + } + + + public Pointcut concretize1(ResolvedTypeX inAspect, IntMap bindings) { + return new ArgsPointcut(arguments.resolveReferences(bindings)); + } + + private Test findResidueNoEllipsis(Shadow shadow, ExposedState state, TypePattern[] patterns) { + int len = shadow.getArgCount(); + //System.err.println("boudn to : " + len + ", " + patterns.length); + if (patterns.length != len) { + throw new RuntimeException("this should never happen"); + //return Literal.FALSE; //??? this should never happen + } + + Test ret = Literal.TRUE; + + for (int i=0; i < len; i++) { + TypeX argType = shadow.getArgType(i); + TypePattern type = patterns[i]; + if (!(type instanceof BindingTypePattern)) { + if (type.matchesInstanceof(shadow.getIWorld().resolve(argType)).alwaysTrue()) { + continue; + } + } + ret = Test.makeAnd(ret, + exposeStateForVar(shadow.getArgVar(i), type, state,shadow.getIWorld())); + } + + return ret; + } + + public Test findResidue(Shadow shadow, ExposedState state) { + int ellipsisCount = arguments.ellipsisCount; + if (ellipsisCount == 0) { + return findResidueNoEllipsis(shadow, state, arguments.getTypePatterns()); + } else if (ellipsisCount == 1) { + TypePattern[] patternsWithEllipsis = arguments.getTypePatterns(); + TypePattern[] patternsWithoutEllipsis = new TypePattern[shadow.getArgCount()]; + int lenWithEllipsis = patternsWithEllipsis.length; + int lenWithoutEllipsis = patternsWithoutEllipsis.length; + // l1+1 >= l0 + int indexWithEllipsis = 0; + int indexWithoutEllipsis = 0; + while (indexWithoutEllipsis < lenWithoutEllipsis) { + TypePattern p = patternsWithEllipsis[indexWithEllipsis++]; + if (p == TypePattern.ELLIPSIS) { + int newLenWithoutEllipsis = + lenWithoutEllipsis - (lenWithEllipsis-indexWithEllipsis); + while (indexWithoutEllipsis < newLenWithoutEllipsis) { + patternsWithoutEllipsis[indexWithoutEllipsis++] = TypePattern.ANY; + } + } else { + patternsWithoutEllipsis[indexWithoutEllipsis++] = p; + } + } + return findResidueNoEllipsis(shadow, state, patternsWithoutEllipsis); + } else { + throw new BetaException("unimplemented"); + } + } + + public String toString() { + return "args" + arguments.toString() + ""; + } +} diff --git a/weaver/src/org/aspectj/weaver/patterns/BasicToken.java b/weaver/src/org/aspectj/weaver/patterns/BasicToken.java new file mode 100644 index 000000000..9a36b35e7 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/patterns/BasicToken.java @@ -0,0 +1,75 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.patterns; + + +public class BasicToken implements IToken { + private String value; + private boolean isIdentifier; + private String literalKind; + + private int start; + private int end; + + public static BasicToken makeOperator(String value, int start, int end) { + return new BasicToken(value.intern(), false, null, start, end); + } + + public static BasicToken makeIdentifier(String value, int start, int end) { + return new BasicToken(value, true, null, start, end); + } + + public static BasicToken makeLiteral(String value, String kind, int start, int end) { + return new BasicToken(value, false, kind.intern(), start, end); + } + + + private BasicToken(String value, boolean isIdentifier, String literalKind, int start, int end) { + this.value = value; + this.isIdentifier = isIdentifier; + this.literalKind = literalKind; + this.start = start; + this.end = end; + } + + public int getStart() { return start; } + public int getEnd() { return end; } + public String getFileName() { return "unknown"; } + + public String getString() { + return value; + } + + public boolean isIdentifier() { + return isIdentifier; + } + + public Pointcut maybeGetParsedPointcut() { + return null; + } + + + + public String toString() { + String s; + if (isIdentifier) s = value; + else s = "'" + value + "'"; + + return s + "@" + start + ":" + end; + } + public String getLiteralKind() { + return literalKind; + } + +} diff --git a/weaver/src/org/aspectj/weaver/patterns/BasicTokenSource.java b/weaver/src/org/aspectj/weaver/patterns/BasicTokenSource.java new file mode 100644 index 000000000..ef4db5bf8 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/patterns/BasicTokenSource.java @@ -0,0 +1,153 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.patterns; + +import java.util.*; + +import org.aspectj.weaver.*; + + +public class BasicTokenSource implements ITokenSource { + private int index = 0; + private IToken[] tokens; + private ISourceContext sourceContext; + + public BasicTokenSource(IToken[] tokens, ISourceContext sourceContext) { + this.tokens = tokens; + this.sourceContext = sourceContext; + } + + public int getIndex() { + return index; + } + + public void setIndex(int newIndex) { + this.index = newIndex; + } + + public IToken next() { + try { + return tokens[index++]; + } catch (ArrayIndexOutOfBoundsException e) { + return IToken.EOF; + } + } + + public IToken peek() { + try { + return tokens[index]; + } catch (ArrayIndexOutOfBoundsException e) { + return IToken.EOF; + } + } + + public IToken peek(int offset) { + try { + return tokens[index+offset]; + } catch (ArrayIndexOutOfBoundsException e) { + return IToken.EOF; + } + } + + public String toString() { + StringBuffer buf = new StringBuffer(); + buf.append("["); + for (int i = 0; i < tokens.length; i++) { + IToken t = tokens[i]; + if (t == null) + break; + if (i > 0) + buf.append(", "); + buf.append(t.toString()); + } + buf.append("]"); + return buf.toString(); + } + + + ////////////////////////////////////////////////////// + // Convenience, maybe just for testing + static ITokenSource makeTokenSource(String input) { + char[] chars = input.toCharArray(); + + int i = 0; + List tokens = new ArrayList(); + + while (i < chars.length) { + char ch = chars[i++]; + switch(ch) { + case ' ': + case '\t': + case '\n': + case '\r': + continue; + case '*': + case '.': + case '(': + case ')': + case '+': + case '[': + case ']': + case ',': + case '!': + case ':': + tokens.add(BasicToken.makeOperator(makeString(ch), i-1, i-1)); + continue; + case '&': + case '|': + if (i == chars.length) { + throw new BCException("bad " + ch); + } + char nextChar = chars[i++]; + if (nextChar == ch) { + tokens.add(BasicToken.makeOperator(makeString(ch, 2), i-2, i-1)); + } else { + throw new RuntimeException("bad " + ch); + } + continue; + + case '\"': + int start0 = i-1; + while (i < chars.length && !(chars[i]=='\"')) i++; + i += 1; + tokens.add(BasicToken.makeLiteral(new String(chars, start0+1, i-start0-2), "string", start0, i-1)); + default: + int start = i-1; + while (i < chars.length && Character.isJavaIdentifierPart(chars[i])) { i++; } + tokens.add(BasicToken.makeIdentifier(new String(chars, start, i-start), start, i-1)); + + } + } + + //System.out.println(tokens); + + return new BasicTokenSource((IToken[])tokens.toArray(new IToken[tokens.size()]), null); + } + + private static String makeString(char ch) { + // slightly inefficient ;-) + return new String(new char[] {ch}); + } + + private static String makeString(char ch, int count) { + // slightly inefficient ;-) + char[] chars = new char[count]; + for (int i=0; i<count; i++) { chars[i] = ch; } + return new String(chars); + } + public ISourceContext getSourceContext() { + return sourceContext; + } + +} diff --git a/weaver/src/org/aspectj/weaver/patterns/BindingTypePattern.java b/weaver/src/org/aspectj/weaver/patterns/BindingTypePattern.java new file mode 100644 index 000000000..836b6ac4c --- /dev/null +++ b/weaver/src/org/aspectj/weaver/patterns/BindingTypePattern.java @@ -0,0 +1,74 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.patterns; + +import java.io.*; + +import org.aspectj.weaver.*; + +public class BindingTypePattern extends ExactTypePattern { + private int formalIndex; + + public BindingTypePattern(TypeX type, int index) { + super(type, false); + this.formalIndex = index; + } + + public BindingTypePattern(FormalBinding binding) { + this(binding.getType(), binding.getIndex()); + } + + public int getFormalIndex() { + return formalIndex; + } + + public boolean equals(Object other) { + if (!(other instanceof BindingTypePattern)) return false; + BindingTypePattern o = (BindingTypePattern)other; + return o.type.equals(this.type) && o.formalIndex == this.formalIndex; + } + public int hashCode() { + int result = 17; + result = 37*result + type.hashCode(); + result = 37*result + formalIndex; + return result; + } + + public void write(DataOutputStream out) throws IOException { + out.writeByte(TypePattern.BINDING); + type.write(out); + out.writeShort((short)formalIndex); + writeLocation(out); + } + + public static TypePattern read(DataInputStream s, ISourceContext context) throws IOException { + TypePattern ret = new BindingTypePattern(TypeX.read(s), s.readShort()); + ret.readLocation(context, s); + return ret; + } + + public TypePattern remapAdviceFormals(IntMap bindings) { + if (!bindings.hasKey(formalIndex)) { + return new ExactTypePattern(type, false); + } else { + int newFormalIndex = bindings.get(formalIndex); + return new BindingTypePattern(type, newFormalIndex); + } + } + + public String toString() { + //Thread.currentThread().dumpStack(); + return "BindingTypePattern(" + type.toString() + ", " + formalIndex + ")"; + } +} diff --git a/weaver/src/org/aspectj/weaver/patterns/Bindings.java b/weaver/src/org/aspectj/weaver/patterns/Bindings.java new file mode 100644 index 000000000..b0a1e4eda --- /dev/null +++ b/weaver/src/org/aspectj/weaver/patterns/Bindings.java @@ -0,0 +1,127 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.patterns; + +import org.aspectj.weaver.BCException; +import org.aspectj.bridge.IMessage; + +public class Bindings { + public static final Bindings NONE = new Bindings(0); + + private BindingTypePattern[] bindings; + + public Bindings(BindingTypePattern[] bindings) { + this.bindings = bindings; + } + + public Bindings(int count) { + this(new BindingTypePattern[count]); + } + + public void register(BindingTypePattern binding, IScope scope) { + int index = binding.getFormalIndex(); + BindingTypePattern existingBinding = bindings[index]; + if (existingBinding != null) { + scope.message(IMessage.ERROR, existingBinding, binding, + "multiple bindings" + index + ", " + binding); + } + bindings[index] = binding; + } + + public void mergeIn(Bindings other, IScope scope) { + for (int i=0, len=other.bindings.length; i < len; i++) { + if (other.bindings[i] != null) { + register(other.bindings[i], scope); + } + } + } + + + + /** + * signals an error if one has a binding and other doesn't + */ + public void checkEquals(Bindings other, IScope scope) { + BindingTypePattern[] b1 = this.bindings; + BindingTypePattern[] b2 = other.bindings; + int len = b1.length; + if (len != b2.length) { + throw new BCException("INSANE"); + } + + for (int i=0; i < len; i++) { + if (b1[i] == null && b2[i] != null) { + scope.message(IMessage.ERROR, b2[i], "inconsistent binding"); + b1[i] = b2[i]; // done just to produce fewer error messages + } else if (b2[i] == null && b1[i] != null) { + scope.message(IMessage.ERROR, b1[i], "inconsistent binding"); + b2[i] = b1[i]; // done just to produce fewer error messages + } + } + } + + public String toString() { + StringBuffer buf = new StringBuffer("Bindings("); + for (int i=0, len=bindings.length; i < len; i++) { + if (i > 0) buf.append(", "); + buf.append(bindings[i]); + } + buf.append(")"); + return buf.toString(); + } + + + public int[] getUsedFormals() { + //System.out.println("used: " + this); + int[] ret = new int[bindings.length]; + int index = 0; + for (int i=0, len=bindings.length; i < len; i++) { + if (bindings[i] != null) { + ret[index++] = i; + } + } + int[] newRet = new int[index]; + System.arraycopy(ret, 0, newRet, 0, index); + //System.out.println("ret: " + index); + return newRet; + } + + + public Bindings copy() { +// int len = bindings.length; +// boolean[] a = new boolean[len]; +// System.arraycopy(bindings, 0, a, 0, len); + return new Bindings((BindingTypePattern[])bindings.clone()); + } + + public void checkAllBound(IScope scope) { + for (int i=0, len=bindings.length; i < len; i++) { + if (bindings[i] == null) { + scope.message(IMessage.ERROR, scope.getFormal(i), "formal unbound in pointcut"); + } + } + + } + + public int size() { return bindings.length; } + + public void checkEmpty(IScope scope, String message) { + for (int i=0, len=bindings.length; i < len; i++) { + if (bindings[i] != null) { + scope.message(IMessage.ERROR, bindings[i], message); + } + } + } + +} diff --git a/weaver/src/org/aspectj/weaver/patterns/CflowPointcut.java b/weaver/src/org/aspectj/weaver/patterns/CflowPointcut.java new file mode 100644 index 000000000..72348d45b --- /dev/null +++ b/weaver/src/org/aspectj/weaver/patterns/CflowPointcut.java @@ -0,0 +1,165 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.patterns; + +import java.io.*; +import java.lang.reflect.Modifier; +import java.util.*; +import java.util.List; + +import org.aspectj.weaver.*; +import org.aspectj.weaver.ast.*; +import org.aspectj.weaver.bcel.*; +import org.aspectj.weaver.bcel.BcelTypeMunger; +import org.aspectj.util.*; + + +public class CflowPointcut extends Pointcut { + private Pointcut entry; + private boolean isBelow; + private int[] freeVars; + + /** + * Used to indicate that we're in the context of a cflow when concretizing if's + * + * Will be removed or replaced with something better when we handle this + * as a non-error + */ + public static final ResolvedPointcutDefinition CFLOW_MARKER = + new ResolvedPointcutDefinition(null, 0, null, TypeX.NONE, Pointcut.makeMatchesNothing(Pointcut.RESOLVED)); + + + public CflowPointcut(Pointcut entry, boolean isBelow, int[] freeVars) { + this.entry = entry; + this.isBelow = isBelow; + this.freeVars = freeVars; + } + + public FuzzyBoolean match(Shadow shadow) { + //??? this is not maximally efficient + return FuzzyBoolean.MAYBE; + } + + public void write(DataOutputStream s) throws IOException { + s.writeByte(Pointcut.CFLOW); + entry.write(s); + s.writeBoolean(isBelow); + FileUtil.writeIntArray(s, freeVars); + writeLocation(s); + } + public static Pointcut read(DataInputStream s, ISourceContext context) throws IOException { + + CflowPointcut ret = new CflowPointcut(Pointcut.read(s, context), s.readBoolean(), FileUtil.readIntArray(s)); + ret.readLocation(context, s); + return ret; + } + + public void resolveBindings(IScope scope, Bindings bindings) { + if (bindings == null) { + entry.resolveBindings(scope, null); + entry.state = RESOLVED; + freeVars = new int[0]; + } else { + //??? for if's sake we might need to be more careful here + Bindings entryBindings = new Bindings(bindings.size()); + + entry.resolveBindings(scope, entryBindings); + entry.state = RESOLVED; + + freeVars = entryBindings.getUsedFormals(); + + bindings.mergeIn(entryBindings, scope); + } + } + + public boolean equals(Object other) { + if (!(other instanceof CflowPointcut)) return false; + CflowPointcut o = (CflowPointcut)other; + return o.entry.equals(this.entry) && o.isBelow == this.isBelow; + } + public int hashCode() { + int result = 17; + result = 37*result + entry.hashCode(); + result = 37*result + (isBelow ? 0 : 1); + return result; + } + public String toString() { + return "cflow" + (isBelow ? "below" : "") + "(" + entry + ")"; + } + + public Test findResidue(Shadow shadow, ExposedState state) { + throw new RuntimeException("unimplemented"); + } + + + public Pointcut concretize1(ResolvedTypeX inAspect, IntMap bindings) { + //make this remap from formal positions to arrayIndices + IntMap entryBindings = new IntMap(); + for (int i=0, len=freeVars.length; i < len; i++) { + int freeVar = freeVars[i]; + //int formalIndex = bindings.get(freeVar); + entryBindings.put(freeVar, i); + } + entryBindings.copyContext(bindings); + //System.out.println(this + " bindings: " + entryBindings); + + World world = inAspect.getWorld(); + + Pointcut concreteEntry; + + CrosscuttingMembers xcut = inAspect.crosscuttingMembers; + Collection previousCflowEntries = xcut.getCflowEntries(); + + entryBindings.pushEnclosingDefinition(CFLOW_MARKER); + try { + concreteEntry = entry.concretize(inAspect, entryBindings); + } finally { + entryBindings.popEnclosingDefinitition(); + } + + List innerCflowEntries = new ArrayList(xcut.getCflowEntries()); + innerCflowEntries.removeAll(previousCflowEntries); + + + ResolvedMember cflowField = new ResolvedMember( + Member.FIELD, inAspect, Modifier.STATIC | Modifier.PUBLIC | Modifier.FINAL, + NameMangler.cflowStack(xcut), + TypeX.forName(NameMangler.CFLOW_STACK_TYPE).getSignature()); + + // add field and initializer to inAspect + //XXX and then that info above needs to be mapped down here to help with + //XXX getting the exposed state right + inAspect.crosscuttingMembers.addConcreteShadowMunger( + Advice.makeCflowEntry(world, concreteEntry, isBelow, cflowField, freeVars.length, innerCflowEntries)); + + inAspect.crosscuttingMembers.addTypeMunger( + world.makeCflowStackFieldAdder(cflowField)); + + + List slots = new ArrayList(); + for (int i=0, len=freeVars.length; i < len; i++) { + int freeVar = freeVars[i]; + int formalIndex = bindings.get(freeVar); + ResolvedTypeX formalType = + bindings.getAdviceSignature().getParameterTypes()[formalIndex].resolve(world); + + ConcreteCflowPointcut.Slot slot = + new ConcreteCflowPointcut.Slot(formalIndex, formalType, i); + slots.add(slot); + } + + return new ConcreteCflowPointcut(cflowField, slots); + } + +} diff --git a/weaver/src/org/aspectj/weaver/patterns/ConcreteCflowPointcut.java b/weaver/src/org/aspectj/weaver/patterns/ConcreteCflowPointcut.java new file mode 100644 index 000000000..3d3301b41 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/patterns/ConcreteCflowPointcut.java @@ -0,0 +1,116 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.patterns; + +import java.io.*; +import java.lang.reflect.Modifier; +import java.util.*; +import java.util.List; + +import org.aspectj.weaver.*; +import org.aspectj.weaver.ast.*; +import org.aspectj.weaver.bcel.*; +import org.aspectj.weaver.bcel.BcelTypeMunger; +import org.aspectj.util.*; + + +public class ConcreteCflowPointcut extends Pointcut { + private Member cflowStackField; + List/*Slot*/ slots; // exposed for testing + + + public ConcreteCflowPointcut(Member cflowStackField, List slots) { + this.cflowStackField = cflowStackField; + this.slots = slots; + } + + public FuzzyBoolean match(Shadow shadow) { + //??? this is not maximally efficient + return FuzzyBoolean.MAYBE; + } + + public void write(DataOutputStream s) throws IOException { + throw new RuntimeException("unimplemented"); + } + + + public void resolveBindings(IScope scope, Bindings bindings) { + throw new RuntimeException("unimplemented"); + } + + + public boolean equals(Object other) { + if (!(other instanceof ConcreteCflowPointcut)) return false; + ConcreteCflowPointcut o = (ConcreteCflowPointcut)other; + return o.cflowStackField.equals(this.cflowStackField); + } + public int hashCode() { + int result = 17; + result = 37*result + cflowStackField.hashCode(); + return result; + } + public String toString() { + return "concretecflow(" + cflowStackField + ")"; + } + + public Test findResidue(Shadow shadow, ExposedState state) { + //System.out.println("find residue: " + this); + for (Iterator i = slots.iterator(); i.hasNext();) { + Slot slot = (Slot) i.next(); + //System.out.println("slot: " + slot.formalIndex); + state.set(slot.formalIndex, + new BcelCflowAccessVar(slot.formalType, cflowStackField, slot.arrayIndex)); + } + + return Test.makeFieldGetCall(cflowStackField, cflowStackIsValidMethod, Expr.NONE); + } + + private static final Member cflowStackIsValidMethod = + Member.method(TypeX.forName(NameMangler.CFLOW_STACK_TYPE), 0, "isValid", "()Z"); + + + public Pointcut concretize1(ResolvedTypeX inAspect, IntMap bindings) { + throw new RuntimeException("unimplemented"); + } + + + public static class Slot { + int formalIndex; + ResolvedTypeX formalType; + int arrayIndex; + + public Slot( + int formalIndex, + ResolvedTypeX formalType, + int arrayIndex) { + this.formalIndex = formalIndex; + this.formalType = formalType; + this.arrayIndex = arrayIndex; + } + + public boolean equals(Object other) { + if (!(other instanceof Slot)) return false; + + Slot o = (Slot)other; + return o.formalIndex == this.formalIndex && + o.arrayIndex == this.arrayIndex && + o.formalType.equals(this.formalType); + } + + public String toString() { + return "Slot(" + formalIndex + ", " + formalType + ", " + arrayIndex + ")"; + } + } + +} diff --git a/weaver/src/org/aspectj/weaver/patterns/Declare.java b/weaver/src/org/aspectj/weaver/patterns/Declare.java new file mode 100644 index 000000000..1dcb38bc0 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/patterns/Declare.java @@ -0,0 +1,54 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.patterns; + +import java.io.*; + +import org.aspectj.weaver.ISourceContext; + +public abstract class Declare extends PatternNode { + public static final byte ERROR_OR_WARNING = 1; + public static final byte PARENTS = 2; + public static final byte SOFT = 3; + public static final byte DOMINATES = 4; + + public static Declare read(DataInputStream s, ISourceContext context) throws IOException { + byte kind = s.readByte(); + switch (kind) { + case ERROR_OR_WARNING: + return DeclareErrorOrWarning.read(s, context); + case DOMINATES: + return DeclareDominates.read(s, context); + case PARENTS: + return DeclareParents.read(s, context); + case SOFT: + return DeclareSoft.read(s, context); + default: + throw new RuntimeException("unimplemented"); + } + } + + /** + * Returns this declare mutated + */ + public abstract void resolve(IScope scope); + + /** + * Indicates if this declare should be treated like advice. If true, the + * declare will have no effect in an abstract aspect. It will be inherited by + * any concrete aspects and will have an effect for each concrete aspect it + * is ultimately inherited by. + */ + public abstract boolean isAdviceLike(); +} diff --git a/weaver/src/org/aspectj/weaver/patterns/DeclareDominates.java b/weaver/src/org/aspectj/weaver/patterns/DeclareDominates.java new file mode 100644 index 000000000..964f6659d --- /dev/null +++ b/weaver/src/org/aspectj/weaver/patterns/DeclareDominates.java @@ -0,0 +1,110 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.patterns; + +import java.io.*; +import java.util.List; + +import org.aspectj.weaver.*; +import org.aspectj.weaver.ResolvedTypeX; + +public class DeclareDominates extends Declare { + private TypePatternList patterns; + + + public DeclareDominates(List patterns) { + this(new TypePatternList(patterns)); + } + + private DeclareDominates(TypePatternList patterns) { + this.patterns = patterns; + } + + public String toString() { + StringBuffer buf = new StringBuffer(); + buf.append("declare dominates: "); + buf.append(patterns); + buf.append(";"); + return buf.toString(); + } + + public boolean equals(Object other) { + if (!(other instanceof DeclareDominates)) return false; + DeclareDominates o = (DeclareDominates)other; + return o.patterns.equals(patterns); + } + + public int hashCode() { + return patterns.hashCode(); + } + + + public void write(DataOutputStream s) throws IOException { + s.writeByte(Declare.DOMINATES); + patterns.write(s); + writeLocation(s); + } + + public static Declare read(DataInputStream s, ISourceContext context) throws IOException { + Declare ret = new DeclareDominates(TypePatternList.read(s, context)); + ret.readLocation(context, s); + return ret; + } + + public void resolve(IScope scope) { + patterns = patterns.resolveBindings(scope, Bindings.NONE, false); + } + + public TypePatternList getPatterns() { + return patterns; + } + + private int matchingIndex(ResolvedTypeX a) { + int knownMatch = -1; + int starMatch = -1; + for (int i=0, len=patterns.size(); i < len; i++) { + TypePattern p = patterns.get(i); + if (p.isStar()) { + starMatch = i; + } else if (p.matchesExactly(a)) { + if (knownMatch != -1) { + throw new BCException("multiple matches: " + this + " with " + a); + } else { + knownMatch = i; + } + } + } + if (knownMatch == -1) return starMatch; + else return knownMatch; + } + + + public int compare(ResolvedTypeX aspect1, ResolvedTypeX aspect2) { + int index1 = matchingIndex(aspect1); + int index2 = matchingIndex(aspect2); + + //System.out.println("a1: " + aspect1 + ", " + aspect2 + " = " + index1 + ", " + index2); + + if (index1 == -1 || index2 == -1) return 0; + + if (index1 == index2) return 0; + else if (index1 > index2) return -1; + else return +1; + } + + public boolean isAdviceLike() { + return false; + } + +} diff --git a/weaver/src/org/aspectj/weaver/patterns/DeclareErrorOrWarning.java b/weaver/src/org/aspectj/weaver/patterns/DeclareErrorOrWarning.java new file mode 100644 index 000000000..f1e84cfa3 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/patterns/DeclareErrorOrWarning.java @@ -0,0 +1,99 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.patterns; + +import java.io.*; + +import org.aspectj.weaver.ISourceContext; + +public class DeclareErrorOrWarning extends Declare { + private boolean isError; + private Pointcut pointcut; + private String message; + + public DeclareErrorOrWarning(boolean isError, Pointcut pointcut, String message) { + this.isError = isError; + this.pointcut = pointcut; + this.message = message; + } + + public String toString() { + StringBuffer buf = new StringBuffer(); + buf.append("declare "); + if (isError) buf.append("error: "); + else buf.append("warning: "); + buf.append(pointcut); + buf.append(": "); + buf.append("\""); + buf.append(message); + buf.append("\";"); + return buf.toString(); + } + + public boolean equals(Object other) { + if (!(other instanceof DeclareErrorOrWarning)) return false; + DeclareErrorOrWarning o = (DeclareErrorOrWarning)other; + return (o.isError == isError) && + o.pointcut.equals(pointcut) && + o.message.equals(message); + } + + public int hashCode() { + int result = isError ? 19 : 23; + result = 37*result + pointcut.hashCode(); + result = 37*result + message.hashCode(); + return result; + } + + + public void write(DataOutputStream s) throws IOException { + s.writeByte(Declare.ERROR_OR_WARNING); + s.writeBoolean(isError); + pointcut.write(s); + s.writeUTF(message); + writeLocation(s); + } + + public static Declare read(DataInputStream s, ISourceContext context) throws IOException { + Declare ret = new DeclareErrorOrWarning( + s.readBoolean(), + Pointcut.read(s, context), + s.readUTF() + ); + ret.readLocation(context, s); + return ret; + } + + + public boolean isError() { + return isError; + } + + public String getMessage() { + return message; + } + + public Pointcut getPointcut() { + return pointcut; + } + + public void resolve(IScope scope) { + pointcut = pointcut.resolve(scope); + } + + public boolean isAdviceLike() { + return true; + } + +} diff --git a/weaver/src/org/aspectj/weaver/patterns/DeclareParents.java b/weaver/src/org/aspectj/weaver/patterns/DeclareParents.java new file mode 100644 index 000000000..d34d69ab1 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/patterns/DeclareParents.java @@ -0,0 +1,100 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.patterns; + +import java.io.*; +import java.util.List; + +import org.aspectj.weaver.*; +import org.aspectj.weaver.ResolvedTypeX; + +public class DeclareParents extends Declare { + private TypePattern child; + private TypePatternList parents; + + + public DeclareParents(TypePattern child, List parents) { + this(child, new TypePatternList(parents)); + } + + private DeclareParents(TypePattern child, TypePatternList parents) { + this.child = child; + this.parents = parents; + } + + public boolean match(ResolvedTypeX typeX) { + return child.matchesStatically(typeX); + } + + + + public String toString() { + StringBuffer buf = new StringBuffer(); + buf.append("declare parents: "); + buf.append(child); + buf.append(" extends "); //extends and implements are treated equivalently + buf.append(parents); + buf.append(";"); + return buf.toString(); + } + + public boolean equals(Object other) { + if (!(other instanceof DeclareParents)) return false; + DeclareParents o = (DeclareParents)other; + return o.child.equals(child) && o.parents.equals(parents); + } + + //??? cache this + public int hashCode() { + int result = 23; + result = 37*result + child.hashCode(); + result = 37*result + parents.hashCode(); + return result; + } + + + public void write(DataOutputStream s) throws IOException { + s.writeByte(Declare.PARENTS); + child.write(s); + parents.write(s); + writeLocation(s); + } + + public static Declare read(DataInputStream s, ISourceContext context) throws IOException { + Declare ret = new DeclareParents(TypePattern.read(s, context), TypePatternList.read(s, context)); + ret.readLocation(context, s); + return ret; + } + + public void resolve(IScope scope) { + child = child.resolveBindings(scope, Bindings.NONE, false); + parents = parents.resolveBindings(scope, Bindings.NONE, false); + for (int i=0; i < parents.size(); i++) { + parents.get(i).assertExactType(scope.getMessageHandler()); + } + } + + public TypePatternList getParents() { + return parents; + } + + public TypePattern getChild() { + return child; + } + + public boolean isAdviceLike() { + return false; + } + +} diff --git a/weaver/src/org/aspectj/weaver/patterns/DeclareSoft.java b/weaver/src/org/aspectj/weaver/patterns/DeclareSoft.java new file mode 100644 index 000000000..b80c17a21 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/patterns/DeclareSoft.java @@ -0,0 +1,87 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.patterns; + +import java.io.*; + +import org.aspectj.weaver.ISourceContext; + +public class DeclareSoft extends Declare { + private TypePattern exception; + private Pointcut pointcut; + + public DeclareSoft(TypePattern exception, Pointcut pointcut) { + this.exception = exception; + this.pointcut = pointcut; + } + + public String toString() { + StringBuffer buf = new StringBuffer(); + buf.append("declare soft: "); + buf.append(exception); + buf.append(": "); + buf.append(pointcut); + buf.append(";"); + return buf.toString(); + } + + public boolean equals(Object other) { + if (!(other instanceof DeclareSoft)) return false; + DeclareSoft o = (DeclareSoft)other; + return + o.pointcut.equals(pointcut) && + o.exception.equals(exception); + } + + public int hashCode() { + int result = 19; + result = 37*result + pointcut.hashCode(); + result = 37*result + exception.hashCode(); + return result; + } + + + public void write(DataOutputStream s) throws IOException { + s.writeByte(Declare.SOFT); + exception.write(s); + pointcut.write(s); + writeLocation(s); + } + + public static Declare read(DataInputStream s, ISourceContext context) throws IOException { + Declare ret = new DeclareSoft( + TypePattern.read(s, context), + Pointcut.read(s, context) + ); + ret.readLocation(context, s); + return ret; + } + + public Pointcut getPointcut() { + return pointcut; + } + + public TypePattern getException() { + return exception; + } + + public void resolve(IScope scope) { + exception = exception.resolveBindings(scope, null, false); + pointcut = pointcut.resolve(scope); + } + + public boolean isAdviceLike() { + return true; + } +} diff --git a/weaver/src/org/aspectj/weaver/patterns/ExactTypePattern.java b/weaver/src/org/aspectj/weaver/patterns/ExactTypePattern.java new file mode 100644 index 000000000..b98463ead --- /dev/null +++ b/weaver/src/org/aspectj/weaver/patterns/ExactTypePattern.java @@ -0,0 +1,82 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.patterns; + +import java.io.*; + +import org.aspectj.weaver.*; +import org.aspectj.util.*; + +public class ExactTypePattern extends TypePattern { + protected TypeX type; + + public ExactTypePattern(TypeX type, boolean includeSubtypes) { + super(includeSubtypes); + this.type = type; + } + + protected boolean matchesExactly(ResolvedTypeX matchType) { + return this.type.equals(matchType); + } + + public TypeX getType() { return type; } + + public FuzzyBoolean matchesInstanceof(ResolvedTypeX matchType) { + // in our world, Object is assignable from anything + if (type.equals(ResolvedTypeX.OBJECT)) return FuzzyBoolean.YES; + + if (type.isAssignableFrom(matchType, matchType.getWorld())) { + return FuzzyBoolean.YES; + } + + return matchType.isCoerceableFrom(type) ? FuzzyBoolean.MAYBE : FuzzyBoolean.NO; + } + + public boolean equals(Object other) { + if (!(other instanceof ExactTypePattern)) return false; + ExactTypePattern o = (ExactTypePattern)other; + return o.type.equals(this.type); + } + + public int hashCode() { + return type.hashCode(); + } + + public void write(DataOutputStream out) throws IOException { + out.writeByte(TypePattern.EXACT); + type.write(out); + out.writeBoolean(includeSubtypes); + writeLocation(out); + } + + public static TypePattern read(DataInputStream s, ISourceContext context) throws IOException { + TypePattern ret = new ExactTypePattern(TypeX.read(s), s.readBoolean()); + ret.readLocation(context, s); + return ret; + } + + public String toString() { + //Thread.currentThread().dumpStack(); + return "ExactTypePattern(" + type.toString() + (includeSubtypes ? "+" : "") + ")"; + } + public TypePattern resolveBindings( + IScope scope, + Bindings bindings, + boolean allowBinding) + { + throw new BCException("trying to re-resolve"); + + } + +} diff --git a/weaver/src/org/aspectj/weaver/patterns/ExposedState.java b/weaver/src/org/aspectj/weaver/patterns/ExposedState.java new file mode 100644 index 000000000..e5b35a03c --- /dev/null +++ b/weaver/src/org/aspectj/weaver/patterns/ExposedState.java @@ -0,0 +1,57 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.patterns; + +import java.util.Arrays; + +import org.aspectj.weaver.*; +import org.aspectj.weaver.ast.*; + +public class ExposedState { + public Var[] vars; + private Expr aspectInstance; + + public ExposedState(int size) { + super(); + vars = new Var[size]; + } + + public ExposedState(Member signature) { + // XXX there maybe something about target for non-static sigs + this(signature.getParameterTypes().length); + } + + public void set(int i, Var var) { + //XXX add sanity checks + vars[i] = var; + } + public Var get(int i) { + return vars[i]; + } + public int size() { + return vars.length; + } + + public Expr getAspectInstance() { + return aspectInstance; + } + + public void setAspectInstance(Expr aspectInstance) { + this.aspectInstance = aspectInstance; + } + + public String toString() { + return "ExposedState(" + Arrays.asList(vars) + ", " + aspectInstance + ")"; + } +} diff --git a/weaver/src/org/aspectj/weaver/patterns/FormalBinding.java b/weaver/src/org/aspectj/weaver/patterns/FormalBinding.java new file mode 100644 index 000000000..be34d4848 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/patterns/FormalBinding.java @@ -0,0 +1,76 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.patterns; + +import org.aspectj.weaver.*; + +public class FormalBinding implements IHasPosition { + private final TypeX type; + private final String name; + private final int index; + private final int start, end; + private final String fileName; + + public FormalBinding(TypeX type, String name, int index, int start, int end, String fileName) { + this.type = type; + this.name = name; + this.index = index; + this.start = start; + this.end = end; + this.fileName = fileName; + } + + public FormalBinding(TypeX type, int index) { + this(type, "unknown", index, 0, 0, "unknown"); + } + + public FormalBinding(TypeX type, String name, int index) { + this(type, name, index, 0, 0, "unknown"); + } + + // ---- + + public String toString() { + return type.toString() + ":" + index; + } + + public String getFileName() { + return fileName; + } + + public int getEnd() { + return end; + } + + public int getStart() { + return start; + } + + public int getIndex() { + return index; + } + + public String getName() { + return name; + } + + public TypeX getType() { + return type; + } + + // ---- + + public static final FormalBinding[] NONE = new FormalBinding[0]; + +} diff --git a/weaver/src/org/aspectj/weaver/patterns/HandlerPointcut.java b/weaver/src/org/aspectj/weaver/patterns/HandlerPointcut.java new file mode 100644 index 000000000..4f9d16596 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/patterns/HandlerPointcut.java @@ -0,0 +1,94 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.patterns; + +import java.io.*; + +import org.apache.bcel.classfile.JavaClass; +import org.aspectj.weaver.*; +import org.aspectj.weaver.ast.*; +import org.aspectj.bridge.MessageUtil; +import org.aspectj.util.FuzzyBoolean; + +/** + * This is a kind of KindedPointcut. This belongs either in + * a hierarchy with it or in a new place to share code + * with other potential future statement-level pointcuts like + * synchronized and throws + */ +public class HandlerPointcut extends Pointcut { + TypePattern exceptionType; + + public HandlerPointcut(TypePattern exceptionType) { + this.exceptionType = exceptionType; + } + + + public FuzzyBoolean match(Shadow shadow) { + if (shadow.getKind() != Shadow.ExceptionHandler) return FuzzyBoolean.NO; + + // we know we have exactly one parameter since we're checking an exception handler + return exceptionType.matches( + shadow.getSignature().getParameterTypes()[0].resolve(shadow.getIWorld()), + TypePattern.STATIC); + } + + public boolean equals(Object other) { + if (!(other instanceof HandlerPointcut)) return false; + HandlerPointcut o = (HandlerPointcut)other; + return o.exceptionType.equals(this.exceptionType); } + + public int hashCode() { + int result = 17; + result = 37*result + exceptionType.hashCode(); + return result; + } + + public String toString() { + StringBuffer buf = new StringBuffer(); + buf.append("handler("); + buf.append(exceptionType.toString()); + buf.append(")"); + return buf.toString(); + } + + + public void write(DataOutputStream s) throws IOException { + s.writeByte(Pointcut.HANDLER); + exceptionType.write(s); + writeLocation(s); + } + + public static Pointcut read(DataInputStream s, ISourceContext context) throws IOException { + HandlerPointcut ret = new HandlerPointcut(TypePattern.read(s, context)); + ret.readLocation(context, s); + return ret; + } + + // XXX note: there is no namebinding in any kinded pointcut. + // still might want to do something for better error messages + // We want to do something here to make sure we don't sidestep the parameter + // list in capturing type identifiers. + public void resolveBindings(IScope scope, Bindings bindings) { + exceptionType = exceptionType.resolveBindings(scope, bindings, false); + //XXX add error if exact binding and not an exception + } + public Test findResidue(Shadow shadow, ExposedState state) { + return match(shadow).alwaysTrue() ? Literal.TRUE : Literal.FALSE; + } + + public Pointcut concretize1(ResolvedTypeX inAspect, IntMap bindings) { + return new HandlerPointcut(exceptionType); + } +} diff --git a/weaver/src/org/aspectj/weaver/patterns/IScope.java b/weaver/src/org/aspectj/weaver/patterns/IScope.java new file mode 100644 index 000000000..d815326cd --- /dev/null +++ b/weaver/src/org/aspectj/weaver/patterns/IScope.java @@ -0,0 +1,47 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.patterns; + +import org.aspectj.weaver.*; +import org.aspectj.bridge.*; +import org.aspectj.bridge.IMessageHandler; + +public interface IScope { + + /** returns the type corresponding to the name in this scope + * returns ResolvedTypeX.MISSING if no such type exists and reports a problem + */ + TypeX lookupType(String name, IHasPosition location); + + World getWorld(); + + ResolvedTypeX getEnclosingType(); + + // these next three are used to create {@link BindingTypePattern} objects. + IMessageHandler getMessageHandler(); + /** returns the formal associated with the name, or null if no such formal exists */ + FormalBinding lookupFormal(String name); + /** returns the formal with the index. Throws ArrayOutOfBounds exception if out of bounds */ + FormalBinding getFormal(int i); + + int getFormalCount(); + + String[] getImportedPrefixes(); + String[] getImportedNames(); + + void message(IMessage.Kind kind, IHasPosition location, String message); + void message(IMessage.Kind kind, IHasPosition location1, IHasPosition location2, String message); + + //ISourceLocation makeSourceLocation(ILocation location); +} diff --git a/weaver/src/org/aspectj/weaver/patterns/IToken.java b/weaver/src/org/aspectj/weaver/patterns/IToken.java new file mode 100644 index 000000000..e7403287a --- /dev/null +++ b/weaver/src/org/aspectj/weaver/patterns/IToken.java @@ -0,0 +1,53 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.patterns; + +import org.aspectj.weaver.*; + +public interface IToken extends IHasPosition { + public static final IToken EOF = BasicToken.makeOperator("<eof>", 0, 0); + + /** + * Returns the string value of this token. + * + * If isIdentifier is false, then this string must be intern'd + * so that == matching can be used. + * + * If isIdentifier is true, interning is not required. + */ + public String getString(); + + /** + * Whether this should be treated as a token or a generic identifier + */ + public boolean isIdentifier(); + + /** + * Whether this should be treated as a literal value + * + * Kinds == "string", ??? + * + * returns null if this isn't a literal + */ + public String getLiteralKind(); + + + /** + * If this token represents a pre-parsed Pointcut, then return it; + * otherwise returns null. + * + * Needed for the implementation of 'if' + */ + public Pointcut maybeGetParsedPointcut(); +} diff --git a/weaver/src/org/aspectj/weaver/patterns/ITokenSource.java b/weaver/src/org/aspectj/weaver/patterns/ITokenSource.java new file mode 100644 index 000000000..4410e60d8 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/patterns/ITokenSource.java @@ -0,0 +1,26 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.patterns; + +import org.aspectj.weaver.ISourceContext; + + +public interface ITokenSource { + public IToken next(); + public IToken peek(); + public IToken peek(int offset); + public int getIndex(); + public void setIndex(int newIndex); + public ISourceContext getSourceContext(); +} diff --git a/weaver/src/org/aspectj/weaver/patterns/IfPointcut.java b/weaver/src/org/aspectj/weaver/patterns/IfPointcut.java new file mode 100644 index 000000000..8a5d54ea9 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/patterns/IfPointcut.java @@ -0,0 +1,137 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.patterns; + +import java.io.*; +import java.util.*; +import java.util.List; + +import org.aspectj.weaver.*; +import org.aspectj.weaver.ast.*; +import org.aspectj.bridge.MessageUtil; +import org.aspectj.util.*; + + +public class IfPointcut extends Pointcut { + public ResolvedMember testMethod; + public int extraParameterFlags; + + public Pointcut residueSource; + int baseArgsCount; + + //XXX some way to compute args + + + public IfPointcut(ResolvedMember testMethod, int extraParameterFlags) { + this.testMethod = testMethod; + this.extraParameterFlags = extraParameterFlags; + } + + public FuzzyBoolean match(Shadow shadow) { + //??? this is not maximally efficient + return FuzzyBoolean.MAYBE; + } + + public void write(DataOutputStream s) throws IOException { + s.writeByte(Pointcut.IF); + testMethod.write(s); + s.writeByte(extraParameterFlags); + writeLocation(s); + } + public static Pointcut read(DataInputStream s, ISourceContext context) throws IOException { + IfPointcut ret = new IfPointcut(ResolvedMember.readResolvedMember(s, context), s.readByte()); + ret.readLocation(context, s); + return ret; + } + + public void resolveBindings(IScope scope, Bindings bindings) { + //??? all we need is good error messages in here in cflow contexts + } + + public boolean equals(Object other) { + if (!(other instanceof IfPointcut)) return false; + IfPointcut o = (IfPointcut)other; + return o.testMethod.equals(this.testMethod); + } + public int hashCode() { + int result = 17; + result = 37*result + testMethod.hashCode(); + return result; + } + public String toString() { + return "if(" + testMethod + ")"; + } + + public Test findResidue(Shadow shadow, ExposedState state) { + if (residueSource == null) return Literal.TRUE; //??? + + ExposedState myState = new ExposedState(baseArgsCount); + //System.out.println(residueSource); + residueSource.findResidue(shadow, myState); // don't care about Test + + //System.out.println(myState); + + List args = new ArrayList(); + for (int i=0; i < baseArgsCount; i++) { + args.add(myState.get(i)); + } + + // handle thisJoinPoint parameters + if ((extraParameterFlags & Advice.ThisJoinPoint) != 0) { + args.add(shadow.getThisJoinPointVar()); + } + + if ((extraParameterFlags & Advice.ThisJoinPointStaticPart) != 0) { + args.add(shadow.getThisJoinPointStaticPartVar()); + } + + if ((extraParameterFlags & Advice.ThisEnclosingJoinPointStaticPart) != 0) { + args.add(shadow.getThisEnclosingJoinPointStaticPartVar()); + } + + return Test.makeCall(testMethod, (Expr[])args.toArray(new Expr[args.size()])); + } + + + public Pointcut concretize(ResolvedTypeX inAspect, IntMap bindings) { + return this.concretize1(inAspect, bindings); + } + + public Pointcut concretize1(ResolvedTypeX inAspect, IntMap bindings) { + IfPointcut ret = new IfPointcut(testMethod, extraParameterFlags); + if (this.state == CONCRETE) return ret; + this.state = CONCRETE; + if (bindings.directlyInAdvice()) { + Advice advice = bindings.getEnclosingAdvice(); + ret.baseArgsCount = advice.getBaseParameterCount(); + ret.residueSource = advice.getPointcut().concretize(inAspect, ret.baseArgsCount, advice); + } else { + ResolvedPointcutDefinition def = bindings.peekEnclosingDefinitition(); + if (def == CflowPointcut.CFLOW_MARKER) { + inAspect.getWorld().getMessageHandler().handleMessage( + MessageUtil.error("if not supported lexically within cflow (compiler limitation)", + null)); + return Pointcut.makeMatchesNothing(Pointcut.CONCRETE); + } + ret.baseArgsCount = def.getParameterTypes().length; + + IntMap newBindings = IntMap.idMap(ret.baseArgsCount); + newBindings.copyContext(bindings); + ret.residueSource = def.getPointcut().concretize(inAspect, newBindings); + } + + return ret; + } + +} diff --git a/weaver/src/org/aspectj/weaver/patterns/KindedPointcut.java b/weaver/src/org/aspectj/weaver/patterns/KindedPointcut.java new file mode 100644 index 000000000..2720596f5 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/patterns/KindedPointcut.java @@ -0,0 +1,115 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.patterns; + +import java.io.*; + +import org.apache.bcel.classfile.JavaClass; +import org.aspectj.weaver.*; +import org.aspectj.weaver.ast.*; +import org.aspectj.bridge.MessageUtil; +import org.aspectj.util.FuzzyBoolean; + +public class KindedPointcut extends Pointcut { + Shadow.Kind kind; + SignaturePattern signature; + + public KindedPointcut( + Shadow.Kind kind, + SignaturePattern signature) { + this.kind = kind; + this.signature = signature; + } + + public boolean fastMatch(JavaClass jc) { return true; } + + public FuzzyBoolean match(Shadow shadow) { + + + if (shadow.getKind() != kind) return FuzzyBoolean.NO; + + if (!signature.matches(shadow.getSignature(), shadow.getIWorld())) return FuzzyBoolean.NO; + + return FuzzyBoolean.YES; + } + + public boolean equals(Object other) { + if (!(other instanceof KindedPointcut)) return false; + KindedPointcut o = (KindedPointcut)other; + return o.kind == this.kind && o.signature.equals(this.signature); + } + + public int hashCode() { + int result = 17; + result = 37*result + kind.hashCode(); + result = 37*result + signature.hashCode(); + return result; + } + + public String toString() { + StringBuffer buf = new StringBuffer(); + buf.append(kind.getSimpleName()); + buf.append("("); + buf.append(signature.toString()); + buf.append(")"); + return buf.toString(); + } + + + public void postRead(ResolvedTypeX enclosingType) { + signature.postRead(enclosingType); + } + + public void write(DataOutputStream s) throws IOException { + s.writeByte(Pointcut.KINDED); + kind.write(s); + signature.write(s); + writeLocation(s); + } + + public static Pointcut read(DataInputStream s, ISourceContext context) throws IOException { + Shadow.Kind kind = Shadow.Kind.read(s); + SignaturePattern sig = SignaturePattern.read(s, context); + KindedPointcut ret = new KindedPointcut(kind, sig); + ret.readLocation(context, s); + return ret; + } + + // XXX note: there is no namebinding in any kinded pointcut. + // still might want to do something for better error messages + // We want to do something here to make sure we don't sidestep the parameter + // list in capturing type identifiers. + public void resolveBindings(IScope scope, Bindings bindings) { + if (kind == Shadow.Initialization) { +// scope.getMessageHandler().handleMessage( +// MessageUtil.error( +// "initialization unimplemented in 1.1beta1", +// this.getSourceLocation())); + } + signature = signature.resolveBindings(scope, bindings); + } + public Test findResidue(Shadow shadow, ExposedState state) { + return match(shadow).alwaysTrue() ? Literal.TRUE : Literal.FALSE; + } + + public Pointcut concretize1(ResolvedTypeX inAspect, IntMap bindings) { + return new KindedPointcut(kind, signature); + //return this; //??? no pointers out of here so we're okay + } + + public Shadow.Kind getKind() { + return kind; + } + +} diff --git a/weaver/src/org/aspectj/weaver/patterns/ModifiersPattern.java b/weaver/src/org/aspectj/weaver/patterns/ModifiersPattern.java new file mode 100644 index 000000000..dc146a0ef --- /dev/null +++ b/weaver/src/org/aspectj/weaver/patterns/ModifiersPattern.java @@ -0,0 +1,92 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.patterns; + +import java.io.*; +import java.lang.reflect.Modifier; +import java.util.*; + +public class ModifiersPattern extends PatternNode { + private int requiredModifiers; + private int forbiddenModifiers; + + public static final ModifiersPattern ANY = new ModifiersPattern(0, 0); + + public ModifiersPattern(int requiredModifiers, int forbiddenModifiers) { + this.requiredModifiers = requiredModifiers; + this.forbiddenModifiers = forbiddenModifiers; + } + + public String toString() { + if (this == ANY) return ""; + + String ret = Modifier.toString(requiredModifiers); + if (forbiddenModifiers == 0) return ret; + else return ret + " !" + Modifier.toString(forbiddenModifiers); + } + + public boolean equals(Object other) { + if (!(other instanceof ModifiersPattern)) return false; + ModifiersPattern o = (ModifiersPattern)other; + return o.requiredModifiers == this.requiredModifiers && + o.forbiddenModifiers == this.forbiddenModifiers; + } + public int hashCode() { + int result = 17; + result = 37*result + requiredModifiers; + result = 37*result + forbiddenModifiers; + return result; + } + + public boolean matches(int modifiers) { + return ((modifiers & requiredModifiers) == requiredModifiers) && + ((modifiers & forbiddenModifiers) == 0); + } + + + public static ModifiersPattern read(DataInputStream s) throws IOException { + int requiredModifiers = s.readShort(); + int forbiddenModifiers = s.readShort(); + if (requiredModifiers == 0 && forbiddenModifiers == 0) return ANY; + return new ModifiersPattern(requiredModifiers, forbiddenModifiers); + } + + /** + * @see org.aspectj.weaver.patterns.PatternNode#write(DataOutputStream) + */ + public void write(DataOutputStream s) throws IOException { + //s.writeByte(MODIFIERS_PATTERN); + s.writeShort(requiredModifiers); + s.writeShort(forbiddenModifiers); + } + + + private static Map modifierFlags = null; + + public static int getModifierFlag(String name) { + if (modifierFlags == null) { + modifierFlags = new HashMap(); + int flag = 1; + while (flag <= Modifier.STRICT) { + String flagName = Modifier.toString(flag); + modifierFlags.put(flagName, new Integer(flag)); + flag = flag << 1; + } + } + Integer flag = (Integer)modifierFlags.get(name); + if (flag == null) return -1; + return flag.intValue(); + } + +} diff --git a/weaver/src/org/aspectj/weaver/patterns/NameBindingPointcut.java b/weaver/src/org/aspectj/weaver/patterns/NameBindingPointcut.java new file mode 100644 index 000000000..0a043ad51 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/patterns/NameBindingPointcut.java @@ -0,0 +1,47 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.patterns; + +import java.io.*; + +import org.aspectj.util.FuzzyBoolean; +import org.aspectj.weaver.*; +import org.aspectj.weaver.ast.*; + +/** + * Common super type for Pointcuts that can bind formal parameters. + * + * @author Erik Hilsdale + * @author Jim Hugunin + */ +public abstract class NameBindingPointcut extends Pointcut { + + public NameBindingPointcut() { + super(); + } + + protected Test exposeStateForVar(Var var,TypePattern type, ExposedState state, World world) { + if (type instanceof BindingTypePattern) { + BindingTypePattern b = (BindingTypePattern)type; + state.set(b.getFormalIndex(), var); + } + TypeX myType = type.getExactType(); //should have failed earlier + + return Test.makeInstanceof(var, myType.resolve(world)); + } + + + + +} diff --git a/weaver/src/org/aspectj/weaver/patterns/NamePattern.java b/weaver/src/org/aspectj/weaver/patterns/NamePattern.java new file mode 100644 index 000000000..043376fc1 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/patterns/NamePattern.java @@ -0,0 +1,158 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.patterns; + +import java.io.*; + +public class NamePattern extends PatternNode { + char[] pattern; + int starCount = 0; + + public static final NamePattern ELLIPSIS = new NamePattern(""); + public static final NamePattern ANY = new NamePattern("*"); + + public NamePattern(String name) { + this(name.toCharArray()); + } + + public NamePattern(char[] pattern) { + this.pattern = pattern; + + for (int i=0, len=pattern.length; i<len; i++) { + if (pattern[i] == '*') starCount++; + } + } + + public boolean matches(char[] a2) { + char[] a1 = pattern; + int len1 = a1.length; + int len2 = a2.length; + if (starCount == 0) { + if (len1 != len2) return false; + for (int i=0; i<len1; i++) { + if (a1[i] != a2[i]) return false; + } + return true; + } else if (starCount == 1) { + // just '*' matches anything + if (len1 == 1) return true; + if (len1 > len2+1) return false; + + int i2=0; + for (int i1=0; i1<len1; i1++) { + char c1 = a1[i1]; + if (c1 == '*') { + i2 = len2 - (len1-(i1+1)); + } else if (c1 != a2[i2++]) { + return false; + } + } + return true; + } else { +// String pattern = new String(a1); +// String target = new String(a2); +// System.err.print("match(\"" + pattern + "\", \"" + target + "\") -> "); + boolean b = outOfStar(a1, a2, 0, 0, len1 - starCount, len2, starCount); +// System.err.println(b); + return b; + } + } + private static boolean outOfStar(final char[] pattern, final char[] target, + int pi, int ti, + int pLeft, int tLeft, + final int starsLeft) { + if (pLeft > tLeft) return false; + while (true) { + // invariant: if (tLeft > 0) then (ti < target.length && pi < pattern.length) + if (tLeft == 0) return true; + if (pLeft == 0) { + return (starsLeft > 0); + } + if (pattern[pi] == '*') { + return inStar(pattern, target, pi+1, ti, pLeft, tLeft, starsLeft-1); + } + if (target[ti] != pattern[pi]) { + return false; + } + pi++; ti++; pLeft--; tLeft--; + } + } + private static boolean inStar(final char[] pattern, final char[] target, + int pi, int ti, + final int pLeft, int tLeft, + int starsLeft) { + // invariant: pLeft > 0, so we know we'll run out of stars and find a real char in pattern + char patternChar = pattern[pi]; + while (patternChar == '*') { + starsLeft--; + patternChar = pattern[++pi]; + } + while (true) { + // invariant: if (tLeft > 0) then (ti < target.length) + if (pLeft > tLeft) return false; + if (target[ti] == patternChar) { + if (outOfStar(pattern, target, pi+1, ti+1, pLeft-1, tLeft-1, starsLeft)) return true; + } + ti++; tLeft--; + } + } + + public boolean matches(String other) { + return matches(other.toCharArray()); + } + + public String toString() { + return new String(pattern); + } + + public boolean equals(Object other) { + if (other instanceof NamePattern) { + NamePattern otherPat = (NamePattern)other; + return otherPat.starCount == this.starCount && + new String(otherPat.pattern).equals(new String(this.pattern)); + } + return false; + } + public int hashCode() { + return new String(pattern).hashCode(); + } + + + public void write(DataOutputStream out) throws IOException { + out.writeUTF(new String(pattern)); + } + + public static NamePattern read(DataInputStream in) throws IOException { + String s = in.readUTF(); + if (s.length() == 0) return ELLIPSIS; + return new NamePattern(s); + } + /** + * Method maybeGetSimpleName. + * @return String + */ + public String maybeGetSimpleName() { + if (starCount == 0 && pattern.length > 0) return new String(pattern); + return null; + } + + /** + * Method isAny. + * @return boolean + */ + public boolean isAny() { + return starCount == 1 && pattern.length == 1; + } + +} diff --git a/weaver/src/org/aspectj/weaver/patterns/NotPointcut.java b/weaver/src/org/aspectj/weaver/patterns/NotPointcut.java new file mode 100644 index 000000000..64b919f69 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/patterns/NotPointcut.java @@ -0,0 +1,96 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.patterns; + +import java.io.*; + +import org.aspectj.weaver.*; +import org.aspectj.weaver.ast.*; +import org.aspectj.util.*; + +public class NotPointcut extends Pointcut { + private Pointcut body; + public NotPointcut(Pointcut left) { + super(); + this.body = left; + } + + /** + * Constructor NotPointcut. + * @param pointcut + * @param startPos + */ + public NotPointcut(Pointcut pointcut, int startPos) { + this(pointcut); + setLocation(pointcut.getSourceContext(), startPos, pointcut.getEnd()); + } + + + /** + * @see org.aspectj.weaver.patterns.Pointcut#match(BcelShadow) + */ + public FuzzyBoolean match(Shadow shadow) { + return body.match(shadow).not(); + } + + public String toString() { + return "!" + body.toString(); + + } + + public boolean equals(Object other) { + if (!(other instanceof NotPointcut)) return false; + NotPointcut o = (NotPointcut)other; + return o.body.equals(body); + } + public int hashCode() { + return 37*23 + body.hashCode(); + } + + + public void resolveBindings(IScope scope, Bindings bindings) { + //Bindings old = bindings.copy(); + + //Bindings newBindings = new Bindings(bindings.size()); + + + body.resolveBindings(scope, null); + + //newBindings.checkEmpty(scope, "negation does not allow binding"); + //bindings.checkEquals(old, scope); + + } + + + public void write(DataOutputStream s) throws IOException { + s.writeByte(Pointcut.NOT); + body.write(s); + writeLocation(s); + } + + public static Pointcut read(DataInputStream s, ISourceContext context) throws IOException { + NotPointcut ret = new NotPointcut(Pointcut.read(s, context)); + ret.readLocation(context, s); + return ret; + } + + public Test findResidue(Shadow shadow, ExposedState state) { + return Test.makeNot(body.findResidue(shadow, state)); + } + + public Pointcut concretize1(ResolvedTypeX inAspect, IntMap bindings) { + return new NotPointcut(body.concretize1(inAspect, bindings)); + } + +} diff --git a/weaver/src/org/aspectj/weaver/patterns/NotTypePattern.java b/weaver/src/org/aspectj/weaver/patterns/NotTypePattern.java new file mode 100644 index 000000000..f62d693c7 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/patterns/NotTypePattern.java @@ -0,0 +1,75 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.patterns; + +import java.io.*; + +import org.aspectj.util.FuzzyBoolean; +import org.aspectj.weaver.*; +import org.aspectj.weaver.ResolvedTypeX; + +/** + * !TypePattern + * + * <p>any binding to formals is explicitly forbidden for any composite, ! is + * just the most obviously wrong case. + * + * @author Erik Hilsdale + * @author Jim Hugunin + */ +public class NotTypePattern extends TypePattern { + private TypePattern pattern; + + public NotTypePattern(TypePattern pattern) { + super(false); //??? we override all methods that care about includeSubtypes + this.pattern = pattern; + setLocation(pattern.getSourceContext(), pattern.getStart(), pattern.getEnd()); + } + + public FuzzyBoolean matchesInstanceof(ResolvedTypeX type) { + return pattern.matchesInstanceof(type).not(); + } + + protected boolean matchesExactly(ResolvedTypeX type) { + return !pattern.matchesExactly(type); + } + + public boolean matchesStatically(ResolvedTypeX type) { + return !pattern.matchesStatically(type); + } + + public void write(DataOutputStream s) throws IOException { + s.writeByte(TypePattern.NOT); + pattern.write(s); + writeLocation(s); + } + + public static TypePattern read(DataInputStream s, ISourceContext context) throws IOException { + TypePattern ret = new NotTypePattern(TypePattern.read(s, context)); + ret.readLocation(context, s); + return ret; + } + + public TypePattern resolveBindings( + IScope scope, + Bindings bindings, + boolean allowBinding) { + pattern = pattern.resolveBindings(scope, bindings, false); + return this; + } + + public String toString() { + return "!" + pattern; + } +} diff --git a/weaver/src/org/aspectj/weaver/patterns/OrPointcut.java b/weaver/src/org/aspectj/weaver/patterns/OrPointcut.java new file mode 100644 index 000000000..fc389acaf --- /dev/null +++ b/weaver/src/org/aspectj/weaver/patterns/OrPointcut.java @@ -0,0 +1,89 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.patterns; + +import java.io.*; + +import org.aspectj.weaver.*; +import org.aspectj.weaver.ast.*; +import org.aspectj.util.*; + +public class OrPointcut extends Pointcut { + private Pointcut left, right; + /** + * Constructor for AndPointcut. + */ + public OrPointcut(Pointcut left, Pointcut right) { + super(); + this.left = left; + this.right = right; + setLocation(left.getSourceContext(), left.getStart(), right.getEnd()); + } + + /** + * @see org.aspectj.weaver.patterns.Pointcut#match(BcelShadow) + */ + public FuzzyBoolean match(Shadow shadow) { + return left.match(shadow).or(right.match(shadow)); + } + + public String toString() { + return "(" + left.toString() + " || " + right.toString() + ")"; + } + + public boolean equals(Object other) { + if (!(other instanceof OrPointcut)) return false; + OrPointcut o = (OrPointcut)other; + return o.left.equals(left) && o.right.equals(right); + } + public int hashCode() { + int result = 31; + result = 37*result + left.hashCode(); + result = 37*result + right.hashCode(); + return result; + } + /** + * @see org.aspectj.weaver.patterns.Pointcut#resolveBindings(IScope, Bindings) + */ + public void resolveBindings(IScope scope, Bindings bindings) { + Bindings old = bindings == null ? null : bindings.copy(); + + left.resolveBindings(scope, bindings); + right.resolveBindings(scope, old); + if (bindings != null) bindings.checkEquals(old, scope); + + } + + public void write(DataOutputStream s) throws IOException { + s.writeByte(Pointcut.OR); + left.write(s); + right.write(s); + writeLocation(s); + } + + public static Pointcut read(DataInputStream s, ISourceContext context) throws IOException { + OrPointcut ret = new OrPointcut(Pointcut.read(s, context), Pointcut.read(s, context)); + ret.readLocation(context, s); + return ret; + + } + public Test findResidue(Shadow shadow, ExposedState state) { + return Test.makeOr(left.findResidue(shadow, state), right.findResidue(shadow, state)); + } + + public Pointcut concretize1(ResolvedTypeX inAspect, IntMap bindings) { + return new OrPointcut(left.concretize1(inAspect, bindings), + right.concretize1(inAspect, bindings)); + } +} diff --git a/weaver/src/org/aspectj/weaver/patterns/OrTypePattern.java b/weaver/src/org/aspectj/weaver/patterns/OrTypePattern.java new file mode 100644 index 000000000..6cb043bc5 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/patterns/OrTypePattern.java @@ -0,0 +1,79 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.patterns; + +import java.io.*; + +import org.aspectj.util.FuzzyBoolean; +import org.aspectj.weaver.*; +import org.aspectj.weaver.ResolvedTypeX; + +/** + * left || right + * + * <p>any binding to formals is explicitly forbidden for any composite by the language + * + * @author Erik Hilsdale + * @author Jim Hugunin + */ +public class OrTypePattern extends TypePattern { + private TypePattern left, right; + + public OrTypePattern(TypePattern left, TypePattern right) { + super(false); //??? we override all methods that care about includeSubtypes + this.left = left; + this.right = right; + setLocation(left.getSourceContext(), left.getStart(), right.getEnd()); + } + + public FuzzyBoolean matchesInstanceof(ResolvedTypeX type) { + return left.matchesInstanceof(type).or(right.matchesInstanceof(type)); + } + + protected boolean matchesExactly(ResolvedTypeX type) { + //??? if these had side-effects, this sort-circuit could be a mistake + return left.matchesExactly(type) || right.matchesExactly(type); + } + + public boolean matchesStatically(ResolvedTypeX type) { + return left.matchesStatically(type) || right.matchesStatically(type); + } + + public void write(DataOutputStream s) throws IOException { + s.writeByte(TypePattern.OR); + left.write(s); + right.write(s); + writeLocation(s); + } + + public static TypePattern read(DataInputStream s, ISourceContext context) throws IOException { + TypePattern ret = new OrTypePattern(TypePattern.read(s, context), TypePattern.read(s, context)); + ret.readLocation(context, s); + return ret; + } + + public TypePattern resolveBindings( + IScope scope, + Bindings bindings, + boolean allowBinding) { + left = left.resolveBindings(scope, bindings, false); + right = right.resolveBindings(scope, bindings, false); + return this; + } + + public String toString() { + return "(" + left.toString() + " || " + right.toString() + ")"; + } + +} diff --git a/weaver/src/org/aspectj/weaver/patterns/ParserException.java b/weaver/src/org/aspectj/weaver/patterns/ParserException.java new file mode 100644 index 000000000..af6f0cb49 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/patterns/ParserException.java @@ -0,0 +1,31 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.patterns; + +import org.aspectj.weaver.*; + + +public class ParserException extends RuntimeException { + private IHasPosition token; + + public ParserException(String message, IHasPosition token) { + super(message); + this.token = token; + } + + public IHasPosition getLocation() { + return token; + } + +} diff --git a/weaver/src/org/aspectj/weaver/patterns/PatternNode.java b/weaver/src/org/aspectj/weaver/patterns/PatternNode.java new file mode 100644 index 000000000..4c09ad4b7 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/patterns/PatternNode.java @@ -0,0 +1,82 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.patterns; + +import java.io.*; + +import org.aspectj.bridge.*; +import org.aspectj.bridge.ISourceLocation; +import org.aspectj.weaver.*; +import org.aspectj.weaver.ISourceContext; + +public abstract class PatternNode implements IHasSourceLocation { + protected int start, end; + protected ISourceContext sourceContext; + + public PatternNode() { + super(); + start = end = -1; + } + + public int getStart() { + return start; + } + + public int getEnd() { + return end; + } + + public ISourceContext getSourceContext() { + return sourceContext; + } + + public String getFileName() { + return "unknown"; + } + + public void setLocation(ISourceContext sourceContext, int start, int end) { + this.sourceContext = sourceContext; + this.start = start; + this.end = end; + } + + public void copyLocationFrom(PatternNode other) { + this.start = other.start; + this.end = other.end; + this.sourceContext = other.sourceContext; + } + + public ISourceLocation getSourceLocation() { + //System.out.println("get context: " + this + " is " + sourceContext); + if (sourceContext == null) { + //System.err.println("no context: " + this); + return null; + } + return sourceContext.makeSourceLocation(this); + } + + public abstract void write(DataOutputStream s) throws IOException; + + + public void writeLocation(DataOutputStream s) throws IOException { + s.writeInt(start); + s.writeInt(end); + } + + public void readLocation(ISourceContext context, DataInputStream s) throws IOException { + start = s.readInt(); + end = s.readInt(); + this.sourceContext = context; + } +} diff --git a/weaver/src/org/aspectj/weaver/patterns/PatternParser.java b/weaver/src/org/aspectj/weaver/patterns/PatternParser.java new file mode 100644 index 000000000..eaeb71f20 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/patterns/PatternParser.java @@ -0,0 +1,781 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.patterns; + +import java.util.*; + +import org.aspectj.weaver.*; + +//XXX doesn't handle errors for extra tokens very well (sometimes ignores) +public class PatternParser { + private ITokenSource tokenSource; + + private ISourceContext sourceContext; + + /** + * Constructor for PatternParser. + */ + public PatternParser(ITokenSource tokenSource) { + super(); + this.tokenSource = tokenSource; + this.sourceContext = tokenSource.getSourceContext(); + } + + public PerClause maybeParsePerClause() { + IToken tok = tokenSource.peek(); + if (tok == IToken.EOF) return null; + if (tok.isIdentifier()) { + String name = tok.getString(); + if (name.equals("issingleton")) { + return parsePerSingleton(); + } else if (name.equals("perthis")) { + return parsePerObject(true); + } else if (name.equals("pertarget")) { + return parsePerObject(false); + } else if (name.equals("percflow")) { + return parsePerCflow(false); + } else if (name.equals("percflowbelow")) { + return parsePerCflow(true); + } else { + return null; + } + } + return null; + } + + private PerClause parsePerCflow(boolean isBelow) { + parseIdentifier(); + eat("("); + Pointcut entry = parsePointcut(); + eat(")"); + return new PerCflow(entry, isBelow); + } + + + private PerClause parsePerObject(boolean isThis) { + parseIdentifier(); + eat("("); + Pointcut entry = parsePointcut(); + eat(")"); + return new PerObject(entry, isThis); + } + + + private PerClause parsePerSingleton() { + parseIdentifier(); + eat("("); + eat(")"); + return new PerSingleton(); + } + + + public Declare parseDeclare() { + int startPos = tokenSource.peek().getStart(); + + eatIdentifier("declare"); + String kind = parseIdentifier(); + eat(":"); + Declare ret; + //XXX beta add soft, dominates + if (kind.equals("error")) { + ret = parseErrorOrWarning(true); + } else if (kind.equals("warning")) { + ret = parseErrorOrWarning(false); + } else if (kind.equals("dominates")) { + ret = parseDominates(); + } else if (kind.equals("parents")) { + ret = parseParents(); + } else if (kind.equals("soft")) { + ret = parseSoft(); + } else { + throw new ParserException("expected one of error, warning, parents, soft, dominates", + tokenSource.peek(-1)); + } + int endPos = tokenSource.peek(-1).getEnd(); + ret.setLocation(sourceContext, startPos, endPos); + return ret; + } + + public DeclareDominates parseDominates() { + List l = new ArrayList(); + do { + l.add(parseTypePattern()); + } while (maybeEat(",")); + + return new DeclareDominates(l); + } + + private Declare parseParents() { + TypePattern p = parseTypePattern(); + IToken t = tokenSource.next(); + if (!(t.getString().equals("extends") || t.getString().equals("implements"))) { + throw new ParserException("extends or implements", t); + } + + List l = new ArrayList(); + do { + l.add(parseTypePattern()); + } while (maybeEat(",")); + + //XXX somewhere in the chain we need to enforce that we have only ExactTypePatterns + + return new DeclareParents(p, l); + } + + private Declare parseSoft() { + TypePattern p = parseTypePattern(); + eat(":"); + Pointcut pointcut = parsePointcut(); + return new DeclareSoft(p, pointcut); + } + + + + private Declare parseErrorOrWarning(boolean isError) { + Pointcut pointcut = parsePointcut(); + eat(":"); + String message = parseStringLiteral(); + return new DeclareErrorOrWarning(isError, pointcut, message); + } + + public Pointcut parsePointcut() { + Pointcut p = parseAtomicPointcut(); + if (maybeEat("&&")) { + p = new AndPointcut(p, parseNotOrPointcut()); + } + + if (maybeEat("||")) { + p = new OrPointcut(p, parsePointcut()); + } + return p; + } + + private Pointcut parseNotOrPointcut() { + Pointcut p = parseAtomicPointcut(); + if (maybeEat("&&")) { + p = new AndPointcut(p, parsePointcut()); + } + return p; + } + + private Pointcut parseAtomicPointcut() { + if (maybeEat("!")) { + int startPos = tokenSource.peek(-1).getStart(); + Pointcut p = new NotPointcut(parseAtomicPointcut(), startPos); + return p; + } + if (maybeEat("(")) { + Pointcut p = parsePointcut(); + eat(")"); + return p; + } + int startPos = tokenSource.peek().getStart(); + Pointcut p = parseSinglePointcut(); + int endPos = tokenSource.peek(-1).getEnd(); + p.setLocation(sourceContext, startPos, endPos); + return p; + } + + + + public Pointcut parseSinglePointcut() { + int start = tokenSource.getIndex(); + IToken t = tokenSource.peek(); + Pointcut p = t.maybeGetParsedPointcut(); + if (p != null) { + tokenSource.next(); + return p; + } + + String kind = parseIdentifier(); + tokenSource.setIndex(start); + if (kind.equals("execution") || kind.equals("call") || + kind.equals("get") || kind.equals("set")) { + return parseKindedPointcut(); + } else if (kind.equals("args")) { + return parseArgsPointcut(); + } else if (kind.equals("this") || kind.equals("target")) { + return parseThisOrTargetPointcut(); + } else if (kind.equals("within")) { + return parseWithinPointcut(); + } else if (kind.equals("withincode")) { + return parseWithinCodePointcut(); + } else if (kind.equals("cflow")) { + return parseCflowPointcut(false); + } else if (kind.equals("cflowbelow")) { + return parseCflowPointcut(true); + } else if (kind.equals("adviceexecution")) { + parseIdentifier(); eat("("); + eat(")"); + return new KindedPointcut(Shadow.AdviceExecution, + new SignaturePattern(Member.ADVICE, ModifiersPattern.ANY, + TypePattern.ANY, TypePattern.ANY, NamePattern.ANY, + TypePatternList.ANY, + ThrowsPattern.ANY)); + } else if (kind.equals("handler")) { + parseIdentifier(); eat("("); + TypePattern typePat = parseTypePattern(); + eat(")"); + return new HandlerPointcut(typePat); + } else if (kind.equals("initialization")) { + parseIdentifier(); eat("("); + SignaturePattern sig = parseConstructorSignaturePattern(); + eat(")"); + return new KindedPointcut(Shadow.Initialization, sig); + } else if (kind.equals("staticinitialization")) { + parseIdentifier(); eat("("); + TypePattern typePat = parseTypePattern(); + eat(")"); + return new KindedPointcut(Shadow.StaticInitialization, + new SignaturePattern(Member.STATIC_INITIALIZATION, ModifiersPattern.ANY, + TypePattern.ANY, typePat, NamePattern.ANY, TypePatternList.EMPTY, + ThrowsPattern.ANY)); + } else if (kind.equals("preinitialization")) { + parseIdentifier(); eat("("); + SignaturePattern sig = parseConstructorSignaturePattern(); + eat(")"); + return new KindedPointcut(Shadow.PreInitialization, sig); + } else { + return parseReferencePointcut(); + } + } + + private SignaturePattern parseConstructorSignaturePattern() { + SignaturePattern ret = parseMethodOrConstructorSignaturePattern(); + if (ret.getKind() == Member.CONSTRUCTOR) return ret; + + throw new ParserException("constructor pattern required, found method pattern", + ret); + } + + + private Pointcut parseWithinCodePointcut() { + parseIdentifier(); + eat("("); + SignaturePattern sig = parseMethodOrConstructorSignaturePattern(); + eat(")"); + return new WithincodePointcut(sig); + } + + private Pointcut parseCflowPointcut(boolean isBelow) { + parseIdentifier(); + eat("("); + Pointcut entry = parsePointcut(); + eat(")"); + return new CflowPointcut(entry, isBelow, null); + } + + /** + * Method parseWithinPointcut. + * @return Pointcut + */ + private Pointcut parseWithinPointcut() { + parseIdentifier(); + eat("("); + TypePattern type = parseTypePattern(); + eat(")"); + return new WithinPointcut(type); + } + + + /** + * Method parseThisOrTargetPointcut. + * @return Pointcut + */ + private Pointcut parseThisOrTargetPointcut() { + String kind = parseIdentifier(); + eat("("); + TypePattern type = parseTypePattern(); + eat(")"); + return new ThisOrTargetPointcut(kind.equals("this"), type); + } + + + /** + * Method parseArgsPointcut. + * @return Pointcut + */ + private Pointcut parseArgsPointcut() { + parseIdentifier(); + TypePatternList arguments = parseArgumentsPattern(); + return new ArgsPointcut(arguments); + } + + + private Pointcut parseReferencePointcut() { + TypePattern onType = parseTypePattern(); + NamePattern name = tryToExtractName(onType); + if (name == null) { + throw new ParserException("name pattern", tokenSource.peek()); + } + if (onType.toString().equals("")) { + onType = null; + } + + TypePatternList arguments = parseArgumentsPattern(); + return new ReferencePointcut(onType, name.maybeGetSimpleName(), arguments); + } + + + public List parseDottedIdentifier() { + List ret = new ArrayList(); + ret.add(parseIdentifier()); + while (maybeEat(".")) { + ret.add(parseIdentifier()); + } + return ret; + } + + + + private KindedPointcut parseKindedPointcut() { + String kind = parseIdentifier(); + eat("("); + SignaturePattern sig; + + Shadow.Kind shadowKind = null; + if (kind.equals("execution")) { + sig = parseMethodOrConstructorSignaturePattern(); + if (sig.getKind() == Member.METHOD) { + shadowKind = Shadow.MethodExecution; + } else if (sig.getKind() == Member.CONSTRUCTOR) { + shadowKind = Shadow.ConstructorExecution; + } + } else if (kind.equals("call")) { + sig = parseMethodOrConstructorSignaturePattern(); + if (sig.getKind() == Member.METHOD) { + shadowKind = Shadow.MethodCall; + } else if (sig.getKind() == Member.CONSTRUCTOR) { + shadowKind = Shadow.ConstructorCall; + } + } else if (kind.equals("get")) { + sig = parseFieldSignaturePattern(); + shadowKind = Shadow.FieldGet; + } else if (kind.equals("set")) { + sig = parseFieldSignaturePattern(); + shadowKind = Shadow.FieldSet; + } else { + throw new ParserException("bad kind: " + kind, tokenSource.peek()); + } + eat(")"); + return new KindedPointcut(shadowKind, sig); + } + + public TypePattern parseTypePattern() { + TypePattern p = parseAtomicTypePattern(); + if (maybeEat("&&")) { + p = new AndTypePattern(p, parseNotOrTypePattern()); + } + + if (maybeEat("||")) { + p = new OrTypePattern(p, parseTypePattern()); + } + return p; + } + + private TypePattern parseNotOrTypePattern() { + TypePattern p = parseAtomicTypePattern(); + if (maybeEat("&&")) { + p = new AndTypePattern(p, parseTypePattern()); + } + return p; + } + + private TypePattern parseAtomicTypePattern() { + if (maybeEat("!")) { + int startPos = tokenSource.peek(-1).getStart(); + //??? we lose source location for true start of !type + TypePattern p = new NotTypePattern(parseAtomicTypePattern()); + return p; + } + if (maybeEat("(")) { + TypePattern p = parseTypePattern(); + eat(")"); + return p; + } + int startPos = tokenSource.peek().getStart(); + TypePattern p = parseSingleTypePattern(); + int endPos = tokenSource.peek(-1).getEnd(); + p.setLocation(sourceContext, startPos, endPos); + return p; + } + + public TypePattern parseSingleTypePattern() { + List names = parseDottedNamePattern(); +// new ArrayList(); +// NamePattern p1 = parseNamePattern(); +// names.add(p1); +// while (maybeEat(".")) { +// if (maybeEat(".")) { +// names.add(NamePattern.ELLIPSIS); +// } +// NamePattern p2 = parseNamePattern(); +// names.add(p2); +// } + int dim = 0; + while (maybeEat("[")) { + eat("]"); + dim++; + } + + + boolean includeSubtypes = maybeEat("+"); + int endPos = tokenSource.peek(-1).getEnd(); + + //??? what about the source location of any's???? + if (names.size() == 1 && ((NamePattern)names.get(0)).isAny() && dim == 0) return TypePattern.ANY; + + return new WildTypePattern(names, includeSubtypes, dim, endPos); + } + + public List parseDottedNamePattern() { + List names = new ArrayList(); + StringBuffer buf = new StringBuffer(); + IToken previous = null; + while (true) { + IToken tok; + int startPos = tokenSource.peek().getStart(); + String afterDot = null; + while (true) { + tok = tokenSource.peek(); + if (previous != null) { + if (!isAdjacent(previous, tok)) break; + } + if (tok.getString() == "*" || tok.isIdentifier()) { + buf.append(tok.getString()); + } else if (tok.getLiteralKind() != null) { + //System.err.println("literal kind: " + tok.getString()); + String s = tok.getString(); + int dot = s.indexOf('.'); + if (dot != -1) { + buf.append(s.substring(0, dot)); + afterDot = s.substring(dot+1); + previous = tokenSource.next(); + break; + } + buf.append(s); // ??? so-so + } else { + break; + } + previous = tokenSource.next(); + //XXX need to handle floats and other fun stuff + } + int endPos = tokenSource.peek(-1).getEnd(); + if (buf.length() == 0 && names.isEmpty()) { + throw new ParserException("expected name pattern", tok); + } + + if (buf.length() == 0) { + names.add(NamePattern.ELLIPSIS); + } else { + checkLegalName(buf.toString(), previous); + NamePattern ret = new NamePattern(buf.toString()); + ret.setLocation(sourceContext, startPos, endPos); + names.add(ret); + } + + if (afterDot == null) { + buf.setLength(0); + if (!maybeEat(".")) break; + else previous = tokenSource.peek(-1); + } else { + buf.setLength(0); + buf.append(afterDot); + afterDot = null; + } + } + //System.err.println("parsed: " + names); + return names; + } + + + + public NamePattern parseNamePattern() { + StringBuffer buf = new StringBuffer(); + IToken previous = null; + IToken tok; + int startPos = tokenSource.peek().getStart(); + while (true) { + tok = tokenSource.peek(); + if (previous != null) { + if (!isAdjacent(previous, tok)) break; + } + if (tok.getString() == "*" || tok.isIdentifier()) { + buf.append(tok.getString()); + } else if (tok.getLiteralKind() != null) { + //System.err.println("literal kind: " + tok.getString()); + String s = tok.getString(); + if (s.indexOf('.') != -1) break; + buf.append(s); // ??? so-so + } else { + break; + } + previous = tokenSource.next(); + //XXX need to handle floats and other fun stuff + } + int endPos = tokenSource.peek(-1).getEnd(); + if (buf.length() == 0) { + throw new ParserException("expected name pattern", tok); + } + + checkLegalName(buf.toString(), previous); + NamePattern ret = new NamePattern(buf.toString()); + ret.setLocation(sourceContext, startPos, endPos); + return ret; + } + + private void checkLegalName(String s, IToken tok) { + char ch = s.charAt(0); + if (!(ch == '*' || Character.isJavaIdentifierStart(ch))) { + throw new ParserException("illegal identifier start (" + ch + ")", tok); + } + + for (int i=1, len=s.length(); i < len; i++) { + ch = s.charAt(i); + if (!(ch == '*' || Character.isJavaIdentifierPart(ch))) { + throw new ParserException("illegal identifier character (" + ch + ")", tok); + } + } + + } + + + private boolean isAdjacent(IToken first, IToken second) { + return first.getEnd() == second.getStart()-1; + } + + + public ModifiersPattern parseModifiersPattern() { + int requiredFlags = 0; + int forbiddenFlags = 0; + int start; + while (true) { + start = tokenSource.getIndex(); + boolean isForbidden = false; + isForbidden = maybeEat("!"); + IToken t = tokenSource.next(); + int flag = ModifiersPattern.getModifierFlag(t.getString()); + if (flag == -1) break; + if (isForbidden) forbiddenFlags |= flag; + else requiredFlags |= flag; + } + + tokenSource.setIndex(start); + if (requiredFlags == 0 && forbiddenFlags == 0) { + return ModifiersPattern.ANY; + } else { + return new ModifiersPattern(requiredFlags, forbiddenFlags); + } + } + + public TypePatternList parseArgumentsPattern() { + List patterns = new ArrayList(); + eat("("); + if (maybeEat(")")) { + return new TypePatternList(); + } + + do { + if (maybeEat(".")) { + eat("."); + patterns.add(TypePattern.ELLIPSIS); + } else { + patterns.add(parseTypePattern()); + } + } while (maybeEat(",")); + eat(")"); + return new TypePatternList(patterns); + } + + public ThrowsPattern parseOptionalThrowsPattern() { + IToken t = tokenSource.peek(); + if (t.isIdentifier() && t.getString().equals("throws")) { + tokenSource.next(); + List required = new ArrayList(); + List forbidden = new ArrayList(); + do { + boolean isForbidden = maybeEat("!"); + //???might want an error for a second ! without a paren + TypePattern p = parseTypePattern(); + if (isForbidden) forbidden.add(p); + else required.add(p); + } while (maybeEat(",")); + return new ThrowsPattern(new TypePatternList(required), new TypePatternList(forbidden)); + } + return ThrowsPattern.ANY; + } + + + public SignaturePattern parseMethodOrConstructorSignaturePattern() { + ModifiersPattern modifiers = parseModifiersPattern(); + TypePattern returnType = parseTypePattern(); + + TypePattern declaringType; + NamePattern name = null; + Member.Kind kind; + // here we can check for 'new' + if (maybeEatNew(returnType)) { + kind = Member.CONSTRUCTOR; + if (returnType.toString().length() == 0) { + declaringType = TypePattern.ANY; + } else { + declaringType = returnType; + } + returnType = TypePattern.ANY; + name = NamePattern.ANY; + } else { + kind = Member.METHOD; + declaringType = parseTypePattern(); + if (maybeEat(".")) { + name = parseNamePattern(); + } else { + name = tryToExtractName(declaringType); + if (name == null) { + throw new ParserException("name pattern", tokenSource.peek()); + } + String simpleName = name.maybeGetSimpleName(); + //XXX should add check for any Java keywords + if (simpleName != null && simpleName.equals("new")) { + throw new ParserException("constructor patterns have no return type", + tokenSource.peek()); + } + if (declaringType.toString().equals("")) { + declaringType = declaringType.ANY; + } + } + } + + TypePatternList parameterTypes = parseArgumentsPattern(); + + ThrowsPattern throwsPattern = parseOptionalThrowsPattern(); + + return new SignaturePattern(kind, modifiers, returnType, declaringType, name, parameterTypes, throwsPattern); + } + + private boolean maybeEatNew(TypePattern returnType) { + if (returnType instanceof WildTypePattern) { + WildTypePattern p = (WildTypePattern)returnType; + if (p.maybeExtractName("new")) return true; + } + int start = tokenSource.getIndex(); + if (maybeEat(".")) { + String id = maybeEatIdentifier(); + if (id != null && id.equals("new")) return true; + tokenSource.setIndex(start); + } + + return false; + } + + + public SignaturePattern parseFieldSignaturePattern() { + ModifiersPattern modifiers = parseModifiersPattern(); + TypePattern returnType = parseTypePattern(); + TypePattern declaringType = parseTypePattern(); + NamePattern name; + //System.err.println("parsed field: " + declaringType.toString()); + if (maybeEat(".")) { + name = parseNamePattern(); + } else { + name = tryToExtractName(declaringType); + if (declaringType.toString().equals("")) { + declaringType = declaringType.ANY; + } + } + return new SignaturePattern(Member.FIELD, modifiers, returnType, + declaringType, name, TypePatternList.ANY, ThrowsPattern.ANY); + } + + + private NamePattern tryToExtractName(TypePattern nextType) { + if (nextType == TypePattern.ANY) { + return NamePattern.ANY; + } else if (nextType instanceof WildTypePattern) { + WildTypePattern p = (WildTypePattern)nextType; + return p.extractName(); + } else { + return null; + } + } + + public String parseStringLiteral() { + IToken token = tokenSource.next(); + String literalKind = token.getLiteralKind(); + if (literalKind == "string") { + return token.getString(); + } + + throw new ParserException("string", token); + } + + public String parseIdentifier() { + IToken token = tokenSource.next(); + if (token.isIdentifier()) return token.getString(); + throw new ParserException("identifier", token); + } + + public void eatIdentifier(String expectedValue) { + IToken next = tokenSource.next(); + if (!next.getString().equals(expectedValue)) { + throw new ParserException(expectedValue, next); + } + } + + public boolean maybeEatIdentifier(String expectedValue) { + IToken next = tokenSource.peek(); + if (next.getString().equals(expectedValue)) { + tokenSource.next(); + return true; + } else { + return false; + } + } + + public void eat(String expectedValue) { + IToken next = tokenSource.next(); + if (next.getString() != expectedValue) { + throw new ParserException(expectedValue, next); + } + } + + public boolean maybeEat(String token) { + IToken next = tokenSource.peek(); + if (next.getString() == token) { + tokenSource.next(); + return true; + } else { + return false; + } + } + + public String maybeEatIdentifier() { + IToken next = tokenSource.peek(); + if (next.isIdentifier()) { + tokenSource.next(); + return next.getString(); + } else { + return null; + } + } + + public boolean peek(String token) { + IToken next = tokenSource.peek(); + return next.getString() == token; + } + + + public PatternParser(String data) { + this(BasicTokenSource.makeTokenSource(data)); + } +} diff --git a/weaver/src/org/aspectj/weaver/patterns/PerCflow.java b/weaver/src/org/aspectj/weaver/patterns/PerCflow.java new file mode 100644 index 000000000..35614041a --- /dev/null +++ b/weaver/src/org/aspectj/weaver/patterns/PerCflow.java @@ -0,0 +1,98 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.patterns; + +import java.io.*; +import java.lang.reflect.Modifier; +import java.util.*; +import java.util.List; + +import org.aspectj.weaver.*; +import org.aspectj.weaver.ast.*; +import org.aspectj.util.FuzzyBoolean; + +public class PerCflow extends PerClause { + private boolean isBelow; + private Pointcut entry; + + public PerCflow(Pointcut entry, boolean isBelow) { + this.entry = entry; + this.isBelow = isBelow; + } + + // ----- + + public FuzzyBoolean match(Shadow shadow) { + return FuzzyBoolean.YES; + } + + public void resolveBindings(IScope scope, Bindings bindings) { + // assert bindings == null; + entry.resolve(scope); + } + + public Test findResidue(Shadow shadow, ExposedState state) { + Expr myInstance = + Expr.makeCallExpr(AjcMemberMaker.perCflowAspectOfMethod(inAspect), + Expr.NONE, inAspect); + state.setAspectInstance(myInstance); + return Test.makeCall(AjcMemberMaker.perCflowHasAspectMethod(inAspect), Expr.NONE); + } + + + public PerClause concretize(ResolvedTypeX inAspect) { + PerCflow ret = new PerCflow(entry, isBelow); + ret.inAspect = inAspect; + if (inAspect.isAbstract()) return ret; + + Member cflowStackField = new ResolvedMember( + Member.FIELD, inAspect, Modifier.STATIC|Modifier.PUBLIC|Modifier.FINAL, + TypeX.forName(NameMangler.CFLOW_STACK_TYPE), NameMangler.PERCFLOW_FIELD_NAME, TypeX.NONE); + + World world = inAspect.getWorld(); + + CrosscuttingMembers xcut = inAspect.crosscuttingMembers; + + Collection previousCflowEntries = xcut.getCflowEntries(); + Pointcut concreteEntry = entry.concretize(inAspect, IntMap.EMPTY); + List innerCflowEntries = new ArrayList(xcut.getCflowEntries()); + innerCflowEntries.removeAll(previousCflowEntries); + + xcut.addConcreteShadowMunger( + Advice.makePerCflowEntry(world, concreteEntry, isBelow, cflowStackField, + inAspect, innerCflowEntries)); + return ret; + } + + public void write(DataOutputStream s) throws IOException { + PERCFLOW.write(s); + entry.write(s); + s.writeBoolean(isBelow); + writeLocation(s); + } + + public static PerClause readPerClause(DataInputStream s, ISourceContext context) throws IOException { + PerCflow ret = new PerCflow(Pointcut.read(s, context), s.readBoolean()); + ret.readLocation(context, s); + return ret; + } + + public PerClause.Kind getKind() { + return PERCFLOW; + } + + public String toString() { + return "percflow(" + inAspect + " on " + entry + ")"; + } +} diff --git a/weaver/src/org/aspectj/weaver/patterns/PerClause.java b/weaver/src/org/aspectj/weaver/patterns/PerClause.java new file mode 100644 index 000000000..7ff029818 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/patterns/PerClause.java @@ -0,0 +1,67 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.patterns; + +import java.io.*; +import java.lang.reflect.Modifier; +import java.util.List; + +import org.aspectj.weaver.*; +import org.aspectj.weaver.ast.Test; +import org.aspectj.util.*; +import org.aspectj.util.FuzzyBoolean; + +public abstract class PerClause extends Pointcut { + protected ResolvedTypeX inAspect; + + public static PerClause readPerClause(DataInputStream s, ISourceContext context) throws IOException { + Kind kind = Kind.read(s); + if (kind == SINGLETON) return PerSingleton.readPerClause(s, context); + else if (kind == PERCFLOW) return PerCflow.readPerClause(s, context); + else if (kind == PEROBJECT) return PerObject.readPerClause(s, context); + else if (kind == FROMSUPER) return PerFromSuper.readPerClause(s, context); + + throw new BCException("unknown kind: " + kind); + } + + public final Pointcut concretize1(ResolvedTypeX inAspect, IntMap bindings) { + throw new RuntimeException("unimplemented: wrong concretize"); + } + + public abstract PerClause concretize(ResolvedTypeX inAspect); + + + public abstract PerClause.Kind getKind(); + + + public static class Kind extends TypeSafeEnum { + public Kind(String name, int key) { super(name, key); } + + public static Kind read(DataInputStream s) throws IOException { + int key = s.readByte(); + switch(key) { + case 1: return SINGLETON; + case 2: return PERCFLOW; + case 3: return PEROBJECT; + case 4: return FROMSUPER; + } + throw new BCException("weird kind " + key); + } + } + + public static final Kind SINGLETON = new Kind("issingleton", 1); + public static final Kind PERCFLOW = new Kind("percflow", 2); + public static final Kind PEROBJECT = new Kind("perobject", 3); + public static final Kind FROMSUPER = new Kind("fromsuper", 4); +} diff --git a/weaver/src/org/aspectj/weaver/patterns/PerFromSuper.java b/weaver/src/org/aspectj/weaver/patterns/PerFromSuper.java new file mode 100644 index 000000000..e0fa53166 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/patterns/PerFromSuper.java @@ -0,0 +1,94 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.patterns; + +import java.io.*; +import java.lang.reflect.Modifier; + +import org.aspectj.weaver.*; +import org.aspectj.weaver.ast.*; +import org.aspectj.bridge.MessageUtil; +import org.aspectj.util.FuzzyBoolean; + +public class PerFromSuper extends PerClause { + private PerClause.Kind kind; + + public PerFromSuper(PerClause.Kind kind) { + this.kind = kind; + } + + public FuzzyBoolean match(Shadow shadow) { + throw new RuntimeException("unimplemented"); + } + + public void resolveBindings(IScope scope, Bindings bindings) { + // this method intentionally left blank + } + + public Test findResidue(Shadow shadow, ExposedState state) { + throw new RuntimeException("unimplemented"); + } + + + public PerClause concretize(ResolvedTypeX inAspect) { + PerClause p = lookupConcretePerClause(inAspect.getSuperclass()); + if (p == null) { + inAspect.getWorld().getMessageHandler().handleMessage( + MessageUtil.error("expected per clause on super aspect not found on " + + inAspect.getSuperclass(), getSourceLocation()) + ); + } + if (p.getKind() != kind) { + inAspect.getWorld().getMessageHandler().handleMessage( + MessageUtil.error("wrong kind of per clause on super, expected " + + kind + " but found " + p.getKind(), + getSourceLocation()) + ); + } + return p.concretize(inAspect); + } + + + + private PerClause lookupConcretePerClause(ResolvedTypeX lookupType) { + PerClause ret = lookupType.getPerClause(); + if (ret == null) return null; + if (ret instanceof PerFromSuper) { + return lookupConcretePerClause(lookupType.getSuperclass()); + } + return ret; + } + + + public void write(DataOutputStream s) throws IOException { + FROMSUPER.write(s); + kind.write(s); + writeLocation(s); + } + + public static PerClause readPerClause(DataInputStream s, ISourceContext context) throws IOException { + PerFromSuper ret = new PerFromSuper(Kind.read(s)); + ret.readLocation(context, s); + return ret; + } + + public String toString() { + return "perFromSuper(" + kind + ", " + inAspect + ")"; + } + + public PerClause.Kind getKind() { + return kind; + } + +} diff --git a/weaver/src/org/aspectj/weaver/patterns/PerObject.java b/weaver/src/org/aspectj/weaver/patterns/PerObject.java new file mode 100644 index 000000000..de73e7066 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/patterns/PerObject.java @@ -0,0 +1,104 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.patterns; + +import java.io.*; +import java.lang.reflect.Modifier; +import java.util.*; +import java.util.List; + +import org.aspectj.weaver.*; +import org.aspectj.weaver.ast.*; +import org.aspectj.util.FuzzyBoolean; + +public class PerObject extends PerClause { + private boolean isThis; + private Pointcut entry; + + public PerObject(Pointcut entry, boolean isThis) { + this.entry = entry; + this.isThis = isThis; + } + + // ----- + + public FuzzyBoolean match(Shadow shadow) { + //System.err.println("matches " + this + " ? " + shadow + ", " + shadow.hasTarget()); + //??? could probably optimize this better by testing could match + if (isThis) return FuzzyBoolean.fromBoolean(shadow.hasThis()); + else return FuzzyBoolean.fromBoolean(shadow.hasTarget()); + } + + public void resolveBindings(IScope scope, Bindings bindings) { + // assert bindings == null; + entry.resolve(scope); + } + + private Var getVar(Shadow shadow) { + return isThis ? shadow.getThisVar() : shadow.getTargetVar(); + } + + public Test findResidue(Shadow shadow, ExposedState state) { + Expr myInstance = + Expr.makeCallExpr(AjcMemberMaker.perObjectAspectOfMethod(inAspect), + new Expr[] {getVar(shadow)}, inAspect); + state.setAspectInstance(myInstance); + return Test.makeCall(AjcMemberMaker.perObjectHasAspectMethod(inAspect), + new Expr[] { getVar(shadow) }); + } + + + + public PerClause concretize(ResolvedTypeX inAspect) { + PerObject ret = new PerObject(entry, isThis); + + ret.inAspect = inAspect; + if (inAspect.isAbstract()) return ret; + + + World world = inAspect.getWorld(); + + Pointcut concreteEntry = entry.concretize(inAspect, IntMap.EMPTY); + //concreteEntry = new AndPointcut(this, concreteEntry); + //concreteEntry.state = Pointcut.CONCRETE; + inAspect.crosscuttingMembers.addConcreteShadowMunger( + Advice.makePerObjectEntry(world, concreteEntry, isThis, inAspect)); + ResolvedTypeMunger munger = + new PerObjectInterfaceTypeMunger(inAspect, concreteEntry); + inAspect.crosscuttingMembers.addTypeMunger(world.concreteTypeMunger(munger, inAspect)); + return ret; + } + + public void write(DataOutputStream s) throws IOException { + PEROBJECT.write(s); + entry.write(s); + s.writeBoolean(isThis); + writeLocation(s); + } + + public static PerClause readPerClause(DataInputStream s, ISourceContext context) throws IOException { + PerClause ret = new PerObject(Pointcut.read(s, context), s.readBoolean()); + ret.readLocation(context, s); + return ret; + } + + public PerClause.Kind getKind() { + return PEROBJECT; + } + + public String toString() { + return "per" + (isThis ? "this" : "target") + + "(" + entry + ")"; + } +} diff --git a/weaver/src/org/aspectj/weaver/patterns/PerSingleton.java b/weaver/src/org/aspectj/weaver/patterns/PerSingleton.java new file mode 100644 index 000000000..e95065ece --- /dev/null +++ b/weaver/src/org/aspectj/weaver/patterns/PerSingleton.java @@ -0,0 +1,68 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.patterns; + +import java.io.*; +import java.lang.reflect.Modifier; + +import org.aspectj.weaver.*; +import org.aspectj.weaver.ast.*; +import org.aspectj.util.FuzzyBoolean; + +public class PerSingleton extends PerClause { + public PerSingleton() { + } + + public FuzzyBoolean match(Shadow shadow) { + return FuzzyBoolean.YES; + } + + public void resolveBindings(IScope scope, Bindings bindings) { + // this method intentionally left blank + } + + public Test findResidue(Shadow shadow, ExposedState state) { + state.setAspectInstance( + Expr.makeFieldGet(AjcMemberMaker.perSingletonField(inAspect), inAspect)); + // XXX we need to think about whether it's null... + return Literal.TRUE; + } + + public PerClause concretize(ResolvedTypeX inAspect) { + PerSingleton ret = new PerSingleton(); + ret.inAspect = inAspect; + return ret; + } + + public void write(DataOutputStream s) throws IOException { + SINGLETON.write(s); + writeLocation(s); + } + + public static PerClause readPerClause(DataInputStream s, ISourceContext context) throws IOException { + PerSingleton ret = new PerSingleton(); + ret.readLocation(context, s); + return ret; + } + + + public PerClause.Kind getKind() { + return SINGLETON; + } + + public String toString() { + return "persingleton(" + inAspect + ")"; + } + +} diff --git a/weaver/src/org/aspectj/weaver/patterns/Pointcut.java b/weaver/src/org/aspectj/weaver/patterns/Pointcut.java new file mode 100644 index 000000000..99f7f6bc8 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/patterns/Pointcut.java @@ -0,0 +1,228 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.patterns; + +import java.io.*; +import java.util.Map; + +import org.apache.bcel.classfile.*; +import org.aspectj.weaver.*; +import org.aspectj.weaver.ast.*; +import org.aspectj.util.*; + +/** + * The lifecycle of Pointcuts is modeled by Pointcut.State. It has three things: + * + * <p>Creation -- SYMBOLIC -- then resolve(IScope) -- RESOLVED -- concretize(...) -- CONCRETE + * + * @author Erik Hilsdale + * @author Jim Hugunin + */ +public abstract class Pointcut extends PatternNode { + public static final class State extends TypeSafeEnum { + public State(String name, int key) { + super(name, key); + } + } + + public static final State SYMBOLIC = new State("symbolic", 0); + public static final State RESOLVED = new State("resolved", 1); + public static final State CONCRETE = new State("concrete", 2); + + + public State state; + + /** + * Constructor for Pattern. + */ + public Pointcut() { + super(); + this.state = SYMBOLIC; + } + + + /** + * Could I match any shadows in this JavaClass + */ + public boolean fastMatch(JavaClass jc) { return true; } + + /** + * Do I really match this shadow? + * XXX implementors need to handle state + */ + public abstract FuzzyBoolean match(Shadow shadow); + + + public static final byte KINDED = 1; + public static final byte WITHIN = 2; + public static final byte THIS_OR_TARGET = 3; + public static final byte ARGS = 4; + public static final byte AND = 5; + public static final byte OR = 6; + public static final byte NOT = 7; + public static final byte REFERENCE = 8; + public static final byte IF = 9; + public static final byte CFLOW = 10; + public static final byte WITHINCODE = 12; + public static final byte HANDLER = 13; + + public static final byte NONE = 20; + + + // internal, only called from resolve + protected abstract void resolveBindings(IScope scope, Bindings bindings); + + /** + * Returns this pointcut mutated + */ + public Pointcut resolve(IScope scope) { + assertState(SYMBOLIC); + Bindings bindingTable = new Bindings(scope.getFormalCount()); + this.resolveBindings(scope, bindingTable); + bindingTable.checkAllBound(scope); + this.state = RESOLVED; + return this; + } + + /** + * Returns a new pointcut + */ + public Pointcut concretize(ResolvedTypeX inAspect, int arity) { + return concretize(inAspect, IntMap.idMap(arity)); + } + + + //XXX this is the signature we're moving to + public Pointcut concretize(ResolvedTypeX inAspect, int arity, Advice advice) { + //if (state == CONCRETE) return this; //??? + IntMap map = IntMap.idMap(arity); + map.setEnclosingAdvice(advice); + return concretize(inAspect, map); + } + + + + public Pointcut concretize(ResolvedTypeX inAspect, IntMap bindings) { + assertState(RESOLVED); + Pointcut ret = this.concretize1(inAspect, bindings); + ret.state = CONCRETE; + return ret; + } + + + /** + * Resolves and removes ReferencePointcuts, replacing with basic ones + * + * @param inAspect the aspect to resolve relative to + * @param bindings a Map from formal index in the current lexical context + * -> formal index in the concrete advice that will run + * + * ??? does this return a new pointcut + * XXX fix implementors to handle state + */ + protected abstract Pointcut concretize1(ResolvedTypeX inAspect, IntMap bindings); + + + //XXX implementors need to handle state + /** + * This can be called from NotPointcut even for Pointcuts that + * don't match the shadow + */ + public abstract Test findResidue(Shadow shadow, ExposedState state); + + //XXX we're not sure whether or not this is needed + //XXX currently it's unused we're keeping it around as a stub + public void postRead(ResolvedTypeX enclosingType) {} + + public static Pointcut read(DataInputStream s, ISourceContext context) throws IOException { + byte kind = s.readByte(); + Pointcut ret; + + switch(kind) { + case KINDED: ret = KindedPointcut.read(s, context); break; + case WITHIN: ret = WithinPointcut.read(s, context); break; + case THIS_OR_TARGET: ret = ThisOrTargetPointcut.read(s, context); break; + case ARGS: ret = ArgsPointcut.read(s, context); break; + case AND: ret = AndPointcut.read(s, context); break; + case OR: ret = OrPointcut.read(s, context); break; + case NOT: ret = NotPointcut.read(s, context); break; + case REFERENCE: ret = ReferencePointcut.read(s, context); break; + case IF: ret = IfPointcut.read(s, context); break; + case CFLOW: ret = CflowPointcut.read(s, context); break; + case WITHINCODE: ret = WithincodePointcut.read(s, context); break; + case HANDLER: ret = HandlerPointcut.read(s, context); break; + + case NONE: ret = makeMatchesNothing(RESOLVED); break; + default: + throw new BCException("unknown kind: " + kind); + } + ret.state = RESOLVED; + return ret; + } + + + //public void prepare(Shadow shadow) {} + + // ---- test method + + public static Pointcut fromString(String str) { + PatternParser parser = new PatternParser(str); + return parser.parsePointcut(); + } + + private static class MatchesNothingPointcut extends Pointcut { + public Test findResidue(Shadow shadow, ExposedState state) { + return Literal.FALSE; // can only get here if an earlier error occurred + } + + public FuzzyBoolean match(Shadow shadow) { + return FuzzyBoolean.NO; + } + + public void resolveBindings(IScope scope, Bindings bindings) { + } + + public void postRead(ResolvedTypeX enclosingType) { + } + + public Pointcut concretize1( + ResolvedTypeX inAspect, + IntMap bindings) { + return this; + } + + + public void write(DataOutputStream s) throws IOException { + s.writeByte(NONE); + } + + public String toString() { return ""; } + } + + //public static Pointcut MatchesNothing = new MatchesNothingPointcut(); + //??? there could possibly be some good optimizations to be done at this point + public static Pointcut makeMatchesNothing(State state) { + Pointcut ret = new MatchesNothingPointcut(); + ret.state = state; + return ret; + } + + public void assertState(State state) { + if (this.state != state) { + throw new BCException("expected state: " + state + " got: " + this.state); + } + } + + +} diff --git a/weaver/src/org/aspectj/weaver/patterns/ReferencePointcut.java b/weaver/src/org/aspectj/weaver/patterns/ReferencePointcut.java new file mode 100644 index 000000000..518172ed5 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/patterns/ReferencePointcut.java @@ -0,0 +1,266 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.patterns; + +import java.io.*; +import java.lang.reflect.Modifier; +import java.util.*; +import java.util.Map; + +import org.apache.bcel.classfile.*; +import org.aspectj.weaver.*; +import org.aspectj.weaver.ast.*; +import org.aspectj.bridge.*; +import org.aspectj.bridge.MessageUtil; +import org.aspectj.util.*; + +/** + */ + +//XXX needs check that arguments contains no WildTypePatterns +public class ReferencePointcut extends Pointcut { + public TypeX onType; + public TypePattern onTypeSymbolic; + public String name; + public TypePatternList arguments; + + //public ResolvedPointcut binding; + + public ReferencePointcut(TypePattern onTypeSymbolic, String name, TypePatternList arguments) { + this.onTypeSymbolic = onTypeSymbolic; + this.name = name; + this.arguments = arguments; + } + + public ReferencePointcut(TypeX onType, String name, TypePatternList arguments) { + this.onType = onType; + this.name = name; + this.arguments = arguments; + } + + /** + * Could I match any shadows in this JavaClass + */ + public boolean fastMatch(JavaClass jc) { return true; } + + /** + * Do I really match this shadow? + */ + public FuzzyBoolean match(Shadow shadow) { + return FuzzyBoolean.NO; + } + + public String toString() { + StringBuffer buf = new StringBuffer(); + if (onType != null) { + buf.append(onType); + buf.append("."); +// for (int i=0, len=fromType.length; i < len; i++) { +// buf.append(fromType[i]); +// buf.append("."); +// } + } + buf.append(name); + buf.append(arguments.toString()); + return buf.toString(); + } + + + public void write(DataOutputStream s) throws IOException { + //XXX ignores onType + s.writeByte(Pointcut.REFERENCE); + if (onType != null) { + s.writeBoolean(true); + onType.write(s); + } else { + s.writeBoolean(false); + } + + s.writeUTF(name); + arguments.write(s); + writeLocation(s); + } + + public static Pointcut read(DataInputStream s, ISourceContext context) throws IOException { + TypeX onType = null; + if (s.readBoolean()) { + onType = TypeX.read(s); + } + ReferencePointcut ret = new ReferencePointcut(onType, s.readUTF(), + TypePatternList.read(s, context)); + ret.readLocation(context, s); + return ret; + } + + public void resolveBindings(IScope scope, Bindings bindings) { + if (onTypeSymbolic != null) { + onType = onTypeSymbolic.resolveExactType(scope, bindings); + } + + ResolvedTypeX searchType; + if (onType != null) { + searchType = scope.getWorld().resolve(onType); + } else { + searchType = scope.getEnclosingType(); + } + + + arguments.resolveBindings(scope, bindings, true); + //XXX ensure that arguments has no ..'s in it + + // check that I refer to a real pointcut declaration and that I match + + ResolvedPointcutDefinition pointcutDef = searchType.findPointcut(name); + // if we're not a static reference, then do a lookup of outers + if (onType == null) { + while (true) { + TypeX declaringType = searchType.getDeclaringType(); + if (declaringType == null) break; + searchType = declaringType.resolve(scope.getWorld()); + pointcutDef = searchType.findPointcut(name); + if (pointcutDef != null) { + // make this a static reference + onType = searchType; + break; + } + } + } + + if (pointcutDef == null) { + scope.message(IMessage.ERROR, this, "can't find referenced pointcut"); + return; + } + + if (Modifier.isAbstract(pointcutDef.getModifiers())) { + if (onType != null) { + scope.message(IMessage.ERROR, this, + "can't make static reference to abstract pointcut"); + return; + } else if (!searchType.isAbstract()) { + scope.message(IMessage.ERROR, this, + "can't use abstract pointcut in concrete context"); + return; + } + } + + + ResolvedTypeX[] parameterTypes = + scope.getWorld().resolve(pointcutDef.getParameterTypes()); + + if (parameterTypes.length != arguments.size()) { + scope.message(IMessage.ERROR, this, "incompatible number of arguments to pointcut, expected " + + parameterTypes.length + " found " + arguments.size()); + return; + } + + + + for (int i=0,len=arguments.size(); i < len; i++) { + TypePattern p = arguments.get(i); + //we are allowed to bind to pointcuts which use subtypes as this is type safe + if (!p.matchesSubtypes(parameterTypes[i])) { + scope.message(IMessage.ERROR, p, "incompatible type, expected " + + parameterTypes[i] + " found " + p); + return; + } + } + } + + public void postRead(ResolvedTypeX enclosingType) { + arguments.postRead(enclosingType); + } + + public Test findResidue(Shadow shadow, ExposedState state) { + throw new RuntimeException("shouldn't happen"); + } + + + //??? This is not thread safe, but this class is not designed for multi-threading + private boolean concretizing = false; + public Pointcut concretize1(ResolvedTypeX searchStart, IntMap bindings) { + if (concretizing) { + Thread.currentThread().dumpStack(); + searchStart.getWorld().getMessageHandler().handleMessage( + MessageUtil.error("circular pointcut declaration involving: " + this, + getSourceLocation())); + return Pointcut.makeMatchesNothing(Pointcut.CONCRETE); + } + + try { + concretizing = true; + + ResolvedPointcutDefinition pointcutDec; + if (onType != null) { + searchStart = onType.resolve(searchStart.getWorld()); + if (searchStart == ResolvedTypeX.MISSING) { + return Pointcut.makeMatchesNothing(Pointcut.CONCRETE); + } + } + pointcutDec = searchStart.findPointcut(name); + if (pointcutDec == null) { + searchStart.getWorld().getMessageHandler().handleMessage( + MessageUtil.error("can't find pointcut \'" + name + "\' on " + searchStart.getName(), + getSourceLocation()) + ); + return Pointcut.makeMatchesNothing(Pointcut.CONCRETE); + } + //??? error if we don't find + + //System.err.println("start: " + searchStart); + ResolvedTypeX[] parameterTypes = searchStart.getWorld().resolve(pointcutDec.getParameterTypes()); + + arguments = arguments.resolveReferences(bindings); + + IntMap newBindings = new IntMap(); + for (int i=0,len=arguments.size(); i < len; i++) { + TypePattern p = arguments.get(i); + //we are allowed to bind to pointcuts which use subtypes as this is type safe + if (!p.matchesSubtypes(parameterTypes[i])) { + throw new BCException("illegal change to pointcut declaration: " + this); + } + + if (p instanceof BindingTypePattern) { + newBindings.put(i, ((BindingTypePattern)p).getFormalIndex()); + } + } + + newBindings.copyContext(bindings); + newBindings.pushEnclosingDefinition(pointcutDec); + try { + return pointcutDec.getPointcut().concretize1(searchStart, newBindings); + } finally { + newBindings.popEnclosingDefinitition(); + } + + } finally { + concretizing = false; + } + } + + public boolean equals(Object other) { + if (!(other instanceof ReferencePointcut)) return false; + ReferencePointcut o = (ReferencePointcut)other; + return o.name.equals(name) && o.arguments.equals(arguments) + && ((o.onType == null) ? (onType == null) : o.onType.equals(onType)); + } + public int hashCode() { + int result = 17; + result = 37*result + ((onType == null) ? 0 : onType.hashCode()); + result = 37*result + arguments.hashCode(); + result = 37*result + name.hashCode(); + return result; + } + + +} diff --git a/weaver/src/org/aspectj/weaver/patterns/SignaturePattern.java b/weaver/src/org/aspectj/weaver/patterns/SignaturePattern.java new file mode 100644 index 000000000..fe47e13f4 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/patterns/SignaturePattern.java @@ -0,0 +1,224 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.patterns; + +import java.io.*; +import java.util.*; +import java.util.Collection; + +import org.aspectj.weaver.*; + + +public class SignaturePattern extends PatternNode { + private Member.Kind kind; + private ModifiersPattern modifiers; + private TypePattern returnType; + private TypePattern declaringType; + private NamePattern name; + private TypePatternList parameterTypes; + private ThrowsPattern throwsPattern; + + public SignaturePattern(Member.Kind kind, ModifiersPattern modifiers, + TypePattern returnType, TypePattern declaringType, + NamePattern name, TypePatternList parameterTypes, + ThrowsPattern throwsPattern) { + this.kind = kind; + this.modifiers = modifiers; + this.returnType = returnType; + this.name = name; + this.declaringType = declaringType; + this.parameterTypes = parameterTypes; + this.throwsPattern = throwsPattern; + } + + + public SignaturePattern resolveBindings(IScope scope, Bindings bindings) { + if (returnType != null) { + returnType = returnType.resolveBindings(scope, bindings, false); + } + if (declaringType != null) { + declaringType = declaringType.resolveBindings(scope, bindings, false); + } + if (parameterTypes != null) { + parameterTypes = parameterTypes.resolveBindings(scope, bindings, false); + } + if (throwsPattern != null) { + throwsPattern = throwsPattern.resolveBindings(scope, bindings); + } + + return this; + } + + + public void postRead(ResolvedTypeX enclosingType) { + if (returnType != null) { + returnType.postRead(enclosingType); + } + if (declaringType != null) { + declaringType.postRead(enclosingType); + } + if (parameterTypes != null) { + parameterTypes.postRead(enclosingType); + } + } + + public boolean matches(Member member, World world) { + //XXX performance gains would come from matching on name before resolving + // to fail fast + ResolvedMember sig = member.resolve(world); + if (sig == null) { + //XXX + if (member.getName().startsWith(NameMangler.PREFIX)) { + return false; + } + world.getLint().unresolvableMember.signal(member.toString(), getSourceLocation()); + return false; + } + if (!modifiers.matches(sig.getModifiers())) return false; + + if (kind == Member.STATIC_INITIALIZATION) { + //System.err.println("match static init: " + sig.getDeclaringType() + " with " + this); + return declaringType.matchesStatically(sig.getDeclaringType().resolve(world)); + } else if (kind == Member.FIELD) { + + if (!returnType.matchesStatically(sig.getReturnType().resolve(world))) return false; + if (!name.matches(sig.getName())) return false; + boolean ret = declaringTypeMatch(member.getDeclaringType(), member, world); + //System.out.println(" ret: " + ret); + return ret; + } else if (kind == Member.METHOD) { + if (!returnType.matchesStatically(sig.getReturnType().resolve(world))) return false; + if (!name.matches(sig.getName())) return false; + if (!parameterTypes.matches(world.resolve(sig.getParameterTypes()), TypePattern.STATIC).alwaysTrue()) { + return false; + } + if (!throwsPattern.matches(sig.getExceptions(), world)) return false; + return declaringTypeMatch(member.getDeclaringType(), member, world); + } else if (kind == Member.CONSTRUCTOR) { + if (!parameterTypes.matches(world.resolve(sig.getParameterTypes()), TypePattern.STATIC).alwaysTrue()) { + return false; + } + if (!throwsPattern.matches(sig.getExceptions(), world)) return false; + return declaringType.matchesStatically(member.getDeclaringType().resolve(world)); + //return declaringTypeMatch(member.getDeclaringType(), member, world); + } + + return false; + } + + private boolean declaringTypeMatch(TypeX onTypeUnresolved, Member member, World world) { + ResolvedTypeX onType = onTypeUnresolved.resolve(world); + + // fastmatch + if (declaringType.matchesStatically(onType)) return true; + + Collection declaringTypes = member.getDeclaringTypes(world); + + for (Iterator i = declaringTypes.iterator(); i.hasNext(); ) { + ResolvedTypeX type = (ResolvedTypeX)i.next(); + if (declaringType.matchesStatically(type)) return true; + } + return false; + } + + + public NamePattern getName() { return name; } + public TypePattern getDeclaringType() { return declaringType; } + + public Member.Kind getKind() { + return kind; + } + + public String toString() { + StringBuffer buf = new StringBuffer(); + if (modifiers != ModifiersPattern.ANY) { + buf.append(modifiers.toString()); + buf.append(' '); + } + + if (kind == Member.STATIC_INITIALIZATION) { + buf.append(declaringType.toString()); + buf.append(".<clinit>()"); + } else if (kind == Member.HANDLER) { + buf.append("handler("); + buf.append(parameterTypes.get(0)); + buf.append(")"); + } else { + if (!(kind == Member.CONSTRUCTOR)) { + buf.append(returnType.toString()); + buf.append(' '); + } + if (declaringType != TypePattern.ANY) { + buf.append(declaringType.toString()); + buf.append('.'); + } + if (kind == Member.CONSTRUCTOR) { + buf.append("new"); + } else { + buf.append(name.toString()); + } + if (kind == Member.METHOD || kind == Member.CONSTRUCTOR) { + buf.append(parameterTypes.toString()); + } + } + return buf.toString(); + } + + public boolean equals(Object other) { + if (!(other instanceof SignaturePattern)) return false; + SignaturePattern o = (SignaturePattern)other; + return o.kind.equals(this.kind) + && o.modifiers.equals(this.modifiers) + && o.returnType.equals(this.returnType) + && o.declaringType.equals(this.declaringType) + && o.name.equals(this.name) + && o.parameterTypes.equals(this.parameterTypes); + } + public int hashCode() { + int result = 17; + result = 37*result + kind.hashCode(); + result = 37*result + modifiers.hashCode(); + result = 37*result + returnType.hashCode(); + result = 37*result + declaringType.hashCode(); + result = 37*result + name.hashCode(); + result = 37*result + parameterTypes.hashCode(); + return result; + } + + public void write(DataOutputStream s) throws IOException { + kind.write(s); + modifiers.write(s); + returnType.write(s); + declaringType.write(s); + name.write(s); + parameterTypes.write(s); + throwsPattern.write(s); + writeLocation(s); + } + + public static SignaturePattern read(DataInputStream s, ISourceContext context) throws IOException { + Member.Kind kind = Member.Kind.read(s); + ModifiersPattern modifiers = ModifiersPattern.read(s); + TypePattern returnType = TypePattern.read(s, context); + TypePattern declaringType = TypePattern.read(s, context); + NamePattern name = NamePattern.read(s); + TypePatternList parameterTypes = TypePatternList.read(s, context); + ThrowsPattern throwsPattern = ThrowsPattern.read(s, context); + SignaturePattern ret = new SignaturePattern(kind, modifiers, returnType, declaringType, + name, parameterTypes, throwsPattern); + ret.readLocation(context, s); + return ret; + } + +} diff --git a/weaver/src/org/aspectj/weaver/patterns/SimpleScope.java b/weaver/src/org/aspectj/weaver/patterns/SimpleScope.java new file mode 100644 index 000000000..2821e67ae --- /dev/null +++ b/weaver/src/org/aspectj/weaver/patterns/SimpleScope.java @@ -0,0 +1,137 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.patterns; + +import org.aspectj.weaver.*; +import org.aspectj.bridge.*; +import org.aspectj.bridge.IMessageHandler; + +public class SimpleScope implements IScope { + + private World world; + private ResolvedTypeX enclosingType; + private FormalBinding[] bindings; + + private String[] importedPrefixes = javaLangPrefixArray; + private String[] importedNames = ZERO_STRINGS; + + public SimpleScope(World world, FormalBinding[] bindings) { + super(); + this.world = world; + this.bindings = bindings; + } + + // ---- impl + + //XXX doesn't report any problems + public TypeX lookupType(String name, IHasPosition location) { + for (int i=0; i<importedNames.length; i++) { + String importedName = importedNames[i]; + if (importedName.endsWith(name)) { + return world.resolve(importedName); + } + } + + for (int i=0; i<importedPrefixes.length; i++) { + String importedPrefix = importedPrefixes[i]; + TypeX tryType = world.resolve(TypeX.forName(importedPrefix + name), true); + if (tryType != ResolvedTypeX.MISSING) { + return tryType; + } + } + + return world.resolve(TypeX.forName(name), true); + } + + + public IMessageHandler getMessageHandler() { + return world.getMessageHandler(); + } + public FormalBinding lookupFormal(String name) { + for (int i = 0, len = bindings.length; i < len; i++) { + if (bindings[i].getName().equals(name)) return bindings[i]; + } + return null; + } + public FormalBinding getFormal(int i) { + return bindings[i]; + } + + public int getFormalCount() { + return bindings.length; + } + + public String[] getImportedNames() { + return importedNames; + } + public String[] getImportedPrefixes() { + return importedPrefixes; + } + + public void setImportedNames(String[] importedNames) { + this.importedNames = importedNames; + } + public void setImportedPrefixes(String[] importedPrefixes) { + this.importedPrefixes = importedPrefixes; + } + + public static FormalBinding[] makeFormalBindings(TypeX[] types, String[] names) { + int len = types.length; + FormalBinding[] bindings = new FormalBinding[len]; + for (int i = 0; i < len; i++) { + bindings[i] = new FormalBinding(types[i], names[i], i); + } + return bindings; + } + + // ---- fields + + public static final String[] ZERO_STRINGS = new String[0]; + + public static final String[] javaLangPrefixArray = + new String[] { "java.lang.", }; + + + + public ISourceLocation makeSourceLocation(IHasPosition location) { + return new SourceLocation(ISourceLocation.NO_FILE, 0); + } + + public void message( + IMessage.Kind kind, + IHasPosition location1, + IHasPosition location2, + String message) { + message(kind, location1, message); + message(kind, location2, message); + } + + public void message( + IMessage.Kind kind, + IHasPosition location, + String message) { + getMessageHandler() + .handleMessage(new Message(message, kind, null, makeSourceLocation(location))); + + } + + public World getWorld() { + return world; + } + + public ResolvedTypeX getEnclosingType() { + return enclosingType; + } + +} diff --git a/weaver/src/org/aspectj/weaver/patterns/ThisOrTargetPointcut.java b/weaver/src/org/aspectj/weaver/patterns/ThisOrTargetPointcut.java new file mode 100644 index 000000000..9baf92e10 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/patterns/ThisOrTargetPointcut.java @@ -0,0 +1,108 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.patterns; + +import java.io.*; + +import org.aspectj.util.FuzzyBoolean; +import org.aspectj.weaver.*; +import org.aspectj.weaver.ast.*; + +// + +/** + * Corresponds to target or this pcd. + * + * <p>type is initially a WildTypePattern. If it stays that way, it's a this(Foo) + * type deal. + * however, the resolveBindings method may convert it to a BindingTypePattern, + * in which + * case, it's a this(foo) type deal. + * + * @author Erik Hilsdale + * @author Jim Hugunin + */ +public class ThisOrTargetPointcut extends NameBindingPointcut { + private boolean isThis; + private TypePattern type; + + public ThisOrTargetPointcut(boolean isThis, TypePattern type) { + this.isThis = isThis; + this.type = type; + } + + private boolean couldMatch(Shadow shadow) { + return isThis ? shadow.hasThis() : shadow.hasTarget(); + } + + public FuzzyBoolean match(Shadow shadow) { + if (!couldMatch(shadow)) return FuzzyBoolean.NO; + TypeX typeToMatch = isThis ? shadow.getThisType() : shadow.getTargetType(); + //if (typeToMatch == ResolvedTypeX.MISSING) return FuzzyBoolean.NO; + + return type.matches(typeToMatch.resolve(shadow.getIWorld()), TypePattern.DYNAMIC); + } + + public void write(DataOutputStream s) throws IOException { + s.writeByte(Pointcut.THIS_OR_TARGET); + s.writeBoolean(isThis); + type.write(s); + writeLocation(s); + } + public static Pointcut read(DataInputStream s, ISourceContext context) throws IOException { + boolean isThis = s.readBoolean(); + TypePattern type = TypePattern.read(s, context); + ThisOrTargetPointcut ret = new ThisOrTargetPointcut(isThis, type); + ret.readLocation(context, s); + return ret; + } + + public void resolveBindings(IScope scope, Bindings bindings) { + type = type.resolveBindings(scope, bindings, true); + // ??? handle non-formal + } + + public void postRead(ResolvedTypeX enclosingType) { + type.postRead(enclosingType); + } + + public boolean equals(Object other) { + if (!(other instanceof ThisOrTargetPointcut)) return false; + ThisOrTargetPointcut o = (ThisOrTargetPointcut)other; + return o.isThis == this.isThis && o.type.equals(this.type); + } + public int hashCode() { + int result = 17; + result = 37*result + (isThis ? 0 : 1); + result = 37*result + type.hashCode(); + return result; + } + public String toString() { + return (isThis ? "this(" : "target(") + type + ")"; + } + + public Test findResidue(Shadow shadow, ExposedState state) { + if (!couldMatch(shadow)) return Literal.FALSE; + + if (type == TypePattern.ANY) return Literal.TRUE; + + Var var = isThis ? shadow.getThisVar() : shadow.getTargetVar(); + return exposeStateForVar(var, type, state, shadow.getIWorld()); + } + + public Pointcut concretize1(ResolvedTypeX inAspect, IntMap bindings) { + return new ThisOrTargetPointcut(isThis, type.remapAdviceFormals(bindings)); + } + +} diff --git a/weaver/src/org/aspectj/weaver/patterns/ThrowsPattern.java b/weaver/src/org/aspectj/weaver/patterns/ThrowsPattern.java new file mode 100644 index 000000000..591787911 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/patterns/ThrowsPattern.java @@ -0,0 +1,110 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.patterns; + +import java.io.*; +import java.util.*; + +import org.aspectj.weaver.*; +import org.aspectj.weaver.TypeX; + + +public class ThrowsPattern extends PatternNode { + private TypePatternList required; + private TypePatternList forbidden; + + public static final ThrowsPattern ANY = + new ThrowsPattern(TypePatternList.EMPTY, TypePatternList.EMPTY); + + public ThrowsPattern(TypePatternList required, TypePatternList forbidden) { + this.required = required; + this.forbidden = forbidden; + } + + public String toString() { + if (this == ANY) return ""; + + + String ret = "throws " + required.toString(); + if (forbidden.size() > 0) { + ret = ret + " !(" + forbidden.toString() + ")"; + } + return ret; + } + + public boolean equals(Object other) { + if (!(other instanceof ThrowsPattern)) return false; + ThrowsPattern o = (ThrowsPattern)other; + return o.required.equals(this.required) && + o.forbidden.equals(this.forbidden); + } + public int hashCode() { + int result = 17; + result = 37*result + required.hashCode(); + result = 37*result + forbidden.hashCode(); + return result; + } + + public ThrowsPattern resolveBindings(IScope scope, Bindings bindings) { + if (this == ANY) return this; + required = required.resolveBindings(scope, bindings, false); + forbidden = forbidden.resolveBindings(scope, bindings, false); + return this; + } + + public boolean matches(TypeX[] tys, World world) { + if (this == ANY) return true; + + //System.out.println("matching: " + this + " with " + Arrays.asList(tys)); + + ResolvedTypeX[] types = world.resolve(tys); + int len = types.length; + for (int j=0, lenj = required.size(); j < lenj; j++) { + if (! matchesAny(required.get(j), types)) { + return false; + } + } + for (int j=0, lenj = forbidden.size(); j < lenj; j++) { + if (matchesAny(forbidden.get(j), types)) { + return false; + } + } + return true; + } + + private boolean matchesAny( + TypePattern typePattern, + ResolvedTypeX[] types) + { + for (int i = types.length - 1; i >= 0; i--) { + if (typePattern.matchesExactly(types[i])) return true; + } + return false; + } + + public static ThrowsPattern read(DataInputStream s, ISourceContext context) throws IOException { + TypePatternList required = TypePatternList.read(s, context); + TypePatternList forbidden = TypePatternList.read(s, context); + if (required.size() == 0 && forbidden.size() == 0) return ANY; + ThrowsPattern ret = new ThrowsPattern(required, forbidden); + //XXXret.readLocation(context, s); + return ret; + } + + public void write(DataOutputStream s) throws IOException { + required.write(s); + forbidden.write(s); + //XXXwriteLocation(s); + } +} diff --git a/weaver/src/org/aspectj/weaver/patterns/TypePattern.java b/weaver/src/org/aspectj/weaver/patterns/TypePattern.java new file mode 100644 index 000000000..341138756 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/patterns/TypePattern.java @@ -0,0 +1,263 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.patterns; + +import java.io.*; +import java.util.*; + +import org.aspectj.weaver.*; +import org.aspectj.bridge.*; +import org.aspectj.bridge.IMessageHandler; +import org.aspectj.util.*; +/** + * On creation, type pattern only contains WildTypePattern nodes, not BindingType or ExactType. + * + * <p>Then we call resolveBindings() during compilation + * During concretization of enclosing pointcuts, we call remapAdviceFormals + * + * @author Erik Hilsdale + * @author Jim Hugunin + */ +public abstract class TypePattern extends PatternNode { + public static class MatchKind { + private String name; + public MatchKind(String name) { this.name = name; } + public String toString() { return name; } + } + + public static final MatchKind STATIC = new MatchKind("STATIC"); + public static final MatchKind DYNAMIC = new MatchKind("DYNAMIC"); + + public static final TypePattern ELLIPSIS = new EllipsisTypePattern(); + public static final TypePattern ANY = new AnyTypePattern(); + + + protected boolean includeSubtypes; + + protected TypePattern(boolean includeSubtypes) { + this.includeSubtypes = includeSubtypes; + } + + //XXX non-final for Not, && and || + public boolean matchesStatically(ResolvedTypeX type) { + if (includeSubtypes) { + return matchesSubtypes(type); + } else { + return matchesExactly(type); + } + } + public abstract FuzzyBoolean matchesInstanceof(ResolvedTypeX type); + + + public final FuzzyBoolean matches(ResolvedTypeX type, MatchKind kind) { + if (kind == STATIC) { + return FuzzyBoolean.fromBoolean(matchesStatically(type)); + } else if (kind == DYNAMIC) { + //System.err.println("matching: " + this + " with " + type); + FuzzyBoolean ret = matchesInstanceof(type); + //System.err.println(" got: " + ret); + return ret; + } else { + throw new IllegalArgumentException("kind must be DYNAMIC or STATIC"); + } + } + + + protected abstract boolean matchesExactly(ResolvedTypeX type); + protected boolean matchesSubtypes(ResolvedTypeX type) { + //System.out.println("matching: " + this + " to " + type); + if (matchesExactly(type)) { + //System.out.println(" true"); + return true; + } + + FuzzyBoolean ret = FuzzyBoolean.NO; // ??? -eh + for (Iterator i = type.getDirectSupertypes(); i.hasNext(); ) { + ResolvedTypeX superType = (ResolvedTypeX)i.next(); + if (matchesSubtypes(superType)) return true; + } + return false; + } + + public TypeX resolveExactType(IScope scope, Bindings bindings) { + TypePattern p = resolveBindings(scope, bindings, false); + if (!p.assertExactType(scope.getMessageHandler())) return ResolvedTypeX.MISSING; + + return ((ExactTypePattern)p).getType(); + } + + public TypeX getExactType() { + if (this instanceof ExactTypePattern) return ((ExactTypePattern)this).getType(); + else return ResolvedTypeX.MISSING; + } + + public boolean assertExactType(IMessageHandler m) { + if (this instanceof ExactTypePattern) return true; + + //XXX should try harder to avoid multiple errors for one problem + m.handleMessage(MessageUtil.error("exact type pattern required", getSourceLocation())); + return false; + } + + /** + * This can modify in place, or return a new TypePattern if the type changes. + */ + public TypePattern resolveBindings(IScope scope, Bindings bindings, boolean allowBinding) { + return this; + } + + public void postRead(ResolvedTypeX enclosingType) { + } + + public boolean isStar() { + return false; + } + + + + /** + * This is called during concretization of pointcuts, it is used by BindingTypePattern + * to return a new BindingTypePattern with a formal index appropiate for the advice, + * rather than for the lexical declaration, i.e. this handles transforamtions through + * named pointcuts. + * <pre> + * pointcut foo(String name): args(name); + * --> This makes a BindingTypePattern(0) pointing to the 0th formal + * + * before(Foo f, String n): this(f) && foo(n) { ... } + * --> when resolveReferences is called on the args from the above, it + * will return a BindingTypePattern(1) + * + * before(Foo f): this(f) && foo(*) { ... } + * --> when resolveReferences is called on the args from the above, it + * will return an ExactTypePattern(String) + * </pre> + */ + public TypePattern remapAdviceFormals(IntMap bindings) { + return this; + } + + + public static final byte WILD = 1; + public static final byte EXACT = 2; + public static final byte BINDING = 3; + public static final byte ELLIPSIS_KEY = 4; + public static final byte ANY_KEY = 5; + public static final byte NOT = 6; + public static final byte OR = 7; + public static final byte AND = 8; + + public static TypePattern read(DataInputStream s, ISourceContext context) throws IOException { + byte key = s.readByte(); + switch(key) { + case WILD: return WildTypePattern.read(s, context); + case EXACT: return ExactTypePattern.read(s, context); + case BINDING: return BindingTypePattern.read(s, context); + case ELLIPSIS_KEY: return ELLIPSIS; + case ANY_KEY: return ANY; + case NOT: return NotTypePattern.read(s, context); + case OR: return OrTypePattern.read(s, context); + case AND: return AndTypePattern.read(s, context); + } + throw new BCException("unknown TypePattern kind: " + key); + } +} + +class EllipsisTypePattern extends TypePattern { + + /** + * Constructor for EllipsisTypePattern. + * @param includeSubtypes + */ + public EllipsisTypePattern() { + super(false); + } + + /** + * @see org.aspectj.weaver.patterns.TypePattern#matchesExactly(IType) + */ + protected boolean matchesExactly(ResolvedTypeX type) { + return false; + } + + /** + * @see org.aspectj.weaver.patterns.TypePattern#matchesInstanceof(IType) + */ + public FuzzyBoolean matchesInstanceof(ResolvedTypeX type) { + return FuzzyBoolean.NO; + } + + /** + * @see org.aspectj.weaver.patterns.PatternNode#write(DataOutputStream) + */ + public void write(DataOutputStream s) throws IOException { + s.writeByte(ELLIPSIS_KEY); + } + + public String toString() { return ".."; } +} + +class AnyTypePattern extends TypePattern { + + /** + * Constructor for EllipsisTypePattern. + * @param includeSubtypes + */ + public AnyTypePattern() { + super(false); + } + + /** + * @see org.aspectj.weaver.patterns.TypePattern#matchesExactly(IType) + */ + protected boolean matchesExactly(ResolvedTypeX type) { + return true; + } + + /** + * @see org.aspectj.weaver.patterns.TypePattern#matchesInstanceof(IType) + */ + public FuzzyBoolean matchesInstanceof(ResolvedTypeX type) { + return FuzzyBoolean.YES; + } + + /** + * @see org.aspectj.weaver.patterns.PatternNode#write(DataOutputStream) + */ + public void write(DataOutputStream s) throws IOException { + s.writeByte(ANY_KEY); + } + + /** + * @see org.aspectj.weaver.patterns.TypePattern#matches(IType, MatchKind) + */ +// public FuzzyBoolean matches(IType type, MatchKind kind) { +// return FuzzyBoolean.YES; +// } + + /** + * @see org.aspectj.weaver.patterns.TypePattern#matchesSubtypes(IType) + */ + protected boolean matchesSubtypes(ResolvedTypeX type) { + return true; + } + + + public boolean isStar() { + return true; + } + + public String toString() { return "*"; } + +} diff --git a/weaver/src/org/aspectj/weaver/patterns/TypePatternList.java b/weaver/src/org/aspectj/weaver/patterns/TypePatternList.java new file mode 100644 index 000000000..1f7abffc0 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/patterns/TypePatternList.java @@ -0,0 +1,234 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.patterns; + +import java.io.*; +import java.util.*; + +import org.aspectj.weaver.*; +import org.aspectj.util.*; + +public class TypePatternList extends PatternNode { + private TypePattern[] typePatterns; + int ellipsisCount = 0; + + public static final TypePatternList EMPTY = + new TypePatternList(new TypePattern[] {}); + + public static final TypePatternList ANY = + new TypePatternList(new TypePattern[] {TypePattern.ELLIPSIS}); + + public TypePatternList() { + typePatterns = new TypePattern[0]; + ellipsisCount = 0; + } + + public TypePatternList(TypePattern[] arguments) { + this.typePatterns = arguments; + for (int i=0; i<arguments.length; i++) { + if (arguments[i] == TypePattern.ELLIPSIS) ellipsisCount++; + } + } + + public TypePatternList(List l) { + this((TypePattern[]) l.toArray(new TypePattern[l.size()])); + } + + public int size() { return typePatterns.length; } + + public TypePattern get(int index) { + return typePatterns[index]; + } + + public String toString() { + StringBuffer buf = new StringBuffer(); + buf.append("("); + for (int i=0, len=typePatterns.length; i < len; i++) { + TypePattern type = typePatterns[i]; + if (i > 0) buf.append(", "); + if (type == TypePattern.ELLIPSIS) { + buf.append(".."); + } else { + buf.append(type.toString()); + } + } + buf.append(")"); + return buf.toString(); + } + + //XXX shares much code with WildTypePattern and with NamePattern + /** + * When called with TypePattern.STATIC this will always return either + * FuzzyBoolean.YES or FuzzyBoolean.NO. + * + * When called with TypePattern.DYNAMIC this could return MAYBE if + * at runtime it would be possible for arguments of the given static + * types to dynamically match this, but it is not known for certain. + * + * This method will never return FuzzyBoolean.NEVER + */ + public FuzzyBoolean matches(ResolvedTypeX[] types, TypePattern.MatchKind kind) { + int nameLength = types.length; + int patternLength = typePatterns.length; + + int nameIndex = 0; + int patternIndex = 0; + + if (ellipsisCount == 0) { + if (nameLength != patternLength) return FuzzyBoolean.NO; + FuzzyBoolean finalReturn = FuzzyBoolean.YES; + while (patternIndex < patternLength) { + FuzzyBoolean ret = typePatterns[patternIndex++].matches(types[nameIndex++], kind); + if (ret == FuzzyBoolean.NO) return ret; + if (ret == FuzzyBoolean.MAYBE) finalReturn = ret; + } + return finalReturn; + } else if (ellipsisCount == 1) { + if (nameLength < patternLength-1) return FuzzyBoolean.NO; + FuzzyBoolean finalReturn = FuzzyBoolean.YES; + while (patternIndex < patternLength) { + TypePattern p = typePatterns[patternIndex++]; + if (p == TypePattern.ELLIPSIS) { + nameIndex = nameLength - (patternLength-patternIndex); + } else { + FuzzyBoolean ret = p.matches(types[nameIndex++], kind); + if (ret == FuzzyBoolean.NO) return ret; + if (ret == FuzzyBoolean.MAYBE) finalReturn = ret; + } + } + return finalReturn; + } else { +// System.err.print("match(" + arguments + ", " + types + ") -> "); + FuzzyBoolean b = outOfStar(typePatterns, types, 0, 0, patternLength - ellipsisCount, nameLength, ellipsisCount, kind); +// System.err.println(b); + return b; + } + } + private static FuzzyBoolean outOfStar(final TypePattern[] pattern, final ResolvedTypeX[] target, + int pi, int ti, + int pLeft, int tLeft, + final int starsLeft, TypePattern.MatchKind kind) { + if (pLeft > tLeft) return FuzzyBoolean.NO; + FuzzyBoolean finalReturn = FuzzyBoolean.YES; + while (true) { + // invariant: if (tLeft > 0) then (ti < target.length && pi < pattern.length) + if (tLeft == 0) return finalReturn; + if (pLeft == 0) { + if (starsLeft > 0) { + return finalReturn; + } else { + return FuzzyBoolean.NO; + } + } + if (pattern[pi] == TypePattern.ELLIPSIS) { + return inStar(pattern, target, pi+1, ti, pLeft, tLeft, starsLeft-1, kind); + } + FuzzyBoolean ret = pattern[pi].matches(target[ti], kind); + if (ret == FuzzyBoolean.NO) return ret; + if (ret == FuzzyBoolean.MAYBE) finalReturn = ret; + pi++; ti++; pLeft--; tLeft--; + } + } + private static FuzzyBoolean inStar(final TypePattern[] pattern, final ResolvedTypeX[] target, + int pi, int ti, + final int pLeft, int tLeft, + int starsLeft, TypePattern.MatchKind kind) { + // invariant: pLeft > 0, so we know we'll run out of stars and find a real char in pattern + TypePattern patternChar = pattern[pi]; + while (patternChar == TypePattern.ELLIPSIS) { + starsLeft--; + patternChar = pattern[++pi]; + } + while (true) { + // invariant: if (tLeft > 0) then (ti < target.length) + if (pLeft > tLeft) return FuzzyBoolean.NO; + FuzzyBoolean ff = patternChar.matches(target[ti], kind); + if (ff.maybeTrue()) { + FuzzyBoolean xx = outOfStar(pattern, target, pi+1, ti+1, pLeft-1, tLeft-1, starsLeft, kind); + if (xx.maybeTrue()) return ff.and(xx); + } + ti++; tLeft--; + } + } + + public TypePatternList resolveBindings(IScope scope, Bindings bindings, boolean allowBinding) { + for (int i=0; i<typePatterns.length; i++) { + TypePattern p = typePatterns[i]; + if (p != null) { + typePatterns[i] = typePatterns[i].resolveBindings(scope, bindings, allowBinding); + } + } + return this; + } + + public TypePatternList resolveReferences(IntMap bindings) { + int len = typePatterns.length; + TypePattern[] ret = new TypePattern[len]; + for (int i=0; i < len; i++) { + ret[i] = typePatterns[i].remapAdviceFormals(bindings); + } + return new TypePatternList(ret); + } + + public void postRead(ResolvedTypeX enclosingType) { + for (int i=0; i<typePatterns.length; i++) { + TypePattern p = typePatterns[i]; + p.postRead(enclosingType); + } + } + + + public boolean equals(Object other) { + if (!(other instanceof TypePatternList)) return false; + TypePatternList o = (TypePatternList)other; + int len = o.typePatterns.length; + if (len != this.typePatterns.length) return false; + for (int i=0; i<len; i++) { + if (!this.typePatterns[i].equals(o.typePatterns[i])) return false; + } + return true; + } + public int hashCode() { + int result = 41; + for (int i = 0, len = typePatterns.length; i < len; i++) { + result = 37*result + typePatterns[i].hashCode(); + } + return result; + } + + + public static TypePatternList read(DataInputStream s, ISourceContext context) throws IOException { + short len = s.readShort(); + TypePattern[] arguments = new TypePattern[len]; + for (int i=0; i<len; i++) { + arguments[i] = TypePattern.read(s, context); + } + TypePatternList ret = new TypePatternList(arguments); + ret.readLocation(context, s); + return ret; + } + + + public void write(DataOutputStream s) throws IOException { + s.writeShort(typePatterns.length); + for (int i=0; i<typePatterns.length; i++) { + typePatterns[i].write(s); + } + writeLocation(s); + } + public TypePattern[] getTypePatterns() { + return typePatterns; + } + +} diff --git a/weaver/src/org/aspectj/weaver/patterns/TypePatternQuestions.java b/weaver/src/org/aspectj/weaver/patterns/TypePatternQuestions.java new file mode 100644 index 000000000..d9ebe0983 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/patterns/TypePatternQuestions.java @@ -0,0 +1,109 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.patterns; + +import java.util.*; +import java.util.HashMap; +import java.util.Map; + +import org.aspectj.weaver.*; +import org.aspectj.weaver.ResolvedTypeX; +import org.aspectj.weaver.TypeX; +import org.aspectj.util.FuzzyBoolean; + + +public class TypePatternQuestions { + private Map questionsAndAnswers = new HashMap(); + + public FuzzyBoolean askQuestion(TypePattern pattern, ResolvedTypeX type, + TypePattern.MatchKind kind) + { + Question question = new Question(pattern, type, kind); + //??? should we use this table to do caching or is that a pessimization + //??? if we do that optimization we can also do error checking that the result + //??? doesn't change + FuzzyBoolean answer = question.ask(); + questionsAndAnswers.put(question, answer); + return answer; + } + + public Question anyChanges() { + for (Iterator i = questionsAndAnswers.entrySet().iterator(); i.hasNext(); ) { + Map.Entry entry = (Map.Entry)i.next(); + Question question = (Question)entry.getKey(); + FuzzyBoolean expectedAnswer = (FuzzyBoolean)entry.getValue(); + + FuzzyBoolean currentAnswer = question.ask(); + //System.out.println(question + ":" + currentAnswer); + if (currentAnswer != expectedAnswer) { + return question; + } + } + + return null; + } + + public String toString() { + StringBuffer buf = new StringBuffer(); + buf.append("TypePatternQuestions{"); + for (Iterator i = questionsAndAnswers.entrySet().iterator(); i.hasNext(); ) { + Map.Entry entry = (Map.Entry)i.next(); + Question question = (Question)entry.getKey(); + FuzzyBoolean expectedAnswer = (FuzzyBoolean)entry.getValue(); + buf.append(question); + buf.append(":"); + buf.append(expectedAnswer); + buf.append(", "); + } + buf.append("}"); + return buf.toString(); + } + + + public class Question { + TypePattern pattern; + ResolvedTypeX type; + TypePattern.MatchKind kind; + + public Question(TypePattern pattern, ResolvedTypeX type, + TypePattern.MatchKind kind) { + super(); + this.pattern = pattern; + this.type = type; + this.kind = kind; + } + + public FuzzyBoolean ask() { + return pattern.matches(type, kind); + } + + public boolean equals(Object other) { + if (!(other instanceof Question)) return false; + Question o = (Question)other; + return o.pattern.equals(pattern) && o.type.equals(type) && o.kind == kind; + } + + public int hashCode() { + int result = 17; + result = 37*result + kind.hashCode(); + result = 37*result + pattern.hashCode(); + result = 37*result + type.hashCode(); + return result; + } + + public String toString() { + return "?(" + pattern + ", " + type + ", " + kind + ")"; + } + } +} diff --git a/weaver/src/org/aspectj/weaver/patterns/WildTypePattern.java b/weaver/src/org/aspectj/weaver/patterns/WildTypePattern.java new file mode 100644 index 000000000..1aed2f2ea --- /dev/null +++ b/weaver/src/org/aspectj/weaver/patterns/WildTypePattern.java @@ -0,0 +1,428 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.patterns; + +import java.io.*; +import java.util.*; + +import org.aspectj.weaver.*; +import org.aspectj.bridge.*; +import org.aspectj.bridge.IMessage; +import org.aspectj.util.*; + +//XXX need to use dim in matching +public class WildTypePattern extends TypePattern { + NamePattern[] namePatterns; + int ellipsisCount; + String[] importedPrefixes; + String[] knownMatches; + int dim; + + WildTypePattern(NamePattern[] namePatterns, boolean includeSubtypes, int dim) { + super(includeSubtypes); + this.namePatterns = namePatterns; + this.ellipsisCount = ellipsisCount; + this.dim = dim; + ellipsisCount = 0; + for (int i=0; i<namePatterns.length; i++) { + if (namePatterns[i] == NamePattern.ELLIPSIS) ellipsisCount++; + } + setLocation(namePatterns[0].getSourceContext(), namePatterns[0].getStart(), namePatterns[namePatterns.length-1].getEnd()); + } + + public WildTypePattern(List names, boolean includeSubtypes, int dim) { + this((NamePattern[])names.toArray(new NamePattern[names.size()]), includeSubtypes, dim); + + } + + public WildTypePattern(List names, boolean includeSubtypes, int dim, int endPos) { + this(names, includeSubtypes, dim); + this.end = endPos; + } + + //XXX inefficient implementation + public static char[][] splitNames(String s) { + List ret = new ArrayList(); + int startIndex = 0; + while (true) { + int breakIndex = s.indexOf('.', startIndex); // what about / + if (breakIndex == -1) breakIndex = s.indexOf('$', startIndex); // we treat $ like . here + if (breakIndex == -1) break; + char[] name = s.substring(startIndex, breakIndex).toCharArray(); + ret.add(name); + startIndex = breakIndex+1; + } + ret.add(s.substring(startIndex).toCharArray()); + return (char[][])ret.toArray(new char[ret.size()][]); + } + + + /** + * @see org.aspectj.weaver.TypePattern#matchesExactly(IType) + */ + protected boolean matchesExactly(ResolvedTypeX type) { + String targetTypeName = type.getName(); + + //XXX hack + if (knownMatches == null) { + return innerMatchesExactly(targetTypeName); + } + + // if our pattern is length 1, then known matches are exact matches + // if it's longer than that, then known matches are prefixes of a sort + if (namePatterns.length == 1) { + for (int i=0, len=knownMatches.length; i < len; i++) { + if (knownMatches[i].equals(targetTypeName)) return true; + } + } else { + for (int i=0, len=knownMatches.length; i < len; i++) { + String knownPrefix = knownMatches[i] + "$"; + if (targetTypeName.startsWith(knownPrefix)) { + int pos = lastIndexOfDotOrDollar(knownMatches[i]); + if (innerMatchesExactly(targetTypeName.substring(pos+1))) { + return true; + } + } + } + } + + + // if any prefixes match, strip the prefix and check that the rest matches + // assumes that prefixes have a dot at the end + for (int i=0, len=importedPrefixes.length; i < len; i++) { + String prefix = importedPrefixes[i]; + if (targetTypeName.startsWith(prefix)) { + if (innerMatchesExactly(targetTypeName.substring(prefix.length()))) { + return true; + } + } + } + + return innerMatchesExactly(targetTypeName); + } + + private int lastIndexOfDotOrDollar(String string) { + int dot = string.lastIndexOf('.'); + int dollar = string.lastIndexOf('$'); + return Math.max(dot, dollar); + } + + + private boolean innerMatchesExactly(String targetTypeName) { + //??? doing this everytime is not very efficient + char[][] names = splitNames(targetTypeName); + + + + return innerMatchesExactly(names); + } + + private boolean innerMatchesExactly(char[][] names) { + + int namesLength = names.length; + int patternsLength = namePatterns.length; + + int namesIndex = 0; + int patternsIndex = 0; + + if (ellipsisCount == 0) { + if (namesLength != patternsLength) return false; + while (patternsIndex < patternsLength) { + if (!namePatterns[patternsIndex++].matches(names[namesIndex++])) { + return false; + } + } + return true; + } else if (ellipsisCount == 1) { + if (namesLength < patternsLength-1) return false; + while (patternsIndex < patternsLength) { + NamePattern p = namePatterns[patternsIndex++]; + if (p == NamePattern.ELLIPSIS) { + namesIndex = namesLength - (patternsLength-patternsIndex); + } else { + if (!p.matches(names[namesIndex++])) { + return false; + } + } + } + return true; + } else { + // System.err.print("match(\"" + Arrays.asList(namePatterns) + "\", \"" + Arrays.asList(names) + "\") -> "); + boolean b = outOfStar(namePatterns, names, 0, 0, patternsLength - ellipsisCount, namesLength, ellipsisCount); + // System.err.println(b); + return b; + } + } + private static boolean outOfStar(final NamePattern[] pattern, final char[][] target, + int pi, int ti, + int pLeft, int tLeft, + final int starsLeft) { + if (pLeft > tLeft) return false; + while (true) { + // invariant: if (tLeft > 0) then (ti < target.length && pi < pattern.length) + if (tLeft == 0) return true; + if (pLeft == 0) { + return (starsLeft > 0); + } + if (pattern[pi] == NamePattern.ELLIPSIS) { + return inStar(pattern, target, pi+1, ti, pLeft, tLeft, starsLeft-1); + } + if (! pattern[pi].matches(target[ti])) { + return false; + } + pi++; ti++; pLeft--; tLeft--; + } + } + private static boolean inStar(final NamePattern[] pattern, final char[][] target, + int pi, int ti, + final int pLeft, int tLeft, + int starsLeft) { + // invariant: pLeft > 0, so we know we'll run out of stars and find a real char in pattern + // of course, we probably can't parse multiple ..'s in a row, but this keeps the algorithm + // exactly parallel with that in NamePattern + NamePattern patternChar = pattern[pi]; + while (patternChar == NamePattern.ELLIPSIS) { + starsLeft--; + patternChar = pattern[++pi]; + } + while (true) { + // invariant: if (tLeft > 0) then (ti < target.length) + if (pLeft > tLeft) return false; + if (patternChar.matches(target[ti])) { + if (outOfStar(pattern, target, pi+1, ti+1, pLeft-1, tLeft-1, starsLeft)) return true; + } + ti++; tLeft--; + } + } + + /** + * @see org.aspectj.weaver.TypePattern#matchesInstanceof(IType) + */ + public FuzzyBoolean matchesInstanceof(ResolvedTypeX type) { + //XXX hack to let unmatched types just silently remain so + if (maybeGetSimpleName() != null) return FuzzyBoolean.NO; + + type.getWorld().getMessageHandler().handleMessage( + new Message("can't do instanceof matching on patterns with wildcards", + IMessage.ERROR, null, getSourceLocation())); + return FuzzyBoolean.NO; + } + + public NamePattern extractName() { + //System.err.println("extract from : " + Arrays.asList(namePatterns)); + int len = namePatterns.length; + NamePattern ret = namePatterns[len-1]; + NamePattern[] newNames = new NamePattern[len-1]; + System.arraycopy(namePatterns, 0, newNames, 0, len-1); + namePatterns = newNames; + //System.err.println(" left : " + Arrays.asList(namePatterns)); + return ret; + } + + /** + * Method maybeExtractName. + * @param string + * @return boolean + */ + public boolean maybeExtractName(String string) { + int len = namePatterns.length; + NamePattern ret = namePatterns[len-1]; + String simple = ret.maybeGetSimpleName(); + if (simple != null && simple.equals(string)) { + extractName(); + return true; + } + return false; + } + + /** + * If this type pattern has no '.' or '*' in it, then + * return a simple string + * + * otherwise, this will return null; + */ + public String maybeGetSimpleName() { + if (namePatterns.length == 1) { + return namePatterns[0].maybeGetSimpleName(); + } + return null; + } + + /** + * If this type pattern has no '*' or '..' in it + */ + public String maybeGetCleanName() { + if (namePatterns.length == 0) { + throw new RuntimeException("bad name: " + namePatterns); + } + //System.out.println("get clean: " + this); + StringBuffer buf = new StringBuffer(); + for (int i=0, len=namePatterns.length; i < len; i++) { + NamePattern p = namePatterns[i]; + String simpleName = p.maybeGetSimpleName(); + if (simpleName == null) return null; + if (i > 0) buf.append("."); + buf.append(simpleName); + } + //System.out.println(buf); + return buf.toString(); + } + + + /** + * Need to determine if I'm really a pattern or a reference to a formal + * + * We may wish to further optimize the case of pattern vs. non-pattern + * + * We will be replaced by what we return + */ + public TypePattern resolveBindings(IScope scope, Bindings bindings, boolean allowBinding) { + if (isStar()) { + return TypePattern.ANY; //??? loses source location + } + + String simpleName = maybeGetSimpleName(); + if (simpleName != null) { + FormalBinding formalBinding = scope.lookupFormal(simpleName); + if (formalBinding != null) { + if (!allowBinding || bindings == null) { + scope.message(IMessage.ERROR, this, "negation doesn't allow binding"); + return this; + } + BindingTypePattern binding = new BindingTypePattern(formalBinding); + binding.copyLocationFrom(this); + bindings.register(binding, scope); + + return binding; + } + } + + String cleanName = maybeGetCleanName(); + if (cleanName != null) { + TypeX type; + + //System.out.println("resolve: " + cleanName); + //??? this loop has too many inefficiencies to count + while ((type = scope.lookupType(cleanName, this)) == ResolvedTypeX.MISSING) { + int lastDot = cleanName.lastIndexOf('.'); + if (lastDot == -1) break; + cleanName = cleanName.substring(0, lastDot) + '$' + cleanName.substring(lastDot+1); + } + if (type == ResolvedTypeX.MISSING) { + if (scope.getWorld().getLint().invalidAbsoluteTypeName.isEnabled()) { + scope.getWorld().getLint().invalidAbsoluteTypeName.signal(cleanName, getSourceLocation()); + } + } else { + TypePattern ret = new ExactTypePattern(type, includeSubtypes); + ret.copyLocationFrom(this); + return ret; + } + } else { + //XXX need to implement behavior for Lint.invalidWildcardTypeName + } + + importedPrefixes = scope.getImportedPrefixes(); + knownMatches = preMatch(scope.getImportedNames()); + + return this; + } + + public boolean isStar() { + return namePatterns.length == 1 && namePatterns[0].isAny(); + } + + /** + * returns those possible matches which I match exactly the last element of + */ + private String[] preMatch(String[] possibleMatches) { + //if (namePatterns.length != 1) return CollectionUtil.NO_STRINGS; + + List ret = new ArrayList(); + for (int i=0, len=possibleMatches.length; i < len; i++) { + char[][] names = splitNames(possibleMatches[i]); //??? not most efficient + if (namePatterns[0].matches(names[names.length-1])) { + ret.add(possibleMatches[i]); + } + } + return (String[])ret.toArray(new String[ret.size()]); + } + + +// public void postRead(ResolvedTypeX enclosingType) { +// this.importedPrefixes = enclosingType.getImportedPrefixes(); +// this.knownNames = prematch(enclosingType.getImportedNames()); +// } + + + public String toString() { + StringBuffer buf = new StringBuffer(); + for (int i=0, len=namePatterns.length; i < len; i++) { + NamePattern name = namePatterns[i]; + if (name == null) { + buf.append("."); + } else { + if (i > 0) buf.append("."); + buf.append(name.toString()); + } + } + return buf.toString(); + } + + public boolean equals(Object other) { + if (!(other instanceof WildTypePattern)) return false; + WildTypePattern o = (WildTypePattern)other; + int len = o.namePatterns.length; + if (len != this.namePatterns.length) return false; + for (int i=0; i < len; i++) { + if (!o.namePatterns[i].equals(this.namePatterns[i])) return false; + } + return true; + } + + public int hashCode() { + int result = 17; + for (int i = 0, len = namePatterns.length; i < len; i++) { + result = 37*result + namePatterns[i].hashCode(); + } + return result; + } + + /** + * @see org.aspectj.weaver.patterns.PatternNode#write(DataOutputStream) + */ + public void write(DataOutputStream s) throws IOException { + s.writeByte(TypePattern.WILD); + s.writeShort(namePatterns.length); + for (int i = 0; i < namePatterns.length; i++) { + namePatterns[i].write(s); + } + s.writeBoolean(includeSubtypes); + s.writeInt(dim); + writeLocation(s); + } + + public static TypePattern read(DataInputStream s, ISourceContext context) throws IOException { + int len = s.readShort(); + NamePattern[] namePatterns = new NamePattern[len]; + for (int i=0; i < len; i++) { + namePatterns[i] = NamePattern.read(s); + } + boolean includeSubtypes = s.readBoolean(); + int dim = s.readInt(); + TypePattern ret = new WildTypePattern(namePatterns, includeSubtypes, dim); + ret.readLocation(context, s); + return ret; + } + +} diff --git a/weaver/src/org/aspectj/weaver/patterns/WithinPointcut.java b/weaver/src/org/aspectj/weaver/patterns/WithinPointcut.java new file mode 100644 index 000000000..2b931985b --- /dev/null +++ b/weaver/src/org/aspectj/weaver/patterns/WithinPointcut.java @@ -0,0 +1,83 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.patterns; + +import java.io.*; + +import org.aspectj.weaver.*; +import org.aspectj.weaver.ast.*; +import org.aspectj.util.*; + +public class WithinPointcut extends Pointcut { + TypePattern type; + + public WithinPointcut(TypePattern type) { + this.type = type; + } + + public FuzzyBoolean match(Shadow shadow) { + TypeX enclosingType = shadow.getEnclosingType(); + while (enclosingType != null) { + if (type.matchesStatically(shadow.getIWorld().resolve(enclosingType))) { + return FuzzyBoolean.YES; + } + enclosingType = enclosingType.getDeclaringType(); + } + return FuzzyBoolean.NO; + } + + public void write(DataOutputStream s) throws IOException { + s.writeByte(Pointcut.WITHIN); + type.write(s); + writeLocation(s); + } + public static Pointcut read(DataInputStream s, ISourceContext context) throws IOException { + TypePattern type = TypePattern.read(s, context); + WithinPointcut ret = new WithinPointcut(type); + ret.readLocation(context, s); + return ret; + } + + public void resolveBindings(IScope scope, Bindings bindings) { + type = type.resolveBindings(scope, bindings, false); + } + + public void postRead(ResolvedTypeX enclosingType) { + type.postRead(enclosingType); + } + + public boolean equals(Object other) { + if (!(other instanceof WithinPointcut)) return false; + WithinPointcut o = (WithinPointcut)other; + return o.type.equals(this.type); + } + public int hashCode() { + int result = 43; + result = 37*result + type.hashCode(); + return result; + } + + public String toString() { + return "within(" + type + ")"; + } + + public Test findResidue(Shadow shadow, ExposedState state) { + return match(shadow).alwaysTrue() ? Literal.TRUE : Literal.FALSE; + } + + + public Pointcut concretize1(ResolvedTypeX inAspect, IntMap bindings) { + return this; //??? no pointers out of here so we're okay + } +} diff --git a/weaver/src/org/aspectj/weaver/patterns/WithincodePointcut.java b/weaver/src/org/aspectj/weaver/patterns/WithincodePointcut.java new file mode 100644 index 000000000..328f31fb4 --- /dev/null +++ b/weaver/src/org/aspectj/weaver/patterns/WithincodePointcut.java @@ -0,0 +1,79 @@ +/* ******************************************************************* + * Copyright (c) 2002 Palo Alto Research Center, Incorporated (PARC). + * All rights reserved. + * This program and the accompanying materials are made available + * under the terms of the Common Public License v1.0 + * which accompanies this distribution and is available at + * http://www.eclipse.org/legal/cpl-v10.html + * + * Contributors: + * Xerox/PARC initial implementation + * ******************************************************************/ + + +package org.aspectj.weaver.patterns; + +import java.io.*; + +import org.aspectj.weaver.*; +import org.aspectj.weaver.ast.*; +import org.aspectj.util.*; + +public class WithincodePointcut extends Pointcut { + SignaturePattern signature; + + public WithincodePointcut(SignaturePattern signature) { + this.signature = signature; + } + + public FuzzyBoolean match(Shadow shadow) { + //This will not match code in local or anonymous classes as if + //they were withincode of the outer signature + return FuzzyBoolean.fromBoolean( + signature.matches(shadow.getEnclosingCodeSignature(), shadow.getIWorld())); + } + + public void write(DataOutputStream s) throws IOException { + s.writeByte(Pointcut.WITHINCODE); + signature.write(s); + writeLocation(s); + } + + public static Pointcut read(DataInputStream s, ISourceContext context) throws IOException { + WithincodePointcut ret = new WithincodePointcut(SignaturePattern.read(s, context)); + ret.readLocation(context, s); + return ret; + } + + public void resolveBindings(IScope scope, Bindings bindings) { + signature = signature.resolveBindings(scope, bindings); + } + + public void postRead(ResolvedTypeX enclosingType) { + signature.postRead(enclosingType); + } + + public boolean equals(Object other) { + if (!(other instanceof WithincodePointcut)) return false; + WithincodePointcut o = (WithincodePointcut)other; + return o.signature.equals(this.signature); + } + public int hashCode() { + int result = 43; + result = 37*result + signature.hashCode(); + return result; + } + + public String toString() { + return "withincode(" + signature + ")"; + } + + public Test findResidue(Shadow shadow, ExposedState state) { + return match(shadow).alwaysTrue() ? Literal.TRUE : Literal.FALSE; + } + + + public Pointcut concretize1(ResolvedTypeX inAspect, IntMap bindings) { + return this; //??? no pointers out of here so we're okay + } +} |