Skip to content

Commit

Permalink
Move rest CardTable functions to AlignedHeapSegmentBase
Browse files Browse the repository at this point in the history
Differential Revision: D65638010
  • Loading branch information
lavenzg authored and facebook-github-bot committed Nov 21, 2024
1 parent 1d8d1e9 commit a18b1b5
Show file tree
Hide file tree
Showing 2 changed files with 46 additions and 42 deletions.
65 changes: 34 additions & 31 deletions include/hermes/VM/AlignedHeapSegment.h
Original file line number Diff line number Diff line change
Expand Up @@ -218,6 +218,40 @@ class AlignedHeapSegment {
return contents()->cardTable_;
}

/// Given a \p cell lives in the memory region of some valid segment \c s,
/// returns a pointer to the CardTable covering the segment containing the
/// cell. Note that this takes a GCCell pointer in order to correctly get
/// the segment starting address for JumboHeapSegment.
///
/// \pre There exists a currently alive heap in which \p cell is allocated.
static CardTable *cardTableCovering(const GCCell *cell) {
return &contents(alignedStorageStart(cell))->cardTable_;
}

/// Find the head of the first cell that extends into the card at index
/// \p cardIdx.
/// \return A cell such that
/// cell <= indexToAddress(cardIdx) < cell->nextCell().
GCCell *getFirstCellHead(size_t cardIdx) {
CardTable &cards = cardTable();
GCCell *cell = cards.firstObjForCard(cardIdx);
assert(cell->isValid() && "Object head doesn't point to a valid object");
return cell;
}

/// Record the head of this cell so it can be found by the card scanner.
static void setCellHead(const GCCell *cellStart, const size_t sz) {
const char *start = reinterpret_cast<const char *>(cellStart);
const char *end = start + sz;
CardTable *cards = cardTableCovering(cellStart);
auto boundary = cards->nextBoundary(start);
// If this object crosses a card boundary, then update boundaries
// appropriately.
if (boundary.address() < end) {
cards->updateBoundaries(&boundary, start, end);
}
}

