Fixed #3640 (False positive: statement begins with numeric code)

This commit is contained in:
Daniel Marjamäki 2012-05-04 17:53:47 +02:00
parent 119b24e363
commit 005ce81689
2 changed files with 30 additions and 9 deletions

View File

@ -925,7 +925,7 @@ std::list<std::string> Preprocessor::getcfgs(const std::string &filedata, const
unsigned int linenr = 0; unsigned int linenr = 0;
std::istringstream istr(filedata); std::istringstream istr(filedata);
std::string line; std::string line;
while (getline(istr, line)) { while (std::getline(istr, line)) {
++linenr; ++linenr;
if (_errorLogger) if (_errorLogger)
@ -949,10 +949,12 @@ std::list<std::string> Preprocessor::getcfgs(const std::string &filedata, const
if (line.compare(0, 8, "#define ") == 0) { if (line.compare(0, 8, "#define ") == 0) {
bool valid = true; bool valid = true;
for (std::string::size_type pos = 8; pos < line.size() && line[pos] != ' '; ++pos) { for (std::string::size_type pos = 8; pos < line.size(); ++pos) {
char ch = line[pos]; char ch = line[pos];
if (ch=='_' || (ch>='a' && ch<='z') || (ch>='A' && ch<='Z') || (pos>8 && ch>='0' && ch<='9')) if (ch=='_' || (ch>='a' && ch<='z') || (ch>='A' && ch<='Z') || (pos>8 && ch>='0' && ch<='9'))
continue; continue;
if (ch==' ' || ch=='(')
break;
valid = false; valid = false;
break; break;
} }
@ -1018,9 +1020,11 @@ std::list<std::string> Preprocessor::getcfgs(const std::string &filedata, const
{ {
std::map<std::string, std::string> varmap; std::map<std::string, std::string> varmap;
for (std::set<std::string>::const_iterator it = defines.begin(); it != defines.end(); ++it) { for (std::set<std::string>::const_iterator it = defines.begin(); it != defines.end(); ++it) {
std::string::size_type pos = it->find("="); std::string::size_type pos = it->find_first_of("=(");
if (pos == std::string::npos) if (pos == std::string::npos)
continue; continue;
if ((*it)[pos] == '(')
continue;
const std::string varname(it->substr(0, pos)); const std::string varname(it->substr(0, pos));
const std::string value(it->substr(pos + 1)); const std::string value(it->substr(pos + 1));
varmap[varname] = value; varmap[varname] = value;
@ -1142,16 +1146,16 @@ std::list<std::string> Preprocessor::getcfgs(const std::string &filedata, const
// Get name of define // Get name of define
std::string defineName(*it2); std::string defineName(*it2);
if (defineName.find("=") != std::string::npos) if (defineName.find_first_of("=(") != std::string::npos)
defineName.erase(defineName.find("=")); defineName.erase(defineName.find_first_of("=("));
// Remove ifdef configurations that match the defineName // Remove ifdef configurations that match the defineName
while ((pos = cfg.find(defineName, pos)) != std::string::npos) { while ((pos = cfg.find(defineName, pos)) != std::string::npos) {
std::string::size_type pos1 = pos; const std::string::size_type pos1 = pos;
++pos; ++pos;
if (pos1 > 0 && cfg[pos1-1] != ';') if (pos1 > 0 && cfg[pos1-1] != ';')
continue; continue;
std::string::size_type pos2 = pos1 + defineName.length(); const std::string::size_type pos2 = pos1 + defineName.length();
if (pos2 < cfg.length() && cfg[pos2] != ';') if (pos2 < cfg.length() && cfg[pos2] != ';')
continue; continue;
--pos; --pos;

View File

@ -212,6 +212,7 @@ private:
TEST_CASE(define_ifdef); TEST_CASE(define_ifdef);
TEST_CASE(define_ifndef1); TEST_CASE(define_ifndef1);
TEST_CASE(define_ifndef2); TEST_CASE(define_ifndef2);
TEST_CASE(ifndef_define);
TEST_CASE(undef_ifdef); TEST_CASE(undef_ifdef);
TEST_CASE(endfile); TEST_CASE(endfile);
@ -2706,8 +2707,7 @@ private:
// Compare results.. // Compare results..
ASSERT_EQUALS("\n\n\n\n", actual[""]); ASSERT_EQUALS("\n\n\n\n", actual[""]);
TODO_ASSERT_EQUALS(1, ASSERT_EQUALS(1U, actual.size());
2, actual.size());
} }
void define_ifndef2() { void define_ifndef2() {
@ -2725,6 +2725,23 @@ private:
ASSERT_EQUALS("\n\n\n\n\n\n$char me;\n", preprocessor.getcode(filedata, "A", "a.cpp")); ASSERT_EQUALS("\n\n\n\n\n\n$char me;\n", preprocessor.getcode(filedata, "A", "a.cpp"));
} }
void ifndef_define() {
const char filedata[] = "#ifndef A\n"
"#define A(x) x\n"
"#endif\n"
"A(123);";
// Preprocess => actual result..
std::istringstream istr(filedata);
std::map<std::string, std::string> actual;
Settings settings;
Preprocessor preprocessor(&settings, this);
preprocessor.preprocess(istr, actual, "file.c");
ASSERT_EQUALS(1U, actual.size());
ASSERT_EQUALS("\n\n\n$123;\n", actual[""]);
}
void undef_ifdef() { void undef_ifdef() {
const char filedata[] = "#undef A\n" const char filedata[] = "#undef A\n"
"#ifdef A\n" "#ifdef A\n"