5
5
private import java as J
6
6
private import semmle.code.java.dataflow.internal.DataFlowPrivate
7
7
private import semmle.code.java.dataflow.internal.ContainerFlow as ContainerFlow
8
+ private import semmle.code.java.dataflow.internal.FlowSummaryImpl as FlowSummaryImpl
8
9
private import semmle.code.java.dataflow.internal.ModelExclusions
9
10
private import semmle.code.java.dataflow.DataFlow as Df
10
11
private import semmle.code.java.dataflow.SSA as Ssa
@@ -21,6 +22,8 @@ class Type = J::Type;
21
22
22
23
class Unit = J:: Unit ;
23
24
25
+ class Callable = J:: Callable ;
26
+
24
27
private J:: Method superImpl ( J:: Method m ) {
25
28
result = m .getAnOverride ( ) and
26
29
not exists ( result .getAnOverride ( ) ) and
@@ -35,15 +38,19 @@ private predicate isInfrequentlyUsed(J::CompilationUnit cu) {
35
38
/**
36
39
* Holds if it is relevant to generate models for `api`.
37
40
*/
38
- private predicate isRelevantForModels ( J :: Callable api ) {
41
+ private predicate isRelevantForModels ( Callable api ) {
39
42
not isUninterestingForModels ( api ) and
40
- not isInfrequentlyUsed ( api .getCompilationUnit ( ) )
43
+ not isInfrequentlyUsed ( api .getCompilationUnit ( ) ) and
44
+ // Disregard all APIs that have a manual model.
45
+ not api = any ( FlowSummaryImpl:: Public:: SummarizedCallable sc | sc .applyManualModel ( ) ) .asCallable ( ) and
46
+ not api =
47
+ any ( FlowSummaryImpl:: Public:: NeutralSummaryCallable sc | sc .hasManualModel ( ) ) .asCallable ( )
41
48
}
42
49
43
50
/**
44
51
* Holds if it is relevant to generate models for `api` based on data flow analysis.
45
52
*/
46
- predicate isRelevantForDataFlowModels ( J :: Callable api ) {
53
+ predicate isRelevantForDataFlowModels ( Callable api ) {
47
54
isRelevantForModels ( api ) and
48
55
( not api .getDeclaringType ( ) instanceof J:: Interface or exists ( api .getBody ( ) ) )
49
56
}
@@ -56,7 +63,7 @@ predicate isRelevantForTypeBasedFlowModels = isRelevantForModels/1;
56
63
* In the Standard library and 3rd party libraries it the Callables that can be called
57
64
* from outside the library itself.
58
65
*/
59
- class TargetApiSpecific extends J :: Callable {
66
+ class TargetApiSpecific extends Callable {
60
67
TargetApiSpecific ( ) {
61
68
this .isPublic ( ) and
62
69
this .fromSource ( ) and
@@ -66,6 +73,15 @@ class TargetApiSpecific extends J::Callable {
66
73
) and
67
74
isRelevantForModels ( this )
68
75
}
76
+
77
+ /**
78
+ * Gets the callable that a model will be lifted to, if any.
79
+ */
80
+ Callable lift ( ) {
81
+ exists ( Method m | m = superImpl ( this ) and m .fromSource ( ) | result = m )
82
+ or
83
+ not exists ( superImpl ( this ) ) and result = this
84
+ }
69
85
}
70
86
71
87
private string isExtensible ( J:: RefType ref ) {
@@ -79,9 +95,7 @@ private string typeAsModel(J::RefType type) {
79
95
}
80
96
81
97
private J:: RefType bestTypeForModel ( TargetApiSpecific api ) {
82
- if exists ( superImpl ( api ) )
83
- then superImpl ( api ) .fromSource ( ) and result = superImpl ( api ) .getDeclaringType ( )
84
- else result = api .getDeclaringType ( )
98
+ result = api .lift ( ) .getDeclaringType ( )
85
99
}
86
100
87
101
/**
@@ -195,7 +209,7 @@ string returnNodeAsOutput(DataFlowImplCommon::ReturnNodeExt node) {
195
209
/**
196
210
* Gets the enclosing callable of `ret`.
197
211
*/
198
- J :: Callable returnNodeEnclosingCallable ( DataFlowImplCommon:: ReturnNodeExt ret ) {
212
+ Callable returnNodeEnclosingCallable ( DataFlowImplCommon:: ReturnNodeExt ret ) {
199
213
result = DataFlowImplCommon:: getNodeEnclosingCallable ( ret ) .asCallable ( )
200
214
}
201
215
0 commit comments