26#ifndef ETL_BIT_STREAM_INCLUDED
27#define ETL_BIT_STREAM_INCLUDED
59 typedef const unsigned char* const_iterator;
75 : pdata(reinterpret_cast<unsigned char*>(begin_))
76 , length_chars(
etl::distance(reinterpret_cast<unsigned char*>(begin_), reinterpret_cast<unsigned char*>(end_)))
85 : pdata(reinterpret_cast<unsigned char*>(begin_))
86 , length_chars(length_)
96 pdata =
reinterpret_cast<unsigned char*
>(begin_);
97 length_chars = length_;
106 set_stream(begin_, etl::distance(
reinterpret_cast<unsigned char*
>(begin_),
reinterpret_cast<unsigned char*
>(end_)));
114 bits_available_in_char = CHAR_BIT;
116 bits_available = CHAR_BIT * length_chars;
124 return (bits_available == 0U);
132 bool success =
false;
134 if (pdata != ETL_NULLPTR)
136 if (bits_available > 0)
138 unsigned char chunk = value ? 1 : 0;
139 put_integral(uint32_t(chunk), 1);
150 template <
typename T>
152 put(T value, uint_least8_t nbits = CHAR_BIT *
sizeof(T))
154 return put_integral(
static_cast<uint32_t
>(value), nbits);
157#if ETL_USING_64BIT_TYPES
161 bool put(int64_t value, uint_least8_t nbits = CHAR_BIT *
sizeof(int64_t))
163 return put_integral(uint64_t(value), nbits);
169 bool put(uint64_t value, uint_least8_t nbits = CHAR_BIT *
sizeof(uint64_t))
171 return put_integral(value, nbits);
178 template <
typename T>
184 unsigned char data[
sizeof(T)];
185 to_bytes(value, data);
187 for (
size_t i = 0UL; i <
sizeof(T); ++i)
189 if (!put_integral(uint32_t(data[i]), CHAR_BIT))
203 bool success =
false;
205 if (pdata != ETL_NULLPTR)
208 if (bits_available > 0U)
221 template <
typename T>
223 get(T& value, uint_least8_t nbits = CHAR_BIT *
sizeof(T))
225 bool success =
false;
226 uint_least8_t
bits = nbits;
228 if (pdata != ETL_NULLPTR)
231 if (bits_available >= nbits)
238 unsigned char mask_width =
static_cast<unsigned char>(etl::min(nbits, bits_available_in_char));
241 chunk_t chunk = get_chunk(mask_width);
244 value |=
static_cast<T
>(chunk << nbits);
252 if (etl::is_signed<T>::value && (
bits != (CHAR_BIT *
sizeof(T))))
264 template <
typename T>
268 bool success =
false;
270 if (pdata != ETL_NULLPTR)
272 uint_least8_t nbits = CHAR_BIT *
sizeof(T);
275 if (bits_available >= nbits)
280 for (
size_t i = 0UL; i <
sizeof(T); ++i)
282 get(data.raw[i], CHAR_BIT);
285 from_bytes(
reinterpret_cast<const unsigned char*
>(data.raw), value);
299 size_t s = char_index;
302 if (bits_available_in_char != CHAR_BIT)
315 return (length_chars * CHAR_BIT) - bits_available;
329 const_iterator
end()
const
331 return pdata +
size();
339 bool put_integral(uint32_t value, uint_least8_t nbits)
341 bool success =
false;
343 if (pdata != ETL_NULLPTR)
346 if (bits_available >= nbits)
351 unsigned char mask_width =
static_cast<unsigned char>(etl::min(nbits, bits_available_in_char));
353 uint32_t mask = ((1U << mask_width) - 1U) << nbits;
358 uint32_t chunk = ((value & mask) >> nbits) << (bits_available_in_char - mask_width);
360 put_chunk(
static_cast<unsigned char>(chunk), mask_width);
370#if ETL_USING_64BIT_TYPES
374 bool put_integral(uint64_t value, uint_least8_t nbits)
376 bool success =
false;
378 if (pdata != ETL_NULLPTR)
381 if (bits_available >= nbits)
386 unsigned char mask_width =
static_cast<unsigned char>(etl::min(nbits, bits_available_in_char));
388 uint64_t mask = ((uint64_t(1U) << mask_width) - 1U) << nbits;
392 uint64_t chunk = ((value & mask) >> nbits) << (bits_available_in_char - mask_width);
394 put_chunk(
static_cast<unsigned char>(chunk), mask_width);
408 void put_chunk(
unsigned char chunk,
unsigned char nbits)
411 if (bits_available_in_char == 8U)
413 pdata[char_index] = 0U;
416 pdata[char_index] |= chunk;
423 unsigned char get_chunk(
unsigned char nbits)
425 unsigned char value = pdata[char_index];
427 value >>= (bits_available_in_char - nbits);
431 if (nbits == CHAR_BIT)
433 mask = etl::integral_limits<unsigned char>::max;
437 mask = (1U << nbits) - 1;
452 bool result = (pdata[char_index] & (1U << (bits_available_in_char - 1U))) != 0U;
462 template <
typename T>
463 void from_bytes(
const unsigned char* data, T& value)
465 etl::uninitialized_buffer_of<T, 1U> temp;
468 if (etl::endianness::value() == etl::endian::little)
470 etl::reverse_copy(data, data +
sizeof(T), temp.raw);
474 etl::copy(data, data +
sizeof(T), temp.raw);
477 value = *
reinterpret_cast<T*
>(temp.raw);
483 template <
typename T>
484 void to_bytes(T value,
unsigned char* data)
486 unsigned char* pf =
reinterpret_cast<unsigned char*
>(&value);
489 if (etl::endianness::value() == etl::endian::little)
491 etl::reverse_copy(pf, pf +
sizeof(T), data);
495 etl::copy(pf, pf +
sizeof(T), data);
503 void step(
unsigned char nbits)
505 bits_available_in_char -= nbits;
507 if (bits_available_in_char == 0)
510 bits_available_in_char = 8;
513 bits_available -= nbits;
516 unsigned char *pdata;
518 unsigned char bits_available_in_char;
520 size_t bits_available;
530 typedef char value_type;
531 typedef value_type* iterator;
532 typedef const value_type* const_iterator;
534 typedef etl::delegate<void(callback_parameter_type)> callback_type;
539 template <
size_t Length>
541 : pdata(span_.
begin())
543 , stream_endianness(stream_endianness_)
544 , callback(callback_)
552 template <
size_t Length>
554 : pdata(reinterpret_cast<char*>(span_.
begin()))
556 , stream_endianness(stream_endianness_)
557 , callback(callback_)
566 : pdata(reinterpret_cast<char*>(begin_))
567 , length_chars(
etl::distance(reinterpret_cast<unsigned char*>(begin_), reinterpret_cast<unsigned char*>(end_)))
568 , stream_endianness(stream_endianness_)
569 , callback(callback_)
578 : pdata(reinterpret_cast<char*>(begin_))
579 , length_chars(length_chars_)
580 , stream_endianness(stream_endianness_)
581 , callback(callback_)
591 bits_available_in_char = CHAR_BIT;
609 return length_chars * CHAR_BIT;
633 unsigned char chunk = value ? 1 : 0;
634 write_data<unsigned char>(chunk, 1);
655 template <
typename T>
659 typedef typename etl::unsigned_type<T>::type unsigned_t;
661 write_data<unsigned_t>(
static_cast<unsigned_t
>(value), nbits);
667 template <
typename T>
669 write(T value, uint_least8_t nbits = CHAR_BIT *
sizeof(T))
692 while (nbits > bits_available_in_char)
694 step(bits_available_in_char);
695 nbits -= bits_available_in_char;
700 step(
static_cast<unsigned char>(nbits));
712 size_t s = char_index;
715 if (bits_available_in_char != CHAR_BIT)
735 template <
size_t Nbits>
738 return bits_available / Nbits;
745 template <
typename T>
757 return bits_available / nbits;
765 return bits_available;
803 const_iterator
end()
const
853 if (callback.is_valid())
855 if (bits_available_in_char != 0U)
870 callback = callback_;
887 template <
typename T>
888 void write_data(T value, uint_least8_t nbits)
891 nbits = (nbits > (CHAR_BIT *
sizeof(T))) ? (CHAR_BIT *
sizeof(T)) : nbits;
893 if (stream_endianness == etl::endian::little)
896 value = value >> ((CHAR_BIT *
sizeof(T)) - nbits);
902 unsigned char mask_width =
static_cast<unsigned char>(etl::min(nbits, bits_available_in_char));
904 T mask = ((T(1U) << mask_width) - 1U) << nbits;
908 T chunk = ((value & mask) >> nbits) << (bits_available_in_char - mask_width);
910 write_chunk(
static_cast<char>(chunk), mask_width);
913 if (callback.is_valid())
922 void write_chunk(
char chunk,
unsigned char nbits)
925 if (bits_available_in_char == CHAR_BIT)
927 pdata[char_index] = 0U;
930 pdata[char_index] |= chunk;
938 void flush_full_bytes()
943 callback(callback_parameter_type(pdata, pdata + char_index));
945 bits_available = CHAR_BIT * length_chars;
947 if (bits_available_in_char != 0U)
950 pdata[0] = pdata[char_index];
951 bits_available -= (CHAR_BIT - bits_available_in_char);
962 void step(
unsigned char nbits)
964 bits_available_in_char -= nbits;
966 if (bits_available_in_char == 0)
969 bits_available_in_char = CHAR_BIT;
972 bits_available -= nbits;
976 const size_t length_chars;
977 const etl::endian stream_endianness;
978 unsigned char bits_available_in_char;
980 size_t bits_available;
981 callback_type callback;
999 return stream.
write(value);
1007 template <
typename T>
1019 template <
typename T>
1023 return stream.
write(value, nbits);
1033 typedef char value_type;
1034 typedef const char* const_iterator;
1039 template <
size_t Length>
1041 : pdata(span_.
begin())
1043 , stream_endianness(stream_endianness_)
1051 template <
size_t Length>
1053 : pdata(reinterpret_cast<const char*>(span_.
begin()))
1055 , stream_endianness(stream_endianness_)
1063 template <
size_t Length>
1065 : pdata(span_.
begin())
1067 , stream_endianness(stream_endianness_)
1075 template <
size_t Length>
1077 : pdata(reinterpret_cast<const char*>(span_.
begin()))
1079 , stream_endianness(stream_endianness_)
1088 : pdata(reinterpret_cast<const char*>(begin_))
1089 , length_chars(
etl::distance(reinterpret_cast<const char*>(begin_), reinterpret_cast<const char*>(end_)))
1090 , stream_endianness(stream_endianness_)
1099 : pdata(reinterpret_cast<const char*>(begin_))
1100 , length_chars(length_)
1101 , stream_endianness(stream_endianness_)
1111 bits_available_in_char = CHAR_BIT;
1113 bits_available = CHAR_BIT * length_chars;
1119 template <
typename T>
1129 template <
typename T>
1135 if (bits_available > 0U)
1146 template <
typename T>
1150 typedef typename etl::unsigned_type<T>::type unsigned_t;
1152 T value = read_value<unsigned_t>(nbits, etl::is_signed<T>::value);
1154 return static_cast<T
>(value);
1160 template <
typename T>
1162 read(uint_least8_t nbits = CHAR_BIT *
sizeof(T))
1167 if (bits_available >= nbits)
1180 return length_chars;
1188 return length_chars * CHAR_BIT;
1238 bool success = (nbits <= bits_available);
1242 while (nbits > bits_available_in_char)
1244 nbits -= bits_available_in_char;
1245 step(bits_available_in_char);
1250 step(
static_cast<unsigned char>(nbits));
1263 template <
typename T>
1264 T read_value(uint_least8_t nbits,
bool is_signed)
1267 nbits = (nbits > (CHAR_BIT *
sizeof(T))) ? (CHAR_BIT *
sizeof(T)) : nbits;
1270 uint_least8_t bits = nbits;
1275 unsigned char mask_width =
static_cast<unsigned char>(etl::min(nbits, bits_available_in_char));
1277 T chunk = get_chunk(mask_width);
1279 nbits -= mask_width;
1280 value |=
static_cast<T
>(chunk << nbits);
1283 if (stream_endianness == etl::endian::little)
1285 value = value << ((CHAR_BIT *
sizeof(T)) - bits);
1289 if (is_signed && (bits != (CHAR_BIT *
sizeof(T))))
1300 unsigned char get_chunk(
unsigned char nbits)
1302 unsigned char value = pdata[char_index];
1303 value >>= (bits_available_in_char - nbits);
1307 if (nbits == CHAR_BIT)
1309 mask = etl::integral_limits<unsigned char>::max;
1313 mask = (1U << nbits) - 1;
1328 bool result = (pdata[char_index] & (1U << (bits_available_in_char - 1U))) != 0U;
1339 void step(
unsigned char nbits)
1341 bits_available_in_char -= nbits;
1343 if (bits_available_in_char == 0)
1346 bits_available_in_char = 8;
1349 bits_available -= nbits;
1353 size_t length_chars;
1354 const etl::endian stream_endianness;
1355 unsigned char bits_available_in_char;
1357 size_t bits_available;
1363 template <
typename T>
1369 template <
typename T>
1378 template <
typename T>
1381 return stream.
read<T>();
1384 template <
typename T>
1387 return stream.
read<T>(nbits);
1405 return stream.
read<
bool>();
Reads bit streams.
Definition bit_stream.h:1030
etl::enable_if< etl::is_same< bool, T >::value, etl::optional< bool > >::type read()
For bool types.
Definition bit_stream.h:1131
const_iterator end() const
Returns end of the stream.
Definition bit_stream.h:1210
bit_stream_reader(const etl::span< const unsigned char, Length > &span_, etl::endian stream_endianness_)
Construct from span.
Definition bit_stream.h:1076
const_iterator cend() const
Returns end of the stream.
Definition bit_stream.h:1218
bit_stream_reader(const etl::span< char, Length > &span_, etl::endian stream_endianness_)
Construct from span.
Definition bit_stream.h:1040
size_t size_bits() const
Returns the number of bits in the stream buffer.
Definition bit_stream.h:1186
etl::enable_if< etl::is_integral< T >::value &&!etl::is_same< bool, T >::value, T >::type read_unchecked(uint_least8_t nbits=CHAR_BIT *sizeof(T))
For integral types.
Definition bit_stream.h:1148
bit_stream_reader(const void *begin_, size_t length_, etl::endian stream_endianness_)
Construct from begin and length.
Definition bit_stream.h:1098
bool skip(size_t nbits)
Definition bit_stream.h:1236
bit_stream_reader(const etl::span< const char, Length > &span_, etl::endian stream_endianness_)
Construct from span.
Definition bit_stream.h:1064
bit_stream_reader(const void *begin_, const void *end_, etl::endian stream_endianness_)
Construct from range.
Definition bit_stream.h:1087
bit_stream_reader(const etl::span< unsigned char, Length > &span_, etl::endian stream_endianness_)
Construct from span.
Definition bit_stream.h:1052
etl::enable_if< etl::is_integral< T >::value &&!etl::is_same< bool, T >::value, etl::optional< T > >::type read(uint_least8_t nbits=CHAR_BIT *sizeof(T))
For integral types.
Definition bit_stream.h:1162
void restart()
Sets the indexes back to the beginning of the stream.
Definition bit_stream.h:1109
const_iterator cbegin() const
Returns start of the stream.
Definition bit_stream.h:1202
etl::enable_if< etl::is_same< bool, T >::value, bool >::type read_unchecked()
For bool types.
Definition bit_stream.h:1121
etl::span< const char > data() const
Returns a span of whole the stream.
Definition bit_stream.h:1226
const_iterator begin() const
Returns start of the stream.
Definition bit_stream.h:1194
size_t size_bytes() const
Returns the number of bytes in the stream buffer.
Definition bit_stream.h:1178
Writes bits streams.
Definition bit_stream.h:527
bit_stream_writer(void *begin_, void *end_, etl::endian stream_endianness_, callback_type callback_=callback_type())
Construct from range.
Definition bit_stream.h:565
etl::enable_if< etl::is_integral< T >::value, bool >::type write(T value, uint_least8_t nbits=CHAR_BIT *sizeof(T))
For integral types.
Definition bit_stream.h:669
size_t size_bytes() const
Returns the number of bytes used in the stream.
Definition bit_stream.h:710
etl::enable_if< etl::is_integral< T >::value, void >::type write_unchecked(T value, uint_least8_t nbits=CHAR_BIT *sizeof(T))
For integral types.
Definition bit_stream.h:657
etl::span< const char > used_data() const
Returns a span of the used portion of the stream.
Definition bit_stream.h:827
bool write(bool value)
Writes a boolean to the stream.
Definition bit_stream.h:640
size_t size_bits() const
Returns the number of bits used in the stream.
Definition bit_stream.h:726
size_t available_bits() const
The number of bits left in the stream.
Definition bit_stream.h:763
const_iterator cend() const
Returns end of the stream.
Definition bit_stream.h:811
callback_type get_callback() const
Gets the function to call after every write.
Definition bit_stream.h:876
bool empty() const
Returns true if the bitsteam indexes have been reset.
Definition bit_stream.h:615
bit_stream_writer(void *begin_, size_t length_chars_, etl::endian stream_endianness_, callback_type callback_=callback_type())
Construct from begin and length.
Definition bit_stream.h:577
bit_stream_writer(const etl::span< unsigned char, Length > &span_, etl::endian stream_endianness_, callback_type callback_=callback_type())
Construct from span.
Definition bit_stream.h:553
bit_stream_writer(const etl::span< char, Length > &span_, etl::endian stream_endianness_, callback_type callback_=callback_type())
Construct from span.
Definition bit_stream.h:540
iterator end()
Returns end of the stream.
Definition bit_stream.h:795
void set_callback(callback_type callback_)
Sets the function to call after every write.
Definition bit_stream.h:868
size_t available() const
Definition bit_stream.h:736
etl::span< char > data()
Returns a span of whole the stream.
Definition bit_stream.h:835
bool full() const
Returns true if the bitsteam indexes have reached the end.
Definition bit_stream.h:623
bool skip(size_t nbits)
Definition bit_stream.h:686
size_t capacity_bits() const
Returns the maximum capacity in bits.
Definition bit_stream.h:607
etl::span< char > used_data()
Returns a span of the used portion of the stream.
Definition bit_stream.h:819
size_t capacity_bytes() const
Returns the maximum capacity in bytes.
Definition bit_stream.h:599
etl::span< const char > data() const
Returns a span of whole the stream.
Definition bit_stream.h:843
void restart()
Sets the indexes back to the beginning of the stream.
Definition bit_stream.h:589
void write_unchecked(bool value)
Writes a boolean to the stream.
Definition bit_stream.h:631
void flush()
Flush the last byte, if partially filled, to the callback, if valid.
Definition bit_stream.h:851
const_iterator cbegin() const
Returns start of the stream.
Definition bit_stream.h:787
size_t available(size_t nbits) const
Definition bit_stream.h:755
const_iterator end() const
Returns end of the stream.
Definition bit_stream.h:803
const_iterator begin() const
Returns start of the stream.
Definition bit_stream.h:779
iterator begin()
Returns start of the stream.
Definition bit_stream.h:771
etl::enable_if< etl::is_integral< T >::value, bool >::type get(T &value, uint_least8_t nbits=CHAR_BIT *sizeof(T))
For integral types.
Definition bit_stream.h:223
etl::enable_if< etl::is_integral< T >::value, bool >::type put(T value, uint_least8_t nbits=CHAR_BIT *sizeof(T))
For integral types.
Definition bit_stream.h:152
bool put(int64_t value, uint_least8_t nbits=CHAR_BIT *sizeof(int64_t))
For 64bit integral types.
Definition bit_stream.h:161
etl::enable_if< etl::is_floating_point< T >::value, bool >::type put(T value)
For floating point types.
Definition bit_stream.h:180
void restart()
Sets the indexes back to the beginning of the stream.
Definition bit_stream.h:112
size_t size() const
Returns the number of bytes used in the stream.
Definition bit_stream.h:297
size_t bits() const
Returns the number of bits used in the stream.
Definition bit_stream.h:313
bit_stream(void *begin_, void *end_)
Construct from range.
Definition bit_stream.h:74
bool put(uint64_t value, uint_least8_t nbits=CHAR_BIT *sizeof(uint64_t))
For 64bit integral types.
Definition bit_stream.h:169
bit_stream(void *begin_, size_t length_)
Construct from begin and length.
Definition bit_stream.h:84
etl::enable_if< etl::is_floating_point< T >::value, bool >::type get(T &value)
For floating point types.
Definition bit_stream.h:266
bool at_end() const
Returns true if the bitsteam indexes have reached the end.
Definition bit_stream.h:122
bool put(bool value)
Writes a boolean to the stream.
Definition bit_stream.h:130
bit_stream()
Default constructor.
Definition bit_stream.h:64
bool get(bool &value)
For bool types.
Definition bit_stream.h:201
void set_stream(void *begin_, size_t length_)
Construct from begin and length.
Definition bit_stream.h:94
void set_stream(void *begin_, void *end_)
Construct from range.
Definition bit_stream.h:104
const_iterator end() const
Returns end of the stream.
Definition bit_stream.h:329
const_iterator begin() const
Returns start of the stream.
Definition bit_stream.h:321
Declaration.
Definition delegate_cpp03.h:191
Definition optional.h:1321
Span - Fixed Extent.
Definition span.h:138
ETL_CONSTEXPR14 etl::enable_if< etl::is_integral< T >::value &&etl::is_unsigned< T >::value &&(etl::integral_limits< T >::bits==16U), T >::type reverse_bits(T value)
Definition binary.h:542
ETL_CONSTEXPR14 TReturn sign_extend(TValue value)
Definition binary.h:272
Definition endianness.h:100
enable_if
Definition type_traits_generator.h:1254
is_same
Definition type_traits_generator.h:1104
is_signed
Definition type_traits_generator.h:1074
make_signed
Definition type_traits_generator.h:1234
make_unsigned
Definition type_traits_generator.h:1244
bitset_ext
Definition absolute.h:39
bool read_unchecked< bool >(etl::bit_stream_reader &stream)
Read an unchecked bool from a stream.
Definition bit_stream.h:1394
etl::optional< T > read(etl::bit_stream_reader &stream)
Read a checked type from a stream.
Definition bit_stream.h:1379
void write_unchecked(etl::bit_stream_writer &stream, bool value)
Definition bit_stream.h:988
etl::optional< bool > read< bool >(etl::bit_stream_reader &stream)
Read a bool from a stream.
Definition bit_stream.h:1403
T read_unchecked(etl::bit_stream_reader &stream)
Read an unchecked type from a stream.
Definition bit_stream.h:1364
bool write(etl::bit_stream_writer &stream, bool value)
Definition bit_stream.h:997