diff --git a/src/de/dhbwstuttgart/bytecode/BytecodeGen.java b/src/de/dhbwstuttgart/bytecode/BytecodeGen.java index dcf98357..2ac10fcc 100644 --- a/src/de/dhbwstuttgart/bytecode/BytecodeGen.java +++ b/src/de/dhbwstuttgart/bytecode/BytecodeGen.java @@ -6,6 +6,7 @@ import java.util.HashSet; import java.util.Iterator; import java.util.LinkedList; import java.util.List; +import java.util.NoSuchElementException; import java.util.Optional; import de.dhbwstuttgart.exceptions.NotImplementedException; @@ -62,6 +63,8 @@ public class BytecodeGen implements ASTVisitor { private String superClass; + private ArrayList tphsClass; + // stores parameter, local vars and the next index on the local variable table, which use for aload_i, astore_i,... HashMap paramsAndLocals = new HashMap<>(); // stores generics and their bounds of class @@ -131,20 +134,44 @@ public class BytecodeGen implements ASTVisitor { superClass = classOrInterface.getSuperClass().acceptTV(new TypeToDescriptor()); resultSet = rs; tphExtractor.setResultSet(resultSet); + // Nur einmal ausführen!! if(!isVisited) { classOrInterface.accept(tphExtractor); getCommonTPHS(tphExtractor); - + + tphsClass = new ArrayList<>(); + for(TypePlaceholder t : tphExtractor.allTPHS.keySet()) { + if(!tphExtractor.allTPHS.get(t)) + tphsClass.add(t); + } + + ArrayList consClass = new ArrayList<>(); + for(TPHConstraint cons : tphExtractor.allCons) { + TypePlaceholder right = null; + for(TypePlaceholder tph : tphsClass) { + if(cons.getLeft().equals(tph.getName())) { + + consClass.add(cons); + right = getTPH(cons.getRight()); + } + } + if(right != null) { + tphsClass.add(right); + removeFromMethod(right.getName()); + right = null; + } + } String sig = null; /* if class has generics then creates signature * Signature looks like: * Superclass */ if(classOrInterface.getGenerics().iterator().hasNext() || !commonPairs.isEmpty() || - classOrInterface.getSuperClass().acceptTV(new TypeToSignature()).contains("<")) { - Signature signature = new Signature(classOrInterface, genericsAndBounds,commonPairs); + classOrInterface.getSuperClass().acceptTV(new TypeToSignature()).contains("<") + || !tphsClass.isEmpty()) { + Signature signature = new Signature(classOrInterface, genericsAndBounds,commonPairs,tphsClass, consClass); sig = signature.toString(); System.out.println("Signature: => " + sig); } @@ -176,6 +203,31 @@ public class BytecodeGen implements ASTVisitor { } + private void removeFromMethod(String name) { + for(MethodAndTPH m : tphExtractor.ListOfMethodsAndTph) { + ArrayList toRemove = new ArrayList<>(); + for(String tph : m.getTphs()) { + if(tph.equals(name)) { + toRemove.add(tph); + } + } + + if(!toRemove.isEmpty()) { + m.getTphs().removeAll(toRemove); + return; + } + } + + } + + private TypePlaceholder getTPH(String name) { + for(TypePlaceholder tph: tphExtractor.allTPHS.keySet()) { + if(tph.getName().equals(name)) + return tph; + } + throw new NoSuchElementException("TPH "+name +" does not exist"); + } + private void getCommonTPHS(TPHExtractor tphExtractor) { // Gemeinsame TPHs ArrayList cTPHs = new ArrayList<>(); @@ -228,7 +280,7 @@ public class BytecodeGen implements ASTVisitor { } String sig = null; if(hasGen) { - HashMap> constraints = Simplify.simplifyConstraints(field.name, tphExtractor); + HashMap> constraints = Simplify.simplifyConstraints(field.name, tphExtractor,tphsClass); Signature signature = new Signature(field, genericsAndBounds,methodParamsAndTypes,resultSet,constraints); sig = signature.toString(); } @@ -307,7 +359,7 @@ public class BytecodeGen implements ASTVisitor { System.out.println("ALL CONST: " + tphExtractor.allCons.size()); tphExtractor.allCons.forEach(c->System.out.println(c.toString())); System.out.println("----------------"); - HashMap> constraints = Simplify.simplifyConstraints(method.name, tphExtractor); + HashMap> constraints = Simplify.simplifyConstraints(method.name, tphExtractor, tphsClass); // ArrayList pairs = simplifyPairs(method.name,tphExtractor.allPairs,tphExtractor.allCons); Signature signature = new Signature(method, genericsAndBoundsMethod, genericsAndBounds,methodParamsAndTypes,resultSet,constraints); sig = signature.toString(); @@ -388,9 +440,21 @@ public class BytecodeGen implements ASTVisitor { @Override public void visit(Field field) { System.out.println("In Field ---"); + String des = "L"; + if(resultSet.resolveType(field.getType()).resolvedType instanceof TypePlaceholder) { + des += Type.getInternalName(Object.class); + } else { + des += resultSet.resolveType(field.getType()).resolvedType.acceptTV(new TypeToDescriptor()); + } + des +=";"; + System.out.println(des); + String sig = resultSet.resolveType(field.getType()).resolvedType.acceptTV(new TypeToSignature()); + System.out.println(sig); + if(sig.charAt(sig.length()-1) != (";").charAt(0)) { + sig +=";"; + } cw.visitField(field.modifier, field.getName(), - "L"+resultSet.resolveType(field.getType()).resolvedType.acceptTV(new TypeToDescriptor())+";", - resultSet.resolveType(field.getType()).resolvedType.acceptTV(new TypeToSignature()), + des, sig, null); } diff --git a/src/de/dhbwstuttgart/bytecode/BytecodeGenMethod.java b/src/de/dhbwstuttgart/bytecode/BytecodeGenMethod.java index 1e1048ff..6fc80347 100644 --- a/src/de/dhbwstuttgart/bytecode/BytecodeGenMethod.java +++ b/src/de/dhbwstuttgart/bytecode/BytecodeGenMethod.java @@ -708,7 +708,13 @@ public class BytecodeGenMethod implements StatementVisitor { @Override public void visit(FieldVar fieldVar) { fieldName = fieldVar.fieldVarName; - fieldDesc = "L" + getResolvedType(fieldVar.getType()) + ";"; + fieldDesc = "L"; + if(resultSet.resolveType(fieldVar.getType()).resolvedType instanceof TypePlaceholder) { + fieldDesc += Type.getInternalName(Object.class); + } else { + fieldDesc += resultSet.resolveType(fieldVar.getType()).resolvedType.acceptTV(new TypeToDescriptor()); + } + fieldDesc +=";"; fieldVar.receiver.accept(this); // test (if) @@ -1405,9 +1411,18 @@ public class BytecodeGenMethod implements StatementVisitor { // array slot onto the top of the operand stack. assignLeftSide.field.receiver.accept(this); this.rightSideTemp.accept(this); + String fDesc = "L"; + if(resultSet.resolveType(assignLeftSide.field.getType()).resolvedType instanceof TypePlaceholder) { + fDesc += Type.getInternalName(Object.class); + } else { + fDesc += resultSet.resolveType(assignLeftSide.field.getType()).resolvedType.acceptTV(new TypeToDescriptor()); + } + fDesc +=";"; + + System.out.println("Receiver = " + getResolvedType(assignLeftSide.field.receiver.getType())); mv.visitFieldInsn(Opcodes.PUTFIELD, getResolvedType(assignLeftSide.field.receiver.getType()), - assignLeftSide.field.fieldVarName, "L"+getResolvedType(assignLeftSide.field.getType())+";"); + assignLeftSide.field.fieldVarName, fDesc); } @Override diff --git a/src/de/dhbwstuttgart/bytecode/signature/Signature.java b/src/de/dhbwstuttgart/bytecode/signature/Signature.java index f74d2eef..995cdf17 100644 --- a/src/de/dhbwstuttgart/bytecode/signature/Signature.java +++ b/src/de/dhbwstuttgart/bytecode/signature/Signature.java @@ -41,11 +41,16 @@ public class Signature { private ResultSet resultSet; private ArrayList commonPairs; private HashMap> methodConstraints; + private ArrayList tphsClass; + private ArrayList consClass; - public Signature(ClassOrInterface classOrInterface, HashMap genericsAndBounds,ArrayList commonPairs) { + public Signature(ClassOrInterface classOrInterface, HashMap genericsAndBounds, + ArrayList commonPairs, ArrayList tphsClass, ArrayList consClass) { this.classOrInterface = classOrInterface; this.genericsAndBounds = genericsAndBounds; this.commonPairs = commonPairs; + this.tphsClass = tphsClass; + this.consClass = consClass; sw = new SignatureWriter(); createSignatureForClassOrInterface(); } @@ -367,7 +372,37 @@ public class Signature { GenericTypeVar g = itr.next(); getBoundsOfTypeVar(g,genericsAndBounds); } - if(!commonPairs.isEmpty()) { + + if(!consClass.isEmpty()) { + ArrayList types = new ArrayList<>(); + ArrayList superTypes = new ArrayList<>(); + + for(TPHConstraint cons : consClass) { + types.add(cons.getLeft()); + superTypes.add(cons.getRight()); + } + + for(TPHConstraint cons : consClass) { + String t = cons.getLeft()+"$"; + String bound = cons.getRight()+"$"; + sw.visitFormalTypeParameter(t); + sw.visitClassBound().visitTypeVariable(bound); + genericsAndBounds.put(t, bound); + } + + for(TPHConstraint cons : consClass) { + if(!types.contains(cons.getRight())) { + String t = cons.getRight()+"$"; + String bound = Type.getInternalName(Object.class); + sw.visitFormalTypeParameter(t); + sw.visitClassBound().visitClassType(bound); + genericsAndBounds.put(t, bound); + sw.visitClassBound().visitEnd(); + } + } + + } + /*if(!commonPairs.isEmpty()) { ArrayList types = new ArrayList<>(); ArrayList superTypes = new ArrayList<>(); @@ -395,6 +430,14 @@ public class Signature { } } } + for(TypePlaceholder t : tphsClass) { + String n = t.getName()+"$"; + String bound = Type.getInternalName(Object.class); + sw.visitFormalTypeParameter(n); + sw.visitClassBound().visitClassType(bound); + genericsAndBounds.put(n, bound); + sw.visitClassBound().visitEnd(); + }*/ String sClass = classOrInterface.getSuperClass().acceptTV(new TypeToSignature()); sw.visitSuperclass().visitClassType(sClass.substring(1, sClass.length()-1)); sw.visitEnd(); diff --git a/src/de/dhbwstuttgart/bytecode/utilities/Simplify.java b/src/de/dhbwstuttgart/bytecode/utilities/Simplify.java index bcaaf2b5..7a121fba 100644 --- a/src/de/dhbwstuttgart/bytecode/utilities/Simplify.java +++ b/src/de/dhbwstuttgart/bytecode/utilities/Simplify.java @@ -16,7 +16,7 @@ import de.dhbwstuttgart.syntaxtree.type.TypePlaceholder; public class Simplify { - public static HashMap> simplifyConstraints(String name, TPHExtractor tphExtractor) { + public static HashMap> simplifyConstraints(String name, TPHExtractor tphExtractor, ArrayList tphsClass) { // 1. check if there are any simple cycles like L set L=R and: // * remove both constraints diff --git a/test/bytecode/FieldTph.java b/test/bytecode/FieldTph.java new file mode 100644 index 00000000..3edb43d1 --- /dev/null +++ b/test/bytecode/FieldTph.java @@ -0,0 +1,43 @@ +package bytecode; + +import static org.junit.Assert.*; + +import java.io.File; +import java.lang.reflect.Field; +import java.net.URL; +import java.net.URLClassLoader; + +import org.junit.BeforeClass; +import org.junit.Test; + +import de.dhbwstuttgart.core.JavaTXCompiler; + +public class FieldTph { + + private static String path; + private static File fileToTest; + private static JavaTXCompiler compiler; + private static ClassLoader loader; + private static Class classToTest; + private static String pathToClassFile; + private static Object instanceOfClass; + + @BeforeClass + public static void setUpBeforeClass() throws Exception { + path = System.getProperty("user.dir")+"/test/bytecode/javFiles/FieldTph.jav"; + fileToTest = new File(path); + compiler = new JavaTXCompiler(fileToTest); + compiler.generateBytecode(System.getProperty("user.dir")+"/testBytecode/generatedBC/"); + pathToClassFile = System.getProperty("user.dir")+"/testBytecode/generatedBC/"; + loader = new URLClassLoader(new URL[] {new URL("file://"+pathToClassFile)}); + classToTest = loader.loadClass("FieldTph"); + instanceOfClass = classToTest.getDeclaredConstructor().newInstance(); + } + + @Test + public void test() { + Field[] fields = classToTest.getFields(); + assertEquals(1, fields.length); + } + +} diff --git a/test/bytecode/FieldTph2Test.java b/test/bytecode/FieldTph2Test.java new file mode 100644 index 00000000..2ed19d33 --- /dev/null +++ b/test/bytecode/FieldTph2Test.java @@ -0,0 +1,52 @@ +package bytecode; + +import static org.junit.Assert.*; + +import java.io.File; +import java.lang.reflect.Field; +import java.lang.reflect.Method; +import java.net.URL; +import java.net.URLClassLoader; + +import org.junit.BeforeClass; +import org.junit.Test; + +import de.dhbwstuttgart.core.JavaTXCompiler; + +public class FieldTph2Test { + + private static String path; + private static File fileToTest; + private static JavaTXCompiler compiler; + private static ClassLoader loader; + private static Class classToTest; + private static String pathToClassFile; + private static Object instanceOfClass; + + @BeforeClass + public static void setUpBeforeClass() throws Exception { + path = System.getProperty("user.dir")+"/test/bytecode/javFiles/FieldTph2.jav"; + fileToTest = new File(path); + compiler = new JavaTXCompiler(fileToTest); + compiler.generateBytecode(System.getProperty("user.dir")+"/testBytecode/generatedBC/"); + pathToClassFile = System.getProperty("user.dir")+"/testBytecode/generatedBC/"; + loader = new URLClassLoader(new URL[] {new URL("file://"+pathToClassFile)}); + classToTest = loader.loadClass("FieldTph2"); + instanceOfClass = classToTest.getDeclaredConstructor().newInstance(); + } + + @Test + public void test() throws Exception { + Field a = classToTest.getDeclaredField("a"); + a.setAccessible(true); + + Method m2 = classToTest.getDeclaredMethod("m2", Object.class); + m2.invoke(instanceOfClass, 1); + + Method m = classToTest.getDeclaredMethod("m", Object.class); + Object result = m.invoke(instanceOfClass, 1); + + assertEquals(1,result); + } + +} diff --git a/test/bytecode/javFiles/FieldTph.jav b/test/bytecode/javFiles/FieldTph.jav new file mode 100644 index 00000000..fc74e539 --- /dev/null +++ b/test/bytecode/javFiles/FieldTph.jav @@ -0,0 +1,4 @@ +public class FieldTph { + a; + +} \ No newline at end of file diff --git a/test/bytecode/javFiles/FieldTph2.jav b/test/bytecode/javFiles/FieldTph2.jav new file mode 100644 index 00000000..7d60b683 --- /dev/null +++ b/test/bytecode/javFiles/FieldTph2.jav @@ -0,0 +1,12 @@ +public class FieldTph2 { + a; + + m(b){ + b = a; + return b; + } + + m2(c){ + a = c; + } +} \ No newline at end of file diff --git a/test/bytecode/simplifyalgo/CycleTest.java b/test/bytecode/simplifyalgo/CycleTest.java index 88697e6d..f847eff4 100644 --- a/test/bytecode/simplifyalgo/CycleTest.java +++ b/test/bytecode/simplifyalgo/CycleTest.java @@ -5,6 +5,7 @@ package bytecode.simplifyalgo; import static org.junit.Assert.*; +import java.util.ArrayList; import java.util.HashMap; import java.util.HashSet; @@ -68,7 +69,7 @@ public class CycleTest { TPHConstraint k = new ExtendsConstraint("A", Type.getInternalName(Object.class), Relation.EXTENDS); result.put(k, equals); - HashMap> sim = Simplify.simplifyConstraints(methName, tphExtractor); + HashMap> sim = Simplify.simplifyConstraints(methName, tphExtractor,new ArrayList<>()); boolean areEquals = SimpleCycle.areMapsEqual(result, sim); assertTrue(areEquals); } diff --git a/test/bytecode/simplifyalgo/SameLeftSide.java b/test/bytecode/simplifyalgo/SameLeftSide.java index 402fe1e1..69d50c63 100644 --- a/test/bytecode/simplifyalgo/SameLeftSide.java +++ b/test/bytecode/simplifyalgo/SameLeftSide.java @@ -2,6 +2,7 @@ package bytecode.simplifyalgo; import static org.junit.Assert.*; +import java.util.ArrayList; import java.util.HashMap; import java.util.HashSet; @@ -76,7 +77,7 @@ public class SameLeftSide { result.put(b, hs); - HashMap> sim = Simplify.simplifyConstraints(methName, tphExtractor); + HashMap> sim = Simplify.simplifyConstraints(methName, tphExtractor,new ArrayList<>()); boolean areEquals = SimpleCycle.areMapsEqual(result, sim); assertTrue(areEquals); } @@ -93,7 +94,7 @@ public class SameLeftSide { hs.add("B"); result.put(c, hs); - HashMap> sim = Simplify.simplifyConstraints(methName2, tphExtractor); + HashMap> sim = Simplify.simplifyConstraints(methName2, tphExtractor,new ArrayList<>()); boolean areEquals = SimpleCycle.areMapsEqual(result, sim); assertTrue(areEquals); } diff --git a/test/bytecode/simplifyalgo/SimpleCycle.java b/test/bytecode/simplifyalgo/SimpleCycle.java index 636eb026..545b0514 100644 --- a/test/bytecode/simplifyalgo/SimpleCycle.java +++ b/test/bytecode/simplifyalgo/SimpleCycle.java @@ -2,6 +2,7 @@ package bytecode.simplifyalgo; import static org.junit.Assert.*; +import java.util.ArrayList; import java.util.HashMap; import java.util.HashSet; @@ -45,7 +46,7 @@ public class SimpleCycle { TPHConstraint k = new ExtendsConstraint("B", Type.getInternalName(Object.class), Relation.EXTENDS); result.put(k, equals); - HashMap> sim = Simplify.simplifyConstraints(methName, tphExtractor); + HashMap> sim = Simplify.simplifyConstraints(methName, tphExtractor,new ArrayList<>()); boolean areEquals = areMapsEqual(result, sim); assertTrue(areEquals); }