forked from llvm/llvm-project
-
Notifications
You must be signed in to change notification settings - Fork 57
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
[AMDGPU] Split vgpr regalloc pipeline (llvm#93526)
Allocating wwm-registers and regular VGPR operands together imposes many challenges in the way the registers are reused during allocation. There are times when regalloc reuses the registers of regular VGPRs operations for wwm-operations in a small range leading to unwantedly clobbering their inactive lanes causing correctness issues which are hard to trace. This patch splits the VGPR allocation pipeline further to allocate wwm-registers first and the regular VGPR operands in a separate pipeline. The splitting would ensure that the physical registers used for wwm allocations won't taken part in the next allocation pipeline to avoid any such clobbering. Change-Id: Ib2c5b9b53944bf78709465a9d1786d129434ce40
- Loading branch information
Showing
88 changed files
with
10,423 additions
and
10,878 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,97 @@ | ||
//===-- AMDGPUReserveWWMRegs.cpp - Add WWM Regs to reserved regs list -----===// | ||
// | ||
// 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 | ||
// | ||
//===----------------------------------------------------------------------===// | ||
// | ||
/// \file | ||
/// This pass should be invoked at the end of wwm-regalloc pipeline. | ||
/// It identifies the WWM regs allocated during this pipeline and add | ||
/// them to the list of reserved registers so that they won't be available for | ||
/// regular VGPR allocation in the subsequent regalloc pipeline. | ||
// | ||
//===----------------------------------------------------------------------===// | ||
|
||
#include "AMDGPU.h" | ||
#include "GCNSubtarget.h" | ||
#include "MCTargetDesc/AMDGPUMCTargetDesc.h" | ||
#include "SIMachineFunctionInfo.h" | ||
#include "llvm/CodeGen/LiveIntervals.h" | ||
#include "llvm/CodeGen/MachineFunctionPass.h" | ||
#include "llvm/CodeGen/VirtRegMap.h" | ||
#include "llvm/InitializePasses.h" | ||
|
||
using namespace llvm; | ||
|
||
#define DEBUG_TYPE "amdgpu-reserve-wwm-regs" | ||
|
||
namespace { | ||
|
||
class AMDGPUReserveWWMRegs : public MachineFunctionPass { | ||
public: | ||
static char ID; | ||
|
||
AMDGPUReserveWWMRegs() : MachineFunctionPass(ID) { | ||
initializeAMDGPUReserveWWMRegsPass(*PassRegistry::getPassRegistry()); | ||
} | ||
|
||
bool runOnMachineFunction(MachineFunction &MF) override; | ||
|
||
StringRef getPassName() const override { | ||
return "AMDGPU Reserve WWM Registers"; | ||
} | ||
|
||
void getAnalysisUsage(AnalysisUsage &AU) const override { | ||
AU.setPreservesAll(); | ||
MachineFunctionPass::getAnalysisUsage(AU); | ||
} | ||
}; | ||
|
||
} // End anonymous namespace. | ||
|
||
INITIALIZE_PASS(AMDGPUReserveWWMRegs, DEBUG_TYPE, | ||
"AMDGPU Reserve WWM Registers", false, false) | ||
|
||
char AMDGPUReserveWWMRegs::ID = 0; | ||
|
||
char &llvm::AMDGPUReserveWWMRegsID = AMDGPUReserveWWMRegs::ID; | ||
|
||
bool AMDGPUReserveWWMRegs::runOnMachineFunction(MachineFunction &MF) { | ||
SIMachineFunctionInfo *MFI = MF.getInfo<SIMachineFunctionInfo>(); | ||
|
||
bool Changed = false; | ||
for (MachineBasicBlock &MBB : MF) { | ||
for (MachineInstr &MI : MBB) { | ||
unsigned Opc = MI.getOpcode(); | ||
if (Opc != AMDGPU::SI_SPILL_S32_TO_VGPR && | ||
Opc != AMDGPU::SI_RESTORE_S32_FROM_VGPR) | ||
continue; | ||
|
||
Register Reg = Opc == AMDGPU::SI_SPILL_S32_TO_VGPR | ||
? MI.getOperand(0).getReg() | ||
: MI.getOperand(1).getReg(); | ||
|
||
assert(Reg.isPhysical() && | ||
"All WWM registers should have been allocated by now."); | ||
|
||
MFI->reserveWWMRegister(Reg); | ||
Changed |= true; | ||
} | ||
} | ||
|
||
// The renamable flag can't be set for reserved registers. Reset the flag for | ||
// MOs involving wwm-regs as they will be reserved during vgpr-regalloc | ||
// pipeline. | ||
const MachineRegisterInfo &MRI = MF.getRegInfo(); | ||
for (Register Reg : MFI->getWWMReservedRegs()) { | ||
for (MachineOperand &MO : MRI.reg_operands(Reg)) | ||
MO.setIsRenamable(false); | ||
} | ||
|
||
// Now clear the NonWWMRegMask earlier set during wwm-regalloc. | ||
MFI->clearNonWWMRegAllocMask(); | ||
|
||
return Changed; | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.