From e20079a5d9fab63060ad38907285393be83e2afa Mon Sep 17 00:00:00 2001 From: IOBYTE Date: Mon, 26 Nov 2018 00:23:12 -0500 Subject: [PATCH] template simplifier: fix 3 function forward declaration bugs (#1493) * fix support for multi token types * fix support for const member functions * fix duplicate template parameters sometimes being inserted --- lib/templatesimplifier.cpp | 28 ++++++++++++++++++++++---- test/testsimplifytemplate.cpp | 37 +++++++++++++++++++++++++++++++++++ 2 files changed, 61 insertions(+), 4 deletions(-) diff --git a/lib/templatesimplifier.cpp b/lib/templatesimplifier.cpp index 1ec20b7b7..f4b7706d6 100644 --- a/lib/templatesimplifier.cpp +++ b/lib/templatesimplifier.cpp @@ -1047,6 +1047,8 @@ void TemplateSimplifier::expandTemplate( start = templateDeclarationToken->next(); end = templateDeclarationNameToken->linkAt(1)->next(); } + while (!Token::Match(end, ";|{")) + end = end->next(); std::map links; while (start && start != end) { @@ -1055,12 +1057,30 @@ void TemplateSimplifier::expandTemplate( ++itype; if (itype < typeParametersInDeclaration.size()) { - dst->insertToken(mTypesUsedInTemplateInstantiation[itype]->str(), "", true); - dst->previous()->isTemplateArg(true); + unsigned int typeindentlevel = 0; + for (const Token *typetok = mTypesUsedInTemplateInstantiation[itype]; + typetok && (typeindentlevel > 0 || !Token::Match(typetok, ",|>")); + typetok = typetok->next()) { + if (Token::simpleMatch(typetok, ". . .")) { + typetok = typetok->tokAt(2); + continue; + } + if (Token::Match(typetok, "%name% <") && templateParameters(typetok->next()) > 0) + ++typeindentlevel; + else if (typeindentlevel > 0 && typetok->str() == ">") + --typeindentlevel; + dst->insertToken(typetok->str(), typetok->originalName(), true); + dst->previous()->isTemplateArg(true); + dst->previous()->isSigned(typetok->isSigned()); + dst->previous()->isUnsigned(typetok->isUnsigned()); + dst->previous()->isLong(typetok->isLong()); + } } else { - if (start->str() == templateDeclarationNameToken->str()) + if (start->str() == templateDeclarationNameToken->str()) { dst->insertToken(newName, "", true); - else + if (start->strAt(1) == "<") + start = start->next()->findClosingBracket(); + } else dst->insertToken(start->str(), "", true); if (start->link()) { if (Token::Match(start, "[|{|(")) { diff --git a/test/testsimplifytemplate.cpp b/test/testsimplifytemplate.cpp index bbe2e3211..786ec42cb 100644 --- a/test/testsimplifytemplate.cpp +++ b/test/testsimplifytemplate.cpp @@ -117,6 +117,8 @@ private: TEST_CASE(template77); TEST_CASE(template78); TEST_CASE(template79); // #5133 + TEST_CASE(template80); + TEST_CASE(template81); TEST_CASE(template_specialization_1); // #7868 - template specialization template struct S> {..}; TEST_CASE(template_specialization_2); // #7868 - template specialization template struct S> {..}; TEST_CASE(template_enum); // #6299 Syntax error in complex enum declaration (including template) @@ -1517,6 +1519,41 @@ private: ASSERT_EQUALS(exp, tok(code)); } + void template80() { + const char code[] = "class Fred {\n" + " template T foo(T t) const { return t; }\n" + "};\n" + "const void * p = Fred::foo(nullptr);"; + const char exp[] = "class Fred { " + "const void * foo ( const void * t ) const ; " + "} ; " + "const void * p ; p = Fred :: foo ( nullptr ) ; " + "const void * Fred :: foo ( const void * t ) const { return t ; }"; + ASSERT_EQUALS(exp, tok(code)); + } + + void template81() { + const char code[] = "template \n" + "struct SortWith {\n" + " SortWith(Type);\n" + "};\n" + "template \n" + "SortWith::SortWith(Type) {}\n" + "int main() {\n" + " SortWith(0);\n" + "}"; + const char exp[] = "template < typename Type > " + "struct SortWith { " + "SortWith ( Type ) ; " + "} ; " + "SortWith :: SortWith ( int ) ; " + "int main ( ) { " + "SortWith ( 0 ) ; " + "} " + "SortWith :: SortWith ( int ) { }"; + ASSERT_EQUALS(exp, tok(code)); + } + void template_specialization_1() { // #7868 - template specialization template struct S> {..}; const char code[] = "template struct C {};\n" "template struct S {a};\n"