Skip to content

Commit 2cc9ef7

Browse files
committed
Introduce bitst and bytest to avoid bit/byte mix-up
Use mp_integers with a unit to clarify when an mp_integer holds a number of bits or a number of bytes. This makes sure that type-inconsistent comparisons as fixed in #7411 result in compile-time errors.
1 parent 709a5d8 commit 2cc9ef7

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

49 files changed

+830
-560
lines changed

src/analyses/goto_rw.cpp

+9-9
Original file line numberDiff line numberDiff line change
@@ -151,7 +151,7 @@ void rw_range_sett::get_objects_byte_extract(
151151
auto object_size_bits_opt = pointer_offset_bits(be.op().type(), ns);
152152
const exprt simp_offset=simplify_expr(be.offset(), ns);
153153

154-
auto index = numeric_cast<mp_integer>(simp_offset);
154+
auto index = numeric_cast<bytest>(simp_offset);
155155
if(
156156
range_start.is_unknown() || !index.has_value() ||
157157
!object_size_bits_opt.has_value())
@@ -160,25 +160,25 @@ void rw_range_sett::get_objects_byte_extract(
160160
}
161161
else
162162
{
163-
*index *= be.get_bits_per_byte();
164-
if(*index >= *object_size_bits_opt)
163+
bitst index_bits = bytes_to_bits(*index, be.get_bits_per_byte());
164+
if(index_bits >= *object_size_bits_opt)
165165
return;
166166

167167
endianness_mapt map(
168168
be.op().type(),
169169
be.id()==ID_byte_extract_little_endian,
170170
ns);
171171
range_spect offset = range_start;
172-
if(*index > 0)
172+
if(index_bits > bitst{0})
173173
{
174174
offset += range_spect::to_range_spect(
175-
map.map_bit(numeric_cast_v<std::size_t>(*index)));
175+
bitst{map.map_bit(numeric_cast_v<std::size_t>(index_bits))});
176176
}
177177
else
178178
{
179179
// outside the bounds of immediate byte-extract operand, might still be in
180180
// bounds of a parent object
181-
offset += range_spect::to_range_spect(*index);
181+
offset += range_spect::to_range_spect(index_bits);
182182
}
183183
get_objects_rec(mode, be.op(), offset, size);
184184
}
@@ -198,7 +198,7 @@ void rw_range_sett::get_objects_shift(
198198
? range_spect::to_range_spect(*op_bits)
199199
: range_spect::unknown();
200200

201-
const auto dist = numeric_cast<mp_integer>(simp_distance);
201+
const auto dist = numeric_cast<bitst>(simp_distance);
202202
if(
203203
range_start.is_unknown() || size.is_unknown() || src_size.is_unknown() ||
204204
!dist.has_value())
@@ -313,7 +313,7 @@ void rw_range_sett::get_objects_index(
313313
get_objects_rec(
314314
mode,
315315
expr.array(),
316-
range_start + range_spect::to_range_spect(*index) * sub_size,
316+
range_start + range_spect::to_range_spect(bitst{*index}) * sub_size,
317317
size);
318318
}
319319
}
@@ -343,7 +343,7 @@ void rw_range_sett::get_objects_array(
343343
range_start.is_unknown() ? range_spect{0} : range_start;
344344
range_spect full_r_e =
345345
size.is_unknown()
346-
? sub_size * range_spect::to_range_spect(expr.operands().size())
346+
? sub_size * range_spect::to_range_spect(bitst{expr.operands().size()})
347347
: full_r_s + size;
348348

349349
for(const auto &op : expr.operands())

src/analyses/goto_rw.h

+7-6
Original file line numberDiff line numberDiff line change
@@ -76,15 +76,15 @@ class range_spect
7676
return *this == unknown();
7777
}
7878

