cppcheck/test/testunusedfunctions.cpp

405 lines
11 KiB
C++
Raw Normal View History

2008-12-18 22:28:57 +01:00
/*
* Cppcheck - A tool for static C/C++ code analysis
2014-02-15 07:45:39 +01:00
* Copyright (C) 2007-2014 Daniel Marjamäki and Cppcheck team.
2008-12-18 22:28:57 +01:00
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
2008-12-18 22:28:57 +01:00
*/
2009-10-25 12:49:06 +01:00
#include "tokenize.h"
2008-12-18 22:28:57 +01:00
#include "testsuite.h"
2009-10-25 12:49:06 +01:00
#include "checkunusedfunctions.h"
2008-12-18 22:28:57 +01:00
#include <sstream>
extern std::ostringstream errout;
2011-10-13 20:53:06 +02:00
class TestUnusedFunctions : public TestFixture {
2008-12-18 22:28:57 +01:00
public:
2014-11-20 10:13:03 +01:00
TestUnusedFunctions() : TestFixture("TestUnusedFunctions")
{
}
2008-12-18 22:28:57 +01:00
private:
2014-11-20 10:13:03 +01:00
void run()
{
TEST_CASE(incondition);
TEST_CASE(return1);
2009-06-05 15:02:26 +02:00
TEST_CASE(return2);
TEST_CASE(callback1);
TEST_CASE(else1);
TEST_CASE(functionpointer);
TEST_CASE(template1);
TEST_CASE(template2);
TEST_CASE(template3);
TEST_CASE(throwIsNotAFunction);
TEST_CASE(unusedError);
TEST_CASE(unusedMain);
TEST_CASE(initializationIsNotAFunction);
TEST_CASE(operator1); // #3195
TEST_CASE(returnRef);
TEST_CASE(attribute); // #3471 - FP __attribute__(constructor)
TEST_CASE(initializer_list);
TEST_CASE(member_function_ternary);
2014-08-31 20:33:27 +02:00
TEST_CASE(boost);
TEST_CASE(multipleFiles); // same function name in multiple files
TEST_CASE(lineNumber); // Ticket 3059
TEST_CASE(ignore_declaration); // ignore declaration
2008-12-18 22:28:57 +01:00
}
2014-11-20 10:13:03 +01:00
void check(const char code[])
{
2008-12-18 22:28:57 +01:00
// Clear the error buffer..
errout.str("");
2010-07-08 13:01:53 +02:00
Settings settings;
settings.addEnabled("unusedFunction");
// Tokenize..
Tokenizer tokenizer(&settings, this);
std::istringstream istr(code);
tokenizer.tokenize(istr, "test.cpp");
// Check for unused functions..
2010-07-08 13:01:53 +02:00
CheckUnusedFunctions checkUnusedFunctions(&tokenizer, &settings, this);
std::list<Check::FileInfo*> fileInfo;
fileInfo.push_back(checkUnusedFunctions.getFileInfo(&tokenizer, &settings));
checkUnusedFunctions.analyseWholeProgram(fileInfo,*this);
delete fileInfo.back();
2008-12-18 22:28:57 +01:00
}
2014-11-20 10:13:03 +01:00
void incondition()
{
check("int f1()\n"
"{\n"
" if (f1())\n"
" { }\n"
"}");
ASSERT_EQUALS("", errout.str());
2008-12-18 22:28:57 +01:00
}
2014-11-20 10:13:03 +01:00
void return1()
{
check("int f1()\n"
"{\n"
" return f1();\n"
"}");
ASSERT_EQUALS("", errout.str());
2008-12-18 22:28:57 +01:00
}
2014-11-20 10:13:03 +01:00
void return2()
{
2009-06-05 15:02:26 +02:00
check("char * foo()\n"
"{\n"
" return *foo();\n"
"}");
2009-06-05 15:02:26 +02:00
ASSERT_EQUALS("", errout.str());
}
2014-11-20 10:13:03 +01:00
void callback1()
{
check("void f1()\n"
"{\n"
" void (*f)() = cond ? f1 : NULL;\n"
"}");
ASSERT_EQUALS("", errout.str());
2008-12-18 22:28:57 +01:00
}
2014-11-20 10:13:03 +01:00
void else1()
{
check("void f1()\n"
"{\n"
" if (cond) ;\n"
" else f1();\n"
"}");
ASSERT_EQUALS("", errout.str());
2008-12-18 22:28:57 +01:00
}
2014-11-20 10:13:03 +01:00
void functionpointer()
{
check("namespace abc {\n"
"void foo() { }\n"
"};\n"
"\n"
"int main()\n"
"{\n"
" f(&abc::foo);\n"
" return 0\n"
"}");
ASSERT_EQUALS("", errout.str());
check("namespace abc {\n"
"void foo() { }\n"
"};\n"
"\n"
"int main()\n"
"{\n"
" f = &abc::foo;\n"
" return 0\n"
"}");
ASSERT_EQUALS("", errout.str());
check("namespace abc {\n" // #3875
"void foo() { }\n"
"};\n"
"\n"
"int main()\n"
"{\n"
" f(abc::foo);\n"
" return 0\n"
"}");
ASSERT_EQUALS("", errout.str());
}
2014-11-20 10:13:03 +01:00
void template1()
{
check("template<class T> void foo() { }\n"
"\n"
"int main()\n"
"{\n"
" foo<int>();\n"
" return 0\n"
"}");
ASSERT_EQUALS("", errout.str());
}
2014-11-20 10:13:03 +01:00
void template2()
{
check("void f() { }\n"
"\n"
"template<class T> void g()\n"
"{\n"
" f();\n"
"}");
ASSERT_EQUALS("", errout.str());
}
2014-11-20 10:13:03 +01:00
void template3() // #4701
{
check("class X {\n"
"public:\n"
" void bar() { foo<int>(0); }\n"
"private:\n"
" template<typename T> void foo( T t ) const;\n"
"};\n"
"template<typename T> void X::foo( T t ) const { }\n");
ASSERT_EQUALS("[test.cpp:3]: (style) The function 'bar' is never used.\n", errout.str());
}
2014-11-20 10:13:03 +01:00
void throwIsNotAFunction()
{
check("struct A {void f() const throw () {}}; int main() {A a; a.f();}");
ASSERT_EQUALS("", errout.str());
}
2014-11-20 10:13:03 +01:00
void unusedError()
{
check("void foo() {}\n"
"int main()\n");
ASSERT_EQUALS("[test.cpp:1]: (style) The function 'foo' is never used.\n", errout.str());
check("void foo() const {}\n"
"int main()\n");
ASSERT_EQUALS("[test.cpp:1]: (style) The function 'foo' is never used.\n", errout.str());
check("void foo() const throw() {}\n"
"int main()\n");
ASSERT_EQUALS("[test.cpp:1]: (style) The function 'foo' is never used.\n", errout.str());
check("void foo() throw() {}\n"
"int main()\n");
ASSERT_EQUALS("[test.cpp:1]: (style) The function 'foo' is never used.\n", errout.str());
}
2014-11-20 10:13:03 +01:00
void unusedMain()
{
check("int main() { }");
ASSERT_EQUALS("", errout.str());
check("int _tmain() { }");
ASSERT_EQUALS("", errout.str());
check("int WinMain() { }");
ASSERT_EQUALS("", errout.str());
}
2014-11-20 10:13:03 +01:00
void initializationIsNotAFunction()
{
check("struct B: N::A {\n"
" B(): N::A() {};\n"
"};");
ASSERT_EQUALS("", errout.str());
}
2014-11-20 10:13:03 +01:00
void operator1()
{
check("struct Foo { void operator()(int a) {} };");
ASSERT_EQUALS("", errout.str());
}
2014-11-20 10:13:03 +01:00
void returnRef()
{
check("int& foo() {return x;}");
ASSERT_EQUALS("[test.cpp:1]: (style) The function 'foo' is never used.\n", errout.str());
}
2014-11-20 10:13:03 +01:00
void attribute() // #3471 - FP __attribute__((constructor))
{
check("void __attribute__((constructor)) f() {}");
ASSERT_EQUALS("", errout.str());
check("void __attribute__((constructor(1000))) f() {}");
ASSERT_EQUALS("", errout.str());
check("void __attribute__((destructor)) f() {}");
ASSERT_EQUALS("", errout.str());
check("void __attribute__((destructor(1000))) f() {}");
ASSERT_EQUALS("", errout.str());
// alternate syntax
check("__attribute__((constructor)) void f() {}");
ASSERT_EQUALS("", errout.str());
check("__attribute__((constructor(1000))) void f() {}");
ASSERT_EQUALS("", errout.str());
check("__attribute__((destructor)) void f() {}");
ASSERT_EQUALS("", errout.str());
check("__attribute__((destructor(1000))) void f() {}");
ASSERT_EQUALS("", errout.str());
// alternate syntax
check("void f() __attribute__((constructor));\n"
"void f() { }");
ASSERT_EQUALS("", errout.str());
check("void f() __attribute__((constructor(1000)));\n"
"void f() { }");
ASSERT_EQUALS("", errout.str());
check("void f() __attribute__((destructor));\n"
"void f() { }");
ASSERT_EQUALS("", errout.str());
check("void f() __attribute__((destructor(1000)));\n"
"void f() { }");
ASSERT_EQUALS("", errout.str());
// Don't crash on wrong syntax
check("int x __attribute__((constructor));\n"
"int x __attribute__((destructor));");
ASSERT_EQUALS("", errout.str());
}
2014-11-20 10:13:03 +01:00
void initializer_list()
{
check("int foo() { return 0; }\n"
"struct A {\n"
" A() : m_i(foo())\n"
" {}\n"
"int m_i;\n"
"};");
ASSERT_EQUALS("", errout.str());
}
2014-11-20 10:13:03 +01:00
void member_function_ternary()
{
check("struct Foo {\n"
" void F1() {}\n"
" void F2() {}\n"
"};\n"
"int main(int argc, char *argv[]) {\n"
" Foo foo;\n"
" void (Foo::*ptr)();\n"
" ptr = (argc > 1 && !strcmp(argv[1], \"F2\")) ? &Foo::F2 : &Foo::F1;\n"
" (foo.*ptr)();\n"
"}");
ASSERT_EQUALS("", errout.str());
}
2014-11-20 10:13:03 +01:00
void boost()
{
2014-08-31 20:33:27 +02:00
check("static void _xy(const char *b, const char *e)\n"
"{}\n"
"parse(line, blanks_p >> ident[&_xy] >> blanks_p >> eol_p).full");
ASSERT_EQUALS("", errout.str());
}
2014-11-20 10:13:03 +01:00
void multipleFiles()
{
CheckUnusedFunctions c;
// Clear the error buffer..
errout.str("");
const char code[] = "static void f() { }";
std::list<Check::FileInfo*> fileInfo;
2011-10-13 20:53:06 +02:00
for (int i = 1; i <= 2; ++i) {
std::ostringstream fname;
fname << "test" << i << ".cpp";
// Clear the error buffer..
errout.str("");
Settings settings;
settings.addEnabled("unusedFunction");
Tokenizer tokenizer(&settings, this);
std::istringstream istr(code);
tokenizer.tokenize(istr, fname.str().c_str());
fileInfo.push_back(c.getFileInfo(&tokenizer, &settings));
}
// Check for unused functions..
c.analyseWholeProgram(fileInfo,*this);
while (!fileInfo.empty()) {
delete fileInfo.back();
fileInfo.pop_back();
}
ASSERT_EQUALS("[test1.cpp:1]: (style) The function 'f' is never used.\n", errout.str());
}
2014-11-20 10:13:03 +01:00
void lineNumber()
{
check("void foo() {}\n"
"void bar() {}\n"
"int main()\n");
ASSERT_EQUALS("[test.cpp:2]: (style) The function 'bar' is never used.\n"
"[test.cpp:1]: (style) The function 'foo' is never used.\n", errout.str());
}
2014-11-20 10:13:03 +01:00
void ignore_declaration()
{
check("void f();\n"
"void f() {}");
ASSERT_EQUALS("[test.cpp:2]: (style) The function 'f' is never used.\n", errout.str());
check("void f(void) {}\n"
"void (*list[])(void) = {f}");
ASSERT_EQUALS("", errout.str());
}
2008-12-18 22:28:57 +01:00
};
REGISTER_TEST(TestUnusedFunctions)