org.graalvm.compiler.lir.framemap.FrameMap Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of compiler Show documentation
Show all versions of compiler Show documentation
The GraalVM compiler and the Graal-truffle optimizer.
/*
* Copyright (c) 2009, 2022, 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. Oracle designates this
* particular file as subject to the "Classpath" exception as provided
* by Oracle in the LICENSE file that accompanied this code.
*
* 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 org.graalvm.compiler.lir.framemap;
import static jdk.vm.ci.code.ValueUtil.asStackSlot;
import static jdk.vm.ci.code.ValueUtil.isStackSlot;
import org.graalvm.compiler.core.common.LIRKind;
import org.graalvm.compiler.core.common.NumUtil;
import org.graalvm.compiler.core.common.PermanentBailoutException;
import org.graalvm.compiler.debug.GraalError;
import jdk.vm.ci.code.Architecture;
import jdk.vm.ci.code.CallingConvention;
import jdk.vm.ci.code.CodeCacheProvider;
import jdk.vm.ci.code.CodeUtil;
import jdk.vm.ci.code.Register;
import jdk.vm.ci.code.RegisterConfig;
import jdk.vm.ci.code.RegisterSaveLayout;
import jdk.vm.ci.code.StackSlot;
import jdk.vm.ci.code.TargetDescription;
import jdk.vm.ci.meta.AllocatableValue;
import jdk.vm.ci.meta.ValueKind;
/**
* This class is used to build the stack frame layout for a compiled method. A {@link StackSlot} is
* used to index slots of the frame relative to the stack pointer. The frame size is only fixed
* after register allocation when all spill slots have been allocated. Both the outgoing argument
* area and the spill area can grow until then. Therefore, outgoing arguments are indexed from the
* stack pointer, while spill slots are indexed from the beginning of the frame (and the total frame
* size has to be added to get the actual offset from the stack pointer).
*/
public abstract class FrameMap {
private final TargetDescription target;
private final RegisterConfig registerConfig;
public interface ReferenceMapBuilderFactory {
ReferenceMapBuilder newReferenceMapBuilder(int totalFrameSize);
}
private final ReferenceMapBuilderFactory referenceMapFactory;
/**
* The final frame size, not including the size of the
* {@link Architecture#getReturnAddressSize() return address slot}. The value is only set after
* register allocation is complete, i.e., after all spill slots have been allocated.
*/
private int frameSize;
/**
* Initial size of the area occupied by spill slots and other stack-allocated memory blocks.
*/
protected int initialSpillSize;
/**
* Size of the area occupied by spill slots and other stack-allocated memory blocks.
*/
protected int spillSize;
/**
* Size of the area occupied by outgoing overflow arguments. This value is adjusted as calling
* conventions for outgoing calls are retrieved. On some platforms, there is a minimum outgoing
* size even if no overflow arguments are on the stack.
*/
protected int outgoingSize;
/**
* Determines if this frame has values on the stack for outgoing calls.
*/
protected boolean hasOutgoingStackArguments;
/**
* Records whether an offset to an incoming stack argument was ever returned by
* {@link #offsetForStackSlot(StackSlot)}.
*/
private boolean accessesCallerFrame;
/**
* Creates a new frame map for the specified method. The given registerConfig is optional, in
* case null is passed the default RegisterConfig from the CodeCacheProvider will be used.
*/
public FrameMap(CodeCacheProvider codeCache, RegisterConfig registerConfig, ReferenceMapBuilderFactory referenceMapFactory) {
this.target = codeCache.getTarget();
this.registerConfig = registerConfig == null ? codeCache.getRegisterConfig() : registerConfig;
this.frameSize = -1;
this.outgoingSize = codeCache.getMinimumOutgoingSize();
this.referenceMapFactory = referenceMapFactory;
}
public RegisterConfig getRegisterConfig() {
return registerConfig;
}
public TargetDescription getTarget() {
return target;
}
protected int returnAddressSize() {
return getTarget().arch.getReturnAddressSize();
}
/**
* Determines if an offset to an incoming stack argument was ever returned by
* {@link #offsetForStackSlot(StackSlot)}.
*/
public boolean accessesCallerFrame() {
return accessesCallerFrame;
}
/**
* Gets the allocated space in the compiled frame, not including any ABI required storage like
* the {@link Architecture#getReturnAddressSize() return address slot} or a saved frame pointer.
* {@link #totalFrameSize} includes all storage that is part of the frame.
*
* @return The size of the allocated frame space (in bytes).
*/
public int frameSize() {
assert frameSize != -1 : "frame size not computed yet";
return frameSize;
}
/**
* Determines if any space is used in the frame apart from the
* {@link Architecture#getReturnAddressSize() return address slot}.
*/
public boolean frameNeedsAllocating() {
int unalignedFrameSize = spillSize - returnAddressSize();
return hasOutgoingStackArguments || unalignedFrameSize != 0;
}
/**
* Gets the total frame size of the compiled frame, including any ABI required storage like the
* {@link Architecture#getReturnAddressSize() return address slot} or a saved frame pointer.
*
* @return The total size of the frame (in bytes).
*/
public abstract int totalFrameSize();
/**
* Gets the current size of this frame. This is the size that would be returned by
* {@link #frameSize()} if {@link #finish()} were called now.
*/
public abstract int currentFrameSize();
/**
* Aligns the given frame size to the stack alignment size and return the aligned size.
*
* @param size the initial frame size to be aligned
* @return the aligned frame size
*/
protected int alignFrameSize(int size) {
return NumUtil.roundUp(size, getTarget().stackAlignment);
}
/**
* Computes the final size of this frame. After this method has been called, methods that change
* the frame size cannot be called anymore, e.g., no more spill slots or outgoing arguments can
* be requested.
*/
public void finish() {
frameSize = currentFrameSize();
if (frameSize > getRegisterConfig().getMaximumFrameSize()) {
throw new PermanentBailoutException("Frame size (%d) exceeded maximum allowed frame size (%d).", frameSize, getRegisterConfig().getMaximumFrameSize());
}
}
/**
* Computes the offset of a stack slot relative to the frame register.
*
* @param slot a stack slot
* @return the offset of the stack slot
*/
public int offsetForStackSlot(StackSlot slot) {
if (slot.isInCallerFrame()) {
accessesCallerFrame = true;
}
return slot.getOffset(totalFrameSize());
}
/**
* Informs the frame map that the compiled code calls a particular method, which may need stack
* space for outgoing arguments.
*
* @param cc The calling convention for the called method.
*/
public void callsMethod(CallingConvention cc) {
reserveOutgoing(cc.getStackSize());
}
/**
* Reserves space for stack-based outgoing arguments.
*
* @param argsSize The amount of space (in bytes) to reserve for stack-based outgoing arguments.
*/
public void reserveOutgoing(int argsSize) {
assert frameSize == -1 : "frame size must not yet be fixed";
outgoingSize = Math.max(outgoingSize, argsSize);
hasOutgoingStackArguments = hasOutgoingStackArguments || argsSize > 0;
}
/**
* Returns the spill slot size for the given {@link ValueKind}. The default value is the size in
* bytes for the target architecture.
*
* @param kind the {@link ValueKind} to be stored in the spill slot.
* @return the size in bytes
*/
public int spillSlotSize(ValueKind> kind) {
return kind.getPlatformKind().getSizeInBytes();
}
/**
* Reserves a spill slot in the frame of the method being compiled. The returned slot is aligned
* on its natural alignment, i.e., an 8-byte spill slot is aligned at an 8-byte boundary, unless
* overridden by a subclass.
*
* @param kind The kind of the spill slot to be reserved.
* @return A spill slot denoting the reserved memory area.
*/
public StackSlot allocateSpillSlot(ValueKind> kind) {
assert frameSize == -1 : "frame size must not yet be fixed";
int size = spillSlotSize(kind);
spillSize = NumUtil.roundUp(spillSize + size, size);
return newStackSlot(kind);
}
private StackSlot newStackSlot(ValueKind> kind) {
return StackSlot.get(kind, -spillSize, true);
}
/**
* Reserves a contiguous and aligned range of memory in the frame of the method being compiled.
*
* @param sizeInBytes the number of bytes to reserve. Must be > 0.
* @param alignmentInBytes the required alignment of the memory. Must be > 0, a power of 2, and
* not exceed the {@link TargetDescription#stackAlignment OS stack frame alignment}.
* @return the first reserved stack slot (i.e., at the lowest address)
*/
public StackSlot allocateStackMemory(int sizeInBytes, int alignmentInBytes) {
assert frameSize == -1 : "frame size must not yet be fixed";
GraalError.guarantee(sizeInBytes > 0, "Invalid size");
GraalError.guarantee(alignmentInBytes > 0 && CodeUtil.isPowerOf2(alignmentInBytes), "Invalid alignment");
if (alignmentInBytes > getTarget().stackAlignment) {
throw GraalError.shouldNotReachHere("Stack memory alignment cannot be larger than OS alignment of stack frames: " + alignmentInBytes + " > " + getTarget().stackAlignment); // ExcludeFromJacocoGeneratedReport
}
spillSize = NumUtil.roundUp(spillSize + sizeInBytes, alignmentInBytes);
return newStackSlot(LIRKind.Illegal);
}
public ReferenceMapBuilder newReferenceMapBuilder() {
return referenceMapFactory.newReferenceMapBuilder(totalFrameSize());
}
public RegisterSaveLayout getRegisterSaveLayout(Register[] savedRegisters, AllocatableValue[] slots) {
Register[] filteredSavedRegisters = filterSavedRegisters(savedRegisters);
int total = 0;
for (int i = 0; i < filteredSavedRegisters.length; i++) {
if (filteredSavedRegisters[i] != null) {
total++;
}
}
Register[] keys = new Register[total];
int[] values = new int[total];
if (total != 0) {
int mapIndex = 0;
for (int i = 0; i < filteredSavedRegisters.length; i++) {
if (filteredSavedRegisters[i] != null) {
keys[mapIndex] = filteredSavedRegisters[i];
assert isStackSlot(slots[i]) : "not a StackSlot: " + slots[i];
StackSlot slot = asStackSlot(slots[i]);
values[mapIndex] = indexForStackSlot(slot);
mapIndex++;
}
}
assert mapIndex == total;
}
return new RegisterSaveLayout(keys, values);
}
/**
* {@link org.graalvm.compiler.lir.StandardOp.SaveRegistersOp} might save more registers than
* {@link RegisterSaveLayout} should know about so permit subclasses to filter the contents.
*/
protected Register[] filterSavedRegisters(Register[] savedRegisters) {
return savedRegisters;
}
/**
* Computes the index of a stack slot relative to slot 0. This is also the bit index of stack
* slots in the reference map.
*
* @param slot a stack slot
* @return the index of the stack slot
*/
private int indexForStackSlot(StackSlot slot) {
assert offsetForStackSlot(slot) % getTarget().wordSize == 0;
return offsetForStackSlot(slot) / getTarget().wordSize;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy