Added test 'CheckVariableScope'. Increased constness.
This commit is contained in:
parent
c366278b48
commit
a9524b9207
|
@ -517,5 +517,92 @@ void CheckUnsignedDivision()
|
||||||
declvar = findtoken(declvar->next, pattern_declvar);
|
declvar = findtoken(declvar->next, pattern_declvar);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
//---------------------------------------------------------------------------
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
//---------------------------------------------------------------------------
|
||||||
|
// Check scope of variables..
|
||||||
|
//---------------------------------------------------------------------------
|
||||||
|
|
||||||
|
static void CheckVariableScope_LookupVar( const TOKEN *tok1, const char varname[] );
|
||||||
|
|
||||||
|
void CheckVariableScope()
|
||||||
|
{
|
||||||
|
// Walk through all tokens..
|
||||||
|
bool func = false;
|
||||||
|
int indentlevel = 0;
|
||||||
|
for ( TOKEN *tok = tokens; tok; tok = tok->next )
|
||||||
|
{
|
||||||
|
if ( tok->str[0] == '{' )
|
||||||
|
{
|
||||||
|
indentlevel++;
|
||||||
|
}
|
||||||
|
if ( tok->str[0] == '}' )
|
||||||
|
{
|
||||||
|
indentlevel--;
|
||||||
|
if ( indentlevel == 0 )
|
||||||
|
func = false;
|
||||||
|
}
|
||||||
|
if ( indentlevel == 0 && match(tok, ") {") )
|
||||||
|
{
|
||||||
|
func = true;
|
||||||
|
}
|
||||||
|
if ( indentlevel > 0 && func && strchr("{};", tok->str[0]) )
|
||||||
|
{
|
||||||
|
// Variable declaration?
|
||||||
|
if (match(tok->next, "var var ;"))
|
||||||
|
CheckVariableScope_LookupVar( tok->next, getstr(tok, 2) );
|
||||||
|
|
||||||
|
// Variable declaration?
|
||||||
|
else if (match(tok->next, "var var ="))
|
||||||
|
CheckVariableScope_LookupVar( tok->next, getstr(tok, 2) );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
//---------------------------------------------------------------------------
|
||||||
|
|
||||||
|
static void CheckVariableScope_LookupVar( const TOKEN *tok1, const char varname[] )
|
||||||
|
{
|
||||||
|
const TOKEN *tok = tok1;
|
||||||
|
|
||||||
|
// Skip the variable declaration..
|
||||||
|
tok = tok->next;
|
||||||
|
while ( tok->str[0] != ';' )
|
||||||
|
tok = tok->next;
|
||||||
|
|
||||||
|
// Check if the variable is used in this indentlevel..
|
||||||
|
bool used = false;
|
||||||
|
int indentlevel = 0;
|
||||||
|
while ( indentlevel >= 0 && tok )
|
||||||
|
{
|
||||||
|
if ( tok->str[0] == '{' )
|
||||||
|
{
|
||||||
|
indentlevel++;
|
||||||
|
}
|
||||||
|
|
||||||
|
else if ( tok->str[0] == '}' )
|
||||||
|
{
|
||||||
|
indentlevel--;
|
||||||
|
}
|
||||||
|
|
||||||
|
else if ( strcmp(tok->str, varname) == 0 )
|
||||||
|
{
|
||||||
|
if ( indentlevel == 0 )
|
||||||
|
return;
|
||||||
|
used = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
tok = tok->next;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Warning if "used" is true
|
||||||
|
std::ostringstream errmsg;
|
||||||
|
errmsg << FileLine(tok1) << " The scope of the variable '" << varname << "' can be limited";
|
||||||
|
ReportErr( errmsg.str() );
|
||||||
|
}
|
||||||
|
//---------------------------------------------------------------------------
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -37,6 +37,9 @@ void CheckCaseWithoutBreak();
|
||||||
// Check for unsigned division that might create bad results
|
// Check for unsigned division that might create bad results
|
||||||
void CheckUnsignedDivision();
|
void CheckUnsignedDivision();
|
||||||
|
|
||||||
|
// Check scope of variables
|
||||||
|
void CheckVariableScope();
|
||||||
|
|
||||||
//---------------------------------------------------------------------------
|
//---------------------------------------------------------------------------
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
|
@ -11,7 +11,7 @@ bool OnlyReportUniqueErrors;
|
||||||
std::ostringstream errout;
|
std::ostringstream errout;
|
||||||
//---------------------------------------------------------------------------
|
//---------------------------------------------------------------------------
|
||||||
|
|
||||||
std::string FileLine(TOKEN *tok)
|
std::string FileLine(const TOKEN *tok)
|
||||||
{
|
{
|
||||||
std::ostringstream ostr;
|
std::ostringstream ostr;
|
||||||
ostr << "[" << Files[tok->FileIndex] << ":" << tok->linenr << "]";
|
ostr << "[" << Files[tok->FileIndex] << ":" << tok->linenr << "]";
|
||||||
|
|
|
@ -8,7 +8,7 @@
|
||||||
|
|
||||||
struct TOKEN;
|
struct TOKEN;
|
||||||
|
|
||||||
std::string FileLine(TOKEN *tok);
|
std::string FileLine(const TOKEN *tok);
|
||||||
|
|
||||||
extern bool OnlyReportUniqueErrors;
|
extern bool OnlyReportUniqueErrors;
|
||||||
|
|
||||||
|
|
21
tests.cpp
21
tests.cpp
|
@ -714,5 +714,26 @@ static void unused_variable()
|
||||||
"}\n";
|
"}\n";
|
||||||
|
|
||||||
|
|
||||||
|
// Scope of variable..
|
||||||
|
const char test2[] = "void f()\n"
|
||||||
|
"{\n"
|
||||||
|
" int i;\n"
|
||||||
|
" if (abc)\n"
|
||||||
|
" {\n"
|
||||||
|
" i = 1;\n"
|
||||||
|
" }\n"
|
||||||
|
"}\n";
|
||||||
|
check( CheckVariableScope, __LINE__, test2, "[test.cpp:3] The scope of the variable 'i' can be limited\n" );
|
||||||
|
/*
|
||||||
|
const char test3[] = "void f()\n"
|
||||||
|
"{\n"
|
||||||
|
" int i = 0;\n"
|
||||||
|
" while (abc)\n"
|
||||||
|
" {\n"
|
||||||
|
" i = i + 1;\n"
|
||||||
|
" }\n"
|
||||||
|
"}\n";
|
||||||
|
check( CheckVariableScope, __LINE__, test3, "" );
|
||||||
|
*/
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1018,12 +1018,12 @@ TOKEN *findtoken(TOKEN *tok1, const char *tokenstr[])
|
||||||
}
|
}
|
||||||
//---------------------------------------------------------------------------
|
//---------------------------------------------------------------------------
|
||||||
|
|
||||||
bool match(TOKEN *tok, const std::string pattern)
|
bool match(const TOKEN *tok, const char pattern[])
|
||||||
{
|
{
|
||||||
if (!tok)
|
if (!tok)
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
const char *p = pattern.c_str();
|
const char *p = pattern;
|
||||||
while (*p)
|
while (*p)
|
||||||
{
|
{
|
||||||
char str[50];
|
char str[50];
|
||||||
|
|
|
@ -36,7 +36,7 @@ void DeallocateTokens();
|
||||||
|
|
||||||
// Helper functions for handling the tokens list..
|
// Helper functions for handling the tokens list..
|
||||||
TOKEN *findtoken(TOKEN *tok1, const char *tokenstr[]);
|
TOKEN *findtoken(TOKEN *tok1, const char *tokenstr[]);
|
||||||
bool match(TOKEN *tok, const std::string pattern);
|
bool match(const TOKEN *tok, const char pattern[]);
|
||||||
TOKEN *gettok(TOKEN *tok, int index);
|
TOKEN *gettok(TOKEN *tok, int index);
|
||||||
const char *getstr(TOKEN *tok, int index);
|
const char *getstr(TOKEN *tok, int index);
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue