Skip to content

Commit 9becd08

Browse files
authored
Merge pull request #15179 from michaelnebel/modelgenrespectmanual
C#/Java: Increase precision of model generation.
2 parents 6bd31de + 37a21ec commit 9becd08

12 files changed

+94
-34
lines changed

csharp/ql/src/utils/modelgenerator/CaptureDiscardedSummaryModels.ql

-15
This file was deleted.

csharp/ql/src/utils/modelgenerator/CaptureNeutralModels.ql

+1-3
Original file line numberDiff line numberDiff line change
@@ -11,7 +11,5 @@ import internal.CaptureModels
1111
import internal.CaptureSummaryFlowQuery
1212

1313
from DataFlowTargetApi api, string noflow
14-
where
15-
noflow = captureNoFlow(api) and
16-
not api.(FlowSummaryImpl::Public::SummarizedCallable).applyManualModel()
14+
where noflow = captureNoFlow(api)
1715
select noflow order by noflow

csharp/ql/src/utils/modelgenerator/CaptureSummaryModels.ql

+1-3
Original file line numberDiff line numberDiff line change
@@ -11,7 +11,5 @@ import internal.CaptureModels
1111
import internal.CaptureSummaryFlowQuery
1212

1313
from DataFlowTargetApi api, string flow
14-
where
15-
flow = captureFlow(api) and
16-
not api.(FlowSummaryImpl::Public::SummarizedCallable).applyManualModel()
14+
where flow = captureFlow(api)
1715
select flow order by flow

csharp/ql/src/utils/modelgenerator/internal/CaptureModelsSpecific.qll

+5-1
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,7 @@ private import dotnet
77
private import semmle.code.csharp.commons.Util as Util
88
private import semmle.code.csharp.commons.Collections as Collections
99
private import semmle.code.csharp.dataflow.internal.DataFlowDispatch
10+
private import semmle.code.csharp.dataflow.internal.FlowSummaryImpl as FlowSummaryImpl
1011
private import semmle.code.csharp.frameworks.system.linq.Expressions
1112
import semmle.code.csharp.dataflow.internal.ExternalFlow as ExternalFlow
1213
import semmle.code.csharp.dataflow.internal.DataFlowImplCommon as DataFlowImplCommon
@@ -37,7 +38,10 @@ private predicate isRelevantForModels(CS::Callable api) {
3738
not api instanceof Util::MainMethod and
3839
not api instanceof CS::Destructor and
3940
not api instanceof CS::AnonymousFunctionExpr and
40-
not api.(CS::Constructor).isParameterless()
41+
not api.(CS::Constructor).isParameterless() and
42+
// Disregard all APIs that have a manual model.
43+
not api = any(FlowSummaryImpl::Public::SummarizedCallable sc | sc.applyManualModel()) and
44+
not api = any(FlowSummaryImpl::Public::NeutralSummaryCallable sc | sc.hasManualModel())
4145
}
4246

