CheckClass: Removed noExplicitCopyMoveConstructorError after discussion in http://sourceforge.net/p/cppcheck/discussion/general/thread/b2ce9d3d/.

This commit is contained in:
Daniel Marjamäki 2015-11-13 12:48:26 +01:00
parent ab90a7eb49
commit b10110b5ac
3 changed files with 11 additions and 22 deletions

View File

@ -241,8 +241,8 @@ void CheckClass::checkExplicitConstructors()
} }
} }
// Abstract classes can't be instantiated. But if there is "misuse" by derived // Abstract classes can't be instantiated. But if there is C++11
// classes then these constructors must be explicit. // "misuse" by derived classes then these constructors must be explicit.
if (isAbstractClass && _settings->standards.cpp != Standards::CPP11) if (isAbstractClass && _settings->standards.cpp != Standards::CPP11)
continue; continue;
@ -256,13 +256,11 @@ void CheckClass::checkExplicitConstructors()
if (!func->isConstructor() || func->isDelete() || (!func->hasBody() && func->access == Private)) if (!func->isConstructor() || func->isDelete() || (!func->hasBody() && func->access == Private))
continue; continue;
if (!func->isExplicit() && func->argCount() == 1) { if (!func->isExplicit() &&
// We must decide, if it is not a copy/move constructor, or it is a copy/move constructor of abstract class. func->argCount() == 1 &&
if (func->type != Function::eCopyConstructor && func->type != Function::eMoveConstructor) { func->type != Function::eCopyConstructor &&
noExplicitConstructorError(func->tokenDef, scope->className, scope->type == Scope::eStruct); func->type != Function::eMoveConstructor) {
} else if (isAbstractClass) { noExplicitConstructorError(func->tokenDef, scope->className, scope->type == Scope::eStruct);
noExplicitCopyMoveConstructorError(func->tokenDef, scope->className, scope->type == Scope::eStruct);
}
} }
} }
} }
@ -788,13 +786,6 @@ void CheckClass::noExplicitConstructorError(const Token *tok, const std::string
reportError(tok, Severity::style, "noExplicitConstructor", message + "\n" + verbose); reportError(tok, Severity::style, "noExplicitConstructor", message + "\n" + verbose);
} }
void CheckClass::noExplicitCopyMoveConstructorError(const Token *tok, const std::string &classname, bool isStruct)
{
const std::string message(std::string(isStruct ? "Abstract struct" : "Abstract class") + " '" + classname + "' has a copy/move constructor that is not explicit.");
const std::string verbose(message + " For abstract classes, even copy/move constructors may be declared explicit, as, by definition, abstract classes cannot be instantiated, and so objects of such type should never be passed by value.");
reportError(tok, Severity::style, "noExplicitCopyMoveConstructor", message + "\n" + verbose);
}
void CheckClass::uninitVarError(const Token *tok, const std::string &classname, const std::string &varname, bool inconclusive) void CheckClass::uninitVarError(const Token *tok, const std::string &classname, const std::string &varname, bool inconclusive)
{ {
reportError(tok, Severity::warning, "uninitMemberVar", "Member variable '" + classname + "::" + varname + "' is not initialized in the constructor.", 0U, inconclusive); reportError(tok, Severity::warning, "uninitMemberVar", "Member variable '" + classname + "::" + varname + "' is not initialized in the constructor.", 0U, inconclusive);

View File

@ -142,7 +142,6 @@ private:
// Reporting errors.. // Reporting errors..
void noConstructorError(const Token *tok, const std::string &classname, bool isStruct); void noConstructorError(const Token *tok, const std::string &classname, bool isStruct);
void noExplicitConstructorError(const Token *tok, const std::string &classname, bool isStruct); void noExplicitConstructorError(const Token *tok, const std::string &classname, bool isStruct);
void noExplicitCopyMoveConstructorError(const Token *tok, const std::string &classname, bool isStruct);
//void copyConstructorMallocError(const Token *cctor, const Token *alloc, const std::string& var_name); //void copyConstructorMallocError(const Token *cctor, const Token *alloc, const std::string& var_name);
void copyConstructorShallowCopyError(const Token *tok, const std::string& varname); void copyConstructorShallowCopyError(const Token *tok, const std::string& varname);
void noCopyConstructorError(const Token *tok, const std::string &classname, bool isStruct); void noCopyConstructorError(const Token *tok, const std::string &classname, bool isStruct);
@ -173,7 +172,6 @@ private:
CheckClass c(0, settings, errorLogger); CheckClass c(0, settings, errorLogger);
c.noConstructorError(0, "classname", false); c.noConstructorError(0, "classname", false);
c.noExplicitConstructorError(0, "classname", false); c.noExplicitConstructorError(0, "classname", false);
c.noExplicitCopyMoveConstructorError(0, "classname", false);
//c.copyConstructorMallocError(0, 0, "var"); //c.copyConstructorMallocError(0, 0, "var");
c.copyConstructorShallowCopyError(0, "var"); c.copyConstructorShallowCopyError(0, "var");
c.noCopyConstructorError(0, "class", false); c.noCopyConstructorError(0, "class", false);
@ -208,7 +206,7 @@ private:
return "Check the code for each class.\n" return "Check the code for each class.\n"
"- Missing constructors and copy constructors\n" "- Missing constructors and copy constructors\n"
//"- Missing allocation of memory in copy constructor\n" //"- Missing allocation of memory in copy constructor\n"
"- Constructors which should be explicit are explicit\n" "- Constructors which should be explicit\n"
"- Are all variables initialized by the constructors?\n" "- Are all variables initialized by the constructors?\n"
"- Are all variables assigned by 'operator='?\n" "- Are all variables assigned by 'operator='?\n"
"- Warn if memset, memcpy etc are used on a class\n" "- Warn if memset, memcpy etc are used on a class\n"

View File

@ -237,13 +237,13 @@ private:
" Class(const Class& other) { }\n" " Class(const Class& other) { }\n"
" virtual int i() = 0;\n" " virtual int i() = 0;\n"
"};"); "};");
ASSERT_EQUALS("[test.cpp:2]: (style) Abstract class 'Class' has a copy/move constructor that is not explicit.\n", errout.str()); ASSERT_EQUALS("", errout.str());
checkExplicitConstructors("class Class {\n" checkExplicitConstructors("class Class {\n"
" Class(Class&& other) { }\n" " Class(Class&& other) { }\n"
" virtual int i() = 0;\n" " virtual int i() = 0;\n"
"};"); "};");
ASSERT_EQUALS("[test.cpp:2]: (style) Abstract class 'Class' has a copy/move constructor that is not explicit.\n", errout.str()); ASSERT_EQUALS("", errout.str());
// #6585 // #6585
checkExplicitConstructors("class Class {\n" checkExplicitConstructors("class Class {\n"
@ -256,7 +256,7 @@ private:
" public: Class(const Class&);\n" " public: Class(const Class&);\n"
" virtual int i() = 0;\n" " virtual int i() = 0;\n"
"};"); "};");
ASSERT_EQUALS("[test.cpp:2]: (style) Abstract class 'Class' has a copy/move constructor that is not explicit.\n", errout.str()); ASSERT_EQUALS("", errout.str());
} }
void checkDuplInheritedMembers(const char code[]) { void checkDuplInheritedMembers(const char code[]) {