Refactor handling of char constants
This commit is contained in:
parent
c5b21d12cf
commit
0022ce8075
|
@ -363,6 +363,25 @@ MathLib::bigint MathLib::toLongNumber(const std::string & str)
|
||||||
return static_cast<bigint>(doubleval);
|
return static_cast<bigint>(doubleval);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (str[0] == '\'' && str.size() >= 3U && str[str.size()-1U] == '\'') {
|
||||||
|
char c;
|
||||||
|
if (str.size() == 3U &&
|
||||||
|
str[0] == '\'' &&
|
||||||
|
str[2] == '\'')
|
||||||
|
c = str[1];
|
||||||
|
else if (str == "\'\\0\'")
|
||||||
|
c = '\0';
|
||||||
|
else if (str == "\'\\n\'")
|
||||||
|
c = '\n';
|
||||||
|
else if (str == "\'\\r\'")
|
||||||
|
c = '\r';
|
||||||
|
else if (str == "\'\\t\'")
|
||||||
|
c = '\t';
|
||||||
|
else
|
||||||
|
throw InternalError(0, "MathLib::toLongNumber: Unhandled char constant " + str);
|
||||||
|
return c & 0xff;
|
||||||
|
}
|
||||||
|
|
||||||
bigint ret = 0;
|
bigint ret = 0;
|
||||||
std::istringstream istr(str);
|
std::istringstream istr(str);
|
||||||
istr >> ret;
|
istr >> ret;
|
||||||
|
|
|
@ -1023,7 +1023,8 @@ bool TemplateSimplifier::simplifyCalculations(Token *_tokens)
|
||||||
}
|
}
|
||||||
|
|
||||||
if (Token::Match(tok->previous(), "(|&&|%oror% %char% %comp% %num% &&|%oror%|)")) {
|
if (Token::Match(tok->previous(), "(|&&|%oror% %char% %comp% %num% &&|%oror%|)")) {
|
||||||
tok->str(MathLib::toString(tok->str()[1] & 0xff));
|
int c = MathLib::toLongNumber(tok->str());
|
||||||
|
tok->str(MathLib::toString(c));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (tok->isNumber()) {
|
if (tok->isNumber()) {
|
||||||
|
|
|
@ -594,29 +594,10 @@ static void setTokenValue(Token* tok, const ValueFlow::Value &value)
|
||||||
static void valueFlowNumber(TokenList *tokenlist)
|
static void valueFlowNumber(TokenList *tokenlist)
|
||||||
{
|
{
|
||||||
for (Token *tok = tokenlist->front(); tok; tok = tok->next()) {
|
for (Token *tok = tokenlist->front(); tok; tok = tok->next()) {
|
||||||
if (tok->isNumber() && MathLib::isInt(tok->str())) {
|
if ((tok->isNumber() && MathLib::isInt(tok->str())) || (tok->tokType() == Token::eChar)) {
|
||||||
ValueFlow::Value value(MathLib::toLongNumber(tok->str()));
|
ValueFlow::Value value(MathLib::toLongNumber(tok->str()));
|
||||||
value.setKnown();
|
value.setKnown();
|
||||||
setTokenValue(tok, value);
|
setTokenValue(tok, value);
|
||||||
} else if (tok->tokType() == Token::eChar) {
|
|
||||||
char c;
|
|
||||||
if (tok->str() == "\'\\n\'")
|
|
||||||
c = '\n';
|
|
||||||
else if (tok->str() == "\'\\r\'")
|
|
||||||
c = '\r';
|
|
||||||
else if (tok->str() == "\'\\t\'")
|
|
||||||
c = '\t';
|
|
||||||
else if (tok->str().size() == 3U &&
|
|
||||||
tok->str()[0] == '\'' &&
|
|
||||||
(tok->str()[1] & 0x80) == 0 &&
|
|
||||||
tok->str()[2] == '\'')
|
|
||||||
c = tok->str()[1];
|
|
||||||
else
|
|
||||||
continue;
|
|
||||||
|
|
||||||
ValueFlow::Value value(c);
|
|
||||||
value.setKnown();
|
|
||||||
setTokenValue(tok, value);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -104,7 +104,7 @@ private:
|
||||||
check("void f()\n"
|
check("void f()\n"
|
||||||
"{\n"
|
"{\n"
|
||||||
" struct hostent *hp;\n"
|
" struct hostent *hp;\n"
|
||||||
" if(!hp = gethostbyname('127.0.0.1')) {\n"
|
" if(!hp = gethostbyname(\"127.0.0.1\")) {\n"
|
||||||
" exit(1);\n"
|
" exit(1);\n"
|
||||||
" }\n"
|
" }\n"
|
||||||
"}");
|
"}");
|
||||||
|
@ -115,7 +115,7 @@ private:
|
||||||
check("void f()\n"
|
check("void f()\n"
|
||||||
"{\n"
|
"{\n"
|
||||||
" long addr;\n"
|
" long addr;\n"
|
||||||
" addr = inet_addr('127.0.0.1');\n"
|
" addr = inet_addr(\"127.0.0.1\");\n"
|
||||||
" if(!hp = gethostbyaddr((char *) &addr, sizeof(addr), AF_INET)) {\n"
|
" if(!hp = gethostbyaddr((char *) &addr, sizeof(addr), AF_INET)) {\n"
|
||||||
" exit(1);\n"
|
" exit(1);\n"
|
||||||
" }\n"
|
" }\n"
|
||||||
|
@ -164,7 +164,7 @@ private:
|
||||||
|
|
||||||
check("const char f()\n"
|
check("const char f()\n"
|
||||||
"{\n"
|
"{\n"
|
||||||
" const char var[6] = 'index';\n"
|
" const char var[6] = \"index\";\n"
|
||||||
" const char i = index(var, 0);\n"
|
" const char i = index(var, 0);\n"
|
||||||
" return i;\n"
|
" return i;\n"
|
||||||
"}");
|
"}");
|
||||||
|
@ -188,7 +188,7 @@ private:
|
||||||
|
|
||||||
check("void f()\n"
|
check("void f()\n"
|
||||||
"{\n"
|
"{\n"
|
||||||
" const char var[7] = 'rindex';\n"
|
" const char var[7] = \"rindex\";\n"
|
||||||
" print(rindex(var, 0));\n"
|
" print(rindex(var, 0));\n"
|
||||||
"}");
|
"}");
|
||||||
ASSERT_EQUALS("[test.cpp:4]: (style) Obsolete function 'rindex' called. It is recommended to use the function 'strrchr' instead.\n", errout.str());
|
ASSERT_EQUALS("[test.cpp:4]: (style) Obsolete function 'rindex' called. It is recommended to use the function 'strrchr' instead.\n", errout.str());
|
||||||
|
|
|
@ -307,10 +307,10 @@ private:
|
||||||
|
|
||||||
// Ticket #5734
|
// Ticket #5734
|
||||||
check("int foo(char c) {\n"
|
check("int foo(char c) {\n"
|
||||||
"return c == '42';}", "test.cpp");
|
"return c == '4';}", "test.cpp");
|
||||||
ASSERT_EQUALS("", errout.str());
|
ASSERT_EQUALS("", errout.str());
|
||||||
check("int foo(char c) {\n"
|
check("int foo(char c) {\n"
|
||||||
"return c == '42';}", "test.c");
|
"return c == '4';}", "test.c");
|
||||||
ASSERT_EQUALS("", errout.str());
|
ASSERT_EQUALS("", errout.str());
|
||||||
check("int foo(char c) {\n"
|
check("int foo(char c) {\n"
|
||||||
"return c == \"42\"[0];}", "test.cpp");
|
"return c == \"42\"[0];}", "test.cpp");
|
||||||
|
|
Loading…
Reference in New Issue