// RUN: %clang_cc1 -triple x86_64 -emit-llvm %s \ // RUN: -o - | FileCheck -check-prefixes=CHECK,NATIVE %s // RUN: %clang_cc1 -triple riscv32 -target-feature -a -emit-llvm %s \ // RUN: -o - | FileCheck -check-prefixes=CHECK,LIBCALL %s void foo(int x) { _Atomic(int) i = 0; _Atomic(short) j = 0; // Check that multiply / divides on atomics produce a cmpxchg loop i *= 2; // NATIVE: mul nsw i32 // NATIVE: cmpxchg ptr {{.*}} seq_cst, align 4 // LIBCALL: mul nsw i32 // LIBCALL: i1 @__atomic_compare_exchange(i32 noundef 4, i /= 2; // NATIVE: sdiv i32 // NATIVE: cmpxchg ptr {{.*}} seq_cst, align 4 // LIBCALL: sdiv i32 // LIBCALL: i1 @__atomic_compare_exchange(i32 noundef 4, j /= x; // NATIVE: sdiv i32 // NATIVE: cmpxchg ptr {{.*}} seq_cst, align 2 // LIBCALL: sdiv i32 // LIBCALL: i1 @__atomic_compare_exchange(i32 noundef 2, } // LIBCALL: declare void @__atomic_load(i32, ptr, ptr, i32) [[LC_ATTRS:#[0-9]+]] // LIBCALL: declare i1 @__atomic_compare_exchange(i32, ptr, ptr, ptr, i32, i32) [[LC_ATTRS:#[0-9]+]] extern _Atomic _Bool b; _Bool bar(void) { // NATIVE-LABEL: @bar // NATIVE: %[[load:.*]] = load atomic i8, ptr @b seq_cst, align 1 // NATIVE: %[[tobool:.*]] = trunc i8 %[[load]] to i1 // NATIVE: ret i1 %[[tobool]] // LIBCALL-LABEL: @bar // LIBCALL: call void @__atomic_load(i32 noundef 1, ptr noundef @b, ptr noundef %atomic-temp, i32 noundef 5) // LIBCALL: %[[load:.*]] = load i8, ptr %atomic-temp // LIBCALL: %[[tobool:.*]] = trunc i8 %[[load]] to i1 // LIBCALL: ret i1 %[[tobool]] return b; } extern _Atomic(_Complex int) x; void baz(int y) { // NATIVE-LABEL: @baz // NATIVE: store atomic i64 {{.*}} seq_cst, align 8 // LIBCALL-LABEL: @baz // LIBCALL: call void @__atomic_store x += y; } // LIBCALL: declare void @__atomic_store(i32, ptr, ptr, i32) [[LC_ATTRS:#[0-9]+]] _Atomic(int) compound_add(_Atomic(int) in) { // CHECK-LABEL: @compound_add // CHECK: [[OLD:%.*]] = atomicrmw add ptr {{.*}}, i32 5 seq_cst, align 4 // CHECK: [[NEW:%.*]] = add i32 [[OLD]], 5 // CHECK: ret i32 [[NEW]] return (in += 5); } _Atomic(int) compound_sub(_Atomic(int) in) { // CHECK-LABEL: @compound_sub // CHECK: [[OLD:%.*]] = atomicrmw sub ptr {{.*}}, i32 5 seq_cst, align 4 // CHECK: [[NEW:%.*]] = sub i32 [[OLD]], 5 // CHECK: ret i32 [[NEW]] return (in -= 5); } _Atomic(int) compound_xor(_Atomic(int) in) { // CHECK-LABEL: @compound_xor // CHECK: [[OLD:%.*]] = atomicrmw xor ptr {{.*}}, i32 5 seq_cst, align 4 // CHECK: [[NEW:%.*]] = xor i32 [[OLD]], 5 // CHECK: ret i32 [[NEW]] return (in ^= 5); } _Atomic(int) compound_or(_Atomic(int) in) { // CHECK-LABEL: @compound_or // CHECK: [[OLD:%.*]] = atomicrmw or ptr {{.*}}, i32 5 seq_cst, align 4 // CHECK: [[NEW:%.*]] = or i32 [[OLD]], 5 // CHECK: ret i32 [[NEW]] return (in |= 5); } _Atomic(int) compound_and(_Atomic(int) in) { // CHECK-LABEL: @compound_and // CHECK: [[OLD:%.*]] = atomicrmw and ptr {{.*}}, i32 5 seq_cst, align 4 // CHECK: [[NEW:%.*]] = and i32 [[OLD]], 5 // CHECK: ret i32 [[NEW]] return (in &= 5); } _Atomic(int) compound_mul(_Atomic(int) in) { // NATIVE-LABEL: @compound_mul // NATIVE: cmpxchg ptr {{%.*}}, i32 {{%.*}}, i32 [[NEW:%.*]] seq_cst seq_cst, align 4 // NATIVE: ret i32 [[NEW]] // LIBCALL-LABEL: @compound_mul // LIBCALL: i1 @__atomic_compare_exchange(i32 noundef 4, return (in *= 5); } // LIBCALL: [[LC_ATTRS]] = { nounwind willreturn }