| // NOTE: Assertions have been autogenerated by utils/update_cc_test_checks.py UTC_ARGS: --version 4 |
| // RUN: %clang_cc1 -triple armv5e-none-linux-gnueabi -emit-llvm %s -o - | FileCheck %s |
| |
| enum memory_order { |
| memory_order_relaxed, memory_order_consume, memory_order_acquire, |
| memory_order_release, memory_order_acq_rel, memory_order_seq_cst |
| }; |
| |
| // CHECK-LABEL: define dso_local ptr @test_c11_atomic_fetch_add_int_ptr( |
| // CHECK-SAME: ptr noundef [[P:%.*]]) #[[ATTR0:[0-9]+]] { |
| // CHECK-NEXT: entry: |
| // CHECK-NEXT: [[P_ADDR:%.*]] = alloca ptr, align 4 |
| // CHECK-NEXT: [[DOTATOMICTMP:%.*]] = alloca i32, align 4 |
| // CHECK-NEXT: [[ATOMIC_TEMP:%.*]] = alloca ptr, align 4 |
| // CHECK-NEXT: store ptr [[P]], ptr [[P_ADDR]], align 4 |
| // CHECK-NEXT: [[TMP0:%.*]] = load ptr, ptr [[P_ADDR]], align 4 |
| // CHECK-NEXT: store i32 12, ptr [[DOTATOMICTMP]], align 4 |
| // CHECK-NEXT: [[TMP1:%.*]] = load i32, ptr [[DOTATOMICTMP]], align 4 |
| // CHECK-NEXT: [[TMP2:%.*]] = atomicrmw add ptr [[TMP0]], i32 [[TMP1]] seq_cst, align 4 |
| // CHECK-NEXT: store i32 [[TMP2]], ptr [[ATOMIC_TEMP]], align 4 |
| // CHECK-NEXT: [[TMP3:%.*]] = load ptr, ptr [[ATOMIC_TEMP]], align 4 |
| // CHECK-NEXT: ret ptr [[TMP3]] |
| // |
| int *test_c11_atomic_fetch_add_int_ptr(_Atomic(int *) *p) { |
| return __c11_atomic_fetch_add(p, 3, memory_order_seq_cst); |
| } |
| |
| // CHECK-LABEL: define dso_local ptr @test_c11_atomic_fetch_sub_int_ptr( |
| // CHECK-SAME: ptr noundef [[P:%.*]]) #[[ATTR0]] { |
| // CHECK-NEXT: entry: |
| // CHECK-NEXT: [[P_ADDR:%.*]] = alloca ptr, align 4 |
| // CHECK-NEXT: [[DOTATOMICTMP:%.*]] = alloca i32, align 4 |
| // CHECK-NEXT: [[ATOMIC_TEMP:%.*]] = alloca ptr, align 4 |
| // CHECK-NEXT: store ptr [[P]], ptr [[P_ADDR]], align 4 |
| // CHECK-NEXT: [[TMP0:%.*]] = load ptr, ptr [[P_ADDR]], align 4 |
| // CHECK-NEXT: store i32 20, ptr [[DOTATOMICTMP]], align 4 |
| // CHECK-NEXT: [[TMP1:%.*]] = load i32, ptr [[DOTATOMICTMP]], align 4 |
| // CHECK-NEXT: [[TMP2:%.*]] = atomicrmw sub ptr [[TMP0]], i32 [[TMP1]] seq_cst, align 4 |
| // CHECK-NEXT: store i32 [[TMP2]], ptr [[ATOMIC_TEMP]], align 4 |
| // CHECK-NEXT: [[TMP3:%.*]] = load ptr, ptr [[ATOMIC_TEMP]], align 4 |
| // CHECK-NEXT: ret ptr [[TMP3]] |
| // |
| int *test_c11_atomic_fetch_sub_int_ptr(_Atomic(int *) *p) { |
| return __c11_atomic_fetch_sub(p, 5, memory_order_seq_cst); |
| } |
| |
| // CHECK-LABEL: define dso_local i32 @test_c11_atomic_fetch_add_int( |
| // CHECK-SAME: ptr noundef [[P:%.*]]) #[[ATTR0]] { |
| // CHECK-NEXT: entry: |
| // CHECK-NEXT: [[P_ADDR:%.*]] = alloca ptr, align 4 |
| // CHECK-NEXT: [[DOTATOMICTMP:%.*]] = alloca i32, align 4 |
| // CHECK-NEXT: [[ATOMIC_TEMP:%.*]] = alloca i32, align 4 |
| // CHECK-NEXT: store ptr [[P]], ptr [[P_ADDR]], align 4 |
| // CHECK-NEXT: [[TMP0:%.*]] = load ptr, ptr [[P_ADDR]], align 4 |
| // CHECK-NEXT: store i32 3, ptr [[DOTATOMICTMP]], align 4 |
| // CHECK-NEXT: [[TMP1:%.*]] = load i32, ptr [[DOTATOMICTMP]], align 4 |
| // CHECK-NEXT: [[TMP2:%.*]] = atomicrmw add ptr [[TMP0]], i32 [[TMP1]] seq_cst, align 4 |
| // CHECK-NEXT: store i32 [[TMP2]], ptr [[ATOMIC_TEMP]], align 4 |
| // CHECK-NEXT: [[TMP3:%.*]] = load i32, ptr [[ATOMIC_TEMP]], align 4 |
| // CHECK-NEXT: ret i32 [[TMP3]] |
| // |
| int test_c11_atomic_fetch_add_int(_Atomic(int) *p) { |
| return __c11_atomic_fetch_add(p, 3, memory_order_seq_cst); |
| } |
| |
| // CHECK-LABEL: define dso_local i32 @test_c11_atomic_fetch_sub_int( |
| // CHECK-SAME: ptr noundef [[P:%.*]]) #[[ATTR0]] { |
| // CHECK-NEXT: entry: |
| // CHECK-NEXT: [[P_ADDR:%.*]] = alloca ptr, align 4 |
| // CHECK-NEXT: [[DOTATOMICTMP:%.*]] = alloca i32, align 4 |
| // CHECK-NEXT: [[ATOMIC_TEMP:%.*]] = alloca i32, align 4 |
| // CHECK-NEXT: store ptr [[P]], ptr [[P_ADDR]], align 4 |
| // CHECK-NEXT: [[TMP0:%.*]] = load ptr, ptr [[P_ADDR]], align 4 |
| // CHECK-NEXT: store i32 5, ptr [[DOTATOMICTMP]], align 4 |
| // CHECK-NEXT: [[TMP1:%.*]] = load i32, ptr [[DOTATOMICTMP]], align 4 |
| // CHECK-NEXT: [[TMP2:%.*]] = atomicrmw sub ptr [[TMP0]], i32 [[TMP1]] seq_cst, align 4 |
| // CHECK-NEXT: store i32 [[TMP2]], ptr [[ATOMIC_TEMP]], align 4 |
| // CHECK-NEXT: [[TMP3:%.*]] = load i32, ptr [[ATOMIC_TEMP]], align 4 |
| // CHECK-NEXT: ret i32 [[TMP3]] |
| // |
| int test_c11_atomic_fetch_sub_int(_Atomic(int) *p) { |
| return __c11_atomic_fetch_sub(p, 5, memory_order_seq_cst); |
| } |
| |
| // CHECK-LABEL: define dso_local ptr @fp2a( |
| // CHECK-SAME: ptr noundef [[P:%.*]]) #[[ATTR0]] { |
| // CHECK-NEXT: entry: |
| // CHECK-NEXT: [[P_ADDR:%.*]] = alloca ptr, align 4 |
| // CHECK-NEXT: [[DOTATOMICTMP:%.*]] = alloca i32, align 4 |
| // CHECK-NEXT: [[ATOMIC_TEMP:%.*]] = alloca ptr, align 4 |
| // CHECK-NEXT: store ptr [[P]], ptr [[P_ADDR]], align 4 |
| // CHECK-NEXT: [[TMP0:%.*]] = load ptr, ptr [[P_ADDR]], align 4 |
| // CHECK-NEXT: store i32 4, ptr [[DOTATOMICTMP]], align 4 |
| // CHECK-NEXT: [[TMP1:%.*]] = load i32, ptr [[DOTATOMICTMP]], align 4 |
| // CHECK-NEXT: [[TMP2:%.*]] = atomicrmw sub ptr [[TMP0]], i32 [[TMP1]] monotonic, align 4 |
| // CHECK-NEXT: store i32 [[TMP2]], ptr [[ATOMIC_TEMP]], align 4 |
| // CHECK-NEXT: [[TMP3:%.*]] = load ptr, ptr [[ATOMIC_TEMP]], align 4 |
| // CHECK-NEXT: ret ptr [[TMP3]] |
| // |
| int *fp2a(int **p) { |
| // Note, the GNU builtins do not multiply by sizeof(T)! |
| return __atomic_fetch_sub(p, 4, memory_order_relaxed); |
| } |
| |
| // CHECK-LABEL: define dso_local i32 @test_atomic_fetch_add( |
| // CHECK-SAME: ptr noundef [[P:%.*]]) #[[ATTR0]] { |
| // CHECK-NEXT: entry: |
| // CHECK-NEXT: [[P_ADDR:%.*]] = alloca ptr, align 4 |
| // CHECK-NEXT: [[DOTATOMICTMP:%.*]] = alloca i32, align 4 |
| // CHECK-NEXT: [[ATOMIC_TEMP:%.*]] = alloca i32, align 4 |
| // CHECK-NEXT: store ptr [[P]], ptr [[P_ADDR]], align 4 |
| // CHECK-NEXT: [[TMP0:%.*]] = load ptr, ptr [[P_ADDR]], align 4 |
| // CHECK-NEXT: store i32 55, ptr [[DOTATOMICTMP]], align 4 |
| // CHECK-NEXT: [[TMP1:%.*]] = load i32, ptr [[DOTATOMICTMP]], align 4 |
| // CHECK-NEXT: [[TMP2:%.*]] = atomicrmw add ptr [[TMP0]], i32 [[TMP1]] seq_cst, align 4 |
| // CHECK-NEXT: store i32 [[TMP2]], ptr [[ATOMIC_TEMP]], align 4 |
| // CHECK-NEXT: [[TMP3:%.*]] = load i32, ptr [[ATOMIC_TEMP]], align 4 |
| // CHECK-NEXT: ret i32 [[TMP3]] |
| // |
| int test_atomic_fetch_add(int *p) { |
| return __atomic_fetch_add(p, 55, memory_order_seq_cst); |
| } |
| |
| // CHECK-LABEL: define dso_local i32 @test_atomic_fetch_sub( |
| // CHECK-SAME: ptr noundef [[P:%.*]]) #[[ATTR0]] { |
| // CHECK-NEXT: entry: |
| // CHECK-NEXT: [[P_ADDR:%.*]] = alloca ptr, align 4 |
| // CHECK-NEXT: [[DOTATOMICTMP:%.*]] = alloca i32, align 4 |
| // CHECK-NEXT: [[ATOMIC_TEMP:%.*]] = alloca i32, align 4 |
| // CHECK-NEXT: store ptr [[P]], ptr [[P_ADDR]], align 4 |
| // CHECK-NEXT: [[TMP0:%.*]] = load ptr, ptr [[P_ADDR]], align 4 |
| // CHECK-NEXT: store i32 55, ptr [[DOTATOMICTMP]], align 4 |
| // CHECK-NEXT: [[TMP1:%.*]] = load i32, ptr [[DOTATOMICTMP]], align 4 |
| // CHECK-NEXT: [[TMP2:%.*]] = atomicrmw sub ptr [[TMP0]], i32 [[TMP1]] seq_cst, align 4 |
| // CHECK-NEXT: store i32 [[TMP2]], ptr [[ATOMIC_TEMP]], align 4 |
| // CHECK-NEXT: [[TMP3:%.*]] = load i32, ptr [[ATOMIC_TEMP]], align 4 |
| // CHECK-NEXT: ret i32 [[TMP3]] |
| // |
| int test_atomic_fetch_sub(int *p) { |
| return __atomic_fetch_sub(p, 55, memory_order_seq_cst); |
| } |
| |
| // CHECK-LABEL: define dso_local i32 @test_atomic_fetch_and( |
| // CHECK-SAME: ptr noundef [[P:%.*]]) #[[ATTR0]] { |
| // CHECK-NEXT: entry: |
| // CHECK-NEXT: [[P_ADDR:%.*]] = alloca ptr, align 4 |
| // CHECK-NEXT: [[DOTATOMICTMP:%.*]] = alloca i32, align 4 |
| // CHECK-NEXT: [[ATOMIC_TEMP:%.*]] = alloca i32, align 4 |
| // CHECK-NEXT: store ptr [[P]], ptr [[P_ADDR]], align 4 |
| // CHECK-NEXT: [[TMP0:%.*]] = load ptr, ptr [[P_ADDR]], align 4 |
| // CHECK-NEXT: store i32 55, ptr [[DOTATOMICTMP]], align 4 |
| // CHECK-NEXT: [[TMP1:%.*]] = load i32, ptr [[DOTATOMICTMP]], align 4 |
| // CHECK-NEXT: [[TMP2:%.*]] = atomicrmw and ptr [[TMP0]], i32 [[TMP1]] seq_cst, align 4 |
| // CHECK-NEXT: store i32 [[TMP2]], ptr [[ATOMIC_TEMP]], align 4 |
| // CHECK-NEXT: [[TMP3:%.*]] = load i32, ptr [[ATOMIC_TEMP]], align 4 |
| // CHECK-NEXT: ret i32 [[TMP3]] |
| // |
| int test_atomic_fetch_and(int *p) { |
| return __atomic_fetch_and(p, 55, memory_order_seq_cst); |
| } |
| |
| // CHECK-LABEL: define dso_local i32 @test_atomic_fetch_or( |
| // CHECK-SAME: ptr noundef [[P:%.*]]) #[[ATTR0]] { |
| // CHECK-NEXT: entry: |
| // CHECK-NEXT: [[P_ADDR:%.*]] = alloca ptr, align 4 |
| // CHECK-NEXT: [[DOTATOMICTMP:%.*]] = alloca i32, align 4 |
| // CHECK-NEXT: [[ATOMIC_TEMP:%.*]] = alloca i32, align 4 |
| // CHECK-NEXT: store ptr [[P]], ptr [[P_ADDR]], align 4 |
| // CHECK-NEXT: [[TMP0:%.*]] = load ptr, ptr [[P_ADDR]], align 4 |
| // CHECK-NEXT: store i32 55, ptr [[DOTATOMICTMP]], align 4 |
| // CHECK-NEXT: [[TMP1:%.*]] = load i32, ptr [[DOTATOMICTMP]], align 4 |
| // CHECK-NEXT: [[TMP2:%.*]] = atomicrmw or ptr [[TMP0]], i32 [[TMP1]] seq_cst, align 4 |
| // CHECK-NEXT: store i32 [[TMP2]], ptr [[ATOMIC_TEMP]], align 4 |
| // CHECK-NEXT: [[TMP3:%.*]] = load i32, ptr [[ATOMIC_TEMP]], align 4 |
| // CHECK-NEXT: ret i32 [[TMP3]] |
| // |
| int test_atomic_fetch_or(int *p) { |
| return __atomic_fetch_or(p, 55, memory_order_seq_cst); |
| } |
| |
| // CHECK-LABEL: define dso_local i32 @test_atomic_fetch_xor( |
| // CHECK-SAME: ptr noundef [[P:%.*]]) #[[ATTR0]] { |
| // CHECK-NEXT: entry: |
| // CHECK-NEXT: [[P_ADDR:%.*]] = alloca ptr, align 4 |
| // CHECK-NEXT: [[DOTATOMICTMP:%.*]] = alloca i32, align 4 |
| // CHECK-NEXT: [[ATOMIC_TEMP:%.*]] = alloca i32, align 4 |
| // CHECK-NEXT: store ptr [[P]], ptr [[P_ADDR]], align 4 |
| // CHECK-NEXT: [[TMP0:%.*]] = load ptr, ptr [[P_ADDR]], align 4 |
| // CHECK-NEXT: store i32 55, ptr [[DOTATOMICTMP]], align 4 |
| // CHECK-NEXT: [[TMP1:%.*]] = load i32, ptr [[DOTATOMICTMP]], align 4 |
| // CHECK-NEXT: [[TMP2:%.*]] = atomicrmw xor ptr [[TMP0]], i32 [[TMP1]] seq_cst, align 4 |
| // CHECK-NEXT: store i32 [[TMP2]], ptr [[ATOMIC_TEMP]], align 4 |
| // CHECK-NEXT: [[TMP3:%.*]] = load i32, ptr [[ATOMIC_TEMP]], align 4 |
| // CHECK-NEXT: ret i32 [[TMP3]] |
| // |
| int test_atomic_fetch_xor(int *p) { |
| return __atomic_fetch_xor(p, 55, memory_order_seq_cst); |
| } |
| |
| // CHECK-LABEL: define dso_local i32 @test_atomic_fetch_nand( |
| // CHECK-SAME: ptr noundef [[P:%.*]]) #[[ATTR0]] { |
| // CHECK-NEXT: entry: |
| // CHECK-NEXT: [[P_ADDR:%.*]] = alloca ptr, align 4 |
| // CHECK-NEXT: [[DOTATOMICTMP:%.*]] = alloca i32, align 4 |
| // CHECK-NEXT: [[ATOMIC_TEMP:%.*]] = alloca i32, align 4 |
| // CHECK-NEXT: store ptr [[P]], ptr [[P_ADDR]], align 4 |
| // CHECK-NEXT: [[TMP0:%.*]] = load ptr, ptr [[P_ADDR]], align 4 |
| // CHECK-NEXT: store i32 55, ptr [[DOTATOMICTMP]], align 4 |
| // CHECK-NEXT: [[TMP1:%.*]] = load i32, ptr [[DOTATOMICTMP]], align 4 |
| // CHECK-NEXT: [[TMP2:%.*]] = atomicrmw nand ptr [[TMP0]], i32 [[TMP1]] seq_cst, align 4 |
| // CHECK-NEXT: store i32 [[TMP2]], ptr [[ATOMIC_TEMP]], align 4 |
| // CHECK-NEXT: [[TMP3:%.*]] = load i32, ptr [[ATOMIC_TEMP]], align 4 |
| // CHECK-NEXT: ret i32 [[TMP3]] |
| // |
| int test_atomic_fetch_nand(int *p) { |
| return __atomic_fetch_nand(p, 55, memory_order_seq_cst); |
| } |
| |
| // CHECK-LABEL: define dso_local i32 @test_atomic_add_fetch( |
| // CHECK-SAME: ptr noundef [[P:%.*]]) #[[ATTR0]] { |
| // CHECK-NEXT: entry: |
| // CHECK-NEXT: [[P_ADDR:%.*]] = alloca ptr, align 4 |
| // CHECK-NEXT: [[DOTATOMICTMP:%.*]] = alloca i32, align 4 |
| // CHECK-NEXT: [[ATOMIC_TEMP:%.*]] = alloca i32, align 4 |
| // CHECK-NEXT: store ptr [[P]], ptr [[P_ADDR]], align 4 |
| // CHECK-NEXT: [[TMP0:%.*]] = load ptr, ptr [[P_ADDR]], align 4 |
| // CHECK-NEXT: store i32 55, ptr [[DOTATOMICTMP]], align 4 |
| // CHECK-NEXT: [[TMP1:%.*]] = load i32, ptr [[DOTATOMICTMP]], align 4 |
| // CHECK-NEXT: [[TMP2:%.*]] = atomicrmw add ptr [[TMP0]], i32 [[TMP1]] seq_cst, align 4 |
| // CHECK-NEXT: [[TMP3:%.*]] = add i32 [[TMP2]], [[TMP1]] |
| // CHECK-NEXT: store i32 [[TMP3]], ptr [[ATOMIC_TEMP]], align 4 |
| // CHECK-NEXT: [[TMP4:%.*]] = load i32, ptr [[ATOMIC_TEMP]], align 4 |
| // CHECK-NEXT: ret i32 [[TMP4]] |
| // |
| int test_atomic_add_fetch(int *p) { |
| return __atomic_add_fetch(p, 55, memory_order_seq_cst); |
| } |
| |
| // CHECK-LABEL: define dso_local i32 @test_atomic_sub_fetch( |
| // CHECK-SAME: ptr noundef [[P:%.*]]) #[[ATTR0]] { |
| // CHECK-NEXT: entry: |
| // CHECK-NEXT: [[P_ADDR:%.*]] = alloca ptr, align 4 |
| // CHECK-NEXT: [[DOTATOMICTMP:%.*]] = alloca i32, align 4 |
| // CHECK-NEXT: [[ATOMIC_TEMP:%.*]] = alloca i32, align 4 |
| // CHECK-NEXT: store ptr [[P]], ptr [[P_ADDR]], align 4 |
| // CHECK-NEXT: [[TMP0:%.*]] = load ptr, ptr [[P_ADDR]], align 4 |
| // CHECK-NEXT: store i32 55, ptr [[DOTATOMICTMP]], align 4 |
| // CHECK-NEXT: [[TMP1:%.*]] = load i32, ptr [[DOTATOMICTMP]], align 4 |
| // CHECK-NEXT: [[TMP2:%.*]] = atomicrmw sub ptr [[TMP0]], i32 [[TMP1]] seq_cst, align 4 |
| // CHECK-NEXT: [[TMP3:%.*]] = sub i32 [[TMP2]], [[TMP1]] |
| // CHECK-NEXT: store i32 [[TMP3]], ptr [[ATOMIC_TEMP]], align 4 |
| // CHECK-NEXT: [[TMP4:%.*]] = load i32, ptr [[ATOMIC_TEMP]], align 4 |
| // CHECK-NEXT: ret i32 [[TMP4]] |
| // |
| int test_atomic_sub_fetch(int *p) { |
| return __atomic_sub_fetch(p, 55, memory_order_seq_cst); |
| } |
| |
| // CHECK-LABEL: define dso_local i32 @test_atomic_and_fetch( |
| // CHECK-SAME: ptr noundef [[P:%.*]]) #[[ATTR0]] { |
| // CHECK-NEXT: entry: |
| // CHECK-NEXT: [[P_ADDR:%.*]] = alloca ptr, align 4 |
| // CHECK-NEXT: [[DOTATOMICTMP:%.*]] = alloca i32, align 4 |
| // CHECK-NEXT: [[ATOMIC_TEMP:%.*]] = alloca i32, align 4 |
| // CHECK-NEXT: store ptr [[P]], ptr [[P_ADDR]], align 4 |
| // CHECK-NEXT: [[TMP0:%.*]] = load ptr, ptr [[P_ADDR]], align 4 |
| // CHECK-NEXT: store i32 55, ptr [[DOTATOMICTMP]], align 4 |
| // CHECK-NEXT: [[TMP1:%.*]] = load i32, ptr [[DOTATOMICTMP]], align 4 |
| // CHECK-NEXT: [[TMP2:%.*]] = atomicrmw and ptr [[TMP0]], i32 [[TMP1]] seq_cst, align 4 |
| // CHECK-NEXT: [[TMP3:%.*]] = and i32 [[TMP2]], [[TMP1]] |
| // CHECK-NEXT: store i32 [[TMP3]], ptr [[ATOMIC_TEMP]], align 4 |
| // CHECK-NEXT: [[TMP4:%.*]] = load i32, ptr [[ATOMIC_TEMP]], align 4 |
| // CHECK-NEXT: ret i32 [[TMP4]] |
| // |
| int test_atomic_and_fetch(int *p) { |
| return __atomic_and_fetch(p, 55, memory_order_seq_cst); |
| } |
| |
| // CHECK-LABEL: define dso_local i32 @test_atomic_or_fetch( |
| // CHECK-SAME: ptr noundef [[P:%.*]]) #[[ATTR0]] { |
| // CHECK-NEXT: entry: |
| // CHECK-NEXT: [[P_ADDR:%.*]] = alloca ptr, align 4 |
| // CHECK-NEXT: [[DOTATOMICTMP:%.*]] = alloca i32, align 4 |
| // CHECK-NEXT: [[ATOMIC_TEMP:%.*]] = alloca i32, align 4 |
| // CHECK-NEXT: store ptr [[P]], ptr [[P_ADDR]], align 4 |
| // CHECK-NEXT: [[TMP0:%.*]] = load ptr, ptr [[P_ADDR]], align 4 |
| // CHECK-NEXT: store i32 55, ptr [[DOTATOMICTMP]], align 4 |
| // CHECK-NEXT: [[TMP1:%.*]] = load i32, ptr [[DOTATOMICTMP]], align 4 |
| // CHECK-NEXT: [[TMP2:%.*]] = atomicrmw or ptr [[TMP0]], i32 [[TMP1]] seq_cst, align 4 |
| // CHECK-NEXT: [[TMP3:%.*]] = or i32 [[TMP2]], [[TMP1]] |
| // CHECK-NEXT: store i32 [[TMP3]], ptr [[ATOMIC_TEMP]], align 4 |
| // CHECK-NEXT: [[TMP4:%.*]] = load i32, ptr [[ATOMIC_TEMP]], align 4 |
| // CHECK-NEXT: ret i32 [[TMP4]] |
| // |
| int test_atomic_or_fetch(int *p) { |
| return __atomic_or_fetch(p, 55, memory_order_seq_cst); |
| } |
| |
| // CHECK-LABEL: define dso_local i32 @test_atomic_xor_fetch( |
| // CHECK-SAME: ptr noundef [[P:%.*]]) #[[ATTR0]] { |
| // CHECK-NEXT: entry: |
| // CHECK-NEXT: [[P_ADDR:%.*]] = alloca ptr, align 4 |
| // CHECK-NEXT: [[DOTATOMICTMP:%.*]] = alloca i32, align 4 |
| // CHECK-NEXT: [[ATOMIC_TEMP:%.*]] = alloca i32, align 4 |
| // CHECK-NEXT: store ptr [[P]], ptr [[P_ADDR]], align 4 |
| // CHECK-NEXT: [[TMP0:%.*]] = load ptr, ptr [[P_ADDR]], align 4 |
| // CHECK-NEXT: store i32 55, ptr [[DOTATOMICTMP]], align 4 |
| // CHECK-NEXT: [[TMP1:%.*]] = load i32, ptr [[DOTATOMICTMP]], align 4 |
| // CHECK-NEXT: [[TMP2:%.*]] = atomicrmw xor ptr [[TMP0]], i32 [[TMP1]] seq_cst, align 4 |
| // CHECK-NEXT: [[TMP3:%.*]] = xor i32 [[TMP2]], [[TMP1]] |
| // CHECK-NEXT: store i32 [[TMP3]], ptr [[ATOMIC_TEMP]], align 4 |
| // CHECK-NEXT: [[TMP4:%.*]] = load i32, ptr [[ATOMIC_TEMP]], align 4 |
| // CHECK-NEXT: ret i32 [[TMP4]] |
| // |
| int test_atomic_xor_fetch(int *p) { |
| return __atomic_xor_fetch(p, 55, memory_order_seq_cst); |
| } |
| |
| // CHECK-LABEL: define dso_local i32 @test_atomic_nand_fetch( |
| // CHECK-SAME: ptr noundef [[P:%.*]]) #[[ATTR0]] { |
| // CHECK-NEXT: entry: |
| // CHECK-NEXT: [[P_ADDR:%.*]] = alloca ptr, align 4 |
| // CHECK-NEXT: [[DOTATOMICTMP:%.*]] = alloca i32, align 4 |
| // CHECK-NEXT: [[ATOMIC_TEMP:%.*]] = alloca i32, align 4 |
| // CHECK-NEXT: store ptr [[P]], ptr [[P_ADDR]], align 4 |
| // CHECK-NEXT: [[TMP0:%.*]] = load ptr, ptr [[P_ADDR]], align 4 |
| // CHECK-NEXT: store i32 55, ptr [[DOTATOMICTMP]], align 4 |
| // CHECK-NEXT: [[TMP1:%.*]] = load i32, ptr [[DOTATOMICTMP]], align 4 |
| // CHECK-NEXT: [[TMP2:%.*]] = atomicrmw nand ptr [[TMP0]], i32 [[TMP1]] seq_cst, align 4 |
| // CHECK-NEXT: [[TMP3:%.*]] = and i32 [[TMP2]], [[TMP1]] |
| // CHECK-NEXT: [[TMP4:%.*]] = xor i32 [[TMP3]], -1 |
| // CHECK-NEXT: store i32 [[TMP4]], ptr [[ATOMIC_TEMP]], align 4 |
| // CHECK-NEXT: [[TMP5:%.*]] = load i32, ptr [[ATOMIC_TEMP]], align 4 |
| // CHECK-NEXT: ret i32 [[TMP5]] |
| // |
| int test_atomic_nand_fetch(int *p) { |
| return __atomic_nand_fetch(p, 55, memory_order_seq_cst); |
| } |