From 319cdc2b43f84a6da31db290862fd07d8bca803a Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Daniel=20Marjam=C3=A4ki?= Date: Sun, 1 Nov 2009 19:03:52 +0100 Subject: [PATCH] Fixed #880 (Tokenizer: The Tokenizer::simplifyTemplates doesn't handle recursive templates) --- lib/tokenize.cpp | 437 ++++++++++++++++++++---------------- test/testclass.cpp | 4 +- test/testsimplifytokens.cpp | 171 +++++++------- 3 files changed, 321 insertions(+), 291 deletions(-) diff --git a/lib/tokenize.cpp b/lib/tokenize.cpp index 82861d748..6f56b3396 100644 --- a/lib/tokenize.cpp +++ b/lib/tokenize.cpp @@ -651,6 +651,41 @@ static bool templateParameters(const Token *tok) } +/** + * Remove "template < ..." they can cause false positives because they are not expanded + */ +static void removeTemplates(Token *tok) +{ + for (; tok; tok = tok->next()) + { + if (! Token::simpleMatch(tok, "template <")) + continue; + + for (const Token *tok2 = tok->next(); tok2; tok2 = tok2->next()) + { + if (tok2->str() == "{") + { + tok2 = tok2->link(); + tok2 = tok2 ? tok2->next() : 0; + Token::eraseTokens(tok, tok2); + tok->str(";"); + break; + } + if (tok2->str() == "(") + { + tok2 = tok2->link(); + if (!tok2) + break; + } + if (tok2->str() == ";") + { + Token::eraseTokens(tok, tok2); + tok->str(";"); + } + } + } +} + void Tokenizer::simplifyTemplates() { // Remove "typename" unless used in template arguments.. @@ -691,24 +726,19 @@ void Tokenizer::simplifyTemplates() continue; tok2 = tok2->previous(); - std::string s(tok2->str()); + std::string s; { - const Token *tok3 = tok2->next(); - while (Token::Match(tok3, "<|, %any%")) + std::ostringstream ostr; + const Token *tok3 = tok2; + for (tok3 = tok2; tok3 && tok3->str() != ">"; tok3 = tok3->next()) { - if (!tok3->next()->isNumber() && !tok3->next()->isName()) - break; - - s += " " + tok3->str() + " " + tok3->strAt(1); - tok3 = tok3->tokAt(2); - if (tok3->str() == "*") - { - tok3 = tok3->next(); - s += " *"; - } + if (tok3 != tok2) + ostr << " "; + ostr << tok3->str(); } if (!Token::Match(tok3, "> (")) continue; + s = ostr.str(); } // save search pattern.. @@ -759,7 +789,10 @@ void Tokenizer::simplifyTemplates() } } if (templates.empty()) + { + removeTemplates(_tokens); return; + } // Locate possible instantiations of templates.. std::list used; @@ -768,20 +801,17 @@ void Tokenizer::simplifyTemplates() // template definition.. skip it if (Token::simpleMatch(tok, "template <")) { - unsigned int indentlevel = 0; for (; tok; tok = tok->next()) { - if (tok->str() == "{") + if (tok->str() == "{" || tok->str() == "(") { - ++indentlevel; - } - else if (tok->str() == "}") - { - if (indentlevel <= 1) + tok = tok->link(); + if (!tok) + break; + if (tok->str() == "}") break; - --indentlevel; } - else if (indentlevel == 0 && tok->str() == ";") + else if (tok->str() == ";") { break; } @@ -789,14 +819,19 @@ void Tokenizer::simplifyTemplates() if (!tok) break; } - else if (Token::Match(tok->previous(), "[{};=] %var% <")) + else if (Token::Match(tok->previous(), "[{};=] %var% <") || + Token::Match(tok->tokAt(-2), "[,:] private|protected|public %var% <")) { if (templateParameters(tok->next())) used.push_back(tok); } } if (used.empty()) + { + removeTemplates(_tokens); return; + } + @@ -870,216 +905,234 @@ void Tokenizer::simplifyTemplates() // expand templates - for (std::list::iterator iter1 = templates.begin(); iter1 != templates.end(); ++iter1) + bool done = false; + while (!done) { - Token *tok = *iter1; - - std::vector type; - for (tok = tok->tokAt(2); tok && tok->str() != ">"; tok = tok->next()) + done = true; + for (std::list::iterator iter1 = templates.begin(); iter1 != templates.end(); ++iter1) { - if (Token::Match(tok, "%var% ,|>")) - type.push_back(tok->str()); - } + Token *tok = *iter1; - // bail out if the end of the file was reached - if (!tok) - break; - - // if this is a template function, get the position of the function name - unsigned int pos = 0; - if (Token::Match(tok, "> %type% *| %var% (")) - pos = 2; - else if (Token::Match(tok, "> %type% %type% *| %var% (")) - pos = 3; - if (pos > 0 && tok->tokAt(pos)->str() == "*") - ++pos; - - // name of template function/class.. - const std::string name(tok->strAt(pos > 0 ? pos : 2)); - - const bool isfunc(pos > 0); - - // locate template usage.. - - std::string s(name + " <"); - for (unsigned int i = 0; i < type.size(); ++i) - { - if (i > 0) - s += ","; - s += " %any% "; - } - const std::string pattern(s + "> "); - - std::string::size_type sz1 = used.size(); - unsigned int recursiveCount = 0; - - for (std::list::iterator iter2 = used.begin(); iter2 != used.end(); ++iter2) - { - // If the size of "used" has changed, simplify calculations - if (sz1 != used.size()) + std::vector type; + for (tok = tok->tokAt(2); tok && tok->str() != ">"; tok = tok->next()) { - sz1 = used.size(); - simplifyCalculations(); - recursiveCount++; - if (recursiveCount > 100) + if (Token::Match(tok, "%var% ,|>")) + type.push_back(tok->str()); + } + + // bail out if the end of the file was reached + if (!tok) + break; + + // if this is a template function, get the position of the function name + unsigned int pos = 0; + if (Token::Match(tok, "> %type% *| %var% (")) + pos = 2; + else if (Token::Match(tok, "> %type% %type% *| %var% (")) + pos = 3; + if (pos > 0 && tok->tokAt(pos)->str() == "*") + ++pos; + + // name of template function/class.. + const std::string name(tok->strAt(pos > 0 ? pos : 2)); + + const bool isfunc(pos > 0); + + // locate template usage.. + + std::string s(name + " <"); + for (unsigned int i = 0; i < type.size(); ++i) + { + if (i > 0) + s += ","; + s += " %any% "; + } + const std::string pattern(s + "> "); + + std::string::size_type sz1 = used.size(); + unsigned int recursiveCount = 0; + + for (std::list::iterator iter2 = used.begin(); iter2 != used.end(); ++iter2) + { + // If the size of "used" has changed, simplify calculations + if (sz1 != used.size()) { - // bail out.. - break; + sz1 = used.size(); + simplifyCalculations(); + recursiveCount++; + if (recursiveCount > 100) + { + // bail out.. + break; + } } - } - Token *tok2 = *iter2; + Token *tok2 = *iter2; - if (tok2->str() != name) - continue; + if (tok2->str() != name) + continue; - if (!Token::Match(tok2, (pattern + (isfunc ? "(" : "%var%")).c_str())) - continue; + if (Token::Match(tok2->previous(), "[;{}=]") && + !Token::Match(tok2, (pattern + (isfunc ? "(" : "%var%")).c_str())) + continue; - // New type.. - std::vector types2; - s = ""; - for (const Token *tok3 = tok2->tokAt(2); tok3->str() != ">"; tok3 = tok3->next()) - { - if (tok3->str() != ",") - types2.push_back(tok3->str()); - s += tok3->str(); - } - const std::string type2(s); - - // New classname/funcname.. - const std::string name2(name + "<" + type2 + ">"); - - if (expandedtemplates.find(name2) == expandedtemplates.end()) - { - expandedtemplates.insert(name2); - // Copy template.. - int _indentlevel = 0; - int _parlevel = 0; - for (const Token *tok3 = _tokens; tok3; tok3 = tok3->next()) + // New type.. + std::vector types2; + s = ""; + for (const Token *tok3 = tok2->tokAt(2); tok3->str() != ">"; tok3 = tok3->next()) { - if (tok3->str() == "{") - ++_indentlevel; - else if (tok3->str() == "}") - --_indentlevel; - else if (tok3->str() == "(") - ++_parlevel; - else if (tok3->str() == ")") - --_parlevel; + if (tok3->str() != ",") + types2.push_back(tok3->str()); + s += tok3->str(); + } + const std::string type2(s); - // Start of template.. - if (tok3 == tok) - { - tok3 = tok3->next(); - } + // New classname/funcname.. + const std::string name2(name + "<" + type2 + ">"); - // member function implemented outside class definition - else if (_indentlevel == 0 && _parlevel == 0 && Token::Match(tok3, (pattern + " :: %var% (").c_str())) - { - addtoken(name2.c_str(), tok3->linenr(), tok3->fileIndex()); - while (tok3->str() != "::") - tok3 = tok3->next(); - } - - // not part of template.. go on to next token - else - continue; - - int indentlevel = 0; - std::stack braces; // holds "{" tokens - std::stack brackets; // holds "(" tokens - - for (; tok3; tok3 = tok3->next()) + if (expandedtemplates.find(name2) == expandedtemplates.end()) + { + expandedtemplates.insert(name2); + // Copy template.. + int _indentlevel = 0; + int _parlevel = 0; + for (const Token *tok3 = _tokens; tok3; tok3 = tok3->next()) { if (tok3->str() == "{") - ++indentlevel; - + ++_indentlevel; else if (tok3->str() == "}") + --_indentlevel; + else if (tok3->str() == "(") + ++_parlevel; + else if (tok3->str() == ")") + --_parlevel; + + // Start of template.. + if (tok3 == tok) { - if (indentlevel <= 1) - { - // there is a bug if indentlevel is 0 - // the "}" token should only be added if indentlevel is 1 but I add it always intentionally - // if indentlevel ever becomes 0, cppcheck will write: - // ### Error: Invalid number of character { - addtoken("}", tok3->linenr(), tok3->fileIndex()); - Token::createMutualLinks(braces.top(), _tokensBack); - braces.pop(); - break; - } - --indentlevel; + tok3 = tok3->next(); } + // member function implemented outside class definition + else if (_indentlevel == 0 && _parlevel == 0 && Token::Match(tok3, (pattern + " :: %var% (").c_str())) { - // search for this token in the type vector - unsigned int itype = 0; - while (itype < type.size() && type[itype] != tok3->str()) - ++itype; + addtoken(name2.c_str(), tok3->linenr(), tok3->fileIndex()); + while (tok3->str() != "::") + tok3 = tok3->next(); + } - // replace type with given type.. - if (itype < type.size()) - addtoken(types2[itype].c_str(), tok3->linenr(), tok3->fileIndex()); + // not part of template.. go on to next token + else + continue; - // replace name.. - else if (Token::Match(tok3, (name + " !!<").c_str())) - addtoken(name2.c_str(), tok3->linenr(), tok3->fileIndex()); + int indentlevel = 0; + std::stack braces; // holds "{" tokens + std::stack brackets; // holds "(" tokens - // copy - else + for (; tok3; tok3 = tok3->next()) + { + if (tok3->str() == "{") + ++indentlevel; + + else if (tok3->str() == "}") { - addtoken(tok3->str().c_str(), tok3->linenr(), tok3->fileIndex()); - if (Token::Match(tok3, (name + " <").c_str())) - used.push_back(_tokensBack); - - // link() newly tokens manually - if (tok3->str() == "{") + if (indentlevel <= 1) { - braces.push(_tokensBack); - } - else if (tok3->str() == "}") - { - assert(braces.empty() == false); + // there is a bug if indentlevel is 0 + // the "}" token should only be added if indentlevel is 1 but I add it always intentionally + // if indentlevel ever becomes 0, cppcheck will write: + // ### Error: Invalid number of character { + addtoken("}", tok3->linenr(), tok3->fileIndex()); Token::createMutualLinks(braces.top(), _tokensBack); braces.pop(); + break; } - else if (tok3->str() == "(") + --indentlevel; + } + + + if (tok3->isName()) + { + // search for this token in the type vector + unsigned int itype = 0; + while (itype < type.size() && type[itype] != tok3->str()) + ++itype; + + // replace type with given type.. + if (itype < type.size()) { - brackets.push(_tokensBack); - } - else if (tok3->str() == ")") - { - assert(brackets.empty() == false); - Token::createMutualLinks(brackets.top(), _tokensBack); - brackets.pop(); + addtoken(types2[itype].c_str(), tok3->linenr(), tok3->fileIndex()); + continue; } } + + // replace name.. + if (Token::Match(tok3, (name + " !!<").c_str())) + { + addtoken(name2.c_str(), tok3->linenr(), tok3->fileIndex()); + continue; + } + + // copy + addtoken(tok3->str().c_str(), tok3->linenr(), tok3->fileIndex()); + if (Token::Match(tok3, "%type% <")) + { + if (!Token::Match(tok3, (name + " <").c_str())) + done = false; + used.push_back(_tokensBack); + } + + // link() newly tokens manually + if (tok3->str() == "{") + { + braces.push(_tokensBack); + } + else if (tok3->str() == "}") + { + assert(braces.empty() == false); + Token::createMutualLinks(braces.top(), _tokensBack); + braces.pop(); + } + else if (tok3->str() == "(") + { + brackets.push(_tokensBack); + } + else if (tok3->str() == ")") + { + assert(brackets.empty() == false); + Token::createMutualLinks(brackets.top(), _tokensBack); + brackets.pop(); + } + } + + assert(braces.empty()); + assert(brackets.empty()); } - - assert(braces.empty()); - assert(brackets.empty()); } - } - // Replace all these template usages.. - s = name + " < " + type2 + " >"; - for (std::string::size_type pos = s.find(","); pos != std::string::npos; pos = s.find(",", pos + 2)) - { - s.insert(pos + 1, " "); - s.insert(pos, " "); - } - for (Token *tok4 = tok2; tok4; tok4 = tok4->next()) - { - if (Token::simpleMatch(tok4, s.c_str())) + // Replace all these template usages.. + s = name + " < " + type2 + " >"; + for (std::string::size_type pos = s.find(","); pos != std::string::npos; pos = s.find(",", pos + 2)) { - tok4->str(name2); - while (tok4->next()->str() != ">") + s.insert(pos + 1, " "); + s.insert(pos, " "); + } + for (Token *tok4 = tok2; tok4; tok4 = tok4->next()) + { + if (Token::simpleMatch(tok4, s.c_str())) + { + tok4->str(name2); + while (tok4->next()->str() != ">") + tok4->deleteNext(); tok4->deleteNext(); - tok4->deleteNext(); + } } } } } + + removeTemplates(_tokens); } //--------------------------------------------------------------------------- diff --git a/test/testclass.cpp b/test/testclass.cpp index 01f3ab98f..bc8538e87 100644 --- a/test/testclass.cpp +++ b/test/testclass.cpp @@ -316,8 +316,6 @@ private: void virtualDestructorTemplate() { - // Base class has protected destructor, it makes Base *p = new Derived(); fail - // during compilation time, so error is not possible. => no error checkVirtualDestructor("template class A\n" "{\n" " public:\n" @@ -333,7 +331,7 @@ private: " public:\n" " ~B(){int a;}\n" "};\n"); - ASSERT_EQUALS("[test.cpp:7]: (error) Class AA which is inherited by class B does not have a virtual destructor\n", errout.str()); + ASSERT_EQUALS("[test.cpp:7]: (error) Class AA which is inherited by class B does not have a virtual destructor\n", errout.str()); } void checkUninitVar(const char code[]) diff --git a/test/testsimplifytokens.cpp b/test/testsimplifytokens.cpp index 3540e4de4..97c28dd8c 100644 --- a/test/testsimplifytokens.cpp +++ b/test/testsimplifytokens.cpp @@ -86,6 +86,7 @@ private: TEST_CASE(template13); TEST_CASE(template14); TEST_CASE(template15); + TEST_CASE(template16); TEST_CASE(template_default_parameter); TEST_CASE(template_typename); @@ -921,8 +922,7 @@ private: const char code[] = "template void f(T val) { T a; }\n" "f(10);"; - const std::string expected("template < classname T > void f ( T val ) { T a ; } " - "f ( 10 ) ; " + const std::string expected("; f ( 10 ) ; " "void f ( int val ) { int a ; }"); ASSERT_EQUALS(expected, sizeof_(code)); @@ -933,7 +933,7 @@ private: const char code[] = "template class Fred { T a; };\n" "Fred fred;"; - const std::string expected("template < classname T > class Fred { T a ; } ; " + const std::string expected("; ; " "Fred fred ; " "class Fred { int a ; }"); @@ -945,7 +945,7 @@ private: const char code[] = "template class Fred { T data[sz]; };\n" "Fred fred;"; - const std::string expected("template < classname T , int sz > class Fred { T data [ sz ] ; } ; " + const std::string expected("; ; " "Fred fred ; " "class Fred { float data [ 4 ] ; }"); @@ -957,7 +957,7 @@ private: const char code[] = "template class Fred { Fred(); };\n" "Fred fred;"; - const std::string expected("template < classname T > class Fred { Fred ( ) ; } ; " + const std::string expected("; ; " "Fred fred ; " "class Fred { Fred ( ) ; }"); @@ -970,8 +970,8 @@ private: "template Fred::Fred() { }\n" "Fred fred;"; - const std::string expected("template < classname T > class Fred { } ; " - "template < classname T > Fred < T > :: Fred ( ) { } " + const std::string expected("; ; " + "; " "Fred fred ; " "class Fred { } " "Fred :: Fred ( ) { }"); @@ -985,7 +985,7 @@ private: "Fred fred1;\n" "Fred fred2;"; - const std::string expected("template < classname T > class Fred { } ;" + const std::string expected("; ;" " Fred fred1 ;" " Fred fred2 ;" " class Fred { }"); @@ -1005,12 +1005,7 @@ private: "\n" "};\n"; - const std::string expected("template < class T > " - "class ABC " - "{ " - "public: " - "typedef ABC < T > m ; " - "} ;"); + const std::string expected("; ;"); ASSERT_EQUALS(expected, sizeof_(code)); } @@ -1026,10 +1021,7 @@ private: " return 0;\n" "}\n"; - const std::string expected("template < typename T > class ABC {" - " public: " - "typedef std :: vector < T > type ; " - "} ; " + const std::string expected("; ; " "int main ( ) { " "std :: vector < int > v ; " "v . push_back ( 4 ) ; " @@ -1050,17 +1042,9 @@ private: " }\n" "};\n"; - const std::string expected("template < typename T > class ABC " - "{" - " public: typedef std :: vector < T > type ;" - " void f ( ) " - "{ " - "std :: vector < int > v ;" - " v . push_back ( 4 ) ;" - " } " - "} ;"); + const std::string expected("; ;"); - TODO_ASSERT_EQUALS(expected, sizeof_(code)); + ASSERT_EQUALS(expected, sizeof_(code)); } } @@ -1084,20 +1068,7 @@ private: "\n" "template inline B h() { return B(); }\n"; - const std::string expected("template < typename T > class A ;" - " template < typename T > class B ;" - "" - " typedef A < int > x ;" - " typedef B < int > y ;" - "" - " template < typename T > class A {" - " void f ( ) {" - " B < T > a = B < T > :: g ( ) ;" - " T b ; b = 0 ;" - " }" - " } ;" - "" - " template < typename T > inline B < T > h ( ) { return B < T > ( ) ; }"); + const std::string expected("; ; ;"); ASSERT_EQUALS(expected, sizeof_(code)); } @@ -1118,12 +1089,7 @@ private: "} ;\n"; // The expected result.. - std::string expected(std::string(code) + " class A { }"); - std::string::size_type pos; - while ((pos = expected.find("\n")) != std::string::npos) - expected[pos] = ' '; - while ((pos = expected.find(" ")) != std::string::npos) - expected.erase(pos, 1); + std::string expected("; ; void f ( ) { A a ; } ; ; class A { }"); ASSERT_EQUALS(expected, sizeof_(code)); } @@ -1139,13 +1105,12 @@ private: "}\n"; // The expected result.. - const std::string expected("template < int ui , typename T > T * foo ( )" - " { return new T [ ui ] ; }" - " void f ( )" - " {" - " foo<3,int> ( ) ;" - " }" - " int * foo<3,int> ( ) { return new int [ 3 ] ; }"); + const std::string expected("; " + "void f ( ) " + "{" + " foo<3,int> ( ) ; " + "} " + "int * foo<3,int> ( ) { return new int [ 3 ] ; }"); ASSERT_EQUALS(expected, sizeof_(code)); } @@ -1160,13 +1125,12 @@ private: "}\n"; // The expected result.. - const std::string expected("template < int ui , typename T > T * foo ( )" - " { return new T [ ui ] ; }" - " void f ( )" - " {" - " char * p ; p = foo<3,char> ( ) ;" - " }" - " char * foo<3,char> ( ) { return new char [ 3 ] ; }"); + const std::string expected("; " + "void f ( ) " + "{" + " char * p ; p = foo<3,char> ( ) ; " + "} " + "char * foo<3,char> ( ) { return new char [ 3 ] ; }"); ASSERT_EQUALS(expected, sizeof_(code)); } @@ -1182,15 +1146,13 @@ private: "}\n"; // The expected result.. - const std::string expected("template < int x , int y , int z >" - " class A : public B < x , y , ( x - y ) ? ( ( y < z ) ? 1 : - 1 ) : 0 >" - " { } ;" - " void f ( )" - " {" - " A<12,12,11> a ;" - " }" - " class A<12,12,11> : public B < 12 , 12 , 0 >" - " { }"); + const std::string expected("; ; " + "void f ( ) " + "{" + " A<12,12,11> a ; " + "} " + "class A<12,12,11> : public B < 12 , 12 , 0 > " + "{ }"); ASSERT_EQUALS(expected, sizeof_(code)); } @@ -1257,8 +1219,7 @@ private: "}\n"; // The expected result.. - const std::string expected("template < int i > void a ( ) " - "{ a < i - 1 > ( ) ; } " + const std::string expected("; " "void a<0> ( ) { } " "int main ( ) " "{ a<2> ( ) ; return 0 ; } " @@ -1268,6 +1229,30 @@ private: ASSERT_EQUALS(expected, sizeof_(code)); } + void template16() + { + const char code[] = "template void a()\n" + "{ }\n" + "\n" + "template void b()\n" + "{ a(); }\n" + "\n" + "int main()\n" + "{\n" + " b<2>();\n" + " return 0;\n" + "}\n"; + + // The expected result.. + const std::string expected("; " + "; " + "int main ( ) { b<2> ( ) ; return 0 ; } " + "void b<2> ( ) { a<2> ( ) ; } " + "void a<2> ( ) { }"); + + ASSERT_EQUALS(expected, sizeof_(code)); + } + void template_default_parameter() { { @@ -1282,18 +1267,16 @@ private: "}\n"; // The expected result.. - const std::string expected("template < class T , int n >" - " class A" - " { T ar [ n ] ; } ;" - " void f ( )" - " {" + const std::string expected("; ; " + "void f ( ) " + "{" " A a1 ;" - " A a2 ;" - " }" - " class A" - " { int ar [ 2 ] ; }" - " class A" - " { int ar [ 3 ] ; }"); + " A a2 ; " + "} " + "class A " + "{ int ar [ 2 ] ; } " + "class A " + "{ int ar [ 3 ] ; }"); ASSERT_EQUALS(expected, sizeof_(code)); } { @@ -1308,16 +1291,14 @@ private: "}\n"; // The expected result.. - const std::string expected("template < class T , int n1 , int n2 >" - " class A" - " { T ar [ n1 + n2 ] ; } ;" - " void f ( )" - " {" + const std::string expected("; ; " + "void f ( ) " + "{" " A a1 ;" - " A a2 ;" - " }" - " class A" - " { int ar [ 5 ] ; }"); + " A a2 ; " + "} " + "class A " + "{ int ar [ 5 ] ; }"); ASSERT_EQUALS(expected, sizeof_(code)); } { @@ -1355,9 +1336,7 @@ private: "{ }"; // The expected result.. - const std::string expected("template < class T >" - " void foo ( T :: t * )" - " { }"); + const std::string expected(";"); ASSERT_EQUALS(expected, sizeof_(code)); }