Skip to content

Commit

Permalink
Merge pull request #17818 from hvitved/rust/summary-stats-perf
Browse files Browse the repository at this point in the history
Rust: Speedup `SummaryStats.ql`
  • Loading branch information
hvitved authored Oct 22, 2024
2 parents 23a1ea7 + c4c936d commit 60c205f
Show file tree
Hide file tree
Showing 3 changed files with 31 additions and 18 deletions.
24 changes: 15 additions & 9 deletions rust/ql/lib/codeql/rust/AstConsistency.qll
Original file line number Diff line number Diff line change
Expand Up @@ -5,35 +5,41 @@
private import rust
private import codeql.rust.elements.internal.generated.ParentChild

private predicate multipleToStrings(Element e) { strictcount(e.toString()) > 1 }

/**
* Holds if `e` has more than one `toString()` result.
*/
query predicate multipleToStrings(Element e, string s) {
s = strictconcat(e.toString(), ", ") and
strictcount(e.toString()) > 1
multipleToStrings(e) and
s = strictconcat(e.toString(), ", ")
}

/**
* Holds if `e` has more than one `Location`.
*/
query predicate multipleLocations(Locatable e) { strictcount(e.getLocation()) > 1 }

private predicate multiplePrimaryQlClasses(Element e) { strictcount(e.getAPrimaryQlClass()) > 1 }

/**
* Holds if `e` has more than one `getPrimaryQlClasses()` result.
*/
query predicate multiplePrimaryQlClasses(Element e, string s) {
s = strictconcat(e.getPrimaryQlClasses(), ", ") and
strictcount(e.getAPrimaryQlClass()) > 1
multiplePrimaryQlClasses(e) and
s = strictconcat(e.getPrimaryQlClasses(), ", ")
}

private Element getParent(Element child) { child = getChildAndAccessor(result, _, _) }

private predicate multipleParents(Element child) { strictcount(getParent(child)) > 1 }

/**
* Holds if `child` has more than one AST parent.
*/
query predicate multipleParents(Element child, Element parent) {
parent = getParent(child) and
strictcount(getParent(child)) > 1
multipleParents(child) and
parent = getParent(child)
}

/**
Expand All @@ -42,14 +48,14 @@ query predicate multipleParents(Element child, Element parent) {
int getAstInconsistencyCounts(string type) {
// total results from all the AST consistency query predicates.
type = "Multiple toStrings" and
result = count(Element e | multipleToStrings(e, _) | e)
result = count(Element e | multipleToStrings(e) | e)
or
type = "Multiple locations" and
result = count(Element e | multipleLocations(e) | e)
or
type = "Multiple primary QL classes" and
result = count(Element e | multiplePrimaryQlClasses(e, _) | e)
result = count(Element e | multiplePrimaryQlClasses(e) | e)
or
type = "Multiple parents" and
result = count(Element e | multipleParents(e, _) | e)
result = count(Element e | multipleParents(e) | e)
}
18 changes: 11 additions & 7 deletions rust/ql/lib/codeql/rust/controlflow/internal/CfgConsistency.qll
Original file line number Diff line number Diff line change
Expand Up @@ -9,11 +9,7 @@ private import codeql.rust.controlflow.ControlFlowGraph
private import codeql.rust.controlflow.internal.ControlFlowGraphImpl as CfgImpl
private import codeql.rust.controlflow.internal.Completion

/**
* All `Expr` nodes are `PostOrderTree`s
*/
query predicate nonPostOrderExpr(Expr e, string cls) {
cls = e.getPrimaryQlClasses() and
private predicate nonPostOrderExpr(Expr e) {
not e instanceof LetExpr and
not e instanceof ParenExpr and
exists(AstNode last, Completion c |
Expand All @@ -23,6 +19,14 @@ query predicate nonPostOrderExpr(Expr e, string cls) {
)
}

/**
* All `Expr` nodes are `PostOrderTree`s
*/
query predicate nonPostOrderExpr(Expr e, string cls) {
nonPostOrderExpr(e) and
cls = e.getPrimaryQlClasses()
}

/**
* Holds if CFG scope `scope` lacks an initial AST node. Overrides shared consistency predicate.
*/
Expand Down Expand Up @@ -84,11 +88,11 @@ int getCfgInconsistencyCounts(string type) {
result = count(CfgImpl::SplitKind sk | nonUniqueListOrder(sk, _) | sk)
or
type = "Multiple toStrings" and
result = count(CfgNode n | multipleToString(n, _) | n)
result = count(CfgNode n | multipleToString(n) | n)
or
type = "CFG scope lacks initial AST node" and
result = count(CfgScope s | scopeNoFirst(s) | s)
or
type = "Non-PostOrderTree Expr node" and
result = count(Expr e | nonPostOrderExpr(e, _) | e)
result = count(Expr e | nonPostOrderExpr(e) | e)
}
7 changes: 5 additions & 2 deletions shared/controlflow/codeql/controlflow/Cfg.qll
Original file line number Diff line number Diff line change
Expand Up @@ -1502,10 +1502,13 @@ module MakeWithSplitting<
strictcount(sk.getListOrder()) > 1
}

/** Holds if `n` has multiple textual representations. */
predicate multipleToString(Node n) { strictcount(n.toString()) > 1 }

/** Holds if `n` has multiple textual representations. */
query predicate multipleToString(Node n, string s) {
s = strictconcat(n.toString(), ",") and
strictcount(n.toString()) > 1
multipleToString(n) and
s = strictconcat(n.toString(), ",")
}

/** Holds if CFG scope `scope` lacks an initial AST node. */
Expand Down

0 comments on commit 60c205f

Please sign in to comment.