| // Copyright (c) 2012, the Dart project authors.  Please see the AUTHORS file | 
 | // for details. All rights reserved. Use of this source code is governed by a | 
 | // BSD-style license that can be found in the LICENSE file. | 
 |  | 
 | #ifndef RUNTIME_VM_BITMAP_H_ | 
 | #define RUNTIME_VM_BITMAP_H_ | 
 |  | 
 | #include "vm/allocation.h" | 
 | #include "vm/datastream.h" | 
 | #include "vm/thread_state.h" | 
 | #include "vm/zone.h" | 
 |  | 
 | namespace dart { | 
 |  | 
 | // BitmapBuilder is used to build a bitmap. The implementation is optimized | 
 | // for a dense set of small bit maps without a fixed upper bound (e.g: a | 
 | // pointer map description of a stack). | 
 | class BitmapBuilder : public ZoneAllocated { | 
 |  public: | 
 |   BitmapBuilder() : length_(0), data_size_in_bytes_(kInlineCapacityInBytes) { | 
 |     memset(data_.inline_, 0, data_size_in_bytes_); | 
 |   } | 
 |  | 
 |   BitmapBuilder(const BitmapBuilder& other) | 
 |       : ZoneAllocated(), | 
 |         length_(other.length_), | 
 |         data_size_in_bytes_(other.data_size_in_bytes_) { | 
 |     if (data_size_in_bytes_ == kInlineCapacityInBytes) { | 
 |       memmove(data_.inline_, other.data_.inline_, kInlineCapacityInBytes); | 
 |     } else { | 
 |       data_.ptr_ = AllocBackingStore(data_size_in_bytes_); | 
 |       memmove(data_.ptr_, other.data_.ptr_, data_size_in_bytes_); | 
 |     } | 
 |   } | 
 |  | 
 |   intptr_t Length() const { return length_; } | 
 |   void SetLength(intptr_t length); | 
 |  | 
 |   // Get/Set individual bits in the bitmap, setting bits beyond the bitmap's | 
 |   // length increases the length and expands the underlying bitmap if | 
 |   // needed. | 
 |   bool Get(intptr_t bit_offset) const; | 
 |   void Set(intptr_t bit_offset, bool value); | 
 |  | 
 |   // Return the bit offset of the highest bit set. | 
 |   intptr_t Maximum() const; | 
 |  | 
 |   // Return the bit offset of the lowest bit set. | 
 |   intptr_t Minimum() const; | 
 |  | 
 |   // Sets min..max (inclusive) to value. | 
 |   void SetRange(intptr_t min, intptr_t max, bool value); | 
 |  | 
 |   void Print() const; | 
 |   void AppendAsBytesTo(BaseWriteStream* stream) const; | 
 |  | 
 |   void Write(BaseWriteStream* stream) const; | 
 |   void Read(ReadStream* stream); | 
 |  | 
 |  private: | 
 |   static constexpr intptr_t kIncrementSizeInBytes = 16; | 
 |   static constexpr intptr_t kInlineCapacityInBytes = 16; | 
 |  | 
 |   bool InRange(intptr_t offset) const { | 
 |     if (offset < 0) { | 
 |       FATAL( | 
 |           "Fatal error in BitmapBuilder::InRange :" | 
 |           " invalid bit_offset, %" Pd "\n", | 
 |           offset); | 
 |     } | 
 |     return (offset < length_); | 
 |   } | 
 |  | 
 |   bool InBackingStore(intptr_t bit_offset) { | 
 |     intptr_t byte_offset = bit_offset >> kBitsPerByteLog2; | 
 |     return byte_offset < data_size_in_bytes_; | 
 |   } | 
 |  | 
 |   uint8_t* BackingStore() { | 
 |     return data_size_in_bytes_ == kInlineCapacityInBytes ? &data_.inline_[0] | 
 |                                                          : data_.ptr_; | 
 |   } | 
 |  | 
 |   const uint8_t* BackingStore() const { | 
 |     return data_size_in_bytes_ == kInlineCapacityInBytes ? &data_.inline_[0] | 
 |                                                          : data_.ptr_; | 
 |   } | 
 |  | 
 |   static uint8_t* AllocBackingStore(intptr_t size_in_bytes) { | 
 |     return ThreadState::Current()->zone()->Alloc<uint8_t>(size_in_bytes); | 
 |   } | 
 |  | 
 |   // Get/Set a bit that is known to be covered by the backing store. | 
 |   bool GetBit(intptr_t bit_offset) const; | 
 |   void SetBit(intptr_t bit_offset, bool value); | 
 |  | 
 |   intptr_t length_; | 
 |  | 
 |   // Backing store for the bitmap.  Reading bits beyond the backing store | 
 |   // (up to length_) is allowed and they are assumed to be false. | 
 |   intptr_t data_size_in_bytes_; | 
 |   union { | 
 |     uint8_t* ptr_; | 
 |     uint8_t inline_[kInlineCapacityInBytes]; | 
 |   } data_; | 
 | }; | 
 |  | 
 | }  // namespace dart | 
 |  | 
 | #endif  // RUNTIME_VM_BITMAP_H_ |