/// Return a reference to the mark bit array covering the memory region
/// managed by this segment.
Contents::MarkBitArray &markBitArray() const {
Expand Down Expand Up @@ -410,15 +444,6 @@ class FixedSizeHeapSegment : public AlignedHeapSegment {
/// \pre There exists a currently alive heap that claims to contain \c ptr.
inline static CardTable *cardTableCovering(const void *ptr);

/// Find the head of the first cell that extends into the card at index
/// \p cardIdx.
/// \return A cell such that
/// cell <= indexToAddress(cardIdx) < cell->nextCell().
inline GCCell *getFirstCellHead(size_t cardIdx);

/// Record the head of this cell so it can be found by the card scanner.
static inline void setCellHead(const GCCell *start, const size_t sz);

/// The largest size the allocation region of an aligned heap segment could
/// be. This is a static override of AlignedHeapSegment::maxSize().
inline static constexpr size_t maxSize();
Expand Down Expand Up @@ -556,28 +581,6 @@ AllocResult FixedSizeHeapSegment::alloc(uint32_t size) {
return {cell, true};
}

GCCell *FixedSizeHeapSegment::getFirstCellHead(size_t cardIdx) {
CardTable &cards = cardTable();
GCCell *cell = cards.firstObjForCard(cardIdx);
assert(cell->isValid() && "Object head doesn't point to a valid object");
return cell;
}

/* static */
void FixedSizeHeapSegment::setCellHead(
const GCCell *cellStart,
const size_t sz) {
const char *start = reinterpret_cast<const char *>(cellStart);
const char *end = start + sz;
CardTable *cards = cardTableCovering(start);
auto boundary = cards->nextBoundary(start);
// If this object crosses a card boundary, then update boundaries
// appropriately.
if (boundary.address() < end) {
cards->updateBoundaries(&boundary, start, end);
}
}

/* static */ CardTable *FixedSizeHeapSegment::cardTableCovering(
const void *ptr) {
return &FixedSizeHeapSegment::contents(storageStart(ptr))->cardTable_;
Expand Down
23 changes: 12 additions & 11 deletions lib/VM/gcs/HadesGC.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -80,7 +80,7 @@ void HadesGC::OldGen::addCellToFreelist(
sz >= sizeof(FreelistCell) &&
"Cannot construct a FreelistCell into an allocation in the OG");
FreelistCell *newFreeCell = constructCell<FreelistCell>(addr, sz);
FixedSizeHeapSegment::setCellHead(static_cast<GCCell *>(addr), sz);
AlignedHeapSegment::setCellHead(static_cast<GCCell *>(addr), sz);
addCellToFreelist(newFreeCell, segBucket);
}

Expand Down Expand Up @@ -124,8 +124,9 @@ void HadesGC::OldGen::addCellToFreelistFromSweep(
size_t newCellSize = freeRangeEnd - freeRangeStart;
// While coalescing, sweeping may generate new cells, so make sure the cell
// head is updated.
FixedSizeHeapSegment::setCellHead(
reinterpret_cast<GCCell *>(freeRangeStart), newCellSize);
if (setHead)
AlignedHeapSegment::setCellHead(
reinterpret_cast<GCCell *>(freeRangeStart), newCellSize);
auto *newCell = constructCell<FreelistCell>(freeRangeStart, newCellSize);
// Get the size bucket for the cell being added;
const uint32_t bucket = getFreelistBucket(newCellSize);
Expand Down Expand Up @@ -217,7 +218,7 @@ GCCell *HadesGC::OldGen::FreelistCell::carve(uint32_t sz) {
char *newCellAddress = reinterpret_cast<char *>(this) + finalSize;
GCCell *const newCell = reinterpret_cast<GCCell *>(newCellAddress);
setSizeFromGC(finalSize);
FixedSizeHeapSegment::setCellHead(newCell, sz);
AlignedHeapSegment::setCellHead(newCell, sz);
return newCell;
}

Expand Down Expand Up @@ -415,7 +416,7 @@ class HadesGC::EvacAcceptor final : public RootAndSlotAcceptor,
assert(currentCell_ && "currentCell_ must be set for compaction");
// If a compaction is about to take place, dirty the card for any newly
// evacuated cells, since the marker may miss them.
FixedSizeHeapSegment::cardTableCovering(heapLoc)
AlignedHeapSegment::cardTableCovering(currentCell_)
->dirtyCardForAddressInLargeObj(heapLoc);
}
return ptr;
Expand All @@ -434,7 +435,7 @@ class HadesGC::EvacAcceptor final : public RootAndSlotAcceptor,
assert(currentCell_ && "currentCell_ must be set for compaction");
// If a compaction is about to take place, dirty the card for any newly
// evacuated cells, since the marker may miss them.
FixedSizeHeapSegment::cardTableCovering(heapLoc)
AlignedHeapSegment::cardTableCovering(currentCell_)
->dirtyCardForAddressInLargeObj(heapLoc);
}
return cptr;
Expand Down Expand Up @@ -658,7 +659,7 @@ class HadesGC::MarkAcceptor final : public RootAndSlotAcceptor {
if (gc.compactee_.contains(cell) && !gc.compactee_.contains(heapLoc)) {
// This is a pointer in the heap pointing into the compactee, dirty the
// corresponding card.
FixedSizeHeapSegment::cardTableCovering(heapLoc)
AlignedHeapSegment::cardTableCovering(currentCell_)
->dirtyCardForAddressInLargeObj(heapLoc);
}
if (AlignedHeapSegment::getCellMarkBit(cell)) {
Expand Down Expand Up @@ -1104,7 +1105,7 @@ bool HadesGC::OldGen::sweepNext(bool backgroundThread) {
assert(
!AlignedHeapSegment::getCellMarkBit(newCell) &&
"Trimmed space cannot be marked");
FixedSizeHeapSegment::setCellHead(newCell, trimmableBytes);
AlignedHeapSegment::setCellHead(newCell, trimmableBytes);
#ifndef NDEBUG
sweepIterator_.trimmedBytes += trimmableBytes;
#endif
Expand Down Expand Up @@ -2247,7 +2248,7 @@ GCCell *HadesGC::OldGen::alloc(uint32_t sz) {
"A newly created segment should always be able to allocate");
// Set the cell head for any successful alloc, so that write barriers can
// move from dirty cards to the head of the object.
FixedSizeHeapSegment::setCellHead(static_cast<GCCell *>(res.ptr), sz);
AlignedHeapSegment::setCellHead(static_cast<GCCell *>(res.ptr), sz);
// Add the segment to segments_ and add the remainder of the segment to the
// free list.
addSegment(std::move(seg.get()));
Expand Down Expand Up @@ -2636,7 +2637,7 @@ bool HadesGC::promoteYoungGenToOldGen() {
// going into OG. This could be done at allocation time, but at a cost
// to YG alloc times for a case that might not come up.
forAllObjsInSegment(youngGen_, [](GCCell *cell) {
FixedSizeHeapSegment::setCellHead(cell, cell->getAllocatedSize());
AlignedHeapSegment::setCellHead(cell, cell->getAllocatedSize());
});
// It is important that this operation is just a move of pointers to
// segments. The addresses have to stay the same or else it would
Expand Down Expand Up @@ -3128,7 +3129,7 @@ void HadesGC::verifyCardTable() {
gc.compactee_.evacContains(valuePtr);
if (!gc.inYoungGen(locPtr) &&
(gc.inYoungGen(valuePtr) || crossRegionCompacteePtr)) {
assert(FixedSizeHeapSegment::cardTableCovering(locPtr)
assert(AlignedHeapSegment::cardTableCovering(currentCell)
->isCardForAddressDirtyInLargeObj(locPtr));
}
}
Expand Down

0 comments on commit a18b1b5

Please sign in to comment.