AST: Refactoring new checkIncorrectLogic checking

This commit is contained in:
Daniel Marjamäki 2013-11-18 23:09:14 +01:00
parent 007e2d9dd5
commit 45ee3b0fba
1 changed files with 61 additions and 43 deletions

View File

@ -1304,6 +1304,34 @@ static bool analyzeLogicOperatorCondition(const Condition& c1, const Condition&
return checkRelation(relation, firstConstant, secondConstant); return checkRelation(relation, firstConstant, secondConstant);
} }
template<class T> static T getvalue(const int test, const T value1, const T value2)
{
// test:
// 1 => return value that is less than both value1 and value2
// 2 => return value1
// 3 => return value that is between value1 and value2
// 4 => return value2
// 5 => return value that is larger than both value1 and value2
switch (test) {
case 1:
return std::min(value1, value2) - (T)1;
break;
case 2:
return value1;
break;
case 3:
return (value1 + value2) / (T)2;
break;
case 4:
return value2;
break;
case 5:
return std::max(value1, value2) + (T)1;
break;
};
return 0;
}
void CheckOther::checkIncorrectLogicOperator() void CheckOther::checkIncorrectLogicOperator()
{ {
bool style = _settings->isEnabled("style"); bool style = _settings->isEnabled("style");
@ -1368,7 +1396,7 @@ void CheckOther::checkIncorrectLogicOperator()
const std::set<std::string> constStandardFunctions; const std::set<std::string> constStandardFunctions;
if (isSameExpression(comp1, comp2, constStandardFunctions)) if (isSameExpression(comp1, comp2, constStandardFunctions))
continue; // same expressions => only report that continue; // same expressions => only report that there are same expressions
if (!isSameExpression(expr1, expr2, constStandardFunctions)) if (!isSameExpression(expr1, expr2, constStandardFunctions))
continue; continue;
@ -1382,49 +1410,36 @@ void CheckOther::checkIncorrectLogicOperator()
// evaluate if expression is always true/false // evaluate if expression is always true/false
bool alwaysTrue = true, alwaysFalse = true; bool alwaysTrue = true, alwaysFalse = true;
bool firstTrue = true, secondTrue = true; bool firstTrue = true, secondTrue = true;
for (int offset = -3; offset <=3; ++offset) { for (int test = 1; test <= 5; ++test) {
for (int selvalue = 1; selvalue <= 2; selvalue++) { // test:
bool res1,res2; // 1 => testvalue is less than both value1 and value2
if (isfloat) { // 2 => testvalue is value1
const double d1 = MathLib::toDoubleNumber(value1); // 3 => testvalue is between value1 and value2
const double d2 = MathLib::toDoubleNumber(value2); // 4 => testvalue value2
double varvalue = (selvalue==1) ? d1 : d2; // 5 => testvalue is larger than both value1 and value2
switch (offset) { bool result1, result2;
case -3: if (isfloat) {
varvalue /= 2.0; const double d1 = MathLib::toDoubleNumber(value1);
break; const double d2 = MathLib::toDoubleNumber(value2);
case -2: const double testvalue = getvalue<double>(test, d1, d2);
varvalue *= 2.0; result1 = checkFloatRelation(op1, testvalue, d1);
break; result2 = checkFloatRelation(op2, testvalue, d2);
case -1: } else {
varvalue -= 1.0; const MathLib::bigint i1 = MathLib::toLongNumber(value1);
break; const MathLib::bigint i2 = MathLib::toLongNumber(value2);
case 1: const MathLib::bigint testvalue = getvalue<MathLib::bigint>(test, i1, i2);
varvalue += 1.0; result1 = checkIntRelation(op1, testvalue, i1);
break; result2 = checkIntRelation(op2, testvalue, i2);
case 2:
varvalue = (d1 + d2) / 2.0;
break;
default:
break;
};
res1 = checkFloatRelation(op1, varvalue, MathLib::toDoubleNumber(value1));
res2 = checkFloatRelation(op2, varvalue, MathLib::toDoubleNumber(value2));
} else {
const MathLib::bigint varvalue = MathLib::toLongNumber(selvalue==1 ? value1 : value2) + offset;
res1 = checkIntRelation(op1, varvalue, MathLib::toLongNumber(value1));
res2 = checkIntRelation(op2, varvalue, MathLib::toLongNumber(value2));
}
if (tok->str() == "&&") {
alwaysTrue &= (res1 && res2);
alwaysFalse &= !(res1 && res2);
} else {
alwaysTrue &= (res1 || res2);
alwaysFalse &= !(res1 || res2);
}
firstTrue &= !(!res1 && res2);
secondTrue &= !(res1 && !res2);
} }
if (tok->str() == "&&") {
alwaysTrue &= (result1 && result2);
alwaysFalse &= !(result1 && result2);
} else {
alwaysTrue &= (result1 || result2);
alwaysFalse &= !(result1 || result2);
}
firstTrue &= !(!result1 && result2);
secondTrue &= !(result1 && !result2);
} }
const std::string cond1str = (expr1->isName() ? expr1->str() : "EXPR") + " " + op1 + " " + value1; const std::string cond1str = (expr1->isName() ? expr1->str() : "EXPR") + " " + op1 + " " + value1;
@ -1437,6 +1452,9 @@ void CheckOther::checkIncorrectLogicOperator()
" is always " + (secondTrue ? "true" : "false") + "."; " is always " + (secondTrue ? "true" : "false") + ".";
redundantConditionError(tok, text); redundantConditionError(tok, text);
} else if (style && firstTrue) { } else if (style && firstTrue) {
//const std::string text = "The comparison " + cond1str + " is always " +
// (firstTrue ? "true" : "false") + " when " +
// cond2str + ".";
const std::string text = "If " + cond2str + ", the comparison " + cond1str + const std::string text = "If " + cond2str + ", the comparison " + cond1str +
" is always " + (firstTrue ? "true" : "false") + "."; " is always " + (firstTrue ? "true" : "false") + ".";
redundantConditionError(tok, text); redundantConditionError(tok, text);