Fixed #8111 (performance warning - member variable at constructor)

This commit is contained in:
Daniel Marjamäki 2019-01-07 21:26:58 +01:00
parent 30d53a911e
commit e4525d56a0
2 changed files with 52 additions and 28 deletions

View File

@ -930,31 +930,47 @@ void CheckClass::initializationListUsage()
break; break;
if (Token::Match(tok, "try|do {")) if (Token::Match(tok, "try|do {"))
break; break;
if (Token::Match(tok, "%var% =") && tok->strAt(-1) != "*") { if (!Token::Match(tok, "%var% =") || tok->strAt(-1) == "*")
continue;
const Variable* var = tok->variable(); const Variable* var = tok->variable();
if (var && var->scope() == owner && !var->isStatic()) { if (!var || var->scope() != owner || var->isStatic())
continue;
if (var->isPointer() || var->isReference() || var->isEnumType() || var->valueType()->type > ValueType::Type::ITERATOR) if (var->isPointer() || var->isReference() || var->isEnumType() || var->valueType()->type > ValueType::Type::ITERATOR)
continue; continue;
// Access local var member in rhs => do not warn
bool localmember = false;
visitAstNodes(tok->next()->astOperand2(),
[&](const Token *rhs) {
if (rhs->str() == "." && rhs->astOperand1() && rhs->astOperand1()->variable() && rhs->astOperand1()->variable()->isLocal())
localmember = true;
return ChildrenToVisit::op1_and_op2;
});
if (localmember)
continue;
bool allowed = true; bool allowed = true;
for (const Token* tok2 = tok->tokAt(2); tok2 && tok2->str() != ";"; tok2 = tok2->next()) { visitAstNodes(tok->next()->astOperand2(),
[&](const Token *tok2) {
const Variable* var2 = tok2->variable(); const Variable* var2 = tok2->variable();
if (var2) { if (var2) {
if (var2->scope() == owner && tok2->strAt(-1)!=".") { // Is there a dependency between two member variables? if (var2->scope() == owner && tok2->strAt(-1)!=".") { // Is there a dependency between two member variables?
allowed = false; allowed = false;
break; return ChildrenToVisit::done;
} else if (var2->isArray() && var2->isLocal()) { // Can't initialize with a local array } else if (var2->isArray() && var2->isLocal()) { // Can't initialize with a local array
allowed = false; allowed = false;
break; return ChildrenToVisit::done;
} }
} else if (tok2->str() == "this") { // 'this' instance is not completely constructed in initialization list } else if (tok2->str() == "this") { // 'this' instance is not completely constructed in initialization list
allowed = false; allowed = false;
break; return ChildrenToVisit::done;
} else if (Token::Match(tok2, "%name% (") && tok2->strAt(-1) != "." && isMemberFunc(owner, tok2)) { // Member function called? } else if (Token::Match(tok2, "%name% (") && tok2->strAt(-1) != "." && isMemberFunc(owner, tok2)) { // Member function called?
allowed = false; allowed = false;
break; return ChildrenToVisit::done;
}
} }
return ChildrenToVisit::op1_and_op2;
});
if (!allowed) if (!allowed)
continue; continue;
@ -962,8 +978,6 @@ void CheckClass::initializationListUsage()
} }
} }
} }
}
}
void CheckClass::suggestInitializationList(const Token* tok, const std::string& varname) void CheckClass::suggestInitializationList(const Token* tok, const std::string& varname)
{ {

View File

@ -6625,6 +6625,16 @@ private:
" Foo(int a, const std::string& b) : Foo(a, bar(b)) {}\n" " Foo(int a, const std::string& b) : Foo(a, bar(b)) {}\n"
"};"); "};");
ASSERT_EQUALS("", errout.str()); ASSERT_EQUALS("", errout.str());
checkInitializationListUsage("class Fred {\n" // #8111
" std::string a;\n"
" Fred() {\n"
" std::ostringstream ostr;\n"
" ostr << x;\n"
" a = ostr.str();\n"
" }\n"
"};");
ASSERT_EQUALS("", errout.str());
} }