blob: 150e3fa1597c778cac32f9d49334109fb77f4fd8 [file] [log] [blame]
// Copyright 2020 the V8 project authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "src/heap/index-generator.h"
namespace v8 {
namespace internal {
IndexGenerator::IndexGenerator(size_t size) : size_(size) {
if (size == 0) return;
base::MutexGuard guard(&lock_);
pending_indices_.push(0);
ranges_to_split_.push({0, size_});
}
base::Optional<size_t> IndexGenerator::GetNext() {
base::MutexGuard guard(&lock_);
if (!pending_indices_.empty()) {
// Return any pending index first.
auto index = pending_indices_.top();
pending_indices_.pop();
return index;
}
if (ranges_to_split_.empty()) return base::nullopt;
// Split the oldest running range in 2 and return the middle index as
// starting point.
auto range = ranges_to_split_.front();
ranges_to_split_.pop();
size_t size = range.second - range.first;
size_t mid = range.first + size / 2;
// Both sides of the range are added to |ranges_to_split_| so they may be
// further split if possible.
if (mid - range.first > 1) ranges_to_split_.push({range.first, mid});
if (range.second - mid > 1) ranges_to_split_.push({mid, range.second});
return mid;
}
void IndexGenerator::GiveBack(size_t index) {
base::MutexGuard guard(&lock_);
// Add |index| to pending indices so GetNext() may return it before anything
// else.
pending_indices_.push(index);
}
} // namespace internal
} // namespace v8