[clang] cbbcc3d - [C2y] Claim conformance to WG14 N3460 (#131196)

via cfe-commits cfe-commits at lists.llvm.org
Fri Mar 14 09:37:12 PDT 2025


Author: Aaron Ballman
Date: 2025-03-14T12:37:07-04:00
New Revision: cbbcc3d13b7747be4587e36ddc0699587ff31904

URL: https://github.com/llvm/llvm-project/commit/cbbcc3d13b7747be4587e36ddc0699587ff31904
DIFF: https://github.com/llvm/llvm-project/commit/cbbcc3d13b7747be4587e36ddc0699587ff31904.diff

LOG: [C2y] Claim conformance to WG14 N3460 (#131196)

This moves some Annex G requirements for complex numbers into the main
body of the standard.

Added: 
    clang/test/C/C2y/n3460.c
    clang/test/C/C2y/n3460_1.c

Modified: 
    clang/www/c_status.html

Removed: 
    


################################################################################
diff  --git a/clang/test/C/C2y/n3460.c b/clang/test/C/C2y/n3460.c
new file mode 100644
index 0000000000000..e21c60b418721
--- /dev/null
+++ b/clang/test/C/C2y/n3460.c
@@ -0,0 +1,51 @@
+// RUN: %clang_cc1 -verify -std=c2y -Wall %s
+
+/* WG14 N3460: Clang 12
+ * Complex operators
+ *
+ * This moves some Annex G requirements into the main body of the standard.
+ */
+
+// CMPLX(0.0, inf) * 2.0, the result should be CMPLX(0.0, inf), not CMPLX(nan, inf)
+static_assert(__builtin_complex(0.0, __builtin_inf()) * 2.0 ==
+              __builtin_complex(0.0, __builtin_inf()));
+
+// CMPLX(0.0, 1.0) * -0.0 is CMPLX(-0.0, -0.0), not CMPLX(-0.0, +0.0)
+static_assert(__builtin_complex(0.0, 1.0) * -0.0 ==
+              __builtin_complex(-0.0, -0.0));
+
+// Testing for -0.0 is a pain because -0.0 == +0.0, so forcefully generate a
+// diagnostic and check the note.
+static_assert(__builtin_complex(0.0, 1.0) * -0.0 == 1); /* expected-error {{static assertion failed due to requirement '__builtin_complex(0., 1.) * -0. == 1'}} \
+                                                           expected-note {{expression evaluates to '(-0 + -0i) == 1'}}
+                                                         */
+
+// CMPLX(0.0, inf) / 2.0, the result should be CMPLX(0.0, inf),
+// not CMPLX(nan, inf)
+static_assert(__builtin_complex(0.0, __builtin_inf()) / 2.0 ==
+              __builtin_complex(0.0, __builtin_inf()));
+
+// CMPLX(2.0, 3.0) * 2.0, the result should be CMPLX(4.0, 6.0)
+static_assert(__builtin_complex(2.0, 3.0) * 2.0 ==
+              __builtin_complex(4.0, 6.0));
+
+// CMPLX(2.0, 4.0) / 2.0, the result should be CMPLX(1.0, 2.0)
+static_assert(__builtin_complex(2.0, 4.0) / 2.0 ==
+              __builtin_complex(1.0, 2.0));
+
+// CMPLX(2.0, 3.0) * CMPLX(4.0, 5.0), the result should be
+// CMPLX(8.0 - 15.0, 12.0 + 10.0)
+static_assert(__builtin_complex(2.0, 3.0) * __builtin_complex(4.0, 5.0) ==
+              __builtin_complex(-7.0, 22.0));
+
+// CMPLX(2.0, 3.0) / CMPLX(4.0, 5.0), the result should be
+// CMPLX((8.0 + 15.0)/(4.0^2 + 5.0^2), (12.0 - 10.0)/(4.0^2 + 5.0^2))
+static_assert(__builtin_complex(2.0, 3.0) / __builtin_complex(4.0, 5.0) ==
+              __builtin_complex(23.0 / 41.0, 2.0 / 41.0));
+
+
+// 2.0 / CMPLX(2.0, 4.0), the result should be
+// CMPLX(4.0 /(2.0^2 + 4.0^2), -8.0/(2.0^2 + 4.0^2))
+static_assert(2.0 / __builtin_complex(2.0, 4.0) ==
+              __builtin_complex(4.0 / 20.0, -8.0 / 20.0));
+

diff  --git a/clang/test/C/C2y/n3460_1.c b/clang/test/C/C2y/n3460_1.c
new file mode 100644
index 0000000000000..2d50b3c23dae8
--- /dev/null
+++ b/clang/test/C/C2y/n3460_1.c
@@ -0,0 +1,80 @@
+// NOTE: Assertions have been autogenerated by utils/update_cc_test_checks.py UTC_ARGS: --version 5
+// RUN: %clang_cc1 -std=c2y -O0 -triple x86_64-unknown-unknown %s -emit-llvm -o - | FileCheck %s
+// This tests the codegen for the same test cases as in n3460.c.
+
+// CHECK-LABEL: define dso_local void @test(
+// CHECK-SAME: ) #[[ATTR0:[0-9]+]] {
+// CHECK-NEXT:  [[ENTRY:.*]]:
+// CHECK-NEXT:    [[A:%.*]] = alloca { double, double }, align 8
+// CHECK-NEXT:    [[B:%.*]] = alloca { double, double }, align 8
+// CHECK-NEXT:    [[C:%.*]] = alloca { double, double }, align 8
+// CHECK-NEXT:    [[D:%.*]] = alloca { double, double }, align 8
+// CHECK-NEXT:    [[E:%.*]] = alloca { double, double }, align 8
+// CHECK-NEXT:    [[F:%.*]] = alloca { double, double }, align 8
+// CHECK-NEXT:    [[G:%.*]] = alloca { double, double }, align 8
+// CHECK-NEXT:    [[H:%.*]] = alloca { double, double }, align 8
+// CHECK-NEXT:    [[A_REALP:%.*]] = getelementptr inbounds nuw { double, double }, ptr [[A]], i32 0, i32 0
+// CHECK-NEXT:    [[A_IMAGP:%.*]] = getelementptr inbounds nuw { double, double }, ptr [[A]], i32 0, i32 1
+// CHECK-NEXT:    store double 0.000000e+00, ptr [[A_REALP]], align 8
+// CHECK-NEXT:    store double 0x7FF0000000000000, ptr [[A_IMAGP]], align 8
+// CHECK-NEXT:    [[B_REALP:%.*]] = getelementptr inbounds nuw { double, double }, ptr [[B]], i32 0, i32 0
+// CHECK-NEXT:    [[B_IMAGP:%.*]] = getelementptr inbounds nuw { double, double }, ptr [[B]], i32 0, i32 1
+// CHECK-NEXT:    store double -0.000000e+00, ptr [[B_REALP]], align 8
+// CHECK-NEXT:    store double -0.000000e+00, ptr [[B_IMAGP]], align 8
+// CHECK-NEXT:    [[C_REALP:%.*]] = getelementptr inbounds nuw { double, double }, ptr [[C]], i32 0, i32 0
+// CHECK-NEXT:    [[C_IMAGP:%.*]] = getelementptr inbounds nuw { double, double }, ptr [[C]], i32 0, i32 1
+// CHECK-NEXT:    store double 0.000000e+00, ptr [[C_REALP]], align 8
+// CHECK-NEXT:    store double 0x7FF0000000000000, ptr [[C_IMAGP]], align 8
+// CHECK-NEXT:    [[D_REALP:%.*]] = getelementptr inbounds nuw { double, double }, ptr [[D]], i32 0, i32 0
+// CHECK-NEXT:    [[D_IMAGP:%.*]] = getelementptr inbounds nuw { double, double }, ptr [[D]], i32 0, i32 1
+// CHECK-NEXT:    store double 4.000000e+00, ptr [[D_REALP]], align 8
+// CHECK-NEXT:    store double 6.000000e+00, ptr [[D_IMAGP]], align 8
+// CHECK-NEXT:    [[E_REALP:%.*]] = getelementptr inbounds nuw { double, double }, ptr [[E]], i32 0, i32 0
+// CHECK-NEXT:    [[E_IMAGP:%.*]] = getelementptr inbounds nuw { double, double }, ptr [[E]], i32 0, i32 1
+// CHECK-NEXT:    store double 1.000000e+00, ptr [[E_REALP]], align 8
+// CHECK-NEXT:    store double 2.000000e+00, ptr [[E_IMAGP]], align 8
+// CHECK-NEXT:    br i1 false, label %[[COMPLEX_MUL_IMAG_NAN:.*]], label %[[COMPLEX_MUL_CONT:.*]], !prof [[PROF2:![0-9]+]]
+// CHECK:       [[COMPLEX_MUL_IMAG_NAN]]:
+// CHECK-NEXT:    br i1 false, label %[[COMPLEX_MUL_LIBCALL:.*]], label %[[COMPLEX_MUL_CONT]], !prof [[PROF2]]
+// CHECK:       [[COMPLEX_MUL_LIBCALL]]:
+// CHECK-NEXT:    [[CALL:%.*]] = call { double, double } @__muldc3(double noundef 2.000000e+00, double noundef 3.000000e+00, double noundef 4.000000e+00, double noundef 5.000000e+00) #[[ATTR1:[0-9]+]]
+// CHECK-NEXT:    [[TMP0:%.*]] = extractvalue { double, double } [[CALL]], 0
+// CHECK-NEXT:    [[TMP1:%.*]] = extractvalue { double, double } [[CALL]], 1
+// CHECK-NEXT:    br label %[[COMPLEX_MUL_CONT]]
+// CHECK:       [[COMPLEX_MUL_CONT]]:
+// CHECK-NEXT:    [[REAL_MUL_PHI:%.*]] = phi double [ -7.000000e+00, %[[ENTRY]] ], [ -7.000000e+00, %[[COMPLEX_MUL_IMAG_NAN]] ], [ [[TMP0]], %[[COMPLEX_MUL_LIBCALL]] ]
+// CHECK-NEXT:    [[IMAG_MUL_PHI:%.*]] = phi double [ 2.200000e+01, %[[ENTRY]] ], [ 2.200000e+01, %[[COMPLEX_MUL_IMAG_NAN]] ], [ [[TMP1]], %[[COMPLEX_MUL_LIBCALL]] ]
+// CHECK-NEXT:    [[F_REALP:%.*]] = getelementptr inbounds nuw { double, double }, ptr [[F]], i32 0, i32 0
+// CHECK-NEXT:    [[F_IMAGP:%.*]] = getelementptr inbounds nuw { double, double }, ptr [[F]], i32 0, i32 1
+// CHECK-NEXT:    store double [[REAL_MUL_PHI]], ptr [[F_REALP]], align 8
+// CHECK-NEXT:    store double [[IMAG_MUL_PHI]], ptr [[F_IMAGP]], align 8
+// CHECK-NEXT:    [[CALL1:%.*]] = call { double, double } @__divdc3(double noundef 2.000000e+00, double noundef 3.000000e+00, double noundef 4.000000e+00, double noundef 5.000000e+00) #[[ATTR1]]
+// CHECK-NEXT:    [[TMP2:%.*]] = extractvalue { double, double } [[CALL1]], 0
+// CHECK-NEXT:    [[TMP3:%.*]] = extractvalue { double, double } [[CALL1]], 1
+// CHECK-NEXT:    [[G_REALP:%.*]] = getelementptr inbounds nuw { double, double }, ptr [[G]], i32 0, i32 0
+// CHECK-NEXT:    [[G_IMAGP:%.*]] = getelementptr inbounds nuw { double, double }, ptr [[G]], i32 0, i32 1
+// CHECK-NEXT:    store double [[TMP2]], ptr [[G_REALP]], align 8
+// CHECK-NEXT:    store double [[TMP3]], ptr [[G_IMAGP]], align 8
+// CHECK-NEXT:    [[CALL2:%.*]] = call { double, double } @__divdc3(double noundef 2.000000e+00, double noundef 0.000000e+00, double noundef 2.000000e+00, double noundef 4.000000e+00) #[[ATTR1]]
+// CHECK-NEXT:    [[TMP4:%.*]] = extractvalue { double, double } [[CALL2]], 0
+// CHECK-NEXT:    [[TMP5:%.*]] = extractvalue { double, double } [[CALL2]], 1
+// CHECK-NEXT:    [[H_REALP:%.*]] = getelementptr inbounds nuw { double, double }, ptr [[H]], i32 0, i32 0
+// CHECK-NEXT:    [[H_IMAGP:%.*]] = getelementptr inbounds nuw { double, double }, ptr [[H]], i32 0, i32 1
+// CHECK-NEXT:    store double [[TMP4]], ptr [[H_REALP]], align 8
+// CHECK-NEXT:    store double [[TMP5]], ptr [[H_IMAGP]], align 8
+// CHECK-NEXT:    ret void
+//
+void test() {
+  _Complex double a  = __builtin_complex(0.0, __builtin_inf()) * 2.0;
+  _Complex double b = __builtin_complex(0.0, 1.0) * -0.0;
+  _Complex double c = __builtin_complex(0.0, __builtin_inf()) / 2.0;
+  _Complex double d = __builtin_complex(2.0, 3.0) * 2.0;
+  _Complex double e = __builtin_complex(2.0, 4.0) / 2.0;
+  _Complex double f = __builtin_complex(2.0, 3.0) * __builtin_complex(4.0, 5.0);
+  _Complex double g = __builtin_complex(2.0, 3.0) / __builtin_complex(4.0, 5.0);
+  _Complex double h = 2.0 / __builtin_complex(2.0, 4.0);
+}
+
+//.
+// CHECK: [[PROF2]] = !{!"branch_weights", i32 1, i32 1048575}
+//.

diff  --git a/clang/www/c_status.html b/clang/www/c_status.html
index f3f9ffadae7ec..d68e8d6441ed2 100644
--- a/clang/www/c_status.html
+++ b/clang/www/c_status.html
@@ -318,7 +318,7 @@ <h2 id="c2y">C2y implementation status</h2>
     <tr>
       <td>Complex operators</td>
       <td><a href="https://www.open-std.org/jtc1/sc22/wg14/www/docs/n3460.pdf">N3460</a></td>
-      <td class="unknown" align="center">Unknown</td>
+      <td class="full" align="center">Clang 12</td>
 	</tr>
 </table>
 </details>


        


More information about the cfe-commits mailing list