From c80dc162d93ab81e11198223d1d210a259f17efb Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Enrico=20Schr=C3=B6dter?= Date: Sun, 20 Mar 2016 18:55:36 +0100 Subject: [PATCH] - Verhindern von der Generierung von doppelte Methoden --- src/de/dhbwstuttgart/syntaxtree/Class.java | 13 +- .../dhbwstuttgart/syntaxtree/Constructor.java | 2 +- src/de/dhbwstuttgart/syntaxtree/Method.java | 201 ++---------------- .../types/AutoOverloadingMultiResults.jav | 8 + .../AutoOverloadingMultiResultsTest.java | 36 ++++ .../types/AutoOverloadingVectorTest.java | 8 +- test/bytecode/types/ExtendsType.jav | 2 +- test/bytecode/types/ExtendsTypeTest.java | 2 +- .../types/ExtendsVectorStringTest.java | 11 +- 9 files changed, 85 insertions(+), 198 deletions(-) create mode 100644 test/bytecode/types/AutoOverloadingMultiResults.jav create mode 100644 test/bytecode/types/AutoOverloadingMultiResultsTest.java diff --git a/src/de/dhbwstuttgart/syntaxtree/Class.java b/src/de/dhbwstuttgart/syntaxtree/Class.java index 006e3537..a746b8c9 100755 --- a/src/de/dhbwstuttgart/syntaxtree/Class.java +++ b/src/de/dhbwstuttgart/syntaxtree/Class.java @@ -7,6 +7,8 @@ import java.util.Collection; import java.util.Enumeration; import java.util.Hashtable; import java.util.Iterator; +import java.util.LinkedList; +import java.util.List; import javax.lang.model.element.Modifier; @@ -72,6 +74,7 @@ public class Class extends GTVDeclarationContext implements AClassOrInterface, I protected UsedId pkgName; protected Modifiers modifiers; protected String name; + private List methodSignaturesAndNames = new LinkedList<>(); /** * @@ -106,7 +109,7 @@ public class Class extends GTVDeclarationContext implements AClassOrInterface, I } //Zuerst die Methoden und Felder abarbeiten: for(Method m : methods){ - m.genByteCode(_cg); + m.genByteCode(_cg, this); } InstructionList fieldInitializations = new InstructionList(); for(FieldDeclaration f : fieldDeclarations){ @@ -1039,6 +1042,14 @@ public class Class extends GTVDeclarationContext implements AClassOrInterface, I public boolean isInterface(){ return false; } + + protected boolean methodExists(String nameAndSignature) { + return methodSignaturesAndNames.contains(nameAndSignature); + } + + protected void addMethod(String nameAndSignature) { + methodSignaturesAndNames.add(nameAndSignature); + } /* private Collection getGenericClasses() { diff --git a/src/de/dhbwstuttgart/syntaxtree/Constructor.java b/src/de/dhbwstuttgart/syntaxtree/Constructor.java index 65a85522..9859a734 100644 --- a/src/de/dhbwstuttgart/syntaxtree/Constructor.java +++ b/src/de/dhbwstuttgart/syntaxtree/Constructor.java @@ -98,7 +98,7 @@ public class Constructor extends Method { } @Override - public void genByteCode(ClassGenerator cg) { + public void genByteCode(ClassGenerator cg, Class classObj) { this.genByteCode(cg, new InstructionList()); } // super statement muss drin sein diff --git a/src/de/dhbwstuttgart/syntaxtree/Method.java b/src/de/dhbwstuttgart/syntaxtree/Method.java index 8977fdd8..8a3370fb 100755 --- a/src/de/dhbwstuttgart/syntaxtree/Method.java +++ b/src/de/dhbwstuttgart/syntaxtree/Method.java @@ -1,6 +1,7 @@ // ino.module.Method.8564.package package de.dhbwstuttgart.syntaxtree; +import java.util.Arrays; // ino.end // ino.module.Method.8564.import import java.util.Enumeration; @@ -465,39 +466,7 @@ public class Method extends Field implements IItemWithOffset, TypeInsertable } @Override - public void wandleRefTypeAttributes2GenericAttributes( - Menge classParalist) { - /* - * Menge paralist = new Menge();//Mit den Generischen Typen - * der Methode paralist.addAll(classParalist); - * paralist.addAll(this.genericMethodParameters); - * - * // Zuerst Returntype untersuchen Type returnType=getType(); Type - * pendantReturnType = null; if(returnType instanceof - * RefType)pendantReturnType = - * ((RefType)returnType).findGenericType(paralist, new - * Menge()); //GenericTypeVar - * pendantReturnType=ClassHelper.findGenericType(returnType, - * paralist,genericMethodParameters); if(pendantReturnType!=null){ - * //Wenn generisch, dann modifizieren setReturnType(pendantReturnType); - * } - * - * // Dann parameterlist untersuchen for(int - * par=0;par()); //GenericTypeVar - * pendantPara=ClassHelper.findGenericType - * (fpType,paralist,genericMethodParameters); if(pendantPara!=null){ - * //Wenn generisch, dann modifizieren fp.setType(pendantPara); } } - * - * // Zuletzt alle Lokalen Variablendeklarationen durchgehen - * if(block!=null){ - * block.wandleRefTypeAttributes2GenericAttributes(paralist - * ,genericMethodParameters); } - */ + public void wandleRefTypeAttributes2GenericAttributes(Menge classParalist) { } public void set_Method_Name(String string) { @@ -520,30 +489,11 @@ public class Method extends Field implements IItemWithOffset, TypeInsertable // TypeCheck, falls es sich um einen RefType handelt: this.returntype = this.returntype.checkTYPE(localAss, this); - /* - * if(this.returntype!=null && (this.returntype instanceof RefType)&& - * !(this.returntype instanceof mycompiler.mytype.Void)){//Sonderfall - * der Methode: Ihr Typ darf Void definiert werden. Type replaceType = - * null; replaceType = ass.getTypeFor((RefType)this.returntype); - * if(replaceType == null)throw new - * TypeinferenceException("Der Typ "+this - * .getType().getName()+" ist nicht korrekt",this); this.returntype = - * replaceType; } - */ // Die Parameter zu den Assumptions hinzufügen: if (this.parameterlist != null) for (FormalParameter param : this.parameterlist) { param.setType(param.getType().checkTYPE(localAss, this)); - /* - * if(param.getType() instanceof RefType) { Type replaceType = - * null; replaceType = ass.getTypeFor((RefType)param.getType()); - * if(replaceType == null) throw new - * TypeinferenceException("Der Typ " - * +param.getType().getName()+" ist nicht korrekt",param); - * param.setType(replaceType); } - */ - localAss.addAssumption(new ParameterAssumption(param)); } ret.add(this.block.TYPEStmt(localAss)); @@ -587,107 +537,15 @@ public class Method extends Field implements IItemWithOffset, TypeInsertable TypeAssumptions ret = new TypeAssumptions(); ret.addAssumption(new MethodAssumption(this, parentClass)); return ret; - /* - * TypeAssumptions assumptions = new TypeAssumptions(); this.assumedType - * = null; //if((this.get_Method_Name().equals(classmember.getName()) || - * this.get_Method_Name().equals("")) && - * ((this.getType().equals(new mycompiler.mytype.Void(0))) || - * this.getType() instanceof TypePlaceholder)){ - * if((this.get_Method_Name().equals(classmember.getName()) || - * this.get_Method_Name().equals(""))) { - * this.set_Method_Name(""); this.assumedType = new - * RefType(classmember.getName(),0); - * this.setReturnType(this.assumedType); this.assumedType = new - * RefType("void",0); //Return constructorReturnStatement = new - * Return(0,0); //constructorReturnStatement.retexpr = - * //this.block.statements.add(constructorReturnStatement); } //hoth: - * 06.04.2006 //durchlaufe Block und suche nach Objektvariablen fuer - * Offset-Markierung Iterator fieldVarIterator = - * assumptions.iterator(); while (fieldVarIterator.hasNext()) { //Wenn - * ObjektVariable CTypeAssumption dieAssum = fieldVarIterator.next(); - * if(dieAssum instanceof CInstVarTypeAssumption) { - * Class.isFirstLocalVarDecl=false; if(this.get_Block() != null) - * this.get_Block - * ().addOffsetsToAssumption(dieAssum,dieAssum.getIdentifier(),true); } - * } - * - * //methodList.addElement(method); - * - * //F�r V_fields_methods: CMethodTypeAssumption methodAssum - * = new CMethodTypeAssumption(classmember.getType(), - * this.get_Method_Name(), this.getType(), - * this.getParameterCount(),this.getLineNumber(),this.getOffset(),new - * Menge(),this.getGenericMethodParameters()); // Typannahme - * bauen... - * - * - * //Methode in V_Fields_methods ablegen //Dabei wird die - * OverloadedMethodID ermittelt !! //=> Method setzenuct - * - * - * assumptions.add(methodAssum); - * this.setOverloadedID(methodAssum.getHashSetKey - * ().getOverloadedMethodID()); - * - * - * //F�r die V_i: CTypeAssumptionSet localAssum = new - * CTypeAssumptionSet(); - * - * //Bauen... ParameterList parameterList = this.getParameterList(); - * if(parameterList!=null){ for(int i=0; - * i()); //fuege Offsets fuer Parameter hinzu, hoth: - * 06.04.2006 Class.isFirstLocalVarDecl=false; - * - * if(this.get_Block() != null) - * this.get_Block().addOffsetsToAssumption(paraAssum - * ,paraAssum.getIdentifier(),true); - * - * methodAssum.addParaAssumption(paraAssum); - * - * // F�r die V_i: CLocalVarTypeAssumption varAssum = new - * CLocalVarTypeAssumption(classmember.getName(), - * this.get_Method_Name(), this.getParameterCount(), - * this.getOverloadedID(),"1", para.get_Name(),para.getType(), - * para.getLineNumber(),para.getOffset(),new Menge()); - * localAssum.addElement(varAssum); - * //rememberLocals.addElement(varAssum); } } //...und - * hinzuf�gen: - * - * assumptions.add(localAssum);//Assumptions für lokale Variablen den - * Assumptions hinzufügen - * - * //Hier wird der Typ der als Assumption eingetragen wird in die - * Variable assumedType dieser Klasse geschrieben: if(this.assumedType - * == null) // Falls der Typ nicht schon gesetzt ist. Das ist der Fall, - * falls die Methode ein Konstruktor ist this.assumedType = - * methodAssum.getAssumedType(); - * - * return assumptions; - */ } @Override public void parserPostProcessing(SyntaxTreeNode parent) { if (this.getType() == null) this.setType(TypePlaceholder.fresh(this)); - // Bei dem Elterntyp der Methode darf es sich nur um eine Klasse - // handeln, daher Cast ohne Prüfung: - // Class parentClass = (Class)parent; if (this.returntype == null) this.returntype = TypePlaceholder.fresh(this); super.parserPostProcessing(parent); - /* - * this.returntype.parserPostProcessing(this); if(this.parameterlist != - * null){ for(FormalParameter fp : this.parameterlist){ - * fp.parserPostProcessing(this); } } for(GenericTypeVar gtv : - * this.getGenericParameter()){ gtv.parserPostProcessing(this); } - */ } @Override @@ -728,7 +586,6 @@ public class Method extends Field implements IItemWithOffset, TypeInsertable DImethod.set_Name(withSignature); ret.set_DeclId(DImethod); Block tempBlock = new Block(); - // tempBlock.setType(new RefType(parent.getName(),0)); ret.set_Block(tempBlock); ret.parserPostProcessing(parent); return ret; @@ -751,7 +608,7 @@ public class Method extends Field implements IItemWithOffset, TypeInsertable return super.equals(obj); } - public void genByteCode(ClassGenerator cg) { + public void genByteCode(ClassGenerator cg, Class classObj) { List typeInterferenceResults = cg.getTypeinferenceResults().getTypeReconstructions(this, cg); for(TypeinferenceResultSet t: typeInterferenceResults){ @@ -771,10 +628,15 @@ public class Method extends Field implements IItemWithOffset, TypeInsertable argumentNames[i] = parameter.getIdentifier(); i++; } - - if(!createdMethods.contains(argumentTypes, new ArgumentTypeEquals())){ - createdMethods.add(argumentTypes); - } + } + + String nameAndSignature = get_Method_Name()+Arrays.toString(argumentTypes); + + Logger.getLogger("nameAndSignature").error(nameAndSignature, Section.CODEGEN); + + if(classObj.methodExists(nameAndSignature)){ + Logger.getLogger("methodExists").debug(this.toString(), Section.CODEGEN); + continue; } short constants = Constants.ACC_PUBLIC; //Per Definition ist jede Methode public @@ -787,45 +649,10 @@ public class Method extends Field implements IItemWithOffset, TypeInsertable //Methode generieren und anfügen: cg.addMethod(method.createMethod(cg, getParameterList(), returnType, get_Block(), t)); + classObj.addMethod(nameAndSignature); - /* - short constants = Constants.ACC_PUBLIC; //Per Definition ist jede Methode public - if(this.modifiers != null && this.modifiers.includesModifier(new Static())) constants += Constants.ACC_STATIC; - - Type returnType = this.getType(); - - //Methode generieren: - MethodGenerator method = new MethodGenerator(constants, returnType.getBytecodeType(cg), argumentTypes , argumentNames, this.get_Method_Name(), parentClass.name, il, _cp); - - //Methode generieren und anfügen: - cg.addMethod(method.createMethod(cg, getParameterList(), returnType, get_Block())); - */ + Logger.getLogger("createMethod").debug(this.toString(), Section.CODEGEN); } } } - -class ArgumentTypeEquals implements Equal{ - - @Override - public boolean equal(org.apache.commons.bcel6.generic.Type[] a, org.apache.commons.bcel6.generic.Type[] b) { - if(a.length != b.length){ - Logger.getLogger("ArgumentTypeEquals").error("false(length)", Section.CODEGEN); - return false; - } - - for(int i = 0; i < a.length; i++){ - if(!a[i].equals(b[i])){ - String as = a[i].toString(); - String bs = b[i].toString(); - Logger.getLogger("ArgumentTypeEquals").error("false "+as+" != "+bs, Section.CODEGEN); - return false; - } - } - - Logger.getLogger("ArgumentTypeEquals").error("true", Section.CODEGEN); - - return true; - } - -} // ino.end diff --git a/test/bytecode/types/AutoOverloadingMultiResults.jav b/test/bytecode/types/AutoOverloadingMultiResults.jav new file mode 100644 index 00000000..d5c0f23f --- /dev/null +++ b/test/bytecode/types/AutoOverloadingMultiResults.jav @@ -0,0 +1,8 @@ +import java.util.Vector; + +class AutoOverloadingMultiResults{ + void method(Integer a) { + b; + b = 1; + } +} \ No newline at end of file diff --git a/test/bytecode/types/AutoOverloadingMultiResultsTest.java b/test/bytecode/types/AutoOverloadingMultiResultsTest.java new file mode 100644 index 00000000..37fa57a7 --- /dev/null +++ b/test/bytecode/types/AutoOverloadingMultiResultsTest.java @@ -0,0 +1,36 @@ +package bytecode.types; + +import static org.junit.Assert.*; + +import java.io.File; +import java.lang.reflect.Constructor; +import java.lang.reflect.Method; +import java.net.URL; +import java.net.URLClassLoader; +import java.util.Vector; + +import org.junit.Test; + +import org.junit.Ignore; + +import bytecode.SourceFileBytecodeTest; + + +public class AutoOverloadingMultiResultsTest extends SourceFileBytecodeTest{ + @Override + protected void init() { + testName = "AutoOverloadingMultiResults"; + rootDirectory = System.getProperty("user.dir")+"/test/bytecode/types/"; + } + + @Test + @Ignore + public void testConstruct() throws Exception{ + ClassLoader classLoader = getClassLoader(); + + Class cls = classLoader.loadClass(testName); + + Object obj = cls.newInstance(); + assertTrue(true); + } +} diff --git a/test/bytecode/types/AutoOverloadingVectorTest.java b/test/bytecode/types/AutoOverloadingVectorTest.java index aa4264b7..8dfe1d70 100644 --- a/test/bytecode/types/AutoOverloadingVectorTest.java +++ b/test/bytecode/types/AutoOverloadingVectorTest.java @@ -43,7 +43,7 @@ public class AutoOverloadingVectorTest extends SourceFileBytecodeTest{ URL url = file.toURL(); URL[] urls = new URL[]{url}; - Class string = classLoader.loadClass("java.lang.String"); + Class string = classLoader.loadClass("java%util%Vector%%java%lang%String%"); Class[] params = new Class[1]; params[0] = string; @@ -69,13 +69,13 @@ public class AutoOverloadingVectorTest extends SourceFileBytecodeTest{ URL url = file.toURL(); URL[] urls = new URL[]{url}; - Integer integer = new Integer(123); + Class integer = classLoader.loadClass("java%util%Vector%%java%lang%Integer%"); Class[] params = new Class[1]; - params[0] = integer.getClass(); + params[0] = integer; Method method = cls.getDeclaredMethod("method2", params); - method.invoke(obj, integer); + method.invoke(obj, integer.newInstance()); assertTrue(true); }catch(Exception e){ throw new RuntimeException(e); diff --git a/test/bytecode/types/ExtendsType.jav b/test/bytecode/types/ExtendsType.jav index 9b03238b..8bd50b6f 100644 --- a/test/bytecode/types/ExtendsType.jav +++ b/test/bytecode/types/ExtendsType.jav @@ -1,6 +1,6 @@ import java.util.Vector; -class SuperType{ +class ExtendsType{ Vector integerVector; void method() { diff --git a/test/bytecode/types/ExtendsTypeTest.java b/test/bytecode/types/ExtendsTypeTest.java index eb753389..a3560b9c 100644 --- a/test/bytecode/types/ExtendsTypeTest.java +++ b/test/bytecode/types/ExtendsTypeTest.java @@ -16,7 +16,7 @@ import bytecode.SourceFileBytecodeTest; public class ExtendsTypeTest extends SourceFileBytecodeTest{ @Override protected void init() { - testName = "SuperType"; + testName = "ExtendsType"; rootDirectory = System.getProperty("user.dir")+"/test/bytecode/types/"; } diff --git a/test/bytecode/types/ExtendsVectorStringTest.java b/test/bytecode/types/ExtendsVectorStringTest.java index d3ed9fb3..f92c61ad 100644 --- a/test/bytecode/types/ExtendsVectorStringTest.java +++ b/test/bytecode/types/ExtendsVectorStringTest.java @@ -4,6 +4,7 @@ import static org.junit.Assert.*; import java.lang.reflect.Constructor; import java.lang.reflect.Method; +import java.util.Vector; import org.junit.BeforeClass; import org.junit.Test; @@ -60,13 +61,13 @@ public class ExtendsVectorStringTest extends ASTBytecodeTest{ Object obj = cls.newInstance(); - Class stringClass = classLoader.loadClass("java.lang.Object"); + Class objectClass = classLoader.loadClass("java.lang.Object"); Class[] params = new Class[1]; - params[0] = stringClass; + params[0] = objectClass; Method method = cls.getDeclaredMethod("add", params); - method.invoke(obj, stringClass.newInstance()); + method.invoke(obj, objectClass.newInstance()); }catch(Exception e){ throw new RuntimeException(e); } @@ -83,5 +84,9 @@ public class ExtendsVectorStringTest extends ASTBytecodeTest{ public String getTestName() { return "ExtendsVectorString"; } + + class StringVector extends Vector{ + + } }