4347
/**
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
extensions:
2+
- addsTo:
3+
pack: codeql/csharp-all
4+
extensible: neutralModel
5+
data:
6+
- [ "NoSummaries", "ManuallyModelled", "HasNeutralSummaryNoFlow", "(System.Object)", "summary", "manual"]
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
extensions:
2+
- addsTo:
3+
pack: codeql/csharp-all
4+
extensible: summaryModel
5+
data:
6+
- [ "NoSummaries", "ManuallyModelled", False, "HasSummary", "(System.Object)", "", "Argument[0]", "ReturnValue", "value", "manual"]
7+
8+
- addsTo:
9+
pack: codeql/csharp-all
10+
extensible: neutralModel
11+
data:
12+
- [ "NoSummaries", "ManuallyModelled", "HasNeutralSummary", "(System.Object)", "summary", "manual"]

csharp/ql/test/utils/modelgenerator/dataflow/NoSummaries.cs

+19
Original file line numberDiff line numberDiff line change
@@ -143,3 +143,22 @@ public ParameterlessConstructor()
143143
IsInitialized = true;
144144
}
145145
}
146+
147+
// No models should be created, if there exist either a manual summary or neutral summary.
148+
public class ManuallyModelled
149+
{
150+
public object HasSummary(object o)
151+
{
152+
return o;
153+
}
154+
155+
public object HasNeutralSummary(object o)
156+
{
157+
return o;
158+
}
159+
160+
public object HasNeutralSummaryNoFlow(object o)
161+
{
162+
return null;
163+
}
164+
}

csharp/ql/test/utils/modelgenerator/typebasedflow/CaptureTypeBasedSummaryModels.expected

-1
Original file line numberDiff line numberDiff line change
@@ -124,7 +124,6 @@
124124
| Summaries;TypeBasedCollection<T>;false;Add;(T);;Argument[0];Argument[this].Element;value;tb-generated |
125125
| Summaries;TypeBasedCollection<T>;false;AddMany;(System.Collections.Generic.IEnumerable<T>);;Argument[0].Element;Argument[this].Element;value;tb-generated |
126126
| Summaries;TypeBasedCollection<T>;false;First;();;Argument[this].Element;ReturnValue;value;tb-generated |
127-
| Summaries;TypeBasedCollection<T>;false;GetEnumerator;();;Argument[this].Element;ReturnValue.SyntheticField[ArgType0];value;tb-generated |
128127
| Summaries;TypeBasedCollection<T>;false;GetMany;();;Argument[this].Element;ReturnValue.Element;value;tb-generated |
129128
| Summaries;TypeBasedComplex<T>;false;AddMany;(System.Collections.Generic.IEnumerable<T>);;Argument[0].Element;Argument[this].SyntheticField[ArgType0];value;tb-generated |
130129
| Summaries;TypeBasedComplex<T>;false;Apply;(System.Func<T,System.Int32>);;Argument[this].SyntheticField[ArgType0];Argument[0].Parameter[0];value;tb-generated |

java/ql/src/utils/modelgenerator/internal/CaptureModelsSpecific.qll

+22-8
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@
55
private import java as J
66
private import semmle.code.java.dataflow.internal.DataFlowPrivate
77
private import semmle.code.java.dataflow.internal.ContainerFlow as ContainerFlow
8+
private import semmle.code.java.dataflow.internal.FlowSummaryImpl as FlowSummaryImpl
89
private import semmle.code.java.dataflow.internal.ModelExclusions
910
private import semmle.code.java.dataflow.DataFlow as Df
1011
private import semmle.code.java.dataflow.SSA as Ssa
@@ -21,6 +22,8 @@ class Type = J::Type;
2122

2223
class Unit = J::Unit;
2324

25+
class Callable = J::Callable;
26+
2427
private J::Method superImpl(J::Method m) {
2528
result = m.getAnOverride() and
2629
not exists(result.getAnOverride()) and
@@ -35,15 +38,19 @@ private predicate isInfrequentlyUsed(J::CompilationUnit cu) {
3538
/**
3639
* Holds if it is relevant to generate models for `api`.
3740
*/
38-
private predicate isRelevantForModels(J::Callable api) {
41+
private predicate isRelevantForModels(Callable api) {
3942
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()
4148
}
4249

4350
/**
4451
* Holds if it is relevant to generate models for `api` based on data flow analysis.
4552
*/
46-
predicate isRelevantForDataFlowModels(J::Callable api) {
53+
predicate isRelevantForDataFlowModels(Callable api) {
4754
isRelevantForModels(api) and
4855
(not api.getDeclaringType() instanceof J::Interface or exists(api.getBody()))
4956
}
@@ -56,7 +63,7 @@ predicate isRelevantForTypeBasedFlowModels = isRelevantForModels/1;
5663
* In the Standard library and 3rd party libraries it the Callables that can be called
5764
* from outside the library itself.
5865
*/
59-
class TargetApiSpecific extends J::Callable {
66+
class TargetApiSpecific extends Callable {
6067
TargetApiSpecific() {
6168
this.isPublic() and
6269
this.fromSource() and
@@ -66,6 +73,15 @@ class TargetApiSpecific extends J::Callable {
6673
) and
6774
isRelevantForModels(this)
6875
}
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+
}
6985
}
7086

7187
private string isExtensible(J::RefType ref) {
@@ -79,9 +95,7 @@ private string typeAsModel(J::RefType type) {
7995
}
8096

8197
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()
8599
}
86100

87101
/**
@@ -195,7 +209,7 @@ string returnNodeAsOutput(DataFlowImplCommon::ReturnNodeExt node) {
195209
/**
196210
* Gets the enclosing callable of `ret`.
197211
*/
198-
J::Callable returnNodeEnclosingCallable(DataFlowImplCommon::ReturnNodeExt ret) {
212+
Callable returnNodeEnclosingCallable(DataFlowImplCommon::ReturnNodeExt ret) {
199213
result = DataFlowImplCommon::getNodeEnclosingCallable(ret).asCallable()
200214
}
201215

java/ql/src/utils/modelgenerator/internal/CaptureSummaryFlowQuery.qll

+4-3
Original file line numberDiff line numberDiff line change
@@ -73,10 +73,11 @@ string captureFlow(DataFlowTargetApi api) {
7373
}
7474

7575
/**
76-
* Gets the neutral summary for `api`, if any.
77-
* A neutral model is generated, if there does not exist any summary model.
76+
* Gets the neutral summary model for `api`, if any.
77+
* A neutral summary model is generated, if we are not generating
78+
* a summary model that applies to `api`.
7879
*/
7980
string captureNoFlow(DataFlowTargetApi api) {
80-
not exists(captureFlow(api)) and
81+
not exists(DataFlowTargetApi api0 | exists(captureFlow(api0)) and api0.lift() = api.lift()) and
8182
result = ModelPrinting::asNeutralSummaryModel(api)
8283
}

java/ql/test/utils/modelgenerator/dataflow/CaptureSummaryModels.expected

+1
Original file line numberDiff line numberDiff line change
@@ -22,6 +22,7 @@
2222
| p;Joiner;false;setEmptyValue;(CharSequence);;Argument[0];Argument[this];taint;df-generated |
2323
| p;Joiner;false;setEmptyValue;(CharSequence);;Argument[this];ReturnValue;value;df-generated |
2424
| p;Joiner;false;toString;();;Argument[this];ReturnValue;taint;df-generated |
25+
| p;MultipleImpl2$IInterface;true;m;(Object);;Argument[0];ReturnValue;taint;df-generated |
2526
| p;MultipleImpls$Strat2;true;getValue;();;Argument[this];ReturnValue;taint;df-generated |
2627
| p;MultipleImpls$Strategy;true;doSomething;(String);;Argument[0];Argument[this];taint;df-generated |
2728
| p;MultipleImpls$Strategy;true;doSomething;(String);;Argument[0];ReturnValue;taint;df-generated |
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,23 @@
1+
package p;
2+
3+
class MultipleImpl2 {
4+
5+
// Multiple implementations of the same interface.
6+
// This is used to test that we only generate a summary model and
7+
// not neutral summary model for `IInterface.m`.
8+
public interface IInterface {
9+
Object m(Object value);
10+
}
11+
12+
public class Impl1 implements IInterface {
13+
public Object m(Object value) {
14+
return null;
15+
}
16+
}
17+
18+
public class Impl2 implements IInterface {
19+
public Object m(Object value) {
20+
return value;
21+
}
22+
}
23+
}

0 commit comments

Comments
 (0)