2014-01-04 20:57:02 +01:00
|
|
|
/*
|
|
|
|
* Cppcheck - A tool for static C/C++ code analysis
|
|
|
|
* Copyright (C) 2007-2013 Daniel Marjamäki and Cppcheck team.
|
|
|
|
*
|
|
|
|
* 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/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "testsuite.h"
|
|
|
|
#include "testutils.h"
|
|
|
|
#include "valueflow.h"
|
|
|
|
#include "tokenize.h"
|
|
|
|
#include "token.h"
|
|
|
|
|
|
|
|
#include <vector>
|
|
|
|
#include <string>
|
|
|
|
|
|
|
|
extern std::ostringstream errout;
|
|
|
|
class TestValueFlow : public TestFixture {
|
|
|
|
public:
|
|
|
|
TestValueFlow() : TestFixture("TestValueFlow") {
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
|
|
|
void run() {
|
2014-01-06 10:09:49 +01:00
|
|
|
TEST_CASE(valueFlowBeforeCondition);
|
2014-01-07 19:20:56 +01:00
|
|
|
TEST_CASE(valueFlowForLoop);
|
2014-01-06 16:37:52 +01:00
|
|
|
TEST_CASE(valueFlowSubFunction);
|
2014-01-04 20:57:02 +01:00
|
|
|
}
|
|
|
|
|
2014-01-11 14:31:51 +01:00
|
|
|
bool testValueOfX(const std::string &code, unsigned int linenr, int value) {
|
2014-01-04 20:57:02 +01:00
|
|
|
Settings settings;
|
|
|
|
settings.valueFlow = true; // temporary flag
|
|
|
|
|
|
|
|
// Tokenize..
|
|
|
|
Tokenizer tokenizer(&settings, this);
|
|
|
|
std::istringstream istr(code);
|
|
|
|
tokenizer.tokenize(istr, "test.cpp");
|
|
|
|
|
|
|
|
for (const Token *tok = tokenizer.tokens(); tok; tok = tok->next()) {
|
|
|
|
if (tok->str() == "x" && tok->linenr() == linenr) {
|
|
|
|
std::list<ValueFlow::Value>::const_iterator it;
|
|
|
|
for (it = tok->values.begin(); it != tok->values.end(); ++it) {
|
|
|
|
if (it->intvalue == value)
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2014-01-06 07:44:58 +01:00
|
|
|
|
|
|
|
void bailout(const char code[]) {
|
|
|
|
Settings settings;
|
|
|
|
settings.valueFlow = true; // temporary flag
|
|
|
|
settings.debugwarnings = true;
|
|
|
|
|
|
|
|
// Tokenize..
|
|
|
|
Tokenizer tokenizer(&settings, this);
|
|
|
|
std::istringstream istr(code);
|
|
|
|
errout.str("");
|
|
|
|
tokenizer.tokenize(istr, "test.cpp");
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-01-04 20:57:02 +01:00
|
|
|
void valueFlowBeforeCondition() {
|
2014-01-08 06:39:15 +01:00
|
|
|
const char *code;
|
|
|
|
|
|
|
|
code = "void f(int x) {\n"
|
|
|
|
" int a = x;\n"
|
|
|
|
" if (x == 123) {}\n"
|
|
|
|
"}";
|
2014-01-04 20:57:02 +01:00
|
|
|
ASSERT_EQUALS(true, testValueOfX(code, 2U, 123));
|
2014-01-06 07:44:58 +01:00
|
|
|
|
2014-01-08 06:53:17 +01:00
|
|
|
code = "void f(unsigned int x) {\n"
|
|
|
|
" int a = x;\n"
|
|
|
|
" if (x >= 1) {}\n"
|
|
|
|
"}";
|
|
|
|
ASSERT_EQUALS(true, testValueOfX(code, 2U, 1));
|
|
|
|
ASSERT_EQUALS(true, testValueOfX(code, 2U, 0));
|
2014-01-08 17:37:39 +01:00
|
|
|
|
|
|
|
code = "void f(int *x) {\n"
|
|
|
|
" *x = 100;\n"
|
|
|
|
" if (x) {}\n"
|
|
|
|
"}";
|
|
|
|
ASSERT_EQUALS(true, testValueOfX(code, 2U, 0));
|
2014-01-08 06:53:17 +01:00
|
|
|
|
2014-01-10 05:47:56 +01:00
|
|
|
code = "extern const int x;\n"
|
|
|
|
"void f() {\n"
|
|
|
|
" int a = x;\n"
|
|
|
|
" if (x == 123) {}\n"
|
|
|
|
"}";
|
|
|
|
ASSERT_EQUALS(true, testValueOfX(code, 3U, 123));
|
|
|
|
|
2014-01-11 15:36:09 +01:00
|
|
|
// assignment
|
|
|
|
code = "void f(int x) {\n"
|
|
|
|
" x = 2 + x;\n"
|
|
|
|
" if (x == 65);\n"
|
|
|
|
"}";
|
|
|
|
ASSERT_EQUALS(false, testValueOfX(code, 2U, 65));
|
|
|
|
|
2014-01-11 20:53:23 +01:00
|
|
|
code = "void f(int x) {\n"
|
|
|
|
" x = y = 2 + x;\n"
|
|
|
|
" if (x == 65);\n"
|
|
|
|
"}";
|
|
|
|
ASSERT_EQUALS(false, testValueOfX(code, 2U, 65));
|
2014-01-11 15:36:09 +01:00
|
|
|
|
2014-01-11 11:30:34 +01:00
|
|
|
// guarding by &&
|
2014-01-11 12:44:55 +01:00
|
|
|
code = "void f(int x) {\n"
|
|
|
|
" if (!x || \n" // <- x can be 0
|
|
|
|
" a/x) {}\n" // <- x can't be 0
|
|
|
|
" if (x==0) {}\n"
|
|
|
|
"}";
|
|
|
|
ASSERT_EQUALS(true, testValueOfX(code, 2U, 0));
|
2014-01-11 11:30:34 +01:00
|
|
|
ASSERT_EQUALS(false, testValueOfX(code, 3U, 0));
|
|
|
|
|
2014-01-11 14:54:10 +01:00
|
|
|
code = "void f(int *x) {\n"
|
2014-01-11 15:18:39 +01:00
|
|
|
" ((x=ret())&&\n"
|
|
|
|
" (*x==0));\n" // <- x is not 0
|
2014-01-11 14:54:10 +01:00
|
|
|
" if (x==0) {}\n"
|
|
|
|
"}";
|
|
|
|
ASSERT_EQUALS(false, testValueOfX(code, 3U, 0));
|
|
|
|
|
2014-01-11 14:31:51 +01:00
|
|
|
// function calls
|
|
|
|
code = "void f(int x) {\n"
|
|
|
|
" a = x;\n"
|
|
|
|
" setx(x);\n"
|
|
|
|
" if (x == 1) {}\n"
|
|
|
|
"}";
|
|
|
|
ASSERT_EQUALS(true, testValueOfX(std::string("void setx(int x);")+code, 2U, 1));
|
|
|
|
ASSERT_EQUALS(false, testValueOfX(std::string("void setx(int &x);")+code, 2U, 1));
|
|
|
|
ASSERT_EQUALS(false, testValueOfX(code, 2U, 1));
|
|
|
|
|
2014-01-11 20:25:49 +01:00
|
|
|
// while, for, do-while
|
2014-01-12 11:58:10 +01:00
|
|
|
code = "void f(int x) {\n" // loop condition, x is not assigned inside loop => use condition
|
2014-01-11 20:25:49 +01:00
|
|
|
" a = x;\n" // x can be 37
|
|
|
|
" while (x == 37) {}\n"
|
|
|
|
"}";
|
|
|
|
ASSERT_EQUALS(true, testValueOfX(code, 2U, 37));
|
|
|
|
|
2014-01-12 11:58:10 +01:00
|
|
|
code = "void f(int x) {\n" // loop condition, x is assigned inside loop => dont use condition
|
2014-01-11 20:25:49 +01:00
|
|
|
" a = x;\n" // don't assume that x can be 37
|
|
|
|
" while (x != 37) { x++; }\n"
|
|
|
|
"}";
|
|
|
|
ASSERT_EQUALS(false, testValueOfX(code, 2U, 37));
|
|
|
|
|
2014-01-12 11:58:10 +01:00
|
|
|
code = "void f(int x) {\n" // condition inside loop, x is NOT assigned inside loop => use condition
|
|
|
|
" a = x;\n"
|
|
|
|
" do {\n"
|
|
|
|
" if (x==76) {}\n"
|
|
|
|
" } while (1);\n"
|
|
|
|
"}";
|
|
|
|
ASSERT_EQUALS(true, testValueOfX(code, 2U, 76));
|
|
|
|
|
|
|
|
code = "void f(int x) {\n" // conditions inside loop, x is assigned inside do-while => dont use condition
|
|
|
|
" a = x;\n"
|
|
|
|
" do {\n"
|
|
|
|
" if (x!=76) { x=do_something(); }\n"
|
|
|
|
" } while (1);\n"
|
|
|
|
"}";
|
|
|
|
ASSERT_EQUALS(false, testValueOfX(code, 2U, 76));
|
|
|
|
|
2014-01-07 19:46:13 +01:00
|
|
|
// bailout: ?:
|
|
|
|
bailout("void f(int x) {\n"
|
|
|
|
" y = ((x<0) ? x : ((x==2)?3:4));\n"
|
|
|
|
"}");
|
2014-01-08 06:39:15 +01:00
|
|
|
ASSERT_EQUALS("[test.cpp:2]: (debug) ValueFlow bailout: no simplification of x within ?: expression\n", errout.str());
|
|
|
|
|
|
|
|
bailout("int f(int x) {\n"
|
|
|
|
" int r = x ? 1 / x : 0;\n"
|
|
|
|
" if (x == 0) {}\n"
|
|
|
|
"}");
|
|
|
|
ASSERT_EQUALS("[test.cpp:2]: (debug) ValueFlow bailout: no simplification of x within ?: expression\n", errout.str());
|
|
|
|
|
|
|
|
code = "void f(int x) {\n"
|
2014-01-10 16:13:39 +01:00
|
|
|
" int a =v x;\n"
|
2014-01-08 06:39:15 +01:00
|
|
|
" a = b ? x/2 : 20/x;\n"
|
|
|
|
" if (x == 123) {}\n"
|
|
|
|
"}";
|
|
|
|
ASSERT_EQUALS(true, testValueOfX(code, 2U, 123));
|
2014-01-07 19:46:13 +01:00
|
|
|
|
2014-01-10 16:13:39 +01:00
|
|
|
code = "void f(int x, int y) {\n"
|
|
|
|
" a = x;\n"
|
|
|
|
" if (y){}\n"
|
|
|
|
" if (x==123){}\n"
|
|
|
|
"}";
|
|
|
|
ASSERT_EQUALS(true, testValueOfX(code, 2U, 123));
|
|
|
|
|
2014-01-11 21:10:01 +01:00
|
|
|
// skip sizeof
|
|
|
|
code = "void f(int *x) {\n"
|
|
|
|
" sizeof(x[0]);\n"
|
|
|
|
" if (x==63){}\n"
|
|
|
|
"}";
|
|
|
|
ASSERT_EQUALS(false, testValueOfX(code, 2U, 63));
|
|
|
|
|
2014-01-06 11:27:56 +01:00
|
|
|
// bailout: if/else/etc
|
|
|
|
bailout("void f(int x) {\n"
|
|
|
|
" if (x != 123) { b = x; }\n"
|
|
|
|
" if (x == 123) {}\n"
|
|
|
|
"}");
|
|
|
|
ASSERT_EQUALS("[test.cpp:2]: (debug) ValueFlow bailout: variable x stopping on }\n", errout.str());
|
|
|
|
|
2014-01-06 07:44:58 +01:00
|
|
|
// bailout: assignment
|
|
|
|
bailout("void f(int x) {\n"
|
|
|
|
" x = y;\n"
|
|
|
|
" if (x == 123) {}\n"
|
|
|
|
"}");
|
|
|
|
ASSERT_EQUALS("[test.cpp:2]: (debug) ValueFlow bailout: assignment of x\n", errout.str());
|
|
|
|
|
|
|
|
// bailout: global variables
|
|
|
|
bailout("int x;\n"
|
|
|
|
"void f() {\n"
|
|
|
|
" int a = x;\n"
|
|
|
|
" if (x == 123) {}\n"
|
|
|
|
"}");
|
|
|
|
ASSERT_EQUALS("[test.cpp:4]: (debug) ValueFlow bailout: global variable x\n", errout.str());
|
2014-01-11 21:21:00 +01:00
|
|
|
|
|
|
|
// bailout: switch
|
2014-01-12 15:07:58 +01:00
|
|
|
// TODO : handle switch/goto more intelligently
|
2014-01-11 21:21:00 +01:00
|
|
|
bailout("void f(int x, int y) {\n"
|
|
|
|
" switch (y) {\n"
|
|
|
|
" case 1: a=x; break;\n"
|
|
|
|
" case 2: if (x==5) {} break;\n"
|
|
|
|
" };\n"
|
|
|
|
"}");
|
|
|
|
ASSERT_EQUALS("[test.cpp:3]: (debug) ValueFlow bailout: variable x stopping on break\n", errout.str());
|
2014-01-12 15:07:58 +01:00
|
|
|
|
|
|
|
bailout("void f(int x, int y) {\n"
|
|
|
|
" switch (y) {\n"
|
|
|
|
" case 1: a=x; return 1;\n"
|
|
|
|
" case 2: if (x==5) {} break;\n"
|
|
|
|
" };\n"
|
|
|
|
"}");
|
|
|
|
ASSERT_EQUALS("[test.cpp:3]: (debug) ValueFlow bailout: variable x stopping on return\n", errout.str());
|
2014-01-04 20:57:02 +01:00
|
|
|
}
|
2014-01-06 16:37:52 +01:00
|
|
|
|
2014-01-07 19:20:56 +01:00
|
|
|
void valueFlowForLoop() {
|
2014-01-10 16:51:58 +01:00
|
|
|
const char *code;
|
|
|
|
|
|
|
|
code = "void f() {\n"
|
|
|
|
" for (int x = 0; x < 10; x++)\n"
|
|
|
|
" a[x] = 0;\n"
|
|
|
|
"}";
|
|
|
|
ASSERT_EQUALS(true, testValueOfX(code, 3U, 0));
|
|
|
|
ASSERT_EQUALS(true, testValueOfX(code, 3U, 9));
|
|
|
|
ASSERT_EQUALS(false, testValueOfX(code, 3U, 10));
|
|
|
|
|
|
|
|
code = "void f() {\n"
|
2014-01-10 18:18:49 +01:00
|
|
|
" for (int x = 0; x < ((short)10); x++)\n"
|
2014-01-10 16:51:58 +01:00
|
|
|
" a[x] = 0;\n"
|
|
|
|
"}";
|
2014-01-07 19:20:56 +01:00
|
|
|
ASSERT_EQUALS(true, testValueOfX(code, 3U, 0));
|
|
|
|
ASSERT_EQUALS(true, testValueOfX(code, 3U, 9));
|
|
|
|
ASSERT_EQUALS(false, testValueOfX(code, 3U, 10));
|
|
|
|
}
|
|
|
|
|
2014-01-06 16:37:52 +01:00
|
|
|
void valueFlowSubFunction() {
|
|
|
|
const char *code;
|
|
|
|
|
|
|
|
code = "void f1(int x) { return x; }\n"
|
|
|
|
"void f2(int x) {\n"
|
|
|
|
" f1(123);\n"
|
|
|
|
"}";
|
|
|
|
ASSERT_EQUALS(true, testValueOfX(code, 1U, 123));
|
|
|
|
|
|
|
|
code = "void f1(int x) { return x; }\n"
|
|
|
|
"void f2(int x) {\n"
|
|
|
|
" f1(x);\n"
|
|
|
|
" if (x==0){}\n"
|
|
|
|
"}";
|
|
|
|
ASSERT_EQUALS(true, testValueOfX(code, 1U, 0));
|
2014-01-08 06:04:51 +01:00
|
|
|
|
|
|
|
code = "void f1(int x) {\n"
|
|
|
|
" if (x == 0) return;\n"
|
|
|
|
" int y = 1234 / x;\n"
|
|
|
|
"}\n"
|
|
|
|
"\n"
|
|
|
|
"void f2() {\n"
|
|
|
|
" f1(0);\n"
|
|
|
|
"}";
|
|
|
|
ASSERT_EQUALS(false, testValueOfX(code, 3U, 0));
|
2014-01-08 16:17:47 +01:00
|
|
|
|
|
|
|
code = "void f1(int x) { a=x; }\n"
|
|
|
|
"void f2(int y) { f1(y<123); }\n";
|
|
|
|
ASSERT_EQUALS(true, testValueOfX(code, 1U, 0));
|
|
|
|
ASSERT_EQUALS(true, testValueOfX(code, 1U, 1));
|
2014-01-11 07:52:25 +01:00
|
|
|
|
|
|
|
code = "void f1(int x) { a=(abc)x; }\n"
|
|
|
|
"void f2(int y) { f1(123); }\n";
|
|
|
|
ASSERT_EQUALS(true, testValueOfX(code, 1U, 123));
|
2014-01-06 16:37:52 +01:00
|
|
|
}
|
2014-01-04 20:57:02 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
REGISTER_TEST(TestValueFlow)
|