Commit 5bc1d291 authored by Ulan Degenbaev's avatar Ulan Degenbaev Committed by Commit Bot

[heap] Make assert in Page::ShrinkToHighWaterMark more robust.

Currently Page::ShrinkToHighWaterMark checks that there is only one
filler in the to-be-freed area at the end of the page. This does not
hold if an allocation observer is active.

We should instead check that the to-be-freed area does not contain
allocated objects and will not contain allocated objects:
1) Following chain of fillers we arrive at the end of the page.
2) The free list of the page is empty.

This patch also changes PagedSpace::ResetFreeList to evict free list
entries of each page, instead of just reseting the global free list.

It also removes invalidation of free list categories.
Now FreeList::EvictFreeListItems simply evicts free list entries without
invalidating free list categories.

Cq-Include-Trybots: master.tryserver.chromium.linux:linux_chromium_rel_ng
Change-Id: I6051578ec2c91c9614d14c7a6ce188d2db5ace3b
Reviewed-on: https://chromium-review.googlesource.com/822571
Commit-Queue: Ulan Degenbaev <ulan@chromium.org>
Reviewed-by: 's avatarMichael Lippautz <mlippautz@chromium.org>
Cr-Commit-Position: refs/heads/master@{#50040}
parent cb35f165
...@@ -847,6 +847,22 @@ size_t Page::AvailableInFreeList() { ...@@ -847,6 +847,22 @@ size_t Page::AvailableInFreeList() {
return sum; return sum;
} }
#ifdef DEBUG
namespace {
// Skips filler starting from the given filler until the end address.
// Returns the first address after the skipped fillers.
Address SkipFillers(HeapObject* filler, Address end) {
Address addr = filler->address();
while (addr < end) {
filler = HeapObject::FromAddress(addr);
CHECK(filler->IsFiller());
addr = filler->address() + filler->Size();
}
return addr;
}
} // anonymous namespace
#endif // DEBUG
size_t Page::ShrinkToHighWaterMark() { size_t Page::ShrinkToHighWaterMark() {
// Shrinking only makes sense outside of the CodeRange, where we don't care // Shrinking only makes sense outside of the CodeRange, where we don't care
// about address space fragmentation. // about address space fragmentation.
...@@ -858,28 +874,12 @@ size_t Page::ShrinkToHighWaterMark() { ...@@ -858,28 +874,12 @@ size_t Page::ShrinkToHighWaterMark() {
HeapObject* filler = HeapObject::FromAddress(HighWaterMark()); HeapObject* filler = HeapObject::FromAddress(HighWaterMark());
if (filler->address() == area_end()) return 0; if (filler->address() == area_end()) return 0;
CHECK(filler->IsFiller()); CHECK(filler->IsFiller());
if (!filler->IsFreeSpace()) return 0; // Ensure that no objects were allocated in [filler, area_end) region.
DCHECK_EQ(area_end(), SkipFillers(filler, area_end()));
#ifdef DEBUG // Ensure that no objects will be allocated on this page.
// Check the the filler is indeed the last filler on the page. DCHECK_EQ(0u, AvailableInFreeList());
HeapObjectIterator it(this);
HeapObject* filler2 = nullptr;
for (HeapObject* obj = it.Next(); obj != nullptr; obj = it.Next()) {
filler2 = HeapObject::FromAddress(obj->address() + obj->Size());
}
if (filler2 == nullptr || filler2->address() == area_end()) return 0;
DCHECK(filler2->IsFiller());
// The deserializer might leave behind fillers. In this case we need to
// iterate even further.
while ((filler2->address() + filler2->Size()) != area_end()) {
filler2 = HeapObject::FromAddress(filler2->address() + filler2->Size());
DCHECK(filler2->IsFiller());
}
DCHECK_EQ(filler->address(), filler2->address());
#endif // DEBUG
size_t unused = RoundDown( size_t unused = RoundDown(static_cast<size_t>(area_end() - filler->address()),
static_cast<size_t>(area_end() - filler->address() - FreeSpace::kSize),
MemoryAllocator::GetCommitPageSize()); MemoryAllocator::GetCommitPageSize());
if (unused > 0) { if (unused > 0) {
DCHECK_EQ(0u, unused % MemoryAllocator::GetCommitPageSize()); DCHECK_EQ(0u, unused % MemoryAllocator::GetCommitPageSize());
...@@ -1552,12 +1552,18 @@ size_t PagedSpace::ShrinkPageToHighWaterMark(Page* page) { ...@@ -1552,12 +1552,18 @@ size_t PagedSpace::ShrinkPageToHighWaterMark(Page* page) {
return unused; return unused;
} }
void PagedSpace::ResetFreeList() {
for (Page* page : *this) {
free_list_.EvictFreeListItems(page);
}
DCHECK(free_list_.IsEmpty());
}
void PagedSpace::ShrinkImmortalImmovablePages() { void PagedSpace::ShrinkImmortalImmovablePages() {
DCHECK(!heap()->deserialization_complete()); DCHECK(!heap()->deserialization_complete());
MemoryChunk::UpdateHighWaterMark(allocation_info_.top()); MemoryChunk::UpdateHighWaterMark(allocation_info_.top());
EmptyAllocationInfo(); EmptyAllocationInfo();
ResetFreeList(); ResetFreeList();
for (Page* page : *this) { for (Page* page : *this) {
DCHECK(page->IsFlagSet(Page::NEVER_EVACUATE)); DCHECK(page->IsFlagSet(Page::NEVER_EVACUATE));
ShrinkPageToHighWaterMark(page); ShrinkPageToHighWaterMark(page);
...@@ -2822,11 +2828,6 @@ void FreeListCategory::Relink() { ...@@ -2822,11 +2828,6 @@ void FreeListCategory::Relink() {
owner()->AddCategory(this); owner()->AddCategory(this);
} }
void FreeListCategory::Invalidate() {
Reset();
type_ = kInvalidCategory;
}
FreeList::FreeList(PagedSpace* owner) : owner_(owner), wasted_bytes_(0) { FreeList::FreeList(PagedSpace* owner) : owner_(owner), wasted_bytes_(0) {
for (int i = kFirstCategory; i < kNumberOfCategories; i++) { for (int i = kFirstCategory; i < kNumberOfCategories; i++) {
categories_[i] = nullptr; categories_[i] = nullptr;
...@@ -3010,14 +3011,10 @@ bool FreeList::Allocate(size_t size_in_bytes) { ...@@ -3010,14 +3011,10 @@ bool FreeList::Allocate(size_t size_in_bytes) {
size_t FreeList::EvictFreeListItems(Page* page) { size_t FreeList::EvictFreeListItems(Page* page) {
size_t sum = 0; size_t sum = 0;
page->ForAllFreeListCategories([this, &sum](FreeListCategory* category) { page->ForAllFreeListCategories([this, &sum](FreeListCategory* category) {
// The category might have been already evicted
// if the page is an evacuation candidate.
if (category->type_ != kInvalidCategory) {
DCHECK_EQ(this, category->owner()); DCHECK_EQ(this, category->owner());
sum += category->available(); sum += category->available();
RemoveCategory(category); RemoveCategory(category);
category->Invalidate(); category->Reset();
}
}); });
return sum; return sum;
} }
......
...@@ -170,8 +170,6 @@ class FreeListCategory { ...@@ -170,8 +170,6 @@ class FreeListCategory {
next_ = nullptr; next_ = nullptr;
} }
void Invalidate();
void Reset(); void Reset();
void ResetStats() { Reset(); } void ResetStats() { Reset(); }
...@@ -2124,7 +2122,7 @@ class V8_EXPORT_PRIVATE PagedSpace ...@@ -2124,7 +2122,7 @@ class V8_EXPORT_PRIVATE PagedSpace
inline bool TryFreeLast(HeapObject* object, int object_size); inline bool TryFreeLast(HeapObject* object, int object_size);
void ResetFreeList() { free_list_.Reset(); } void ResetFreeList();
void PauseAllocationObservers() override; void PauseAllocationObservers() override;
......
...@@ -677,8 +677,8 @@ TEST(ShrinkPageToHighWaterMarkFreeSpaceEnd) { ...@@ -677,8 +677,8 @@ TEST(ShrinkPageToHighWaterMarkFreeSpaceEnd) {
// Reset space so high water mark is consistent. // Reset space so high water mark is consistent.
PagedSpace* old_space = CcTest::heap()->old_space(); PagedSpace* old_space = CcTest::heap()->old_space();
old_space->ResetFreeList();
old_space->EmptyAllocationInfo(); old_space->EmptyAllocationInfo();
old_space->ResetFreeList();
HeapObject* filler = HeapObject* filler =
HeapObject::FromAddress(array->address() + array->Size()); HeapObject::FromAddress(array->address() + array->Size());
...@@ -728,8 +728,8 @@ TEST(ShrinkPageToHighWaterMarkOneWordFiller) { ...@@ -728,8 +728,8 @@ TEST(ShrinkPageToHighWaterMarkOneWordFiller) {
// Reset space so high water mark and fillers are consistent. // Reset space so high water mark and fillers are consistent.
PagedSpace* old_space = CcTest::heap()->old_space(); PagedSpace* old_space = CcTest::heap()->old_space();
old_space->ResetFreeList();
old_space->EmptyAllocationInfo(); old_space->EmptyAllocationInfo();
old_space->ResetFreeList();
HeapObject* filler = HeapObject* filler =
HeapObject::FromAddress(array->address() + array->Size()); HeapObject::FromAddress(array->address() + array->Size());
...@@ -755,8 +755,8 @@ TEST(ShrinkPageToHighWaterMarkTwoWordFiller) { ...@@ -755,8 +755,8 @@ TEST(ShrinkPageToHighWaterMarkTwoWordFiller) {
// Reset space so high water mark and fillers are consistent. // Reset space so high water mark and fillers are consistent.
PagedSpace* old_space = CcTest::heap()->old_space(); PagedSpace* old_space = CcTest::heap()->old_space();
old_space->ResetFreeList();
old_space->EmptyAllocationInfo(); old_space->EmptyAllocationInfo();
old_space->ResetFreeList();
HeapObject* filler = HeapObject* filler =
HeapObject::FromAddress(array->address() + array->Size()); HeapObject::FromAddress(array->address() + array->Size());
......
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