Skip to content

Commit b1a57c3

Browse files
committed
[AMDGPU] Fix hidden kernarg preload count inconsistency
It is possible that the number of hidden arguments that are selected to be preloaded in AMDGPULowerKernel arguments and isel can differ. This isn't an issue with explicit arguments since isel can lower the argument correctly either way, but with hidden arguments we may have alignment issues if we try to load these hidden arguments that were added to the kernel signature. The reason for the mismatch is that isel reserves an extra synthetic user SGPR for module LDS. Instead of teaching lowerFormalArguments how to handle these properly it makes more sense and is less expensive to fix the mismatch and assert if we ever run into this issue again. We should never be trying to lower these in the normal way. In a future change we probably want to revise how we track "synthetic" user SGPRs and unify the handling in GCNUserSGPRUsageInfo. Sometimes synthetic SGPRSs are considered user SGPRs and sometimes they are not. Until then this patch resolves the inconsistency, fixes the bug, and is otherwise a NFC.
1 parent 3155199 commit b1a57c3

File tree

4 files changed

+62
-23
lines changed

4 files changed

+62
-23
lines changed

llvm/lib/Target/AMDGPU/AMDGPULowerKernelArguments.cpp

Lines changed: 6 additions & 6 deletions
Original file line numberDiff line numberDiff line change
@@ -143,17 +143,17 @@ class PreloadKernelArgInfo {
143143
// Returns the maximum number of user SGPRs that we have available to preload
144144
// arguments.
145145
void setInitialFreeUserSGPRsCount() {
146-
const unsigned MaxUserSGPRs = ST.getMaxNumUserSGPRs();
147146
GCNUserSGPRUsageInfo UserSGPRInfo(F, ST);
148-
149-
NumFreeUserSGPRs = MaxUserSGPRs - UserSGPRInfo.getNumUsedUserSGPRs();
147+
NumFreeUserSGPRs =
148+
UserSGPRInfo.getNumFreeUserSGPRs() - 1 /* Synthetic SGPRs*/;
150149
}
151150

152151
bool tryAllocPreloadSGPRs(unsigned AllocSize, uint64_t ArgOffset,
153152
uint64_t LastExplicitArgOffset) {
154153
// Check if this argument may be loaded into the same register as the
155154
// previous argument.
156-
if (!isAligned(Align(4), ArgOffset) && AllocSize < 4)
155+
if (ArgOffset == LastExplicitArgOffset && !isAligned(Align(4), ArgOffset) &&
156+
AllocSize < 4)
157157
return true;
158158

159159
// Pad SGPRs for kernarg alignment.
@@ -169,6 +169,7 @@ class PreloadKernelArgInfo {
169169

170170
// Try to allocate SGPRs to preload implicit kernel arguments.
171171
void tryAllocImplicitArgPreloadSGPRs(uint64_t ImplicitArgsBaseOffset,
172+
uint64_t LastExplicitArgOffset,
172173
IRBuilder<> &Builder) {
173174
Function *ImplicitArgPtr = Intrinsic::getDeclarationIfExists(
174175
F.getParent(), Intrinsic::amdgcn_implicitarg_ptr);
@@ -214,7 +215,6 @@ class PreloadKernelArgInfo {
214215
// argument can actually be preloaded.
215216
std::sort(ImplicitArgLoads.begin(), ImplicitArgLoads.end(), less_second());
216217

217-
uint64_t LastExplicitArgOffset = ImplicitArgsBaseOffset;
218218
// If we fail to preload any implicit argument we know we don't have SGPRs
219219
// to preload any subsequent ones with larger offsets. Find the first
220220
// argument that we cannot preload.
@@ -474,7 +474,7 @@ static bool lowerKernelArguments(Function &F, const TargetMachine &TM) {
474474
uint64_t ImplicitArgsBaseOffset =
475475
alignTo(ExplicitArgOffset, ST.getAlignmentForImplicitArgPtr()) +
476476
BaseOffset;
477-
PreloadInfo.tryAllocImplicitArgPreloadSGPRs(ImplicitArgsBaseOffset,
477+
PreloadInfo.tryAllocImplicitArgPreloadSGPRs(ImplicitArgsBaseOffset, ExplicitArgOffset,
478478
Builder);
479479
}
480480

llvm/lib/Target/AMDGPU/SIISelLowering.cpp

Lines changed: 8 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -2999,6 +2999,14 @@ SDValue SITargetLowering::LowerFormalArguments(
29992999
NewArg = DAG.getMergeValues({NewArg, Chain}, DL);
30003000
}
30013001
} else {
3002+
#ifndef NDEBUG
3003+
if (Arg.isOrigArg()) {
3004+
Argument *OrigArg = Fn.getArg(Arg.getOrigArgIndex());
3005+
assert(!OrigArg->hasAttribute("amdgpu-hidden-argument") &&
3006+
"Hidden arguments should be preloaded");
3007+
}
3008+
#endif // NDEBUG
3009+
30023010
NewArg =
30033011
lowerKernargMemParameter(DAG, VT, MemVT, DL, Chain, Offset,
30043012
Alignment, Ins[i].Flags.isSExt(), &Ins[i]);

llvm/test/CodeGen/AMDGPU/preload-implicit-kernargs.ll

Lines changed: 44 additions & 15 deletions
Original file line numberDiff line numberDiff line change
@@ -439,13 +439,13 @@ define amdgpu_kernel void @preload_workgroup_size_xyz(ptr addrspace(1) inreg %ou
439439
; GFX90a: s_trap 2 ; Kernarg preload header. Trap with incompatible firmware that doesn't support preloading kernel arguments.
440440
; GFX90a-NEXT: .fill 63, 4, 0xbf800000 ; s_nop 0
441441
; GFX90a-NEXT: ; %bb.0:
442+
; GFX90a-NEXT: v_mov_b32_e32 v3, 0
443+
; GFX90a-NEXT: global_load_ushort v2, v3, s[4:5] offset:24
442444
; GFX90a-NEXT: s_lshr_b32 s0, s11, 16
443445
; GFX90a-NEXT: s_and_b32 s1, s11, 0xffff
444-
; GFX90a-NEXT: s_and_b32 s2, s12, 0xffff
445-
; GFX90a-NEXT: v_mov_b32_e32 v3, 0
446446
; GFX90a-NEXT: v_mov_b32_e32 v0, s1
447447
; GFX90a-NEXT: v_mov_b32_e32 v1, s0
448-
; GFX90a-NEXT: v_mov_b32_e32 v2, s2
448+
; GFX90a-NEXT: s_waitcnt vmcnt(0)
449449
; GFX90a-NEXT: global_store_dwordx3 v3, v[0:2], s[6:7]
450450
; GFX90a-NEXT: s_endpgm
451451
%imp_arg_ptr = call ptr addrspace(4) @llvm.amdgcn.implicitarg.ptr()
@@ -554,27 +554,28 @@ define amdgpu_kernel void @preloadremainder_xyz(ptr addrspace(1) inreg %out) #0
554554
; GFX940: s_trap 2 ; Kernarg preload header. Trap with incompatible firmware that doesn't support preloading kernel arguments.
555555
; GFX940-NEXT: .fill 63, 4, 0xbf800000 ; s_nop 0
556556
; GFX940-NEXT: ; %bb.0:
557-
; GFX940-NEXT: s_lshr_b32 s0, s9, 16
558-
; GFX940-NEXT: s_lshr_b32 s1, s8, 16
559-
; GFX940-NEXT: s_and_b32 s4, s9, 0xffff
560557
; GFX940-NEXT: v_mov_b32_e32 v3, 0
561-
; GFX940-NEXT: v_mov_b32_e32 v0, s1
562-
; GFX940-NEXT: v_mov_b32_e32 v1, s4
563-
; GFX940-NEXT: v_mov_b32_e32 v2, s0
558+
; GFX940-NEXT: global_load_ushort v2, v3, s[0:1] offset:30
559+
; GFX940-NEXT: s_lshr_b32 s0, s8, 16
560+
; GFX940-NEXT: s_and_b32 s1, s9, 0xffff
561+
; GFX940-NEXT: v_mov_b32_e32 v0, s0
562+
; GFX940-NEXT: v_mov_b32_e32 v1, s1
563+
; GFX940-NEXT: s_waitcnt vmcnt(0)
564564
; GFX940-NEXT: global_store_dwordx3 v3, v[0:2], s[2:3] sc0 sc1
565565
; GFX940-NEXT: s_endpgm
566566
;
567567
; GFX90a-LABEL: preloadremainder_xyz:
568568
; GFX90a: s_trap 2 ; Kernarg preload header. Trap with incompatible firmware that doesn't support preloading kernel arguments.
569569
; GFX90a-NEXT: .fill 63, 4, 0xbf800000 ; s_nop 0
570570
; GFX90a-NEXT: ; %bb.0:
571-
; GFX90a-NEXT: s_lshr_b32 s0, s13, 16
572-
; GFX90a-NEXT: s_lshr_b32 s1, s12, 16
573-
; GFX90a-NEXT: s_and_b32 s2, s13, 0xffff
574571
; GFX90a-NEXT: v_mov_b32_e32 v3, 0
575-
; GFX90a-NEXT: v_mov_b32_e32 v0, s1
576-
; GFX90a-NEXT: v_mov_b32_e32 v1, s2
577-
; GFX90a-NEXT: v_mov_b32_e32 v2, s0
572+
; GFX90a-NEXT: global_load_dword v0, v3, s[4:5] offset:26
573+
; GFX90a-NEXT: global_load_ushort v2, v3, s[4:5] offset:30
574+
; GFX90a-NEXT: s_lshr_b32 s0, s12, 16
575+
; GFX90a-NEXT: s_waitcnt vmcnt(1)
576+
; GFX90a-NEXT: v_lshrrev_b32_e32 v1, 16, v0
577+
; GFX90a-NEXT: v_mov_b32_e32 v0, s0
578+
; GFX90a-NEXT: s_waitcnt vmcnt(0)
578579
; GFX90a-NEXT: global_store_dwordx3 v3, v[0:2], s[6:7]
579580
; GFX90a-NEXT: s_endpgm
580581
%imp_arg_ptr = call ptr addrspace(4) @llvm.amdgcn.implicitarg.ptr()
@@ -626,4 +627,32 @@ define amdgpu_kernel void @no_free_sgprs_preloadremainder_z(ptr addrspace(1) inr
626627
ret void
627628
}
628629

630+
; Check for consistency between isel and earlier passes preload SGPR accounting.
631+
632+
define amdgpu_kernel void @preload_block_max_user_sgprs(ptr addrspace(1) inreg %out, i192 inreg %t0, i32 inreg %t1) #0 {
633+
; GFX940-LABEL: preload_block_max_user_sgprs:
634+
; GFX940: s_trap 2 ; Kernarg preload header. Trap with incompatible firmware that doesn't support preloading kernel arguments.
635+
; GFX940-NEXT: .fill 63, 4, 0xbf800000 ; s_nop 0
636+
; GFX940-NEXT: ; %bb.0:
637+
; GFX940-NEXT: v_mov_b32_e32 v0, 0
638+
; GFX940-NEXT: v_mov_b32_e32 v1, s12
639+
; GFX940-NEXT: global_store_dword v0, v1, s[2:3] sc0 sc1
640+
; GFX940-NEXT: s_endpgm
641+
;
642+
; GFX90a-LABEL: preload_block_max_user_sgprs:
643+
; GFX90a: s_trap 2 ; Kernarg preload header. Trap with incompatible firmware that doesn't support preloading kernel arguments.
644+
; GFX90a-NEXT: .fill 63, 4, 0xbf800000 ; s_nop 0
645+
; GFX90a-NEXT: ; %bb.0:
646+
; GFX90a-NEXT: s_load_dword s0, s[4:5], 0x28
647+
; GFX90a-NEXT: v_mov_b32_e32 v0, 0
648+
; GFX90a-NEXT: s_waitcnt lgkmcnt(0)
649+
; GFX90a-NEXT: v_mov_b32_e32 v1, s0
650+
; GFX90a-NEXT: global_store_dword v0, v1, s[6:7]
651+
; GFX90a-NEXT: s_endpgm
652+
%imp_arg_ptr = call ptr addrspace(4) @llvm.amdgcn.implicitarg.ptr()
653+
%load = load i32, ptr addrspace(4) %imp_arg_ptr
654+
store i32 %load, ptr addrspace(1) %out
655+
ret void
656+
}
657+
629658
attributes #0 = { "amdgpu-no-agpr" "amdgpu-no-completion-action" "amdgpu-no-default-queue" "amdgpu-no-dispatch-id" "amdgpu-no-dispatch-ptr" "amdgpu-no-heap-ptr" "amdgpu-no-hostcall-ptr" "amdgpu-no-lds-kernel-id" "amdgpu-no-multigrid-sync-arg" "amdgpu-no-queue-ptr" "amdgpu-no-workgroup-id-x" "amdgpu-no-workgroup-id-y" "amdgpu-no-workgroup-id-z" "amdgpu-no-workitem-id-x" "amdgpu-no-workitem-id-y" "amdgpu-no-workitem-id-z" "uniform-work-group-size"="false" }

llvm/test/CodeGen/AMDGPU/preload-kernargs-IR-lowering.ll

Lines changed: 4 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -187,15 +187,17 @@ define amdgpu_kernel void @test_preload_IR_lowering_kernel_8(ptr addrspace(1) %i
187187
; PRELOAD-8-LABEL: define {{[^@]+}}@test_preload_IR_lowering_kernel_8
188188
; PRELOAD-8-SAME: (ptr addrspace(1) inreg [[IN:%.*]], ptr addrspace(1) inreg [[IN1:%.*]], ptr addrspace(1) inreg [[IN2:%.*]], ptr addrspace(1) inreg [[IN3:%.*]], ptr addrspace(1) inreg [[OUT:%.*]], ptr addrspace(1) inreg [[OUT1:%.*]], ptr addrspace(1) inreg [[OUT2:%.*]], ptr addrspace(1) inreg [[OUT3:%.*]]) #[[ATTR0]] {
189189
; PRELOAD-8-NEXT: [[TEST_PRELOAD_IR_LOWERING_KERNEL_8_KERNARG_SEGMENT:%.*]] = call nonnull align 16 dereferenceable(64) ptr addrspace(4) @llvm.amdgcn.kernarg.segment.ptr()
190+
; PRELOAD-8-NEXT: [[OUT2_KERNARG_OFFSET:%.*]] = getelementptr inbounds i8, ptr addrspace(4) [[TEST_PRELOAD_IR_LOWERING_KERNEL_8_KERNARG_SEGMENT]], i64 48
191+
; PRELOAD-8-NEXT: [[OUT2_LOAD:%.*]] = load ptr addrspace(1), ptr addrspace(4) [[OUT2_KERNARG_OFFSET]], align 16, !invariant.load [[META0:![0-9]+]]
190192
; PRELOAD-8-NEXT: [[OUT3_KERNARG_OFFSET:%.*]] = getelementptr inbounds i8, ptr addrspace(4) [[TEST_PRELOAD_IR_LOWERING_KERNEL_8_KERNARG_SEGMENT]], i64 56
191-
; PRELOAD-8-NEXT: [[OUT3_LOAD:%.*]] = load ptr addrspace(1), ptr addrspace(4) [[OUT3_KERNARG_OFFSET]], align 8, !invariant.load [[META0:![0-9]+]]
193+
; PRELOAD-8-NEXT: [[OUT3_LOAD:%.*]] = load ptr addrspace(1), ptr addrspace(4) [[OUT3_KERNARG_OFFSET]], align 8, !invariant.load [[META0]]
192194
; PRELOAD-8-NEXT: [[LOAD:%.*]] = load i32, ptr addrspace(1) [[IN]], align 4
193195
; PRELOAD-8-NEXT: [[LOAD1:%.*]] = load i32, ptr addrspace(1) [[IN1]], align 4
194196
; PRELOAD-8-NEXT: [[LOAD2:%.*]] = load i32, ptr addrspace(1) [[IN2]], align 4
195197
; PRELOAD-8-NEXT: [[LOAD3:%.*]] = load i32, ptr addrspace(1) [[IN3]], align 4
196198
; PRELOAD-8-NEXT: store i32 [[LOAD]], ptr addrspace(1) [[OUT]], align 4
197199
; PRELOAD-8-NEXT: store i32 [[LOAD1]], ptr addrspace(1) [[OUT1]], align 4
198-
; PRELOAD-8-NEXT: store i32 [[LOAD2]], ptr addrspace(1) [[OUT2]], align 4
200+
; PRELOAD-8-NEXT: store i32 [[LOAD2]], ptr addrspace(1) [[OUT2_LOAD]], align 4
199201
; PRELOAD-8-NEXT: store i32 [[LOAD3]], ptr addrspace(1) [[OUT3_LOAD]], align 4
200202
; PRELOAD-8-NEXT: ret void
201203
;

0 commit comments

Comments
 (0)