Chromium Code Reviews| Index: src/heap/remembered-set.cc |
| diff --git a/src/heap/remembered-set.cc b/src/heap/remembered-set.cc |
| index 403c99b05712ec53d2920b40e5ee6ff9e3a2f84c..260f268ccde9bfee03e30a6e9626742d2d5a827e 100644 |
| --- a/src/heap/remembered-set.cc |
| +++ b/src/heap/remembered-set.cc |
| @@ -9,6 +9,7 @@ |
| #include "src/heap/slot-set.h" |
| #include "src/heap/spaces.h" |
| #include "src/heap/store-buffer.h" |
| +#include "src/macro-assembler.h" |
| namespace v8 { |
| namespace internal { |
| @@ -16,16 +17,38 @@ namespace internal { |
| template <PointerDirection direction> |
| void RememberedSet<direction>::ClearInvalidSlots(Heap* heap) { |
| STATIC_ASSERT(direction == OLD_TO_NEW); |
| - PageIterator it(heap->old_space()); |
| - MemoryChunk* chunk; |
| - while (it.has_next()) { |
| - chunk = it.next(); |
| - SlotSet* slots = GetSlotSet(chunk); |
| - if (slots != nullptr) { |
| - slots->Iterate([heap, chunk](Address addr) { |
| - Object** slot = reinterpret_cast<Object**>(addr); |
| - return IsValidSlot(heap, chunk, slot) ? KEEP_SLOT : REMOVE_SLOT; |
| - }); |
| + { |
| + PageIterator it(heap->old_space()); |
| + MemoryChunk* chunk; |
| + while (it.has_next()) { |
| + chunk = it.next(); |
| + { |
| + SlotSet* slots = GetSlotSet(chunk); |
| + if (slots != nullptr) { |
| + slots->Iterate([heap, chunk](Address addr) { |
| + Object** slot = reinterpret_cast<Object**>(addr); |
| + return IsValidSlot(heap, chunk, slot) ? KEEP_SLOT : REMOVE_SLOT; |
| + }); |
| + } |
| + } |
| + } |
| + } |
| + { |
| + PageIterator it(heap->code_space()); |
| + MemoryChunk* chunk; |
| + while (it.has_next()) { |
| + chunk = it.next(); |
| + TypedSlotSet* slots = GetTypedSlotSet(chunk); |
| + if (slots != nullptr) { |
| + slots->Iterate( |
| + [heap, chunk](SlotType type, Address host_addr, Address addr) { |
| + if (IsValidTypedSlot(heap, chunk, type, host_addr, addr)) { |
|
ulan
2016/06/08 14:37:00
Just use Marking::IsBlack(Marking::MarkBitFrom(hos
ahaas
2016/06/09 10:34:32
Done.
|
| + return KEEP_SLOT; |
| + } else { |
| + return REMOVE_SLOT; |
| + } |
| + }); |
| + } |
| } |
| } |
| } |
| @@ -69,6 +92,51 @@ bool RememberedSet<direction>::IsValidSlot(Heap* heap, MemoryChunk* chunk, |
| chunk, reinterpret_cast<Address>(slot)); |
| } |
| +template <PointerDirection direction> |
| +bool RememberedSet<direction>::IsValidTypedSlot(Heap* heap, MemoryChunk* chunk, |
| + SlotType slot_type, |
| + Address host_addr, |
| + Address slot) { |
| + STATIC_ASSERT(direction == OLD_TO_NEW); |
| + Object* object; |
| + Code* code = reinterpret_cast<Code*>(host_addr); |
| + switch (slot_type) { |
| + case CODE_TARGET_SLOT: { |
| + RelocInfo rinfo(heap->isolate(), slot, RelocInfo::CODE_TARGET, 0, code); |
| + object = rinfo.target_object(); |
| + break; |
| + } |
| + case CELL_TARGET_SLOT: { |
| + RelocInfo rinfo(heap->isolate(), slot, RelocInfo::CELL, 0, code); |
| + object = rinfo.target_object(); |
| + break; |
| + } |
| + case EMBEDDED_OBJECT_SLOT: { |
| + RelocInfo rinfo(heap->isolate(), slot, RelocInfo::EMBEDDED_OBJECT, 0, |
| + code); |
| + object = rinfo.target_object(); |
| + break; |
| + } |
| + case OBJECT_SLOT: { |
| + object = *reinterpret_cast<Object**>(slot); |
| + break; |
| + } |
| + default: |
| + printf("Slot type: %d\n", slot_type); |
| + UNREACHABLE(); |
| + break; |
| + } |
| + |
| + if (!heap->InNewSpace(object)) { |
| + return false; |
| + } |
| + HeapObject* heap_object = HeapObject::cast(object); |
| + // If the target object is not black, the source slot must be part |
| + // of a non-black (dead) object. |
| + return Marking::IsBlack(Marking::MarkBitFrom(heap_object)) && |
| + Marking::IsBlack(Marking::MarkBitFrom(host_addr)); |
| +} |
| + |
| template void RememberedSet<OLD_TO_NEW>::ClearInvalidSlots(Heap* heap); |
| template void RememberedSet<OLD_TO_NEW>::VerifyValidSlots(Heap* heap); |
| template void RememberedSet<OLD_TO_OLD>::VerifyValidSlots(Heap* heap); |