Depends on D152879. Specification PR: riscv-non-isa/rvv-intrinsic-doc#226 This patch adds variant of `vfadd` that models the rounding mode control. The added variant has suffix `_rm` appended to differentiate from the existing ones that does not alternate `frm` and uses whatever is inside. The value `7` is used to indicate no rounding mode change. Reusing the semantic from the rounding mode encoding for scalar floating-point instructions. Additional data member `HasFRMRoundModeOp` is added so we can append `_rm` suffix for the fadd variants that models rounding mode control. Additional data member `IsRVVFixedPoint` is added so we can define pseudo instructions with rounding mode operand and distinguish the instructions between fixed-point and floating-point. Reviewed By: craig.topper, kito-cheng Differential Revision: https://reviews.llvm.org/D152996
136 lines
4.3 KiB
C++
136 lines
4.3 KiB
C++
//===-- RISCVInsertReadWriteCSR.cpp - Insert Read/Write of RISC-V CSR -----===//
|
|
//
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
// This file implements the machine function pass to insert read/write of CSR-s
|
|
// of the RISC-V instructions.
|
|
//
|
|
// Currently the pass implements naive insertion of a write to vxrm before an
|
|
// RVV fixed-point instruction.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "MCTargetDesc/RISCVBaseInfo.h"
|
|
#include "RISCV.h"
|
|
#include "RISCVSubtarget.h"
|
|
#include "llvm/CodeGen/MachineFunctionPass.h"
|
|
using namespace llvm;
|
|
|
|
#define DEBUG_TYPE "riscv-insert-read-write-csr"
|
|
#define RISCV_INSERT_READ_WRITE_CSR_NAME "RISC-V Insert Read/Write CSR Pass"
|
|
|
|
namespace {
|
|
|
|
class RISCVInsertReadWriteCSR : public MachineFunctionPass {
|
|
const TargetInstrInfo *TII;
|
|
|
|
public:
|
|
static char ID;
|
|
|
|
RISCVInsertReadWriteCSR() : MachineFunctionPass(ID) {
|
|
initializeRISCVInsertReadWriteCSRPass(*PassRegistry::getPassRegistry());
|
|
}
|
|
|
|
bool runOnMachineFunction(MachineFunction &MF) override;
|
|
|
|
void getAnalysisUsage(AnalysisUsage &AU) const override {
|
|
AU.setPreservesCFG();
|
|
MachineFunctionPass::getAnalysisUsage(AU);
|
|
}
|
|
|
|
StringRef getPassName() const override {
|
|
return RISCV_INSERT_READ_WRITE_CSR_NAME;
|
|
}
|
|
|
|
private:
|
|
bool emitWriteRoundingMode(MachineBasicBlock &MBB);
|
|
};
|
|
|
|
} // end anonymous namespace
|
|
|
|
char RISCVInsertReadWriteCSR::ID = 0;
|
|
|
|
INITIALIZE_PASS(RISCVInsertReadWriteCSR, DEBUG_TYPE,
|
|
RISCV_INSERT_READ_WRITE_CSR_NAME, false, false)
|
|
|
|
// Returns the index to the rounding mode immediate value if any, otherwise the
|
|
// function will return None.
|
|
static std::optional<unsigned> getRoundModeIdx(const MachineInstr &MI) {
|
|
uint64_t TSFlags = MI.getDesc().TSFlags;
|
|
if (!RISCVII::hasRoundModeOp(TSFlags))
|
|
return std::nullopt;
|
|
|
|
// The operand order
|
|
// -------------------------------------
|
|
// | n-1 (if any) | n-2 | n-3 | n-4 |
|
|
// | policy | sew | vl | rm |
|
|
// -------------------------------------
|
|
return MI.getNumExplicitOperands() - RISCVII::hasVecPolicyOp(TSFlags) - 3;
|
|
}
|
|
|
|
// This function inserts a write to vxrm when encountering an RVV fixed-point
|
|
// instruction.
|
|
bool RISCVInsertReadWriteCSR::emitWriteRoundingMode(MachineBasicBlock &MBB) {
|
|
bool Changed = false;
|
|
for (MachineInstr &MI : MBB) {
|
|
if (auto RoundModeIdx = getRoundModeIdx(MI)) {
|
|
if (RISCVII::usesVXRM(MI.getDesc().TSFlags)) {
|
|
unsigned VXRMImm = MI.getOperand(*RoundModeIdx).getImm();
|
|
|
|
Changed = true;
|
|
|
|
BuildMI(MBB, MI, MI.getDebugLoc(), TII->get(RISCV::WriteVXRMImm))
|
|
.addImm(VXRMImm);
|
|
MI.addOperand(MachineOperand::CreateReg(RISCV::VXRM, /*IsDef*/ false,
|
|
/*IsImp*/ true));
|
|
} else { // FRM
|
|
unsigned FRMImm = MI.getOperand(*RoundModeIdx).getImm();
|
|
|
|
// The value is a hint to this pass to not alter the frm value.
|
|
if (FRMImm == RISCVFPRndMode::DYN)
|
|
continue;
|
|
|
|
Changed = true;
|
|
|
|
// Save
|
|
MachineRegisterInfo *MRI = &MBB.getParent()->getRegInfo();
|
|
Register SavedFRM = MRI->createVirtualRegister(&RISCV::GPRRegClass);
|
|
BuildMI(MBB, MI, MI.getDebugLoc(), TII->get(RISCV::SwapFRMImm),
|
|
SavedFRM)
|
|
.addImm(FRMImm);
|
|
MI.addOperand(MachineOperand::CreateReg(RISCV::FRM, /*IsDef*/ false,
|
|
/*IsImp*/ true));
|
|
// Restore
|
|
MachineInstrBuilder MIB =
|
|
BuildMI(*MBB.getParent(), {}, TII->get(RISCV::WriteFRM))
|
|
.addReg(SavedFRM);
|
|
MBB.insertAfter(MI, MIB);
|
|
}
|
|
}
|
|
}
|
|
return Changed;
|
|
}
|
|
|
|
bool RISCVInsertReadWriteCSR::runOnMachineFunction(MachineFunction &MF) {
|
|
// Skip if the vector extension is not enabled.
|
|
const RISCVSubtarget &ST = MF.getSubtarget<RISCVSubtarget>();
|
|
if (!ST.hasVInstructions())
|
|
return false;
|
|
|
|
TII = ST.getInstrInfo();
|
|
|
|
bool Changed = false;
|
|
|
|
for (MachineBasicBlock &MBB : MF)
|
|
Changed |= emitWriteRoundingMode(MBB);
|
|
|
|
return Changed;
|
|
}
|
|
|
|
FunctionPass *llvm::createRISCVInsertReadWriteCSRPass() {
|
|
return new RISCVInsertReadWriteCSR();
|
|
}
|