From c8f2c7c81753d8a11604603d1514459fb90fb3c7 Mon Sep 17 00:00:00 2001 From: Jeroen Ketema Date: Thu, 26 Dec 2024 21:51:59 +0100 Subject: [PATCH] C++: Fix printing of concept id type arguments in PrintAST If a type would be used in multiple places in the AST, rendering of the AST would be broken. --- cpp/ql/lib/semmle/code/cpp/PrintAST.qll | 50 +++++++++++++++++-------- 1 file changed, 35 insertions(+), 15 deletions(-) diff --git a/cpp/ql/lib/semmle/code/cpp/PrintAST.qll b/cpp/ql/lib/semmle/code/cpp/PrintAST.qll index c29624ede01c..7636f1372590 100644 --- a/cpp/ql/lib/semmle/code/cpp/PrintAST.qll +++ b/cpp/ql/lib/semmle/code/cpp/PrintAST.qll @@ -114,6 +114,9 @@ private newtype TPrintAstNode = TConceptIdExprArgumentsNode(ConceptIdExpr concept) { shouldPrintDeclaration(getAnEnclosingDeclaration(concept)) } or + TConceptIdExprTypeArgumentNode(Type type, ConceptIdExpr concept, int childIndex) { + type = concept.getTemplateArgument(childIndex) + } or TConstructorInitializersNode(Constructor ctor) { ctor.hasEntryPoint() and shouldPrintDeclaration(ctor) @@ -601,19 +604,6 @@ class ParameterNode extends AstNode { } } -/** - * A node representing a `Type`. - */ -class TypeNode extends AstNode { - Type t; - - TypeNode() { t = ast } - - final override PrintAstNode getChildInternal(int childIndex) { none() } - - final override string getChildAccessorPredicateInternal(int childIndex) { none() } -} - /** * A node representing an `Initializer`. */ @@ -645,8 +635,12 @@ class ConceptIdExprArgumentsNode extends PrintAstNode, TConceptIdExprArgumentsNo final override Location getLocation() { result = getRepresentativeLocation(concept) } - override AstNode getChildInternal(int childIndex) { - result.getAst() = concept.getTemplateArgument(childIndex) + override PrintAstNode getChildInternal(int childIndex) { + exists(Locatable arg | arg = concept.getTemplateArgument(childIndex) | + result.(ConceptIdExprTypeArgumentNode).isArgumentNode(arg, concept, childIndex) + or + result.(ExprNode).getAst() = arg + ) } override string getChildAccessorPredicateInternal(int childIndex) { @@ -660,6 +654,32 @@ class ConceptIdExprArgumentsNode extends PrintAstNode, TConceptIdExprArgumentsNo final ConceptIdExpr getConceptIdExpr() { result = concept } } +/** + * A node representing a type argument of a `ConceptIdExpr`. + */ +class ConceptIdExprTypeArgumentNode extends PrintAstNode, TConceptIdExprTypeArgumentNode { + Type type; + ConceptIdExpr concept; + int index; + + ConceptIdExprTypeArgumentNode() { this = TConceptIdExprTypeArgumentNode(type, concept, index) } + + final override string toString() { result = qlClass(type) + type.toString() } + + final override Location getLocation() { result = getRepresentativeLocation(type) } + + override AstNode getChildInternal(int childIndex) { none() } + + override string getChildAccessorPredicateInternal(int childIndex) { none() } + + /** + * Holds if `t` is the `i`th template argument of `c`. + */ + predicate isArgumentNode(Type t, ConceptIdExpr c, int i) { + type = t and concept = c and index = i + } +} + /** * A node representing the parameters of a `Function`. */