-
Notifications
You must be signed in to change notification settings - Fork 196
Expand file tree
/
Copy pathPointerAnalysis.java
More file actions
209 lines (197 loc) · 9.54 KB
/
PointerAnalysis.java
File metadata and controls
209 lines (197 loc) · 9.54 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
/*
* Tai-e: A Static Analysis Framework for Java
*
* Copyright (C) 2022 Tian Tan <tiantan@nju.edu.cn>
* Copyright (C) 2022 Yue Li <yueli@nju.edu.cn>
*
* This file is part of Tai-e.
*
* Tai-e is free software: you can redistribute it and/or modify
* it under the terms of the GNU Lesser General Public License
* as published by the Free Software Foundation, either version 3
* of the License, or (at your option) any later version.
*
* Tai-e is distributed in the hope that it will be useful,but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General
* Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with Tai-e. If not, see <https://www.gnu.org/licenses/>.
*/
package pascal.taie.analysis.pta;
import org.apache.logging.log4j.Level;
import pascal.taie.World;
import pascal.taie.analysis.ProgramAnalysis;
import pascal.taie.analysis.pta.core.cs.element.MapBasedCSManager;
import pascal.taie.analysis.pta.core.cs.selector.ContextSelector;
import pascal.taie.analysis.pta.core.cs.selector.ContextSelectorFactory;
import pascal.taie.analysis.pta.core.heap.AllocationSiteBasedModel;
import pascal.taie.analysis.pta.core.heap.HeapModel;
import pascal.taie.analysis.pta.core.solver.CutShortcutSolver;
import pascal.taie.analysis.pta.core.solver.DefaultSolver;
import pascal.taie.analysis.pta.core.solver.Solver;
import pascal.taie.analysis.pta.plugin.AnalysisTimer;
import pascal.taie.analysis.pta.plugin.ClassInitializer;
import pascal.taie.analysis.pta.plugin.CompositePlugin;
import pascal.taie.analysis.pta.plugin.EntryPointHandler;
import pascal.taie.analysis.pta.plugin.Plugin;
import pascal.taie.analysis.pta.plugin.ReferenceHandler;
import pascal.taie.analysis.pta.plugin.ResultProcessor;
import pascal.taie.analysis.pta.plugin.ThreadHandler;
import pascal.taie.analysis.pta.plugin.cutshortcut.container.ContainerAccessHandler;
import pascal.taie.analysis.pta.plugin.cutshortcut.container.MakeDefaultContainerConfig;
import pascal.taie.analysis.pta.plugin.cutshortcut.field.FieldAccessHandler;
import pascal.taie.analysis.pta.plugin.cutshortcut.localflow.LocalFlowHandler;
import pascal.taie.analysis.pta.plugin.exception.ExceptionAnalysis;
import pascal.taie.analysis.pta.plugin.invokedynamic.InvokeDynamicAnalysis;
import pascal.taie.analysis.pta.plugin.invokedynamic.Java9StringConcatHandler;
import pascal.taie.analysis.pta.plugin.invokedynamic.LambdaAnalysis;
import pascal.taie.analysis.pta.plugin.natives.NativeModeller;
import pascal.taie.analysis.pta.plugin.reflection.ReflectionAnalysis;
import pascal.taie.analysis.pta.plugin.taint.TaintAnalysis;
import pascal.taie.analysis.pta.toolkit.CollectionMethods;
import pascal.taie.analysis.pta.toolkit.mahjong.Mahjong;
import pascal.taie.analysis.pta.toolkit.scaler.Scaler;
import pascal.taie.analysis.pta.toolkit.zipper.Zipper;
import pascal.taie.config.AnalysisConfig;
import pascal.taie.config.AnalysisOptions;
import pascal.taie.config.ConfigException;
import pascal.taie.util.AnalysisException;
import pascal.taie.util.Monitor;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;
import java.util.List;
public class PointerAnalysis extends ProgramAnalysis<PointerAnalysisResult> {
public static final String ID = "pta";
public PointerAnalysis(AnalysisConfig config) {
super(config);
}
@Override
public PointerAnalysisResult analyze() {
AnalysisOptions options = getOptions();
HeapModel heapModel = new AllocationSiteBasedModel(options);
ContextSelector selector = null;
String advanced = options.getString("advanced");
String solverType = options.has("solver") ? options.getString("solver"): "default"; // which solver to use
String cs = options.getString("cs");
if (advanced != null) {
if (advanced.equals("collection"))
selector = ContextSelectorFactory.makeSelectiveSelector(cs,
new CollectionMethods(World.get().getClassHierarchy()).get());
else {
// run context-insensitive analysis as pre-analysis
PointerAnalysisResult preResult = runAnalysis(heapModel, ContextSelectorFactory.makeCISelector(), solverType);
if (advanced.startsWith("scaler"))
selector = Monitor.runAndCount(() -> ContextSelectorFactory
.makeGuidedSelector(Scaler.run(preResult, advanced)),
"Scaler", Level.INFO);
else if (advanced.startsWith("zipper"))
selector = Monitor.runAndCount(() -> ContextSelectorFactory
.makeSelectiveSelector(cs, Zipper.run(preResult, advanced)),
"Zipper", Level.INFO);
else if (advanced.equals("mahjong"))
heapModel = Monitor.runAndCount(() -> Mahjong.run(preResult, options),
"Mahjong", Level.INFO);
else
throw new IllegalArgumentException(
"Illegal advanced analysis argument: " + advanced);
}
}
if (selector == null) {
selector = ContextSelectorFactory.makePlainSelector(cs);
}
return runAnalysis(heapModel, selector, solverType);
}
private PointerAnalysisResult runAnalysis(HeapModel heapModel,
ContextSelector selector,
String solverType) {
AnalysisOptions options = getOptions();
Solver solver;
if (solverType.equals("default"))
solver = new DefaultSolver(options, heapModel, selector, new MapBasedCSManager());
// cut-shortcut
else if (solverType.equals("csc"))
solver = new CutShortcutSolver(options, heapModel, selector, new MapBasedCSManager());
else
throw new IllegalArgumentException("Illegal solver type: " + solverType);
// The initialization of some Plugins may read the fields in solver,
// e.g., contextSelector or csManager, thus we initialize Plugins
// after setting all other fields of solver.setPlugin(solver, options);
setPlugin(solver, options, solverType);
solver.solve();
return solver.getResult();
}
private static void setPlugin(Solver solver, AnalysisOptions options, String solverType) {
CompositePlugin plugin = new CompositePlugin();
// add builtin plugins
// To record elapsed time precisely, AnalysisTimer should be added at first.
plugin.addPlugin(
new AnalysisTimer(),
new EntryPointHandler(),
new ClassInitializer(),
new ThreadHandler(),
new NativeModeller(),
new ExceptionAnalysis()
);
if (solverType.equals("csc")) {
MakeDefaultContainerConfig.make();
plugin.addPlugin(
new LocalFlowHandler(),
new FieldAccessHandler(),
new ContainerAccessHandler()
);
}
int javaVersion = World.get().getOptions().getJavaVersion();
if (javaVersion < 9) {
// current reference handler doesn't support Java 9+
plugin.addPlugin(new ReferenceHandler());
}
if (javaVersion >= 8) {
plugin.addPlugin(new LambdaAnalysis());
}
if (javaVersion >= 9) {
plugin.addPlugin(new Java9StringConcatHandler());
}
if (options.getString("reflection-inference") != null ||
options.getString("reflection-log") != null) {
plugin.addPlugin(new ReflectionAnalysis());
}
if (options.getBoolean("handle-invokedynamic") &&
InvokeDynamicAnalysis.useMethodHandle()) {
plugin.addPlugin(new InvokeDynamicAnalysis());
}
if (options.getString("taint-config") != null
|| !((List<String>) options.get("taint-config-providers")).isEmpty()) {
plugin.addPlugin(new TaintAnalysis());
}
plugin.addPlugin(new ResultProcessor());
// add plugins specified in options
// noinspection unchecked
addPlugins(plugin, (List<String>) options.get("plugins"));
// connects plugins and solver
plugin.setSolver(solver);
solver.setPlugin(plugin);
}
private static void addPlugins(CompositePlugin plugin,
List<String> pluginClasses) {
for (String pluginClass : pluginClasses) {
try {
Class<?> clazz = Class.forName(pluginClass);
Constructor<?> ctor = clazz.getConstructor();
Plugin newPlugin = (Plugin) ctor.newInstance();
plugin.addPlugin(newPlugin);
} catch (ClassNotFoundException e) {
throw new ConfigException(
"Plugin class " + pluginClass + " is not found");
} catch (IllegalAccessException | NoSuchMethodException e) {
throw new AnalysisException("Failed to get constructor of " +
pluginClass + ", does the plugin class" +
" provide a public non-arg constructor?");
} catch (InvocationTargetException | InstantiationException e) {
throw new AnalysisException(
"Failed to create plugin instance for " + pluginClass, e);
}
}
}
}