P4C
The P4 Compiler
Loading...
Searching...
No Matches
symbitmatrix.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_SYMBITMATRIX_H_
18#define LIB_SYMBITMATRIX_H_
19
20#include "bitvec.h"
21
22/* A symmetric bit matrix, held in a bit vector. We only store a triangular submatrix which
23 * is used for both halves, so modifying one bit modifies both sides, keeping the matrix
24 * always symmetric. Iterating over the matrix only iterates over the lower triangle */
25class SymBitMatrix : private bitvec {
26 public:
27 nonconst_bitref operator()(unsigned r, unsigned c) {
28 if (r < c) std::swap(r, c);
29 return bitvec::operator[]((r * r + r) / 2 + c);
30 }
31 bool operator()(unsigned r, unsigned c) const {
32 if (r < c) std::swap(r, c);
33 return bitvec::operator[]((r * r + r) / 2 + c);
34 }
35 unsigned size() const {
36 if (empty()) return 0;
37 unsigned m = *max();
38 unsigned r = 1;
39 while ((r * r + r) / 2 <= m) r++;
40 return r;
41 }
42 using bitvec::clear;
43 using bitvec::empty;
44 using bitvec::operator bool;
45
46 private:
47 using bitvec::operator|=;
48 template <class T>
49 class rowref {
50 friend class SymBitMatrix;
51 T &self;
52 unsigned row;
53 rowref(T &s, unsigned r) : self(s), row(r) {}
54
55 public:
56 rowref(const rowref &) = default;
57 rowref(rowref &&) = default;
58 explicit operator bool() const {
59 if (row < bits_per_unit) {
60 if (self.getrange((row * row + row) / 2, row + 1) != 0) return true;
61 } else {
62 if (self.getslice((row * row + row) / 2, row + 1)) return true;
63 }
64 const auto size = self.size();
65 for (auto c = row + 1; c < size; ++c)
66 if (self(row, c)) return true;
67 return false;
68 }
69 operator bitvec() const {
70 auto rv = self.getslice((row * row + row) / 2, row + 1);
71 const auto size = self.size();
72 for (auto c = row + 1; c < size; ++c)
73 if (self(row, c)) rv[c] = 1;
74 return rv;
75 }
76 };
77 class nonconst_rowref : public rowref<SymBitMatrix> {
78 public:
79 friend class SymBitMatrix;
80 using rowref<SymBitMatrix>::rowref;
81 void operator|=(bitvec a) const {
82 for (size_t v : a) {
83 self(row, v) = 1;
84 }
85 }
86 nonconst_bitref operator[](unsigned col) const { return self(row, col); }
87 };
88 class const_rowref : public rowref<const SymBitMatrix> {
89 public:
90 friend class SymBitMatrix;
91 using rowref<const SymBitMatrix>::rowref;
92 bool operator[](unsigned col) const { return self(row, col); }
93 };
94
95 public:
96 nonconst_rowref operator[](unsigned r) { return nonconst_rowref(*this, r); }
97 const_rowref operator[](unsigned r) const { return const_rowref(*this, r); }
98
99 bool operator==(const SymBitMatrix &a) const { return bitvec::operator==(a); }
100 bool operator!=(const SymBitMatrix &a) const { return bitvec::operator!=(a); }
101 bool operator|=(const SymBitMatrix &a) { return bitvec::operator|=(a); }
102};
103
104#endif /* LIB_SYMBITMATRIX_H_ */
Definition symbitmatrix.h:25
Definition bitvec.h:174
Definition bitvec.h:119