2021-04-14 15:49:54 +00:00
|
|
|
// 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 <algorithm>
|
|
|
|
#include <iterator>
|
|
|
|
#include <numeric>
|
|
|
|
|
|
|
|
#include "include/cppgc/allocation.h"
|
|
|
|
#include "include/cppgc/heap-consistency.h"
|
|
|
|
#include "include/cppgc/persistent.h"
|
|
|
|
#include "include/cppgc/prefinalizer.h"
|
|
|
|
#include "src/heap/cppgc/globals.h"
|
|
|
|
#include "src/heap/cppgc/heap-visitor.h"
|
|
|
|
#include "src/heap/cppgc/heap.h"
|
2021-05-17 10:23:17 +00:00
|
|
|
#include "src/heap/cppgc/object-view.h"
|
2021-04-14 15:49:54 +00:00
|
|
|
#include "test/unittests/heap/cppgc/tests.h"
|
|
|
|
#include "testing/gtest/include/gtest/gtest.h"
|
|
|
|
|
|
|
|
namespace cppgc {
|
|
|
|
namespace internal {
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
class WorkloadsTest : public testing::TestWithHeap {
|
|
|
|
public:
|
|
|
|
void ConservativeGC() {
|
|
|
|
internal::Heap::From(GetHeap())->CollectGarbage(
|
|
|
|
Heap::Config::ConservativeAtomicConfig());
|
|
|
|
}
|
|
|
|
void PreciseGC() {
|
|
|
|
internal::Heap::From(GetHeap())->CollectGarbage(
|
|
|
|
Heap::Config::PreciseAtomicConfig());
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
class SuperClass;
|
|
|
|
|
|
|
|
class PointsBack final : public GarbageCollected<PointsBack> {
|
|
|
|
public:
|
|
|
|
PointsBack() { ++alive_count_; }
|
|
|
|
~PointsBack() { --alive_count_; }
|
|
|
|
|
|
|
|
void SetBackPointer(SuperClass* back_pointer) {
|
|
|
|
back_pointer_ = back_pointer;
|
|
|
|
}
|
|
|
|
|
|
|
|
SuperClass* BackPointer() const { return back_pointer_; }
|
|
|
|
|
|
|
|
void Trace(Visitor* visitor) const { visitor->Trace(back_pointer_); }
|
|
|
|
|
|
|
|
static int alive_count_;
|
|
|
|
|
|
|
|
private:
|
|
|
|
WeakMember<SuperClass> back_pointer_;
|
|
|
|
};
|
|
|
|
int PointsBack::alive_count_ = 0;
|
|
|
|
|
|
|
|
class SuperClass : public GarbageCollected<SuperClass> {
|
|
|
|
public:
|
|
|
|
explicit SuperClass(PointsBack* points_back) : points_back_(points_back) {
|
|
|
|
points_back_->SetBackPointer(this);
|
|
|
|
++alive_count_;
|
|
|
|
}
|
|
|
|
virtual ~SuperClass() { --alive_count_; }
|
|
|
|
|
|
|
|
void InvokeConservativeGCAndExpect(WorkloadsTest* test, SuperClass* target,
|
|
|
|
PointsBack* points_back,
|
|
|
|
int super_class_count) {
|
|
|
|
test->ConservativeGC();
|
|
|
|
EXPECT_EQ(points_back, target->GetPointsBack());
|
|
|
|
EXPECT_EQ(super_class_count, SuperClass::alive_count_);
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void Trace(Visitor* visitor) const { visitor->Trace(points_back_); }
|
|
|
|
|
|
|
|
PointsBack* GetPointsBack() const { return points_back_.Get(); }
|
|
|
|
|
|
|
|
static int alive_count_;
|
|
|
|
|
|
|
|
private:
|
|
|
|
Member<PointsBack> points_back_;
|
|
|
|
};
|
|
|
|
int SuperClass::alive_count_ = 0;
|
|
|
|
|
|
|
|
class SubData final : public GarbageCollected<SubData> {
|
|
|
|
public:
|
|
|
|
SubData() { ++alive_count_; }
|
|
|
|
~SubData() { --alive_count_; }
|
|
|
|
|
|
|
|
void Trace(Visitor* visitor) const {}
|
|
|
|
|
|
|
|
static int alive_count_;
|
|
|
|
};
|
|
|
|
int SubData::alive_count_ = 0;
|
|
|
|
|
|
|
|
class SubClass final : public SuperClass {
|
|
|
|
public:
|
|
|
|
explicit SubClass(AllocationHandle& allocation_handle,
|
|
|
|
PointsBack* points_back)
|
|
|
|
: SuperClass(points_back),
|
|
|
|
data_(MakeGarbageCollected<SubData>(allocation_handle)) {
|
|
|
|
++alive_count_;
|
|
|
|
}
|
|
|
|
~SubClass() final { --alive_count_; }
|
|
|
|
|
|
|
|
void Trace(Visitor* visitor) const final {
|
|
|
|
visitor->Trace(data_);
|
|
|
|
SuperClass::Trace(visitor);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int alive_count_;
|
|
|
|
|
|
|
|
private:
|
|
|
|
Member<SubData> data_;
|
|
|
|
};
|
|
|
|
int SubClass::alive_count_ = 0;
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
TEST_F(WorkloadsTest, Transition) {
|
|
|
|
PointsBack::alive_count_ = 0;
|
|
|
|
SuperClass::alive_count_ = 0;
|
|
|
|
SubClass::alive_count_ = 0;
|
|
|
|
SubData::alive_count_ = 0;
|
|
|
|
|
|
|
|
Persistent<PointsBack> points_back1 =
|
|
|
|
MakeGarbageCollected<PointsBack>(GetAllocationHandle());
|
|
|
|
Persistent<PointsBack> points_back2 =
|
|
|
|
MakeGarbageCollected<PointsBack>(GetAllocationHandle());
|
|
|
|
Persistent<SuperClass> super_class =
|
|
|
|
MakeGarbageCollected<SuperClass>(GetAllocationHandle(), points_back1);
|
|
|
|
Persistent<SubClass> sub_class = MakeGarbageCollected<SubClass>(
|
|
|
|
GetAllocationHandle(), GetAllocationHandle(), points_back2);
|
|
|
|
EXPECT_EQ(2, PointsBack::alive_count_);
|
|
|
|
EXPECT_EQ(2, SuperClass::alive_count_);
|
|
|
|
EXPECT_EQ(1, SubClass::alive_count_);
|
|
|
|
EXPECT_EQ(1, SubData::alive_count_);
|
|
|
|
|
|
|
|
PreciseGC();
|
|
|
|
EXPECT_EQ(2, PointsBack::alive_count_);
|
|
|
|
EXPECT_EQ(2, SuperClass::alive_count_);
|
|
|
|
EXPECT_EQ(1, SubClass::alive_count_);
|
|
|
|
EXPECT_EQ(1, SubData::alive_count_);
|
|
|
|
|
|
|
|
super_class->InvokeConservativeGCAndExpect(this, super_class.Release(),
|
|
|
|
points_back1.Get(), 2);
|
|
|
|
PreciseGC();
|
|
|
|
EXPECT_EQ(2, PointsBack::alive_count_);
|
|
|
|
EXPECT_EQ(1, SuperClass::alive_count_);
|
|
|
|
EXPECT_EQ(1, SubClass::alive_count_);
|
|
|
|
EXPECT_EQ(1, SubData::alive_count_);
|
|
|
|
EXPECT_EQ(nullptr, points_back1->BackPointer());
|
|
|
|
|
|
|
|
points_back1.Release();
|
|
|
|
PreciseGC();
|
|
|
|
EXPECT_EQ(1, PointsBack::alive_count_);
|
|
|
|
EXPECT_EQ(1, SuperClass::alive_count_);
|
|
|
|
EXPECT_EQ(1, SubClass::alive_count_);
|
|
|
|
EXPECT_EQ(1, SubData::alive_count_);
|
|
|
|
|
|
|
|
sub_class->InvokeConservativeGCAndExpect(this, sub_class.Release(),
|
|
|
|
points_back2.Get(), 1);
|
|
|
|
PreciseGC();
|
|
|
|
EXPECT_EQ(1, PointsBack::alive_count_);
|
|
|
|
EXPECT_EQ(0, SuperClass::alive_count_);
|
|
|
|
EXPECT_EQ(0, SubClass::alive_count_);
|
|
|
|
EXPECT_EQ(0, SubData::alive_count_);
|
|
|
|
EXPECT_EQ(nullptr, points_back2->BackPointer());
|
|
|
|
|
|
|
|
points_back2.Release();
|
|
|
|
PreciseGC();
|
|
|
|
EXPECT_EQ(0, PointsBack::alive_count_);
|
|
|
|
EXPECT_EQ(0, SuperClass::alive_count_);
|
|
|
|
EXPECT_EQ(0, SubClass::alive_count_);
|
|
|
|
EXPECT_EQ(0, SubData::alive_count_);
|
|
|
|
|
|
|
|
EXPECT_EQ(super_class, sub_class);
|
|
|
|
}
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
class DynamicallySizedObject final
|
|
|
|
: public GarbageCollected<DynamicallySizedObject> {
|
|
|
|
public:
|
|
|
|
static DynamicallySizedObject* Create(AllocationHandle& allocation_handle,
|
|
|
|
size_t size) {
|
|
|
|
CHECK_GT(size, sizeof(DynamicallySizedObject));
|
|
|
|
return MakeGarbageCollected<DynamicallySizedObject>(
|
|
|
|
allocation_handle,
|
|
|
|
AdditionalBytes(size - sizeof(DynamicallySizedObject)));
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t Get(int i) { return *(reinterpret_cast<uint8_t*>(this) + i); }
|
|
|
|
|
|
|
|
void Trace(Visitor* visitor) const {}
|
|
|
|
};
|
|
|
|
|
|
|
|
class ObjectSizeCounter final : private HeapVisitor<ObjectSizeCounter> {
|
|
|
|
friend class HeapVisitor<ObjectSizeCounter>;
|
|
|
|
|
|
|
|
public:
|
2021-05-27 16:09:53 +00:00
|
|
|
size_t GetSize(RawHeap& heap) {
|
2021-04-14 15:49:54 +00:00
|
|
|
Traverse(heap);
|
|
|
|
return accumulated_size_;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
2021-05-27 16:09:53 +00:00
|
|
|
static size_t ObjectSize(const HeapObjectHeader& header) {
|
|
|
|
return ObjectView(header).Size();
|
2021-04-14 15:49:54 +00:00
|
|
|
}
|
|
|
|
|
2021-05-27 16:09:53 +00:00
|
|
|
bool VisitHeapObjectHeader(HeapObjectHeader& header) {
|
|
|
|
if (header.IsFree()) return true;
|
2021-04-14 15:49:54 +00:00
|
|
|
accumulated_size_ += ObjectSize(header);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t accumulated_size_ = 0;
|
|
|
|
};
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
TEST_F(WorkloadsTest, BasicFunctionality) {
|
|
|
|
static_assert(kAllocationGranularity % 4 == 0,
|
|
|
|
"Allocation granularity is expected to be a multiple of 4");
|
|
|
|
Heap* heap = internal::Heap::From(GetHeap());
|
|
|
|
size_t initial_object_payload_size =
|
2021-05-27 16:09:53 +00:00
|
|
|
ObjectSizeCounter().GetSize(heap->raw_heap());
|
2021-04-14 15:49:54 +00:00
|
|
|
{
|
|
|
|
// When the test starts there may already have been leaked some memory
|
|
|
|
// on the heap, so we establish a base line.
|
|
|
|
size_t base_level = initial_object_payload_size;
|
|
|
|
bool test_pages_allocated = !base_level;
|
2021-04-14 17:12:51 +00:00
|
|
|
if (test_pages_allocated) {
|
2021-04-14 15:49:54 +00:00
|
|
|
EXPECT_EQ(0ul, heap->stats_collector()->allocated_memory_size());
|
2021-04-14 17:12:51 +00:00
|
|
|
}
|
2021-04-14 15:49:54 +00:00
|
|
|
|
|
|
|
// This allocates objects on the general heap which should add a page of
|
|
|
|
// memory.
|
|
|
|
DynamicallySizedObject* alloc32 =
|
|
|
|
DynamicallySizedObject::Create(GetAllocationHandle(), 32);
|
|
|
|
memset(alloc32, 40, 32);
|
|
|
|
DynamicallySizedObject* alloc64 =
|
|
|
|
DynamicallySizedObject::Create(GetAllocationHandle(), 64);
|
|
|
|
memset(alloc64, 27, 64);
|
|
|
|
|
|
|
|
size_t total = 96;
|
|
|
|
|
|
|
|
EXPECT_EQ(base_level + total,
|
2021-05-27 16:09:53 +00:00
|
|
|
ObjectSizeCounter().GetSize(heap->raw_heap()));
|
2021-04-14 15:49:54 +00:00
|
|
|
if (test_pages_allocated) {
|
|
|
|
EXPECT_EQ(kPageSize * 2,
|
|
|
|
heap->stats_collector()->allocated_memory_size());
|
|
|
|
}
|
|
|
|
|
|
|
|
EXPECT_EQ(alloc32->Get(0), 40);
|
|
|
|
EXPECT_EQ(alloc32->Get(31), 40);
|
|
|
|
EXPECT_EQ(alloc64->Get(0), 27);
|
|
|
|
EXPECT_EQ(alloc64->Get(63), 27);
|
|
|
|
|
|
|
|
ConservativeGC();
|
|
|
|
|
|
|
|
EXPECT_EQ(alloc32->Get(0), 40);
|
|
|
|
EXPECT_EQ(alloc32->Get(31), 40);
|
|
|
|
EXPECT_EQ(alloc64->Get(0), 27);
|
|
|
|
EXPECT_EQ(alloc64->Get(63), 27);
|
|
|
|
}
|
|
|
|
|
|
|
|
PreciseGC();
|
|
|
|
size_t total = 0;
|
2021-05-27 16:09:53 +00:00
|
|
|
size_t base_level = ObjectSizeCounter().GetSize(heap->raw_heap());
|
2021-04-14 15:49:54 +00:00
|
|
|
bool test_pages_allocated = !base_level;
|
2021-04-14 17:12:51 +00:00
|
|
|
if (test_pages_allocated) {
|
2021-04-14 15:49:54 +00:00
|
|
|
EXPECT_EQ(0ul, heap->stats_collector()->allocated_memory_size());
|
2021-04-14 17:12:51 +00:00
|
|
|
}
|
2021-04-14 15:49:54 +00:00
|
|
|
|
|
|
|
size_t big = 1008;
|
|
|
|
Persistent<DynamicallySizedObject> big_area =
|
|
|
|
DynamicallySizedObject::Create(GetAllocationHandle(), big);
|
|
|
|
total += big;
|
|
|
|
|
|
|
|
size_t persistent_count = 0;
|
|
|
|
const size_t kNumPersistents = 100000;
|
|
|
|
Persistent<DynamicallySizedObject>* persistents[kNumPersistents];
|
|
|
|
|
|
|
|
for (int i = 0; i < 1000; i++) {
|
|
|
|
size_t size = 128 + i * 8;
|
|
|
|
total += size;
|
|
|
|
persistents[persistent_count++] = new Persistent<DynamicallySizedObject>(
|
|
|
|
DynamicallySizedObject::Create(GetAllocationHandle(), size));
|
|
|
|
// The allocations in the loop may trigger GC with lazy sweeping.
|
|
|
|
heap->sweeper().FinishIfRunning();
|
|
|
|
EXPECT_EQ(base_level + total,
|
2021-05-27 16:09:53 +00:00
|
|
|
ObjectSizeCounter().GetSize(heap->raw_heap()));
|
2021-04-14 15:49:54 +00:00
|
|
|
if (test_pages_allocated) {
|
|
|
|
EXPECT_EQ(0ul, heap->stats_collector()->allocated_memory_size() &
|
|
|
|
(kPageSize - 1));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
DynamicallySizedObject* alloc32b(
|
|
|
|
DynamicallySizedObject::Create(GetAllocationHandle(), 32));
|
|
|
|
memset(alloc32b, 40, 32);
|
|
|
|
DynamicallySizedObject* alloc64b(
|
|
|
|
DynamicallySizedObject::Create(GetAllocationHandle(), 64));
|
|
|
|
memset(alloc64b, 27, 64);
|
|
|
|
EXPECT_TRUE(alloc32b != alloc64b);
|
|
|
|
|
|
|
|
total += 96;
|
|
|
|
EXPECT_EQ(base_level + total,
|
2021-05-27 16:09:53 +00:00
|
|
|
ObjectSizeCounter().GetSize(heap->raw_heap()));
|
2021-04-14 15:49:54 +00:00
|
|
|
if (test_pages_allocated) {
|
|
|
|
EXPECT_EQ(0ul, heap->stats_collector()->allocated_memory_size() &
|
|
|
|
(kPageSize - 1));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
PreciseGC();
|
|
|
|
total -= 96;
|
|
|
|
if (test_pages_allocated) {
|
|
|
|
EXPECT_EQ(0ul, heap->stats_collector()->allocated_memory_size() &
|
|
|
|
(kPageSize - 1));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Clear the persistent, so that the big area will be garbage collected.
|
|
|
|
big_area.Release();
|
|
|
|
PreciseGC();
|
|
|
|
|
|
|
|
total -= big;
|
2021-05-27 16:09:53 +00:00
|
|
|
EXPECT_EQ(base_level + total, ObjectSizeCounter().GetSize(heap->raw_heap()));
|
2021-04-14 15:49:54 +00:00
|
|
|
if (test_pages_allocated) {
|
|
|
|
EXPECT_EQ(0ul, heap->stats_collector()->allocated_memory_size() &
|
|
|
|
(kPageSize - 1));
|
|
|
|
}
|
|
|
|
|
2021-05-27 16:09:53 +00:00
|
|
|
EXPECT_EQ(base_level + total, ObjectSizeCounter().GetSize(heap->raw_heap()));
|
2021-04-14 15:49:54 +00:00
|
|
|
if (test_pages_allocated) {
|
|
|
|
EXPECT_EQ(0ul, heap->stats_collector()->allocated_memory_size() &
|
|
|
|
(kPageSize - 1));
|
|
|
|
}
|
|
|
|
|
|
|
|
for (size_t i = 0; i < persistent_count; i++) {
|
|
|
|
delete persistents[i];
|
|
|
|
persistents[i] = nullptr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace internal
|
|
|
|
} // namespace cppgc
|