diff options
author | Andreas Kling <kling@serenityos.org> | 2021-09-11 16:44:40 +0200 |
---|---|---|
committer | Andreas Kling <kling@serenityos.org> | 2021-09-11 16:52:03 +0200 |
commit | c364520c2402b4b395f4e43fcfe3d4af4d546810 (patch) | |
tree | 0de3e6d60caea72831f3807b473b20d1bb2acdf5 /Userland/Libraries/LibJS/Heap/Heap.cpp | |
parent | 57371f7608b0d3251be03192e03a70b8b36ea3f2 (diff) | |
download | serenity-c364520c2402b4b395f4e43fcfe3d4af4d546810.zip |
LibJS+js+test-js: Add GC debug mode that keeps cells "alive" as zombies
This patch adds a `-z` option to js and test-js. When run in this mode,
garbage cells are never actually destroyed. We instead keep them around
in a special zombie state.
This allows us to validate that zombies don't get marked in future GC
scans (since there were not supposed to be any more references!) :^)
Cells get notified when they become a zombie (via did_become_zombie())
and this is used by WeakContainer cells to deregister themselves from
the heap.
Diffstat (limited to 'Userland/Libraries/LibJS/Heap/Heap.cpp')
-rw-r--r-- | Userland/Libraries/LibJS/Heap/Heap.cpp | 14 |
1 files changed, 13 insertions, 1 deletions
diff --git a/Userland/Libraries/LibJS/Heap/Heap.cpp b/Userland/Libraries/LibJS/Heap/Heap.cpp index 4c5b9292d0..4e7ccdf6fb 100644 --- a/Userland/Libraries/LibJS/Heap/Heap.cpp +++ b/Userland/Libraries/LibJS/Heap/Heap.cpp @@ -183,6 +183,13 @@ public: if (cell.is_marked()) return; dbgln_if(HEAP_DEBUG, " ! {}", &cell); + + if (cell.state() == Cell::State::Zombie) { + dbgln("BUG! Marking a zombie cell, {} @ {:p}", cell.class_name(), &cell); + cell.vm().dump_backtrace(); + VERIFY_NOT_REACHED(); + } + cell.set_marked(true); cell.visit_edges(*this); } @@ -223,7 +230,12 @@ void Heap::sweep_dead_cells(bool print_report, const Core::ElapsedTimer& measure dbgln_if(HEAP_DEBUG, " ~ {}", cell); if (should_store_swept_cells) swept_cells.append(cell); - block.deallocate(cell); + if (m_zombify_dead_cells) { + cell->set_state(Cell::State::Zombie); + cell->did_become_zombie(); + } else { + block.deallocate(cell); + } ++collected_cells; collected_cell_bytes += block.cell_size(); } else { |