Skip to content

Commit 285893a

Browse files
committed
Conversion check: fix off-by-one error for float-to-unsigned
We use less-than in the comparison, so make the bound one larger than the largest representable value (as we do in the other cases of float-to-signed/unsigned conversion checks). Fixes: #8131
1 parent 69bb2b6 commit 285893a

File tree

3 files changed

+29
-1
lines changed

3 files changed

+29
-1
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,20 @@
1+
#include <stdint.h>
2+
3+
int main()
4+
{
5+
uint32_t u = 0xffffffffu;
6+
7+
// since the double type seems to be implemented with a IEEE 754 binary64
8+
// format in CBMC, which has 53 bits of mantissa, double has enough bits to
9+
// represent the exact value of u, use, e.g., http://weitz.de/ieee/ to check;
10+
// therefore, C17, section 6.3.1.4, paragraph 2 says that this is
11+
// defined behavior and d should store the exact value that u stores
12+
double d = u;
13+
14+
// defined behavior behavior as well, by C17 section 6.3.1.4, paragraph 1,
15+
// because the 'unsigned' type can represent the value; however,
16+
// cbmc --conversion-check used to complain
17+
u = (uint32_t)d;
18+
19+
return 0;
20+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
CORE
2+
float_conversion.c
3+
4+
^VERIFICATION SUCCESSFUL$
5+
^EXIT=0$
6+
^SIGNAL=0$
7+
--
8+
^warning: ignoring

src/ansi-c/goto_check_c.cpp

+1-1
Original file line numberDiff line numberDiff line change
@@ -796,7 +796,7 @@ void goto_check_ct::conversion_check(const exprt &expr, const guardt &guard)
796796
{
797797
// Note that the fractional part is truncated!
798798
ieee_floatt upper(to_floatbv_type(old_type));
799-
upper.from_integer(power(2, new_width) - 1);
799+
upper.from_integer(power(2, new_width));
800800
const binary_relation_exprt no_overflow_upper(
801801
op, ID_lt, upper.to_expr());
802802

0 commit comments

Comments
 (0)