mirror of
https://github.com/dart-lang/sdk
synced 2024-11-02 12:24:24 +00:00
63e6041ca9
TEST=ci Change-Id: Ic6bc784605e10760bb28ea6df34242336a33b4d0 Reviewed-on: https://dart-review.googlesource.com/c/sdk/+/286947 Reviewed-by: Alexander Aprelev <aam@google.com> Commit-Queue: Ryan Macnak <rmacnak@google.com>
110 lines
3.4 KiB
C++
110 lines
3.4 KiB
C++
// 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_
|