P4C
The P4 Compiler
All Classes Namespaces Functions Variables Typedefs Enumerations Enumerator Friends Modules Pages
bitrange.h
1/*
2Copyright 2013-present Barefoot Networks, Inc.
3
4Licensed under the Apache License, Version 2.0 (the "License");
5you may not use this file except in compliance with the License.
6You may obtain a copy of the License at
7
8 http://www.apache.org/licenses/LICENSE-2.0
9
10Unless required by applicable law or agreed to in writing, software
11distributed under the License is distributed on an "AS IS" BASIS,
12WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13See the License for the specific language governing permissions and
14limitations under the License.
15*/
16
17#ifndef LIB_BITRANGE_H_
18#define LIB_BITRANGE_H_
19
20#include <algorithm>
21#include <iosfwd>
22#include <limits>
23#include <optional>
24#include <utility>
25
26#include "absl/numeric/bits.h"
27#include "bitvec.h"
28#include "exceptions.h"
29#include "hash.h"
30
31/* iterate over ranges of contiguous bits in a bitvector */
32class bitranges {
33 bitvec tmp;
34 const bitvec &bits;
35 struct iter {
36 bool valid = true;
38 std::pair<int, int> range;
39
40 iter &operator++() {
41 if (ptr) {
42 range.first = range.second = ptr.index();
43 while (++ptr && range.second + 1 == ptr.index()) ++range.second;
44 } else {
45 valid = false;
46 }
47 return *this;
48 }
49 std::pair<int, int> operator*() { return range; }
50 bool operator==(iter &a) const { return valid == a.valid && ptr == a.ptr; }
51 bool operator!=(iter &a) const { return !(*this == a); }
52 explicit iter(bitvec::const_bitref p) : ptr(p) { ++*this; }
53 };
54
55 public:
56 explicit bitranges(const bitvec &b) : bits(b) {}
57 explicit bitranges(bitvec &&b) : tmp(b), bits(tmp) {}
58 explicit bitranges(uintptr_t b) : tmp(b), bits(tmp) {}
59 iter begin() const { return iter(bits.begin()); }
60 iter end() const { return iter(bits.end()); }
61};
62
63class JSONGenerator;
64class JSONLoader;
65
66namespace BitRange {
67
68namespace Detail {
75constexpr inline int divideFloor(int dividend, int divisor) {
76#if defined(__GNUC__) || defined(__clang__)
77 // Code to enable compiler folding when the divisor is a power-of-two compile-time constant
78 // In this case, compiler should fold to a right-shift
79 // FIXME: Replace absl with std after moving to C++20
80 unsigned u_divisor = static_cast<unsigned>(divisor);
81 if (__builtin_constant_p(u_divisor) && absl::has_single_bit(u_divisor))
82 return dividend >> (absl::bit_width(u_divisor) - 1);
83#endif // defined(__GNUC__) || defined(__clang__)
84
85 const int quotient = dividend / divisor;
86 const int remainder = dividend % divisor;
87 if ((remainder != 0) && ((remainder < 0) != (divisor < 0))) return quotient - 1;
88 return quotient;
89}
90
98constexpr int modulo(int dividend, int divisor) {
99 return (dividend % divisor) * ((dividend < 0) != (divisor < 0) ? -1 : 1);
100}
101
115constexpr inline int moduloFloor(const int dividend, const int divisor) {
116#if defined(__GNUC__) || defined(__clang__)
117 // Code to enable compiler folding when the divisor is a power-of-two compile-time constant
118 // In this case, compiler should fold to a bitwise-and
119 // FIXME: Replace absl with std after moving to C++20
120 if (__builtin_constant_p(divisor) && absl::has_single_bit(static_cast<unsigned>(divisor)))
121 return dividend & (divisor - 1);
122#endif // defined(__GNUC__) || defined(__clang__)
123
124 const int remainder = modulo(dividend, divisor);
125 if (remainder == 0 || dividend >= 0) return remainder;
126 return divisor - remainder;
127}
128} // namespace Detail
129
139struct FromTo {
140 FromTo(int from, int to) : from(from), to(to) {}
141 FromTo(const FromTo &) = delete;
142 FromTo &operator=(const FromTo &) = delete;
143 const int from;
144 const int to;
145};
146
156struct StartLen {
157 StartLen(int start, int len) : start(start), len(len) {}
158 StartLen(const StartLen &) = delete;
159 StartLen &operator=(const StartLen &) = delete;
160 const int start;
161 const int len;
162};
163
188struct ZeroToMax {};
189
207struct MinToMax {};
208
210void rangeToJSON(JSONGenerator &json, int lo, int hi);
211std::pair<int, int> rangeFromJSON(JSONLoader &json);
212
213} // namespace BitRange
214
216enum class RangeUnit : uint8_t {
217 Bit = 0,
218 Byte = 1
219};
220
222enum class Endian : uint8_t {
223 Network = 0,
224 Little = 1,
225};
226
250template <RangeUnit Unit, Endian Order>
252 static constexpr RangeUnit unit = Unit;
253 static constexpr Endian order = Order;
254
255 using FromTo = BitRange::FromTo;
259
260 HalfOpenRange() : lo(0), hi(0) {}
261 HalfOpenRange(int lo, int hi) : lo(lo), hi(hi) {}
262 HalfOpenRange(FromTo &&fromTo) // NOLINT(runtime/explicit)
263 : lo(fromTo.from), hi(fromTo.to + 1) {}
264 HalfOpenRange(StartLen &&startLen) // NOLINT(runtime/explicit)
265 : lo(startLen.start), hi(startLen.start + startLen.len) {}
266 HalfOpenRange(ZeroToMax &&) // NOLINT(runtime/explicit)
267 : lo(0), hi(std::numeric_limits<int>::max()) {}
268 HalfOpenRange(MinToMax &&) // NOLINT(runtime/explicit)
269 : lo(std::numeric_limits<int>::min()), hi(std::numeric_limits<int>::max()) {}
270 explicit HalfOpenRange(std::pair<int, int> range) : lo(range.first), hi(range.second) {}
271
273 ssize_t size() const { return ssize_t(hi) - ssize_t(lo); }
274
278 if (empty()) return HalfOpenRange(0, 0);
279 return *this;
280 }
281
287 auto asBits = toUnit<RangeUnit::Bit>();
288 return {asBits.lo, asBits.lo + size};
289 }
290
294 const int resizedLo = empty() ? 0 : lo;
295 if (Unit == RangeUnit::Byte) return {resizedLo, resizedLo + size};
296 return {resizedLo, resizedLo + size * 8};
297 }
298
304 auto asBits = toUnit<RangeUnit::Bit>();
305 return {asBits.lo + offset, asBits.hi + offset};
306 }
307
312 if (empty()) return HalfOpenRange();
313 if (Unit == RangeUnit::Byte) return {lo + offset, hi + offset};
314 return {lo + offset * 8, hi + offset * 8};
315 }
316
318 int loByte() const {
319 if (empty()) return 0;
320 return Unit == RangeUnit::Byte ? lo : BitRange::Detail::divideFloor(lo, 8);
321 }
322
324 int hiByte() const {
325 if (empty()) return 0;
326 return Unit == RangeUnit::Byte ? hi - 1 : BitRange::Detail::divideFloor(hi - 1, 8);
327 }
328
332 int nextByte() const { return empty() ? 0 : hiByte() + 1; }
333
336 bool isLoAligned() const {
337 return (empty() || Unit == RangeUnit::Byte) ? true
338 : BitRange::Detail::moduloFloor(lo, 8) == 0;
339 }
340
344 bool isHiAligned() const {
345 return (empty() || Unit == RangeUnit::Byte) ? true
346 : BitRange::Detail::moduloFloor(hi, 8) == 0;
347 }
348
349 bool operator==(HalfOpenRange other) const {
350 if (empty()) return other.empty();
351 return other.lo == lo && other.hi == hi;
352 }
353 bool operator!=(HalfOpenRange other) const { return !(*this == other); }
354
356 bool empty() const { return lo == hi; }
357
359 bool contains(int index) const { return index >= lo && index < hi; }
360
363 bool contains(HalfOpenRange other) const { return intersectWith(other) == other; }
364
368 bool overlaps(HalfOpenRange a) const { return !intersectWith(a).empty(); }
369 bool overlaps(int l, int h) const { return !intersectWith(l, h).empty(); }
370
375 HalfOpenRange intersectWith(int l, int h) const {
376 HalfOpenRange rv = {std::max(lo, l), std::min(hi, h)};
377 if (rv.hi <= rv.lo) return {0, 0};
378 return rv;
379 }
380 HalfOpenRange operator&(HalfOpenRange a) const { return intersectWith(a); }
381 HalfOpenRange operator&=(HalfOpenRange a) {
382 *this = intersectWith(a);
383 return *this;
384 }
385
391 HalfOpenRange unionWith(int l, int h) const {
392 if (empty()) return {l, h};
393 if (l == h) return *this;
394 return HalfOpenRange(std::min(lo, l), std::max(hi, h));
395 }
396 HalfOpenRange operator|(HalfOpenRange a) const { return unionWith(a); }
397 HalfOpenRange operator|=(HalfOpenRange a) {
398 *this = unionWith(a);
399 return *this;
400 }
401
424 template <Endian DestOrder>
426 if (DestOrder == Order) return HalfOpenRange<Unit, DestOrder>(lo, hi);
427 switch (DestOrder) {
428 case Endian::Network:
429 case Endian::Little:
430 return HalfOpenRange<Unit, DestOrder>(spaceSize - hi, spaceSize - lo);
431 }
432 BUG("Unexpected ordering");
433 }
434
450 template <RangeUnit DestUnit>
452 if (DestUnit == Unit) return HalfOpenRange<DestUnit, Order>(lo, hi);
454 switch (DestUnit) {
455 case RangeUnit::Bit:
456 return HalfOpenRange<DestUnit, Order>(lo * 8, hi * 8);
457 case RangeUnit::Byte:
459 }
460 BUG("Unexpected unit");
461 }
462
464 void toJSON(JSONGenerator &json) const { BitRange::rangeToJSON(json, lo, hi); }
465 static HalfOpenRange fromJSON(JSONLoader &json) {
466 return HalfOpenRange(BitRange::rangeFromJSON(json));
467 }
468
470 bool operator<(const HalfOpenRange &other) const {
471 if (lo != other.lo) return lo < other.lo;
472 return hi < other.hi;
473 }
474
475 friend size_t hash_value(const HalfOpenRange &r) { return Util::Hash{}(r.lo, r.hi); }
476
480 int lo;
481
486 int hi;
487};
488
510template <RangeUnit Unit, Endian Order>
512 static constexpr RangeUnit unit = Unit;
513 static constexpr Endian order = Order;
514
515 using FromTo = BitRange::FromTo;
519
520 ClosedRange() : lo(0), hi(0) {} // FIXME: default is [0,0]? This is just wrong ...
521 constexpr ClosedRange(int lo, int hi) : lo(lo), hi(hi) {}
522 ClosedRange(FromTo &&fromTo) // NOLINT(runtime/explicit)
523 : lo(fromTo.from), hi(fromTo.to) {}
524 ClosedRange(StartLen &&startLen) // NOLINT(runtime/explicit)
525 : lo(startLen.start), hi(startLen.start + startLen.len - 1) {}
526 ClosedRange(ZeroToMax &&) // NOLINT(runtime/explicit)
527 : lo(0), hi(std::numeric_limits<int>::max() - 1) {}
528 ClosedRange(MinToMax &&) // NOLINT(runtime/explicit)
529 : lo(std::numeric_limits<int>::min()), hi(std::numeric_limits<int>::max() - 1) {}
530 explicit ClosedRange(std::pair<int, int> range) : lo(range.first), hi(range.second) {}
531 ClosedRange(const HalfOpenRange<Unit, Order> &r) // NOLINT(runtime/explicit)
532 : lo(r.lo), hi(r.hi - 1) {
533 BUG_CHECK(!r.empty(), "can't convert empty range to Closed");
534 }
535
537 ssize_t size() const { return ssize_t(hi) - ssize_t(lo) + 1; }
538
541 BUG_CHECK(size != 0, "Resizing ClosedRange to zero size");
542 auto asBits = toUnit<RangeUnit::Bit>();
543 return {asBits.lo, asBits.lo + size - 1};
544 }
545
548 BUG_CHECK(size != 0, "Resizing ClosedRange to zero size");
549 if (Unit == RangeUnit::Byte) return {lo, lo + size - 1};
550 return {lo, lo + size * 8 - 1};
551 }
552
555 auto asBits = toUnit<RangeUnit::Bit>();
556 return {asBits.lo + offset, asBits.hi + offset};
557 }
558
560 ClosedRange shiftedByBytes(int offset) const {
561 if (Unit == RangeUnit::Byte) return {lo + offset, hi + offset};
562 return {lo + offset * 8, hi + offset * 8};
563 }
564
566 int loByte() const {
567 return Unit == RangeUnit::Byte ? lo : BitRange::Detail::divideFloor(lo, 8);
568 }
569
571 int hiByte() const {
572 return Unit == RangeUnit::Byte ? hi : BitRange::Detail::divideFloor(hi, 8);
573 }
574
576 int nextByte() const { return hiByte() + 1; }
577
579 bool isLoAligned() const {
580 return Unit == RangeUnit::Byte ? true : BitRange::Detail::moduloFloor(lo, 8) == 0;
581 }
582
584 bool isHiAligned() const {
585 return Unit == RangeUnit::Byte ? true : BitRange::Detail::moduloFloor(hi + 1, 8) == 0;
586 }
587
588 bool operator==(ClosedRange other) const { return (other.lo == lo) && (other.hi == hi); }
589 bool operator!=(ClosedRange other) const { return !(*this == other); }
590
592 bool contains(int index) const { return (index >= lo) && (index <= hi); }
593
595 bool contains(ClosedRange other) const {
596 auto intersection = intersectWith(other);
597 return intersection.lo == other.lo && intersection.size() == other.size();
598 }
599
601 bool overlaps(ClosedRange a) const { return !intersectWith(a).empty(); }
602 bool overlaps(int l, int h) const { return !intersectWith(l, h).empty(); }
603
611 HalfOpenRange<Unit, Order> intersectWith(int l, int h) const {
612 return HalfOpenRange<Unit, Order>(lo, hi + 1)
614 }
615 HalfOpenRange<Unit, Order> operator&(ClosedRange a) const { return intersectWith(a); }
617 *this = intersectWith(a);
618 return *this;
619 }
620
622 ClosedRange unionWith(ClosedRange a) const { return unionWith(a.lo, a.hi); }
623 ClosedRange unionWith(int l, int h) const {
624 return ClosedRange(std::min(lo, l), std::max(hi, h));
625 }
626 ClosedRange operator|(ClosedRange a) const { return unionWith(a); }
627 ClosedRange operator|=(ClosedRange a) {
628 *this = unionWith(a);
629 return *this;
630 }
631
633 template <Endian DestOrder>
635 BUG_CHECK(spaceSize > 0, "Can't represent an empty range");
636 if (DestOrder == Order) return ClosedRange<Unit, DestOrder>(lo, hi);
637 switch (DestOrder) {
638 case Endian::Network:
639 case Endian::Little:
640 return ClosedRange<Unit, DestOrder>((spaceSize - 1) - hi, (spaceSize - 1) - lo);
641 }
642 BUG("Unexpected ordering");
643 }
644
646 template <RangeUnit DestUnit>
648 if (DestUnit == Unit) return ClosedRange<DestUnit, Order>(lo, hi);
649 switch (DestUnit) {
650 case RangeUnit::Bit:
651 return ClosedRange<DestUnit, Order>(lo * 8, hi * 8 + 7);
652 case RangeUnit::Byte:
654 }
655 BUG("Unexpected unit");
656 }
657
659 void toJSON(JSONGenerator &json) const { BitRange::rangeToJSON(json, lo, hi); }
660 static ClosedRange fromJSON(JSONLoader &json) {
661 return ClosedRange(BitRange::rangeFromJSON(json));
662 }
663
665 bool operator<(const ClosedRange &other) const {
666 if (lo != other.lo) return lo < other.lo;
667 return hi < other.hi;
668 }
669
670 friend size_t hash_value(const ClosedRange &r) { return Util::Hash{}(r.lo, r.hi); }
671
674 cstring formatAsSlice(int spaceSize) const {
675 auto r = toOrder<Endian::Little>(spaceSize);
676 auto hi = r.hi;
677 auto lo = r.lo;
678 if (Unit == RangeUnit::Byte) {
679 lo = lo * 8;
680 hi = hi * 8 + 7;
681 } else {
682 BUG_CHECK(Unit == RangeUnit::Bit, "mismatch range units");
683 }
684 std::stringstream out;
685 out << "[" << hi << ":" << lo << "]";
686 return cstring(out.str());
687 }
688
692 int lo;
693
698 int hi;
699};
700
713template <RangeUnit Unit, Endian Order>
714std::pair<HalfOpenRange<Unit, Order>, HalfOpenRange<Unit, Order>> operator-(
716 HalfOpenRange<Unit, Order> empty = {0, 0};
717 HalfOpenRange<Unit, Order> intersect = left.intersectWith(right);
718 if (intersect.empty())
719 return left.lo < right.lo ? std::make_pair(left, empty) : std::make_pair(empty, left);
720
722 left.lo == intersect.lo ? empty : BitRange::FromTo(left.lo, intersect.lo - 1);
724 left.hi == intersect.hi ? empty : BitRange::FromTo(intersect.hi, left.hi - 1);
725
726 return {lower, upper};
727}
728
731template <RangeUnit Unit, Endian Order>
732std::pair<HalfOpenRange<Unit, Order>, HalfOpenRange<Unit, Order>> operator-(
734 return toHalfOpenRange(left) - toHalfOpenRange(right);
735}
736
739template <RangeUnit Unit, Endian Order>
740HalfOpenRange<Unit, Order> toHalfOpenRange(ClosedRange<Unit, Order> closedRange) {
741 return HalfOpenRange<Unit, Order>(closedRange.lo, closedRange.hi + 1);
742}
743
746template <RangeUnit Unit, Endian Order>
747std::optional<ClosedRange<Unit, Order>> toClosedRange(HalfOpenRange<Unit, Order> halfOpenRange) {
748 if (halfOpenRange.empty()) return std::nullopt;
749 return ClosedRange<Unit, Order>(halfOpenRange.lo, halfOpenRange.hi - 1);
750}
751
755
759
763
767
768std::ostream &toStream(std::ostream &out, RangeUnit unit, Endian order, int lo, int hi,
769 bool closed);
770
771template <RangeUnit Unit, Endian Order>
772std::ostream &operator<<(std::ostream &out, const HalfOpenRange<Unit, Order> &range) {
773 return toStream(out, Unit, Order, range.lo, range.hi, false);
774}
775
776template <RangeUnit Unit, Endian Order>
777std::ostream &operator<<(std::ostream &out, const ClosedRange<Unit, Order> &range) {
778 return toStream(out, Unit, Order, range.lo, range.hi, true);
779}
780
781// Hashing specializations
782namespace std {
783template <RangeUnit Unit, Endian Order>
784struct hash<HalfOpenRange<Unit, Order>> {
785 std::size_t operator()(const HalfOpenRange<Unit, Order> &r) const {
786 return Util::Hash{}(r.lo, r.hi);
787 }
788};
789
790template <RangeUnit Unit, Endian Order>
791struct hash<ClosedRange<Unit, Order>> {
792 std::size_t operator()(const ClosedRange<Unit, Order> &r) const {
793 return Util::Hash{}(r.lo, r.hi);
794 }
795};
796} // namespace std
797
798namespace Util {
799template <RangeUnit Unit, Endian Order>
800struct Hasher<HalfOpenRange<Unit, Order>> {
801 size_t operator()(const HalfOpenRange<Unit, Order> &r) const {
802 return Util::Hash{}(r.lo, r.hi);
803 }
804};
805
806template <RangeUnit Unit, Endian Order>
807struct Hasher<ClosedRange<Unit, Order>> {
808 size_t operator()(const ClosedRange<Unit, Order> &r) const { return Util::Hash{}(r.lo, r.hi); }
809};
810} // namespace Util
811
812#endif /* LIB_BITRANGE_H_ */
Definition json_generator.h:34
Definition json_loader.h:36
Definition bitrange.h:32
Definition bitvec.h:200
Definition bitvec.h:119
Definition cstring.h:80
STL namespace.
Definition bitrange.h:139
Definition bitrange.h:207
Definition bitrange.h:156
Definition bitrange.h:188
Definition bitrange.h:511
HalfOpenRange< Unit, Order > intersectWith(ClosedRange a) const
Definition bitrange.h:608
ClosedRange resizedToBytes(int size) const
Definition bitrange.h:547
ClosedRange< Unit, DestOrder > toOrder(int spaceSize) const
Definition bitrange.h:634
int nextByte() const
Definition bitrange.h:576
bool operator<(const ClosedRange &other) const
Definition bitrange.h:665
int lo
Definition bitrange.h:692
bool isLoAligned() const
Definition bitrange.h:579
ClosedRange shiftedByBytes(int offset) const
Definition bitrange.h:560
bool contains(int index) const
Definition bitrange.h:592
bool isHiAligned() const
Definition bitrange.h:584
ssize_t size() const
Definition bitrange.h:537
ClosedRange< RangeUnit::Bit, Order > shiftedByBits(int offset) const
Definition bitrange.h:554
ClosedRange unionWith(ClosedRange a) const
Definition bitrange.h:622
int loByte() const
Definition bitrange.h:566
bool contains(ClosedRange other) const
Definition bitrange.h:595
cstring formatAsSlice(int spaceSize) const
Definition bitrange.h:674
ClosedRange< DestUnit, Order > toUnit() const
Definition bitrange.h:647
ClosedRange< RangeUnit::Bit, Order > resizedToBits(int size) const
Definition bitrange.h:540
bool overlaps(ClosedRange a) const
Definition bitrange.h:601
int hiByte() const
Definition bitrange.h:571
int hi
Definition bitrange.h:698
void toJSON(JSONGenerator &json) const
JSON serialization/deserialization.
Definition bitrange.h:659
Definition bitrange.h:251
ssize_t size() const
Definition bitrange.h:273
HalfOpenRange< Unit, Order > shiftedByBytes(int offset) const
Definition bitrange.h:311
bool contains(int index) const
Definition bitrange.h:359
HalfOpenRange< RangeUnit::Bit, Order > shiftedByBits(int offset) const
Definition bitrange.h:302
bool operator<(const HalfOpenRange &other) const
Total ordering, first by lo, then by hi.
Definition bitrange.h:470
HalfOpenRange< Unit, DestOrder > toOrder(int spaceSize) const
Definition bitrange.h:425
bool isLoAligned() const
Definition bitrange.h:336
HalfOpenRange resizedToBytes(int size) const
Definition bitrange.h:293
int lo
Definition bitrange.h:480
bool contains(HalfOpenRange other) const
Definition bitrange.h:363
HalfOpenRange< DestUnit, Order > toUnit() const
Definition bitrange.h:451
int hi
Definition bitrange.h:486
bool isHiAligned() const
Definition bitrange.h:344
HalfOpenRange< RangeUnit::Bit, Order > resizedToBits(int size) const
Definition bitrange.h:285
int hiByte() const
Definition bitrange.h:324
HalfOpenRange intersectWith(HalfOpenRange a) const
Definition bitrange.h:374
int nextByte() const
Definition bitrange.h:332
bool overlaps(HalfOpenRange a) const
Definition bitrange.h:368
HalfOpenRange canonicalize() const
Definition bitrange.h:277
HalfOpenRange unionWith(HalfOpenRange a) const
Definition bitrange.h:390
bool empty() const
Definition bitrange.h:356
void toJSON(JSONGenerator &json) const
JSON serialization/deserialization.
Definition bitrange.h:464
int loByte() const
Definition bitrange.h:318
Definition hash.h:125
Definition hash.h:123