R600/SI: Don't move operands that are required to be SGPRs

e.g. v_cndmask_b32 requires the condition operand be an SGPR.
If one of the source operands were an SGPR, that would be considered
the one SGPR use and the condition operand would be illegally moved.

git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@218529 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
Matt Arsenault 2014-09-26 17:54:52 +00:00
parent 5b199b585c
commit 8a70e28114
2 changed files with 53 additions and 10 deletions

View File

@ -1357,8 +1357,27 @@ void SIInstrInfo::legalizeOperands(MachineInstr *MI) const {
// XXX - Do any VOP3 instructions read VCC? // XXX - Do any VOP3 instructions read VCC?
// Legalize VOP3 // Legalize VOP3
if (isVOP3(MI->getOpcode())) { if (isVOP3(MI->getOpcode())) {
int VOP3Idx[3] = {Src0Idx, Src1Idx, Src2Idx}; const MCInstrDesc &Desc = get(MI->getOpcode());
unsigned SGPRReg = AMDGPU::NoRegister; unsigned SGPRReg = AMDGPU::NoRegister;
int VOP3Idx[3] = { Src0Idx, Src1Idx, Src2Idx };
// First we need to consider the instruction's operand requirements before
// legalizing. Some operands are required to be SGPRs, but we are still
// bound by the constant bus requirement to only use one.
//
// If the operand's class is an SGPR, we can never move it.
for (unsigned i = 0; i < 3; ++i) {
int Idx = VOP3Idx[i];
if (Idx == -1)
break;
if (RI.isSGPRClassID(Desc.OpInfo[Idx].RegClass)) {
SGPRReg = MI->getOperand(Idx).getReg();
break;
}
}
for (unsigned i = 0; i < 3; ++i) { for (unsigned i = 0; i < 3; ++i) {
int Idx = VOP3Idx[i]; int Idx = VOP3Idx[i];
if (Idx == -1) if (Idx == -1)

View File

@ -1,14 +1,38 @@
; RUN: llc < %s -march=r600 -mcpu=SI -verify-machineinstrs | FileCheck --check-prefix=SI %s ; RUN: llc -march=r600 -mcpu=SI -verify-machineinstrs < %s | FileCheck -check-prefix=SI %s
; SI: @v_cnd_nan declare i32 @llvm.r600.read.tidig.x() #1
; SI: V_CNDMASK_B32_e64 v{{[0-9]}},
; SI-LABEL: @v_cnd_nan_nosgpr
; SI: V_CNDMASK_B32_e64 v{{[0-9]}}, v{{[0-9]}}, -1, s{{\[[0-9]+:[0-9]+\]}}
; SI-DAG: v{{[0-9]}} ; SI-DAG: v{{[0-9]}}
; All nan values are converted to 0xffffffff ; All nan values are converted to 0xffffffff
; SI-DAG: -1 ; SI: S_ENDPGM
define void @v_cnd_nan(float addrspace(1)* %out, i32 %c, float %f) { define void @v_cnd_nan_nosgpr(float addrspace(1)* %out, i32 %c, float addrspace(1)* %fptr) #0 {
entry: %idx = call i32 @llvm.r600.read.tidig.x() #1
%0 = icmp ne i32 %c, 0 %f.gep = getelementptr float addrspace(1)* %fptr, i32 %idx
%1 = select i1 %0, float 0xFFFFFFFFE0000000, float %f %f = load float addrspace(1)* %fptr
store float %1, float addrspace(1)* %out %setcc = icmp ne i32 %c, 0
%select = select i1 %setcc, float 0xFFFFFFFFE0000000, float %f
store float %select, float addrspace(1)* %out
ret void ret void
} }
; This requires slightly trickier SGPR operand legalization since the
; single constant bus SGPR usage is the last operand, and it should
; never be moved.
; SI-LABEL: @v_cnd_nan
; SI: V_CNDMASK_B32_e64 v{{[0-9]}}, v{{[0-9]}}, -1, s{{\[[0-9]+:[0-9]+\]}}
; SI-DAG: v{{[0-9]}}
; All nan values are converted to 0xffffffff
; SI: S_ENDPGM
define void @v_cnd_nan(float addrspace(1)* %out, i32 %c, float %f) #0 {
%setcc = icmp ne i32 %c, 0
%select = select i1 %setcc, float 0xFFFFFFFFE0000000, float %f
store float %select, float addrspace(1)* %out
ret void
}
attributes #0 = { nounwind }
attributes #1 = { nounwind readnone }