1 /* Routines for restoring various data types from a file stream. This deals
2 with various data types like strings, integers, enums, etc.
4 Copyright (C) 2011-2024 Free Software Foundation, Inc.
5 Contributed by Diego Novillo <dnovillo@google.com>
7 This file is part of GCC.
9 GCC is free software; you can redistribute it and/or modify it under
10 the terms of the GNU General Public License as published by the Free
11 Software Foundation; either version 3, or (at your option) any later
14 GCC is distributed in the hope that it will be useful, but WITHOUT ANY
15 WARRANTY; without even the implied warranty of MERCHANTABILITY or
16 FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
19 You should have received a copy of the GNU General Public License
20 along with GCC; see the file COPYING3. If not see
21 <http://www.gnu.org/licenses/>. */
25 #include "coretypes.h"
30 #include "data-streamer.h"
31 #include "value-range.h"
32 #include "streamer-hooks.h"
34 /* Read a string from the string table in DATA_IN using input block
35 IB. Write the length to RLEN. */
38 string_for_index (class data_in
*data_in
, unsigned int loc
, unsigned int *rlen
)
49 /* Get the string stored at location LOC in DATA_IN->STRINGS. */
50 lto_input_block
str_tab (data_in
->strings
, loc
- 1, data_in
->strings_len
, NULL
);
51 len
= streamer_read_uhwi (&str_tab
);
54 if (str_tab
.p
+ len
> data_in
->strings_len
)
55 internal_error ("bytecode stream: string too long for the string table");
57 result
= (const char *)(data_in
->strings
+ str_tab
.p
);
63 /* Read a string from the string table in DATA_IN using input block
64 IB. Write the length to RLEN. */
67 streamer_read_indexed_string (class data_in
*data_in
,
68 class lto_input_block
*ib
, unsigned int *rlen
)
70 return string_for_index (data_in
, streamer_read_uhwi (ib
), rlen
);
74 /* Read a NULL terminated string from the string table in DATA_IN. */
77 streamer_read_string (class data_in
*data_in
, class lto_input_block
*ib
)
82 ptr
= streamer_read_indexed_string (data_in
, ib
, &len
);
85 if (ptr
[len
- 1] != '\0')
86 internal_error ("bytecode stream: found non-null terminated string");
92 /* Read a string from the string table in DATA_IN using the bitpack BP.
93 Write the length to RLEN. */
96 bp_unpack_indexed_string (class data_in
*data_in
,
97 struct bitpack_d
*bp
, unsigned int *rlen
)
99 return string_for_index (data_in
, bp_unpack_var_len_unsigned (bp
), rlen
);
103 /* Read a NULL terminated string from the string table in DATA_IN. */
106 bp_unpack_string (class data_in
*data_in
, struct bitpack_d
*bp
)
111 ptr
= bp_unpack_indexed_string (data_in
, bp
, &len
);
114 if (ptr
[len
- 1] != '\0')
115 internal_error ("bytecode stream: found non-null terminated string");
121 /* Read an unsigned HOST_WIDE_INT number from IB. */
123 unsigned HOST_WIDE_INT
124 streamer_read_uhwi (class lto_input_block
*ib
)
126 unsigned HOST_WIDE_INT result
;
128 unsigned HOST_WIDE_INT byte
;
129 unsigned int p
= ib
->p
;
130 unsigned int len
= ib
->len
;
132 const char *data
= ib
->data
;
134 if ((result
& 0x80) != 0)
141 result
|= (byte
& 0x7f) << shift
;
144 while ((byte
& 0x80) != 0);
147 /* We check for section overrun after the fact for performance reason. */
149 lto_section_overrun (ib
);
156 /* Read a HOST_WIDE_INT number from IB. */
159 streamer_read_hwi (class lto_input_block
*ib
)
161 HOST_WIDE_INT result
= 0;
163 unsigned HOST_WIDE_INT byte
;
167 byte
= streamer_read_uchar (ib
);
168 result
|= (byte
& 0x7f) << shift
;
170 if ((byte
& 0x80) == 0)
172 if ((shift
< HOST_BITS_PER_WIDE_INT
) && (byte
& 0x40))
173 result
|= - (HOST_WIDE_INT_1U
<< shift
);
180 /* Read a poly_uint64 from IB. */
183 streamer_read_poly_uint64 (class lto_input_block
*ib
)
185 using coeff_type
= poly_int_traits
<poly_uint64
>::coeff_type
;
186 return poly_int_read_common
<coeff_type
> (streamer_read_uhwi
, ib
);
189 /* Read a poly_int64 from IB. */
192 streamer_read_poly_int64 (class lto_input_block
*ib
)
194 using coeff_type
= poly_int_traits
<poly_int64
>::coeff_type
;
195 return poly_int_read_common
<coeff_type
> (streamer_read_hwi
, ib
);
198 /* Read gcov_type value from IB. */
201 streamer_read_gcov_count (class lto_input_block
*ib
)
203 gcov_type ret
= streamer_read_hwi (ib
);
207 /* Read REAL_VALUE_TYPE from IB. */
210 streamer_read_real_value (class lto_input_block
*ib
, REAL_VALUE_TYPE
*r
)
212 struct bitpack_d bp
= streamer_read_bitpack (ib
);
213 bp_unpack_real_value (&bp
, r
);
217 streamer_read_value_range (class lto_input_block
*ib
, data_in
*data_in
,
220 // Read the common fields to all vranges.
221 value_range_kind kind
= streamer_read_enum (ib
, value_range_kind
, VR_LAST
);
222 gcc_checking_assert (kind
!= VR_UNDEFINED
);
223 tree type
= stream_read_tree (ib
, data_in
);
225 // Initialize the value_range to the correct type.
228 if (is_a
<irange
> (vr
))
230 irange
&r
= as_a
<irange
> (vr
);
232 unsigned HOST_WIDE_INT num_pairs
= streamer_read_uhwi (ib
);
233 for (unsigned i
= 0; i
< num_pairs
; ++i
)
235 wide_int lb
= streamer_read_wide_int (ib
);
236 wide_int ub
= streamer_read_wide_int (ib
);
237 int_range
<2> tmp (type
, lb
, ub
);
240 wide_int value
= streamer_read_wide_int (ib
);
241 wide_int mask
= streamer_read_wide_int (ib
);
242 irange_bitmask
bm (value
, mask
);
243 r
.update_bitmask (bm
);
246 if (is_a
<frange
> (vr
))
248 frange
&r
= as_a
<frange
> (vr
);
250 // Stream in NAN bits.
251 struct bitpack_d bp
= streamer_read_bitpack (ib
);
252 bool pos_nan
= (bool) bp_unpack_value (&bp
, 1);
253 bool neg_nan
= (bool) bp_unpack_value (&bp
, 1);
254 nan_state
nan (pos_nan
, neg_nan
);
257 r
.set_nan (type
, nan
);
260 REAL_VALUE_TYPE lb
, ub
;
261 streamer_read_real_value (ib
, &lb
);
262 streamer_read_real_value (ib
, &ub
);
263 r
.set (type
, lb
, ub
, nan
);
267 if (is_a
<prange
> (vr
))
269 prange
&r
= as_a
<prange
> (vr
);
270 wide_int lb
= streamer_read_wide_int (ib
);
271 wide_int ub
= streamer_read_wide_int (ib
);
272 r
.set (type
, lb
, ub
);
273 wide_int value
= streamer_read_wide_int (ib
);
274 wide_int mask
= streamer_read_wide_int (ib
);
275 irange_bitmask
bm (value
, mask
);
276 r
.update_bitmask (bm
);
282 /* Read the physical representation of a wide_int val from
286 streamer_read_wide_int (class lto_input_block
*ib
)
288 HOST_WIDE_INT abuf
[WIDE_INT_MAX_INL_ELTS
], *a
= abuf
;
290 int prec
= streamer_read_uhwi (ib
);
291 int len
= streamer_read_uhwi (ib
);
292 if (UNLIKELY (len
> WIDE_INT_MAX_INL_ELTS
))
293 a
= XALLOCAVEC (HOST_WIDE_INT
, len
);
294 for (i
= 0; i
< len
; i
++)
295 a
[i
] = streamer_read_hwi (ib
);
296 return wide_int::from_array (a
, len
, prec
);
299 /* Read the physical representation of a widest_int val from
303 streamer_read_widest_int (class lto_input_block
*ib
)
305 HOST_WIDE_INT abuf
[WIDE_INT_MAX_INL_ELTS
], *a
= abuf
;
307 int prec ATTRIBUTE_UNUSED
= streamer_read_uhwi (ib
);
308 int len
= streamer_read_uhwi (ib
);
309 if (UNLIKELY (len
> WIDE_INT_MAX_INL_ELTS
))
310 a
= XALLOCAVEC (HOST_WIDE_INT
, len
);
311 for (i
= 0; i
< len
; i
++)
312 a
[i
] = streamer_read_hwi (ib
);
313 return widest_int::from_array (a
, len
);