1 // Copyright 2012 the V8 project authors. All rights reserved. 2 // Use of this source code is governed by a BSD-style license that can be 3 // found in the LICENSE file. 4 5 #ifndef V8_UTILS_BIT_VECTOR_H_ 6 #define V8_UTILS_BIT_VECTOR_H_ 7 8 #include "src/base/bits.h" 9 #include "src/utils/allocation.h" 10 #include "src/zone/zone.h" 11 12 namespace v8 { 13 namespace internal { 14 15 class V8_EXPORT_PRIVATE BitVector : public ZoneObject { 16 public: 17 union DataStorage { 18 uintptr_t* ptr_; // valid if data_length_ > 1 19 uintptr_t inline_; // valid if data_length_ == 1 20 DataStorage(uintptr_t value)21 explicit DataStorage(uintptr_t value) : inline_(value) {} 22 }; 23 24 // Iterator for the elements of this BitVector. 25 class Iterator { 26 public: 27 V8_EXPORT_PRIVATE inline void operator++() { 28 current_++; 29 30 // Skip zeroed words. 31 while (current_value_ == 0) { 32 current_index_++; 33 if (Done()) return; 34 DCHECK(!target_->is_inline()); 35 current_value_ = target_->data_.ptr_[current_index_]; 36 current_ = current_index_ << kDataBitShift; 37 } 38 39 // Skip zeroed bits. 40 uintptr_t trailing_zeros = base::bits::CountTrailingZeros(current_value_); 41 current_ += trailing_zeros; 42 current_value_ >>= trailing_zeros; 43 44 // Get current_value ready for next advance. 45 current_value_ >>= 1; 46 } 47 48 int operator*() const { 49 DCHECK(!Done()); 50 return current_; 51 } 52 53 bool operator!=(const Iterator& other) const { 54 // "other" is required to be the end sentinel value, to avoid us needing 55 // to compare exact "current" values. 56 DCHECK(other.Done()); 57 DCHECK_EQ(target_, other.target_); 58 return current_index_ != other.current_index_; 59 } 60 61 private: 62 static constexpr struct StartTag { 63 } kStartTag = {}; 64 static constexpr struct EndTag { 65 } kEndTag = {}; 66 Iterator(const BitVector * target,StartTag)67 explicit Iterator(const BitVector* target, StartTag) 68 : target_(target), 69 current_value_(target->is_inline() ? target->data_.inline_ 70 : target->data_.ptr_[0]), 71 current_index_(0), 72 current_(-1) { 73 ++(*this); 74 } Iterator(const BitVector * target,EndTag)75 explicit Iterator(const BitVector* target, EndTag) 76 : target_(target), 77 current_value_(0), 78 current_index_(target->data_length_), 79 current_(-1) { 80 DCHECK(Done()); 81 } 82 Done()83 bool Done() const { return current_index_ >= target_->data_length_; } 84 85 const BitVector* target_; 86 uintptr_t current_value_; 87 int current_index_; 88 int current_; 89 90 friend class BitVector; 91 }; 92 93 static const int kDataLengthForInline = 1; 94 static const int kDataBits = kBitsPerSystemPointer; 95 static const int kDataBitShift = kBitsPerSystemPointerLog2; 96 static const uintptr_t kOne = 1; // This saves some static_casts. 97 BitVector()98 BitVector() : length_(0), data_length_(kDataLengthForInline), data_(0) {} 99 BitVector(int length,Zone * zone)100 BitVector(int length, Zone* zone) 101 : length_(length), data_length_(SizeFor(length)), data_(0) { 102 DCHECK_LE(0, length); 103 if (!is_inline()) { 104 data_.ptr_ = zone->NewArray<uintptr_t>(data_length_); 105 Clear(); 106 } 107 // Otherwise, clearing is implicit 108 } 109 BitVector(const BitVector & other,Zone * zone)110 BitVector(const BitVector& other, Zone* zone) 111 : length_(other.length_), 112 data_length_(other.data_length_), 113 data_(other.data_.inline_) { 114 if (!is_inline()) { 115 data_.ptr_ = zone->NewArray<uintptr_t>(data_length_); 116 for (int i = 0; i < other.data_length_; i++) { 117 data_.ptr_[i] = other.data_.ptr_[i]; 118 } 119 } 120 } 121 SizeFor(int length)122 static int SizeFor(int length) { 123 if (length <= kDataBits) { 124 return kDataLengthForInline; 125 } 126 127 int data_length = 1 + ((length - 1) / kDataBits); 128 DCHECK_GT(data_length, kDataLengthForInline); 129 return data_length; 130 } 131 CopyFrom(const BitVector & other)132 void CopyFrom(const BitVector& other) { 133 DCHECK_EQ(other.length(), length()); 134 if (is_inline()) { 135 DCHECK(other.is_inline()); 136 data_.inline_ = other.data_.inline_; 137 } else { 138 for (int i = 0; i < data_length_; i++) { 139 data_.ptr_[i] = other.data_.ptr_[i]; 140 } 141 } 142 } 143 Resize(int new_length,Zone * zone)144 void Resize(int new_length, Zone* zone) { 145 DCHECK_GT(new_length, length()); 146 int new_data_length = SizeFor(new_length); 147 if (new_data_length > data_length_) { 148 DataStorage old_data = data_; 149 int old_data_length = data_length_; 150 151 // Make sure the new data length is large enough to need allocation. 152 DCHECK_GT(new_data_length, kDataLengthForInline); 153 data_.ptr_ = zone->NewArray<uintptr_t>(new_data_length); 154 data_length_ = new_data_length; 155 156 // Copy over the data. 157 if (old_data_length == kDataLengthForInline) { 158 data_.ptr_[0] = old_data.inline_; 159 } else { 160 for (int i = 0; i < old_data_length; i++) { 161 data_.ptr_[i] = old_data.ptr_[i]; 162 } 163 } 164 // Zero out the rest of the data. 165 for (int i = old_data_length; i < data_length_; i++) { 166 data_.ptr_[i] = 0; 167 } 168 } 169 length_ = new_length; 170 } 171 Contains(int i)172 bool Contains(int i) const { 173 DCHECK(i >= 0 && i < length()); 174 uintptr_t block = is_inline() ? data_.inline_ : data_.ptr_[i / kDataBits]; 175 return (block & (kOne << (i % kDataBits))) != 0; 176 } 177 Add(int i)178 void Add(int i) { 179 DCHECK(i >= 0 && i < length()); 180 if (is_inline()) { 181 data_.inline_ |= (kOne << i); 182 } else { 183 data_.ptr_[i / kDataBits] |= (kOne << (i % kDataBits)); 184 } 185 } 186 AddAll()187 void AddAll() { 188 // TODO(leszeks): This sets bits outside of the length of this bit-vector, 189 // which is observable if we resize it or copy from it. If this is a 190 // problem, we should clear the high bits either on add, or on resize/copy. 191 if (is_inline()) { 192 data_.inline_ = -1; 193 } else { 194 memset(data_.ptr_, -1, sizeof(uintptr_t) * data_length_); 195 } 196 } 197 Remove(int i)198 void Remove(int i) { 199 DCHECK(i >= 0 && i < length()); 200 if (is_inline()) { 201 data_.inline_ &= ~(kOne << i); 202 } else { 203 data_.ptr_[i / kDataBits] &= ~(kOne << (i % kDataBits)); 204 } 205 } 206 Union(const BitVector & other)207 void Union(const BitVector& other) { 208 DCHECK(other.length() == length()); 209 if (is_inline()) { 210 DCHECK(other.is_inline()); 211 data_.inline_ |= other.data_.inline_; 212 } else { 213 for (int i = 0; i < data_length_; i++) { 214 data_.ptr_[i] |= other.data_.ptr_[i]; 215 } 216 } 217 } 218 UnionIsChanged(const BitVector & other)219 bool UnionIsChanged(const BitVector& other) { 220 DCHECK(other.length() == length()); 221 if (is_inline()) { 222 DCHECK(other.is_inline()); 223 uintptr_t old_data = data_.inline_; 224 data_.inline_ |= other.data_.inline_; 225 return data_.inline_ != old_data; 226 } else { 227 bool changed = false; 228 for (int i = 0; i < data_length_; i++) { 229 uintptr_t old_data = data_.ptr_[i]; 230 data_.ptr_[i] |= other.data_.ptr_[i]; 231 if (data_.ptr_[i] != old_data) changed = true; 232 } 233 return changed; 234 } 235 } 236 Intersect(const BitVector & other)237 void Intersect(const BitVector& other) { 238 DCHECK(other.length() == length()); 239 if (is_inline()) { 240 DCHECK(other.is_inline()); 241 data_.inline_ &= other.data_.inline_; 242 } else { 243 for (int i = 0; i < data_length_; i++) { 244 data_.ptr_[i] &= other.data_.ptr_[i]; 245 } 246 } 247 } 248 IntersectIsChanged(const BitVector & other)249 bool IntersectIsChanged(const BitVector& other) { 250 DCHECK(other.length() == length()); 251 if (is_inline()) { 252 DCHECK(other.is_inline()); 253 uintptr_t old_data = data_.inline_; 254 data_.inline_ &= other.data_.inline_; 255 return data_.inline_ != old_data; 256 } else { 257 bool changed = false; 258 for (int i = 0; i < data_length_; i++) { 259 uintptr_t old_data = data_.ptr_[i]; 260 data_.ptr_[i] &= other.data_.ptr_[i]; 261 if (data_.ptr_[i] != old_data) changed = true; 262 } 263 return changed; 264 } 265 } 266 Subtract(const BitVector & other)267 void Subtract(const BitVector& other) { 268 DCHECK(other.length() == length()); 269 if (is_inline()) { 270 DCHECK(other.is_inline()); 271 data_.inline_ &= ~other.data_.inline_; 272 } else { 273 for (int i = 0; i < data_length_; i++) { 274 data_.ptr_[i] &= ~other.data_.ptr_[i]; 275 } 276 } 277 } 278 Clear()279 void Clear() { 280 if (is_inline()) { 281 data_.inline_ = 0; 282 } else { 283 for (int i = 0; i < data_length_; i++) { 284 data_.ptr_[i] = 0; 285 } 286 } 287 } 288 IsEmpty()289 bool IsEmpty() const { 290 if (is_inline()) { 291 return data_.inline_ == 0; 292 } else { 293 for (int i = 0; i < data_length_; i++) { 294 if (data_.ptr_[i] != 0) return false; 295 } 296 return true; 297 } 298 } 299 Equals(const BitVector & other)300 bool Equals(const BitVector& other) const { 301 DCHECK(other.length() == length()); 302 if (is_inline()) { 303 DCHECK(other.is_inline()); 304 return data_.inline_ == other.data_.inline_; 305 } else { 306 for (int i = 0; i < data_length_; i++) { 307 if (data_.ptr_[i] != other.data_.ptr_[i]) return false; 308 } 309 return true; 310 } 311 } 312 313 int Count() const; 314 length()315 int length() const { return length_; } 316 begin()317 Iterator begin() const { return Iterator(this, Iterator::kStartTag); } 318 end()319 Iterator end() const { return Iterator(this, Iterator::kEndTag); } 320 321 #ifdef DEBUG 322 void Print() const; 323 #endif 324 325 MOVE_ONLY_NO_DEFAULT_CONSTRUCTOR(BitVector); 326 327 private: 328 int length_; 329 int data_length_; 330 DataStorage data_; 331 is_inline()332 bool is_inline() const { return data_length_ == kDataLengthForInline; } 333 }; 334 335 class GrowableBitVector { 336 public: GrowableBitVector()337 GrowableBitVector() : bits_() {} GrowableBitVector(int length,Zone * zone)338 GrowableBitVector(int length, Zone* zone) : bits_(length, zone) {} 339 Contains(int value)340 bool Contains(int value) const { 341 if (!InBitsRange(value)) return false; 342 return bits_.Contains(value); 343 } 344 Add(int value,Zone * zone)345 void Add(int value, Zone* zone) { 346 EnsureCapacity(value, zone); 347 bits_.Add(value); 348 } 349 Clear()350 void Clear() { bits_.Clear(); } 351 length()352 int length() const { return bits_.length(); } 353 Equals(const GrowableBitVector & other)354 bool Equals(const GrowableBitVector& other) const { 355 return length() == other.length() && bits_.Equals(other.bits_); 356 } 357 begin()358 BitVector::Iterator begin() const { return bits_.begin(); } 359 end()360 BitVector::Iterator end() const { return bits_.end(); } 361 362 private: 363 static constexpr int kInitialLength = 1024; 364 InBitsRange(int value)365 bool InBitsRange(int value) const { return bits_.length() > value; } 366 EnsureCapacity(int value,Zone * zone)367 void EnsureCapacity(int value, Zone* zone) { 368 if (InBitsRange(value)) return; 369 int new_length = 370 base::bits::RoundUpToPowerOfTwo32(static_cast<uint32_t>(value)); 371 new_length = std::min(new_length, kInitialLength); 372 while (new_length <= value) new_length *= 2; 373 bits_.Resize(new_length, zone); 374 } 375 376 BitVector bits_; 377 }; 378 379 } // namespace internal 380 } // namespace v8 381 382 #endif // V8_UTILS_BIT_VECTOR_H_ 383