/* * Copyright (c) 2010, 2012, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License version 2 only, as * published by the Free Software Foundation. * * This code is distributed in the hope that it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License * version 2 for more details (a copy is included in the LICENSE file that * accompanied this code). * * You should have received a copy of the GNU General Public License version * 2 along with this work; if not, write to the Free Software Foundation, * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. * * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA * or visit www.oracle.com if you need additional information or have any * questions. */ package com.sun.max.vm.heap.gcx; import static com.sun.max.vm.VMOptions.*; import com.sun.max.annotate.*; import com.sun.max.memory.*; import com.sun.max.unsafe.*; import com.sun.max.vm.*; import com.sun.max.vm.actor.holder.*; import com.sun.max.vm.heap.*; import com.sun.max.vm.layout.*; import com.sun.max.vm.reference.*; import com.sun.max.vm.runtime.*; /** * Fixed size marking stack for heap tracer. * Heap tracers specify an overflow handler to recover from overflow, * and a {@link CellVisitor} to be used when draining or flushing the stack. * * Currently, the marking stack drains itself when reaching end of capacity. * Overflows typically take place while the stack is draining. */ public class MarkingStack { private static final VMIntOption markingStackSizeOption = register(new VMIntOption("-XX:MarkingStackSize=", 16 * 1024, "Size of the marking stack in number of references."), MaxineVM.Phase.PRISTINE); abstract static class MarkingStackCellVisitor { abstract void visitPoppedCell(Pointer cell); abstract void visitFlushedCell(Pointer cell); } /** * Interface that a heap tracer has to implement to handle overflow of the marking stack. * The heap tracer is responsible for recovering from the overflow. */ interface OverflowHandler { void recoverFromOverflow(); } private Address base; private int last; private int drainThreshold; private int topIndex = 0; private Pointer draining = Pointer.zero(); private OverflowHandler overflowHandler; private MarkingStackCellVisitor drainingCellVisitor; /** * Sets an overflow handler. * @param handler an marking stack overflow handler. */ void setOverflowHandler(OverflowHandler handler) { overflowHandler = handler; } MarkingStack() { } void initialize(MarkingStackCellVisitor cellVisitor) { drainingCellVisitor = cellVisitor; // TODO (ld) a better solution might be to allocate this in the heap, outside of the covered area, as a reference array, // Root marking will skip it. // Same with the other GC data structures (i.e., rescan map and mark bitmap) final int length = markingStackSizeOption.getValue(); final int size = length << Word.widthValue().log2numberOfBytes; base = Memory.allocate(Size.fromInt(size)); if (base.isZero()) { MaxineVM.reportPristineMemoryFailure("marking stack", "allocate", Size.fromInt(size)); } last = length - 1; drainThreshold = (length * 2) / 3; } Size length() { return Size.fromInt(last + 1); } @INLINE final boolean isEmpty() { return topIndex == 0; } final void reset() { topIndex = 0; } void push(Pointer cell) { if (MaxineVM.isDebug()) { final Pointer origin = Layout.cellToOrigin(cell); Reference hubRef = Layout.readHubReference(origin); FatalError.check(!hubRef.isZero() && hubRef.toJava() instanceof Hub, "Invalid pointer pushed on marking stack"); } if (topIndex < last) { base.asPointer().setWord(topIndex++, cell); if (MaxineVM.isDebug() && Heap.logAllGC()) { Log.print("MarkingStack.push("); Log.print(cell); Log.println(")"); } return; } if (!draining.isZero()) { if (MaxineVM.isDebug() && Heap.logAllGC()) { Log.println("MarkingStack.push initiates overflow recovery"); } // We're already draining. So this is an overflow situation. Store the cell in the last slot of the stack (reserved for overflow). base.asPointer().setWord(topIndex++, cell); // Set draining back to false. The recovering will empty the marking stack. overflowHandler.recoverFromOverflow(); if (MaxineVM.isDebug() && Heap.logAllGC()) { Log.println("MarkingStack.push ends overflow recovery"); } } else { if (MaxineVM.isDebug() && Heap.logAllGC()) { Log.println("MarkingStack.push initiates draining"); } // Start draining with the cell requested to be pushed. draining = cell; drainingCellVisitor.visitPoppedCell(cell); // Drain further while we're at it. while (topIndex > drainThreshold) { draining = base.asPointer().getWord(--topIndex).asPointer(); drainingCellVisitor.visitPoppedCell(draining); } draining = Pointer.zero(); if (MaxineVM.isDebug() && Heap.logAllGC()) { Log.println("MarkingStack.push ends draining"); } } } void drain() { if (MaxineVM.isDebug() && Heap.logAllGC()) { Log.println("MarkingStack begin draining"); } if (MaxineVM.isDebug()) { FatalError.check(draining.isZero(), "Cannot drain an already draining marking stack"); } while (topIndex > 0) { draining = base.asPointer().getWord(--topIndex).asPointer(); drainingCellVisitor.visitPoppedCell(draining); } draining = Pointer.zero(); if (MaxineVM.isDebug() && Heap.logAllGC()) { Log.println("MarkingStack ends draining"); } } void flush() { if (!draining.isZero()) { drainingCellVisitor.visitFlushedCell(draining); draining = Pointer.zero(); } while (topIndex > 0) { drainingCellVisitor.visitFlushedCell(base.asPointer().getWord(--topIndex).asPointer()); } if (MaxineVM.isDebug() && Heap.logAllGC()) { Log.println("MarkingStack flushed"); } } void print() { if (!draining.isZero()) { Log.print(" "); Log.print(draining); Log.println(" [d]"); } int index = topIndex; while (index > 0) { Log.print(" "); Log.println(base.asPointer().getWord(--index).asPointer()); } } }