; NOTE: Assertions have been autogenerated by utils/update_test_checks.py UTC_ARGS: --version 5 ; RUN: opt -S -O1 -mtriple=amdgcn-amd-amdhsa %s -o - | FileCheck --check-prefix=INFER %s ; RUN: opt -S -O2 -mtriple=amdgcn-amd-amdhsa %s -o - | FileCheck --check-prefix=INFER %s ; RUN: opt -S -O3 -mtriple=amdgcn-amd-amdhsa %s -o - | FileCheck --check-prefix=INFER %s ; RUN: opt -S -passes='lto' -mtriple=amdgcn-amd-amdhsa %s -o - | FileCheck --check-prefix=NO-INFER %s ; RUN: opt -S -passes='lto' -mtriple=amdgcn-amd-amdhsa %s -o - | FileCheck --check-prefix=INFER %s ; RUN: opt -S -passes='lto' -mtriple=amdgcn-amd-amdhsa %s -o - | FileCheck --check-prefix=INFER %s %struct.data = type { ptr, i32 } declare void @external_use(i32) define internal void @callee(ptr %ptr_as0, i32 %val) { ; NO-INFER-LABEL: define internal void @callee( ; NO-INFER-SAME: ptr [[PTR_AS0:%.*]], i32 [[VAL:%.*]]) #[[ATTR1:[0-9]+]] { ; NO-INFER-NEXT: [[LOADED:%.*]] = load i32, ptr [[PTR_AS0]], align 4 ; NO-INFER-NEXT: [[COMPUTED:%.*]] = add i32 [[LOADED]], [[VAL]] ; NO-INFER-NEXT: store i32 [[COMPUTED]], ptr [[PTR_AS0]], align 4 ; NO-INFER-NEXT: call void @external_use(i32 [[COMPUTED]]) ; NO-INFER-NEXT: ret void ; %loaded = load i32, ptr %ptr_as0, align 4 %computed = add i32 %loaded, %val store i32 %computed, ptr %ptr_as0, align 4 call void @external_use(i32 %computed) ret void } define void @caller(ptr addrspace(1) %ptr_as1, i32 %value) { ; INFER-LABEL: define void @caller( ; INFER-SAME: ptr addrspace(1) captures(none) [[PTR_AS1:%.*]], i32 [[VALUE:%.*]]) local_unnamed_addr #[[ATTR1:[0-9]+]] { ; INFER-NEXT: [[LOADED_I:%.*]] = load i32, ptr addrspace(1) [[PTR_AS1]], align 4 ; INFER-NEXT: [[COMPUTED_I:%.*]] = add i32 [[LOADED_I]], [[VALUE]] ; INFER-NEXT: store i32 [[COMPUTED_I]], ptr addrspace(1) [[PTR_AS1]], align 4 ; INFER-NEXT: tail call void @external_use(i32 [[COMPUTED_I]]) ; INFER-NEXT: ret void ; ; NO-INFER-LABEL: define void @caller( ; NO-INFER-SAME: ptr addrspace(1) [[PTR_AS1:%.*]], i32 [[VALUE:%.*]]) #[[ATTR1]] { ; NO-INFER-NEXT: [[DATA:%.*]] = alloca [[STRUCT_DATA:%.*]], align 8, addrspace(5) ; NO-INFER-NEXT: [[VAL_FIELD:%.*]] = getelementptr [[STRUCT_DATA]], ptr addrspace(5) [[DATA]], i32 0, i32 1 ; NO-INFER-NEXT: store i32 [[VALUE]], ptr addrspace(5) [[VAL_FIELD]], align 4 ; NO-INFER-NEXT: [[GENERIC_INPUT:%.*]] = addrspacecast ptr addrspace(1) [[PTR_AS1]] to ptr ; NO-INFER-NEXT: store ptr [[GENERIC_INPUT]], ptr addrspace(5) [[DATA]], align 8 ; NO-INFER-NEXT: [[RETRIEVED_PTR:%.*]] = load ptr, ptr addrspace(5) [[DATA]], align 8 ; NO-INFER-NEXT: [[RETRIEVED_VAL:%.*]] = load i32, ptr addrspace(5) [[VAL_FIELD]], align 4 ; NO-INFER-NEXT: call void @callee(ptr [[RETRIEVED_PTR]], i32 [[RETRIEVED_VAL]]) ; NO-INFER-NEXT: ret void ; %data = alloca %struct.data, align 8, addrspace(5) %ptr_field = getelementptr %struct.data, ptr addrspace(5) %data, i32 0, i32 0 %val_field = getelementptr %struct.data, ptr addrspace(5) %data, i32 0, i32 1 store i32 %value, ptr addrspace(5) %val_field, align 4 %generic_input = addrspacecast ptr addrspace(1) %ptr_as1 to ptr store ptr %generic_input, ptr addrspace(5) %ptr_field, align 8 %retrieved_ptr = load ptr, ptr addrspace(5) %ptr_field, align 8 %retrieved_val = load i32, ptr addrspace(5) %val_field, align 4 call void @callee(ptr %retrieved_ptr, i32 %retrieved_val) ret void }