blob: 1e9ed4a7f336627acdf376e051441ba6c5ceb79c [file] [edit]
; NOTE: Assertions have been autogenerated by utils/update_test_checks.py
; RUN: opt -S -mtriple=powerpc64-unknown-unknown -passes='require<libcall-lowering-info>,atomic-expand' %s | FileCheck %s
define float @test_atomicrmw_fadd_f32(ptr %ptr, float %value) {
; CHECK-LABEL: @test_atomicrmw_fadd_f32(
; CHECK-NEXT: call void @llvm.ppc.sync()
; CHECK-NEXT: [[TMP6:%.*]] = load atomic i32, ptr [[PTR:%.*]] monotonic, align 4
; CHECK-NEXT: [[TMP1:%.*]] = bitcast i32 [[TMP6]] to float
; CHECK-NEXT: br label [[ATOMICRMW_START:%.*]]
; CHECK: atomicrmw.start:
; CHECK-NEXT: [[LOADED:%.*]] = phi float [ [[TMP1]], [[TMP0:%.*]] ], [ [[TMP5:%.*]], [[CMPXCHG_END:%.*]] ]
; CHECK-NEXT: [[NEW:%.*]] = fadd float [[LOADED]], [[VALUE:%.*]]
; CHECK-NEXT: [[TMP2:%.*]] = bitcast float [[NEW]] to i32
; CHECK-NEXT: [[TMP3:%.*]] = bitcast float [[LOADED]] to i32
; CHECK-NEXT: br label [[CMPXCHG_START:%.*]]
; CHECK: cmpxchg.start:
; CHECK-NEXT: [[LARX:%.*]] = call i32 @llvm.ppc.lwarx(ptr [[PTR]])
; CHECK-NEXT: [[SHOULD_STORE:%.*]] = icmp eq i32 [[LARX]], [[TMP3]]
; CHECK-NEXT: br i1 [[SHOULD_STORE]], label [[CMPXCHG_FENCEDSTORE:%.*]], label [[CMPXCHG_NOSTORE:%.*]], !prof [[PROF0:![0-9]+]]
; CHECK: cmpxchg.fencedstore:
; CHECK-NEXT: br label [[CMPXCHG_TRYSTORE:%.*]]
; CHECK: cmpxchg.trystore:
; CHECK-NEXT: [[LOADED_TRYSTORE:%.*]] = phi i32 [ [[LARX]], [[CMPXCHG_FENCEDSTORE]] ]
; CHECK-NEXT: [[STCX:%.*]] = call i32 @llvm.ppc.stwcx(ptr [[PTR]], i32 [[TMP2]])
; CHECK-NEXT: [[TMP4:%.*]] = xor i32 [[STCX]], 1
; CHECK-NEXT: [[SUCCESS1:%.*]] = icmp eq i32 [[TMP4]], 0
; CHECK-NEXT: br i1 [[SUCCESS1]], label [[CMPXCHG_SUCCESS:%.*]], label [[CMPXCHG_START]], !prof [[PROF0]]
; CHECK: cmpxchg.releasedload:
; CHECK-NEXT: unreachable
; CHECK: cmpxchg.success:
; CHECK-NEXT: br label [[CMPXCHG_END]]
; CHECK: cmpxchg.nostore:
; CHECK-NEXT: [[LOADED_NOSTORE:%.*]] = phi i32 [ [[LARX]], [[CMPXCHG_START]] ]
; CHECK-NEXT: br label [[CMPXCHG_FAILURE:%.*]]
; CHECK: cmpxchg.failure:
; CHECK-NEXT: [[LOADED_FAILURE:%.*]] = phi i32 [ [[LOADED_NOSTORE]], [[CMPXCHG_NOSTORE]] ]
; CHECK-NEXT: br label [[CMPXCHG_END]]
; CHECK: cmpxchg.end:
; CHECK-NEXT: [[LOADED_EXIT:%.*]] = phi i32 [ [[LOADED_TRYSTORE]], [[CMPXCHG_SUCCESS]] ], [ [[LOADED_FAILURE]], [[CMPXCHG_FAILURE]] ]
; CHECK-NEXT: [[SUCCESS2:%.*]] = phi i1 [ true, [[CMPXCHG_SUCCESS]] ], [ false, [[CMPXCHG_FAILURE]] ]
; CHECK-NEXT: [[TMP5]] = bitcast i32 [[LOADED_EXIT]] to float
; CHECK-NEXT: br i1 [[SUCCESS2]], label [[ATOMICRMW_END:%.*]], label [[ATOMICRMW_START]]
; CHECK: atomicrmw.end:
; CHECK-NEXT: call void @llvm.ppc.lwsync()
; CHECK-NEXT: ret float [[TMP5]]
;
%res = atomicrmw fadd ptr %ptr, float %value seq_cst
ret float %res
}
define float @test_atomicrmw_fsub_f32(ptr %ptr, float %value) {
; CHECK-LABEL: @test_atomicrmw_fsub_f32(
; CHECK-NEXT: call void @llvm.ppc.sync()
; CHECK-NEXT: [[TMP6:%.*]] = load atomic i32, ptr [[PTR:%.*]] monotonic, align 4
; CHECK-NEXT: [[TMP1:%.*]] = bitcast i32 [[TMP6]] to float
; CHECK-NEXT: br label [[ATOMICRMW_START:%.*]]
; CHECK: atomicrmw.start:
; CHECK-NEXT: [[LOADED:%.*]] = phi float [ [[TMP1]], [[TMP0:%.*]] ], [ [[TMP5:%.*]], [[CMPXCHG_END:%.*]] ]
; CHECK-NEXT: [[NEW:%.*]] = fsub float [[LOADED]], [[VALUE:%.*]]
; CHECK-NEXT: [[TMP2:%.*]] = bitcast float [[NEW]] to i32
; CHECK-NEXT: [[TMP3:%.*]] = bitcast float [[LOADED]] to i32
; CHECK-NEXT: br label [[CMPXCHG_START:%.*]]
; CHECK: cmpxchg.start:
; CHECK-NEXT: [[LARX:%.*]] = call i32 @llvm.ppc.lwarx(ptr [[PTR]])
; CHECK-NEXT: [[SHOULD_STORE:%.*]] = icmp eq i32 [[LARX]], [[TMP3]]
; CHECK-NEXT: br i1 [[SHOULD_STORE]], label [[CMPXCHG_FENCEDSTORE:%.*]], label [[CMPXCHG_NOSTORE:%.*]], !prof [[PROF0]]
; CHECK: cmpxchg.fencedstore:
; CHECK-NEXT: br label [[CMPXCHG_TRYSTORE:%.*]]
; CHECK: cmpxchg.trystore:
; CHECK-NEXT: [[LOADED_TRYSTORE:%.*]] = phi i32 [ [[LARX]], [[CMPXCHG_FENCEDSTORE]] ]
; CHECK-NEXT: [[STCX:%.*]] = call i32 @llvm.ppc.stwcx(ptr [[PTR]], i32 [[TMP2]])
; CHECK-NEXT: [[TMP4:%.*]] = xor i32 [[STCX]], 1
; CHECK-NEXT: [[SUCCESS1:%.*]] = icmp eq i32 [[TMP4]], 0
; CHECK-NEXT: br i1 [[SUCCESS1]], label [[CMPXCHG_SUCCESS:%.*]], label [[CMPXCHG_START]], !prof [[PROF0]]
; CHECK: cmpxchg.releasedload:
; CHECK-NEXT: unreachable
; CHECK: cmpxchg.success:
; CHECK-NEXT: br label [[CMPXCHG_END]]
; CHECK: cmpxchg.nostore:
; CHECK-NEXT: [[LOADED_NOSTORE:%.*]] = phi i32 [ [[LARX]], [[CMPXCHG_START]] ]
; CHECK-NEXT: br label [[CMPXCHG_FAILURE:%.*]]
; CHECK: cmpxchg.failure:
; CHECK-NEXT: [[LOADED_FAILURE:%.*]] = phi i32 [ [[LOADED_NOSTORE]], [[CMPXCHG_NOSTORE]] ]
; CHECK-NEXT: br label [[CMPXCHG_END]]
; CHECK: cmpxchg.end:
; CHECK-NEXT: [[LOADED_EXIT:%.*]] = phi i32 [ [[LOADED_TRYSTORE]], [[CMPXCHG_SUCCESS]] ], [ [[LOADED_FAILURE]], [[CMPXCHG_FAILURE]] ]
; CHECK-NEXT: [[SUCCESS2:%.*]] = phi i1 [ true, [[CMPXCHG_SUCCESS]] ], [ false, [[CMPXCHG_FAILURE]] ]
; CHECK-NEXT: [[TMP5]] = bitcast i32 [[LOADED_EXIT]] to float
; CHECK-NEXT: br i1 [[SUCCESS2]], label [[ATOMICRMW_END:%.*]], label [[ATOMICRMW_START]]
; CHECK: atomicrmw.end:
; CHECK-NEXT: call void @llvm.ppc.lwsync()
; CHECK-NEXT: ret float [[TMP5]]
;
%res = atomicrmw fsub ptr %ptr, float %value seq_cst
ret float %res
}