java - 如何从 Soot 生成的调用图中排除 java 标准库?

标签 java static-analysis call-graph soot

我目前正在开发一个自动代码文档工具。为此,我使用 Soot 来构建调用图。然而,Soot 似乎在此调用图中包含了标准 java 库。这当然是不可取的,因为我只对我将为其生成文档的程序的实际类感兴趣。

这是我用来测试调用图的程序:

    public static void main(String[] args) throws FileNotFoundException {

        List<String> argsList = new ArrayList<String>(Arrays.asList(new String[0]));
        argsList.addAll(Arrays.asList(new String[]{
            "-w",
            "-no-bodies-for-excluded",
            "-process-dir",
            args[1], //directory with the java files
            "-src-prec",
            "java",
            "-main-class",
            args[2] //main class
        }));

        String[] trueArgs = argsList.toArray(new String[0]);
        Main.v().run(trueArgs);
        CallGraph cg = Scene.v().getCallGraph();
        visit(cg , Scene.v().getEntryPoints().get(0));
    }

使用以下函数迭代调用图(取自 this question ):

    private static void visit(CallGraph cg, SootMethod method) {
          String identifier = method.getSignature();
          visited.put(method.getSignature(), true);
          dot.drawNode(identifier);
          // iterate over unvisited parents
          Iterator<MethodOrMethodContext> ptargets = new Sources(cg.edgesInto(method));
          if (ptargets != null) {
            while (ptargets.hasNext()) {
                SootMethod parent = (SootMethod) ptargets.next();
                if (!visited.containsKey(parent.getSignature())) visit(cg, parent);
            }
          }
          // iterate over unvisited children
          Iterator<MethodOrMethodContext> ctargets = new Targets(cg.edgesOutOf(method));
          if (ctargets != null) {
            while (ctargets.hasNext()) {
               SootMethod child = (SootMethod) ctargets.next();
               dot.drawEdge(identifier, child.getSignature());
               System.out.println(method + " may call " + child);
               if (!visited.containsKey(child.getSignature())) visit(cg, child);
            }
          }
    }

但是,在测试时我一直在记录此类通话:

[...]
<callgraphs.A: void <init>()> may call <java.lang.Object: void <init>()>
<java.lang.Thread: void <init>(java.lang.ThreadGroup,java.lang.Runnable)> may call <java.lang.Object: void <clinit>()>
<java.lang.Thread: void <init>(java.lang.ThreadGroup,java.lang.String)> may call <java.lang.Object: void <clinit>()>
<java.lang.Thread: void <init>(java.lang.ThreadGroup,java.lang.String)> may call <java.lang.Object: void <init>()>
<java.lang.Thread: void <init>(java.lang.ThreadGroup,java.lang.String)> may call <java.lang.Thread: void init(java.lang.ThreadGroup,java.lang.Runnable,java.lang.String,long)>
<java.lang.Thread: void <init>(java.lang.ThreadGroup,java.lang.String)> may call <java.lang.Object: void <init>()>
[...]

随后是 java 库之间的大量调用。

有没有办法让 Soot 忽略标准 java 库?

最佳答案

好的,我找到了解决方案。

我们只是抑制来自 java 包中的类的 visit 调用。

所以通过使用child.isJavaLibraryMethod()来检查它是否来自java包。

如果它来自 java 包,我们就不会使用该类调用 visit,因此通过添加对父调用和子调用的检查并抑制输出,我们可以获得正确的调用图。 (作为奖励,它的速度要快得多,因为您不再遍历 java 库。

所以代码改为:

private static void visit(CallGraph cg, SootMethod method) {
          String identifier = method.getSignature();
          visited.put(method.getSignature(), true);
          dot.drawNode(identifier);
          // iterate over unvisited parents
          Iterator<MethodOrMethodContext> ptargets = new Sources(cg.edgesInto(method));
          if (ptargets != null) {
            while (ptargets.hasNext()) {
                SootMethod parent = (SootMethod) ptargets.next();
                if (!visited.containsKey(parent.getSignature()) && !parent.isJavaLibraryMethod()) visit(cg, parent);
            }
          }
          // iterate over unvisited children
          Iterator<MethodOrMethodContext> ctargets = new Targets(cg.edgesOutOf(method));
          if (ctargets != null) {
            while (ctargets.hasNext()) {
               SootMethod child = (SootMethod) ctargets.next();
               dot.drawEdge(identifier, child.getSignature());
               if (!child.isJavaLibraryMethod())System.out.println(method + " may call " + child);
               if (!visited.containsKey(child.getSignature()) && !child.isJavaLibraryMethod()) visit(cg, child);
            }
          }
    }

关于java - 如何从 Soot 生成的调用图中排除 java 标准库?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/59912631/

相关文章:

C# 抽象 Dispose 方法

c - 为 C 代码生成调用图

ocaml - OCaml 或 Reason 的调用图生成器

java - 使用java设置Spinner的宽度而不引用任何xml

java - Quartz 中的嵌套作业

java - 埃拉托斯特尼筛法 - HeapMemoryOutOfSpace

c++ - 在同一项目中使用 C 和 C++ 的 SourceMonitor

java - 应用程序初始化时的启动屏幕

java - 静态线程分析: Good idea?

c++ - gprof - 文件缺少调用图数据