aboutsummaryrefslogtreecommitdiffstats
path: root/bcel-builder/verifier-src
diff options
context:
space:
mode:
Diffstat (limited to 'bcel-builder/verifier-src')
-rw-r--r--bcel-builder/verifier-src/org/aspectj/apache/bcel/verifier/statics/Pass2Verifier.java1349
-rw-r--r--bcel-builder/verifier-src/org/aspectj/apache/bcel/verifier/statics/Pass3aVerifier.java18
2 files changed, 746 insertions, 621 deletions
diff --git a/bcel-builder/verifier-src/org/aspectj/apache/bcel/verifier/statics/Pass2Verifier.java b/bcel-builder/verifier-src/org/aspectj/apache/bcel/verifier/statics/Pass2Verifier.java
index f6b15dbb1..b02c35128 100644
--- a/bcel-builder/verifier-src/org/aspectj/apache/bcel/verifier/statics/Pass2Verifier.java
+++ b/bcel-builder/verifier-src/org/aspectj/apache/bcel/verifier/statics/Pass2Verifier.java
@@ -105,199 +105,176 @@ import org.aspectj.apache.bcel.verifier.exc.ClassConstraintException;
import org.aspectj.apache.bcel.verifier.exc.LocalVariableInfoInconsistentException;
/**
- * This PassVerifier verifies a class file according to
- * pass 2 as described in The Java Virtual Machine
- * Specification, 2nd edition.
- * More detailed information is to be found at the do_verify()
- * method's documentation.
- *
- * @version $Id: Pass2Verifier.java,v 1.4 2009/09/09 19:56:20 aclement Exp $
+ * This PassVerifier verifies a class file according to pass 2 as described in The Java Virtual Machine Specification, 2nd edition.
+ * More detailed information is to be found at the do_verify() method's documentation.
+ *
+ * @version $Id: Pass2Verifier.java,v 1.5 2009/09/10 15:35:05 aclement Exp $
* @author <A HREF="http://www.inf.fu-berlin.de/~ehaase"/>Enver Haase</A>
* @see #do_verify()
*/
-public final class Pass2Verifier extends PassVerifier implements Constants{
+public final class Pass2Verifier extends PassVerifier implements Constants {
/**
- * The LocalVariableInfo instances used by Pass3bVerifier.
- * localVariablesInfos[i] denotes the information for the
- * local variables of method number i in the
- * JavaClass this verifier operates on.
+ * The LocalVariableInfo instances used by Pass3bVerifier. localVariablesInfos[i] denotes the information for the local
+ * variables of method number i in the JavaClass this verifier operates on.
*/
private LocalVariablesInfo[] localVariablesInfos;
-
+
/** The Verifier that created this. */
private Verifier myOwner;
/**
* Should only be instantiated by a Verifier.
- *
+ *
* @see Verifier
*/
- public Pass2Verifier(Verifier owner){
+ public Pass2Verifier(Verifier owner) {
myOwner = owner;
}
/**
- * Returns a LocalVariablesInfo object containing information
- * about the usage of the local variables in the Code attribute
- * of the said method or <B>null</B> if the class file this
- * Pass2Verifier operates on could not be pass-2-verified correctly.
- * The method number method_nr is the method you get using
- * <B>Repository.lookupClass(myOwner.getClassname()).getMethods()[method_nr];</B>.
- * You should not add own information. Leave that to JustIce.
+ * Returns a LocalVariablesInfo object containing information about the usage of the local variables in the Code attribute of
+ * the said method or <B>null</B> if the class file this Pass2Verifier operates on could not be pass-2-verified correctly. The
+ * method number method_nr is the method you get using
+ * <B>Repository.lookupClass(myOwner.getClassname()).getMethods()[method_nr];</B>. You should not add own information. Leave
+ * that to JustIce.
*/
- public LocalVariablesInfo getLocalVariablesInfo(int method_nr){
- if (this.verify() != VerificationResult.VR_OK) return null; // It's cached, don't worry.
- if (method_nr < 0 || method_nr >= localVariablesInfos.length){
+ public LocalVariablesInfo getLocalVariablesInfo(int method_nr) {
+ if (this.verify() != VerificationResult.VR_OK)
+ return null; // It's cached, don't worry.
+ if (method_nr < 0 || method_nr >= localVariablesInfos.length) {
throw new AssertionViolatedException("Method number out of range.");
}
return localVariablesInfos[method_nr];
}
-
+
/**
- * Pass 2 is the pass where static properties of the
- * class file are checked without looking into "Code"
- * arrays of methods.
- * This verification pass is usually invoked when
- * a class is resolved; and it may be possible that
- * this verification pass has to load in other classes
- * such as superclasses or implemented interfaces.
- * Therefore, Pass 1 is run on them.<BR>
- * Note that most referenced classes are <B>not</B> loaded
- * in for verification or for an existance check by this
- * pass; only the syntactical correctness of their names
- * and descriptors (a.k.a. signatures) is checked.<BR>
- * Very few checks that conceptually belong here
- * are delayed until pass 3a in JustIce. JustIce does
- * not only check for syntactical correctness but also
- * for semantical sanity - therefore it needs access to
- * the "Code" array of methods in a few cases. Please
- * see the pass 3a documentation, too.
- *
+ * Pass 2 is the pass where static properties of the class file are checked without looking into "Code" arrays of methods. This
+ * verification pass is usually invoked when a class is resolved; and it may be possible that this verification pass has to load
+ * in other classes such as superclasses or implemented interfaces. Therefore, Pass 1 is run on them.<BR>
+ * Note that most referenced classes are <B>not</B> loaded in for verification or for an existance check by this pass; only the
+ * syntactical correctness of their names and descriptors (a.k.a. signatures) is checked.<BR>
+ * Very few checks that conceptually belong here are delayed until pass 3a in JustIce. JustIce does not only check for
+ * syntactical correctness but also for semantical sanity - therefore it needs access to the "Code" array of methods in a few
+ * cases. Please see the pass 3a documentation, too.
+ *
* @see org.aspectj.apache.bcel.verifier.statics.Pass3aVerifier
*/
- public VerificationResult do_verify(){
+ @Override
+ public VerificationResult do_verify() {
VerificationResult vr1 = myOwner.doPass1();
- if (vr1.equals(VerificationResult.VR_OK)){
-
+ if (vr1.equals(VerificationResult.VR_OK)) {
+
// For every method, we could have information about the local variables out of LocalVariableTable attributes of
// the Code attributes.
localVariablesInfos = new LocalVariablesInfo[Repository.lookupClass(myOwner.getClassName()).getMethods().length];
VerificationResult vr = VerificationResult.VR_OK; // default.
- try{
+ try {
constant_pool_entries_satisfy_static_constraints();
field_and_method_refs_are_valid();
every_class_has_an_accessible_superclass();
final_methods_are_not_overridden();
- }
- catch (ClassConstraintException cce){
+ } catch (ClassConstraintException cce) {
vr = new VerificationResult(VerificationResult.VERIFIED_REJECTED, cce.getMessage());
}
return vr;
- }
- else
+ } else
return VerificationResult.VR_NOTYET;
}
/**
- * Ensures that every class has a super class and that
- * <B>final</B> classes are not subclassed.
- * This means, the class this Pass2Verifier operates
- * on has proper super classes (transitively) up to
- * java.lang.Object.
- * The reason for really loading (and Pass1-verifying)
- * all of those classes here is that we need them in
- * Pass2 anyway to verify no final methods are overridden
+ * Ensures that every class has a super class and that <B>final</B> classes are not subclassed. This means, the class this
+ * Pass2Verifier operates on has proper super classes (transitively) up to java.lang.Object. The reason for really loading (and
+ * Pass1-verifying) all of those classes here is that we need them in Pass2 anyway to verify no final methods are overridden
* (that could be declared anywhere in the ancestor hierarchy).
- *
+ *
* @throws ClassConstraintException otherwise.
*/
- private void every_class_has_an_accessible_superclass(){
+ private void every_class_has_an_accessible_superclass() {
HashSet<String> hs = new HashSet<String>(); // save class names to detect circular inheritance
JavaClass jc = Repository.lookupClass(myOwner.getClassName());
int supidx = -1;
- while (supidx != 0){
+ while (supidx != 0) {
supidx = jc.getSuperclassNameIndex();
-
- if (supidx == 0){
- if (jc != Repository.lookupClass(Type.OBJECT.getClassName())){
- throw new ClassConstraintException("Superclass of '"+jc.getClassName()+"' missing but not "+Type.OBJECT.getClassName()+" itself!");
+
+ if (supidx == 0) {
+ if (jc != Repository.lookupClass(Type.OBJECT.getClassName())) {
+ throw new ClassConstraintException("Superclass of '" + jc.getClassName() + "' missing but not "
+ + Type.OBJECT.getClassName() + " itself!");
}
- }
- else{
+ } else {
String supername = jc.getSuperclassName();
- if (! hs.add(supername)){ // If supername already is in the list
+ if (!hs.add(supername)) { // If supername already is in the list
throw new ClassConstraintException("Circular superclass hierarchy detected.");
}
Verifier v = VerifierFactory.getVerifier(supername);
VerificationResult vr = v.doPass1();
- if (vr != VerificationResult.VR_OK){
- throw new ClassConstraintException("Could not load in ancestor class '"+supername+"'.");
+ if (vr != VerificationResult.VR_OK) {
+ throw new ClassConstraintException("Could not load in ancestor class '" + supername + "'.");
}
jc = Repository.lookupClass(supername);
- if (jc.isFinal()){
- throw new ClassConstraintException("Ancestor class '"+supername+"' has the FINAL access modifier and must therefore not be subclassed.");
+ if (jc.isFinal()) {
+ throw new ClassConstraintException("Ancestor class '" + supername
+ + "' has the FINAL access modifier and must therefore not be subclassed.");
}
}
}
}
/**
- * Ensures that <B>final</B> methods are not overridden.
- * <B>Precondition to run this method:
- * constant_pool_entries_satisfy_static_constraints() and
- * every_class_has_an_accessible_superclass() have to be invoked before
+ * Ensures that <B>final</B> methods are not overridden. <B>Precondition to run this method:
+ * constant_pool_entries_satisfy_static_constraints() and every_class_has_an_accessible_superclass() have to be invoked before
* (in that order).</B>
- *
+ *
* @throws ClassConstraintException otherwise.
* @see #constant_pool_entries_satisfy_static_constraints()
* @see #every_class_has_an_accessible_superclass()
*/
- private void final_methods_are_not_overridden(){
+ private void final_methods_are_not_overridden() {
HashMap<String, String> hashmap = new HashMap<String, String>();
JavaClass jc = Repository.lookupClass(myOwner.getClassName());
-
+
int supidx = -1;
- while (supidx != 0){
+ while (supidx != 0) {
supidx = jc.getSuperclassNameIndex();
Method[] methods = jc.getMethods();
- for (int i=0; i<methods.length; i++){
- String name_and_sig = (methods[i].getName()+methods[i].getSignature());
-
- if (hashmap.containsKey(name_and_sig)){
- if (methods[i].isFinal()){
- throw new ClassConstraintException("Method '"+name_and_sig+"' in class '"+hashmap.get(name_and_sig)+"' overrides the final (not-overridable) definition in class '"+jc.getClassName()+"'.");
- }
- else{
- if (!methods[i].isStatic()){ // static methods don't inherit
+ for (int i = 0; i < methods.length; i++) {
+ String name_and_sig = (methods[i].getName() + methods[i].getSignature());
+
+ if (hashmap.containsKey(name_and_sig)) {
+ if (methods[i].isFinal()) {
+ throw new ClassConstraintException("Method '" + name_and_sig + "' in class '" + hashmap.get(name_and_sig)
+ + "' overrides the final (not-overridable) definition in class '" + jc.getClassName() + "'.");
+ } else {
+ if (!methods[i].isStatic()) { // static methods don't inherit
hashmap.put(name_and_sig, jc.getClassName());
}
}
- }
- else{
- if (!methods[i].isStatic()){ // static methods don't inherit
+ } else {
+ if (!methods[i].isStatic()) { // static methods don't inherit
hashmap.put(name_and_sig, jc.getClassName());
}
}
}
-
- jc = Repository.lookupClass(jc.getSuperclassName()); // Well, for OBJECT this returns OBJECT so it works (could return anything but must not throw an Exception).
+
+ jc = Repository.lookupClass(jc.getSuperclassName()); // Well, for OBJECT this returns OBJECT so it works (could return
+ // anything but must not throw an Exception).
}
}
/**
- * Ensures that the constant pool entries satisfy the static constraints
- * as described in The Java Virtual Machine Specification, 2nd Edition.
- *
+ * Ensures that the constant pool entries satisfy the static constraints as described in The Java Virtual Machine Specification,
+ * 2nd Edition.
+ *
* @throws ClassConstraintException otherwise.
*/
- private void constant_pool_entries_satisfy_static_constraints(){
+ private void constant_pool_entries_satisfy_static_constraints() {
// Most of the consistency is handled internally by BCEL; here
// we only have to verify if the indices of the constants point
// to constants of the appropriate type and such.
@@ -306,19 +283,16 @@ public final class Pass2Verifier extends PassVerifier implements Constants{
}
/**
- * A Visitor class that ensures the constant pool satisfies the static
- * constraints.
- * The visitXXX() methods throw ClassConstraintException instances otherwise.
- *
- * @see #constant_pool_entries_satisfy_static_constraints()
+ * A Visitor class that ensures the constant pool satisfies the static constraints. The visitXXX() methods throw
+ * ClassConstraintException instances otherwise.
+ *
+ * @see #constant_pool_entries_satisfy_static_constraints()
*/
- private class CPESSC_Visitor extends org.aspectj.apache.bcel.verifier.EmptyClassVisitor{
+ private class CPESSC_Visitor extends org.aspectj.apache.bcel.verifier.EmptyClassVisitor {
private Class CONST_Class;
/*
- private Class CONST_Fieldref;
- private Class CONST_Methodref;
- private Class CONST_InterfaceMethodref;
- */
+ * private Class CONST_Fieldref; private Class CONST_Methodref; private Class CONST_InterfaceMethodref;
+ */
private Class<ConstantString> CONST_String;
private Class<ConstantInteger> CONST_Integer;
private Class<ConstantFloat> CONST_Float;
@@ -336,17 +310,17 @@ public final class Pass2Verifier extends PassVerifier implements Constants{
private HashSet<String> field_names_and_desc = new HashSet<String>();
private HashSet<String> method_names_and_desc = new HashSet<String>();
- private CPESSC_Visitor(JavaClass _jc){
+ private CPESSC_Visitor(JavaClass _jc) {
jc = _jc;
cp = _jc.getConstantPool();
cplen = cp.getLength();
CONST_Class = org.aspectj.apache.bcel.classfile.ConstantClass.class;
/*
- CONST_Fieldref = org.aspectj.apache.bcel.classfile.ConstantFieldref.class;
- CONST_Methodref = org.aspectj.apache.bcel.classfile.ConstantMethodref.class;
- CONST_InterfaceMethodref = org.aspectj.apache.bcel.classfile.ConstantInterfaceMethodref.class;
- */
+ * CONST_Fieldref = org.aspectj.apache.bcel.classfile.ConstantFieldref.class; CONST_Methodref =
+ * org.aspectj.apache.bcel.classfile.ConstantMethodref.class; CONST_InterfaceMethodref =
+ * org.aspectj.apache.bcel.classfile.ConstantInterfaceMethodref.class;
+ */
CONST_String = org.aspectj.apache.bcel.classfile.ConstantString.class;
CONST_Integer = org.aspectj.apache.bcel.classfile.ConstantInteger.class;
CONST_Float = org.aspectj.apache.bcel.classfile.ConstantFloat.class;
@@ -359,20 +333,23 @@ public final class Pass2Verifier extends PassVerifier implements Constants{
carrier.visit();
}
- private void checkIndex(Node referrer, int index, Class shouldbe){
- if ((index < 0) || (index >= cplen)){
- throw new ClassConstraintException("Invalid index '"+index+"' used by '"+tostring(referrer)+"'.");
+ private void checkIndex(Node referrer, int index, Class shouldbe) {
+ if ((index < 0) || (index >= cplen)) {
+ throw new ClassConstraintException("Invalid index '" + index + "' used by '" + tostring(referrer) + "'.");
}
Constant c = cp.getConstant(index);
- if (! shouldbe.isInstance(c)){
+ if (!shouldbe.isInstance(c)) {
/* String isnot = shouldbe.toString().substring(shouldbe.toString().lastIndexOf(".")+1); //Cut all before last "." */
- throw new ClassCastException("Illegal constant '"+tostring(c)+"' at index '"+index+"'. '"+tostring(referrer)+"' expects a '"+shouldbe+"'.");
+ throw new ClassCastException("Illegal constant '" + tostring(c) + "' at index '" + index + "'. '"
+ + tostring(referrer) + "' expects a '" + shouldbe + "'.");
}
}
- ///////////////////////////////////////
+
+ // /////////////////////////////////////
// ClassFile structure (vmspec2 4.1) //
- ///////////////////////////////////////
- public void visitJavaClass(JavaClass obj){
+ // /////////////////////////////////////
+ @Override
+ public void visitJavaClass(JavaClass obj) {
Attribute[] atts = obj.getAttributes();
boolean foundSourceFile = false;
boolean foundInnerClasses = false;
@@ -381,521 +358,631 @@ public final class Pass2Verifier extends PassVerifier implements Constants{
// This is a costly check; existing verifiers don't do it!
boolean hasInnerClass = new InnerClassDetector(jc).innerClassReferenced();
- for (int i=0; i<atts.length; i++){
- if ((! (atts[i] instanceof SourceFile)) &&
- (! (atts[i] instanceof Deprecated)) &&
- (! (atts[i] instanceof InnerClasses)) &&
- (! (atts[i] instanceof Synthetic))){
- addMessage("Attribute '"+tostring(atts[i])+"' as an attribute of the ClassFile structure '"+tostring(obj)+"' is unknown and will therefore be ignored.");
+ for (int i = 0; i < atts.length; i++) {
+ if ((!(atts[i] instanceof SourceFile)) && (!(atts[i] instanceof Deprecated))
+ && (!(atts[i] instanceof InnerClasses)) && (!(atts[i] instanceof Synthetic))) {
+ addMessage("Attribute '" + tostring(atts[i]) + "' as an attribute of the ClassFile structure '" + tostring(obj)
+ + "' is unknown and will therefore be ignored.");
}
- if (atts[i] instanceof SourceFile){
- if (foundSourceFile == false) foundSourceFile = true;
- else throw new ClassConstraintException("A ClassFile structure (like '"+tostring(obj)+"') may have no more than one SourceFile attribute."); //vmspec2 4.7.7
+ if (atts[i] instanceof SourceFile) {
+ if (foundSourceFile == false)
+ foundSourceFile = true;
+ else
+ throw new ClassConstraintException("A ClassFile structure (like '" + tostring(obj)
+ + "') may have no more than one SourceFile attribute."); // vmspec2 4.7.7
}
- if (atts[i] instanceof InnerClasses){
- if (foundInnerClasses == false) foundInnerClasses = true;
- else{
- if (hasInnerClass){
- throw new ClassConstraintException("A Classfile structure (like '"+tostring(obj)+"') must have exactly one InnerClasses attribute if at least one Inner Class is referenced (which is the case). More than one InnerClasses attribute was found.");
+ if (atts[i] instanceof InnerClasses) {
+ if (foundInnerClasses == false)
+ foundInnerClasses = true;
+ else {
+ if (hasInnerClass) {
+ throw new ClassConstraintException(
+ "A Classfile structure (like '"
+ + tostring(obj)
+ + "') must have exactly one InnerClasses attribute if at least one Inner Class is referenced (which is the case). More than one InnerClasses attribute was found.");
}
}
- if (!hasInnerClass){
- addMessage("No referenced Inner Class found, but InnerClasses attribute '"+tostring(atts[i])+"' found. Strongly suggest removal of that attribute.");
+ if (!hasInnerClass) {
+ addMessage("No referenced Inner Class found, but InnerClasses attribute '" + tostring(atts[i])
+ + "' found. Strongly suggest removal of that attribute.");
}
}
}
- if (hasInnerClass && !foundInnerClasses){
- //throw new ClassConstraintException("A Classfile structure (like '"+tostring(obj)+"') must have exactly one InnerClasses attribute if at least one Inner Class is referenced (which is the case). No InnerClasses attribute was found.");
- //vmspec2, page 125 says it would be a constraint: but existing verifiers
- //don't check it and javac doesn't satisfy it when it comes to anonymous
- //inner classes
- addMessage("A Classfile structure (like '"+tostring(obj)+"') must have exactly one InnerClasses attribute if at least one Inner Class is referenced (which is the case). No InnerClasses attribute was found.");
+ if (hasInnerClass && !foundInnerClasses) {
+ // throw new
+ // ClassConstraintException("A Classfile structure (like '"+tostring(obj)+"') must have exactly one InnerClasses attribute if at least one Inner Class is referenced (which is the case). No InnerClasses attribute was found.");
+ // vmspec2, page 125 says it would be a constraint: but existing verifiers
+ // don't check it and javac doesn't satisfy it when it comes to anonymous
+ // inner classes
+ addMessage("A Classfile structure (like '"
+ + tostring(obj)
+ + "') must have exactly one InnerClasses attribute if at least one Inner Class is referenced (which is the case). No InnerClasses attribute was found.");
}
}
- /////////////////////////////
+
+ // ///////////////////////////
// CONSTANTS (vmspec2 4.4) //
- /////////////////////////////
- public void visitConstantClass(ConstantClass obj){
- if (obj.getTag() != Constants.CONSTANT_Class){
- throw new ClassConstraintException("Wrong constant tag in '"+tostring(obj)+"'.");
+ // ///////////////////////////
+ @Override
+ public void visitConstantClass(ConstantClass obj) {
+ if (obj.getTag() != Constants.CONSTANT_Class) {
+ throw new ClassConstraintException("Wrong constant tag in '" + tostring(obj) + "'.");
}
checkIndex(obj, obj.getNameIndex(), CONST_Utf8);
}
- public void visitConstantFieldref(ConstantFieldref obj){
- if (obj.getTag() != Constants.CONSTANT_Fieldref){
- throw new ClassConstraintException("Wrong constant tag in '"+tostring(obj)+"'.");
+
+ @Override
+ public void visitConstantFieldref(ConstantFieldref obj) {
+ if (obj.getTag() != Constants.CONSTANT_Fieldref) {
+ throw new ClassConstraintException("Wrong constant tag in '" + tostring(obj) + "'.");
}
checkIndex(obj, obj.getClassIndex(), CONST_Class);
checkIndex(obj, obj.getNameAndTypeIndex(), CONST_NameAndType);
}
- public void visitConstantMethodref(ConstantMethodref obj){
- if (obj.getTag() != Constants.CONSTANT_Methodref){
- throw new ClassConstraintException("Wrong constant tag in '"+tostring(obj)+"'.");
+
+ @Override
+ public void visitConstantMethodref(ConstantMethodref obj) {
+ if (obj.getTag() != Constants.CONSTANT_Methodref) {
+ throw new ClassConstraintException("Wrong constant tag in '" + tostring(obj) + "'.");
}
checkIndex(obj, obj.getClassIndex(), CONST_Class);
checkIndex(obj, obj.getNameAndTypeIndex(), CONST_NameAndType);
}
- public void visitConstantInterfaceMethodref(ConstantInterfaceMethodref obj){
- if (obj.getTag() != Constants.CONSTANT_InterfaceMethodref){
- throw new ClassConstraintException("Wrong constant tag in '"+tostring(obj)+"'.");
+
+ @Override
+ public void visitConstantInterfaceMethodref(ConstantInterfaceMethodref obj) {
+ if (obj.getTag() != Constants.CONSTANT_InterfaceMethodref) {
+ throw new ClassConstraintException("Wrong constant tag in '" + tostring(obj) + "'.");
}
checkIndex(obj, obj.getClassIndex(), CONST_Class);
checkIndex(obj, obj.getNameAndTypeIndex(), CONST_NameAndType);
}
- public void visitConstantString(ConstantString obj){
- if (obj.getTag() != Constants.CONSTANT_String){
- throw new ClassConstraintException("Wrong constant tag in '"+tostring(obj)+"'.");
+
+ @Override
+ public void visitConstantString(ConstantString obj) {
+ if (obj.getTag() != Constants.CONSTANT_String) {
+ throw new ClassConstraintException("Wrong constant tag in '" + tostring(obj) + "'.");
}
checkIndex(obj, obj.getStringIndex(), CONST_Utf8);
}
- public void visitConstantInteger(ConstantInteger obj){
- if (obj.getTag() != Constants.CONSTANT_Integer){
- throw new ClassConstraintException("Wrong constant tag in '"+tostring(obj)+"'.");
+
+ @Override
+ public void visitConstantInteger(ConstantInteger obj) {
+ if (obj.getTag() != Constants.CONSTANT_Integer) {
+ throw new ClassConstraintException("Wrong constant tag in '" + tostring(obj) + "'.");
}
// no indices to check
}
- public void visitConstantFloat(ConstantFloat obj){
- if (obj.getTag() != Constants.CONSTANT_Float){
- throw new ClassConstraintException("Wrong constant tag in '"+tostring(obj)+"'.");
+
+ @Override
+ public void visitConstantFloat(ConstantFloat obj) {
+ if (obj.getTag() != Constants.CONSTANT_Float) {
+ throw new ClassConstraintException("Wrong constant tag in '" + tostring(obj) + "'.");
}
- //no indices to check
+ // no indices to check
}
- public void visitConstantLong(ConstantLong obj){
- if (obj.getTag() != Constants.CONSTANT_Long){
- throw new ClassConstraintException("Wrong constant tag in '"+tostring(obj)+"'.");
+
+ @Override
+ public void visitConstantLong(ConstantLong obj) {
+ if (obj.getTag() != Constants.CONSTANT_Long) {
+ throw new ClassConstraintException("Wrong constant tag in '" + tostring(obj) + "'.");
}
- //no indices to check
+ // no indices to check
}
- public void visitConstantDouble(ConstantDouble obj){
- if (obj.getTag() != Constants.CONSTANT_Double){
- throw new ClassConstraintException("Wrong constant tag in '"+tostring(obj)+"'.");
+
+ @Override
+ public void visitConstantDouble(ConstantDouble obj) {
+ if (obj.getTag() != Constants.CONSTANT_Double) {
+ throw new ClassConstraintException("Wrong constant tag in '" + tostring(obj) + "'.");
}
- //no indices to check
+ // no indices to check
}
- public void visitConstantNameAndType(ConstantNameAndType obj){
- if (obj.getTag() != Constants.CONSTANT_NameAndType){
- throw new ClassConstraintException("Wrong constant tag in '"+tostring(obj)+"'.");
+
+ @Override
+ public void visitConstantNameAndType(ConstantNameAndType obj) {
+ if (obj.getTag() != Constants.CONSTANT_NameAndType) {
+ throw new ClassConstraintException("Wrong constant tag in '" + tostring(obj) + "'.");
}
checkIndex(obj, obj.getNameIndex(), CONST_Utf8);
- //checkIndex(obj, obj.getDescriptorIndex(), CONST_Utf8); //inconsistently named in BCEL, see below.
+ // checkIndex(obj, obj.getDescriptorIndex(), CONST_Utf8); //inconsistently named in BCEL, see below.
checkIndex(obj, obj.getSignatureIndex(), CONST_Utf8);
}
- public void visitConstantUtf8(ConstantUtf8 obj){
- if (obj.getTag() != Constants.CONSTANT_Utf8){
- throw new ClassConstraintException("Wrong constant tag in '"+tostring(obj)+"'.");
+
+ @Override
+ public void visitConstantUtf8(ConstantUtf8 obj) {
+ if (obj.getTag() != Constants.CONSTANT_Utf8) {
+ throw new ClassConstraintException("Wrong constant tag in '" + tostring(obj) + "'.");
}
- //no indices to check
+ // no indices to check
}
- //////////////////////////
+
+ // ////////////////////////
// FIELDS (vmspec2 4.5) //
- //////////////////////////
- public void visitField(Field obj){
-
- if (jc.isClass()){
- int maxone=0;
- if (obj.isPrivate()) maxone++;
- if (obj.isProtected()) maxone++;
- if (obj.isPublic()) maxone++;
- if (maxone > 1){
- throw new ClassConstraintException("Field '"+tostring(obj)+"' must only have at most one of its ACC_PRIVATE, ACC_PROTECTED, ACC_PUBLIC modifiers set.");
+ // ////////////////////////
+ @Override
+ public void visitField(Field obj) {
+
+ if (jc.isClass()) {
+ int maxone = 0;
+ if (obj.isPrivate())
+ maxone++;
+ if (obj.isProtected())
+ maxone++;
+ if (obj.isPublic())
+ maxone++;
+ if (maxone > 1) {
+ throw new ClassConstraintException("Field '" + tostring(obj)
+ + "' must only have at most one of its ACC_PRIVATE, ACC_PROTECTED, ACC_PUBLIC modifiers set.");
}
- if (obj.isFinal() && obj.isVolatile()){
- throw new ClassConstraintException("Field '"+tostring(obj)+"' must only have at most one of its ACC_FINAL, ACC_VOLATILE modifiers set.");
+ if (obj.isFinal() && obj.isVolatile()) {
+ throw new ClassConstraintException("Field '" + tostring(obj)
+ + "' must only have at most one of its ACC_FINAL, ACC_VOLATILE modifiers set.");
}
- }
- else{ // isInterface!
- if (!obj.isPublic()){
- throw new ClassConstraintException("Interface field '"+tostring(obj)+"' must have the ACC_PUBLIC modifier set but hasn't!");
+ } else { // isInterface!
+ if (!obj.isPublic()) {
+ throw new ClassConstraintException("Interface field '" + tostring(obj)
+ + "' must have the ACC_PUBLIC modifier set but hasn't!");
}
- if (!obj.isStatic()){
- throw new ClassConstraintException("Interface field '"+tostring(obj)+"' must have the ACC_STATIC modifier set but hasn't!");
+ if (!obj.isStatic()) {
+ throw new ClassConstraintException("Interface field '" + tostring(obj)
+ + "' must have the ACC_STATIC modifier set but hasn't!");
}
- if (!obj.isFinal()){
- throw new ClassConstraintException("Interface field '"+tostring(obj)+"' must have the ACC_FINAL modifier set but hasn't!");
+ if (!obj.isFinal()) {
+ throw new ClassConstraintException("Interface field '" + tostring(obj)
+ + "' must have the ACC_FINAL modifier set but hasn't!");
}
}
- if ((obj.getModifiers() & ~(ACC_PUBLIC|ACC_PRIVATE|ACC_PROTECTED|ACC_STATIC|ACC_FINAL|ACC_VOLATILE|ACC_TRANSIENT)) > 0){
- addMessage("Field '"+tostring(obj)+"' has access flag(s) other than ACC_PUBLIC, ACC_PRIVATE, ACC_PROTECTED, ACC_STATIC, ACC_FINAL, ACC_VOLATILE, ACC_TRANSIENT set (ignored).");
+ if ((obj.getModifiers() & ~(ACC_PUBLIC | ACC_PRIVATE | ACC_PROTECTED | ACC_STATIC | ACC_FINAL | ACC_VOLATILE | ACC_TRANSIENT)) > 0) {
+ addMessage("Field '"
+ + tostring(obj)
+ + "' has access flag(s) other than ACC_PUBLIC, ACC_PRIVATE, ACC_PROTECTED, ACC_STATIC, ACC_FINAL, ACC_VOLATILE, ACC_TRANSIENT set (ignored).");
}
checkIndex(obj, obj.getNameIndex(), CONST_Utf8);
String name = obj.getName();
- if (! validFieldName(name)){
- throw new ClassConstraintException("Field '"+tostring(obj)+"' has illegal name '"+obj.getName()+"'.");
+ if (!validFieldName(name)) {
+ throw new ClassConstraintException("Field '" + tostring(obj) + "' has illegal name '" + obj.getName() + "'.");
}
// A descriptor is often named signature in BCEL
checkIndex(obj, obj.getSignatureIndex(), CONST_Utf8);
- String sig = ((ConstantUtf8) (cp.getConstant(obj.getSignatureIndex()))).getBytes(); // Field or Method signature(=descriptor)
+ String sig = ((ConstantUtf8) (cp.getConstant(obj.getSignatureIndex()))).getValue(); // Field or Method
+ // signature(=descriptor)
- try{
- Type.getType(sig); /* Don't need the return value */
- }
- catch (ClassFormatError cfe){ // sometimes BCEL is a little harsh describing exceptional situations.
- throw new ClassConstraintException("Illegal descriptor (==signature) '"+sig+"' used by '"+tostring(obj)+"'.");
+ try {
+ Type.getType(sig); /* Don't need the return value */
+ } catch (ClassFormatError cfe) { // sometimes BCEL is a little harsh describing exceptional situations.
+ throw new ClassConstraintException("Illegal descriptor (==signature) '" + sig + "' used by '" + tostring(obj)
+ + "'.");
}
- String nameanddesc = (name+sig);
- if (field_names_and_desc.contains(nameanddesc)){
- throw new ClassConstraintException("No two fields (like '"+tostring(obj)+"') are allowed have same names and descriptors!");
+ String nameanddesc = (name + sig);
+ if (field_names_and_desc.contains(nameanddesc)) {
+ throw new ClassConstraintException("No two fields (like '" + tostring(obj)
+ + "') are allowed have same names and descriptors!");
}
- if (field_names.contains(name)){
- addMessage("More than one field of name '"+name+"' detected (but with different type descriptors). This is very unusual.");
+ if (field_names.contains(name)) {
+ addMessage("More than one field of name '" + name
+ + "' detected (but with different type descriptors). This is very unusual.");
}
field_names_and_desc.add(nameanddesc);
field_names.add(name);
Attribute[] atts = obj.getAttributes();
- for (int i=0; i<atts.length; i++){
- if ((! (atts[i] instanceof ConstantValue)) &&
- (! (atts[i] instanceof Synthetic)) &&
- (! (atts[i] instanceof Deprecated))){
- addMessage("Attribute '"+tostring(atts[i])+"' as an attribute of Field '"+tostring(obj)+"' is unknown and will therefore be ignored.");
+ for (int i = 0; i < atts.length; i++) {
+ if ((!(atts[i] instanceof ConstantValue)) && (!(atts[i] instanceof Synthetic))
+ && (!(atts[i] instanceof Deprecated))) {
+ addMessage("Attribute '" + tostring(atts[i]) + "' as an attribute of Field '" + tostring(obj)
+ + "' is unknown and will therefore be ignored.");
}
- if (! (atts[i] instanceof ConstantValue)){
- addMessage("Attribute '"+tostring(atts[i])+"' as an attribute of Field '"+tostring(obj)+"' is not a ConstantValue and is therefore only of use for debuggers and such.");
+ if (!(atts[i] instanceof ConstantValue)) {
+ addMessage("Attribute '" + tostring(atts[i]) + "' as an attribute of Field '" + tostring(obj)
+ + "' is not a ConstantValue and is therefore only of use for debuggers and such.");
}
}
}
- ///////////////////////////
+
+ // /////////////////////////
// METHODS (vmspec2 4.6) //
- ///////////////////////////
- public void visitMethod(Method obj){
+ // /////////////////////////
+ @Override
+ public void visitMethod(Method obj) {
checkIndex(obj, obj.getNameIndex(), CONST_Utf8);
String name = obj.getName();
- if (! validMethodName(name, true)){
- throw new ClassConstraintException("Method '"+tostring(obj)+"' has illegal name '"+name+"'.");
+ if (!validMethodName(name, true)) {
+ throw new ClassConstraintException("Method '" + tostring(obj) + "' has illegal name '" + name + "'.");
}
// A descriptor is often named signature in BCEL
checkIndex(obj, obj.getSignatureIndex(), CONST_Utf8);
- String sig = ((ConstantUtf8) (cp.getConstant(obj.getSignatureIndex()))).getBytes(); // Method's signature(=descriptor)
+ String sig = ((ConstantUtf8) (cp.getConstant(obj.getSignatureIndex()))).getValue(); // Method's signature(=descriptor)
Type t;
Type[] ts; // needed below the try block.
- try{
- t = Type.getReturnType(sig);
+ try {
+ t = Type.getReturnType(sig);
ts = Type.getArgumentTypes(sig);
- }
- catch (ClassFormatError cfe){
+ } catch (ClassFormatError cfe) {
// Well, BCEL sometimes is a little harsh describing exceptional situations.
- throw new ClassConstraintException("Illegal descriptor (==signature) '"+sig+"' used by Method '"+tostring(obj)+"'.");
+ throw new ClassConstraintException("Illegal descriptor (==signature) '" + sig + "' used by Method '"
+ + tostring(obj) + "'.");
}
// Check if referenced objects exist.
Type act = t;
- if (act instanceof ArrayType) act = ((ArrayType) act).getBasicType();
- if (act instanceof ObjectType){
- Verifier v = VerifierFactory.getVerifier( ((ObjectType) act).getClassName() );
+ if (act instanceof ArrayType)
+ act = ((ArrayType) act).getBasicType();
+ if (act instanceof ObjectType) {
+ Verifier v = VerifierFactory.getVerifier(((ObjectType) act).getClassName());
VerificationResult vr = v.doPass1();
if (vr != VerificationResult.VR_OK) {
- throw new ClassConstraintException("Method '"+tostring(obj)+"' has a return type that does not pass verification pass 1: '"+vr+"'.");
+ throw new ClassConstraintException("Method '" + tostring(obj)
+ + "' has a return type that does not pass verification pass 1: '" + vr + "'.");
}
}
- for (int i=0; i<ts.length; i++){
+ for (int i = 0; i < ts.length; i++) {
act = ts[i];
- if (act instanceof ArrayType) act = ((ArrayType) act).getBasicType();
- if (act instanceof ObjectType){
- Verifier v = VerifierFactory.getVerifier( ((ObjectType) act).getClassName() );
+ if (act instanceof ArrayType)
+ act = ((ArrayType) act).getBasicType();
+ if (act instanceof ObjectType) {
+ Verifier v = VerifierFactory.getVerifier(((ObjectType) act).getClassName());
VerificationResult vr = v.doPass1();
if (vr != VerificationResult.VR_OK) {
- throw new ClassConstraintException("Method '"+tostring(obj)+"' has an argument type that does not pass verification pass 1: '"+vr+"'.");
+ throw new ClassConstraintException("Method '" + tostring(obj)
+ + "' has an argument type that does not pass verification pass 1: '" + vr + "'.");
}
}
}
- // Nearly forgot this! Funny return values are allowed, but a non-empty arguments list makes a different method out of it!
- if (name.equals(STATIC_INITIALIZER_NAME) && (ts.length != 0)){
- throw new ClassConstraintException("Method '"+tostring(obj)+"' has illegal name '"+name+"'. It's name resembles the class or interface initialization method which it isn't because of its arguments (==descriptor).");
- }
-
- if (jc.isClass()){
- int maxone=0;
- if (obj.isPrivate()) maxone++;
- if (obj.isProtected()) maxone++;
- if (obj.isPublic()) maxone++;
- if (maxone > 1){
- throw new ClassConstraintException("Method '"+tostring(obj)+"' must only have at most one of its ACC_PRIVATE, ACC_PROTECTED, ACC_PUBLIC modifiers set.");
+ // Nearly forgot this! Funny return values are allowed, but a non-empty arguments list makes a different method out of
+ // it!
+ if (name.equals(STATIC_INITIALIZER_NAME) && (ts.length != 0)) {
+ throw new ClassConstraintException(
+ "Method '"
+ + tostring(obj)
+ + "' has illegal name '"
+ + name
+ + "'. It's name resembles the class or interface initialization method which it isn't because of its arguments (==descriptor).");
+ }
+
+ if (jc.isClass()) {
+ int maxone = 0;
+ if (obj.isPrivate())
+ maxone++;
+ if (obj.isProtected())
+ maxone++;
+ if (obj.isPublic())
+ maxone++;
+ if (maxone > 1) {
+ throw new ClassConstraintException("Method '" + tostring(obj)
+ + "' must only have at most one of its ACC_PRIVATE, ACC_PROTECTED, ACC_PUBLIC modifiers set.");
}
- if (obj.isAbstract()){
- if (obj.isFinal()) throw new ClassConstraintException("Abstract method '"+tostring(obj)+"' must not have the ACC_FINAL modifier set.");
- if (obj.isNative()) throw new ClassConstraintException("Abstract method '"+tostring(obj)+"' must not have the ACC_NATIVE modifier set.");
- if (obj.isPrivate()) throw new ClassConstraintException("Abstract method '"+tostring(obj)+"' must not have the ACC_PRIVATE modifier set.");
- if (obj.isStatic()) throw new ClassConstraintException("Abstract method '"+tostring(obj)+"' must not have the ACC_STATIC modifier set.");
- if (obj.isStrictfp()) throw new ClassConstraintException("Abstract method '"+tostring(obj)+"' must not have the ACC_STRICT modifier set.");
- if (obj.isSynchronized()) throw new ClassConstraintException("Abstract method '"+tostring(obj)+"' must not have the ACC_SYNCHRONIZED modifier set.");
+ if (obj.isAbstract()) {
+ if (obj.isFinal())
+ throw new ClassConstraintException("Abstract method '" + tostring(obj)
+ + "' must not have the ACC_FINAL modifier set.");
+ if (obj.isNative())
+ throw new ClassConstraintException("Abstract method '" + tostring(obj)
+ + "' must not have the ACC_NATIVE modifier set.");
+ if (obj.isPrivate())
+ throw new ClassConstraintException("Abstract method '" + tostring(obj)
+ + "' must not have the ACC_PRIVATE modifier set.");
+ if (obj.isStatic())
+ throw new ClassConstraintException("Abstract method '" + tostring(obj)
+ + "' must not have the ACC_STATIC modifier set.");
+ if (obj.isStrictfp())
+ throw new ClassConstraintException("Abstract method '" + tostring(obj)
+ + "' must not have the ACC_STRICT modifier set.");
+ if (obj.isSynchronized())
+ throw new ClassConstraintException("Abstract method '" + tostring(obj)
+ + "' must not have the ACC_SYNCHRONIZED modifier set.");
}
- }
- else{ // isInterface!
- if (!name.equals(STATIC_INITIALIZER_NAME)){//vmspec2, p.116, 2nd paragraph
- if (!obj.isPublic()){
- throw new ClassConstraintException("Interface method '"+tostring(obj)+"' must have the ACC_PUBLIC modifier set but hasn't!");
+ } else { // isInterface!
+ if (!name.equals(STATIC_INITIALIZER_NAME)) {// vmspec2, p.116, 2nd paragraph
+ if (!obj.isPublic()) {
+ throw new ClassConstraintException("Interface method '" + tostring(obj)
+ + "' must have the ACC_PUBLIC modifier set but hasn't!");
}
- if (!obj.isAbstract()){
- throw new ClassConstraintException("Interface method '"+tostring(obj)+"' must have the ACC_STATIC modifier set but hasn't!");
+ if (!obj.isAbstract()) {
+ throw new ClassConstraintException("Interface method '" + tostring(obj)
+ + "' must have the ACC_STATIC modifier set but hasn't!");
}
- if ( obj.isPrivate() ||
- obj.isProtected() ||
- obj.isStatic() ||
- obj.isFinal() ||
- obj.isSynchronized() ||
- obj.isNative() ||
- obj.isStrictfp() ){
- throw new ClassConstraintException("Interface method '"+tostring(obj)+"' must not have any of the ACC_PRIVATE, ACC_PROTECTED, ACC_STATIC, ACC_FINAL, ACC_SYNCHRONIZED, ACC_NATIVE, ACC_ABSTRACT, ACC_STRICT modifiers set.");
+ if (obj.isPrivate() || obj.isProtected() || obj.isStatic() || obj.isFinal() || obj.isSynchronized()
+ || obj.isNative() || obj.isStrictfp()) {
+ throw new ClassConstraintException(
+ "Interface method '"
+ + tostring(obj)
+ + "' must not have any of the ACC_PRIVATE, ACC_PROTECTED, ACC_STATIC, ACC_FINAL, ACC_SYNCHRONIZED, ACC_NATIVE, ACC_ABSTRACT, ACC_STRICT modifiers set.");
}
}
}
// A specific instance initialization method... (vmspec2,Page 116).
- if (name.equals(CONSTRUCTOR_NAME)){
- //..may have at most one of ACC_PRIVATE, ACC_PROTECTED, ACC_PUBLIC set: is checked above.
- //..may also have ACC_STRICT set, but none of the other flags in table 4.5 (vmspec2, page 115)
- if ( obj.isStatic() ||
- obj.isFinal() ||
- obj.isSynchronized() ||
- obj.isNative() ||
- obj.isAbstract() ){
- throw new ClassConstraintException("Instance initialization method '"+tostring(obj)+"' must not have any of the ACC_STATIC, ACC_FINAL, ACC_SYNCHRONIZED, ACC_NATIVE, ACC_ABSTRACT modifiers set.");
+ if (name.equals(CONSTRUCTOR_NAME)) {
+ // ..may have at most one of ACC_PRIVATE, ACC_PROTECTED, ACC_PUBLIC set: is checked above.
+ // ..may also have ACC_STRICT set, but none of the other flags in table 4.5 (vmspec2, page 115)
+ if (obj.isStatic() || obj.isFinal() || obj.isSynchronized() || obj.isNative() || obj.isAbstract()) {
+ throw new ClassConstraintException(
+ "Instance initialization method '"
+ + tostring(obj)
+ + "' must not have any of the ACC_STATIC, ACC_FINAL, ACC_SYNCHRONIZED, ACC_NATIVE, ACC_ABSTRACT modifiers set.");
}
}
// Class and interface initialization methods...
- if (name.equals(STATIC_INITIALIZER_NAME)){
- if ((obj.getModifiers() & (~ACC_STRICT)) > 0){
- addMessage("Class or interface initialization method '"+tostring(obj)+"' has superfluous access modifier(s) set: everything but ACC_STRICT is ignored.");
+ if (name.equals(STATIC_INITIALIZER_NAME)) {
+ if ((obj.getModifiers() & (~ACC_STRICT)) > 0) {
+ addMessage("Class or interface initialization method '" + tostring(obj)
+ + "' has superfluous access modifier(s) set: everything but ACC_STRICT is ignored.");
}
- if (obj.isAbstract()){
- throw new ClassConstraintException("Class or interface initialization method '"+tostring(obj)+"' must not be abstract. This contradicts the Java Language Specification, Second Edition (which omits this constraint) but is common practice of existing verifiers.");
+ if (obj.isAbstract()) {
+ throw new ClassConstraintException(
+ "Class or interface initialization method '"
+ + tostring(obj)
+ + "' must not be abstract. This contradicts the Java Language Specification, Second Edition (which omits this constraint) but is common practice of existing verifiers.");
}
}
- if ((obj.getModifiers() & ~(ACC_PUBLIC|ACC_PRIVATE|ACC_PROTECTED|ACC_STATIC|ACC_FINAL|ACC_SYNCHRONIZED|ACC_NATIVE|ACC_ABSTRACT|ACC_STRICT)) > 0){
- addMessage("Method '"+tostring(obj)+"' has access flag(s) other than ACC_PUBLIC, ACC_PRIVATE, ACC_PROTECTED, ACC_STATIC, ACC_FINAL, ACC_SYNCHRONIZED, ACC_NATIVE, ACC_ABSTRACT, ACC_STRICT set (ignored).");
+ if ((obj.getModifiers() & ~(ACC_PUBLIC | ACC_PRIVATE | ACC_PROTECTED | ACC_STATIC | ACC_FINAL | ACC_SYNCHRONIZED
+ | ACC_NATIVE | ACC_ABSTRACT | ACC_STRICT)) > 0) {
+ addMessage("Method '"
+ + tostring(obj)
+ + "' has access flag(s) other than ACC_PUBLIC, ACC_PRIVATE, ACC_PROTECTED, ACC_STATIC, ACC_FINAL, ACC_SYNCHRONIZED, ACC_NATIVE, ACC_ABSTRACT, ACC_STRICT set (ignored).");
}
- String nameanddesc = (name+sig);
- if (method_names_and_desc.contains(nameanddesc)){
- throw new ClassConstraintException("No two methods (like '"+tostring(obj)+"') are allowed have same names and desciptors!");
+ String nameanddesc = (name + sig);
+ if (method_names_and_desc.contains(nameanddesc)) {
+ throw new ClassConstraintException("No two methods (like '" + tostring(obj)
+ + "') are allowed have same names and desciptors!");
}
method_names_and_desc.add(nameanddesc);
Attribute[] atts = obj.getAttributes();
int num_code_atts = 0;
- for (int i=0; i<atts.length; i++){
- if ((! (atts[i] instanceof Code)) &&
- (! (atts[i] instanceof ExceptionTable)) &&
- (! (atts[i] instanceof Synthetic)) &&
- (! (atts[i] instanceof Deprecated))){
- addMessage("Attribute '"+tostring(atts[i])+"' as an attribute of Method '"+tostring(obj)+"' is unknown and will therefore be ignored.");
+ for (int i = 0; i < atts.length; i++) {
+ if ((!(atts[i] instanceof Code)) && (!(atts[i] instanceof ExceptionTable)) && (!(atts[i] instanceof Synthetic))
+ && (!(atts[i] instanceof Deprecated))) {
+ addMessage("Attribute '" + tostring(atts[i]) + "' as an attribute of Method '" + tostring(obj)
+ + "' is unknown and will therefore be ignored.");
}
- if ((! (atts[i] instanceof Code)) &&
- (! (atts[i] instanceof ExceptionTable))){
- addMessage("Attribute '"+tostring(atts[i])+"' as an attribute of Method '"+tostring(obj)+"' is neither Code nor Exceptions and is therefore only of use for debuggers and such.");
+ if ((!(atts[i] instanceof Code)) && (!(atts[i] instanceof ExceptionTable))) {
+ addMessage("Attribute '" + tostring(atts[i]) + "' as an attribute of Method '" + tostring(obj)
+ + "' is neither Code nor Exceptions and is therefore only of use for debuggers and such.");
}
- if ((atts[i] instanceof Code) && (obj.isNative() || obj.isAbstract())){
- throw new ClassConstraintException("Native or abstract methods like '"+tostring(obj)+"' must not have a Code attribute like '"+tostring(atts[i])+"'."); //vmspec2 page120, 4.7.3
+ if ((atts[i] instanceof Code) && (obj.isNative() || obj.isAbstract())) {
+ throw new ClassConstraintException("Native or abstract methods like '" + tostring(obj)
+ + "' must not have a Code attribute like '" + tostring(atts[i]) + "'."); // vmspec2 page120, 4.7.3
}
- if (atts[i] instanceof Code) num_code_atts++;
+ if (atts[i] instanceof Code)
+ num_code_atts++;
}
- if ( !obj.isNative() && !obj.isAbstract() && num_code_atts != 1){
- throw new ClassConstraintException("Non-native, non-abstract methods like '"+tostring(obj)+"' must have exactly one Code attribute (found: "+num_code_atts+").");
+ if (!obj.isNative() && !obj.isAbstract() && num_code_atts != 1) {
+ throw new ClassConstraintException("Non-native, non-abstract methods like '" + tostring(obj)
+ + "' must have exactly one Code attribute (found: " + num_code_atts + ").");
}
}
- ///////////////////////////////////////////////////////
+
+ // /////////////////////////////////////////////////////
// ClassFile-structure-ATTRIBUTES (vmspec2 4.1, 4.7) //
- ///////////////////////////////////////////////////////
- public void visitSourceFile(SourceFile obj){//vmspec2 4.7.7
+ // /////////////////////////////////////////////////////
+ public void visitSourceFile(SourceFile obj) {// vmspec2 4.7.7
// zero or one SourceFile attr per ClassFile: see visitJavaClass()
checkIndex(obj, obj.getNameIndex(), CONST_Utf8);
- String name = ((ConstantUtf8) cp.getConstant(obj.getNameIndex())).getBytes();
- if (! name.equals("SourceFile")){
- throw new ClassConstraintException("The SourceFile attribute '"+tostring(obj)+"' is not correctly named 'SourceFile' but '"+name+"'.");
+ String name = ((ConstantUtf8) cp.getConstant(obj.getNameIndex())).getValue();
+ if (!name.equals("SourceFile")) {
+ throw new ClassConstraintException("The SourceFile attribute '" + tostring(obj)
+ + "' is not correctly named 'SourceFile' but '" + name + "'.");
}
checkIndex(obj, obj.getSourceFileIndex(), CONST_Utf8);
- String sourcefilename = ((ConstantUtf8) cp.getConstant(obj.getSourceFileIndex())).getBytes(); //==obj.getSourceFileName() ?
+ String sourcefilename = ((ConstantUtf8) cp.getConstant(obj.getSourceFileIndex())).getValue(); // ==obj.getSourceFileName()
+ // ?
String sourcefilenamelc = sourcefilename.toLowerCase();
- if ( (sourcefilename.indexOf('/') != -1) ||
- (sourcefilename.indexOf('\\') != -1) ||
- (sourcefilename.indexOf(':') != -1) ||
- (sourcefilenamelc.lastIndexOf(".java") == -1) ){
- addMessage("SourceFile attribute '"+tostring(obj)+"' has a funny name: remember not to confuse certain parsers working on javap's output. Also, this name ('"+sourcefilename+"') is considered an unqualified (simple) file name only.");
+ if ((sourcefilename.indexOf('/') != -1) || (sourcefilename.indexOf('\\') != -1) || (sourcefilename.indexOf(':') != -1)
+ || (sourcefilenamelc.lastIndexOf(".java") == -1)) {
+ addMessage("SourceFile attribute '"
+ + tostring(obj)
+ + "' has a funny name: remember not to confuse certain parsers working on javap's output. Also, this name ('"
+ + sourcefilename + "') is considered an unqualified (simple) file name only.");
}
}
- public void visitDeprecated(Deprecated obj){//vmspec2 4.7.10
+
+ public void visitDeprecated(Deprecated obj) {// vmspec2 4.7.10
checkIndex(obj, obj.getNameIndex(), CONST_Utf8);
- String name = ((ConstantUtf8) cp.getConstant(obj.getNameIndex())).getBytes();
- if (! name.equals("Deprecated")){
- throw new ClassConstraintException("The Deprecated attribute '"+tostring(obj)+"' is not correctly named 'Deprecated' but '"+name+"'.");
+ String name = ((ConstantUtf8) cp.getConstant(obj.getNameIndex())).getValue();
+ if (!name.equals("Deprecated")) {
+ throw new ClassConstraintException("The Deprecated attribute '" + tostring(obj)
+ + "' is not correctly named 'Deprecated' but '" + name + "'.");
}
}
- public void visitSynthetic(Synthetic obj){//vmspec2 4.7.6
+
+ public void visitSynthetic(Synthetic obj) {// vmspec2 4.7.6
checkIndex(obj, obj.getNameIndex(), CONST_Utf8);
- String name = ((ConstantUtf8) cp.getConstant(obj.getNameIndex())).getBytes();
- if (! name.equals("Synthetic")){
- throw new ClassConstraintException("The Synthetic attribute '"+tostring(obj)+"' is not correctly named 'Synthetic' but '"+name+"'.");
+ String name = ((ConstantUtf8) cp.getConstant(obj.getNameIndex())).getValue();
+ if (!name.equals("Synthetic")) {
+ throw new ClassConstraintException("The Synthetic attribute '" + tostring(obj)
+ + "' is not correctly named 'Synthetic' but '" + name + "'.");
}
}
- public void visitInnerClasses(InnerClasses obj){//vmspec2 4.7.5
+
+ public void visitInnerClasses(InnerClasses obj) {// vmspec2 4.7.5
// exactly one InnerClasses attr per ClassFile if some inner class is refernced: see visitJavaClass()
checkIndex(obj, obj.getNameIndex(), CONST_Utf8);
- String name = ((ConstantUtf8) cp.getConstant(obj.getNameIndex())).getBytes();
- if (! name.equals("InnerClasses")){
- throw new ClassConstraintException("The InnerClasses attribute '"+tostring(obj)+"' is not correctly named 'InnerClasses' but '"+name+"'.");
+ String name = ((ConstantUtf8) cp.getConstant(obj.getNameIndex())).getValue();
+ if (!name.equals("InnerClasses")) {
+ throw new ClassConstraintException("The InnerClasses attribute '" + tostring(obj)
+ + "' is not correctly named 'InnerClasses' but '" + name + "'.");
}
InnerClass[] ics = obj.getInnerClasses();
- for (int i=0; i<ics.length; i++){
+ for (int i = 0; i < ics.length; i++) {
checkIndex(obj, ics[i].getInnerClassIndex(), CONST_Class);
int outer_idx = ics[i].getOuterClassIndex();
- if (outer_idx != 0){
+ if (outer_idx != 0) {
checkIndex(obj, outer_idx, CONST_Class);
}
int innername_idx = ics[i].getInnerNameIndex();
- if (innername_idx != 0){
+ if (innername_idx != 0) {
checkIndex(obj, innername_idx, CONST_Utf8);
}
int acc = ics[i].getInnerAccessFlags();
- acc = acc & (~ (ACC_PUBLIC | ACC_PRIVATE | ACC_PROTECTED | ACC_STATIC | ACC_FINAL | ACC_INTERFACE | ACC_ABSTRACT));
- if (acc != 0){
- addMessage("Unknown access flag for inner class '"+tostring(ics[i])+"' set (InnerClasses attribute '"+tostring(obj)+"').");
+ acc = acc & (~(ACC_PUBLIC | ACC_PRIVATE | ACC_PROTECTED | ACC_STATIC | ACC_FINAL | ACC_INTERFACE | ACC_ABSTRACT));
+ if (acc != 0) {
+ addMessage("Unknown access flag for inner class '" + tostring(ics[i]) + "' set (InnerClasses attribute '"
+ + tostring(obj) + "').");
}
}
// Semantical consistency is not yet checked by Sun, see vmspec2 4.7.5.
// [marked TODO in JustIce]
}
- ////////////////////////////////////////////////////////
+
+ // //////////////////////////////////////////////////////
// field_info-structure-ATTRIBUTES (vmspec2 4.5, 4.7) //
- ////////////////////////////////////////////////////////
- public void visitConstantValue(ConstantValue obj){//vmspec2 4.7.2
+ // //////////////////////////////////////////////////////
+ public void visitConstantValue(ConstantValue obj) {// vmspec2 4.7.2
// Despite its name, this really is an Attribute,
// not a constant!
checkIndex(obj, obj.getNameIndex(), CONST_Utf8);
- String name = ((ConstantUtf8) cp.getConstant(obj.getNameIndex())).getBytes();
- if (! name.equals("ConstantValue")){
- throw new ClassConstraintException("The ConstantValue attribute '"+tostring(obj)+"' is not correctly named 'ConstantValue' but '"+name+"'.");
+ String name = ((ConstantUtf8) cp.getConstant(obj.getNameIndex())).getValue();
+ if (!name.equals("ConstantValue")) {
+ throw new ClassConstraintException("The ConstantValue attribute '" + tostring(obj)
+ + "' is not correctly named 'ConstantValue' but '" + name + "'.");
}
Object pred = carrier.predecessor();
- if (pred instanceof Field){ //ConstantValue attributes are quite senseless if the predecessor is not a field.
+ if (pred instanceof Field) { // ConstantValue attributes are quite senseless if the predecessor is not a field.
Field f = (Field) pred;
// Field constraints have been checked before -- so we are safe using their type information.
- Type field_type = Type.getType(((ConstantUtf8) (cp.getConstant(f.getSignatureIndex()))).getBytes());
+ Type field_type = Type.getType(((ConstantUtf8) (cp.getConstant(f.getSignatureIndex()))).getValue());
int index = obj.getConstantValueIndex();
- if ((index < 0) || (index >= cplen)){
- throw new ClassConstraintException("Invalid index '"+index+"' used by '"+tostring(obj)+"'.");
+ if ((index < 0) || (index >= cplen)) {
+ throw new ClassConstraintException("Invalid index '" + index + "' used by '" + tostring(obj) + "'.");
}
Constant c = cp.getConstant(index);
- if (CONST_Long.isInstance(c) && field_type.equals(Type.LONG)){
+ if (CONST_Long.isInstance(c) && field_type.equals(Type.LONG)) {
return;
}
- if (CONST_Float.isInstance(c) && field_type.equals(Type.FLOAT)){
+ if (CONST_Float.isInstance(c) && field_type.equals(Type.FLOAT)) {
return;
}
- if (CONST_Double.isInstance(c) && field_type.equals(Type.DOUBLE)){
+ if (CONST_Double.isInstance(c) && field_type.equals(Type.DOUBLE)) {
return;
}
- if (CONST_Integer.isInstance(c) && (field_type.equals(Type.INT) || field_type.equals(Type.SHORT) || field_type.equals(Type.CHAR) || field_type.equals(Type.BYTE) || field_type.equals(Type.BOOLEAN))){
+ if (CONST_Integer.isInstance(c)
+ && (field_type.equals(Type.INT) || field_type.equals(Type.SHORT) || field_type.equals(Type.CHAR)
+ || field_type.equals(Type.BYTE) || field_type.equals(Type.BOOLEAN))) {
return;
}
- if (CONST_String.isInstance(c) && field_type.equals(Type.STRING)){
+ if (CONST_String.isInstance(c) && field_type.equals(Type.STRING)) {
return;
}
- throw new ClassConstraintException("Illegal type of ConstantValue '"+obj+"' embedding Constant '"+c+"'. It is referenced by field '"+tostring(f)+"' expecting a different type: '"+field_type+"'.");
+ throw new ClassConstraintException("Illegal type of ConstantValue '" + obj + "' embedding Constant '" + c
+ + "'. It is referenced by field '" + tostring(f) + "' expecting a different type: '" + field_type + "'.");
}
}
+
// SYNTHETIC: see above
// DEPRECATED: see above
- /////////////////////////////////////////////////////////
+ // ///////////////////////////////////////////////////////
// method_info-structure-ATTRIBUTES (vmspec2 4.6, 4.7) //
- /////////////////////////////////////////////////////////
- public void visitCode(Code obj){//vmspec2 4.7.3
+ // ///////////////////////////////////////////////////////
+ public void visitCode(Code obj) {// vmspec2 4.7.3
// No code attribute allowed for native or abstract methods: see visitMethod(Method).
// Code array constraints are checked in Pass3 (3a and 3b).
checkIndex(obj, obj.getNameIndex(), CONST_Utf8);
- String name = ((ConstantUtf8) cp.getConstant(obj.getNameIndex())).getBytes();
- if (! name.equals("Code")){
- throw new ClassConstraintException("The Code attribute '"+tostring(obj)+"' is not correctly named 'Code' but '"+name+"'.");
+ String name = ((ConstantUtf8) cp.getConstant(obj.getNameIndex())).getValue();
+ if (!name.equals("Code")) {
+ throw new ClassConstraintException("The Code attribute '" + tostring(obj) + "' is not correctly named 'Code' but '"
+ + name + "'.");
}
Method m = null; // satisfy compiler
- if (!(carrier.predecessor() instanceof Method)){
- addMessage("Code attribute '"+tostring(obj)+"' is not declared in a method_info structure but in '"+carrier.predecessor()+"'. Ignored.");
+ if (!(carrier.predecessor() instanceof Method)) {
+ addMessage("Code attribute '" + tostring(obj) + "' is not declared in a method_info structure but in '"
+ + carrier.predecessor() + "'. Ignored.");
return;
- }
- else{
- m = (Method) carrier.predecessor(); // we can assume this method was visited before;
- // i.e. the data consistency was verified.
+ } else {
+ m = (Method) carrier.predecessor(); // we can assume this method was visited before;
+ // i.e. the data consistency was verified.
}
- if (obj.getCode().length == 0){
- throw new ClassConstraintException("Code array of Code attribute '"+tostring(obj)+"' (method '"+m+"') must not be empty.");
+ if (obj.getCode().length == 0) {
+ throw new ClassConstraintException("Code array of Code attribute '" + tostring(obj) + "' (method '" + m
+ + "') must not be empty.");
}
- //In JustIce, the check for correct offsets into the code array is delayed to Pass 3a.
+ // In JustIce, the check for correct offsets into the code array is delayed to Pass 3a.
CodeException[] exc_table = obj.getExceptionTable();
- for (int i=0; i<exc_table.length; i++){
+ for (int i = 0; i < exc_table.length; i++) {
int exc_index = exc_table[i].getCatchType();
- if (exc_index != 0){ // if 0, it catches all Throwables
+ if (exc_index != 0) { // if 0, it catches all Throwables
checkIndex(obj, exc_index, CONST_Class);
ConstantClass cc = (ConstantClass) (cp.getConstant(exc_index));
- checkIndex(cc, cc.getNameIndex(), CONST_Utf8); // cannot be sure this ConstantClass has already been visited (checked)!
- String cname = ((ConstantUtf8) cp.getConstant(cc.getNameIndex())).getBytes().replace('/','.');
+ checkIndex(cc, cc.getNameIndex(), CONST_Utf8); // cannot be sure this ConstantClass has already been visited
+ // (checked)!
+ String cname = ((ConstantUtf8) cp.getConstant(cc.getNameIndex())).getValue().replace('/', '.');
Verifier v = VerifierFactory.getVerifier(cname);
VerificationResult vr = v.doPass1();
- if (vr != VerificationResult.VR_OK){
- throw new ClassConstraintException("Code attribute '"+tostring(obj)+"' (method '"+m+"') has an exception_table entry '"+tostring(exc_table[i])+"' that references '"+cname+"' as an Exception but it does not pass verification pass 1: "+vr);
- }
- else{
+ if (vr != VerificationResult.VR_OK) {
+ throw new ClassConstraintException("Code attribute '" + tostring(obj) + "' (method '" + m
+ + "') has an exception_table entry '" + tostring(exc_table[i]) + "' that references '" + cname
+ + "' as an Exception but it does not pass verification pass 1: " + vr);
+ } else {
// We cannot safely trust any other "instanceof" mechanism. We need to transitively verify
// the ancestor hierarchy.
JavaClass e = Repository.lookupClass(cname);
JavaClass t = Repository.lookupClass(Type.THROWABLE.getClassName());
JavaClass o = Repository.lookupClass(Type.OBJECT.getClassName());
- while (e != o){
- if (e == t) break; // It's a subclass of Throwable, OKAY, leave.
+ while (e != o) {
+ if (e == t)
+ break; // It's a subclass of Throwable, OKAY, leave.
v = VerifierFactory.getVerifier(e.getSuperclassName());
vr = v.doPass1();
- if (vr != VerificationResult.VR_OK){
- throw new ClassConstraintException("Code attribute '"+tostring(obj)+"' (method '"+m+"') has an exception_table entry '"+tostring(exc_table[i])+"' that references '"+cname+"' as an Exception but '"+e.getSuperclassName()+"' in the ancestor hierachy does not pass verification pass 1: "+vr);
- }
- else{
+ if (vr != VerificationResult.VR_OK) {
+ throw new ClassConstraintException("Code attribute '" + tostring(obj) + "' (method '" + m
+ + "') has an exception_table entry '" + tostring(exc_table[i]) + "' that references '"
+ + cname + "' as an Exception but '" + e.getSuperclassName()
+ + "' in the ancestor hierachy does not pass verification pass 1: " + vr);
+ } else {
e = Repository.lookupClass(e.getSuperclassName());
}
}
- if (e != t) throw new ClassConstraintException("Code attribute '"+tostring(obj)+"' (method '"+m+"') has an exception_table entry '"+tostring(exc_table[i])+"' that references '"+cname+"' as an Exception but it is not a subclass of '"+t.getClassName()+"'.");
+ if (e != t)
+ throw new ClassConstraintException("Code attribute '" + tostring(obj) + "' (method '" + m
+ + "') has an exception_table entry '" + tostring(exc_table[i]) + "' that references '" + cname
+ + "' as an Exception but it is not a subclass of '" + t.getClassName() + "'.");
}
}
}
@@ -905,190 +992,226 @@ public final class Pass2Verifier extends PassVerifier implements Constants{
// TODO: rework it.
int method_number = -1;
Method[] ms = Repository.lookupClass(myOwner.getClassName()).getMethods();
- for (int mn=0; mn<ms.length; mn++){
- if (m == ms[mn]){
+ for (int mn = 0; mn < ms.length; mn++) {
+ if (m == ms[mn]) {
method_number = mn;
break;
}
}
- if (method_number < 0){ // Mmmmh. Can we be sure BCEL does not sometimes instantiate new objects?
- throw new AssertionViolatedException("Could not find a known BCEL Method object in the corresponding BCEL JavaClass object.");
+ if (method_number < 0) { // Mmmmh. Can we be sure BCEL does not sometimes instantiate new objects?
+ throw new AssertionViolatedException(
+ "Could not find a known BCEL Method object in the corresponding BCEL JavaClass object.");
}
localVariablesInfos[method_number] = new LocalVariablesInfo(obj.getMaxLocals());
int num_of_lvt_attribs = 0;
// Now iterate through the attributes the Code attribute has.
Attribute[] atts = obj.getAttributes();
- for (int a=0; a<atts.length; a++){
- if ((! (atts[a] instanceof LineNumberTable)) &&
- (! (atts[a] instanceof LocalVariableTable))){
- addMessage("Attribute '"+tostring(atts[a])+"' as an attribute of Code attribute '"+tostring(obj)+"' (method '"+m+"') is unknown and will therefore be ignored.");
- }
- else{// LineNumberTable or LocalVariableTable
- addMessage("Attribute '"+tostring(atts[a])+"' as an attribute of Code attribute '"+tostring(obj)+"' (method '"+m+"') will effectively be ignored and is only useful for debuggers and such.");
+ for (int a = 0; a < atts.length; a++) {
+ if ((!(atts[a] instanceof LineNumberTable)) && (!(atts[a] instanceof LocalVariableTable))) {
+ addMessage("Attribute '" + tostring(atts[a]) + "' as an attribute of Code attribute '" + tostring(obj)
+ + "' (method '" + m + "') is unknown and will therefore be ignored.");
+ } else {// LineNumberTable or LocalVariableTable
+ addMessage("Attribute '" + tostring(atts[a]) + "' as an attribute of Code attribute '" + tostring(obj)
+ + "' (method '" + m + "') will effectively be ignored and is only useful for debuggers and such.");
}
- //LocalVariableTable check (partially delayed to Pass3a).
- //Here because its easier to collect the information of the
- //(possibly more than one) LocalVariableTables belonging to
- //one certain Code attribute.
- if (atts[a] instanceof LocalVariableTable){ // checks conforming to vmspec2 4.7.9
+ // LocalVariableTable check (partially delayed to Pass3a).
+ // Here because its easier to collect the information of the
+ // (possibly more than one) LocalVariableTables belonging to
+ // one certain Code attribute.
+ if (atts[a] instanceof LocalVariableTable) { // checks conforming to vmspec2 4.7.9
LocalVariableTable lvt = (LocalVariableTable) atts[a];
checkIndex(lvt, lvt.getNameIndex(), CONST_Utf8);
- String lvtname = ((ConstantUtf8) cp.getConstant(lvt.getNameIndex())).getBytes();
- if (! lvtname.equals("LocalVariableTable")){
- throw new ClassConstraintException("The LocalVariableTable attribute '"+tostring(lvt)+"' is not correctly named 'LocalVariableTable' but '"+lvtname+"'.");
+ String lvtname = ((ConstantUtf8) cp.getConstant(lvt.getNameIndex())).getValue();
+ if (!lvtname.equals("LocalVariableTable")) {
+ throw new ClassConstraintException("The LocalVariableTable attribute '" + tostring(lvt)
+ + "' is not correctly named 'LocalVariableTable' but '" + lvtname + "'.");
}
Code code = obj;
- //In JustIce, the check for correct offsets into the code array is delayed to Pass 3a.
+ // In JustIce, the check for correct offsets into the code array is delayed to Pass 3a.
LocalVariable[] localvariables = lvt.getLocalVariableTable();
- for (int i=0; i<localvariables.length; i++){
+ for (int i = 0; i < localvariables.length; i++) {
checkIndex(lvt, localvariables[i].getNameIndex(), CONST_Utf8);
- String localname = ((ConstantUtf8) cp.getConstant(localvariables[i].getNameIndex())).getBytes();
- if (!validJavaIdentifier(localname)){
- throw new ClassConstraintException("LocalVariableTable '"+tostring(lvt)+"' references a local variable by the name '"+localname+"' which is not a legal Java simple name.");
+ String localname = ((ConstantUtf8) cp.getConstant(localvariables[i].getNameIndex())).getValue();
+ if (!validJavaIdentifier(localname)) {
+ throw new ClassConstraintException("LocalVariableTable '" + tostring(lvt)
+ + "' references a local variable by the name '" + localname
+ + "' which is not a legal Java simple name.");
}
checkIndex(lvt, localvariables[i].getSignatureIndex(), CONST_Utf8);
- String localsig = ((ConstantUtf8) (cp.getConstant(localvariables[i].getSignatureIndex()))).getBytes(); // Local signature(=descriptor)
+ String localsig = ((ConstantUtf8) (cp.getConstant(localvariables[i].getSignatureIndex()))).getValue(); // Local
+ // signature(=descriptor)
Type t;
- try{
+ try {
t = Type.getType(localsig);
- }
- catch (ClassFormatError cfe){ // sometimes BCEL is a little harsh describing exceptional situations.
- throw new ClassConstraintException("Illegal descriptor (==signature) '"+localsig+"' used by LocalVariable '"+tostring(localvariables[i])+"' referenced by '"+tostring(lvt)+"'.");
+ } catch (ClassFormatError cfe) { // sometimes BCEL is a little harsh describing exceptional situations.
+ throw new ClassConstraintException("Illegal descriptor (==signature) '" + localsig
+ + "' used by LocalVariable '" + tostring(localvariables[i]) + "' referenced by '"
+ + tostring(lvt) + "'.");
}
int localindex = localvariables[i].getIndex();
- if ( ( (t==Type.LONG || t==Type.DOUBLE)? localindex+1:localindex) >= code.getMaxLocals()){
- throw new ClassConstraintException("LocalVariableTable attribute '"+tostring(lvt)+"' references a LocalVariable '"+tostring(localvariables[i])+"' with an index that exceeds the surrounding Code attribute's max_locals value of '"+code.getMaxLocals()+"'.");
+ if (((t == Type.LONG || t == Type.DOUBLE) ? localindex + 1 : localindex) >= code.getMaxLocals()) {
+ throw new ClassConstraintException("LocalVariableTable attribute '" + tostring(lvt)
+ + "' references a LocalVariable '" + tostring(localvariables[i])
+ + "' with an index that exceeds the surrounding Code attribute's max_locals value of '"
+ + code.getMaxLocals() + "'.");
}
- try{
- localVariablesInfos[method_number].add(localindex, localname, localvariables[i].getStartPC(), localvariables[i].getLength(), t);
- }
- catch(LocalVariableInfoInconsistentException lviie){
- throw new ClassConstraintException("Conflicting information in LocalVariableTable '"+tostring(lvt)+"' found in Code attribute '"+tostring(obj)+"' (method '"+tostring(m)+"'). "+lviie.getMessage());
+ try {
+ localVariablesInfos[method_number].add(localindex, localname, localvariables[i].getStartPC(),
+ localvariables[i].getLength(), t);
+ } catch (LocalVariableInfoInconsistentException lviie) {
+ throw new ClassConstraintException("Conflicting information in LocalVariableTable '" + tostring(lvt)
+ + "' found in Code attribute '" + tostring(obj) + "' (method '" + tostring(m) + "'). "
+ + lviie.getMessage());
}
}// for all local variables localvariables[i] in the LocalVariableTable attribute atts[a] END
num_of_lvt_attribs++;
- if (num_of_lvt_attribs > obj.getMaxLocals()){
- throw new ClassConstraintException("Number of LocalVariableTable attributes of Code attribute '"+tostring(obj)+"' (method '"+tostring(m)+"') exceeds number of local variable slots '"+obj.getMaxLocals()+"' ('There may be no more than one LocalVariableTable attribute per local variable in the Code attribute.').");
+ if (num_of_lvt_attribs > obj.getMaxLocals()) {
+ throw new ClassConstraintException(
+ "Number of LocalVariableTable attributes of Code attribute '"
+ + tostring(obj)
+ + "' (method '"
+ + tostring(m)
+ + "') exceeds number of local variable slots '"
+ + obj.getMaxLocals()
+ + "' ('There may be no more than one LocalVariableTable attribute per local variable in the Code attribute.').");
}
}// if atts[a] instanceof LocalVariableTable END
}// for all attributes atts[a] END
}// visitCode(Code) END
- public void visitExceptionTable(ExceptionTable obj){//vmspec2 4.7.4
+ public void visitExceptionTable(ExceptionTable obj) {// vmspec2 4.7.4
// incorrectly named, it's the Exceptions attribute (vmspec2 4.7.4)
checkIndex(obj, obj.getNameIndex(), CONST_Utf8);
- String name = ((ConstantUtf8) cp.getConstant(obj.getNameIndex())).getBytes();
- if (! name.equals("Exceptions")){
- throw new ClassConstraintException("The Exceptions attribute '"+tostring(obj)+"' is not correctly named 'Exceptions' but '"+name+"'.");
+ String name = ((ConstantUtf8) cp.getConstant(obj.getNameIndex())).getValue();
+ if (!name.equals("Exceptions")) {
+ throw new ClassConstraintException("The Exceptions attribute '" + tostring(obj)
+ + "' is not correctly named 'Exceptions' but '" + name + "'.");
}
int[] exc_indices = obj.getExceptionIndexTable();
- for (int i=0; i<exc_indices.length; i++){
+ for (int i = 0; i < exc_indices.length; i++) {
checkIndex(obj, exc_indices[i], CONST_Class);
ConstantClass cc = (ConstantClass) (cp.getConstant(exc_indices[i]));
- checkIndex(cc, cc.getNameIndex(), CONST_Utf8); // cannot be sure this ConstantClass has already been visited (checked)!
- String cname = ((ConstantUtf8) cp.getConstant(cc.getNameIndex())).getBytes().replace('/','.'); //convert internal notation on-the-fly to external notation
+ checkIndex(cc, cc.getNameIndex(), CONST_Utf8); // cannot be sure this ConstantClass has already been visited
+ // (checked)!
+ String cname = ((ConstantUtf8) cp.getConstant(cc.getNameIndex())).getValue().replace('/', '.'); // convert internal
+ // notation
+ // on-the-fly to
+ // external notation
Verifier v = VerifierFactory.getVerifier(cname);
VerificationResult vr = v.doPass1();
- if (vr != VerificationResult.VR_OK){
- throw new ClassConstraintException("Exceptions attribute '"+tostring(obj)+"' references '"+cname+"' as an Exception but it does not pass verification pass 1: "+vr);
- }
- else{
+ if (vr != VerificationResult.VR_OK) {
+ throw new ClassConstraintException("Exceptions attribute '" + tostring(obj) + "' references '" + cname
+ + "' as an Exception but it does not pass verification pass 1: " + vr);
+ } else {
// We cannot safely trust any other "instanceof" mechanism. We need to transitively verify
// the ancestor hierarchy.
JavaClass e = Repository.lookupClass(cname);
JavaClass t = Repository.lookupClass(Type.THROWABLE.getClassName());
JavaClass o = Repository.lookupClass(Type.OBJECT.getClassName());
- while (e != o){
- if (e == t) break; // It's a subclass of Throwable, OKAY, leave.
+ while (e != o) {
+ if (e == t)
+ break; // It's a subclass of Throwable, OKAY, leave.
v = VerifierFactory.getVerifier(e.getSuperclassName());
vr = v.doPass1();
- if (vr != VerificationResult.VR_OK){
- throw new ClassConstraintException("Exceptions attribute '"+tostring(obj)+"' references '"+cname+"' as an Exception but '"+e.getSuperclassName()+"' in the ancestor hierachy does not pass verification pass 1: "+vr);
- }
- else{
+ if (vr != VerificationResult.VR_OK) {
+ throw new ClassConstraintException("Exceptions attribute '" + tostring(obj) + "' references '" + cname
+ + "' as an Exception but '" + e.getSuperclassName()
+ + "' in the ancestor hierachy does not pass verification pass 1: " + vr);
+ } else {
e = Repository.lookupClass(e.getSuperclassName());
}
}
- if (e != t) throw new ClassConstraintException("Exceptions attribute '"+tostring(obj)+"' references '"+cname+"' as an Exception but it is not a subclass of '"+t.getClassName()+"'.");
+ if (e != t)
+ throw new ClassConstraintException("Exceptions attribute '" + tostring(obj) + "' references '" + cname
+ + "' as an Exception but it is not a subclass of '" + t.getClassName() + "'.");
}
}
}
+
// SYNTHETIC: see above
// DEPRECATED: see above
- //////////////////////////////////////////////////////////////
+ // ////////////////////////////////////////////////////////////
// code_attribute-structure-ATTRIBUTES (vmspec2 4.7.3, 4.7) //
- //////////////////////////////////////////////////////////////
- public void visitLineNumberTable(LineNumberTable obj){//vmspec2 4.7.8
+ // ////////////////////////////////////////////////////////////
+ public void visitLineNumberTable(LineNumberTable obj) {// vmspec2 4.7.8
checkIndex(obj, obj.getNameIndex(), CONST_Utf8);
- String name = ((ConstantUtf8) cp.getConstant(obj.getNameIndex())).getBytes();
- if (! name.equals("LineNumberTable")){
- throw new ClassConstraintException("The LineNumberTable attribute '"+tostring(obj)+"' is not correctly named 'LineNumberTable' but '"+name+"'.");
+ String name = ((ConstantUtf8) cp.getConstant(obj.getNameIndex())).getValue();
+ if (!name.equals("LineNumberTable")) {
+ throw new ClassConstraintException("The LineNumberTable attribute '" + tostring(obj)
+ + "' is not correctly named 'LineNumberTable' but '" + name + "'.");
}
- //In JustIce,this check is delayed to Pass 3a.
- //LineNumber[] linenumbers = obj.getLineNumberTable();
+ // In JustIce,this check is delayed to Pass 3a.
+ // LineNumber[] linenumbers = obj.getLineNumberTable();
// ...validity check...
}
- public void visitLocalVariableTable(LocalVariableTable obj){//vmspec2 4.7.9
- //In JustIce,this check is partially delayed to Pass 3a.
- //The other part can be found in the visitCode(Code) method.
+
+ public void visitLocalVariableTable(LocalVariableTable obj) {// vmspec2 4.7.9
+ // In JustIce,this check is partially delayed to Pass 3a.
+ // The other part can be found in the visitCode(Code) method.
}
- ////////////////////////////////////////////////////
+
+ // //////////////////////////////////////////////////
// MISC-structure-ATTRIBUTES (vmspec2 4.7.1, 4.7) //
- ////////////////////////////////////////////////////
- public void visitUnknown(Unknown obj){//vmspec2 4.7.1
+ // //////////////////////////////////////////////////
+ public void visitUnknown(Unknown obj) {// vmspec2 4.7.1
// Represents an unknown attribute.
checkIndex(obj, obj.getNameIndex(), CONST_Utf8);
// Maybe only misnamed? Give a (warning) message.
- addMessage("Unknown attribute '"+tostring(obj)+"'. This attribute is not known in any context!");
+ addMessage("Unknown attribute '" + tostring(obj) + "'. This attribute is not known in any context!");
}
- //////////
+
+ // ////////
// BCEL //
- //////////
- public void visitLocalVariable(LocalVariable obj){
+ // ////////
+ public void visitLocalVariable(LocalVariable obj) {
// This does not represent an Attribute but is only
// related to internal BCEL data representation.
// see visitLocalVariableTable(LocalVariableTable)
}
- public void visitCodeException(CodeException obj){
+
+ public void visitCodeException(CodeException obj) {
// Code constraints are checked in Pass3 (3a and 3b).
// This does not represent an Attribute but is only
// related to internal BCEL data representation.
// see visitCode(Code)
}
- public void visitConstantPool(ConstantPool obj){
+
+ public void visitConstantPool(ConstantPool obj) {
// No need to. We're piggybacked by the DescendingVisitor.
// This does not represent an Attribute but is only
// related to internal BCEL data representation.
}
- public void visitInnerClass(InnerClass obj){
+
+ public void visitInnerClass(InnerClass obj) {
// This does not represent an Attribute but is only
// related to internal BCEL data representation.
}
- public void visitLineNumber(LineNumber obj){
+
+ public void visitLineNumber(LineNumber obj) {
// This does not represent an Attribute but is only
// related to internal BCEL data representation.
@@ -1097,277 +1220,271 @@ public final class Pass2Verifier extends PassVerifier implements Constants{
}
/**
- * Ensures that the ConstantCP-subclassed entries of the constant
- * pool are valid. According to "Yellin: Low Level Security in Java",
- * this method does not verify the existence of referenced entities
- * (such as classes) but only the formal correctness (such as well-formed
- * signatures).
- * The visitXXX() methods throw ClassConstraintException instances otherwise.
- * <B>Precondition: index-style cross referencing in the constant
- * pool must be valid. Simply invoke constant_pool_entries_satisfy_static_constraints()
- * before.</B>
- *
+ * Ensures that the ConstantCP-subclassed entries of the constant pool are valid. According to
+ * "Yellin: Low Level Security in Java", this method does not verify the existence of referenced entities (such as classes) but
+ * only the formal correctness (such as well-formed signatures). The visitXXX() methods throw ClassConstraintException instances
+ * otherwise. <B>Precondition: index-style cross referencing in the constant pool must be valid. Simply invoke
+ * constant_pool_entries_satisfy_static_constraints() before.</B>
+ *
* @throws ClassConstraintException otherwise.
* @see #constant_pool_entries_satisfy_static_constraints()
*/
- private void field_and_method_refs_are_valid(){
+ private void field_and_method_refs_are_valid() {
JavaClass jc = Repository.lookupClass(myOwner.getClassName());
DescendingVisitor v = new DescendingVisitor(jc, new FAMRAV_Visitor(jc));
v.visit();
}
/**
- * A Visitor class that ensures the ConstantCP-subclassed entries
- * of the constant pool are valid.
- * <B>Precondition: index-style cross referencing in the constant
- * pool must be valid.</B>
- *
- * @see #constant_pool_entries_satisfy_static_constraints()
+ * A Visitor class that ensures the ConstantCP-subclassed entries of the constant pool are valid. <B>Precondition: index-style
+ * cross referencing in the constant pool must be valid.</B>
+ *
+ * @see #constant_pool_entries_satisfy_static_constraints()
* @see org.aspectj.apache.bcel.classfile.ConstantCP
*/
- private class FAMRAV_Visitor extends EmptyClassVisitor{
+ private class FAMRAV_Visitor extends EmptyClassVisitor {
private final ConstantPool cp; // ==jc.getConstantPool() -- only here to save typing work.
- private FAMRAV_Visitor(JavaClass _jc){
+
+ private FAMRAV_Visitor(JavaClass _jc) {
cp = _jc.getConstantPool();
}
-
- public void visitConstantFieldref(ConstantFieldref obj){
- if (obj.getTag() != Constants.CONSTANT_Fieldref){
- throw new ClassConstraintException("ConstantFieldref '"+tostring(obj)+"' has wrong tag!");
+
+ public void visitConstantFieldref(ConstantFieldref obj) {
+ if (obj.getTag() != Constants.CONSTANT_Fieldref) {
+ throw new ClassConstraintException("ConstantFieldref '" + tostring(obj) + "' has wrong tag!");
}
int name_and_type_index = obj.getNameAndTypeIndex();
ConstantNameAndType cnat = (ConstantNameAndType) (cp.getConstant(name_and_type_index));
- String name = ((ConstantUtf8) (cp.getConstant(cnat.getNameIndex()))).getBytes(); // Field or Method name
- if (!validFieldName(name)){
- throw new ClassConstraintException("Invalid field name '"+name+"' referenced by '"+tostring(obj)+"'.");
+ String name = ((ConstantUtf8) (cp.getConstant(cnat.getNameIndex()))).getValue(); // Field or Method name
+ if (!validFieldName(name)) {
+ throw new ClassConstraintException("Invalid field name '" + name + "' referenced by '" + tostring(obj) + "'.");
}
-
+
int class_index = obj.getClassIndex();
ConstantClass cc = (ConstantClass) (cp.getConstant(class_index));
- String className = ((ConstantUtf8) (cp.getConstant(cc.getNameIndex()))).getBytes(); // Class Name in internal form
- if (! validClassName(className)){
- throw new ClassConstraintException("Illegal class name '"+className+"' used by '"+tostring(obj)+"'.");
+ String className = ((ConstantUtf8) (cp.getConstant(cc.getNameIndex()))).getValue(); // Class Name in internal form
+ if (!validClassName(className)) {
+ throw new ClassConstraintException("Illegal class name '" + className + "' used by '" + tostring(obj) + "'.");
}
- String sig = ((ConstantUtf8) (cp.getConstant(cnat.getSignatureIndex()))).getBytes(); // Field or Method signature(=descriptor)
-
- try{
+ String sig = ((ConstantUtf8) (cp.getConstant(cnat.getSignatureIndex()))).getValue(); // Field or Method
+ // signature(=descriptor)
+
+ try {
Type.getType(sig); /* Don't need the return value */
- }
- catch (ClassFormatError cfe){
+ } catch (ClassFormatError cfe) {
// Well, BCEL sometimes is a little harsh describing exceptional situations.
- throw new ClassConstraintException("Illegal descriptor (==signature) '"+sig+"' used by '"+tostring(obj)+"'.");
+ throw new ClassConstraintException("Illegal descriptor (==signature) '" + sig + "' used by '" + tostring(obj)
+ + "'.");
}
}
- public void visitConstantMethodref(ConstantMethodref obj){
- if (obj.getTag() != Constants.CONSTANT_Methodref){
- throw new ClassConstraintException("ConstantMethodref '"+tostring(obj)+"' has wrong tag!");
+ public void visitConstantMethodref(ConstantMethodref obj) {
+ if (obj.getTag() != Constants.CONSTANT_Methodref) {
+ throw new ClassConstraintException("ConstantMethodref '" + tostring(obj) + "' has wrong tag!");
}
int name_and_type_index = obj.getNameAndTypeIndex();
ConstantNameAndType cnat = (ConstantNameAndType) (cp.getConstant(name_and_type_index));
- String name = ((ConstantUtf8) (cp.getConstant(cnat.getNameIndex()))).getBytes(); // Field or Method name
- if (!validClassMethodName(name)){
- throw new ClassConstraintException("Invalid (non-interface) method name '"+name+"' referenced by '"+tostring(obj)+"'.");
+ String name = ((ConstantUtf8) (cp.getConstant(cnat.getNameIndex()))).getValue(); // Field or Method name
+ if (!validClassMethodName(name)) {
+ throw new ClassConstraintException("Invalid (non-interface) method name '" + name + "' referenced by '"
+ + tostring(obj) + "'.");
}
int class_index = obj.getClassIndex();
ConstantClass cc = (ConstantClass) (cp.getConstant(class_index));
- String className = ((ConstantUtf8) (cp.getConstant(cc.getNameIndex()))).getBytes(); // Class Name in internal form
- if (! validClassName(className)){
- throw new ClassConstraintException("Illegal class name '"+className+"' used by '"+tostring(obj)+"'.");
+ String className = ((ConstantUtf8) (cp.getConstant(cc.getNameIndex()))).getValue(); // Class Name in internal form
+ if (!validClassName(className)) {
+ throw new ClassConstraintException("Illegal class name '" + className + "' used by '" + tostring(obj) + "'.");
}
- String sig = ((ConstantUtf8) (cp.getConstant(cnat.getSignatureIndex()))).getBytes(); // Field or Method signature(=descriptor)
-
- try{
- Type t = Type.getReturnType(sig);
- if ( name.equals(CONSTRUCTOR_NAME) && (t != Type.VOID) ){
+ String sig = ((ConstantUtf8) (cp.getConstant(cnat.getSignatureIndex()))).getValue(); // Field or Method
+ // signature(=descriptor)
+
+ try {
+ Type t = Type.getReturnType(sig);
+ if (name.equals(CONSTRUCTOR_NAME) && (t != Type.VOID)) {
throw new ClassConstraintException("Instance initialization method must have VOID return type.");
}
- }
- catch (ClassFormatError cfe){
+ } catch (ClassFormatError cfe) {
// Well, BCEL sometimes is a little harsh describing exceptional situations.
- throw new ClassConstraintException("Illegal descriptor (==signature) '"+sig+"' used by '"+tostring(obj)+"'.");
+ throw new ClassConstraintException("Illegal descriptor (==signature) '" + sig + "' used by '" + tostring(obj)
+ + "'.");
}
}
- public void visitConstantInterfaceMethodref(ConstantInterfaceMethodref obj){
- if (obj.getTag() != Constants.CONSTANT_InterfaceMethodref){
- throw new ClassConstraintException("ConstantInterfaceMethodref '"+tostring(obj)+"' has wrong tag!");
+ public void visitConstantInterfaceMethodref(ConstantInterfaceMethodref obj) {
+ if (obj.getTag() != Constants.CONSTANT_InterfaceMethodref) {
+ throw new ClassConstraintException("ConstantInterfaceMethodref '" + tostring(obj) + "' has wrong tag!");
}
int name_and_type_index = obj.getNameAndTypeIndex();
ConstantNameAndType cnat = (ConstantNameAndType) (cp.getConstant(name_and_type_index));
- String name = ((ConstantUtf8) (cp.getConstant(cnat.getNameIndex()))).getBytes(); // Field or Method name
- if (!validInterfaceMethodName(name)){
- throw new ClassConstraintException("Invalid (interface) method name '"+name+"' referenced by '"+tostring(obj)+"'.");
+ String name = ((ConstantUtf8) (cp.getConstant(cnat.getNameIndex()))).getValue(); // Field or Method name
+ if (!validInterfaceMethodName(name)) {
+ throw new ClassConstraintException("Invalid (interface) method name '" + name + "' referenced by '" + tostring(obj)
+ + "'.");
}
int class_index = obj.getClassIndex();
ConstantClass cc = (ConstantClass) (cp.getConstant(class_index));
- String className = ((ConstantUtf8) (cp.getConstant(cc.getNameIndex()))).getBytes(); // Class Name in internal form
- if (! validClassName(className)){
- throw new ClassConstraintException("Illegal class name '"+className+"' used by '"+tostring(obj)+"'.");
+ String className = ((ConstantUtf8) (cp.getConstant(cc.getNameIndex()))).getValue(); // Class Name in internal form
+ if (!validClassName(className)) {
+ throw new ClassConstraintException("Illegal class name '" + className + "' used by '" + tostring(obj) + "'.");
}
- String sig = ((ConstantUtf8) (cp.getConstant(cnat.getSignatureIndex()))).getBytes(); // Field or Method signature(=descriptor)
-
- try{
- Type t = Type.getReturnType(sig);
- if ( name.equals(STATIC_INITIALIZER_NAME) && (t != Type.VOID) ){
- addMessage("Class or interface initialization method '"+STATIC_INITIALIZER_NAME+"' usually has VOID return type instead of '"+t+"'. Note this is really not a requirement of The Java Virtual Machine Specification, Second Edition.");
+ String sig = ((ConstantUtf8) (cp.getConstant(cnat.getSignatureIndex()))).getValue(); // Field or Method
+ // signature(=descriptor)
+
+ try {
+ Type t = Type.getReturnType(sig);
+ if (name.equals(STATIC_INITIALIZER_NAME) && (t != Type.VOID)) {
+ addMessage("Class or interface initialization method '" + STATIC_INITIALIZER_NAME
+ + "' usually has VOID return type instead of '" + t
+ + "'. Note this is really not a requirement of The Java Virtual Machine Specification, Second Edition.");
}
- }
- catch (ClassFormatError cfe){
+ } catch (ClassFormatError cfe) {
// Well, BCEL sometimes is a little harsh describing exceptional situations.
- throw new ClassConstraintException("Illegal descriptor (==signature) '"+sig+"' used by '"+tostring(obj)+"'.");
+ throw new ClassConstraintException("Illegal descriptor (==signature) '" + sig + "' used by '" + tostring(obj)
+ + "'.");
}
}
-
+
}
/**
- * This method returns true if and only if the supplied String
- * represents a valid Java class name.
+ * This method returns true if and only if the supplied String represents a valid Java class name.
*/
- private static final boolean validClassName(String name){
- /*
- * TODO: implement.
- * Are there any restrictions?
- */
+ private static final boolean validClassName(String name) {
+ /*
+ * TODO: implement. Are there any restrictions?
+ */
return true;
}
+
/**
- * This method returns true if and only if the supplied String
- * represents a valid method name.
- * This is basically the same as a valid identifier name in the
- * Java programming language, but the special name for
- * the instance initialization method is allowed and the special name
- * for the class/interface initialization method may be allowed.
+ * This method returns true if and only if the supplied String represents a valid method name. This is basically the same as a
+ * valid identifier name in the Java programming language, but the special name for the instance initialization method is
+ * allowed and the special name for the class/interface initialization method may be allowed.
*/
- private static boolean validMethodName(String name, boolean allowStaticInit){
- if (validJavaLangMethodName(name)) return true;
-
- if (allowStaticInit){
+ private static boolean validMethodName(String name, boolean allowStaticInit) {
+ if (validJavaLangMethodName(name))
+ return true;
+
+ if (allowStaticInit) {
return (name.equals(CONSTRUCTOR_NAME) || name.equals(STATIC_INITIALIZER_NAME));
- }
- else{
+ } else {
return name.equals(CONSTRUCTOR_NAME);
}
}
/**
- * This method returns true if and only if the supplied String
- * represents a valid method name that may be referenced by
+ * This method returns true if and only if the supplied String represents a valid method name that may be referenced by
* ConstantMethodref objects.
*/
- private static boolean validClassMethodName(String name){
+ private static boolean validClassMethodName(String name) {
return validMethodName(name, false);
}
/**
- * This method returns true if and only if the supplied String
- * represents a valid Java programming language method name stored as a simple
- * (non-qualified) name.
- * Conforming to: The Java Virtual Machine Specification, Second Edition, §2.7, §2.7.1, §2.2.
+ * This method returns true if and only if the supplied String represents a valid Java programming language method name stored
+ * as a simple (non-qualified) name. Conforming to: The Java Virtual Machine Specification, Second Edition, §2.7, §2.7.1, §2.2.
*/
- private static boolean validJavaLangMethodName(String name){
- if (!Character.isJavaIdentifierStart(name.charAt(0))) return false;
-
- for (int i=1; i<name.length(); i++){
- if (!Character.isJavaIdentifierPart(name.charAt(i))) return false;
+ private static boolean validJavaLangMethodName(String name) {
+ if (!Character.isJavaIdentifierStart(name.charAt(0)))
+ return false;
+
+ for (int i = 1; i < name.length(); i++) {
+ if (!Character.isJavaIdentifierPart(name.charAt(i)))
+ return false;
}
return true;
}
/**
- * This method returns true if and only if the supplied String
- * represents a valid Java interface method name that may be
+ * This method returns true if and only if the supplied String represents a valid Java interface method name that may be
* referenced by ConstantInterfaceMethodref objects.
*/
- private static boolean validInterfaceMethodName(String name){
+ private static boolean validInterfaceMethodName(String name) {
// I guess we should assume special names forbidden here.
- if (name.startsWith("<")) return false;
+ if (name.startsWith("<"))
+ return false;
return validJavaLangMethodName(name);
}
/**
- * This method returns true if and only if the supplied String
- * represents a valid Java identifier (so-called simple name).
+ * This method returns true if and only if the supplied String represents a valid Java identifier (so-called simple name).
*/
- private static boolean validJavaIdentifier(String name){
+ private static boolean validJavaIdentifier(String name) {
// vmspec2 2.7, vmspec2 2.2
- if (!Character.isJavaIdentifierStart(name.charAt(0))) return false;
-
- for (int i=1; i<name.length(); i++){
- if (!Character.isJavaIdentifierPart(name.charAt(i))) return false;
+ if (!Character.isJavaIdentifierStart(name.charAt(0)))
+ return false;
+
+ for (int i = 1; i < name.length(); i++) {
+ if (!Character.isJavaIdentifierPart(name.charAt(i)))
+ return false;
}
return true;
}
/**
- * This method returns true if and only if the supplied String
- * represents a valid Java field name.
+ * This method returns true if and only if the supplied String represents a valid Java field name.
*/
- private static boolean validFieldName(String name){
+ private static boolean validFieldName(String name) {
// vmspec2 2.7, vmspec2 2.2
return validJavaIdentifier(name);
}
/**
- * This class serves for finding out if a given JavaClass' ConstantPool
- * references an Inner Class.
- * The Java Virtual Machine Specification, Second Edition is not very precise
- * about when an "InnerClasses" attribute has to appear. However, it states that
- * there has to be exactly one InnerClasses attribute in the ClassFile structure
- * if the constant pool of a class or interface refers to any class or interface
- * "that is not a member of a package". Sun does not mean "member of the default
- * package". In "Inner Classes Specification" they point out how a "bytecode name"
- * is derived so one has to deduce what a class name of a class "that is not a
- * member of a package" looks like: there is at least one character in the byte-
- * code name that cannot be part of a legal Java Language Class name (and not equal
- * to '/'). This assumption is wrong as the delimiter is '$' for which
- * Character.isJavaIdentifierPart() == true.
- * Hence, you really run into trouble if you have a toplevel class called
- * "A$XXX" and another toplevel class called "A" with in inner class called "XXX".
- * JustIce cannot repair this; please note that existing verifiers at this
- * time even fail to detect missing InnerClasses attributes in pass 2.
+ * This class serves for finding out if a given JavaClass' ConstantPool references an Inner Class. The Java Virtual Machine
+ * Specification, Second Edition is not very precise about when an "InnerClasses" attribute has to appear. However, it states
+ * that there has to be exactly one InnerClasses attribute in the ClassFile structure if the constant pool of a class or
+ * interface refers to any class or interface "that is not a member of a package". Sun does not mean "member of the default
+ * package". In "Inner Classes Specification" they point out how a "bytecode name" is derived so one has to deduce what a class
+ * name of a class "that is not a member of a package" looks like: there is at least one character in the byte- code name that
+ * cannot be part of a legal Java Language Class name (and not equal to '/'). This assumption is wrong as the delimiter is '$'
+ * for which Character.isJavaIdentifierPart() == true. Hence, you really run into trouble if you have a toplevel class called
+ * "A$XXX" and another toplevel class called "A" with in inner class called "XXX". JustIce cannot repair this; please note that
+ * existing verifiers at this time even fail to detect missing InnerClasses attributes in pass 2.
*/
- private class InnerClassDetector extends EmptyClassVisitor{
+ private class InnerClassDetector extends EmptyClassVisitor {
private boolean hasInnerClass = false;
private JavaClass jc;
private ConstantPool cp;
- private InnerClassDetector(){} // Don't use.
+
+ private InnerClassDetector() {
+ } // Don't use.
+
/** Constructs an InnerClassDetector working on the JavaClass _jc. */
- public InnerClassDetector(JavaClass _jc){
+ public InnerClassDetector(JavaClass _jc) {
jc = _jc;
cp = jc.getConstantPool();
(new DescendingVisitor(jc, this)).visit();
}
+
/**
- * Returns if the JavaClass this InnerClassDetector is working on
- * has an Inner Class reference in its constant pool.
+ * Returns if the JavaClass this InnerClassDetector is working on has an Inner Class reference in its constant pool.
*/
- public boolean innerClassReferenced(){
+ public boolean innerClassReferenced() {
return hasInnerClass;
}
+
/** This method casually visits ConstantClass references. */
- public void visitConstantClass(ConstantClass obj){
+ public void visitConstantClass(ConstantClass obj) {
Constant c = cp.getConstant(obj.getNameIndex());
- if (c instanceof ConstantUtf8){ //Ignore the case where it's not a ConstantUtf8 here, we'll find out later.
- String classname = ((ConstantUtf8) c).getBytes();
- if (classname.startsWith(jc.getClassName().replace('.','/')+"$")){
+ if (c instanceof ConstantUtf8) { // Ignore the case where it's not a ConstantUtf8 here, we'll find out later.
+ String classname = ((ConstantUtf8) c).getValue();
+ if (classname.startsWith(jc.getClassName().replace('.', '/') + "$")) {
hasInnerClass = true;
}
}
}
}
-
+
/**
* This method is here to save typing work and improve code readability.
*/
- private static String tostring(Node n){
+ private static String tostring(Node n) {
return new StringRepresentation(n).toString();
}
}
diff --git a/bcel-builder/verifier-src/org/aspectj/apache/bcel/verifier/statics/Pass3aVerifier.java b/bcel-builder/verifier-src/org/aspectj/apache/bcel/verifier/statics/Pass3aVerifier.java
index b6ae90b20..8afbdb5ea 100644
--- a/bcel-builder/verifier-src/org/aspectj/apache/bcel/verifier/statics/Pass3aVerifier.java
+++ b/bcel-builder/verifier-src/org/aspectj/apache/bcel/verifier/statics/Pass3aVerifier.java
@@ -111,7 +111,7 @@ import org.aspectj.apache.bcel.verifier.exc.StaticCodeInstructionOperandConstrai
* This PassVerifier verifies a class file according to pass 3, static part as described in The Java Virtual Machine Specification,
* 2nd edition. More detailed information is to be found at the do_verify() method's documentation.
*
- * @version $Id: Pass3aVerifier.java,v 1.4 2008/08/28 00:02:13 aclement Exp $
+ * @version $Id: Pass3aVerifier.java,v 1.5 2009/09/10 15:35:06 aclement Exp $
* @author <A HREF="http://www.inf.fu-berlin.de/~ehaase"/>Enver Haase</A>
* @see #do_verify()
*/
@@ -151,6 +151,7 @@ public final class Pass3aVerifier extends PassVerifier {
*
* @throws InvalidMethodException if the method to verify does not exist.
*/
+ @Override
public VerificationResult do_verify() {
if (myOwner.doPass2().equals(VerificationResult.VR_OK)) {
// Okay, class file was loaded correctly by Pass 1
@@ -482,6 +483,7 @@ public final class Pass3aVerifier extends PassVerifier {
/**
* Assures the generic preconditions of a LoadClass instance. The referenced class is loaded and pass2-verified.
*/
+ @Override
public void visitLoadClass(Instruction o) {
ObjectType t = o.getLoadClassType(cpg);
if (t != null) {// null means "no class is loaded"
@@ -503,6 +505,7 @@ public final class Pass3aVerifier extends PassVerifier {
/** Checks if the constraints of operands of the said instruction(s) are satisfied. */
// LDC and LDC_W (LDC_W is a subclass of LDC in BCEL's model)
+ @Override
public void visitLDC(Instruction o) {
indexValid(o, o.getIndex());
Constant c = cpg.getConstant(o.getIndex());
@@ -515,6 +518,7 @@ public final class Pass3aVerifier extends PassVerifier {
/** Checks if the constraints of operands of the said instruction(s) are satisfied. */
// LDC2_W
+ @Override
public void visitLDC2_W(Instruction o) {
indexValid(o, o.getIndex());
Constant c = cpg.getConstant(o.getIndex());
@@ -571,6 +575,7 @@ public final class Pass3aVerifier extends PassVerifier {
}
/** Checks if the constraints of operands of the said instruction(s) are satisfied. */
+ @Override
public void visitInvokeInstruction(InvokeInstruction o) {
indexValid(o, o.getIndex());
if (o.getOpcode() == Constants.INVOKEVIRTUAL || o.getOpcode() == Constants.INVOKESPECIAL
@@ -582,10 +587,10 @@ public final class Pass3aVerifier extends PassVerifier {
// Constants are okay due to pass2.
ConstantNameAndType cnat = (ConstantNameAndType) cpg.getConstant(((ConstantMethodref) c).getNameAndTypeIndex());
ConstantUtf8 cutf8 = (ConstantUtf8) cpg.getConstant(cnat.getNameIndex());
- if (cutf8.getBytes().equals(Constants.CONSTRUCTOR_NAME) && !(o.getOpcode() == Constants.INVOKESPECIAL)) {
+ if (cutf8.getValue().equals(Constants.CONSTRUCTOR_NAME) && !(o.getOpcode() == Constants.INVOKESPECIAL)) {
constraintViolated(o, "Only INVOKESPECIAL is allowed to invoke instance initialization methods.");
}
- if (!cutf8.getBytes().equals(Constants.CONSTRUCTOR_NAME) && cutf8.getBytes().startsWith("<")) {
+ if (!cutf8.getValue().equals(Constants.CONSTRUCTOR_NAME) && cutf8.getValue().startsWith("<")) {
constraintViolated(
o,
"No method with a name beginning with '<' other than the instance initialization methods may be called by the method invocation instructions.");
@@ -604,7 +609,7 @@ public final class Pass3aVerifier extends PassVerifier {
// Invoked method must not be <init> or <clinit>
ConstantNameAndType cnat = (ConstantNameAndType) cpg.getConstant(((ConstantInterfaceMethodref) c)
.getNameAndTypeIndex());
- String name = ((ConstantUtf8) cpg.getConstant(cnat.getNameIndex())).getBytes();
+ String name = ((ConstantUtf8) cpg.getConstant(cnat.getNameIndex())).getValue();
if (name.equals(Constants.CONSTRUCTOR_NAME)) {
constraintViolated(o, "Method to invoke must not be '" + Constants.CONSTRUCTOR_NAME + "'.");
}
@@ -647,6 +652,7 @@ public final class Pass3aVerifier extends PassVerifier {
}
/** Checks if the constraints of operands of the said instruction(s) are satisfied. */
+ @Override
public void visitINSTANCEOF(Instruction o) {
indexValid(o, o.getIndex());
Constant c = cpg.getConstant(o.getIndex());
@@ -656,6 +662,7 @@ public final class Pass3aVerifier extends PassVerifier {
}
/** Checks if the constraints of operands of the said instruction(s) are satisfied. */
+ @Override
public void visitCHECKCAST(Instruction o) {
indexValid(o, o.getIndex());
Constant c = cpg.getConstant(o.getIndex());
@@ -665,6 +672,7 @@ public final class Pass3aVerifier extends PassVerifier {
}
/** Checks if the constraints of operands of the said instruction(s) are satisfied. */
+ @Override
public void visitNEW(Instruction o) {
indexValid(o, o.getIndex());
Constant c = cpg.getConstant(o.getIndex());
@@ -672,7 +680,7 @@ public final class Pass3aVerifier extends PassVerifier {
constraintViolated(o, "Expecting a CONSTANT_Class operand, but found a '" + c + "'.");
} else {
ConstantUtf8 cutf8 = (ConstantUtf8) cpg.getConstant(((ConstantClass) c).getNameIndex());
- Type t = Type.getType("L" + cutf8.getBytes() + ";");
+ Type t = Type.getType("L" + cutf8.getValue() + ";");
if (t instanceof ArrayType) {
constraintViolated(o, "NEW must not be used to create an array.");
}