Fixed ticket #3528 (Tokenizer: improve simplifyFunctionParameters to take count of square brackets)
This commit is contained in:
parent
9f139cf414
commit
5953ed7318
169
lib/tokenize.cpp
169
lib/tokenize.cpp
|
@ -1992,7 +1992,8 @@ bool Tokenizer::tokenize(std::istream &code,
|
||||||
|
|
||||||
// Convert K&R function declarations to modern C
|
// Convert K&R function declarations to modern C
|
||||||
simplifyVarDecl(true);
|
simplifyVarDecl(true);
|
||||||
simplifyFunctionParameters();
|
if (!simplifyFunctionParameters())
|
||||||
|
return false;
|
||||||
|
|
||||||
// check for simple syntax errors..
|
// check for simple syntax errors..
|
||||||
for (const Token *tok = _tokens; tok; tok = tok->next()) {
|
for (const Token *tok = _tokens; tok; tok = tok->next()) {
|
||||||
|
@ -4801,7 +4802,7 @@ void Tokenizer::simplifyCasts()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void Tokenizer::simplifyFunctionParameters()
|
bool Tokenizer::simplifyFunctionParameters()
|
||||||
{
|
{
|
||||||
for (Token *tok = _tokens; tok; tok = tok->next()) {
|
for (Token *tok = _tokens; tok; tok = tok->next()) {
|
||||||
if (tok->str() == "{" || tok->str() == "[" || tok->str() == "(") {
|
if (tok->str() == "{" || tok->str() == "[" || tok->str() == "(") {
|
||||||
|
@ -4812,79 +4813,123 @@ void Tokenizer::simplifyFunctionParameters()
|
||||||
else if (Token::Match(tok, "%var% ( %var% [,)]")) {
|
else if (Token::Match(tok, "%var% ( %var% [,)]")) {
|
||||||
// We have found old style function, now we need to change it
|
// We have found old style function, now we need to change it
|
||||||
|
|
||||||
// backup pointer to the '(' token
|
// First step: Get list of argument names in parenthesis
|
||||||
Token * const tok1 = tok->next();
|
std::map<std::string, Token *> argumentNames;
|
||||||
|
|
||||||
// Get list of argument names
|
|
||||||
std::map<std::string, Token*> argumentNames;
|
|
||||||
bool bailOut = false;
|
bool bailOut = false;
|
||||||
for (tok = tok->tokAt(2); tok; tok = tok->tokAt(2)) {
|
Token * tokparam = NULL;
|
||||||
if (!Token::Match(tok, "%var% [,)]")) {
|
|
||||||
bailOut = true;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (argumentNames.find(tok->str()) != argumentNames.end()) {
|
//floating token used to check for parameters
|
||||||
// Invalid code, two arguments with the same name.
|
Token *tok1 = tok;
|
||||||
// syntaxError(tok);
|
|
||||||
// If you uncomment it, testrunner will fail:
|
|
||||||
// testclass.cpp:3910
|
|
||||||
// because of void Fred::foo5(int, int).
|
|
||||||
// how should this be handled?
|
|
||||||
bailOut = true;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
argumentNames[tok->str()] = tok;
|
//goto '('
|
||||||
if (tok->next()->str() == ")") {
|
|
||||||
tok = tok->tokAt(2);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (bailOut) {
|
|
||||||
tok = tok1->link();
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
Token *start = tok;
|
|
||||||
while (tok && tok->str() != "{") {
|
|
||||||
if (tok->str() == ";") {
|
|
||||||
tok = tok->previous();
|
|
||||||
// Move tokens from start to tok into the place of
|
|
||||||
// argumentNames[tok->str()] and remove the ";"
|
|
||||||
|
|
||||||
if (argumentNames.find(tok->str()) == argumentNames.end()) {
|
|
||||||
bailOut = true;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Remove the following ";"
|
|
||||||
Token *temp = tok->tokAt(2);
|
|
||||||
tok->deleteNext();
|
|
||||||
|
|
||||||
// Replace "x" with "int x" or similar
|
|
||||||
Token::replace(argumentNames[tok->str()], start, tok);
|
|
||||||
argumentNames.erase(tok->str());
|
|
||||||
tok = temp;
|
|
||||||
start = tok;
|
|
||||||
} else {
|
|
||||||
tok = tok->next();
|
tok = tok->next();
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (Token::simpleMatch(tok, "{"))
|
while (NULL != (tok1 = tok1->tokAt(2))) {
|
||||||
tok = tok->link();
|
if (!Token::Match(tok1, "%var% [,)]")) {
|
||||||
|
bailOut = true;
|
||||||
if (tok == NULL) {
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//same parameters: take note of the parameter
|
||||||
|
if (argumentNames.find(tok1->str()) != argumentNames.end())
|
||||||
|
tokparam = tok1;
|
||||||
|
else
|
||||||
|
argumentNames[tok1->str()] = tok1;
|
||||||
|
|
||||||
|
if (tok1->next()->str() == ")") {
|
||||||
|
tok1 = tok1->tokAt(2);
|
||||||
|
//expect at least a type name after round brace..
|
||||||
|
if (!tok1 || !tok1->isName())
|
||||||
|
bailOut = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (bailOut) {
|
if (bailOut) {
|
||||||
|
tok = tok->link();
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//there should be the sequence '; {' after the round parenthesis
|
||||||
|
if (!Token::findsimplematch(tok1, "; {")) {
|
||||||
|
tok = tok->link();
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Last step: check out if the declarations between ')' and '{' match the parameters list
|
||||||
|
std::map<std::string, Token *> argumentNames2;
|
||||||
|
|
||||||
|
while (tok1 && tok1->str() != "{") {
|
||||||
|
if (tok1->str() == ";") {
|
||||||
|
if (tokparam) {
|
||||||
|
syntaxError(tokparam);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
Token *tok2 = tok1->previous();
|
||||||
|
while (tok2->str() == "]")
|
||||||
|
tok2 = tok2->link()->previous();
|
||||||
|
|
||||||
|
//it should be a name..
|
||||||
|
if (!tok2->isName()) {
|
||||||
|
bailOut = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (argumentNames2.find(tok2->str()) != argumentNames2.end()) {
|
||||||
|
//same parameter names...
|
||||||
|
syntaxError(tok1);
|
||||||
|
return false;
|
||||||
|
} else
|
||||||
|
argumentNames2[tok2->str()] = tok2;
|
||||||
|
|
||||||
|
if (argumentNames.find(tok2->str()) == argumentNames.end()) {
|
||||||
|
//non-matching parameter... bailout
|
||||||
|
bailOut = true;
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
tok1 = tok1->next();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (bailOut) {
|
||||||
|
tok = tok->link();
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
//the two containers should hold the same size..
|
||||||
|
if (argumentNames.size() != argumentNames2.size()) {
|
||||||
|
tok = tok->link();
|
||||||
|
//error if the second container is not empty
|
||||||
|
if (!argumentNames2.empty()) {
|
||||||
|
syntaxError(tok);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
while (tok->str() != ")") {
|
||||||
|
//initialize start and end tokens to be moved
|
||||||
|
Token *declStart = argumentNames2[tok->next()->str()];
|
||||||
|
Token *declEnd = declStart;
|
||||||
|
while (declStart->previous()->str() != ";" && declStart->previous()->str() != ")")
|
||||||
|
declStart = declStart->previous();
|
||||||
|
while (declEnd->next()->str() != ";" && declEnd->next()->str() != "{")
|
||||||
|
declEnd = declEnd->next();
|
||||||
|
|
||||||
|
//remove ';' after declaration
|
||||||
|
declEnd->deleteNext();
|
||||||
|
|
||||||
|
//replace the parameter name in the parenthesis with all the declaration
|
||||||
|
Token::replace(tok->next(), declStart, declEnd);
|
||||||
|
|
||||||
|
//since there are changes to tokens, put tok where tok1 is
|
||||||
|
tok = declEnd->next();
|
||||||
|
}
|
||||||
|
//goto forward and continue
|
||||||
|
tok = tok->next()->link();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
void Tokenizer::simplifyPointerToStandardType()
|
void Tokenizer::simplifyPointerToStandardType()
|
||||||
|
|
|
@ -449,8 +449,9 @@ public:
|
||||||
/**
|
/**
|
||||||
* Simplify functions like "void f(x) int x; {"
|
* Simplify functions like "void f(x) int x; {"
|
||||||
* into "void f(int x) {"
|
* into "void f(int x) {"
|
||||||
|
* @return false only if there's a syntax error
|
||||||
*/
|
*/
|
||||||
void simplifyFunctionParameters();
|
bool simplifyFunctionParameters();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Simplify templates
|
* Simplify templates
|
||||||
|
|
|
@ -170,11 +170,10 @@ private:
|
||||||
TEST_CASE(strcat1);
|
TEST_CASE(strcat1);
|
||||||
TEST_CASE(strcat2);
|
TEST_CASE(strcat2);
|
||||||
|
|
||||||
// Syntax error
|
|
||||||
TEST_CASE(argumentsWithSameName)
|
|
||||||
|
|
||||||
TEST_CASE(simplifyAtol)
|
TEST_CASE(simplifyAtol)
|
||||||
|
|
||||||
TEST_CASE(simplifyHexInString)
|
TEST_CASE(simplifyHexInString)
|
||||||
|
|
||||||
TEST_CASE(simplifyTypedef1)
|
TEST_CASE(simplifyTypedef1)
|
||||||
TEST_CASE(simplifyTypedef2)
|
TEST_CASE(simplifyTypedef2)
|
||||||
TEST_CASE(simplifyTypedef3)
|
TEST_CASE(simplifyTypedef3)
|
||||||
|
@ -3392,25 +3391,6 @@ private:
|
||||||
ASSERT_EQUALS(expect, tok(code));
|
ASSERT_EQUALS(expect, tok(code));
|
||||||
}
|
}
|
||||||
|
|
||||||
void argumentsWithSameName() {
|
|
||||||
// This code has syntax error, two variables can not have the same name
|
|
||||||
{
|
|
||||||
const char code[] = "void foo(x, x)\n"
|
|
||||||
" int x;\n"
|
|
||||||
" int x;\n"
|
|
||||||
"{}\n";
|
|
||||||
ASSERT_EQUALS("void foo ( x , x ) int x ; int x ; { }", tok(code));
|
|
||||||
}
|
|
||||||
|
|
||||||
{
|
|
||||||
const char code[] = "void foo(x, y)\n"
|
|
||||||
" int x;\n"
|
|
||||||
" int x;\n"
|
|
||||||
"{}\n";
|
|
||||||
ASSERT_EQUALS("void foo ( int x , y ) int x ; { }", tok(code));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void simplifyAtol() {
|
void simplifyAtol() {
|
||||||
ASSERT_EQUALS("a = std :: atol ( x ) ;", tok("a = std::atol(x);"));
|
ASSERT_EQUALS("a = std :: atol ( x ) ;", tok("a = std::atol(x);"));
|
||||||
ASSERT_EQUALS("a = atol ( \"text\" ) ;", tok("a = atol(\"text\");"));
|
ASSERT_EQUALS("a = atol ( \"text\" ) ;", tok("a = atol(\"text\");"));
|
||||||
|
|
|
@ -240,7 +240,8 @@ private:
|
||||||
|
|
||||||
TEST_CASE(macrodoublesharp);
|
TEST_CASE(macrodoublesharp);
|
||||||
|
|
||||||
TEST_CASE(simplify_function_parameters);
|
TEST_CASE(simplifyFunctionParameters);
|
||||||
|
TEST_CASE(simplifyFunctionParametersErrors);
|
||||||
|
|
||||||
TEST_CASE(removeParentheses1); // Ticket #61
|
TEST_CASE(removeParentheses1); // Ticket #61
|
||||||
TEST_CASE(removeParentheses2);
|
TEST_CASE(removeParentheses2);
|
||||||
|
@ -3985,7 +3986,7 @@ private:
|
||||||
ASSERT_EQUALS("DBG ( fmt , args . . . ) printf ( fmt , ## args ) ", ostr.str());
|
ASSERT_EQUALS("DBG ( fmt , args . . . ) printf ( fmt , ## args ) ", ostr.str());
|
||||||
}
|
}
|
||||||
|
|
||||||
void simplify_function_parameters() {
|
void simplifyFunctionParameters() {
|
||||||
{
|
{
|
||||||
const char code[] = "char a [ ABC ( DEF ) ] ;";
|
const char code[] = "char a [ ABC ( DEF ) ] ;";
|
||||||
ASSERT_EQUALS(code, tokenizeAndStringify(code, true));
|
ASSERT_EQUALS(code, tokenizeAndStringify(code, true));
|
||||||
|
@ -3998,6 +3999,7 @@ private:
|
||||||
|
|
||||||
ASSERT_EQUALS("void f ( int x ) { }", tokenizeAndStringify("void f(x) int x; { }", true));
|
ASSERT_EQUALS("void f ( int x ) { }", tokenizeAndStringify("void f(x) int x; { }", true));
|
||||||
ASSERT_EQUALS("void f ( int x , char y ) { }", tokenizeAndStringify("void f(x,y) int x; char y; { }", true));
|
ASSERT_EQUALS("void f ( int x , char y ) { }", tokenizeAndStringify("void f(x,y) int x; char y; { }", true));
|
||||||
|
ASSERT_EQUALS("int main ( int argc , char * argv [ ] ) { }", tokenizeAndStringify("int main(argc,argv) int argc; char *argv[]; { }", true));
|
||||||
|
|
||||||
// #1067 - Not simplified. Feel free to fix so it is simplified correctly but this syntax is obsolete.
|
// #1067 - Not simplified. Feel free to fix so it is simplified correctly but this syntax is obsolete.
|
||||||
ASSERT_EQUALS("int ( * d ( a , b , c ) ) ( ) int a ; int b ; int c ; { }", tokenizeAndStringify("int (*d(a,b,c))()int a,b,c; { }", true));
|
ASSERT_EQUALS("int ( * d ( a , b , c ) ) ( ) int a ; int b ; int c ; { }", tokenizeAndStringify("int (*d(a,b,c))()int a,b,c; { }", true));
|
||||||
|
@ -4014,6 +4016,32 @@ private:
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void simplifyFunctionParametersErrors() {
|
||||||
|
//same parameters...
|
||||||
|
tokenizeAndStringify("void foo(x, x)\n"
|
||||||
|
" int x;\n"
|
||||||
|
" int x;\n"
|
||||||
|
"{}\n");
|
||||||
|
ASSERT_EQUALS("[test.cpp:1]: (error) syntax error\n", errout.str());
|
||||||
|
|
||||||
|
tokenizeAndStringify("void foo(x, y)\n"
|
||||||
|
" int x;\n"
|
||||||
|
" int x;\n"
|
||||||
|
"{}\n");
|
||||||
|
ASSERT_EQUALS("[test.cpp:3]: (error) syntax error\n", errout.str());
|
||||||
|
|
||||||
|
tokenizeAndStringify("void foo(int, int)\n"
|
||||||
|
"{}\n");
|
||||||
|
ASSERT_EQUALS("", errout.str());
|
||||||
|
|
||||||
|
//non-matching arguments after round braces
|
||||||
|
tokenizeAndStringify("void foo(x, y, z)\n"
|
||||||
|
" int x;\n"
|
||||||
|
" int y;\n"
|
||||||
|
"{}\n");
|
||||||
|
ASSERT_EQUALS("[test.cpp:1]: (error) syntax error\n", errout.str());
|
||||||
|
}
|
||||||
|
|
||||||
// Simplify "((..))" into "(..)"
|
// Simplify "((..))" into "(..)"
|
||||||
void removeParentheses1() {
|
void removeParentheses1() {
|
||||||
const char code[] = "void foo()"
|
const char code[] = "void foo()"
|
||||||
|
|
Loading…
Reference in New Issue