2020-04-06 13:29:41 +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 "src/heap/cppgc/heap.h"
|
2020-04-28 20:16:14 +00:00
|
|
|
|
|
|
|
#include <algorithm>
|
|
|
|
#include <iterator>
|
|
|
|
#include <numeric>
|
|
|
|
|
2020-04-06 13:29:41 +00:00
|
|
|
#include "include/cppgc/allocation.h"
|
2020-04-28 20:16:14 +00:00
|
|
|
#include "src/heap/cppgc/globals.h"
|
2020-04-06 13:29:41 +00:00
|
|
|
#include "test/unittests/heap/cppgc/tests.h"
|
|
|
|
#include "testing/gtest/include/gtest/gtest.h"
|
|
|
|
|
|
|
|
namespace cppgc {
|
|
|
|
namespace internal {
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
class GCHeapTest : public testing::TestWithHeap {
|
|
|
|
public:
|
|
|
|
void ConservativeGC() {
|
|
|
|
internal::Heap::From(GetHeap())->CollectGarbage(
|
2020-06-15 20:56:18 +00:00
|
|
|
Heap::Config::ConservativeAtomicConfig());
|
2020-04-06 13:29:41 +00:00
|
|
|
}
|
|
|
|
void PreciseGC() {
|
|
|
|
internal::Heap::From(GetHeap())->CollectGarbage(
|
2020-06-15 20:56:18 +00:00
|
|
|
Heap::Config::PreciseAtomicConfig());
|
2020-04-06 13:29:41 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
class Foo : public GarbageCollected<Foo> {
|
|
|
|
public:
|
|
|
|
static size_t destructor_callcount;
|
|
|
|
|
|
|
|
Foo() { destructor_callcount = 0; }
|
|
|
|
~Foo() { destructor_callcount++; }
|
2020-05-13 23:20:23 +00:00
|
|
|
|
|
|
|
void Trace(cppgc::Visitor*) const {}
|
2020-04-06 13:29:41 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
size_t Foo::destructor_callcount;
|
|
|
|
|
2020-04-28 20:16:14 +00:00
|
|
|
template <size_t Size>
|
|
|
|
class GCed : public GarbageCollected<Foo> {
|
|
|
|
public:
|
2020-05-13 23:20:23 +00:00
|
|
|
void Trace(cppgc::Visitor*) const {}
|
2020-04-28 20:16:14 +00:00
|
|
|
char buf[Size];
|
|
|
|
};
|
|
|
|
|
2020-04-06 13:29:41 +00:00
|
|
|
} // namespace
|
|
|
|
|
|
|
|
TEST_F(GCHeapTest, PreciseGCReclaimsObjectOnStack) {
|
2020-06-10 22:28:41 +00:00
|
|
|
Foo* volatile do_not_access =
|
|
|
|
MakeGarbageCollected<Foo>(GetAllocationHandle());
|
2020-04-07 06:48:15 +00:00
|
|
|
USE(do_not_access);
|
2020-04-06 13:29:41 +00:00
|
|
|
EXPECT_EQ(0u, Foo::destructor_callcount);
|
|
|
|
PreciseGC();
|
|
|
|
EXPECT_EQ(1u, Foo::destructor_callcount);
|
2020-04-07 06:48:15 +00:00
|
|
|
PreciseGC();
|
|
|
|
EXPECT_EQ(1u, Foo::destructor_callcount);
|
2020-04-06 13:29:41 +00:00
|
|
|
}
|
|
|
|
|
2020-04-07 09:42:21 +00:00
|
|
|
namespace {
|
|
|
|
|
|
|
|
const void* ConservativeGCReturningObject(cppgc::Heap* heap,
|
|
|
|
const void* volatile object) {
|
|
|
|
internal::Heap::From(heap)->CollectGarbage(
|
2020-06-15 20:56:18 +00:00
|
|
|
Heap::Config::ConservativeAtomicConfig());
|
2020-04-07 09:42:21 +00:00
|
|
|
return object;
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
2020-04-28 20:16:14 +00:00
|
|
|
TEST_F(GCHeapTest, ConservativeGCRetainsObjectOnStack) {
|
2020-06-10 22:28:41 +00:00
|
|
|
Foo* volatile object = MakeGarbageCollected<Foo>(GetAllocationHandle());
|
2020-04-06 13:29:41 +00:00
|
|
|
EXPECT_EQ(0u, Foo::destructor_callcount);
|
2020-04-07 09:42:21 +00:00
|
|
|
EXPECT_EQ(object, ConservativeGCReturningObject(GetHeap(), object));
|
2020-04-06 13:29:41 +00:00
|
|
|
EXPECT_EQ(0u, Foo::destructor_callcount);
|
|
|
|
PreciseGC();
|
|
|
|
EXPECT_EQ(1u, Foo::destructor_callcount);
|
2020-04-07 06:48:15 +00:00
|
|
|
PreciseGC();
|
|
|
|
EXPECT_EQ(1u, Foo::destructor_callcount);
|
2020-04-06 13:29:41 +00:00
|
|
|
}
|
|
|
|
|
2020-04-28 20:16:14 +00:00
|
|
|
TEST_F(GCHeapTest, ObjectPayloadSize) {
|
|
|
|
static constexpr size_t kNumberOfObjectsPerArena = 16;
|
|
|
|
static constexpr size_t kObjectSizes[] = {1, 32, 64, 128,
|
|
|
|
2 * kLargeObjectSizeThreshold};
|
|
|
|
|
2020-06-08 09:53:26 +00:00
|
|
|
Heap::From(GetHeap())->CollectGarbage(
|
|
|
|
GarbageCollector::Config::ConservativeAtomicConfig());
|
2020-04-28 20:16:14 +00:00
|
|
|
|
2020-06-15 09:37:43 +00:00
|
|
|
Heap::NoGCScope no_gc_scope(*Heap::From(GetHeap()));
|
2020-06-10 07:10:20 +00:00
|
|
|
|
2020-04-28 20:16:14 +00:00
|
|
|
for (size_t k = 0; k < kNumberOfObjectsPerArena; ++k) {
|
2020-06-10 22:28:41 +00:00
|
|
|
MakeGarbageCollected<GCed<kObjectSizes[0]>>(GetAllocationHandle());
|
|
|
|
MakeGarbageCollected<GCed<kObjectSizes[1]>>(GetAllocationHandle());
|
|
|
|
MakeGarbageCollected<GCed<kObjectSizes[2]>>(GetAllocationHandle());
|
|
|
|
MakeGarbageCollected<GCed<kObjectSizes[3]>>(GetAllocationHandle());
|
|
|
|
MakeGarbageCollected<GCed<kObjectSizes[4]>>(GetAllocationHandle());
|
2020-04-28 20:16:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
size_t aligned_object_sizes[arraysize(kObjectSizes)];
|
|
|
|
std::transform(std::cbegin(kObjectSizes), std::cend(kObjectSizes),
|
|
|
|
std::begin(aligned_object_sizes), [](size_t size) {
|
|
|
|
return RoundUp(size, kAllocationGranularity);
|
|
|
|
});
|
|
|
|
const size_t expected_size = std::accumulate(
|
|
|
|
std::cbegin(aligned_object_sizes), std::cend(aligned_object_sizes), 0u,
|
|
|
|
[](size_t acc, size_t size) {
|
|
|
|
return acc + kNumberOfObjectsPerArena * size;
|
|
|
|
});
|
|
|
|
// TODO(chromium:1056170): Change to EXPECT_EQ when proper sweeping is
|
|
|
|
// implemented.
|
|
|
|
EXPECT_LE(expected_size, Heap::From(GetHeap())->ObjectPayloadSize());
|
|
|
|
}
|
|
|
|
|
2020-04-06 13:29:41 +00:00
|
|
|
} // namespace internal
|
|
|
|
} // namespace cppgc
|