Commit e861dbbc authored by tzik's avatar tzik Committed by Commit Bot

Make DetachableVector accessible from builtins

This CL updates DetachableVector to store the data at a known place
instead of in an std::vector<>, so that builtins can update it directly.

Bug: v8:8124
Change-Id: Iba5fb2e9d4e0ddc689d0f7eeaea40bc3218edf3a
Reviewed-on: https://chromium-review.googlesource.com/c/1297783
Commit-Queue: Taiju Tsuiki <tzik@chromium.org>
Reviewed-by: 's avatarPeter Marshall <petermarshall@chromium.org>
Cr-Commit-Position: refs/heads/master@{#57452}
parent fd1b8bbf
...@@ -1955,6 +1955,7 @@ v8_source_set("v8_base") { ...@@ -1955,6 +1955,7 @@ v8_source_set("v8_base") {
"src/deoptimize-reason.h", "src/deoptimize-reason.h",
"src/deoptimizer.cc", "src/deoptimizer.cc",
"src/deoptimizer.h", "src/deoptimizer.h",
"src/detachable-vector.cc",
"src/detachable-vector.h", "src/detachable-vector.h",
"src/disasm.h", "src/disasm.h",
"src/disassembler.cc", "src/disassembler.cc",
......
...@@ -10556,6 +10556,7 @@ void HandleScopeImplementer::IterateThis(RootVisitor* v) { ...@@ -10556,6 +10556,7 @@ void HandleScopeImplementer::IterateThis(RootVisitor* v) {
DetachableVector<Context*>* context_lists[2] = {&saved_contexts_, DetachableVector<Context*>* context_lists[2] = {&saved_contexts_,
&entered_contexts_}; &entered_contexts_};
for (unsigned i = 0; i < arraysize(context_lists); i++) { for (unsigned i = 0; i < arraysize(context_lists); i++) {
context_lists[i]->shrink_to_fit();
if (context_lists[i]->empty()) continue; if (context_lists[i]->empty()) continue;
ObjectSlot start(reinterpret_cast<Address>(&context_lists[i]->front())); ObjectSlot start(reinterpret_cast<Address>(&context_lists[i]->front()));
v->VisitRootPointers(Root::kHandleScope, nullptr, start, v->VisitRootPointers(Root::kHandleScope, nullptr, start,
......
// Copyright 2018 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/detachable-vector.h"
namespace v8 {
namespace internal {
const size_t DetachableVectorBase::kMinimumCapacity = 8;
const size_t DetachableVectorBase::kDataOffset =
offsetof(DetachableVectorBase, data_);
const size_t DetachableVectorBase::kCapacityOffset =
offsetof(DetachableVectorBase, capacity_);
const size_t DetachableVectorBase::kSizeOffset =
offsetof(DetachableVectorBase, size_);
} // namespace internal
} // namespace v8
...@@ -5,65 +5,96 @@ ...@@ -5,65 +5,96 @@
#ifndef V8_DETACHABLE_VECTOR_H_ #ifndef V8_DETACHABLE_VECTOR_H_
#define V8_DETACHABLE_VECTOR_H_ #define V8_DETACHABLE_VECTOR_H_
#include <vector> #include <stddef.h>
#include <algorithm>
#include "src/base/logging.h"
#include "src/base/macros.h"
namespace v8 { namespace v8 {
namespace internal { namespace internal {
// This class wraps a std::vector and provides a few of the common member class V8_EXPORT_PRIVATE DetachableVectorBase {
// functions for accessing the data. It acts as a lazy wrapper of the vector,
// not initiliazing the backing store until push_back() is first called. Two
// extra methods are also provided: free() and detach(), which allow for manual
// control of the backing store. This is currently required for use in the
// HandleScopeImplementer. Any other class should just use a std::vector
// directly.
template <typename T>
class DetachableVector {
public: public:
DetachableVector() : vector_(nullptr) {} // Clear our reference to the backing store. Does not delete it!
void detach() {
data_ = nullptr;
capacity_ = 0;
size_ = 0;
}
void pop_back() { --size_; }
size_t capacity() const { return capacity_; }
size_t size() const { return size_; }
bool empty() const { return size_ == 0; }
~DetachableVector() { delete vector_; } static const size_t kMinimumCapacity;
static const size_t kDataOffset;
static const size_t kCapacityOffset;
static const size_t kSizeOffset;
protected:
void* data_ = nullptr;
size_t capacity_ = 0;
size_t size_ = 0;
};
// This class wraps an array and provides a few of the common member
// functions for accessing the data. Two extra methods are also provided: free()
// and detach(), which allow for manual control of the backing store. This is
// currently required for use in the HandleScopeImplementer. Any other class
// should just use a std::vector.
template <typename T>
class DetachableVector : public DetachableVectorBase {
public:
DetachableVector() = default;
~DetachableVector() { delete[] data(); }
void push_back(const T& value) { void push_back(const T& value) {
ensureAttached(); if (size_ == capacity_) {
vector_->push_back(value); size_t new_capacity = std::max(kMinimumCapacity, 2 * capacity_);
Resize(new_capacity);
}
data()[size_] = value;
++size_;
} }
// Free the backing store and clear our reference to it. // Free the backing store and clear our reference to it.
void free() { void free() {
delete vector_; delete[] data();
vector_ = nullptr; data_ = nullptr;
capacity_ = 0;
size_ = 0;
} }
// Clear our reference to the backing store. Does not delete it! T& at(size_t i) const {
void detach() { vector_ = nullptr; } DCHECK_LT(i, size_);
return data()[i];
T& at(typename std::vector<T>::size_type i) const { return vector_->at(i); }
T& back() const { return vector_->back(); }
T& front() const { return vector_->front(); }
void pop_back() { vector_->pop_back(); }
typename std::vector<T>::size_type size() const {
if (vector_) return vector_->size();
return 0;
} }
T& back() const { return at(size_ - 1); }
T& front() const { return at(0); }
bool empty() const { void shrink_to_fit() {
if (vector_) return vector_->empty(); size_t new_capacity = std::max(size_, kMinimumCapacity);
return true; if (new_capacity < capacity_ / 2) {
Resize(new_capacity);
}
} }
private: private:
std::vector<T>* vector_; T* data() const { return static_cast<T*>(data_); }
// Attach a vector backing store if not present. void Resize(size_t new_capacity) {
void ensureAttached() { DCHECK_LE(size_, new_capacity);
if (vector_ == nullptr) { T* new_data_ = new T[new_capacity];
vector_ = new std::vector<T>();
} std::copy(data(), data() + size_, new_data_);
delete[] data();
data_ = new_data_;
capacity_ = new_capacity;
} }
}; };
......
...@@ -63,5 +63,65 @@ TEST(DetachableVector, DetachLeaksBackingStore) { ...@@ -63,5 +63,65 @@ TEST(DetachableVector, DetachLeaksBackingStore) {
// The destructor of v2 will release the backing store. // The destructor of v2 will release the backing store.
} }
TEST(DetachableVector, PushAndPopWithReallocation) {
DetachableVector<size_t> v;
const size_t kMinimumCapacity = DetachableVector<size_t>::kMinimumCapacity;
EXPECT_EQ(0u, v.capacity());
EXPECT_EQ(0u, v.size());
v.push_back(0);
EXPECT_EQ(kMinimumCapacity, v.capacity());
EXPECT_EQ(1u, v.size());
// Push values until the reallocation happens.
for (size_t i = 1; i <= kMinimumCapacity; ++i) {
v.push_back(i);
}
EXPECT_EQ(2 * kMinimumCapacity, v.capacity());
EXPECT_EQ(kMinimumCapacity + 1, v.size());
EXPECT_EQ(kMinimumCapacity, v.back());
v.pop_back();
v.push_back(100);
EXPECT_EQ(100u, v.back());
v.pop_back();
EXPECT_EQ(kMinimumCapacity - 1, v.back());
}
TEST(DetachableVector, ShrinkToFit) {
DetachableVector<size_t> v;
const size_t kMinimumCapacity = DetachableVector<size_t>::kMinimumCapacity;
// shrink_to_fit doesn't affect the empty capacity DetachableVector.
EXPECT_EQ(0u, v.capacity());
v.shrink_to_fit();
EXPECT_EQ(0u, v.capacity());
// Do not shrink the buffer if it's smaller than kMinimumCapacity.
v.push_back(0);
EXPECT_EQ(kMinimumCapacity, v.capacity());
v.shrink_to_fit();
EXPECT_EQ(kMinimumCapacity, v.capacity());
// Fill items to |v| until the buffer grows twice.
for (size_t i = 0; i < 2 * kMinimumCapacity; ++i) {
v.push_back(i);
}
EXPECT_EQ(2 * kMinimumCapacity + 1, v.size());
EXPECT_EQ(4 * kMinimumCapacity, v.capacity());
// Do not shrink the buffer if the number of unused slots is not large enough.
v.shrink_to_fit();
EXPECT_EQ(2 * kMinimumCapacity + 1, v.size());
EXPECT_EQ(4 * kMinimumCapacity, v.capacity());
v.pop_back();
v.pop_back();
v.shrink_to_fit();
EXPECT_EQ(2 * kMinimumCapacity - 1, v.size());
EXPECT_EQ(2 * kMinimumCapacity - 1, v.capacity());
}
} // namespace internal } // namespace internal
} // namespace v8 } // namespace v8
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment