xref: /aosp_15_r20/art/libartbase/base/bit_vector-inl.h (revision 795d594fd825385562da6b089ea9b2033f3abf5a)
1*795d594fSAndroid Build Coastguard Worker /*
2*795d594fSAndroid Build Coastguard Worker  * Copyright (C) 2013 The Android Open Source Project
3*795d594fSAndroid Build Coastguard Worker  *
4*795d594fSAndroid Build Coastguard Worker  * Licensed under the Apache License, Version 2.0 (the "License");
5*795d594fSAndroid Build Coastguard Worker  * you may not use this file except in compliance with the License.
6*795d594fSAndroid Build Coastguard Worker  * You may obtain a copy of the License at
7*795d594fSAndroid Build Coastguard Worker  *
8*795d594fSAndroid Build Coastguard Worker  *      http://www.apache.org/licenses/LICENSE-2.0
9*795d594fSAndroid Build Coastguard Worker  *
10*795d594fSAndroid Build Coastguard Worker  * Unless required by applicable law or agreed to in writing, software
11*795d594fSAndroid Build Coastguard Worker  * distributed under the License is distributed on an "AS IS" BASIS,
12*795d594fSAndroid Build Coastguard Worker  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*795d594fSAndroid Build Coastguard Worker  * See the License for the specific language governing permissions and
14*795d594fSAndroid Build Coastguard Worker  * limitations under the License.
15*795d594fSAndroid Build Coastguard Worker  */
16*795d594fSAndroid Build Coastguard Worker 
17*795d594fSAndroid Build Coastguard Worker #ifndef ART_LIBARTBASE_BASE_BIT_VECTOR_INL_H_
18*795d594fSAndroid Build Coastguard Worker #define ART_LIBARTBASE_BASE_BIT_VECTOR_INL_H_
19*795d594fSAndroid Build Coastguard Worker 
20*795d594fSAndroid Build Coastguard Worker #include "bit_vector.h"
21*795d594fSAndroid Build Coastguard Worker 
22*795d594fSAndroid Build Coastguard Worker #include <android-base/logging.h>
23*795d594fSAndroid Build Coastguard Worker 
24*795d594fSAndroid Build Coastguard Worker #include "bit_utils.h"
25*795d594fSAndroid Build Coastguard Worker 
26*795d594fSAndroid Build Coastguard Worker namespace art {
27*795d594fSAndroid Build Coastguard Worker 
28*795d594fSAndroid Build Coastguard Worker inline bool BitVector::IndexIterator::operator==(const IndexIterator& other) const {
29*795d594fSAndroid Build Coastguard Worker   DCHECK(bit_storage_ == other.bit_storage_);
30*795d594fSAndroid Build Coastguard Worker   DCHECK_EQ(storage_size_, other.storage_size_);
31*795d594fSAndroid Build Coastguard Worker   return bit_index_ == other.bit_index_;
32*795d594fSAndroid Build Coastguard Worker }
33*795d594fSAndroid Build Coastguard Worker 
34*795d594fSAndroid Build Coastguard Worker inline uint32_t BitVector::IndexIterator::operator*() const {
35*795d594fSAndroid Build Coastguard Worker   DCHECK_LT(bit_index_, BitSize());
36*795d594fSAndroid Build Coastguard Worker   return bit_index_;
37*795d594fSAndroid Build Coastguard Worker }
38*795d594fSAndroid Build Coastguard Worker 
39*795d594fSAndroid Build Coastguard Worker inline BitVector::IndexIterator& BitVector::IndexIterator::operator++() {
40*795d594fSAndroid Build Coastguard Worker   DCHECK_LT(bit_index_, BitSize());
41*795d594fSAndroid Build Coastguard Worker   bit_index_ = FindIndex(bit_index_ + 1u);
42*795d594fSAndroid Build Coastguard Worker   return *this;
43*795d594fSAndroid Build Coastguard Worker }
44*795d594fSAndroid Build Coastguard Worker 
45*795d594fSAndroid Build Coastguard Worker inline BitVector::IndexIterator BitVector::IndexIterator::operator++(int) {
46*795d594fSAndroid Build Coastguard Worker   IndexIterator result(*this);
47*795d594fSAndroid Build Coastguard Worker   ++*this;
48*795d594fSAndroid Build Coastguard Worker   return result;
49*795d594fSAndroid Build Coastguard Worker }
50*795d594fSAndroid Build Coastguard Worker 
FindIndex(uint32_t start_index)51*795d594fSAndroid Build Coastguard Worker inline uint32_t BitVector::IndexIterator::FindIndex(uint32_t start_index) const {
52*795d594fSAndroid Build Coastguard Worker   DCHECK_LE(start_index, BitSize());
53*795d594fSAndroid Build Coastguard Worker   uint32_t word_index = start_index / kWordBits;
54*795d594fSAndroid Build Coastguard Worker   if (UNLIKELY(word_index == storage_size_)) {
55*795d594fSAndroid Build Coastguard Worker     return start_index;
56*795d594fSAndroid Build Coastguard Worker   }
57*795d594fSAndroid Build Coastguard Worker   uint32_t word = bit_storage_[word_index];
58*795d594fSAndroid Build Coastguard Worker   // Mask out any bits in the first word we've already considered.
59*795d594fSAndroid Build Coastguard Worker   word &= static_cast<uint32_t>(-1) << (start_index & 0x1f);
60*795d594fSAndroid Build Coastguard Worker   while (word == 0u) {
61*795d594fSAndroid Build Coastguard Worker     ++word_index;
62*795d594fSAndroid Build Coastguard Worker     if (UNLIKELY(word_index == storage_size_)) {
63*795d594fSAndroid Build Coastguard Worker       return BitSize();
64*795d594fSAndroid Build Coastguard Worker     }
65*795d594fSAndroid Build Coastguard Worker     word = bit_storage_[word_index];
66*795d594fSAndroid Build Coastguard Worker   }
67*795d594fSAndroid Build Coastguard Worker   return word_index * 32u + CTZ(word);
68*795d594fSAndroid Build Coastguard Worker }
69*795d594fSAndroid Build Coastguard Worker 
IndexIterator(const BitVector * bit_vector,begin_tag)70*795d594fSAndroid Build Coastguard Worker inline BitVector::IndexIterator::IndexIterator(const BitVector* bit_vector, begin_tag)
71*795d594fSAndroid Build Coastguard Worker   : bit_storage_(bit_vector->GetRawStorage()),
72*795d594fSAndroid Build Coastguard Worker     storage_size_(bit_vector->storage_size_),
73*795d594fSAndroid Build Coastguard Worker     bit_index_(FindIndex(0u)) { }
74*795d594fSAndroid Build Coastguard Worker 
IndexIterator(const BitVector * bit_vector,end_tag)75*795d594fSAndroid Build Coastguard Worker inline BitVector::IndexIterator::IndexIterator(const BitVector* bit_vector, end_tag)
76*795d594fSAndroid Build Coastguard Worker   : bit_storage_(bit_vector->GetRawStorage()),
77*795d594fSAndroid Build Coastguard Worker     storage_size_(bit_vector->storage_size_),
78*795d594fSAndroid Build Coastguard Worker     bit_index_(BitSize()) { }
79*795d594fSAndroid Build Coastguard Worker 
begin()80*795d594fSAndroid Build Coastguard Worker inline BitVector::IndexIterator BitVector::IndexContainer::begin() const {
81*795d594fSAndroid Build Coastguard Worker   return IndexIterator(bit_vector_, IndexIterator::begin_tag());
82*795d594fSAndroid Build Coastguard Worker }
83*795d594fSAndroid Build Coastguard Worker 
end()84*795d594fSAndroid Build Coastguard Worker inline BitVector::IndexIterator BitVector::IndexContainer::end() const {
85*795d594fSAndroid Build Coastguard Worker   return IndexIterator(bit_vector_, IndexIterator::end_tag());
86*795d594fSAndroid Build Coastguard Worker }
87*795d594fSAndroid Build Coastguard Worker 
ClearAllBits()88*795d594fSAndroid Build Coastguard Worker inline void BitVector::ClearAllBits() {
89*795d594fSAndroid Build Coastguard Worker   memset(storage_, 0, storage_size_ * kWordBytes);
90*795d594fSAndroid Build Coastguard Worker }
91*795d594fSAndroid Build Coastguard Worker 
Equal(const BitVector * src)92*795d594fSAndroid Build Coastguard Worker inline bool BitVector::Equal(const BitVector* src) const {
93*795d594fSAndroid Build Coastguard Worker   return (storage_size_ == src->GetStorageSize()) &&
94*795d594fSAndroid Build Coastguard Worker     (expandable_ == src->IsExpandable()) &&
95*795d594fSAndroid Build Coastguard Worker     (memcmp(storage_, src->GetRawStorage(), storage_size_ * sizeof(uint32_t)) == 0);
96*795d594fSAndroid Build Coastguard Worker }
97*795d594fSAndroid Build Coastguard Worker 
98*795d594fSAndroid Build Coastguard Worker }  // namespace art
99*795d594fSAndroid Build Coastguard Worker 
100*795d594fSAndroid Build Coastguard Worker #endif  // ART_LIBARTBASE_BASE_BIT_VECTOR_INL_H_
101