2019-12-20 15:01:42 +00:00
|
|
|
// Copyright 2019 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.
|
|
|
|
|
2020-08-10 14:38:05 +00:00
|
|
|
#include "src/heap/marking-worklist.h"
|
|
|
|
|
2019-12-20 15:01:42 +00:00
|
|
|
#include <cmath>
|
|
|
|
#include <limits>
|
|
|
|
|
|
|
|
#include "src/heap/heap-inl.h"
|
|
|
|
#include "src/heap/heap.h"
|
2020-08-10 14:38:05 +00:00
|
|
|
#include "src/heap/marking-worklist-inl.h"
|
2019-12-20 15:01:42 +00:00
|
|
|
#include "test/unittests/test-utils.h"
|
|
|
|
#include "testing/gtest/include/gtest/gtest.h"
|
|
|
|
|
|
|
|
namespace v8 {
|
|
|
|
namespace internal {
|
|
|
|
|
|
|
|
using MarkingWorklistTest = TestWithContext;
|
|
|
|
|
|
|
|
TEST_F(MarkingWorklistTest, PushPop) {
|
2020-08-10 14:38:05 +00:00
|
|
|
MarkingWorklists holder;
|
|
|
|
MarkingWorklists::Local worklists(&holder);
|
2019-12-20 15:01:42 +00:00
|
|
|
HeapObject pushed_object =
|
|
|
|
ReadOnlyRoots(i_isolate()->heap()).undefined_value();
|
|
|
|
worklists.Push(pushed_object);
|
|
|
|
HeapObject popped_object;
|
|
|
|
EXPECT_TRUE(worklists.Pop(&popped_object));
|
|
|
|
EXPECT_EQ(popped_object, pushed_object);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(MarkingWorklistTest, PushPopOnHold) {
|
2020-08-10 14:38:05 +00:00
|
|
|
MarkingWorklists holder;
|
|
|
|
MarkingWorklists::Local worklists(&holder);
|
2019-12-20 15:01:42 +00:00
|
|
|
HeapObject pushed_object =
|
|
|
|
ReadOnlyRoots(i_isolate()->heap()).undefined_value();
|
|
|
|
worklists.PushOnHold(pushed_object);
|
|
|
|
HeapObject popped_object;
|
|
|
|
EXPECT_TRUE(worklists.PopOnHold(&popped_object));
|
|
|
|
EXPECT_EQ(popped_object, pushed_object);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(MarkingWorklistTest, PushPopEmbedder) {
|
2020-08-10 14:38:05 +00:00
|
|
|
MarkingWorklists holder;
|
|
|
|
MarkingWorklists::Local worklists(&holder);
|
2019-12-20 15:01:42 +00:00
|
|
|
HeapObject pushed_object =
|
|
|
|
ReadOnlyRoots(i_isolate()->heap()).undefined_value();
|
2021-12-03 12:53:10 +00:00
|
|
|
worklists.PushWrapper(pushed_object);
|
2019-12-20 15:01:42 +00:00
|
|
|
HeapObject popped_object;
|
2021-12-03 12:53:10 +00:00
|
|
|
EXPECT_TRUE(worklists.PopWrapper(&popped_object));
|
2019-12-20 15:01:42 +00:00
|
|
|
EXPECT_EQ(popped_object, pushed_object);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(MarkingWorklistTest, MergeOnHold) {
|
2020-08-10 14:38:05 +00:00
|
|
|
MarkingWorklists holder;
|
|
|
|
MarkingWorklists::Local main_worklists(&holder);
|
|
|
|
MarkingWorklists::Local worker_worklists(&holder);
|
2019-12-20 15:01:42 +00:00
|
|
|
HeapObject pushed_object =
|
|
|
|
ReadOnlyRoots(i_isolate()->heap()).undefined_value();
|
|
|
|
worker_worklists.PushOnHold(pushed_object);
|
2020-08-10 14:38:05 +00:00
|
|
|
worker_worklists.Publish();
|
2019-12-20 15:01:42 +00:00
|
|
|
main_worklists.MergeOnHold();
|
|
|
|
HeapObject popped_object;
|
|
|
|
EXPECT_TRUE(main_worklists.Pop(&popped_object));
|
|
|
|
EXPECT_EQ(popped_object, pushed_object);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(MarkingWorklistTest, ShareWorkIfGlobalPoolIsEmpty) {
|
2020-08-10 14:38:05 +00:00
|
|
|
MarkingWorklists holder;
|
|
|
|
MarkingWorklists::Local main_worklists(&holder);
|
|
|
|
MarkingWorklists::Local worker_worklists(&holder);
|
2019-12-20 15:01:42 +00:00
|
|
|
HeapObject pushed_object =
|
|
|
|
ReadOnlyRoots(i_isolate()->heap()).undefined_value();
|
|
|
|
main_worklists.Push(pushed_object);
|
2020-08-10 14:38:05 +00:00
|
|
|
main_worklists.ShareWork();
|
2019-12-20 15:01:42 +00:00
|
|
|
HeapObject popped_object;
|
|
|
|
EXPECT_TRUE(worker_worklists.Pop(&popped_object));
|
|
|
|
EXPECT_EQ(popped_object, pushed_object);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(MarkingWorklistTest, ContextWorklistsPushPop) {
|
|
|
|
const Address context = 0xabcdef;
|
2020-08-10 14:38:05 +00:00
|
|
|
MarkingWorklists holder;
|
2019-12-20 15:01:42 +00:00
|
|
|
holder.CreateContextWorklists({context});
|
2020-08-10 14:38:05 +00:00
|
|
|
MarkingWorklists::Local worklists(&holder);
|
2019-12-20 15:01:42 +00:00
|
|
|
HeapObject pushed_object =
|
|
|
|
ReadOnlyRoots(i_isolate()->heap()).undefined_value();
|
|
|
|
worklists.SwitchToContext(context);
|
|
|
|
worklists.Push(pushed_object);
|
|
|
|
worklists.SwitchToShared();
|
|
|
|
HeapObject popped_object;
|
|
|
|
EXPECT_TRUE(worklists.Pop(&popped_object));
|
|
|
|
EXPECT_EQ(popped_object, pushed_object);
|
|
|
|
holder.ReleaseContextWorklists();
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(MarkingWorklistTest, ContextWorklistsEmpty) {
|
|
|
|
const Address context = 0xabcdef;
|
2020-08-10 14:38:05 +00:00
|
|
|
MarkingWorklists holder;
|
2019-12-20 15:01:42 +00:00
|
|
|
holder.CreateContextWorklists({context});
|
2020-08-10 14:38:05 +00:00
|
|
|
MarkingWorklists::Local worklists(&holder);
|
2019-12-20 15:01:42 +00:00
|
|
|
HeapObject pushed_object =
|
|
|
|
ReadOnlyRoots(i_isolate()->heap()).undefined_value();
|
|
|
|
worklists.SwitchToContext(context);
|
|
|
|
worklists.Push(pushed_object);
|
|
|
|
EXPECT_FALSE(worklists.IsEmpty());
|
|
|
|
worklists.SwitchToShared();
|
|
|
|
EXPECT_FALSE(worklists.IsEmpty());
|
|
|
|
HeapObject popped_object;
|
|
|
|
EXPECT_TRUE(worklists.Pop(&popped_object));
|
|
|
|
EXPECT_EQ(popped_object, pushed_object);
|
|
|
|
EXPECT_TRUE(worklists.IsEmpty());
|
|
|
|
holder.ReleaseContextWorklists();
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(MarkingWorklistTest, ContextWorklistCrossTask) {
|
|
|
|
const Address context1 = 0x1abcdef;
|
|
|
|
const Address context2 = 0x2abcdef;
|
2020-08-10 14:38:05 +00:00
|
|
|
MarkingWorklists holder;
|
2019-12-20 15:01:42 +00:00
|
|
|
holder.CreateContextWorklists({context1, context2});
|
2020-08-10 14:38:05 +00:00
|
|
|
MarkingWorklists::Local main_worklists(&holder);
|
|
|
|
MarkingWorklists::Local worker_worklists(&holder);
|
2019-12-20 15:01:42 +00:00
|
|
|
HeapObject pushed_object =
|
|
|
|
ReadOnlyRoots(i_isolate()->heap()).undefined_value();
|
|
|
|
main_worklists.SwitchToContext(context1);
|
|
|
|
main_worklists.Push(pushed_object);
|
2020-08-10 14:38:05 +00:00
|
|
|
main_worklists.ShareWork();
|
2019-12-20 15:01:42 +00:00
|
|
|
worker_worklists.SwitchToContext(context2);
|
|
|
|
HeapObject popped_object;
|
|
|
|
EXPECT_TRUE(worker_worklists.Pop(&popped_object));
|
|
|
|
EXPECT_EQ(popped_object, pushed_object);
|
|
|
|
EXPECT_EQ(context1, worker_worklists.Context());
|
|
|
|
holder.ReleaseContextWorklists();
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace internal
|
|
|
|
} // namespace v8
|