xref: /llvm-project/clang/test/CodeGen/AArch64/strictfp-builtins.c (revision 207e5ccceec8d3cc3f32723e78f2a142bc61b07d)
1*207e5cccSFangrui Song // RUN: %clang_cc1 %s -emit-llvm -ffp-exception-behavior=maytrap -o - -triple arm64-none-linux-gnu | FileCheck %s
2*207e5cccSFangrui Song 
3*207e5cccSFangrui Song // Test that the constrained intrinsics are picking up the exception
4*207e5cccSFangrui Song // metadata from the AST instead of the global default from the command line.
5*207e5cccSFangrui Song 
6*207e5cccSFangrui Song #pragma float_control(except, on)
7*207e5cccSFangrui Song 
8*207e5cccSFangrui Song int printf(const char *, ...);
9*207e5cccSFangrui Song 
10*207e5cccSFangrui Song // CHECK-LABEL: @p(
11*207e5cccSFangrui Song // CHECK-NEXT:  entry:
12*207e5cccSFangrui Song // CHECK-NEXT:    [[STR_ADDR:%.*]] = alloca ptr, align 8
13*207e5cccSFangrui Song // CHECK-NEXT:    [[X_ADDR:%.*]] = alloca i32, align 4
14*207e5cccSFangrui Song // CHECK-NEXT:    store ptr [[STR:%.*]], ptr [[STR_ADDR]], align 8
15*207e5cccSFangrui Song // CHECK-NEXT:    store i32 [[X:%.*]], ptr [[X_ADDR]], align 4
16*207e5cccSFangrui Song // CHECK-NEXT:    [[TMP0:%.*]] = load ptr, ptr [[STR_ADDR]], align 8
17*207e5cccSFangrui Song // CHECK-NEXT:    [[TMP1:%.*]] = load i32, ptr [[X_ADDR]], align 4
18*207e5cccSFangrui Song // CHECK-NEXT:    [[CALL:%.*]] = call i32 (ptr, ...) @printf(ptr noundef @.str, ptr noundef [[TMP0]], i32 noundef [[TMP1]])  [[ATTR4:#.*]]
19*207e5cccSFangrui Song // CHECK-NEXT:    ret void
20*207e5cccSFangrui Song //
21*207e5cccSFangrui Song void p(char *str, int x) {
22*207e5cccSFangrui Song   printf("%s: %d\n", str, x);
23*207e5cccSFangrui Song }
24*207e5cccSFangrui Song 
25*207e5cccSFangrui Song #define P(n,args) p(#n #args, __builtin_##n args)
26*207e5cccSFangrui Song 
27*207e5cccSFangrui Song // CHECK-LABEL: @test_long_double_isinf(
28*207e5cccSFangrui Song // CHECK-NEXT:  entry:
29*207e5cccSFangrui Song // CHECK-NEXT:    [[LD_ADDR:%.*]] = alloca fp128, align 16
30*207e5cccSFangrui Song // CHECK-NEXT:    store fp128 [[D:%.*]], ptr [[LD_ADDR]], align 16
31*207e5cccSFangrui Song // CHECK-NEXT:    [[TMP0:%.*]] = load fp128, ptr [[LD_ADDR]], align 16
32*207e5cccSFangrui Song // CHECK-NEXT:    [[TMP1:%.*]] = call i1 @llvm.is.fpclass.f128(fp128 [[TMP0]], i32 516)
33*207e5cccSFangrui Song // CHECK-NEXT:    [[RES:%.*]] = zext i1 [[TMP1]] to i32
34*207e5cccSFangrui Song // CHECK-NEXT:    call void @p(ptr noundef @.str.[[#STRID:1]], i32 noundef [[RES]]) [[ATTR4]]
35*207e5cccSFangrui Song // CHECK-NEXT:    ret void
36*207e5cccSFangrui Song //
37*207e5cccSFangrui Song void test_long_double_isinf(long double ld) {
38*207e5cccSFangrui Song   P(isinf, (ld));
39*207e5cccSFangrui Song 
40*207e5cccSFangrui Song   return;
41*207e5cccSFangrui Song }
42*207e5cccSFangrui Song 
43*207e5cccSFangrui Song // CHECK-LABEL: @test_long_double_isfinite(
44*207e5cccSFangrui Song // CHECK-NEXT:  entry:
45*207e5cccSFangrui Song // CHECK-NEXT:    [[LD_ADDR:%.*]] = alloca fp128, align 16
46*207e5cccSFangrui Song // CHECK-NEXT:    store fp128 [[D:%.*]], ptr [[LD_ADDR]], align 16
47*207e5cccSFangrui Song // CHECK-NEXT:    [[TMP0:%.*]] = load fp128, ptr [[LD_ADDR]], align 16
48*207e5cccSFangrui Song // CHECK-NEXT:    [[TMP1:%.*]] = call i1 @llvm.is.fpclass.f128(fp128 [[TMP0]], i32 504)
49*207e5cccSFangrui Song // CHECK-NEXT:    [[RES:%.*]] = zext i1 [[TMP1]] to i32
50*207e5cccSFangrui Song // CHECK-NEXT:    call void @p(ptr noundef @.str.[[#STRID:STRID+1]], i32 noundef [[RES]]) [[ATTR4]]
51*207e5cccSFangrui Song // CHECK-NEXT:    ret void
52*207e5cccSFangrui Song //
53*207e5cccSFangrui Song void test_long_double_isfinite(long double ld) {
54*207e5cccSFangrui Song   P(isfinite, (ld));
55*207e5cccSFangrui Song 
56*207e5cccSFangrui Song   return;
57*207e5cccSFangrui Song }
58*207e5cccSFangrui Song 
59*207e5cccSFangrui Song // CHECK-LABEL: @test_long_double_isnan(
60*207e5cccSFangrui Song // CHECK-NEXT:  entry:
61*207e5cccSFangrui Song // CHECK-NEXT:    [[LD_ADDR:%.*]] = alloca fp128, align 16
62*207e5cccSFangrui Song // CHECK-NEXT:    store fp128 [[D:%.*]], ptr [[LD_ADDR]], align 16
63*207e5cccSFangrui Song // CHECK-NEXT:    [[TMP0:%.*]] = load fp128, ptr [[LD_ADDR]], align 16
64*207e5cccSFangrui Song // CHECK-NEXT:    [[TMP1:%.*]] = call i1 @llvm.is.fpclass.f128(fp128 [[TMP0]], i32 3)
65*207e5cccSFangrui Song // CHECK-NEXT:    [[RES:%.*]] = zext i1 [[TMP1]] to i32
66*207e5cccSFangrui Song // CHECK-NEXT:    call void @p(ptr noundef @.str.[[#STRID:STRID+1]], i32 noundef [[RES]])
67*207e5cccSFangrui Song // CHECK-NEXT:    ret void
68*207e5cccSFangrui Song //
69*207e5cccSFangrui Song void test_long_double_isnan(long double ld) {
70*207e5cccSFangrui Song   P(isnan, (ld));
71*207e5cccSFangrui Song 
72*207e5cccSFangrui Song   return;
73*207e5cccSFangrui Song }
74