mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2024-12-12 13:30:51 +00:00
24a071b8c5
Summary: When -mno-odd-spreg is in effect, 32-bit floating point values are not permitted in odd FPU registers. The option also prohibits 32-bit and 64-bit floating point comparison results from being written to odd registers. This option has three purposes: * It allows support for certain MIPS implementations such as loongson-3a that do not allow the use of odd registers for single precision arithmetic. * When using -mfpxx, -mno-odd-spreg is the default and this allows us to statically check that code is compliant with the O32 FPXX ABI since mtc1/mfc1 instructions to/from odd registers are guaranteed not to appear for any reason. Once this has been established, the user can then re-enable -modd-spreg to regain the use of all 32 single-precision registers. * When using -mfp64 and -mno-odd-spreg together, an O32 extension named O32 FP64A is used as the ABI. This is intended to provide almost all functionality of an FR=1 processor but can also be executed on a FR=0 core with the assistance of a hardware compatibility mode which emulates FR=0 behaviour on an FR=1 processor. * Added '.module oddspreg' and '.module nooddspreg' each of which update the .MIPS.abiflags section appropriately * Moved setFpABI() call inside emitDirectiveModuleFP() so that the caller doesn't have to remember to do it. * MipsABIFlags now calculates the flags1 and flags2 member on demand rather than trying to maintain them in the same format they will be emitted in. There is one portion of the -mfp64 and -mno-odd-spreg combination that is not implemented yet. Moves to/from odd-numbered double-precision registers must not use mtc1. I will fix this in a follow-up. Differential Revision: http://reviews.llvm.org/D4383 git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@212717 91177308-0d34-0410-b5e6-96231b3b80d8
61 lines
2.1 KiB
C++
61 lines
2.1 KiB
C++
//===-- MipsABIFlagsSection.cpp - Mips ELF ABI Flags Section ---*- C++ -*--===//
|
|
//
|
|
// The LLVM Compiler Infrastructure
|
|
//
|
|
// This file is distributed under the University of Illinois Open Source
|
|
// License. See LICENSE.TXT for details.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "MipsABIFlagsSection.h"
|
|
|
|
using namespace llvm;
|
|
|
|
uint8_t MipsABIFlagsSection::getFpABIValue() {
|
|
switch (FpABI) {
|
|
case FpABIKind::ANY:
|
|
return Val_GNU_MIPS_ABI_FP_ANY;
|
|
case FpABIKind::XX:
|
|
return Val_GNU_MIPS_ABI_FP_XX;
|
|
case FpABIKind::S32:
|
|
return Val_GNU_MIPS_ABI_FP_DOUBLE;
|
|
case FpABIKind::S64:
|
|
if (Is32BitABI)
|
|
return OddSPReg ? Val_GNU_MIPS_ABI_FP_64 : Val_GNU_MIPS_ABI_FP_64A;
|
|
return Val_GNU_MIPS_ABI_FP_DOUBLE;
|
|
default:
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
StringRef MipsABIFlagsSection::getFpABIString(FpABIKind Value) {
|
|
switch (Value) {
|
|
case FpABIKind::XX:
|
|
return "xx";
|
|
case FpABIKind::S32:
|
|
return "32";
|
|
case FpABIKind::S64:
|
|
return "64";
|
|
default:
|
|
llvm_unreachable("unsupported fp abi value");
|
|
}
|
|
}
|
|
|
|
namespace llvm {
|
|
MCStreamer &operator<<(MCStreamer &OS, MipsABIFlagsSection &ABIFlagsSection) {
|
|
// Write out a Elf_Internal_ABIFlags_v0 struct
|
|
OS.EmitIntValue(ABIFlagsSection.getVersionValue(), 2); // version
|
|
OS.EmitIntValue(ABIFlagsSection.getISALevelValue(), 1); // isa_level
|
|
OS.EmitIntValue(ABIFlagsSection.getISARevisionValue(), 1); // isa_rev
|
|
OS.EmitIntValue(ABIFlagsSection.getGPRSizeValue(), 1); // gpr_size
|
|
OS.EmitIntValue(ABIFlagsSection.getCPR1SizeValue(), 1); // cpr1_size
|
|
OS.EmitIntValue(ABIFlagsSection.getCPR2SizeValue(), 1); // cpr2_size
|
|
OS.EmitIntValue(ABIFlagsSection.getFpABIValue(), 1); // fp_abi
|
|
OS.EmitIntValue(ABIFlagsSection.getISAExtensionSetValue(), 4); // isa_ext
|
|
OS.EmitIntValue(ABIFlagsSection.getASESetValue(), 4); // ases
|
|
OS.EmitIntValue(ABIFlagsSection.getFlags1Value(), 4); // flags1
|
|
OS.EmitIntValue(ABIFlagsSection.getFlags2Value(), 4); // flags2
|
|
return OS;
|
|
}
|
|
}
|