Skip to content

Commit

Permalink
C++: Add support for requires clauses and constraints on template par…
Browse files Browse the repository at this point in the history
…ameters
  • Loading branch information
jketema committed Dec 26, 2024
1 parent a77b8f6 commit 3b6878d
Show file tree
Hide file tree
Showing 5 changed files with 65 additions and 0 deletions.
21 changes: 21 additions & 0 deletions cpp/ql/lib/semmle/code/cpp/Function.qll
Original file line number Diff line number Diff line change
Expand Up @@ -715,6 +715,27 @@ class FunctionDeclarationEntry extends DeclarationEntry, @fun_decl {
* specification.
*/
predicate isNoExcept() { fun_decl_empty_noexcept(underlyingElement(this)) }

/**
* Gets a requires clause if this declaration is a template with such a clause.
*/
Expr getARequiresClause() { fun_requires(underlyingElement(this), _, unresolveElement(result)) }

/**
* Gets the requires clause that appears after the template argument list if this
* declaration is a template with such a clause.
*/
Expr getTemplateRequiresClause() {
fun_requires(underlyingElement(this), 1, unresolveElement(result))
}

/**
* Gets the requires clause that appears after the declarator if this declaration
* is a template with such a clause.
*/
Expr getFunctionRequiresClause() {
fun_requires(underlyingElement(this), 2, unresolveElement(result))
}
}

/**
Expand Down
7 changes: 7 additions & 0 deletions cpp/ql/lib/semmle/code/cpp/TemplateParameter.qll
Original file line number Diff line number Diff line change
Expand Up @@ -59,6 +59,13 @@ class TypeTemplateParameter extends UserType, TemplateParameterImpl {
override string getAPrimaryQlClass() { result = "TypeTemplateParameter" }

override predicate involvesTemplateParameter() { any() }

/**
* Get the type constraint of this type template parameter.
*/
Expr getTypeConstraint() {
type_template_type_constraint(underlyingElement(this), unresolveElement(result))
}
}

/**
Expand Down
5 changes: 5 additions & 0 deletions cpp/ql/lib/semmle/code/cpp/UserType.qll
Original file line number Diff line number Diff line change
Expand Up @@ -129,4 +129,9 @@ class TypeDeclarationEntry extends DeclarationEntry, @type_decl {
* class or typedef.
*/
predicate isTopLevel() { type_decl_top(underlyingElement(this)) }

/**
* Gets the requires clause if this declaration is a template with such a clause.
*/
Expr getRequiresClause() { type_requires(underlyingElement(this), unresolveElement(result)) }
}
5 changes: 5 additions & 0 deletions cpp/ql/lib/semmle/code/cpp/Variable.qll
Original file line number Diff line number Diff line change
Expand Up @@ -278,6 +278,11 @@ class VariableDeclarationEntry extends DeclarationEntry, @var_decl {

/** Holds if this declaration is a template specialization. */
predicate isSpecialization() { var_specialized(underlyingElement(this)) }

/**
* Gets the requires clause if this declaration is a template with such a clause.
*/
Expr getRequiresClause() { var_requires(underlyingElement(this), unresolveElement(result)) }
}

/**
Expand Down
27 changes: 27 additions & 0 deletions cpp/ql/lib/semmlecode.cpp.dbscheme
Original file line number Diff line number Diff line change
Expand Up @@ -480,6 +480,20 @@ fun_decl_typedef_type(
int typedeftype_id: @usertype ref
);

/*
case @fun_requires.kind of
1 = @template_attached
| 2 = @function_attached
;
*/

#keyset[id, kind]
fun_requires(
int id: @fun_decl ref,
int kind: int ref,
int constraint: @expr ref
);

param_decl_bind(
unique int id: @var_decl ref,
int index: int ref,
Expand All @@ -501,6 +515,10 @@ var_decl_specifiers(
string name: string ref
)
is_structured_binding(unique int id: @variable ref);
var_requires(
unique int id: @var_decl ref,
int constraint: @expr ref
);

type_decls(
unique int id: @type_decl,
Expand All @@ -511,6 +529,10 @@ type_def(unique int id: @type_decl ref);
type_decl_top(
unique int type_decl: @type_decl ref
);
type_requires(
unique int id: @type_decl ref,
int constraint: @expr ref
);

namespace_decls(
unique int id: @namespace_decl,
Expand Down Expand Up @@ -790,6 +812,11 @@ nontype_template_parameters(
int id: @expr ref
);

type_template_type_constraint(
unique int id: @usertype ref,
int constraint: @expr ref
);

mangled_name(
unique int id: @declaration ref,
int mangled_name : @mangledname,
Expand Down

0 comments on commit 3b6878d

Please sign in to comment.