2016-02-08 08:51:02 +00:00
|
|
|
// Copyright 2016 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.
|
|
|
|
|
2022-10-06 08:36:56 +00:00
|
|
|
#include "src/heap/slot-set.h"
|
|
|
|
|
2016-02-08 08:51:02 +00:00
|
|
|
#include <limits>
|
2016-10-21 09:04:50 +00:00
|
|
|
#include <map>
|
2016-02-08 08:51:02 +00:00
|
|
|
|
2019-05-24 13:51:59 +00:00
|
|
|
#include "src/common/globals.h"
|
2016-02-08 08:51:02 +00:00
|
|
|
#include "src/heap/spaces.h"
|
2018-10-23 23:02:20 +00:00
|
|
|
#include "src/objects/slots.h"
|
2016-02-08 08:51:02 +00:00
|
|
|
#include "testing/gtest/include/gtest/gtest.h"
|
|
|
|
|
|
|
|
namespace v8 {
|
|
|
|
namespace internal {
|
|
|
|
|
2019-12-06 14:49:21 +00:00
|
|
|
TEST(PossiblyEmptyBucketsTest, WordsForBuckets) {
|
|
|
|
EXPECT_EQ(
|
|
|
|
PossiblyEmptyBuckets::WordsForBuckets(PossiblyEmptyBuckets::kBitsPerWord),
|
|
|
|
1U);
|
|
|
|
EXPECT_EQ(PossiblyEmptyBuckets::WordsForBuckets(
|
|
|
|
PossiblyEmptyBuckets::kBitsPerWord - 1),
|
|
|
|
1U);
|
|
|
|
EXPECT_EQ(PossiblyEmptyBuckets::WordsForBuckets(
|
|
|
|
PossiblyEmptyBuckets::kBitsPerWord + 1),
|
|
|
|
2U);
|
|
|
|
|
|
|
|
EXPECT_EQ(PossiblyEmptyBuckets::WordsForBuckets(
|
|
|
|
5 * PossiblyEmptyBuckets::kBitsPerWord - 1),
|
|
|
|
5U);
|
|
|
|
EXPECT_EQ(PossiblyEmptyBuckets::WordsForBuckets(
|
|
|
|
5 * PossiblyEmptyBuckets::kBitsPerWord),
|
|
|
|
5U);
|
|
|
|
EXPECT_EQ(PossiblyEmptyBuckets::WordsForBuckets(
|
|
|
|
5 * PossiblyEmptyBuckets::kBitsPerWord + 1),
|
|
|
|
6U);
|
|
|
|
}
|
|
|
|
|
2019-10-30 09:56:04 +00:00
|
|
|
TEST(SlotSet, BucketsForSize) {
|
|
|
|
EXPECT_EQ(static_cast<size_t>(SlotSet::kBucketsRegularPage),
|
|
|
|
SlotSet::BucketsForSize(Page::kPageSize));
|
|
|
|
|
|
|
|
EXPECT_EQ(static_cast<size_t>(SlotSet::kBucketsRegularPage) * 2,
|
|
|
|
SlotSet::BucketsForSize(Page::kPageSize * 2));
|
|
|
|
}
|
|
|
|
|
2019-12-06 14:49:21 +00:00
|
|
|
TEST(PossiblyEmptyBuckets, ContainsAndInsert) {
|
|
|
|
static const int kBuckets = 100;
|
|
|
|
PossiblyEmptyBuckets possibly_empty_buckets;
|
|
|
|
possibly_empty_buckets.Insert(0, kBuckets);
|
|
|
|
int last = sizeof(uintptr_t) * kBitsPerByte - 2;
|
|
|
|
possibly_empty_buckets.Insert(last, kBuckets);
|
|
|
|
EXPECT_TRUE(possibly_empty_buckets.Contains(0));
|
|
|
|
EXPECT_TRUE(possibly_empty_buckets.Contains(last));
|
|
|
|
possibly_empty_buckets.Insert(last + 1, kBuckets);
|
|
|
|
EXPECT_TRUE(possibly_empty_buckets.Contains(0));
|
|
|
|
EXPECT_TRUE(possibly_empty_buckets.Contains(last));
|
|
|
|
EXPECT_TRUE(possibly_empty_buckets.Contains(last + 1));
|
|
|
|
}
|
|
|
|
|
2016-02-25 17:28:50 +00:00
|
|
|
TEST(TypedSlotSet, Iterate) {
|
|
|
|
TypedSlotSet set(0);
|
2017-01-19 21:34:00 +00:00
|
|
|
// These two constants must be static as a workaround
|
|
|
|
// for a MSVC++ bug about lambda captures, see the discussion at
|
|
|
|
// https://social.msdn.microsoft.com/Forums/SqlServer/4abf18bd-4ae4-4c72-ba3e-3b13e7909d5f
|
|
|
|
static const int kDelta = 10000001;
|
2016-02-25 17:28:50 +00:00
|
|
|
int added = 0;
|
2018-12-10 16:13:45 +00:00
|
|
|
for (uint32_t i = 0; i < TypedSlotSet::kMaxOffset; i += kDelta) {
|
2022-01-20 10:53:34 +00:00
|
|
|
SlotType type =
|
|
|
|
static_cast<SlotType>(i % static_cast<uint8_t>(SlotType::kCleared));
|
2018-12-10 16:13:45 +00:00
|
|
|
set.Insert(type, i);
|
2016-02-25 17:28:50 +00:00
|
|
|
++added;
|
|
|
|
}
|
|
|
|
int iterated = 0;
|
2016-09-26 15:54:03 +00:00
|
|
|
set.Iterate(
|
2018-12-10 16:13:45 +00:00
|
|
|
[&iterated](SlotType type, Address addr) {
|
2018-04-13 22:28:05 +00:00
|
|
|
uint32_t i = static_cast<uint32_t>(addr);
|
2022-01-20 10:53:34 +00:00
|
|
|
EXPECT_EQ(i % static_cast<uint8_t>(SlotType::kCleared),
|
|
|
|
static_cast<uint32_t>(type));
|
2016-11-11 14:55:47 +00:00
|
|
|
EXPECT_EQ(0u, i % kDelta);
|
2016-09-26 15:54:03 +00:00
|
|
|
++iterated;
|
|
|
|
return i % 2 == 0 ? KEEP_SLOT : REMOVE_SLOT;
|
|
|
|
},
|
|
|
|
TypedSlotSet::KEEP_EMPTY_CHUNKS);
|
2016-02-25 17:28:50 +00:00
|
|
|
EXPECT_EQ(added, iterated);
|
|
|
|
iterated = 0;
|
2016-09-26 15:54:03 +00:00
|
|
|
set.Iterate(
|
2018-12-10 16:13:45 +00:00
|
|
|
[&iterated](SlotType type, Address addr) {
|
2018-04-13 22:28:05 +00:00
|
|
|
uint32_t i = static_cast<uint32_t>(addr);
|
2016-11-11 14:55:47 +00:00
|
|
|
EXPECT_EQ(0u, i % 2);
|
2016-09-26 15:54:03 +00:00
|
|
|
++iterated;
|
|
|
|
return KEEP_SLOT;
|
|
|
|
},
|
|
|
|
TypedSlotSet::KEEP_EMPTY_CHUNKS);
|
2016-02-25 17:28:50 +00:00
|
|
|
EXPECT_EQ(added / 2, iterated);
|
|
|
|
}
|
|
|
|
|
2018-11-21 10:34:07 +00:00
|
|
|
TEST(TypedSlotSet, ClearInvalidSlots) {
|
2016-10-21 09:04:50 +00:00
|
|
|
TypedSlotSet set(0);
|
|
|
|
const int kHostDelta = 100;
|
|
|
|
uint32_t entries = 10;
|
|
|
|
for (uint32_t i = 0; i < entries; i++) {
|
2022-01-20 10:53:34 +00:00
|
|
|
SlotType type =
|
|
|
|
static_cast<SlotType>(i % static_cast<uint8_t>(SlotType::kCleared));
|
2018-12-10 16:13:45 +00:00
|
|
|
set.Insert(type, i * kHostDelta);
|
2016-10-21 09:04:50 +00:00
|
|
|
}
|
|
|
|
|
2022-04-06 11:01:03 +00:00
|
|
|
TypedSlotSet::FreeRangesMap invalid_ranges;
|
2016-10-21 09:04:50 +00:00
|
|
|
for (uint32_t i = 1; i < entries; i += 2) {
|
|
|
|
invalid_ranges.insert(
|
|
|
|
std::pair<uint32_t, uint32_t>(i * kHostDelta, i * kHostDelta + 1));
|
|
|
|
}
|
|
|
|
|
2018-11-21 10:34:07 +00:00
|
|
|
set.ClearInvalidSlots(invalid_ranges);
|
2022-04-06 11:01:03 +00:00
|
|
|
for (TypedSlotSet::FreeRangesMap::iterator it = invalid_ranges.begin();
|
2016-10-21 09:04:50 +00:00
|
|
|
it != invalid_ranges.end(); ++it) {
|
|
|
|
uint32_t start = it->first;
|
|
|
|
uint32_t end = it->second;
|
|
|
|
set.Iterate(
|
2018-12-10 16:13:45 +00:00
|
|
|
[=](SlotType slot_type, Address slot_addr) {
|
|
|
|
CHECK(slot_addr < start || slot_addr >= end);
|
2016-10-21 09:04:50 +00:00
|
|
|
return KEEP_SLOT;
|
|
|
|
},
|
|
|
|
TypedSlotSet::KEEP_EMPTY_CHUNKS);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-11-21 10:34:07 +00:00
|
|
|
TEST(TypedSlotSet, Merge) {
|
|
|
|
TypedSlotSet set0(0), set1(0);
|
|
|
|
static const uint32_t kEntries = 10000;
|
|
|
|
for (uint32_t i = 0; i < kEntries; i++) {
|
2022-01-20 10:53:34 +00:00
|
|
|
set0.Insert(SlotType::kEmbeddedObjectFull, 2 * i);
|
|
|
|
set1.Insert(SlotType::kEmbeddedObjectFull, 2 * i + 1);
|
2018-11-21 10:34:07 +00:00
|
|
|
}
|
|
|
|
uint32_t count = 0;
|
|
|
|
set0.Merge(&set1);
|
|
|
|
set0.Iterate(
|
2018-12-10 16:13:45 +00:00
|
|
|
[&count](SlotType slot_type, Address slot_addr) {
|
2018-11-21 10:34:07 +00:00
|
|
|
if (count < kEntries) {
|
2018-12-10 16:13:45 +00:00
|
|
|
CHECK_EQ(slot_addr % 2, 0);
|
2018-11-21 10:34:07 +00:00
|
|
|
} else {
|
2018-12-10 16:13:45 +00:00
|
|
|
CHECK_EQ(slot_addr % 2, 1);
|
2018-11-21 10:34:07 +00:00
|
|
|
}
|
|
|
|
++count;
|
|
|
|
return KEEP_SLOT;
|
|
|
|
},
|
|
|
|
TypedSlotSet::KEEP_EMPTY_CHUNKS);
|
|
|
|
CHECK_EQ(2 * kEntries, count);
|
|
|
|
set1.Iterate(
|
2018-12-10 16:13:45 +00:00
|
|
|
[](SlotType slot_type, Address slot_addr) {
|
2018-11-21 10:34:07 +00:00
|
|
|
CHECK(false); // Unreachable.
|
|
|
|
return KEEP_SLOT;
|
|
|
|
},
|
|
|
|
TypedSlotSet::KEEP_EMPTY_CHUNKS);
|
|
|
|
}
|
|
|
|
|
2016-02-08 08:51:02 +00:00
|
|
|
} // namespace internal
|
|
|
|
} // namespace v8
|