package kr.ac.snu.selab.soot.analyzer;
import java.util.HashSet;
import java.util.Map;
import java.util.Set;
import kr.ac.snu.selab.soot.graph.MetaInfo;
import kr.ac.snu.selab.soot.graph.Path;
import soot.Hierarchy;
import soot.SootClass;
import soot.SootMethod;
import soot.jimple.toolkits.callgraph.CallGraph;
public class VisitorPattern extends PatternAnalysis {
public PatternAnalysisResult perform(Map<String, SootClass> classMap, Hierarchy hierarchy, CallGraph cg, AnalysisUtil au) {
PatternAnalysisResult result = new PatternAnalysisResult();
result.setPatternName("Visitor");
Set<SootClass> interfaceTypes = au.interfaceTypes(classMap);
for (SootClass interfaceType : interfaceTypes) {
RoleRepository roles = new RoleRepository();
Map<String, MetaInfo> metaInfoMap = au.metaInfoMap(classMap.values());
Set<Path<MetaInfo>> abstractReferenceFlows = new HashSet<Path<MetaInfo>>();
abstractReferenceFlows = au.abstractReferenceFlows(interfaceType, classMap, hierarchy, cg, metaInfoMap, roles);
result.addReferenceFlowsPerType(interfaceType, abstractReferenceFlows);
au.analyzeRole(interfaceType, metaInfoMap, roles, classMap, hierarchy, cg);
for (MetaInfo metaInfoOfCaller : roles.callers()) {
for (Role role : metaInfoOfCaller.callers()) {
Caller caller = (Caller)role;
SootMethod calledMethod = caller.calledMethod();
SootClass callerClass = caller.declaringClass();
if (au.doesHaveParamSubtypeIncluding(calledMethod, callerClass, classMap, hierarchy)) {
result.addInterfaceType(interfaceType);
if (result.rolesPerType().containsKey(interfaceType)) {
result.rolesPerType().get(interfaceType).addCaller(metaInfoOfCaller);
}
else {
RoleRepository relatedRoles = new RoleRepository();
relatedRoles.addCaller(metaInfoOfCaller);
result.addRoles(interfaceType, relatedRoles);
}
}
}
}
}
return result;
}
}