In this blog post I hope to convince you that the "old school" conversion checks are bad. These are used in many projects today.
Here is unsafe code that has potential loss of sign:
(1)
signed int svar;
void foo(unsigned int uvar)
{
if (svar > uvar)
{
/* do stuff */
}
}
Assuming that svar can have arbitrary values, there can be loss of sign in the comparison. When a signed int value and a unsigned int value is compared in C/C++ code, the signed value is casted to an unsigned value. It means that a negative value is converted to a large positive value before the comparison. Then you get the wrong comparison result.
Logically, this code is the same as (1):
(2)
int svar;
void foo(unsigned int uvar)
{
if ((unsigned int)svar > uvar)
{
/* do stuff */
}
}
Please note that this code is just as unsafe as the first code.
So if you want to compare a signed and unsigned variable that has arbitrary values, how do you do this safely? If you can cast the operands into a larger signed data type then that would make the code safe:
(3)
int svar;
void foo(unsigned int uvar)
{
if ((long long)svar > (long long)uvar)
{
/* do stuff */
}
}
The other way to write safe code is to add an extra comparison:
(4)
int svar;
void foo(unsigned int uvar)
{
if (svar > 0 && svar > uvar)
{
/* do stuff */
}
}
The advantage with (4) against (3) is that it avoids casts. These casts could easily cause bugs or conceal bugs in the future.
Old school tools
Let us look at the old school tools that will warn about loss of sign and loss of precision.The old school tools will just think that if there is no cast then the code is unsafe and if there is a cast then the code is safe. The old school tools warn about (1) and (4). They do not warn about (2) and (3). This means:
- Warnings are written about both safe and unsafe code.
- You are not safe at all just because the tool is silent.
When there is a warning the developer must suppress it with a cast. Such casts are unsafe, as they can introduce new problems. For instance I often see size_t casted to int etc to fix such warning. Even if that cast to int is safe today, the cast can conceal bugs or cause bugs in the future.
If the warning is really correct and the developer wants to fix the bug, then it is important that the developer choose the right cast. I think it's likely that mistakes are sometimes made, the result will probably not be (3) but instead some variation of (2). Unfortunately the old school tools do not enforce that the problem is fixed properly.
Proper checking
People should use proper static analysis instead to detect these problems.A proper check would use valueflow analysis to determine what the possible values of the operands are. If the signed operand can be negative then there can be loss of sign.
I would expect that proper analysis would only show ~1 % as many warnings as the old school tools. Because old school tools are wrong so often. You would get a lot fewer casts in the code and this would make it much safer. You could just use casts when it was really needed.
There is for instance a proper check (alpha.core.Conversion) in the Clang analyzer that warns when a negative value is implicitly casted to unsigned and when a large value is implictly truncated.
There is no proper check in Cppcheck right now for this. It might be added in the future.