79-
static range_spect to_range_spect(const mp_integer &size)
79+
static range_spect to_range_spect(const bitst &size)
8080
{
8181
// The size need not fit into the analysis platform's width of a signed long
8282
// (as an example, it could be an unsigned size_t max, or perhaps the source
8383
// platform has much wider types than the analysis platform.
84-
if(!size.is_long())
84+
if(!size.get().is_long())
8585
return range_spect::unknown();
8686

87-
mp_integer::llong_t ll = size.to_long();
87+
mp_integer::llong_t ll = size.get().to_long();
8888
if(
8989
ll > std::numeric_limits<range_spect::value_type>::max() ||
9090
ll < std::numeric_limits<range_spect::value_type>::min())
@@ -128,7 +128,7 @@ class range_spect
128128
{
129129
if(is_unknown() || other.is_unknown())
130130
return range_spect::unknown();
131-
return range_spect::to_range_spect(mp_integer{v} + mp_integer{other.v});
131+
return range_spect::to_range_spect(bitst{v} + bitst{other.v});
132132
}
133133

134134
range_spect &operator+=(const range_spect &other)
@@ -142,7 +142,7 @@ class range_spect
142142
{
143143
if(is_unknown() || other.is_unknown())
144144
return range_spect::unknown();
145-
return range_spect::to_range_spect(mp_integer{v} - mp_integer{other.v});
145+
return range_spect::to_range_spect(bitst{v} - bitst{other.v});
146146
}
147147

148148
range_spect &operator-=(const range_spect &other)
@@ -156,7 +156,8 @@ class range_spect
156156
{
157157
if(is_unknown() || other.is_unknown())
158158
return range_spect::unknown();
159-
return range_spect::to_range_spect(mp_integer{v} * mp_integer{other.v});
159+
return range_spect::to_range_spect(
160+
bitst{mp_integer{v} * mp_integer{other.v}});
160161
}
161162

162163
private:

src/ansi-c/ansi_c_internal_additions.cpp

+4-3
Original file line numberDiff line numberDiff line change
@@ -125,9 +125,10 @@ static std::string architecture_string(const std::string &value, const char *s)
125125
template <typename T>
126126
static std::string architecture_string(T value, const char *s)
127127
{
128-
return std::string("const " CPROVER_PREFIX "integer " CPROVER_PREFIX
129-
"architecture_") +
130-
std::string(s) + "=" + std::to_string(value) + ";\n";
128+
std::ostringstream os;
129+
os << "const " << CPROVER_PREFIX "integer " << CPROVER_PREFIX "architecture_"
130+
<< s << '=' << value << ";\n";
131+
return os.str();
131132
}
132133

133134
/// The maximum allocation size is determined by the number of bits that

src/ansi-c/c_typecheck_expr.cpp

+1-1
Original file line numberDiff line numberDiff line change
@@ -1055,7 +1055,7 @@ void c_typecheck_baset::typecheck_expr_alignof(exprt &expr)
10551055
}
10561056

10571057
// we only care about the type
1058-
mp_integer a=alignment(argument_type, *this);
1058+
bytest a = alignment(argument_type, *this);
10591059

10601060
exprt tmp=from_integer(a, size_type());
10611061
tmp.add_source_location()=expr.source_location();

src/ansi-c/expr2c.cpp

+3-2
Original file line numberDiff line numberDiff line change
@@ -3723,8 +3723,9 @@ std::string expr2ct::convert_with_precedence(
37233723
else if(src.id()==ID_bswap)
37243724
return convert_function(
37253725
src,
3726-
"__builtin_bswap" + integer2string(*pointer_offset_bits(
3727-
to_unary_expr(src).op().type(), ns)));
3726+
"__builtin_bswap" +
3727+
integer2string(
3728+
pointer_offset_bits(to_unary_expr(src).op().type(), ns)->get()));
37283729

37293730
else if(has_prefix(src.id_string(), "byte_extract"))
37303731
return convert_byte_extract(to_byte_extract_expr(src), precedence = 16);

0 commit comments

Comments
 (0)