From 965d1b1d7db7d231cdecfaec438165ba4ccb9c16 Mon Sep 17 00:00:00 2001 From: yeom Date: Fri, 5 Oct 2012 16:25:27 +0000 Subject: [PATCH] changes. --- Robust/src/Analysis/SSJava/FlowGraph.java | 106 +- Robust/src/Analysis/SSJava/FlowNode.java | 73 +- .../src/Analysis/SSJava/GlobalFlowGraph.java | 217 +++ .../src/Analysis/SSJava/GlobalFlowNode.java | 94 + Robust/src/Analysis/SSJava/Location.java | 6 +- .../Analysis/SSJava/LocationInference.java | 1704 +++++------------ 6 files changed, 902 insertions(+), 1298 deletions(-) create mode 100644 Robust/src/Analysis/SSJava/GlobalFlowGraph.java create mode 100644 Robust/src/Analysis/SSJava/GlobalFlowNode.java diff --git a/Robust/src/Analysis/SSJava/FlowGraph.java b/Robust/src/Analysis/SSJava/FlowGraph.java index e81fe7d0..e2783bf6 100644 --- a/Robust/src/Analysis/SSJava/FlowGraph.java +++ b/Robust/src/Analysis/SSJava/FlowGraph.java @@ -95,18 +95,13 @@ public class FlowGraph { return mapParamDescToIdx; } - public FlowNode createIntermediateNode(MethodDescriptor md) { - + public FlowNode createIntermediateNode() { NTuple tuple = new NTuple(); Descriptor interDesc = new InterDescriptor(LocationInference.INTERLOC + interseed); tuple.add(interDesc); interseed++; - NTuple locTuple = new NTuple(); - Location interLoc = new Location(md, interDesc); - locTuple.add(interLoc); - - FlowNode newNode = new FlowNode(locTuple); + FlowNode newNode = new FlowNode(tuple); newNode.setIntermediate(true); mapDescTupleToInferNode.put(tuple, newNode); @@ -140,6 +135,18 @@ public class FlowGraph { return mapIdxToFlowNode.get(idx); } + public Set getEdgeSet() { + Set edgeSet = new HashSet(); + + Set nodeSet = getNodeSet(); + for (Iterator iterator = nodeSet.iterator(); iterator.hasNext();) { + FlowNode flowNode = (FlowNode) iterator.next(); + edgeSet.addAll(getOutEdgeSet(flowNode)); + } + + return edgeSet; + } + public Set getNodeSet() { Set set = new HashSet(); set.addAll(mapDescTupleToInferNode.values()); @@ -279,10 +286,8 @@ public class FlowGraph { public FlowNode createNewFlowNode(NTuple tuple) { - NTuple locTuple = translateToLocationTuple(tuple); if (!mapDescTupleToInferNode.containsKey(tuple)) { - - FlowNode node = new FlowNode(locTuple); + FlowNode node = new FlowNode(tuple); mapDescTupleToInferNode.put(tuple, node); // nodeSet.add(node); @@ -361,6 +366,20 @@ public class FlowGraph { return set; } + public Set getReachableSetFrom(NTuple prefix) { + Set reachableSet = new HashSet(); + + Set nodeSet = getNodeSet(); + for (Iterator iterator = nodeSet.iterator(); iterator.hasNext();) { + FlowNode flowNode = (FlowNode) iterator.next(); + if (flowNode.getCurrentDescTuple().startsWith(prefix)) { + recurReachableSetFrom(flowNode, reachableSet); + } + } + + return reachableSet; + } + // private void getReachFlowNodeSetFrom(FlowNode fn, Set visited) { // // for (Iterator iterator = fn.getOutEdgeSet().iterator(); @@ -380,6 +399,20 @@ public class FlowGraph { // // } + private void recurReachableSetFrom(FlowNode curNode, Set reachableSet) { + + Set edgeSet = getOutEdgeSet(curNode); + for (Iterator iterator = edgeSet.iterator(); iterator.hasNext();) { + FlowEdge edge = (FlowEdge) iterator.next(); + FlowNode dstNode = getFlowNode(edge.getEndTuple()); + if (!reachableSet.contains(dstNode)) { + reachableSet.add(dstNode); + recurReachableSetFrom(dstNode, reachableSet); + } + } + + } + public Set> getReachableFlowTupleSet(Set> visited, FlowNode fn) { Set outEdgeSet = getOutEdgeSet(fn); @@ -387,7 +420,7 @@ public class FlowGraph { for (Iterator iterator = outEdgeSet.iterator(); iterator.hasNext();) { FlowEdge edge = (FlowEdge) iterator.next(); - if (fn.getLocTuple().equals(edge.getInitTuple())) { + if (fn.getDescTuple().equals(edge.getInitTuple())) { FlowNode dstNode = getFlowNode(edge.getEndTuple()); NTuple dstTuple = getLocationTuple(dstNode); @@ -405,51 +438,6 @@ public class FlowGraph { return getLocationTuple(getFlowNode(descTuple)); } - public NTuple translateToLocationTuple(NTuple descTuple) { - - NTuple locTuple = new NTuple(); - ClassDescriptor cd = null; - - Descriptor localDesc = descTuple.get(0); - - if (localDesc instanceof InterDescriptor) { - Location interLoc = new Location(md, localDesc); - locTuple.add(interLoc); - } else if (localDesc.getSymbol().equals(LocationInference.TOPLOC)) { - Location topLoc = new Location(md, Location.TOP); - topLoc.setLocDescriptor(LocationInference.TOPDESC); - locTuple.add(topLoc); - } else if (localDesc.getSymbol().equals(LocationInference.GLOBALLOC)) { - Location globalLoc = new Location(md, LocationInference.GLOBALLOC); - globalLoc.setLocDescriptor(LocationInference.GLOBALDESC); - locTuple.add(globalLoc); - } else { - // normal case - for (int i = 0; i < descTuple.size(); i++) { - Descriptor curDesc = descTuple.get(i); - Location loc; - if (i == 0) { - loc = new Location(md, curDesc.getSymbol()); - loc.setLocDescriptor(curDesc); - cd = ((VarDescriptor) curDesc).getType().getClassDesc(); - } else { - loc = new Location(cd, curDesc.getSymbol()); - loc.setLocDescriptor(curDesc); - - if (curDesc instanceof FieldDescriptor) { - cd = ((FieldDescriptor) curDesc).getType().getClassDesc(); - } else { - cd = ((LocationDescriptor) curDesc).getEnclosingClassDesc(); - } - - } - locTuple.add(loc); - } - } - - return locTuple; - } - public NTuple getLocationTuple(FlowNode fn) { if (!mapFlowNodeToLocTuple.containsKey(fn)) { @@ -623,8 +611,8 @@ public class FlowGraph { FlowNode u = flowEdge.getSrc(); FlowNode v = flowEdge.getDst(); - if (u.getLocTuple().equals(flowEdge.getInitTuple()) - && v.getLocTuple().equals(flowEdge.getEndTuple())) { + if (u.getDescTuple().equals(flowEdge.getInitTuple()) + && v.getDescTuple().equals(flowEdge.getEndTuple())) { // only draw an edge of the actual value flow if (!addedEdgeSet.contains(flowEdge)) { @@ -675,7 +663,7 @@ public class FlowGraph { while (iter.hasNext()) { FlowNode node = iter.next(); - if (node.getLocTuple().size() == 1) { + if (node.getDescTuple().size() == 1) { // here, we just care about the local variable if (node.getFieldNodeSet().size() > 0) { drawSubgraph(node, bw, addedEdgeSet); diff --git a/Robust/src/Analysis/SSJava/FlowNode.java b/Robust/src/Analysis/SSJava/FlowNode.java index 45de718a..cdd9bd47 100644 --- a/Robust/src/Analysis/SSJava/FlowNode.java +++ b/Robust/src/Analysis/SSJava/FlowNode.java @@ -11,7 +11,7 @@ import IR.VarDescriptor; public class FlowNode { // descriptor tuple is a unique identifier of the flow node - private NTuple locTuple; + private NTuple descTuple; // if the infer node represents the base type of field access, // this set contains fields of the base type @@ -40,26 +40,26 @@ public class FlowNode { return fieldNodeSet; } - public FlowNode(NTuple tuple) { + public FlowNode(NTuple tuple) { this.isSkeleton = false; this.isIntermediate = false; - NTuple base = null; - Location loc = null; + NTuple base = null; + Descriptor desc = null; if (tuple.size() > 1) { base = tuple.subList(0, tuple.size() - 1); - loc = tuple.get(tuple.size() - 1); + desc = tuple.get(tuple.size() - 1); } else { base = tuple; } fieldNodeSet = new HashSet(); - locTuple = new NTuple(); + descTuple = new NTuple(); if (base != null) { - locTuple.addAll(base); + descTuple.addAll(base); } - if (loc != null) { - locTuple.add(loc); + if (desc != null) { + descTuple.add(desc); } } @@ -76,8 +76,12 @@ public class FlowNode { fieldNodeSet.add(node); } - public NTuple getLocTuple() { - return locTuple; + public NTuple getDescTuple() { + return descTuple; + } + + public Descriptor getOwnDescriptor() { + return descTuple.get(descTuple.size() - 1); } public boolean isReturn() { @@ -88,24 +92,35 @@ public class FlowNode { this.isReturn = isReturn; } + public boolean isPrimitiveType() { + Descriptor desc = descTuple.get(descTuple.size() - 1); + if (desc instanceof VarDescriptor) { + return ((VarDescriptor) desc).getType().isPrimitive(); + } else if (desc instanceof FieldDescriptor) { + return ((FieldDescriptor) desc).getType().isPrimitive(); + } + return false; + } + public String toString() { String rtr = "[FlowNode]:"; if (isSkeleton()) { rtr += "SKELETON:"; } - rtr += ":" + locTuple; + rtr += ":" + descTuple; return rtr; } + public int hashCode() { - return 7 + locTuple.hashCode(); + return 7 + descTuple.hashCode(); } public boolean equals(Object obj) { if (obj instanceof FlowNode) { FlowNode in = (FlowNode) obj; - if (locTuple.equals(in.getLocTuple())) { + if (descTuple.equals(in.getDescTuple())) { return true; } } @@ -116,8 +131,8 @@ public class FlowNode { public String getID() { String id = ""; - for (int i = 0; i < locTuple.size(); i++) { - id += locTuple.get(i).getSymbol(); + for (int i = 0; i < descTuple.size(); i++) { + id += descTuple.get(i).getSymbol(); } return id; } @@ -125,11 +140,11 @@ public class FlowNode { public String getPrettyID() { String id = "<"; String property = ""; - for (int i = 0; i < locTuple.size(); i++) { + for (int i = 0; i < descTuple.size(); i++) { if (i != 0) { id += ","; } - id += locTuple.get(i).getSymbol(); + id += descTuple.get(i).getSymbol(); } id += ">"; @@ -160,22 +175,10 @@ public class FlowNode { return isDeclarationNode; } - public NTuple getCurrentLocTuple() { - if (compLoc == null) { - return locTuple; - } - NTuple curLocTuple = new NTuple(); - for (int i = 0; i < compLoc.getSize(); i++) { - Location locElement = compLoc.get(i); - curLocTuple.add(locElement); - } - return curLocTuple; - } - public NTuple getCurrentDescTuple() { if (compLoc == null) { - return getDescTuple(); + return descTuple; } NTuple curDescTuple = new NTuple(); @@ -194,12 +197,4 @@ public class FlowNode { this.isSkeleton = isSkeleton; } - public NTuple getDescTuple() { - NTuple descTuple = new NTuple(); - for (int i = 0; i < locTuple.size(); i++) { - descTuple.add(locTuple.get(i).getLocDescriptor()); - } - return descTuple; - } - } diff --git a/Robust/src/Analysis/SSJava/GlobalFlowGraph.java b/Robust/src/Analysis/SSJava/GlobalFlowGraph.java new file mode 100644 index 00000000..490ab585 --- /dev/null +++ b/Robust/src/Analysis/SSJava/GlobalFlowGraph.java @@ -0,0 +1,217 @@ +package Analysis.SSJava; + +import java.io.BufferedWriter; +import java.io.FileWriter; +import java.io.IOException; +import java.util.HashMap; +import java.util.HashSet; +import java.util.Iterator; +import java.util.Map; +import java.util.Set; + +import IR.Descriptor; +import IR.MethodDescriptor; + +public class GlobalFlowGraph { + + MethodDescriptor md; + + Map, GlobalFlowNode> mapLocTupleToNode; + Map> mapFlowNodeToOutNodeSet; + + Map mapLocationToInferCompositeLocation; + + Map, NTuple>> mapCalleeToMapCallerArgToCalleeArg; + + public GlobalFlowGraph(MethodDescriptor md) { + this.md = md; + this.mapLocTupleToNode = new HashMap, GlobalFlowNode>(); + this.mapFlowNodeToOutNodeSet = new HashMap>(); + this.mapLocationToInferCompositeLocation = new HashMap(); + this.mapCalleeToMapCallerArgToCalleeArg = + new HashMap, NTuple>>(); + } + + public GlobalFlowNode getFlowNode(NTuple locTuple) { + if (!mapLocTupleToNode.containsKey(locTuple)) { + GlobalFlowNode node = createNewGlobalFlowNode(locTuple); + mapLocTupleToNode.put(locTuple, node); + } + return mapLocTupleToNode.get(locTuple); + } + + private GlobalFlowNode createNewGlobalFlowNode(NTuple locTuple) { + GlobalFlowNode node = new GlobalFlowNode(locTuple); + return node; + } + + public void addMapLocationToInferCompositeLocation(Location loc, CompositeLocation newCompLoc) { + if (mapLocationToInferCompositeLocation.containsKey(loc)) { + // need to do a sanity check + // if the old composite location has the same prefix of the new composite location, + // replace old one with new one + // if both old and new do not share the prefix, throw error + CompositeLocation oldCompLoc = mapLocationToInferCompositeLocation.get(loc); + + if (newCompLoc.getSize() > oldCompLoc.getSize()) { + for (int i = 0; i < oldCompLoc.getSize(); i++) { + Location oldLocElement = oldCompLoc.get(i); + Location newLocElement = newCompLoc.get(i); + + if (!oldLocElement.equals(newLocElement)) { + throw new Error("Failed to generate a composite location"); + } + } + mapLocationToInferCompositeLocation.put(loc, newCompLoc); + } + + } else { + mapLocationToInferCompositeLocation.put(loc, newCompLoc); + } + } + + public CompositeLocation getCompositeLocation(Location loc) { + if (!mapLocationToInferCompositeLocation.containsKey(loc)) { + CompositeLocation compLoc = new CompositeLocation(); + compLoc.addLocation(loc); + mapLocationToInferCompositeLocation.put(loc, compLoc); + } + return mapLocationToInferCompositeLocation.get(loc); + } + + public void addValueFlowEdge(NTuple fromLocTuple, NTuple toLocTuple) { + + GlobalFlowNode fromNode = getFlowNode(fromLocTuple); + GlobalFlowNode toNode = getFlowNode(toLocTuple); + + if (!mapFlowNodeToOutNodeSet.containsKey(fromNode)) { + mapFlowNodeToOutNodeSet.put(fromNode, new HashSet()); + } + mapFlowNodeToOutNodeSet.get(fromNode).add(toNode); + + System.out.println("create a global edge from " + fromNode + " to " + toNode); + + } + + public Set getNodeSet() { + Set nodeSet = new HashSet(); + nodeSet.addAll(mapLocTupleToNode.values()); + return nodeSet; + } + + public Set getOutNodeSet(GlobalFlowNode node) { + + if (!mapFlowNodeToOutNodeSet.containsKey(node)) { + mapFlowNodeToOutNodeSet.put(node, new HashSet()); + } + + return mapFlowNodeToOutNodeSet.get(node); + } + + public void writeGraph(String suffix) { + + String graphName = "flowgraph_" + md.toString() + suffix; + graphName = graphName.replaceAll("[\\W]", ""); + + try { + BufferedWriter bw = new BufferedWriter(new FileWriter(graphName + ".dot")); + bw.write("digraph " + graphName + " {\n"); + bw.write("compound=true;\n"); + + // then visit every flow node + + // Iterator iter = nodeSet.iterator(); + Iterator iter = getNodeSet().iterator(); + + Set addedNodeSet = new HashSet(); + + while (iter.hasNext()) { + GlobalFlowNode srcNode = iter.next(); + + Set outNodeSet = getOutNodeSet(srcNode); + for (Iterator iterator = outNodeSet.iterator(); iterator.hasNext();) { + GlobalFlowNode dstNode = (GlobalFlowNode) iterator.next(); + + if (!addedNodeSet.contains(srcNode)) { + drawNode(srcNode, bw); + } + + if (!addedNodeSet.contains(dstNode)) { + drawNode(dstNode, bw); + } + + bw.write("" + srcNode.getID() + " -> " + dstNode.getID() + ";\n"); + + } + + // if (node.getDescTuple().size() == 1) { + // // here, we just care about the local variable + // if (node.getFieldNodeSet().size() > 0) { + // drawSubgraph(node, bw, addedEdgeSet); + // } + // } + // drawEdges(node, bw, addedNodeSet, addedEdgeSet); + + } + + bw.write("}\n"); + bw.close(); + + } catch (IOException e) { + e.printStackTrace(); + } + } + + private void drawNode(GlobalFlowNode node, BufferedWriter bw) throws IOException { + bw.write(node.getID() + " [label=\"" + node.getPrettyID() + "\"]" + ";\n"); + } + + public Set getIncomingNodeSet(GlobalFlowNode node) { + + Set incomingNodeSet = new HashSet(); + recurIncomingNodeSet(node, incomingNodeSet); + + return incomingNodeSet; + } + + private void recurIncomingNodeSet(GlobalFlowNode node, Set visited) { + + for (Iterator iterator = getNodeSet().iterator(); iterator.hasNext();) { + GlobalFlowNode curNode = (GlobalFlowNode) iterator.next(); + + Set outNodeSet = getOutNodeSet(curNode); + + for (Iterator iterator2 = outNodeSet.iterator(); iterator2.hasNext();) { + GlobalFlowNode outNode = (GlobalFlowNode) iterator2.next(); + + if (outNode.equals(node)) { + if (!visited.contains(curNode)) { + visited.add(curNode); + recurIncomingNodeSet(curNode, visited); + } + } + } + } + + } + + public Set getReachableNodeSetFrom(GlobalFlowNode node) { + + Set reachableNodeSet = new HashSet(); + recurReachableNodeSet(node, reachableNodeSet); + + return reachableNodeSet; + } + + private void recurReachableNodeSet(GlobalFlowNode node, Set reachableNodeSet) { + Set outNodeSet = getOutNodeSet(node); + for (Iterator iterator = outNodeSet.iterator(); iterator.hasNext();) { + GlobalFlowNode outNode = (GlobalFlowNode) iterator.next(); + if (!reachableNodeSet.contains(outNode)) { + reachableNodeSet.add(outNode); + recurReachableNodeSet(outNode, reachableNodeSet); + } + } + } + +} diff --git a/Robust/src/Analysis/SSJava/GlobalFlowNode.java b/Robust/src/Analysis/SSJava/GlobalFlowNode.java new file mode 100644 index 00000000..a1371d79 --- /dev/null +++ b/Robust/src/Analysis/SSJava/GlobalFlowNode.java @@ -0,0 +1,94 @@ +package Analysis.SSJava; + +import IR.Descriptor; + +public class GlobalFlowNode { + + NTuple locTuple; + CompositeLocation compLoc; + + public GlobalFlowNode(NTuple in) { + locTuple = in; + } + + public int hashCode() { + return locTuple.hashCode(); + } + + public NTuple getLocTuple() { + return locTuple; + } + + public boolean equals(Object obj) { + + if (obj instanceof GlobalFlowNode) { + GlobalFlowNode in = (GlobalFlowNode) obj; + if (locTuple.equals(in.getLocTuple())) { + return true; + } + } + + return false; + + } + + public String toString() { + return locTuple.toString(); + } + + public NTuple getDescTuple() { + NTuple descTuple = new NTuple(); + + for (int i = 0; i < locTuple.size(); i++) { + descTuple.add(locTuple.get(i).getLocDescriptor()); + } + + return descTuple; + } + + public String getID() { + + NTuple descTuple = getDescTuple(); + String id = ""; + for (int i = 0; i < descTuple.size(); i++) { + id += descTuple.get(i).getSymbol(); + } + return id; + } + + public String getPrettyID() { + + NTuple descTuple = getDescTuple(); + + String id = "<"; + String property = ""; + for (int i = 0; i < descTuple.size(); i++) { + if (i == 0) { + id += locTuple.get(i); + } else { + id += ","; + id += descTuple.get(i).getSymbol(); + } + } + id += ">"; + + if (compLoc != null) { + id += " " + compLoc; + } + + // if (isReturn()) { + // property += "R"; + // } + // + // if (isSkeleton()) { + // property += "S"; + // } + + if (property.length() > 0) { + property = " [" + property + "]"; + } + + return id + property; + } + +} diff --git a/Robust/src/Analysis/SSJava/Location.java b/Robust/src/Analysis/SSJava/Location.java index c91b472f..e2e25a0c 100644 --- a/Robust/src/Analysis/SSJava/Location.java +++ b/Robust/src/Analysis/SSJava/Location.java @@ -14,10 +14,10 @@ public class Location implements TypeExtension { String loc; Descriptor locDesc; - public Location(Descriptor enclosingDesc, Descriptor locDescriptor) { + public Location(Descriptor enclosingDesc, Descriptor locDesc) { this.d = enclosingDesc; - this.locDesc = locDescriptor; - this.loc = locDescriptor.getSymbol(); + this.locDesc = locDesc; + this.loc = locDesc.getSymbol(); } public Location(Descriptor d, String loc) { diff --git a/Robust/src/Analysis/SSJava/LocationInference.java b/Robust/src/Analysis/SSJava/LocationInference.java index 36645aa1..32b3a124 100644 --- a/Robust/src/Analysis/SSJava/LocationInference.java +++ b/Robust/src/Analysis/SSJava/LocationInference.java @@ -114,7 +114,7 @@ public class LocationInference { // maps a method descriptor to a sub global flow graph that captures all value flows caused by the // set of callees reachable from the method - private Map mapMethodDescriptorToSubGlobalFlowGraph; + private Map mapMethodDescriptorToSubGlobalFlowGraph; public static final String GLOBALLOC = "GLOBALLOC"; @@ -167,7 +167,7 @@ public class LocationInference { this.mapDescToLocationSummary = new HashMap(); - mapMethodDescriptorToSubGlobalFlowGraph = new HashMap(); + mapMethodDescriptorToSubGlobalFlowGraph = new HashMap(); } @@ -215,7 +215,9 @@ public class LocationInference { // 1) construct value flow graph constructFlowGraph(); - constructGlobalFlowGraph(); + assignCompositeLocation(); + + // constructGlobalFlowGraph(); System.exit(0); @@ -246,7 +248,7 @@ public class LocationInference { // 2) construct lattices inferLattices(); - simplifyLattices(); + // simplifyLattices(); // 3) check properties checkLattices(); @@ -261,6 +263,15 @@ public class LocationInference { } + private void assignCompositeLocation() { + calculateGlobalValueFlowCompositeLocation(); + translateCompositeLocationAssignmentToFlowGraph(); + } + + private void translateCompositeLocationAssignmentToFlowGraph() { + + } + private void constructGlobalFlowGraph() { System.out.println(""); @@ -278,15 +289,16 @@ public class LocationInference { FlowGraph flowGraph = getFlowGraph(md); FlowGraph subGlobalFlowGraph = flowGraph.clone(); - mapMethodDescriptorToSubGlobalFlowGraph.put(md, subGlobalFlowGraph); + + // mapMethodDescriptorToSubGlobalFlowGraph.put(md, subGlobalFlowGraph); - addValueFlowsFromCalleeSubGlobalFlowGraph(md, subGlobalFlowGraph); + // addValueFlowsFromCalleeSubGlobalFlowGraph(md, subGlobalFlowGraph); - try { - subGlobalFlowGraph.writeGraph("_SUBGLOBAL"); - } catch (IOException e) { - e.printStackTrace(); - } + // try { + // subGlobalFlowGraph.writeGraph("_SUBGLOBAL"); + // } catch (IOException e) { + // e.printStackTrace(); + // } // FlowGraph fg = new FlowGraph(md, mapParamDescToIdx); // mapMethodDescriptorToFlowGraph.put(md, fg); // analyzeMethodBody(md.getClassDesc(), md); @@ -296,156 +308,171 @@ public class LocationInference { // DEBUG: write a global flow graph MethodDescriptor mdContainingSSJavaLoop = ssjava.getMethodContainingSSJavaLoop(); - FlowGraph globalFlowGraph = getSubGlobalFlowGraph(mdContainingSSJavaLoop); + // FlowGraph globalFlowGraph = getSubGlobalFlowGraph(mdContainingSSJavaLoop); // System.out.println("GLOBAL NODE SET=" + globalFlowGraph.getNodeSet()); - assignCompositeLocation(globalFlowGraph); - try { - globalFlowGraph.writeGraph("_GLOBAL"); - } catch (IOException e) { - e.printStackTrace(); - } + // assignCompositeLocation(globalFlowGraph); + // try { + // globalFlowGraph.writeGraph("_GLOBAL"); + // } catch (IOException e) { + // e.printStackTrace(); + // } // _debug_printGraph(); } - private void assignCompositeLocation(FlowGraph globalFlowGraph) { - Set nodeSet = globalFlowGraph.getNodeSet(); - - for (Iterator iterator = nodeSet.iterator(); iterator.hasNext();) { - FlowNode flowNode = (FlowNode) iterator.next(); - Set inNodeSet = globalFlowGraph.getIncomingFlowNodeSet(flowNode); - Set reachableNodeSet = globalFlowGraph.getReachFlowNodeSetFrom(flowNode); - - // System.out.println("flowNode=" + flowNode + " incoming=" + inNodeSet); - // System.out.println("reachableNodeSet=" + reachableNodeSet); + private void calculateGlobalValueFlowCompositeLocation() { - Map, Set>> mapPrefixToIncomingLocTupleSet = - new HashMap, Set>>(); + System.out.println("SSJAVA: Calculate composite locations in the global value flow graph"); + MethodDescriptor methodDescEventLoop = ssjava.getMethodContainingSSJavaLoop(); + GlobalFlowGraph graph = getSubGlobalFlowGraph(methodDescEventLoop); - List> prefixList = new ArrayList>(); + Set> prefixSet = new HashSet>(); - for (Iterator iterator2 = inNodeSet.iterator(); iterator2.hasNext();) { - FlowNode inNode = (FlowNode) iterator2.next(); + Set nodeSet = graph.getNodeSet(); - NTuple inNodeTuple = inNode.getCurrentDescTuple(); + next: for (Iterator iterator = nodeSet.iterator(); iterator.hasNext();) { + GlobalFlowNode node = (GlobalFlowNode) iterator.next(); + Set incomingNodeSet = graph.getIncomingNodeSet(node); - // CompositeLocation inNodeInferredLoc = - // generateInferredCompositeLocation(methodInfo, inNodeTuple); - // NTuple inNodeInferredLocTuple = inNodeInferredLoc.getTuple(); + List> prefixList = calculatePrefixList(graph, node); - for (int i = 1; i < inNodeTuple.size(); i++) { - NTuple prefix = inNodeTuple.subList(0, i); - if (!prefixList.contains(prefix)) { - prefixList.add(prefix); - } - } - } + Set reachNodeSet = graph.getReachableNodeSetFrom(node); - Collections.sort(prefixList, new Comparator>() { - public int compare(NTuple arg0, NTuple arg1) { - int s0 = arg0.size(); - int s1 = arg1.size(); - if (s0 > s1) { - return -1; - } else if (s0 == s1) { - return 0; - } else { - return 1; - } - } - }); + // System.out.println("node=" + node + " inNodeSet=" + incomingNodeSet + // + " reachableNodeSet=" + reachNodeSet); - // find out reachable nodes that have the longest common prefix for (int i = 0; i < prefixList.size(); i++) { - NTuple curPrefix = prefixList.get(i); - Set> reachableCommonPrefixSet = new HashSet>(); - - for (Iterator iterator2 = reachableNodeSet.iterator(); iterator2.hasNext();) { - FlowNode reachableNode = (FlowNode) iterator2.next(); - NTuple reachLocTuple = reachableNode.getCurrentDescTuple(); - if (reachLocTuple.startsWith(curPrefix)) { - reachableCommonPrefixSet.add(reachLocTuple); + NTuple curPrefix = prefixList.get(i); + Set> reachableCommonPrefixSet = new HashSet>(); + + for (Iterator iterator2 = reachNodeSet.iterator(); iterator2.hasNext();) { + GlobalFlowNode reachNode = (GlobalFlowNode) iterator2.next(); + if (reachNode.getLocTuple().startsWith(curPrefix)) { + reachableCommonPrefixSet.add(reachNode.getLocTuple()); } } if (!reachableCommonPrefixSet.isEmpty()) { - // found reachable nodes that start with the prefix curPrefix - // need to assign a composite location - // System.out.println("-prefixList=" + prefixList); - // System.out.println("-reachableCommonPrefixSet=" + reachableCommonPrefixSet); - // System.out.println("-curPrefix=" + curPrefix); - - // first, check if there are more than one the set of locations that has - // the same length of the longest reachable prefix, no way to assign - // a composite location to the input local var - prefixSanityCheck(prefixList, i, globalFlowGraph, reachableNodeSet); - - MethodDescriptor topMethodDesc = globalFlowGraph.getMethodDescriptor(); - CompositeLocation newCompLoc = generateCompositeLocation(curPrefix, topMethodDesc); - - System.out.println("SET COMPOSITE LOCATION=" + newCompLoc + " to " + flowNode); - flowNode.setCompositeLocation(newCompLoc); + CompositeLocation newCompLoc = generateCompositeLocation(curPrefix); + System.out.println("NEED TO ASSIGN COMP LOC TO " + node + " with prefix=" + curPrefix); + System.out.println("- newCompLoc=" + newCompLoc); + // flowNode.setCompositeLocation(newCompLoc); + continue next; } + } } - + // Set inNodeSet = graph.getIncomingNodeSetWithPrefix(prefix); + // System.out.println("inNodeSet=" + inNodeSet + " from=" + node); } - private CompositeLocation generateCompositeLocation(NTuple curPrefix, - MethodDescriptor md) { - CompositeLocation newCompLoc = new CompositeLocation(); + private List> calculatePrefixList(GlobalFlowGraph graph, GlobalFlowNode node) { - Descriptor enclosingDesc = md; - for (int i = 0; i < curPrefix.size(); i++) { - Descriptor curDesc = curPrefix.get(i); - Location loc = new Location(enclosingDesc, curDesc.getSymbol()); - newCompLoc.addLocation(loc); - if (i == 0) { - VarDescriptor varDesc = (VarDescriptor) curDesc; - enclosingDesc = varDesc.getType().getClassDesc(); - } else { - FieldDescriptor fieldDesc = (FieldDescriptor) curDesc; - enclosingDesc = fieldDesc.getType().getClassDesc(); + System.out.println("\n##### calculatePrefixList=" + node); + + Set incomingNodeSet = graph.getIncomingNodeSet(node); + + List> prefixList = new ArrayList>(); + + for (Iterator iterator = incomingNodeSet.iterator(); iterator.hasNext();) { + GlobalFlowNode inNode = (GlobalFlowNode) iterator.next(); + NTuple inNodeTuple = inNode.getLocTuple(); + + for (int i = 1; i < inNodeTuple.size(); i++) { + NTuple prefix = inNodeTuple.subList(0, i); + if (!prefixList.contains(prefix)) { + prefixList.add(prefix); + } } } - LocationDescriptor newLocDescriptor = generateNewLocationDescriptor(); - newLocDescriptor.setEnclosingClassDesc((ClassDescriptor) enclosingDesc); + Collections.sort(prefixList, new Comparator>() { + public int compare(NTuple arg0, NTuple arg1) { + int s0 = arg0.size(); + int s1 = arg1.size(); + if (s0 > s1) { + return -1; + } else if (s0 == s1) { + return 0; + } else { + return 1; + } + } + }); + return prefixList; + } - Location newLoc = new Location(enclosingDesc, newLocDescriptor.getSymbol()); - newLoc.setLocDescriptor(newLocDescriptor); - newCompLoc.addLocation(newLoc); + private GlobalFlowGraph constructSubGlobalFlowGraph(FlowGraph flowGraph) { - return newCompLoc; + MethodDescriptor md = flowGraph.getMethodDescriptor(); + + GlobalFlowGraph globalGraph = new GlobalFlowGraph(md); + + // Set nodeSet = flowGraph.getNodeSet(); + Set edgeSet = flowGraph.getEdgeSet(); + + for (Iterator iterator = edgeSet.iterator(); iterator.hasNext();) { + + FlowEdge edge = (FlowEdge) iterator.next(); + NTuple srcDescTuple = edge.getInitTuple(); + NTuple dstDescTuple = edge.getEndTuple(); + + // here only keep the first element(method location) of the descriptor tuple + NTuple srcLocTuple = translateToLocTuple(md, srcDescTuple); + // Location srcMethodLoc = srcLocTuple.get(0); + // Descriptor srcVarDesc = srcMethodLoc.getLocDescriptor(); + // // if (flowGraph.isParamDesc(srcVarDesc) && (!srcVarDesc.equals(md.getThis()))) { + // if (!srcVarDesc.equals(md.getThis())) { + // srcLocTuple = new NTuple(); + // Location loc = new Location(md, srcVarDesc); + // srcLocTuple.add(loc); + // } + // + NTuple dstLocTuple = translateToLocTuple(md, dstDescTuple); + // Location dstMethodLoc = dstLocTuple.get(0); + // Descriptor dstVarDesc = dstMethodLoc.getLocDescriptor(); + // if (!dstVarDesc.equals(md.getThis())) { + // dstLocTuple = new NTuple(); + // Location loc = new Location(md, dstVarDesc); + // dstLocTuple.add(loc); + // } + + globalGraph.addValueFlowEdge(srcLocTuple, dstLocTuple); + + } + + return globalGraph; } - private void prefixSanityCheck(List> prefixList, int curIdx, - FlowGraph globalFlowGraph, Set reachableNodeSet) { + private NTuple translateToLocTuple(MethodDescriptor md, NTuple descTuple) { + + NTuple locTuple = new NTuple(); - NTuple curPrefix = prefixList.get(curIdx); + Descriptor enclosingDesc = md; + for (int i = 0; i < descTuple.size(); i++) { + Descriptor desc = descTuple.get(i); - for (int i = curIdx + 1; i < prefixList.size(); i++) { - NTuple prefixTuple = prefixList.get(i); + Location loc = new Location(enclosingDesc, desc); + locTuple.add(loc); - if (curPrefix.startsWith(prefixTuple)) { - continue; + if (desc instanceof VarDescriptor) { + enclosingDesc = ((VarDescriptor) desc).getType().getClassDesc(); + } else if (desc instanceof FieldDescriptor) { + enclosingDesc = ((FieldDescriptor) desc).getType().getClassDesc(); + } else { + // TODO: inter descriptor case + enclosingDesc = desc; } - for (Iterator iterator2 = reachableNodeSet.iterator(); iterator2.hasNext();) { - FlowNode reachableNode = (FlowNode) iterator2.next(); - NTuple reachLocTuple = reachableNode.getCurrentDescTuple(); - if (reachLocTuple.startsWith(prefixTuple)) { - throw new Error( - "Failed to generate a composite location because there is more than one prefix which is reach to the current node."); - } - } } + return locTuple; + } private void addValueFlowsFromCalleeSubGlobalFlowGraph(MethodDescriptor mdCaller, - FlowGraph subGlobalFlowGraph) { + GlobalFlowGraph subGlobalFlowGraph) { // the transformation for a call site propagates flows through parameters // if the method is virtual, it also grab all relations from any possible @@ -469,7 +496,6 @@ public class LocationInference { for (Iterator iterator2 = setPossibleCallees.iterator(); iterator2.hasNext();) { MethodDescriptor possibleMdCallee = (MethodDescriptor) iterator2.next(); propagateValueFlowsToCallerFromSubGlobalFlowGraph(min, mdCaller, possibleMdCallee); - // propagateFlowsToCallerWithNoCompositeLocation(min, mdCaller, possibleMdCallee); } } @@ -481,30 +507,204 @@ public class LocationInference { NTuple baseTuple = mapMethodInvokeNodeToBaseTuple.get(min); - FlowGraph callerSubGlobalGraph = getSubGlobalFlowGraph(mdCaller); - FlowGraph calleeSubGlobalGraph = getSubGlobalFlowGraph(possibleMdCallee); + NTuple baseLocTuple = translateToLocTuple(mdCaller, baseTuple); - int numParam = calleeSubGlobalGraph.getNumParameters(); - for (int idx = 0; idx < numParam; idx++) { - FlowNode paramNode = calleeSubGlobalGraph.getParamFlowNode(idx); - NTuple argTuple = mapMethodInvokeNodeToArgIdxMap.get(min).get(idx); - System.out.println("argTupleSet=" + argTuple + " param=" + paramNode); - // here, it adds all value flows reachable from the paramNode in the callee's flow graph - addValueFlowsFromCalleeParam(min, calleeSubGlobalGraph, paramNode, callerSubGlobalGraph, - argTuple, baseTuple); + FlowGraph calleeFlowGraph = getFlowGraph(possibleMdCallee); + + GlobalFlowGraph callerSubGlobalGraph = getSubGlobalFlowGraph(mdCaller); + GlobalFlowGraph calleeSubGlobalGraph = getSubGlobalFlowGraph(possibleMdCallee); + + Set calleeNodeSet = calleeSubGlobalGraph.getNodeSet(); + for (Iterator iterator = calleeNodeSet.iterator(); iterator.hasNext();) { + GlobalFlowNode calleeNode = (GlobalFlowNode) iterator.next(); + addValueFlowFromCalleeNode(min, mdCaller, possibleMdCallee, calleeNode); + } + + // int numParam = calleeFlowGraph.getNumParameters(); + // for (int idx = 0; idx < numParam; idx++) { + // + // FlowNode paramNode = calleeFlowGraph.getParamFlowNode(idx); + // + // NTuple paramLocTuple = + // translateToLocTuple(possibleMdCallee, paramNode.getCurrentDescTuple()); + // + // GlobalFlowNode globalParamNode = calleeSubGlobalGraph.getFlowNode(paramLocTuple); + // + // NTuple argTuple = mapMethodInvokeNodeToArgIdxMap.get(min).get(idx); + // + // NTuple argLocTuple = translateToLocTuple(mdCaller, argTuple); + // + // System.out.println("argTupleSet=" + argLocTuple + " param=" + paramLocTuple); + // // here, it adds all value flows reachable from the paramNode in the callee's flow graph + // + // addValueFlowsFromCalleeParam(mdCaller, argLocTuple, baseLocTuple, possibleMdCallee, + // globalParamNode); + // } + // + // // TODO + // // FlowGraph callerSubGlobalGraph = getSubGlobalFlowGraph(mdCaller); + // // FlowGraph calleeSubGlobalGraph = getSubGlobalFlowGraph(possibleMdCallee); + // // + // // int numParam = calleeSubGlobalGraph.getNumParameters(); + // // for (int idx = 0; idx < numParam; idx++) { + // // FlowNode paramNode = calleeSubGlobalGraph.getParamFlowNode(idx); + // // NTuple argTuple = mapMethodInvokeNodeToArgIdxMap.get(min).get(idx); + // // System.out.println("argTupleSet=" + argTuple + " param=" + paramNode); + // // // here, it adds all value flows reachable from the paramNode in the callee's flow graph + // // addValueFlowsFromCalleeParam(min, calleeSubGlobalGraph, paramNode, callerSubGlobalGraph, + // // argTuple, baseTuple); + // // } + + } + + private void addValueFlowFromCalleeNode(MethodInvokeNode min, MethodDescriptor mdCaller, + MethodDescriptor mdCallee, GlobalFlowNode calleeSrcNode) { + + GlobalFlowGraph calleeSubGlobalGraph = getSubGlobalFlowGraph(mdCallee); + GlobalFlowGraph callerSubGlobalGraph = getSubGlobalFlowGraph(mdCaller); + + NTuple srcNodeLocTuple = + translateToCallerLocTuple(min, mdCallee, mdCaller, calleeSrcNode.getLocTuple()); + + Set outNodeSet = calleeSubGlobalGraph.getOutNodeSet(calleeSrcNode); + + for (Iterator iterator = outNodeSet.iterator(); iterator.hasNext();) { + GlobalFlowNode outNode = (GlobalFlowNode) iterator.next(); + NTuple dstNodeLocTuple = + translateToCallerLocTuple(min, mdCallee, mdCaller, outNode.getLocTuple()); + callerSubGlobalGraph.addValueFlowEdge(srcNodeLocTuple, dstNodeLocTuple); + } + + } + + private NTuple translateToCallerLocTuple(MethodInvokeNode min, + MethodDescriptor mdCallee, MethodDescriptor mdCaller, NTuple nodeLocTuple) { + + FlowGraph calleeFlowGraph = getFlowGraph(mdCallee); + + NTuple nodeDescTuple = translateToDescTuple(nodeLocTuple); + + if (calleeFlowGraph.isParameter(nodeDescTuple)) { + int paramIdx = calleeFlowGraph.getParamIdx(nodeDescTuple); + NTuple argDescTuple = mapMethodInvokeNodeToArgIdxMap.get(min).get(paramIdx); + NTuple argLocTuple = translateToLocTuple(mdCaller, argDescTuple); + + NTuple callerLocTuple = new NTuple(); + + callerLocTuple.addAll(argLocTuple); + for (int i = 1; i < nodeLocTuple.size(); i++) { + callerLocTuple.add(nodeLocTuple.get(i)); + } + return callerLocTuple; + } else { + return nodeLocTuple; + } + + } + + private NTuple translateToDescTuple(NTuple locTuple) { + + NTuple descTuple = new NTuple(); + for (int i = 0; i < locTuple.size(); i++) { + descTuple.add(locTuple.get(i).getLocDescriptor()); } + return descTuple; + + } + + private void addValueFlowsFromCalleeParam(MethodDescriptor mdCaller, + NTuple argLocTuple, NTuple baseLocTuple, MethodDescriptor mdCallee, + GlobalFlowNode globalParamNode) { + + Set visited = new HashSet(); + visited.add(globalParamNode); + recurAddValueFlowsFromCalleeParam(mdCaller, argLocTuple, baseLocTuple, mdCallee, + globalParamNode); + + } + + private void recurAddValueFlowsFromCalleeParam(MethodDescriptor mdCaller, + NTuple argLocTuple, NTuple baseLocTuple, MethodDescriptor mdCallee, + GlobalFlowNode calleeCurNode) { + + // FlowGraph calleeFlowGraph = getFlowGraph(mdCallee); + // GlobalFlowGraph calleeSubGlobalGraph = getSubGlobalFlowGraph(mdCallee); + // + // NTuple curNodeLocTuple = calleeCurNode.getLocTuple(); + // NTuple curNodeDescTuple = calleeCurNode.getDescTuple(); + // if (calleeFlowGraph.isParameter(curNodeDescTuple)) { + // curNodeLocTuple = translateToCaller(argLocTuple, curNodeLocTuple); + // } + // + // Set outNodeSet = calleeSubGlobalGraph.getOutNodeSet(calleeCurNode); + // for (Iterator iterator = outNodeSet.iterator(); iterator.hasNext();) { + // GlobalFlowNode outNode = (GlobalFlowNode) iterator.next(); + // + // NTuple curNodeLocTuple = calleeCurNode.getLocTuple(); + // NTuple curNodeDescTuple = calleeCurNode.getDescTuple(); + // if (calleeFlowGraph.isParameter(curNodeDescTuple)) { + // curNodeLocTuple = translateToCaller(argLocTuple, curNodeLocTuple); + // } + // + // outNode.getDescTuple(); + // + // if (calleeFlowGraph.is) + // + // if (calleeSubGlobalGraph.isParameter(srcDescTuple)) { + // // destination node is started with 'parameter' + // // need to translate it in terms of the caller's a node + // srcDescTuple = + // translateToCaller(min, calleeSubGlobalGraph.getParamIdx(srcDescTuple), srcDescTuple); + // } + // + // } + // + // Set edgeSet = calleeSubGlobalGraph.getOutEdgeSetStartingFrom(calleeSrcNode); + // for (Iterator iterator = edgeSet.iterator(); iterator.hasNext();) { + // FlowEdge flowEdge = (FlowEdge) iterator.next(); + // + // NTuple srcDescTuple = flowEdge.getInitTuple(); + // NTuple dstDescTuple = flowEdge.getEndTuple(); + // + // FlowNode dstNode = calleeSubGlobalGraph.getFlowNode(dstDescTuple); + // + // if (calleeSubGlobalGraph.isParameter(srcDescTuple)) { + // // destination node is started with 'parameter' + // // need to translate it in terms of the caller's a node + // srcDescTuple = + // translateToCaller(min, calleeSubGlobalGraph.getParamIdx(srcDescTuple), srcDescTuple); + // } + // + // if (calleeSubGlobalGraph.isParameter(dstDescTuple)) { + // // destination node is started with 'parameter' + // // need to translate it in terms of the caller's a node + // dstDescTuple = + // translateToCaller(min, calleeSubGlobalGraph.getParamIdx(dstDescTuple), dstDescTuple); + // } + // + // callerSubGlobalGraph.addValueFlowEdge(srcDescTuple, dstDescTuple); + // + // if (!visited.contains(dstNode)) { + // visited.add(dstNode); + // recurAddValueFlowsFromCalleeParam(min, calleeSubGlobalGraph, dstNode, callerSubGlobalGraph, + // dstDescTuple, visited, baseTuple); + // } + // + // } } - private void addValueFlowsFromCalleeParam(MethodInvokeNode min, FlowGraph calleeSubGlobalGraph, - FlowNode paramNode, FlowGraph callerSubGlobalGraph, NTuple argTuple, - NTuple baseTuple) { + private NTuple translateToCaller(NTuple argLocTuple, + NTuple curNodeLocTuple) { + + NTuple callerLocTuple = new NTuple(); - Set visited = new HashSet(); + callerLocTuple.addAll(argLocTuple); + for (int i = 1; i < curNodeLocTuple.size(); i++) { + callerLocTuple.add(curNodeLocTuple.get(i)); + } - visited.add(paramNode); - recurAddValueFlowsFromCalleeParam(min, calleeSubGlobalGraph, paramNode, callerSubGlobalGraph, - argTuple, visited, baseTuple); + return callerLocTuple; } private void recurAddValueFlowsFromCalleeParam(MethodInvokeNode min, @@ -567,16 +767,20 @@ public class LocationInference { return callerTuple; } - private NTuple translateToCaller(NTuple dstDescTuple, - NTuple baseTuple) { - NTuple callerDescTuple = new NTuple(); + private NTuple traslateToCalleeParamTupleToCallerArgTuple( + NTuple calleeInitTuple, NTuple callerSrcTuple) { + + NTuple callerInitTuple = new NTuple(); + + for (int i = 0; i < callerSrcTuple.size(); i++) { + callerInitTuple.add(callerSrcTuple.get(i)); + } - callerDescTuple.addAll(baseTuple); - for (int i = 1; i < dstDescTuple.size(); i++) { - callerDescTuple.add(dstDescTuple.get(i)); + for (int i = 1; i < calleeInitTuple.size(); i++) { + callerInitTuple.add(calleeInitTuple.get(i)); } - return callerDescTuple; + return callerInitTuple; } public LocationSummary getLocationSummary(Descriptor d) { @@ -606,14 +810,16 @@ public class LocationInference { for (int paramIdx = 0; paramIdx < flowGraph.getNumParameters(); paramIdx++) { FlowNode flowNode = flowGraph.getParamFlowNode(paramIdx); - NTuple locTuple = flowNode.getLocTuple(); + NTuple descTuple = flowNode.getDescTuple(); CompositeLocation assignedCompLoc = flowNode.getCompositeLocation(); CompositeLocation inferredCompLoc; if (assignedCompLoc != null) { inferredCompLoc = translateCompositeLocation(assignedCompLoc); } else { - Location loc = locTuple.get(0); + Descriptor locDesc = descTuple.get(0); + Location loc = new Location(md, locDesc.getSymbol()); + loc.setLocDescriptor(locDesc); inferredCompLoc = new CompositeLocation(loc); } System.out.println("-paramIdx=" + paramIdx + " infer=" + inferredCompLoc); @@ -1286,7 +1492,7 @@ public class LocationInference { Set nodeSet = fg.getNodeSet(); for (Iterator iterator = nodeSet.iterator(); iterator.hasNext();) { FlowNode flowNode = (FlowNode) iterator.next(); - if (flowNode.getLocTuple().get(0).equals(md.getThis())) { + if (flowNode.getDescTuple().get(0).equals(md.getThis())) { return true; } } @@ -1377,160 +1583,6 @@ public class LocationInference { } - private void simplifyLattices() { - - setupToAnalyze(); - - while (!toAnalyzeIsEmpty()) { - ClassDescriptor cd = toAnalyzeNext(); - setupToAnalazeMethod(cd); - - SSJavaLattice classLattice = cd2lattice.get(cd); - if (classLattice != null) { - System.out.println("@@@check lattice=" + cd); - checkLatticeProperty(cd, classLattice); - } - - while (!toAnalyzeMethodIsEmpty()) { - MethodDescriptor md = toAnalyzeMethodNext(); - SSJavaLattice methodLattice = md2lattice.get(md); - if (methodLattice != null) { - System.out.println("@@@check lattice=" + md); - checkLatticeProperty(md, methodLattice); - } - } - } - - setupToAnalyze(); - - while (!toAnalyzeIsEmpty()) { - ClassDescriptor cd = toAnalyzeNext(); - - setupToAnalazeMethod(cd); - - SSJavaLattice classLattice = cd2lattice.get(cd); - if (classLattice != null) { - classLattice.removeRedundantEdges(); - } - - while (!toAnalyzeMethodIsEmpty()) { - MethodDescriptor md = toAnalyzeMethodNext(); - SSJavaLattice methodLattice = md2lattice.get(md); - if (methodLattice != null) { - methodLattice.removeRedundantEdges(); - } - } - } - - } - - private boolean checkLatticeProperty(Descriptor d, SSJavaLattice lattice) { - // if two elements has the same incoming node set, - // we need to merge two elements ... - - boolean isUpdated; - boolean isModified = false; - do { - isUpdated = removeNodeSharingSameIncomingNodes(d, lattice); - if (!isModified && isUpdated) { - isModified = true; - } - } while (isUpdated); - - return isModified; - } - - private boolean removeNodeSharingSameIncomingNodes(Descriptor d, SSJavaLattice lattice) { - LocationInfo locInfo = getLocationInfo(d); - Map> map = lattice.getIncomingElementMap(); - Set keySet = map.keySet(); - for (Iterator iterator = keySet.iterator(); iterator.hasNext();) { - String key = (String) iterator.next(); - Set incomingSetKey = map.get(key); - - // System.out.println("key=" + key + " incomingSetKey=" + - // incomingSetKey); - if (incomingSetKey.size() > 0) { - for (Iterator iterator2 = keySet.iterator(); iterator2.hasNext();) { - String cur = (String) iterator2.next(); - if (!cur.equals(key)) { - Set incomingSetCur = map.get(cur); - if (incomingSetCur.equals(incomingSetKey)) { - if (!(incomingSetCur.size() == 1 && incomingSetCur.contains(lattice.getTopItem()))) { - // NEED TO MERGE HERE!!!! - System.out.println("@@@Try merge=" + cur + " " + key); - - Set mergeSet = new HashSet(); - mergeSet.add(cur); - mergeSet.add(key); - - String newMergeLoc = "MLoc" + (SSJavaLattice.seed++); - - System.out.println("---ASSIGN NEW MERGE LOC=" + newMergeLoc + " to " + mergeSet); - lattice.mergeIntoNewLocation(mergeSet, newMergeLoc); - - for (Iterator miterator = mergeSet.iterator(); miterator.hasNext();) { - String oldLocSymbol = (String) miterator.next(); - - Set> inferLocSet = - locInfo.getRelatedInferLocSet(oldLocSymbol); - System.out.println("---update related locations=" + inferLocSet - + " oldLocSymbol=" + oldLocSymbol); - - for (Iterator miterator2 = inferLocSet.iterator(); miterator2.hasNext();) { - Pair pair = - (Pair) miterator2.next(); - Descriptor enclosingDesc = pair.getFirst(); - Descriptor desc = pair.getSecond(); - - System.out.println("---inferLoc pair=" + pair); - - CompositeLocation inferLoc = - getLocationInfo(enclosingDesc).getInferLocation(desc); - System.out.println("oldLoc=" + inferLoc); - // if (curMethodInfo.md.equals(enclosingDesc)) { - // inferLoc = curMethodInfo.getInferLocation(desc); - // } else { - // inferLoc = - // getLocationInfo(enclosingDesc).getInferLocation(desc); - // } - - Location locElement = inferLoc.get(inferLoc.getSize() - 1); - - locElement.setLocIdentifier(newMergeLoc); - locInfo.addMapLocSymbolToRelatedInferLoc(newMergeLoc, enclosingDesc, desc); - - // if (curMethodInfo.md.equals(enclosingDesc)) { - // inferLoc = curMethodInfo.getInferLocation(desc); - // } else { - // inferLoc = - // getLocationInfo(enclosingDesc).getInferLocation(desc); - // } - - inferLoc = getLocationInfo(enclosingDesc).getInferLocation(desc); - System.out.println("---New Infer Loc=" + inferLoc); - - } - - locInfo.removeRelatedInferLocSet(oldLocSymbol, newMergeLoc); - - } - - for (Iterator iterator3 = mergeSet.iterator(); iterator3.hasNext();) { - String oldLoc = (String) iterator3.next(); - lattice.remove(oldLoc); - } - return true; - } - } - } - } - } - - } - return false; - } - private void checkLattices() { LinkedList descriptorListToAnalyze = ssjava.getSortedDescriptors(); @@ -1592,77 +1644,6 @@ public class LocationInference { } private void inferLattices() { - - // do fixed-point analysis - - ssjava.init(); - LinkedList descriptorListToAnalyze = ssjava.getSortedDescriptors(); - - // Collections.sort(descriptorListToAnalyze, new - // Comparator() { - // public int compare(MethodDescriptor o1, MethodDescriptor o2) { - // return o1.getSymbol().compareToIgnoreCase(o2.getSymbol()); - // } - // }); - - // current descriptors to visit in fixed-point interprocedural analysis, - // prioritized by - // dependency in the call graph - methodDescriptorsToVisitStack.clear(); - - // descriptorListToAnalyze.removeFirst(); - - Set methodDescriptorToVistSet = new HashSet(); - methodDescriptorToVistSet.addAll(descriptorListToAnalyze); - - while (!descriptorListToAnalyze.isEmpty()) { - MethodDescriptor md = descriptorListToAnalyze.removeFirst(); - methodDescriptorsToVisitStack.add(md); - } - - // analyze scheduled methods until there are no more to visit - while (!methodDescriptorsToVisitStack.isEmpty()) { - // start to analyze leaf node - MethodDescriptor md = methodDescriptorsToVisitStack.pop(); - - SSJavaLattice methodLattice = - new SSJavaLattice(SSJavaAnalysis.TOP, SSJavaAnalysis.BOTTOM); - - MethodLocationInfo methodInfo = new MethodLocationInfo(md); - curMethodInfo = methodInfo; - - System.out.println(); - System.out.println("SSJAVA: Inferencing the lattice from " + md); - - try { - analyzeMethodLattice(md, methodLattice, methodInfo); - } catch (CyclicFlowException e) { - throw new Error("Fail to generate the method lattice for " + md); - } - - SSJavaLattice prevMethodLattice = getMethodLattice(md); - MethodLocationInfo prevMethodInfo = getMethodLocationInfo(md); - - if ((!methodLattice.equals(prevMethodLattice)) || (!methodInfo.equals(prevMethodInfo))) { - - setMethodLattice(md, methodLattice); - setMethodLocInfo(md, methodInfo); - - // results for callee changed, so enqueue dependents caller for - // further analysis - Iterator depsItr = ssjava.getDependents(md).iterator(); - while (depsItr.hasNext()) { - MethodDescriptor methodNext = depsItr.next(); - if (!methodDescriptorsToVisitStack.contains(methodNext) - && methodDescriptorToVistSet.contains(methodNext)) { - methodDescriptorsToVisitStack.add(methodNext); - } - } - - } - - } - } private void calculateExtraLocations() { @@ -1770,88 +1751,8 @@ public class LocationInference { return desc; } - private void analyzeMethodLattice(MethodDescriptor md, SSJavaLattice methodLattice, - MethodLocationInfo methodInfo) throws CyclicFlowException { - - // first take a look at method invocation nodes to newly added relations - // from the callee - analyzeLatticeMethodInvocationNode(md, methodLattice, methodInfo); - - if (!md.isStatic()) { - // set the this location - String thisLocSymbol = md.getThis().getSymbol(); - methodInfo.setThisLocName(thisLocSymbol); - } - - // set the global location - methodInfo.setGlobalLocName(LocationInference.GLOBALLOC); - methodInfo.mapDescriptorToLocation(GLOBALDESC, new CompositeLocation( - new Location(md, GLOBALLOC))); - - // visit each node of method flow graph - FlowGraph fg = getFlowGraph(md); - Set nodeSet = fg.getNodeSet(); - - // for the method lattice, we need to look at the first element of - // NTuple - for (Iterator iterator = nodeSet.iterator(); iterator.hasNext();) { - FlowNode srcNode = (FlowNode) iterator.next(); - - Set outEdgeSet = fg.getOutEdgeSet(srcNode); - for (Iterator iterator2 = outEdgeSet.iterator(); iterator2.hasNext();) { - FlowEdge outEdge = (FlowEdge) iterator2.next(); - FlowNode dstNode = outEdge.getDst(); - - NTuple srcNodeTuple = srcNode.getDescTuple(); - NTuple dstNodeTuple = dstNode.getDescTuple(); - - if (outEdge.getInitTuple().equals(srcNodeTuple) - && outEdge.getEndTuple().equals(dstNodeTuple)) { - - if ((srcNodeTuple.size() > 1 && dstNodeTuple.size() > 1) - && srcNodeTuple.get(0).equals(dstNodeTuple.get(0))) { - - // value flows between fields - Descriptor desc = srcNodeTuple.get(0); - ClassDescriptor classDesc; - - if (desc.equals(GLOBALDESC)) { - classDesc = md.getClassDesc(); - } else { - VarDescriptor varDesc = (VarDescriptor) srcNodeTuple.get(0); - classDesc = varDesc.getType().getClassDesc(); - } - extractRelationFromFieldFlows(classDesc, srcNode, dstNode, 1); - - } else { - // value flow between local var - local var or local var - field - addRelationToLattice(md, methodLattice, methodInfo, srcNode, dstNode); - } - } - } - } - - // create mapping from param idx to inferred composite location - - int offset; - if (!md.isStatic()) { - // add 'this' reference location - offset = 1; - methodInfo.addMapParamIdxToInferLoc(0, methodInfo.getInferLocation(md.getThis())); - } else { - offset = 0; - } - - for (int idx = 0; idx < md.numParameters(); idx++) { - Descriptor paramDesc = md.getParameter(idx); - CompositeLocation inferParamLoc = methodInfo.getInferLocation(paramDesc); - methodInfo.addMapParamIdxToInferLoc(idx + offset, inferParamLoc); - } - - } - - private void calculateExtraLocations(MethodDescriptor md) { - // calcualte pcloc, returnloc,... + private void calculateExtraLocations(MethodDescriptor md) { + // calcualte pcloc, returnloc,... SSJavaLattice methodLattice = getMethodLattice(md); MethodLocationInfo methodInfo = getMethodLocationInfo(md); @@ -1873,123 +1774,120 @@ public class LocationInference { Map mapParamToLoc = methodInfo.getMapParamIdxToInferLoc(); Set paramIdxSet = mapParamToLoc.keySet(); - try { - if (!ssjava.getMethodContainingSSJavaLoop().equals(md)) { - // calculate the initial program counter location - // PC location is higher than location types of all parameters - String pcLocSymbol = "PCLOC"; - - Set paramInFlowSet = new HashSet(); + if (!ssjava.getMethodContainingSSJavaLoop().equals(md)) { + // calculate the initial program counter location + // PC location is higher than location types of all parameters + String pcLocSymbol = "PCLOC"; - for (Iterator iterator = paramIdxSet.iterator(); iterator.hasNext();) { - Integer paramIdx = (Integer) iterator.next(); + Set paramInFlowSet = new HashSet(); - FlowNode paramFlowNode = fg.getParamFlowNode(paramIdx); + for (Iterator iterator = paramIdxSet.iterator(); iterator.hasNext();) { + Integer paramIdx = (Integer) iterator.next(); - if (fg.getIncomingFlowNodeSet(paramFlowNode).size() > 0) { - // parameter has in-value flows - CompositeLocation inferLoc = mapParamToLoc.get(paramIdx); - paramInFlowSet.add(inferLoc); - } - } + FlowNode paramFlowNode = fg.getParamFlowNode(paramIdx); - if (paramInFlowSet.size() > 0) { - CompositeLocation lowestLoc = getLowest(methodLattice, paramInFlowSet); - assert (lowestLoc != null); - methodInfo.setPCLoc(lowestLoc); + if (fg.getIncomingFlowNodeSet(paramFlowNode).size() > 0) { + // parameter has in-value flows + CompositeLocation inferLoc = mapParamToLoc.get(paramIdx); + paramInFlowSet.add(inferLoc); } + } + if (paramInFlowSet.size() > 0) { + CompositeLocation lowestLoc = getLowest(methodLattice, paramInFlowSet); + assert (lowestLoc != null); + methodInfo.setPCLoc(lowestLoc); } - // calculate a return location - // the return location type is lower than all parameters and location - // types - // of return values - if (!md.getReturnType().isVoid()) { - // first, generate the set of return value location types that starts - // with - // 'this' reference - - Set inferFieldReturnLocSet = new HashSet(); - - Set paramFlowNode = getParamNodeFlowingToReturnValue(md); - Set inferParamLocSet = new HashSet(); - if (paramFlowNode != null) { - for (Iterator iterator = paramFlowNode.iterator(); iterator.hasNext();) { - FlowNode fn = (FlowNode) iterator.next(); - CompositeLocation inferLoc = - generateInferredCompositeLocation(methodInfo, getFlowGraph(md).getLocationTuple(fn)); - inferParamLocSet.add(inferLoc); - } - } + } - Set returnNodeSet = fg.getReturnNodeSet(); + // calculate a return location + // the return location type is lower than all parameters and location + // types + // of return values + if (!md.getReturnType().isVoid()) { + // first, generate the set of return value location types that starts + // with + // 'this' reference - skip: for (Iterator iterator = returnNodeSet.iterator(); iterator.hasNext();) { - FlowNode returnNode = (FlowNode) iterator.next(); - CompositeLocation inferReturnLoc = - generateInferredCompositeLocation(methodInfo, fg.getLocationTuple(returnNode)); - if (inferReturnLoc.get(0).getLocIdentifier().equals("this")) { - // if the location type of the return value matches "this" reference - // then, check whether this return value is equal to/lower than all - // of - // parameters that possibly flow into the return values - for (Iterator iterator2 = inferParamLocSet.iterator(); iterator2.hasNext();) { - CompositeLocation paramInferLoc = (CompositeLocation) iterator2.next(); - - if ((!paramInferLoc.equals(inferReturnLoc)) - && !isGreaterThan(methodLattice, paramInferLoc, inferReturnLoc)) { - continue skip; - } - } - inferFieldReturnLocSet.add(inferReturnLoc); + Set inferFieldReturnLocSet = new HashSet(); - } + Set paramFlowNode = getParamNodeFlowingToReturnValue(md); + Set inferParamLocSet = new HashSet(); + if (paramFlowNode != null) { + for (Iterator iterator = paramFlowNode.iterator(); iterator.hasNext();) { + FlowNode fn = (FlowNode) iterator.next(); + CompositeLocation inferLoc = + generateInferredCompositeLocation(methodInfo, getFlowGraph(md).getLocationTuple(fn)); + inferParamLocSet.add(inferLoc); } + } - if (inferFieldReturnLocSet.size() > 0) { + Set returnNodeSet = fg.getReturnNodeSet(); - CompositeLocation returnLoc = getLowest(methodLattice, inferFieldReturnLocSet); - if (returnLoc == null) { - // in this case, assign <'this',bottom> to the RETURNLOC - returnLoc = new CompositeLocation(new Location(md, md.getThis().getSymbol())); - returnLoc.addLocation(new Location(md.getClassDesc(), getLattice(md.getClassDesc()) - .getBottomItem())); - } - methodInfo.setReturnLoc(returnLoc); + skip: for (Iterator iterator = returnNodeSet.iterator(); iterator.hasNext();) { + FlowNode returnNode = (FlowNode) iterator.next(); + CompositeLocation inferReturnLoc = + generateInferredCompositeLocation(methodInfo, fg.getLocationTuple(returnNode)); + if (inferReturnLoc.get(0).getLocIdentifier().equals("this")) { + // if the location type of the return value matches "this" reference + // then, check whether this return value is equal to/lower than all + // of + // parameters that possibly flow into the return values + for (Iterator iterator2 = inferParamLocSet.iterator(); iterator2.hasNext();) { + CompositeLocation paramInferLoc = (CompositeLocation) iterator2.next(); - } else { - String returnLocSymbol = "RETURNLOC"; - CompositeLocation returnLocInferLoc = - new CompositeLocation(new Location(md, returnLocSymbol)); - methodInfo.setReturnLoc(returnLocInferLoc); - - for (Iterator iterator = paramIdxSet.iterator(); iterator.hasNext();) { - Integer paramIdx = (Integer) iterator.next(); - CompositeLocation inferLoc = mapParamToLoc.get(paramIdx); - String paramLocLocalSymbol = inferLoc.get(0).getLocIdentifier(); - if (!methodLattice.isGreaterThan(paramLocLocalSymbol, returnLocSymbol)) { - addRelationHigherToLower(methodLattice, methodInfo, paramLocLocalSymbol, - returnLocSymbol); + if ((!paramInferLoc.equals(inferReturnLoc)) + && !isGreaterThan(methodLattice, paramInferLoc, inferReturnLoc)) { + continue skip; } } + inferFieldReturnLocSet.add(inferReturnLoc); - for (Iterator iterator = returnNodeSet.iterator(); iterator.hasNext();) { - FlowNode returnNode = (FlowNode) iterator.next(); - CompositeLocation inferLoc = - generateInferredCompositeLocation(methodInfo, fg.getLocationTuple(returnNode)); - if (!isGreaterThan(methodLattice, inferLoc, returnLocInferLoc)) { - addRelation(methodLattice, methodInfo, inferLoc, returnLocInferLoc); - } + } + } + + if (inferFieldReturnLocSet.size() > 0) { + + CompositeLocation returnLoc = getLowest(methodLattice, inferFieldReturnLocSet); + if (returnLoc == null) { + // in this case, assign <'this',bottom> to the RETURNLOC + returnLoc = new CompositeLocation(new Location(md, md.getThis().getSymbol())); + returnLoc.addLocation(new Location(md.getClassDesc(), getLattice(md.getClassDesc()) + .getBottomItem())); + } + methodInfo.setReturnLoc(returnLoc); + + } else { + String returnLocSymbol = "RETURNLOC"; + CompositeLocation returnLocInferLoc = + new CompositeLocation(new Location(md, returnLocSymbol)); + methodInfo.setReturnLoc(returnLocInferLoc); + + for (Iterator iterator = paramIdxSet.iterator(); iterator.hasNext();) { + Integer paramIdx = (Integer) iterator.next(); + CompositeLocation inferLoc = mapParamToLoc.get(paramIdx); + String paramLocLocalSymbol = inferLoc.get(0).getLocIdentifier(); + if (!methodLattice.isGreaterThan(paramLocLocalSymbol, returnLocSymbol)) { + // TODO + // addRelationHigherToLower(methodLattice, methodInfo, paramLocLocalSymbol, + // returnLocSymbol); } + } + for (Iterator iterator = returnNodeSet.iterator(); iterator.hasNext();) { + FlowNode returnNode = (FlowNode) iterator.next(); + CompositeLocation inferLoc = + generateInferredCompositeLocation(methodInfo, fg.getLocationTuple(returnNode)); + if (!isGreaterThan(methodLattice, inferLoc, returnLocInferLoc)) { + // TODO + // addRelation(methodLattice, methodInfo, inferLoc, returnLocInferLoc); + } } } - } catch (CyclicFlowException e) { - e.printStackTrace(); - } + } } private Set getHigherLocSymbolThan(SSJavaLattice lattice, String loc) { @@ -2106,50 +2004,6 @@ public class LocationInference { return false; } - private void recursiveAddRelationToLattice(int idx, MethodDescriptor md, - CompositeLocation srcInferLoc, CompositeLocation dstInferLoc) throws CyclicFlowException { - - String srcLocSymbol = srcInferLoc.get(idx).getLocIdentifier(); - String dstLocSymbol = dstInferLoc.get(idx).getLocIdentifier(); - - if (srcLocSymbol.equals(dstLocSymbol)) { - recursiveAddRelationToLattice(idx + 1, md, srcInferLoc, dstInferLoc); - } else { - - Descriptor parentDesc = srcInferLoc.get(idx).getDescriptor(); - LocationInfo locInfo = getLocationInfo(parentDesc); - - addRelationHigherToLower(getLattice(parentDesc), getLocationInfo(parentDesc), srcLocSymbol, - dstLocSymbol); - } - - } - - // private void propagateFlowsFromCallee(MethodInvokeNode min, MethodDescriptor mdCaller, - // MethodDescriptor mdCallee) { - // - // // the transformation for a call site propagates all relations between - // // parameters from the callee - // // if the method is virtual, it also grab all relations from any possible - // // callees - // - // Set setPossibleCallees = new HashSet(); - // if (mdCallee.isStatic()) { - // setPossibleCallees.add(mdCallee); - // } else { - // Set calleeSet = ssjava.getCallGraph().getMethods(mdCallee); - // // removes method descriptors that are not invoked by the caller - // calleeSet.retainAll(mapMethodToCalleeSet.get(mdCaller)); - // setPossibleCallees.addAll(calleeSet); - // } - // - // for (Iterator iterator2 = setPossibleCallees.iterator(); iterator2.hasNext();) { - // MethodDescriptor possibleMdCallee = (MethodDescriptor) iterator2.next(); - // propagateFlowsToCaller(min, mdCaller, possibleMdCallee); - // } - // - // } - private void contributeCalleeFlows(MethodInvokeNode min, MethodDescriptor mdCaller, MethodDescriptor mdCallee) { @@ -2159,7 +2013,7 @@ public class LocationInference { } - private FlowGraph getSubGlobalFlowGraph(MethodDescriptor md) { + private GlobalFlowGraph getSubGlobalFlowGraph(MethodDescriptor md) { return mapMethodDescriptorToSubGlobalFlowGraph.get(md); } @@ -2407,6 +2261,38 @@ public class LocationInference { return tuple; } + private CompositeLocation generateCompositeLocation(NTuple prefixLocTuple) { + + System.out.println("generateCompositeLocation=" + prefixLocTuple); + + CompositeLocation newCompLoc = new CompositeLocation(); + for (int i = 0; i < prefixLocTuple.size(); i++) { + newCompLoc.addLocation(prefixLocTuple.get(i)); + } + + Descriptor lastDescOfPrefix = prefixLocTuple.get(prefixLocTuple.size() - 1).getLocDescriptor(); + + ClassDescriptor enclosingDescriptor; + if (lastDescOfPrefix instanceof FieldDescriptor) { + enclosingDescriptor = ((FieldDescriptor) lastDescOfPrefix).getType().getClassDesc(); + // System.out.println("enclosingDescriptor0=" + enclosingDescriptor); + } else { + // var descriptor case + enclosingDescriptor = ((VarDescriptor) lastDescOfPrefix).getType().getClassDesc(); + } + // System.out.println("enclosingDescriptor=" + enclosingDescriptor); + + LocationDescriptor newLocDescriptor = generateNewLocationDescriptor(); + newLocDescriptor.setEnclosingClassDesc(enclosingDescriptor); + + Location newLoc = new Location(enclosingDescriptor, newLocDescriptor.getSymbol()); + newLoc.setLocDescriptor(newLocDescriptor); + newCompLoc.addLocation(newLoc); + + // System.out.println("--newCompLoc=" + newCompLoc); + return newCompLoc; + } + private CompositeLocation generateCompositeLocation(MethodDescriptor md, NTuple paramPrefix) { @@ -2604,104 +2490,6 @@ public class LocationInference { return idx; } - private void analyzeLatticeMethodInvocationNode(MethodDescriptor mdCaller, - SSJavaLattice methodLattice, MethodLocationInfo methodInfo) - throws CyclicFlowException { - - // the transformation for a call site propagates all relations between - // parameters from the callee - // if the method is virtual, it also grab all relations from any possible - // callees - - Set setMethodInvokeNode = - mapMethodDescriptorToMethodInvokeNodeSet.get(mdCaller); - - if (setMethodInvokeNode != null) { - - for (Iterator iterator = setMethodInvokeNode.iterator(); iterator.hasNext();) { - MethodInvokeNode min = (MethodInvokeNode) iterator.next(); - MethodDescriptor mdCallee = min.getMethod(); - Set setPossibleCallees = new HashSet(); - if (mdCallee.isStatic()) { - setPossibleCallees.add(mdCallee); - } else { - Set calleeSet = ssjava.getCallGraph().getMethods(mdCallee); - // removes method descriptors that are not invoked by the caller - calleeSet.retainAll(mapMethodToCalleeSet.get(mdCaller)); - setPossibleCallees.addAll(calleeSet); - } - - for (Iterator iterator2 = setPossibleCallees.iterator(); iterator2.hasNext();) { - MethodDescriptor possibleMdCallee = (MethodDescriptor) iterator2.next(); - propagateRelationToCaller(min, mdCaller, possibleMdCallee, methodLattice, methodInfo); - } - - } - } - - } - - private void propagateRelationToCaller(MethodInvokeNode min, MethodDescriptor mdCaller, - MethodDescriptor possibleMdCallee, SSJavaLattice methodLattice, - MethodLocationInfo methodInfo) throws CyclicFlowException { - - SSJavaLattice calleeLattice = getMethodLattice(possibleMdCallee); - MethodLocationInfo calleeLocInfo = getMethodLocationInfo(possibleMdCallee); - FlowGraph calleeFlowGraph = getFlowGraph(possibleMdCallee); - - int numParam = calleeLocInfo.getNumParam(); - for (int i = 0; i < numParam; i++) { - CompositeLocation param1 = calleeLocInfo.getParamCompositeLocation(i); - for (int k = 0; k < numParam; k++) { - if (i != k) { - CompositeLocation param2 = calleeLocInfo.getParamCompositeLocation(k); - - if (isGreaterThan(getLattice(possibleMdCallee), param1, param2)) { - // NodeTupleSet argDescTupleSet1 = getNodeTupleSetByArgIdx(min, i); - // NodeTupleSet argDescTupleSet2 = getNodeTupleSetByArgIdx(min, k); - NodeTupleSet argDescTupleSet1 = null; - NodeTupleSet argDescTupleSet2 = null; - - // the callee has the relation in which param1 is higher than param2 - // therefore, the caller has to have the relation in which arg1 is - // higher than arg2 - - for (Iterator> iterator = argDescTupleSet1.iterator(); iterator - .hasNext();) { - NTuple argDescTuple1 = iterator.next(); - - for (Iterator> iterator2 = argDescTupleSet2.iterator(); iterator2 - .hasNext();) { - NTuple argDescTuple2 = iterator2.next(); - - // retreive inferred location by the local var descriptor - - NTuple tuple1 = getFlowGraph(mdCaller).getLocationTuple(argDescTuple1); - NTuple tuple2 = getFlowGraph(mdCaller).getLocationTuple(argDescTuple2); - - // CompositeLocation higherInferLoc = - // methodInfo.getInferLocation(argTuple1.get(0)); - // CompositeLocation lowerInferLoc = - // methodInfo.getInferLocation(argTuple2.get(0)); - - CompositeLocation inferLoc1 = generateInferredCompositeLocation(methodInfo, tuple1); - CompositeLocation inferLoc2 = generateInferredCompositeLocation(methodInfo, tuple2); - - // addRelation(methodLattice, methodInfo, inferLoc1, inferLoc2); - - addFlowGraphEdge(mdCaller, argDescTuple1, argDescTuple2); - - } - - } - - } - } - } - } - - } - private CompositeLocation generateInferredCompositeLocation(MethodLocationInfo methodInfo, NTuple tuple) { @@ -2741,36 +2529,6 @@ public class LocationInference { return inferLoc; } - private void addRelation(SSJavaLattice methodLattice, MethodLocationInfo methodInfo, - CompositeLocation srcInferLoc, CompositeLocation dstInferLoc) throws CyclicFlowException { - - System.out.println("addRelation --- srcInferLoc=" + srcInferLoc + " dstInferLoc=" - + dstInferLoc); - String srcLocalLocSymbol = srcInferLoc.get(0).getLocIdentifier(); - String dstLocalLocSymbol = dstInferLoc.get(0).getLocIdentifier(); - - if (srcInferLoc.getSize() == 1 && dstInferLoc.getSize() == 1) { - // add a new relation to the local lattice - addRelationHigherToLower(methodLattice, methodInfo, srcLocalLocSymbol, dstLocalLocSymbol); - } else if (srcInferLoc.getSize() > 1 && dstInferLoc.getSize() > 1) { - // both src and dst have assigned to a composite location - - if (!srcLocalLocSymbol.equals(dstLocalLocSymbol)) { - addRelationHigherToLower(methodLattice, methodInfo, srcLocalLocSymbol, dstLocalLocSymbol); - } else { - recursivelyAddRelation(1, srcInferLoc, dstInferLoc); - } - } else { - // either src or dst has assigned to a composite location - if (!srcLocalLocSymbol.equals(dstLocalLocSymbol)) { - addRelationHigherToLower(methodLattice, methodInfo, srcLocalLocSymbol, dstLocalLocSymbol); - } - } - - System.out.println(); - - } - public LocationInfo getLocationInfo(Descriptor d) { if (d instanceof MethodDescriptor) { return getMethodLocationInfo((MethodDescriptor) d); @@ -2799,101 +2557,6 @@ public class LocationInference { } - private void addRelationToLattice(MethodDescriptor md, SSJavaLattice methodLattice, - MethodLocationInfo methodInfo, FlowNode srcNode, FlowNode dstNode) throws CyclicFlowException { - - System.out.println(); - System.out.println("### addRelationToLattice src=" + srcNode + " dst=" + dstNode); - - // add a new binary relation of dstNode < srcNode - FlowGraph flowGraph = getFlowGraph(md); - try { - System.out.println("***** src composite case::"); - calculateCompositeLocation(flowGraph, methodLattice, methodInfo, srcNode, null); - - CompositeLocation srcInferLoc = - generateInferredCompositeLocation(methodInfo, flowGraph.getLocationTuple(srcNode)); - CompositeLocation dstInferLoc = - generateInferredCompositeLocation(methodInfo, flowGraph.getLocationTuple(dstNode)); - addRelation(methodLattice, methodInfo, srcInferLoc, dstInferLoc); - } catch (CyclicFlowException e) { - // there is a cyclic value flow... try to calculate a composite location - // for the destination node - System.out.println("***** dst composite case::"); - calculateCompositeLocation(flowGraph, methodLattice, methodInfo, dstNode, srcNode); - CompositeLocation srcInferLoc = - generateInferredCompositeLocation(methodInfo, flowGraph.getLocationTuple(srcNode)); - CompositeLocation dstInferLoc = - generateInferredCompositeLocation(methodInfo, flowGraph.getLocationTuple(dstNode)); - try { - addRelation(methodLattice, methodInfo, srcInferLoc, dstInferLoc); - } catch (CyclicFlowException e1) { - throw new Error("Failed to merge cyclic value flows into a shared location."); - } - } - - } - - private void recursivelyAddRelation(int idx, CompositeLocation srcInferLoc, - CompositeLocation dstInferLoc) throws CyclicFlowException { - - String srcLocSymbol = srcInferLoc.get(idx).getLocIdentifier(); - String dstLocSymbol = dstInferLoc.get(idx).getLocIdentifier(); - - Descriptor parentDesc = srcInferLoc.get(idx).getDescriptor(); - - if (srcLocSymbol.equals(dstLocSymbol)) { - // check if it is the case of shared location - if (srcInferLoc.getSize() == (idx + 1) && dstInferLoc.getSize() == (idx + 1)) { - Location inferLocElement = srcInferLoc.get(idx); - System.out.println("SET SHARED LOCATION=" + inferLocElement); - getLattice(inferLocElement.getDescriptor()) - .addSharedLoc(inferLocElement.getLocIdentifier()); - } else if (srcInferLoc.getSize() > (idx + 1) && dstInferLoc.getSize() > (idx + 1)) { - recursivelyAddRelation(idx + 1, srcInferLoc, dstInferLoc); - } - } else { - addRelationHigherToLower(getLattice(parentDesc), getLocationInfo(parentDesc), srcLocSymbol, - dstLocSymbol); - } - } - - private void recursivelyAddCompositeRelation(MethodDescriptor md, FlowGraph flowGraph, - MethodLocationInfo methodInfo, FlowNode srcNode, FlowNode dstNode, Descriptor srcDesc, - Descriptor dstDesc) throws CyclicFlowException { - - CompositeLocation inferSrcLoc; - CompositeLocation inferDstLoc = methodInfo.getInferLocation(dstDesc); - - if (srcNode.getLocTuple().size() > 1) { - // field access - inferSrcLoc = new CompositeLocation(); - - NTuple locTuple = flowGraph.getLocationTuple(srcNode); - for (int i = 0; i < locTuple.size(); i++) { - inferSrcLoc.addLocation(locTuple.get(i)); - } - - } else { - inferSrcLoc = methodInfo.getInferLocation(srcDesc); - } - - if (dstNode.getLocTuple().size() > 1) { - // field access - inferDstLoc = new CompositeLocation(); - - NTuple locTuple = flowGraph.getLocationTuple(dstNode); - for (int i = 0; i < locTuple.size(); i++) { - inferDstLoc.addLocation(locTuple.get(i)); - } - - } else { - inferDstLoc = methodInfo.getInferLocation(dstDesc); - } - - recursiveAddRelationToLattice(1, md, inferSrcLoc, inferDstLoc); - } - private void addPrefixMapping(Map, Set>> map, NTuple prefix, NTuple element) { @@ -2903,254 +2566,6 @@ public class LocationInference { map.get(prefix).add(element); } - private boolean calculateCompositeLocation(FlowGraph flowGraph, - SSJavaLattice methodLattice, MethodLocationInfo methodInfo, FlowNode flowNode, - FlowNode srcNode) throws CyclicFlowException { - - Descriptor localVarDesc = flowNode.getDescTuple().get(0); - NTuple flowNodelocTuple = flowGraph.getLocationTuple(flowNode); - - if (localVarDesc.equals(methodInfo.getMethodDesc())) { - return false; - } - - Set inNodeSet = flowGraph.getIncomingFlowNodeSet(flowNode); - Set reachableNodeSet = flowGraph.getReachFlowNodeSetFrom(flowNode); - - Map, Set>> mapPrefixToIncomingLocTupleSet = - new HashMap, Set>>(); - - List> prefixList = new ArrayList>(); - - for (Iterator iterator = inNodeSet.iterator(); iterator.hasNext();) { - FlowNode inNode = (FlowNode) iterator.next(); - NTuple inNodeTuple = flowGraph.getLocationTuple(inNode); - - CompositeLocation inNodeInferredLoc = - generateInferredCompositeLocation(methodInfo, inNodeTuple); - - NTuple inNodeInferredLocTuple = inNodeInferredLoc.getTuple(); - - for (int i = 1; i < inNodeInferredLocTuple.size(); i++) { - NTuple prefix = inNodeInferredLocTuple.subList(0, i); - if (!prefixList.contains(prefix)) { - prefixList.add(prefix); - } - addPrefixMapping(mapPrefixToIncomingLocTupleSet, prefix, inNodeInferredLocTuple); - } - } - - Collections.sort(prefixList, new Comparator>() { - public int compare(NTuple arg0, NTuple arg1) { - int s0 = arg0.size(); - int s1 = arg1.size(); - if (s0 > s1) { - return -1; - } else if (s0 == s1) { - return 0; - } else { - return 1; - } - } - }); - - // System.out.println("prefixList=" + prefixList); - // System.out.println("reachableNodeSet=" + reachableNodeSet); - - // find out reachable nodes that have the longest common prefix - for (int i = 0; i < prefixList.size(); i++) { - NTuple curPrefix = prefixList.get(i); - Set> reachableCommonPrefixSet = new HashSet>(); - - for (Iterator iterator2 = reachableNodeSet.iterator(); iterator2.hasNext();) { - FlowNode reachableNode = (FlowNode) iterator2.next(); - NTuple reachLocTuple = flowGraph.getLocationTuple(reachableNode); - CompositeLocation reachLocInferLoc = - generateInferredCompositeLocation(methodInfo, reachLocTuple); - if (reachLocInferLoc.getTuple().startsWith(curPrefix)) { - reachableCommonPrefixSet.add(reachLocTuple); - } - } - - if (!reachableCommonPrefixSet.isEmpty()) { - // found reachable nodes that start with the prefix curPrefix - // need to assign a composite location - - // first, check if there are more than one the set of locations that has - // the same length of the longest reachable prefix, no way to assign - // a composite location to the input local var - // prefixSanityCheck(prefixList, i, flowGraph, reachableNodeSet); - - Set> incomingCommonPrefixSet = - mapPrefixToIncomingLocTupleSet.get(curPrefix); - - int idx = curPrefix.size(); - NTuple element = incomingCommonPrefixSet.iterator().next(); - Descriptor desc = element.get(idx).getDescriptor(); - - SSJavaLattice lattice = getLattice(desc); - LocationInfo locInfo = getLocationInfo(desc); - - CompositeLocation inferLocation = - generateInferredCompositeLocation(methodInfo, flowNodelocTuple); - - // methodInfo.getInferLocation(localVarDesc); - CompositeLocation newInferLocation = new CompositeLocation(); - - if (inferLocation.getTuple().startsWith(curPrefix)) { - // the same infer location is already existed. no need to do - // anything - System.out.println("NO ATTEMPT TO MAKE A COMPOSITE LOCATION curPrefix=" + curPrefix); - - // TODO: refactoring! - if (srcNode != null) { - CompositeLocation newLoc = new CompositeLocation(); - String newLocSymbol = "Loc" + (SSJavaLattice.seed++); - for (int locIdx = 0; locIdx < curPrefix.size(); locIdx++) { - newLoc.addLocation(curPrefix.get(locIdx)); - } - Location newLocationElement = new Location(desc, newLocSymbol); - newLoc.addLocation(newLocationElement); - - Descriptor srcLocalVar = srcNode.getDescTuple().get(0); - methodInfo.mapDescriptorToLocation(srcLocalVar, newLoc.clone()); - addMapLocSymbolToInferredLocation(methodInfo.getMethodDesc(), srcLocalVar, newLoc); - methodInfo.removeMaplocalVarToLocSet(srcLocalVar); - - // add the field/var descriptor to the set of the location symbol - int lastIdx = srcNode.getLocTuple().size() - 1; - Descriptor lastFlowNodeDesc = srcNode.getDescTuple().get(lastIdx); - NTuple srcNodelocTuple = flowGraph.getLocationTuple(srcNode); - Descriptor enclosinglastLastFlowNodeDesc = srcNodelocTuple.get(lastIdx).getDescriptor(); - - CompositeLocation newlyInferredLocForFlowNode = - generateInferredCompositeLocation(methodInfo, srcNodelocTuple); - Location lastInferLocElement = - newlyInferredLocForFlowNode.get(newlyInferredLocForFlowNode.getSize() - 1); - Descriptor enclosingLastInferLocElement = lastInferLocElement.getDescriptor(); - - // getLocationInfo(enclosingLastInferLocElement).addMapLocSymbolToDescSet( - // lastInferLocElement.getLocIdentifier(), lastFlowNodeDesc); - getLocationInfo(enclosingLastInferLocElement).addMapLocSymbolToRelatedInferLoc( - lastInferLocElement.getLocIdentifier(), enclosinglastLastFlowNodeDesc, - lastFlowNodeDesc); - - System.out.println("@@@@@@@ ASSIGN " + newLoc + " to SRC=" + srcNode); - } - - return true; - } else { - // assign a new composite location - - // String oldMethodLocationSymbol = - // inferLocation.get(0).getLocIdentifier(); - String newLocSymbol = "Loc" + (SSJavaLattice.seed++); - for (int locIdx = 0; locIdx < curPrefix.size(); locIdx++) { - newInferLocation.addLocation(curPrefix.get(locIdx)); - } - Location newLocationElement = new Location(desc, newLocSymbol); - newInferLocation.addLocation(newLocationElement); - - // maps local variable to location types of the common prefix - methodInfo.mapDescriptorToLocation(localVarDesc, newInferLocation.clone()); - - // methodInfo.mapDescriptorToLocation(localVarDesc, newInferLocation); - addMapLocSymbolToInferredLocation(methodInfo.getMethodDesc(), localVarDesc, - newInferLocation); - methodInfo.removeMaplocalVarToLocSet(localVarDesc); - - // add the field/var descriptor to the set of the location symbol - int lastIdx = flowNode.getLocTuple().size() - 1; - Descriptor lastFlowNodeDesc = flowNode.getDescTuple().get(lastIdx); - Descriptor enclosinglastLastFlowNodeDesc = flowNodelocTuple.get(lastIdx).getDescriptor(); - - CompositeLocation newlyInferredLocForFlowNode = - generateInferredCompositeLocation(methodInfo, flowNodelocTuple); - Location lastInferLocElement = - newlyInferredLocForFlowNode.get(newlyInferredLocForFlowNode.getSize() - 1); - Descriptor enclosingLastInferLocElement = lastInferLocElement.getDescriptor(); - - // getLocationInfo(enclosingLastInferLocElement).addMapLocSymbolToDescSet( - // lastInferLocElement.getLocIdentifier(), lastFlowNodeDesc); - getLocationInfo(enclosingLastInferLocElement).addMapLocSymbolToRelatedInferLoc( - lastInferLocElement.getLocIdentifier(), enclosinglastLastFlowNodeDesc, - lastFlowNodeDesc); - - // clean up the previous location - // Location prevInferLocElement = - // inferLocation.get(inferLocation.getSize() - 1); - // Descriptor prevEnclosingDesc = prevInferLocElement.getDescriptor(); - // - // SSJavaLattice targetLattice; - // LocationInfo targetInfo; - // if (prevEnclosingDesc.equals(methodInfo.getMethodDesc())) { - // targetLattice = methodLattice; - // targetInfo = methodInfo; - // } else { - // targetLattice = getLattice(prevInferLocElement.getDescriptor()); - // targetInfo = getLocationInfo(prevInferLocElement.getDescriptor()); - // } - // - // Set> associstedDescSet = - // targetInfo.getRelatedInferLocSet(prevInferLocElement.getLocIdentifier()); - // - // if (associstedDescSet.size() == 1) { - // targetLattice.remove(prevInferLocElement.getLocIdentifier()); - // } else { - // associstedDescSet.remove(lastFlowNodeDesc); - // } - - } - - System.out.println("curPrefix=" + curPrefix); - System.out.println("ASSIGN NEW COMPOSITE LOCATION =" + newInferLocation + " to " - + flowNode); - - String newlyInsertedLocName = - newInferLocation.get(newInferLocation.getSize() - 1).getLocIdentifier(); - - System.out.println("-- add in-flow"); - for (Iterator iterator = incomingCommonPrefixSet.iterator(); iterator.hasNext();) { - NTuple tuple = (NTuple) iterator.next(); - Location loc = tuple.get(idx); - String higher = loc.getLocIdentifier(); - addRelationHigherToLower(lattice, locInfo, higher, newlyInsertedLocName); - } - - System.out.println("-- add out flow"); - for (Iterator iterator = reachableCommonPrefixSet.iterator(); iterator.hasNext();) { - NTuple tuple = (NTuple) iterator.next(); - if (tuple.size() > idx) { - Location loc = tuple.get(idx); - String lower = loc.getLocIdentifier(); - // String lower = - // locInfo.getFieldInferLocation(loc.getLocDescriptor()).getLocIdentifier(); - addRelationHigherToLower(lattice, locInfo, newlyInsertedLocName, lower); - } - } - - return true; - } - - } - - return false; - - } - - private void addMapLocSymbolToInferredLocation(MethodDescriptor md, Descriptor localVar, - CompositeLocation inferLoc) { - - Location locElement = inferLoc.get((inferLoc.getSize() - 1)); - Descriptor enclosingDesc = locElement.getDescriptor(); - LocationInfo locInfo = getLocationInfo(enclosingDesc); - locInfo.addMapLocSymbolToRelatedInferLoc(locElement.getLocIdentifier(), md, localVar); - } - - private boolean isCompositeLocation(CompositeLocation cl) { - return cl.getSize() > 1; - } - private boolean containsNonPrimitiveElement(Set descSet) { for (Iterator iterator = descSet.iterator(); iterator.hasNext();) { Descriptor desc = (Descriptor) iterator.next(); @@ -3171,93 +2586,6 @@ public class LocationInference { return false; } - private void addRelationHigherToLower(SSJavaLattice lattice, LocationInfo locInfo, - String higher, String lower) throws CyclicFlowException { - - System.out.println("---addRelationHigherToLower " + higher + " -> " + lower - + " to the lattice of " + locInfo.getDescIdentifier()); - // if (higher.equals(lower) && lattice.isSharedLoc(higher)) { - // return; - // } - Set cycleElementSet = lattice.getPossibleCycleElements(higher, lower); - - boolean hasNonPrimitiveElement = false; - for (Iterator iterator = cycleElementSet.iterator(); iterator.hasNext();) { - String cycleElementLocSymbol = (String) iterator.next(); - - Set descSet = locInfo.getDescSet(cycleElementLocSymbol); - if (containsNonPrimitiveElement(descSet)) { - hasNonPrimitiveElement = true; - break; - } - } - - if (hasNonPrimitiveElement) { - System.out.println("#Check cycle= " + lower + " < " + higher + " cycleElementSet=" - + cycleElementSet); - // if there is non-primitive element in the cycle, no way to merge cyclic - // elements into the shared location - throw new CyclicFlowException(); - } - - if (cycleElementSet.size() > 0) { - - String newSharedLoc = "SharedLoc" + (SSJavaLattice.seed++); - - System.out.println("---ASSIGN NEW SHARED LOC=" + newSharedLoc + " to " + cycleElementSet); - lattice.mergeIntoNewLocation(cycleElementSet, newSharedLoc); - lattice.addSharedLoc(newSharedLoc); - - for (Iterator iterator = cycleElementSet.iterator(); iterator.hasNext();) { - String oldLocSymbol = (String) iterator.next(); - - Set> inferLocSet = locInfo.getRelatedInferLocSet(oldLocSymbol); - System.out.println("---update related locations=" + inferLocSet); - for (Iterator iterator2 = inferLocSet.iterator(); iterator2.hasNext();) { - Pair pair = (Pair) iterator2.next(); - Descriptor enclosingDesc = pair.getFirst(); - Descriptor desc = pair.getSecond(); - - CompositeLocation inferLoc; - if (curMethodInfo.md.equals(enclosingDesc)) { - inferLoc = curMethodInfo.getInferLocation(desc); - } else { - inferLoc = getLocationInfo(enclosingDesc).getInferLocation(desc); - } - - Location locElement = inferLoc.get(inferLoc.getSize() - 1); - - locElement.setLocIdentifier(newSharedLoc); - locInfo.addMapLocSymbolToRelatedInferLoc(newSharedLoc, enclosingDesc, desc); - - if (curMethodInfo.md.equals(enclosingDesc)) { - inferLoc = curMethodInfo.getInferLocation(desc); - } else { - inferLoc = getLocationInfo(enclosingDesc).getInferLocation(desc); - } - System.out.println("---New Infer Loc=" + inferLoc); - - } - locInfo.removeRelatedInferLocSet(oldLocSymbol, newSharedLoc); - - } - - lattice.addSharedLoc(newSharedLoc); - - } else if (!lattice.isGreaterThan(higher, lower)) { - lattice.addRelationHigherToLower(higher, lower); - } - } - - private void replaceOldLocWithNewLoc(SSJavaLattice methodLattice, String oldLocSymbol, - String newLocSymbol) { - - if (methodLattice.containsKey(oldLocSymbol)) { - methodLattice.substituteLocation(oldLocSymbol, newLocSymbol); - } - - } - public boolean isPrimitiveLocalVariable(FlowNode node) { VarDescriptor varDesc = (VarDescriptor) node.getDescTuple().get(0); return varDesc.getType().isPrimitive(); @@ -3316,43 +2644,6 @@ public class LocationInference { } - private void extractRelationFromFieldFlows(ClassDescriptor cd, FlowNode srcNode, - FlowNode dstNode, int idx) throws CyclicFlowException { - - if (srcNode.getLocTuple().get(idx).equals(dstNode.getLocTuple().get(idx)) - && srcNode.getLocTuple().size() > (idx + 1) && dstNode.getLocTuple().size() > (idx + 1)) { - // value flow between fields: we don't need to add a binary relation - // for this case - - Descriptor desc = srcNode.getDescTuple().get(idx); - ClassDescriptor classDesc; - - if (idx == 0) { - classDesc = ((VarDescriptor) desc).getType().getClassDesc(); - } else { - classDesc = ((FieldDescriptor) desc).getType().getClassDesc(); - } - - extractRelationFromFieldFlows(classDesc, srcNode, dstNode, idx + 1); - - } else { - - Descriptor srcFieldDesc = srcNode.getDescTuple().get(idx); - Descriptor dstFieldDesc = dstNode.getDescTuple().get(idx); - - // add a new binary relation of dstNode < srcNode - SSJavaLattice fieldLattice = getFieldLattice(cd); - LocationInfo fieldInfo = getFieldLocationInfo(cd); - - String srcSymbol = fieldInfo.getFieldInferLocation(srcFieldDesc).getLocIdentifier(); - String dstSymbol = fieldInfo.getFieldInferLocation(dstFieldDesc).getLocIdentifier(); - - addRelationHigherToLower(fieldLattice, fieldInfo, srcSymbol, dstSymbol); - - } - - } - public SSJavaLattice getFieldLattice(ClassDescriptor cd) { if (!cd2lattice.containsKey(cd)) { cd2lattice.put(cd, new SSJavaLattice(SSJavaAnalysis.TOP, SSJavaAnalysis.BOTTOM)); @@ -3450,6 +2741,16 @@ public class LocationInference { mapMethodDescriptorToFlowGraph.put(md, fg); analyzeMethodBody(md.getClassDesc(), md); + + System.out.println("##constructSubGlobalFlowGraph"); + GlobalFlowGraph subGlobalFlowGraph = constructSubGlobalFlowGraph(fg); + mapMethodDescriptorToSubGlobalFlowGraph.put(md, subGlobalFlowGraph); + + // TODO + System.out.println("##addValueFlowsFromCalleeSubGlobalFlowGraph"); + addValueFlowsFromCalleeSubGlobalFlowGraph(md, subGlobalFlowGraph); + subGlobalFlowGraph.writeGraph("_SUBGLOBAL"); + propagateFlowsFromCalleesWithNoCompositeLocation(md); // assignCompositeLocation(md); @@ -3688,7 +2989,7 @@ public class LocationInference { if (newImplicitTupleSet.size() > 1) { // need to create an intermediate node for the GLB of conditional locations & implicit flows - NTuple interTuple = getFlowGraph(md).createIntermediateNode(md).getDescTuple(); + NTuple interTuple = getFlowGraph(md).createIntermediateNode().getDescTuple(); for (Iterator> idxIter = newImplicitTupleSet.iterator(); idxIter.hasNext();) { NTuple tuple = idxIter.next(); addFlowGraphEdge(md, tuple, interTuple); @@ -3745,7 +3046,7 @@ public class LocationInference { currentFlowTupleSet.addTupleSet(implicitFlowTupleSet); if (currentFlowTupleSet.size() > 1) { - FlowNode meetNode = fg.createIntermediateNode(md); + FlowNode meetNode = fg.createIntermediateNode(); for (Iterator iterator = currentFlowTupleSet.iterator(); iterator.hasNext();) { NTuple currentFlowTuple = (NTuple) iterator.next(); fg.addValueFlowEdge(currentFlowTuple, meetNode.getDescTuple()); @@ -3776,7 +3077,7 @@ public class LocationInference { if (newImplicitTupleSet.size() > 1) { // need to create an intermediate node for the GLB of conditional locations & implicit flows - NTuple interTuple = getFlowGraph(md).createIntermediateNode(md).getDescTuple(); + NTuple interTuple = getFlowGraph(md).createIntermediateNode().getDescTuple(); for (Iterator> idxIter = newImplicitTupleSet.iterator(); idxIter .hasNext();) { NTuple tuple = idxIter.next(); @@ -3826,7 +3127,7 @@ public class LocationInference { implicitFlowTupleSet, false); // /////////// - NTuple interTuple = getFlowGraph(md).createIntermediateNode(md).getDescTuple(); + NTuple interTuple = getFlowGraph(md).createIntermediateNode().getDescTuple(); for (Iterator> idxIter = condTupleNode.iterator(); idxIter.hasNext();) { NTuple tuple = idxIter.next(); @@ -3877,7 +3178,7 @@ public class LocationInference { if (newImplicitTupleSet.size() > 1) { // need to create an intermediate node for the GLB of conditional locations & implicit flows - NTuple interTuple = getFlowGraph(md).createIntermediateNode(md).getDescTuple(); + NTuple interTuple = getFlowGraph(md).createIntermediateNode().getDescTuple(); for (Iterator> idxIter = newImplicitTupleSet.iterator(); idxIter.hasNext();) { NTuple tuple = idxIter.next(); addFlowGraphEdge(md, tuple, interTuple); @@ -3913,7 +3214,7 @@ public class LocationInference { // creates edges from RHS to LHS NTuple interTuple = null; if (nodeSetRHS.size() > 1) { - interTuple = getFlowGraph(md).createIntermediateNode(md).getDescTuple(); + interTuple = getFlowGraph(md).createIntermediateNode().getDescTuple(); } for (Iterator> iter = nodeSetRHS.iterator(); iter.hasNext();) { @@ -4079,6 +3380,8 @@ public class LocationInference { private void analyzeFlowMethodInvokeNode(MethodDescriptor md, SymbolTable nametable, MethodInvokeNode min, NodeTupleSet nodeSet, NodeTupleSet implicitFlowTupleSet) { + System.out.println("analyzeFlowMethodInvokeNode=" + min.printNode(0)); + if (nodeSet == null) { nodeSet = new NodeTupleSet(); } @@ -4126,6 +3429,7 @@ public class LocationInference { } else { // TODO Set inFlowSet = calleeFlowGraph.getIncomingFlowNodeSet(returnNode); + System.out.println("inFlowSet=" + inFlowSet + " from retrunNode=" + returnNode); for (Iterator iterator2 = inFlowSet.iterator(); iterator2.hasNext();) { FlowNode inFlowNode = (FlowNode) iterator2.next(); if (inFlowNode.getDescTuple().startsWith(calleeMethodDesc.getThis())) { @@ -4154,12 +3458,13 @@ public class LocationInference { int idx = i + offset; NodeTupleSet argTupleSet = new NodeTupleSet(); analyzeFlowExpressionNode(md, nametable, en, argTupleSet, false); - // if argument is liternal node, argTuple is set to NULL. + // if argument is liternal node, argTuple is set to NULL NTuple argTuple = new NTuple(); + System.out.println("-argTupleSet=" + argTupleSet + " from en=" + en.printNode(0)); if (argTupleSet.size() > 1) { NTuple interTuple = - getFlowGraph(md).createIntermediateNode(md).getDescTuple(); + getFlowGraph(md).createIntermediateNode().getDescTuple(); for (Iterator> idxIter = argTupleSet.iterator(); idxIter.hasNext();) { NTuple tuple = idxIter.next(); addFlowGraphEdge(md, tuple, interTuple); @@ -4182,13 +3487,18 @@ public class LocationInference { // propagateFlowsFromCallee(min, md, min.getMethod()); + System.out.println("min nodeSet=" + nodeSet); } } private boolean hasInFlowTo(FlowGraph fg, FlowNode inNode, Set nodeSet) { // return true if inNode has in-flows to nodeSet - Set reachableSet = fg.getReachFlowNodeSetFrom(inNode); + + // Set reachableSet = fg.getReachFlowNodeSetFrom(inNode); + Set reachableSet = fg.getReachableSetFrom(inNode.getDescTuple()); + System.out.println("inNode=" + inNode + " reachalbeSet=" + reachableSet); + for (Iterator iterator = reachableSet.iterator(); iterator.hasNext();) { FlowNode fn = (FlowNode) iterator.next(); if (nodeSet.contains(fn)) { @@ -4508,7 +3818,7 @@ public class LocationInference { // creates edges from RHS to LHS NTuple interTuple = null; if (nodeSetRHS.size() > 1) { - interTuple = getFlowGraph(md).createIntermediateNode(md).getDescTuple(); + interTuple = getFlowGraph(md).createIntermediateNode().getDescTuple(); } for (Iterator> iter = nodeSetRHS.iterator(); iter.hasNext();) { -- 2.34.1