forked from Harrm/scale-codec
-
Notifications
You must be signed in to change notification settings - Fork 7
/
Copy pathbit_vector.hpp
1554 lines (1409 loc) · 51.9 KB
/
bit_vector.hpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
/**
* Copyright Quadrivium LLC
* All Rights Reserved
* SPDX-License-Identifier: Apache-2.0
*/
#pragma once
#include <array>
#include <limits>
#include <vector>
#include <scale/detail/compact_integer.hpp>
#include <scale/detail/decomposable_type_traits.hpp>
#include <scale/types.hpp>
namespace scale {
/**
* @class SmallBitVector
* @brief Defines structure for encoding and decoding bit vectors.
*
* This struct provides an encoding mechanism compatible with Rust's
* `BitVec<u8, Lsb0>`. It supports efficient serialization and deserialization
* using SCALE encoding.
*
* Internal Storage:
* - Uses a single integral type T to store both size and data bits.
* - The upper bits store the size, while the lower bits store the data.
* - Efficient memory usage by storing size in the most significant bits.
*
* Example Usage:
* @code
* scale::SmallBitVector<> vec;
* vec.push_back(true);
* vec.push_back(false);
* vec.push_back(true);
* for (auto bit : vec) {
* std::cout << bit;
* }
* // Output: 101
* @endcode
*/
template <typename T = uintmax_t>
requires std::is_integral_v<T> and std::is_unsigned_v<T>
class SmallBitVector {
#ifdef CFG_TESTING
public:
#endif
//==========================================================================
// Static Constants
//==========================================================================
/// @brief Number of bits in the type T.
static constexpr size_t all_bits = std::numeric_limits<T>::digits;
/**
* @brief Number of bits needed to store the size.
* @details Calculated based on the total number of bits in T.
*/
static constexpr size_t size_bits = (all_bits >= 522) ? 10
: (all_bits >= 265) ? 9
: (all_bits >= 136) ? 8
: (all_bits >= 71) ? 7
: (all_bits >= 38) ? 6
: (all_bits >= 21) ? 5
: (all_bits >= 12) ? 4
: 3;
/// @brief Number of bits available for data.
static constexpr size_t data_bits = all_bits - size_bits;
/// @brief Mask for extracting size bits.
static constexpr T size_mask = static_cast<T>(-1) << data_bits;
/// @brief Mask for extracting data bits.
static constexpr T data_mask = static_cast<T>(-1) >> size_bits;
public:
//==========================================================================
// Nested Types
//==========================================================================
/**
* @brief Proxy class for accessing and modifying individual bits.
*
* This class provides a proxy for modifying bits in the SmallBitVector.
*/
class BitReference {
public:
/**
* @brief Constructs a BitReference.
* @param vec Reference to the SmallBitVector.
* @param index Index of the bit.
*/
BitReference(SmallBitVector &vec, size_t index) noexcept
: vec_(vec), index_(index) {}
/**
* @brief Converts the bit to a boolean value.
* @return True if the bit is set, false otherwise.
*/
operator bool() const noexcept {
return static_cast<const SmallBitVector &>(vec_)[index_];
}
/**
* @brief Assigns a boolean value to the bit.
* @param value The value to assign.
* @return Reference to this BitReference.
*/
BitReference &operator=(bool value) noexcept {
if (value) {
vec_.bits_ |= (static_cast<T>(1) << index_);
} else {
vec_.bits_ &= ~(static_cast<T>(1) << index_);
}
return *this;
}
private:
SmallBitVector &vec_;
size_t index_;
};
/**
* @brief Iterator for traversing bits in SmallBitVector.
*
* This iterator provides random access to bits stored in the
* SmallBitVector. It supports standard iterator operations such as
* increment, dereference, and comparison.
*/
class Iterator {
public:
/// @brief Iterator category indicating random access capability.
using iterator_category = std::random_access_iterator_tag;
/// @brief Type of values pointed to by the iterator.
using value_type = bool;
/// @brief Type used for distance between iterators.
using difference_type = std::ptrdiff_t;
/**
* @brief Constructs an iterator for SmallBitVector.
* @param vec Reference to the SmallBitVector.
* @param pos Position of the current bit in the vector.
*/
Iterator(const SmallBitVector &vec, size_t pos) noexcept
: vec_(&vec), pos_(pos) {}
/**
* @brief Dereferences the iterator to access the current bit.
* @return True if the current bit is set, false otherwise.
*/
bool operator*() const noexcept {
return (*vec_)[pos_];
}
/**
* @brief Prefix increment. Moves the iterator to the next bit.
* @return Reference to the updated iterator.
*/
Iterator &operator++() noexcept {
++pos_;
return *this;
}
/**
* @brief Postfix increment. Moves the iterator to the next bit.
* @return Copy of the iterator before increment.
*/
Iterator operator++(int) noexcept {
Iterator tmp = *this;
++(*this);
return tmp;
}
/**
* @brief Difference operator for calculating distance between iterators.
* @param other Iterator to calculate difference with.
* @return Difference in positions.
*/
difference_type operator-(const Iterator &other) const noexcept {
return static_cast<difference_type>(pos_)
- static_cast<difference_type>(other.pos_);
}
/**
* @brief Equality comparison operator.
* @param other Iterator to compare with.
* @return True if both iterators are at the same position.
*/
bool operator==(const Iterator &other) const noexcept {
return pos_ == other.pos_;
}
/**
* @brief Inequality comparison operator.
* @param other Iterator to compare with.
* @return True if iterators are at different positions.
*/
bool operator!=(const Iterator &other) const noexcept {
return !(*this == other);
}
/**
* @brief Returns the current position of the iterator.
* @return Current bit position.
*/
size_t pos() const noexcept {
return pos_;
}
private:
const SmallBitVector *vec_;
size_t pos_;
};
//==========================================================================
// Constructors and Assignment Operators
//==========================================================================
/// @brief Default constructor. Initializes the vector as empty.
SmallBitVector() noexcept : bits_(0) {}
/**
* @brief Constructs a SmallBitVector from an unsigned integral value.
* @tparam I Integral type that fits within T.
* @param bits Initial bits for the vector.
*/
template <typename I>
requires std::is_integral_v<I> and std::is_unsigned_v<I>
and (sizeof(I) <= sizeof(T))
explicit SmallBitVector(I bits) noexcept : bits_(bits) {}
/// @brief Default copy constructor.
SmallBitVector(const SmallBitVector &) noexcept = default;
/// @brief Default move constructor.
SmallBitVector(SmallBitVector &&) noexcept = default;
/// @brief Default copy assignment operator.
SmallBitVector &operator=(const SmallBitVector &) noexcept = default;
/// @brief Default move assignment operator.
SmallBitVector &operator=(SmallBitVector &&) noexcept = default;
/**
* @brief Constructs a SmallBitVector from a collection of boolean values.
* @tparam InputIt Type of the input iterator.
* @param first Iterator pointing to the first element in the collection.
* @param last Iterator pointing to one past the last element in the
* collection.
* @throws std::overflow_error If the collection size exceeds the capacity.
*/
template <typename InputIt>
SmallBitVector(InputIt first, InputIt last) {
size_t new_size = std::distance(first, last);
if (new_size > data_bits) {
throw std::overflow_error("Collection size exceeds capacity");
}
T new_data = 0;
size_t index = 0;
for (auto it = first; it != last; ++it) {
if (*it) {
new_data |= (static_cast<T>(1) << index);
}
++index;
}
bits_ = (new_size << data_bits) | (new_data & data_mask);
}
/**
* @brief Constructs a SmallBitVector from an initializer list of boolean
* values.
* @param init_list Initializer list of boolean values.
* @throws std::overflow_error If the list size exceeds the capacity.
*/
SmallBitVector(std::initializer_list<bool> init_list)
: SmallBitVector(init_list.begin(), init_list.end()) {}
/**
* @brief Constructs a SmallBitVector from a collection of boolean values.
* @tparam Collection Type of the collection (e.g., std::vector<bool>,
* std::list<bool>).
* @param collection The collection containing boolean values.
* @throws std::overflow_error If the collection size exceeds the capacity.
*/
template <typename Collection>
requires std::is_class_v<Collection>
explicit SmallBitVector(const Collection &collection)
: SmallBitVector(collection.begin(), collection.end()) {}
/**
* @brief Assigns the contents of a collection to the SmallBitVector.
* @tparam Collection Type of the collection (e.g., std::vector<bool>,
* std::list<bool>).
* @param collection The collection containing boolean values.
* @return Reference to this SmallBitVector.
* @throws std::overflow_error If the collection size exceeds the capacity.
*/
template <typename Collection>
SmallBitVector &operator=(const Collection &collection) {
size_t new_size = collection.size();
if (new_size > data_bits) {
throw std::overflow_error("Collection size exceeds capacity");
}
T new_data = 0;
size_t index = 0;
for (const auto &value : collection) {
if (value) {
new_data |= (static_cast<T>(1) << index);
}
++index;
}
bits_ = (new_size << data_bits) | (new_data & data_mask);
return *this;
}
//==========================================================================
// Capacity and Element Access
//==========================================================================
/**
* @brief Converts the SmallBitVector to the underlying type T.
* @return The bits stored in the vector.
*/
explicit operator T() const noexcept {
return bits_;
}
/**
* @brief Returns the number of elements (bits) in the vector.
* @return The current size of the vector.
*/
[[nodiscard]] size_t size() const noexcept {
return bits_ >> data_bits;
}
/**
* @brief Returns the maximum capacity of the vector.
* @return Maximum number of bits that can be stored.
*/
[[nodiscard]] size_t capacity() const noexcept {
return data_bits;
}
/**
* @brief Checks if the vector is empty.
* @return True if the vector is empty, false otherwise.
*/
[[nodiscard]] bool empty() const noexcept {
return size() == 0;
}
/**
* @brief Provides read-only access to the bit at the given index.
* @param index Position of the bit to access.
* @return True if the bit is set, false otherwise.
*/
bool operator[](size_t index) const noexcept {
return (bits_ >> index) & static_cast<T>(1);
}
/**
* @brief Provides mutable access to the bit at the given index.
* @param index Position of the bit to access.
* @return A BitReference proxy for modifying the bit.
*/
BitReference operator[](size_t index) noexcept {
return BitReference(*this, index);
}
/**
* @brief Accesses the bit at the given index with bounds checking.
* @param index Position of the bit to access.
* @return True if the bit is set, false otherwise.
* @throws std::out_of_range If the index is out of bounds.
*/
bool at(size_t index) const {
if (index >= size()) {
throw std::out_of_range("Index out of bound");
}
return (*this)[index];
}
/**
* @brief Returns the raw data bits of the vector.
* @return The data bits stored in the vector.
*/
[[nodiscard]] T data() const noexcept {
return bits_ & data_mask;
}
//==========================================================================
// Modifiers
//==========================================================================
/**
* @brief Clears the vector, setting all bits to zero.
*/
void clear() noexcept {
bits_ = 0;
}
/**
* @brief Reserves capacity for at least the specified number of bits.
* @param new_capacity Minimum number of bits to reserve.
* @throws std::overflow_error If the requested capacity exceeds the maximum
* capacity.
*/
void reserve(size_t new_capacity) {
if (new_capacity > data_bits) {
throw std::overflow_error(
"Requested capacity exceeds maximum capacity");
}
// No actual allocation needed since the capacity is fixed by data_bits.
}
/**
* @brief Resizes the vector to the new size.
* @param new_size New size of the vector.
* @throws std::out_of_range If the new size exceeds capacity.
*/
void resize(size_t new_size) {
if (new_size > data_bits) {
throw std::out_of_range("New size exceeds capacity");
}
// Preserve current data bits up to new_size.
T current_data = bits_ & data_mask;
T new_data = current_data & ((static_cast<T>(1) << new_size) - 1);
bits_ = (new_size << data_bits) | new_data;
}
/**
* @brief Resizes the vector to the new size and fills new bits with a given
* value.
* @param new_size New size of the vector.
* @param value Value to fill new bits (true or false).
* @throws std::out_of_range If the new size exceeds capacity.
*/
void resize(size_t new_size, bool value) {
size_t current_size = size();
if (new_size == current_size) return;
if (new_size > data_bits) {
throw std::out_of_range("New size exceeds capacity");
}
T current_data = bits_ & data_mask;
if (new_size > current_size) {
// Expanding the vector
size_t added_bits = new_size - current_size;
T new_bits = value ? ((static_cast<T>(1) << added_bits) - 1) : 0;
new_bits <<= current_size;
current_data |= new_bits;
} else {
// Shrinking the vector
current_data &= ((static_cast<T>(1) << new_size) - 1);
}
bits_ = (new_size << data_bits) | current_data;
}
/**
* @brief Adds a new bit to the end of the vector.
* @param value Value of the new bit (true or false).
* @throws std::overflow_error If the vector exceeds its capacity.
*/
void push_back(bool value) {
size_t current_size = size();
if (current_size >= data_bits) {
throw std::overflow_error("Exceeded maximum capacity");
}
// Update size.
bits_ = ((current_size + 1) << data_bits) | (bits_ & data_mask);
if (value) {
bits_ |= static_cast<T>(1) << current_size;
}
}
/**
* @brief Removes the last bit from the vector.
* @throws std::out_of_range If the vector is empty.
*/
void pop_back() {
size_t current_size = size();
if (current_size == 0) {
throw std::out_of_range("pop_back on empty vector");
}
// Clear the last bit.
bits_ &= ~(static_cast<T>(1) << (current_size - 1));
// Update size.
bits_ = ((current_size - 1) << data_bits) | (bits_ & data_mask);
}
/**
* @brief Inserts a bit at the specified position.
* @param pos Position at which to insert the new bit.
* @param value Value of the new bit.
* @throws std::overflow_error If the vector exceeds its capacity.
* @throws std::out_of_range If the position is out of bounds.
*/
void insert(size_t pos, bool value) {
size_t current_size = size();
if (current_size >= data_bits) {
throw std::overflow_error("Exceeded maximum capacity");
}
if (pos > current_size) {
throw std::out_of_range("Insert position out of range");
}
T data_val = bits_ & data_mask;
// Shift bits at and above pos to the left by one.
T lower = data_val & ((static_cast<T>(1) << pos) - 1);
T upper = data_val & ~((static_cast<T>(1) << pos) - 1);
upper <<= 1;
T new_data = lower | upper;
// Set the inserted bit.
if (value) {
new_data |= static_cast<T>(1) << pos;
} else {
new_data &= ~(static_cast<T>(1) << pos);
}
bits_ = ((current_size + 1) << data_bits) | (new_data & data_mask);
}
/**
* @brief Inserts multiple bits with the same value at the specified
* position.
* @param pos Iterator pointing to the position at which to insert the new
* bits.
* @param count Number of bits to insert.
* @param value The value to assign to each inserted bit (true or false).
* @throws std::overflow_error If the vector exceeds its capacity.
* @throws std::out_of_range If the position is out of bounds.
*/
void insert(Iterator pos, size_t count, bool value) {
size_t current_size = size();
if (count == 0) return;
if (pos.pos() > current_size) {
throw std::out_of_range("Insert position out of range");
}
if (current_size + count > data_bits) {
throw std::overflow_error("Exceeded maximum capacity");
}
T data_val = bits_ & data_mask;
T lower = data_val & ((static_cast<T>(1) << pos.pos()) - 1);
T upper = data_val & ~((static_cast<T>(1) << pos.pos()) - 1);
upper <<= count;
T new_data = lower | upper;
size_t insert_pos = pos.pos();
for (size_t i = 0; i < count; ++i) {
if (value) {
new_data |= (static_cast<T>(1) << insert_pos);
} else {
new_data &= ~(static_cast<T>(1) << insert_pos);
}
++insert_pos;
}
bits_ = ((current_size + count) << data_bits) | (new_data & data_mask);
}
/**
* @brief Inserts a range of bits into the vector at the specified position.
* @tparam InputIt Type of the input iterator.
* @param pos Iterator pointing to the position at which to insert the new
* bits.
* @param first Iterator pointing to the first element in the range.
* @param last Iterator pointing to one past the last element in the range.
* @throws std::overflow_error If the vector exceeds its capacity.
* @throws std::out_of_range If the position is out of bounds.
*/
template <typename InputIt>
void insert(Iterator pos, InputIt first, InputIt last) {
size_t current_size = size();
size_t new_bits_count = std::distance(first, last);
if (new_bits_count == 0) return;
if (pos.pos() > current_size) {
throw std::out_of_range("Insert position out of range");
}
if (current_size + new_bits_count > data_bits) {
throw std::overflow_error("Exceeded maximum capacity");
}
T data_val = bits_ & data_mask;
T lower = data_val & ((static_cast<T>(1) << pos.pos()) - 1);
T upper = data_val & ~((static_cast<T>(1) << pos.pos()) - 1);
upper <<= new_bits_count;
T new_data = lower | upper;
size_t insert_pos = pos.pos();
for (auto it = first; it != last; ++it) {
if (*it) {
new_data |= (static_cast<T>(1) << insert_pos);
} else {
new_data &= ~(static_cast<T>(1) << insert_pos);
}
++insert_pos;
}
bits_ = ((current_size + new_bits_count) << data_bits)
| (new_data & data_mask);
}
/**
* @brief Erases the bit at the specified position.
* @param pos Position of the bit to erase.
* @throws std::out_of_range If the position is out of bounds.
*/
void erase(size_t pos) {
size_t current_size = size();
if (pos >= current_size) {
throw std::out_of_range("Erase position out of range");
}
T data_val = bits_ & data_mask;
T lower = data_val & ((static_cast<T>(1) << pos) - 1);
T upper = data_val >> (pos + 1);
T new_data = lower | (upper << pos);
bits_ = ((current_size - 1) << data_bits) | (new_data & data_mask);
}
/**
* @brief Assigns the vector with the specified number of bits, all set to a
* given value.
* @param count Number of bits to assign.
* @param value The value to assign to each bit.
* @throws std::out_of_range If count exceeds capacity.
*/
void assign(size_t count, bool value) {
if (count > data_bits) {
throw std::out_of_range("Assign count exceeds capacity");
}
T new_data = value ? ((static_cast<T>(1) << count) - 1) : 0;
bits_ = (count << data_bits) | (new_data & data_mask);
}
/**
* @brief Swaps the contents of this vector with another.
* @param other The other SmallBitVector to swap with.
*/
void swap(SmallBitVector &other) noexcept {
std::swap(bits_, other.bits_);
}
//==========================================================================
// Iterators
//==========================================================================
/**
* @brief Returns an iterator to the beginning of the vector.
* @return Iterator pointing to the first bit.
*/
Iterator begin() const noexcept {
return Iterator(*this, 0);
}
/**
* @brief Returns an iterator to the end of the vector.
* @return Iterator pointing to one past the last bit.
*/
Iterator end() const noexcept {
return Iterator(*this, size());
}
/**
* @brief Checks if two SmallBitVector objects are equal.
* @param other The SmallBitVector to compare with.
* @return true if both SmallBitVectors are equal, false otherwise
*/
bool operator==(const SmallBitVector &other) const noexcept {
return this->size() == other.size() && this->data() == other.data();
}
/**
* @brief Checks if two SmallBitVector objects are not equal.
* @param other The SmallBitVector to compare with.
* @return true if SmallBitVectors are not equal, false otherwise
*/
bool operator!=(const SmallBitVector &other) const noexcept {
return !(*this == other);
}
//==========================================================================
// Friend Functions (Encoding/Decoding)
//==========================================================================
/**
* @brief Encodes the bit vector into a SCALE encoder using bit
* manipulation.
* @param bit_vector The bit vector to encode.
* @param encoder The encoder instance to write into.
*/
friend void encode(const SmallBitVector &bit_vector, Encoder &encoder) {
size_t size = bit_vector.size();
encode(as_compact(size), encoder);
T data_val = bit_vector.bits_ & data_mask;
for (size_t i = 0; i < size; i += CHAR_BIT) {
encoder.put(static_cast<uint8_t>(data_val >> i));
}
}
/**
* @brief Decodes a BitVec from a SCALE decoder.
* @param v The BitVec instance to populate.
* @param decoder The decoder instance to read from.
* @throws DecodeError::NOT_ENOUGH_DATA If there is insufficient data.
*/
friend void decode(SmallBitVector &v, Decoder &decoder) {
size_t size;
decode(as_compact(size), decoder);
size_t byte_count = (size + CHAR_BIT - 1) / CHAR_BIT;
if (!decoder.has(byte_count)) {
raise(DecodeError::NOT_ENOUGH_DATA);
}
T bits = 0;
for (size_t i = 0; i < size; i += CHAR_BIT) {
uintmax_t byte = decoder.take();
bits |= static_cast<T>(byte) << i;
}
v = SmallBitVector(bits);
}
private:
//==========================================================================
// Private Data Members
//==========================================================================
/// Internal storage combining size and data bits.
T bits_;
};
/**
* @class BitVector
* @brief A dynamic bit vector with Small Buffer Optimization (SBO).
*
* This class implements a bit vector with the following features:
* - Small Buffer Optimization: Stores small bit sequences in a fixed-size
* array.
* - Dynamic resizing with automatic transition to dynamic vector storage.
* - Proxy class BitReference for safe bit manipulation.
* - Random-access iterators for compatibility with STL algorithms.
*/
class BitVector {
public:
//==========================================================================
// Nested Classes
//==========================================================================
/**
* @class BitReference
* @brief Proxy class for accessing and modifying individual bits.
*
* This class allows safe access and modification of bits within
BitVector.
* It supports conversion to bool and assignment of boolean values.
*/
class BitReference {
public:
/**
* @brief Constructs a BitReference.
* @param vec Reference to the parent BitVector.
* @param index Index of the bit within the BitVector.
*/
BitReference(BitVector &vec, size_t index) noexcept
: vec_(vec), index_(index) {}
/**
* @brief Converts the bit to a boolean value.
* @return true if the bit is set, false otherwise.
*/
operator bool() const noexcept {
return static_cast<const BitVector &>(vec_)[index_];
}
/**
* @brief Assigns a boolean value to the bit.
* @param value The value to assign to the bit.
* @return Reference to this BitReference.
*/
BitReference &operator=(bool value) noexcept {
auto data = vec_.sbf_ ? vec_.arr_.data() : vec_.vec_.data();
auto &byte = data[index_ / CHAR_BIT];
auto bit = index_ % CHAR_BIT;
if (value) {
byte |= (static_cast<uint8_t>(1) << bit); // Set the bit
} else {
byte &= ~(static_cast<uint8_t>(1) << bit); // Clear the bit
}
return *this;
}
/**
* @brief Assigns the value of another BitReference.
* @param other The BitReference to copy from.
* @return Reference to this BitReference.
*/
BitReference &operator=(const BitReference &other) noexcept {
return *this = static_cast<bool>(other);
}
/**
* @brief Equality comparison operator.
* @param other The BitReference to compare with.
* @return true if the bits are equal, false otherwise.
*/
bool operator==(const BitReference &other) const noexcept {
return static_cast<bool>(*this) == static_cast<bool>(other);
}
/**
* @brief Inequality comparison operator.
* @param other The BitReference to compare with.
* @return true if the bits are not equal, false otherwise.
*/
bool operator!=(const BitReference &other) const noexcept {
return !(*this == other);
}
private:
BitVector &vec_; ///< Reference to the parent BitVector
size_t index_; ///< Index of the bit within the BitVector
};
/**
* @class Iterator
* @brief Iterator for BitVector, providing random access to its
elements.
* This iterator allows traversal over a BitVector in a random-access
* manner. It provides the standard iterator interface, including
* dereference, increment (both prefix and postfix), and comparison
* operators.
*
* This version supports std::distance by implementing subtraction and
* comparison operators, enabling the use of random access iterator
* features. Designed with C++20 concepts and modern practices.
*
* This iterator works with BitReference proxy class, since references to
* individual bits cannot be used.
*/
class Iterator {
public:
/// Concept of the iterator for C++20 ranges.
using iterator_concept = std::random_access_iterator_tag;
/// Category of the iterator, indicating random access capabilities.
using iterator_category = std::random_access_iterator_tag;
/// Type of the elements pointed to by the iterator (bool for BitVector).
using value_type = bool;
/// Type used for distance between iterators.
using difference_type = std::ptrdiff_t;
/// Pointer type (not applicable but required for iterator traits).
using pointer = const bool *;
/// Reference type (using proxy class for a bit reference).
using reference = BitReference;
/**
* @brief Constructs an iterator for the given BitVector at a specified
* position.
* @param vec Reference to the BitVector to iterate over.
* @param pos Initial position of the iterator within the BitVector.
*/
constexpr Iterator(BitVector &vec, size_t pos) noexcept
: vec_(std::addressof(vec)), pos_(pos) {}
/**
* @brief Constructs an iterator for the given const BitVector at a
* specified position.
* @param vec Reference to the const BitVector to iterate over.
* @param pos Initial position of the iterator within the BitVector.
*/
constexpr Iterator(const BitVector &vec, size_t pos) noexcept
: vec_(std::addressof(vec)), pos_(pos) {}
/**
* @brief Dereferences the iterator to access the current element.
* @return The BitReference at the current iterator position.
* @note This uses a proxy class since references to bits are not
* possible.
*/
[[nodiscard]] BitReference operator*() noexcept {
return BitReference{*const_cast<BitVector *>(vec_), pos_};
}
/**
* @brief Dereferences the iterator to access the current element
(const
* version).
* @return The value of the bit at the current iterator position as
bool.
* @note Const version returns a bool since no modification is allowed.
*/
[[nodiscard]] bool operator*() const noexcept {
return (*vec_)[pos_];
}
/**
* @brief Prefix increment operator.
* Moves the iterator to the next position.
* @return Reference to the incremented iterator.
*/
Iterator &operator++() noexcept {
++pos_;
return *this;
}
/**
* @brief Postfix increment operator.
* Moves the iterator to the next position, returning the previous
state.
* @return Copy of the iterator before incrementing.
*/
Iterator operator++(int) noexcept {
Iterator tmp = *this;
++(*this);
return tmp;
}
/**
* @brief Equality comparison operator.
* Checks if two iterators are equal.
* @param other Iterator to compare with.
* @return True if iterators are equal, false otherwise.
*/
bool operator==(const Iterator &other) const noexcept {
return pos_ == other.pos_;
}
/**
* @brief Inequality comparison operator.
* Checks if two iterators are not equal.
* @param other Iterator to compare with.
* @return True if iterators are not equal, false otherwise.
*/
bool operator!=(const Iterator &other) const noexcept {
return !(*this == other);
}
/**
* @brief Subtraction operator for distance calculation.
* Allows calculation of the distance between two iterators.
* @param lhs Left-hand side iterator.
* @param rhs Right-hand side iterator.
* @return Distance between the iterators as a difference_type.
*/
friend difference_type operator-(const Iterator &lhs,
const Iterator &rhs) noexcept {
return static_cast<difference_type>(lhs.pos_)
- static_cast<difference_type>(rhs.pos_);
}
private:
/// Pointer to the BitVector being iterated over.
const BitVector *vec_;
/// Current position of the iterator within the BitVector.
size_t pos_;
};
//==========================================================================
// Constructors, Destructor, and Assignment Operators
//==========================================================================
/**
* @brief Constructs an empty BitVector.
* Initializes the BitVector with a size of 0 and enables small buffer
* optimization (SBO) by default.
*/
BitVector() : size_(0), sbf_(1), arr_{0} {}
/**
* @brief Constructs a BitVector from an initializer list.
* @param init_list The initializer list with boolean values.
*/
BitVector(std::initializer_list<bool> init_list) {
resize(init_list.size());
size_t index = 0;
for (bool value : init_list) {
(*this)[index++] = value;
}
}
/**
* @brief Constructs a BitVector from any container of boolean values.
* This constructor allows initializing the BitVector using any container
* that supports `std::begin()` and `std::end()`, including
* std::vector<bool>.
* @tparam Container A container type that holds boolean values.
* @param container The container with boolean values to initialize the
* BitVector.
*/
template <typename Container>
explicit BitVector(const Container &container) {
size_t count = std::distance(std::begin(container), std::end(container));
resize(count);
size_t index = 0;
for (const auto &value : container) {
(*this)[index++] = value;
}