blob: 2ff74388f70f81909c23625b984464656ef9185e [file] [log] [blame]
/*
* Copyright (c) 2015, 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 org.graalvm.compiler.core.amd64;
import static org.graalvm.compiler.asm.amd64.AMD64Assembler.OperandSize.QWORD;
import static org.graalvm.compiler.asm.amd64.AMD64Assembler.OperandSize.WORD;
import java.util.HashMap;
import java.util.Map;
import org.graalvm.compiler.core.common.LIRKind;
import org.graalvm.compiler.lir.VirtualStackSlot;
import org.graalvm.compiler.lir.amd64.AMD64LIRInstruction;
import org.graalvm.compiler.lir.amd64.AMD64Move.AMD64PushPopStackMove;
import org.graalvm.compiler.lir.framemap.FrameMapBuilder;
import org.graalvm.compiler.lir.gen.LIRGeneratorTool.MoveFactory;
import jdk.vm.ci.amd64.AMD64Kind;
import jdk.vm.ci.code.Architecture;
import jdk.vm.ci.code.Register;
import jdk.vm.ci.code.RegisterArray;
import jdk.vm.ci.code.RegisterConfig;
import jdk.vm.ci.meta.AllocatableValue;
import jdk.vm.ci.meta.PlatformKind;
public abstract class AMD64MoveFactoryBase implements MoveFactory {
private final BackupSlotProvider backupSlotProvider;
private static class RegisterBackupPair {
public final Register register;
public final VirtualStackSlot backupSlot;
RegisterBackupPair(Register register, VirtualStackSlot backupSlot) {
this.register = register;
this.backupSlot = backupSlot;
}
}
public static final class BackupSlotProvider {
private final FrameMapBuilder frameMapBuilder;
private Map<PlatformKind.Key, RegisterBackupPair> categorized;
public BackupSlotProvider(FrameMapBuilder frameMapBuilder) {
this.frameMapBuilder = frameMapBuilder;
}
protected RegisterBackupPair getScratchRegister(PlatformKind kind) {
PlatformKind.Key key = kind.getKey();
if (categorized == null) {
categorized = new HashMap<>();
} else if (categorized.containsKey(key)) {
return categorized.get(key);
}
RegisterConfig registerConfig = frameMapBuilder.getRegisterConfig();
RegisterArray availableRegister = registerConfig.filterAllocatableRegisters(kind, registerConfig.getAllocatableRegisters());
assert availableRegister != null && availableRegister.size() > 1;
Register scratchRegister = availableRegister.get(0);
Architecture arch = frameMapBuilder.getCodeCache().getTarget().arch;
LIRKind largestKind = LIRKind.value(arch.getLargestStorableKind(scratchRegister.getRegisterCategory()));
VirtualStackSlot backupSlot = frameMapBuilder.allocateSpillSlot(largestKind);
RegisterBackupPair value = new RegisterBackupPair(scratchRegister, backupSlot);
categorized.put(key, value);
return value;
}
}
public AMD64MoveFactoryBase(BackupSlotProvider backupSlotProvider) {
this.backupSlotProvider = backupSlotProvider;
}
@Override
public final AMD64LIRInstruction createStackMove(AllocatableValue result, AllocatableValue input) {
AMD64Kind kind = (AMD64Kind) result.getPlatformKind();
switch (kind.getSizeInBytes()) {
case 2:
return new AMD64PushPopStackMove(WORD, result, input);
case 8:
return new AMD64PushPopStackMove(QWORD, result, input);
default:
RegisterBackupPair backup = backupSlotProvider.getScratchRegister(input.getPlatformKind());
Register scratchRegister = backup.register;
VirtualStackSlot backupSlot = backup.backupSlot;
return createStackMove(result, input, scratchRegister, backupSlot);
}
}
public abstract AMD64LIRInstruction createStackMove(AllocatableValue result, AllocatableValue input, Register scratchRegister, AllocatableValue backupSlot);
}