blob: 36c9030041a96c322636985159df60d16097dfee [file] [log] [blame]
// Copyright (c) 2013, 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.
#include "vm/heap/weak_table.h"
#include "platform/assert.h"
#include "vm/raw_object.h"
namespace dart {
intptr_t WeakTable::SizeFor(intptr_t count, intptr_t size) {
intptr_t result = size;
if (count <= (size / 4)) {
// Reduce the capacity.
result = size / 2;
} else {
// Increase the capacity.
result = size * 2;
if (result < size) {
FATAL(
"Reached impossible state of having more weak table entries"
" than memory available for heap objects.");
}
}
if (result < kMinSize) {
result = kMinSize;
}
return result;
}
void WeakTable::SetValueExclusive(ObjectPtr key, intptr_t val) {
const intptr_t mask = size() - 1;
intptr_t idx = Hash(key) & mask;
intptr_t empty_idx = -1;
ObjectPtr obj = ObjectAtExclusive(idx);
while (obj != static_cast<ObjectPtr>(kNoEntry)) {
if (obj == key) {
SetValueAt(idx, val);
return;
} else if ((empty_idx < 0) &&
(static_cast<intptr_t>(obj) == kDeletedEntry)) {
empty_idx = idx; // Insert at this location if not found.
}
idx = (idx + 1) & mask;
obj = ObjectAtExclusive(idx);
}
if (val == 0) {
// Do not enter an invalid value. Associating 0 with a key deletes it from
// this weak table above in SetValueAt. If the key was not present in the
// weak table we are done.
return;
}
if (empty_idx >= 0) {
// We will be reusing a slot below.
set_used(used() - 1);
idx = empty_idx;
}
ASSERT(!IsValidEntryAtExclusive(idx));
// Set the key and value.
SetObjectAt(idx, key);
SetValueAt(idx, val);
// Update the counts.
set_used(used() + 1);
set_count(count() + 1);
// Rehash if needed to ensure that there are empty slots available.
if (used_ >= limit()) {
Rehash();
}
}
bool WeakTable::MarkValueExclusive(ObjectPtr key, intptr_t val) {
const intptr_t mask = size() - 1;
intptr_t idx = Hash(key) & mask;
intptr_t empty_idx = -1;
ObjectPtr obj = ObjectAtExclusive(idx);
while (obj != static_cast<ObjectPtr>(kNoEntry)) {
if (obj == key) {
return false;
} else if ((empty_idx < 0) &&
(static_cast<intptr_t>(obj) == kDeletedEntry)) {
empty_idx = idx; // Insert at this location if not found.
}
idx = (idx + 1) & mask;
obj = ObjectAtExclusive(idx);
}
ASSERT(val != 0);
if (empty_idx >= 0) {
// We will be reusing a slot below.
set_used(used() - 1);
idx = empty_idx;
}
ASSERT(!IsValidEntryAtExclusive(idx));
// Set the key and value.
SetObjectAt(idx, key);
SetValueAt(idx, val);
// Update the counts.
set_used(used() + 1);
set_count(count() + 1);
// Rehash if needed to ensure that there are empty slots available.
if (used_ >= limit()) {
Rehash();
}
return true;
}
void WeakTable::Reset() {
intptr_t* old_data = data_;
used_ = 0;
count_ = 0;
size_ = kMinSize;
free(old_data);
data_ = reinterpret_cast<intptr_t*>(malloc(size_ * kEntrySize * kWordSize));
for (intptr_t i = 0; i < size_; i++) {
data_[ObjectIndex(i)] = kNoEntry;
data_[ValueIndex(i)] = kNoValue;
}
}
void WeakTable::Forward(ObjectPointerVisitor* visitor) {
if (used_ == 0) return;
for (intptr_t i = 0; i < size_; i++) {
if (IsValidEntryAtExclusive(i)) {
visitor->VisitPointer(ObjectPointerAt(i));
}
}
Rehash();
}
void WeakTable::Rehash() {
intptr_t old_size = size();
intptr_t* old_data = data_;
intptr_t new_size = SizeFor(count(), size());
ASSERT(Utils::IsPowerOfTwo(new_size));
intptr_t* new_data =
reinterpret_cast<intptr_t*>(malloc(new_size * kEntrySize * kWordSize));
for (intptr_t i = 0; i < new_size; i++) {
new_data[ObjectIndex(i)] = kNoEntry;
new_data[ValueIndex(i)] = kNoValue;
}
intptr_t mask = new_size - 1;
set_used(0);
for (intptr_t i = 0; i < old_size; i++) {
if (IsValidEntryAtExclusive(i)) {
// Find the new hash location for this entry.
ObjectPtr key = ObjectAtExclusive(i);
intptr_t idx = Hash(key) & mask;
ObjectPtr obj = static_cast<ObjectPtr>(new_data[ObjectIndex(idx)]);
while (obj != static_cast<ObjectPtr>(kNoEntry)) {
ASSERT(obj != key); // Duplicate entry is not expected.
idx = (idx + 1) & mask;
obj = static_cast<ObjectPtr>(new_data[ObjectIndex(idx)]);
}
new_data[ObjectIndex(idx)] = static_cast<intptr_t>(key);
new_data[ValueIndex(idx)] = ValueAtExclusive(i);
set_used(used() + 1);
}
}
// We should only have used valid entries.
ASSERT(used() == count());
// Switch to using the newly allocated backing store.
size_ = new_size;
data_ = new_data;
free(old_data);
}
} // namespace dart