1 // RUN: %clang_cc1 -std=c++2a -emit-llvm %s -o - -triple %itanium_abi_triple | \
2 // RUN:    FileCheck %s \
3 // RUN:          '-DWE="class.std::__1::weak_equality"' \
4 // RUN:          '-DSO="class.std::__1::strong_ordering"' \
5 // RUN:          '-DSE="class.std::__1::strong_equality"' \
6 // RUN:          '-DPO="class.std::__1::partial_ordering"' \
7 // RUN:           -DEQ=0 -DLT=-1 -DGT=1 -DUNORD=-127 -DNE=1
8 
9 #include "Inputs/std-compare.h"
10 
11 // Ensure we don't emit definitions for the global variables
12 // since the builtins shouldn't ODR use them.
13 // CHECK-NOT: constant %[[SO]]
14 // CHECK-NOT: constant %[[SE]]
15 // CHECK-NOT: constant %[[WE]]
16 // CHECK-NOT: constant %[[PO]]
17 
18 // CHECK-LABEL: @_Z11test_signedii
test_signed(int x,int y)19 auto test_signed(int x, int y) {
20   // CHECK: %[[DEST:retval|agg.result]]
21   // CHECK: %cmp.lt = icmp slt i32 %0, %1
22   // CHECK: %sel.lt = select i1 %cmp.lt, i8 [[LT]], i8 [[GT]]
23   // CHECK: %cmp.eq = icmp eq i32 %0, %1
24   // CHECK: %sel.eq = select i1 %cmp.eq, i8 [[EQ]], i8 %sel.lt
25   // CHECK: %__value_ = getelementptr inbounds %[[SO]], %[[SO]]* %[[DEST]]
26   // CHECK: store i8 %sel.eq, i8* %__value_, align 1
27   // CHECK: ret
28   return x <=> y;
29 }
30 
31 // CHECK-LABEL: @_Z13test_unsignedjj
test_unsigned(unsigned x,unsigned y)32 auto test_unsigned(unsigned x, unsigned y) {
33   // CHECK: %[[DEST:retval|agg.result]]
34   // CHECK: %cmp.lt = icmp ult i32 %0, %1
35   // CHECK: %sel.lt = select i1 %cmp.lt, i8 [[LT]], i8 [[GT]]
36   // CHECK: %cmp.eq = icmp eq i32 %0, %1
37   // CHECK: %sel.eq = select i1 %cmp.eq, i8 [[EQ]], i8 %sel.lt
38   // CHECK: %__value_ = getelementptr inbounds %[[SO]], %[[SO]]* %[[DEST]]
39   // CHECK: store i8 %sel.eq, i8* %__value_
40   // CHECK: ret
41   return x <=> y;
42 }
43 
44 // CHECK-LABEL: @_Z10float_testdd
float_test(double x,double y)45 auto float_test(double x, double y) {
46   // CHECK: %[[DEST:retval|agg.result]]
47   // CHECK: %cmp.eq = fcmp oeq double %0, %1
48   // CHECK: %sel.eq = select i1 %cmp.eq, i8 [[EQ]], i8 [[UNORD]]
49   // CHECK: %cmp.gt = fcmp ogt double %0, %1
50   // CHECK: %sel.gt = select i1 %cmp.gt, i8 [[GT]], i8 %sel.eq
51   // CHECK: %cmp.lt = fcmp olt double %0, %1
52   // CHECK: %sel.lt = select i1 %cmp.lt, i8 [[LT]], i8 %sel.gt
53   // CHECK: %__value_ = getelementptr inbounds %[[PO]], %[[PO]]* %[[DEST]]
54   // CHECK: store i8 %sel.lt, i8* %__value_
55   // CHECK: ret
56   return x <=> y;
57 }
58 
59 // CHECK-LABEL: @_Z8ptr_testPiS_
ptr_test(int * x,int * y)60 auto ptr_test(int *x, int *y) {
61   // CHECK: %[[DEST:retval|agg.result]]
62   // CHECK: %cmp.lt = icmp ult i32* %0, %1
63   // CHECK: %sel.lt = select i1 %cmp.lt, i8 [[LT]], i8 [[GT]]
64   // CHECK: %cmp.eq = icmp eq i32* %0, %1
65   // CHECK: %sel.eq = select i1 %cmp.eq, i8 [[EQ]], i8 %sel.lt
66   // CHECK: %__value_ = getelementptr inbounds %[[SO]], %[[SO]]* %[[DEST]]
67   // CHECK: store i8 %sel.eq, i8* %__value_, align 1
68   // CHECK: ret
69   return x <=> y;
70 }
71 
72 struct MemPtr {};
73 using MemPtrT = void (MemPtr::*)();
74 using MemDataT = int(MemPtr::*);
75 
76 // CHECK-LABEL: @_Z12mem_ptr_testM6MemPtrFvvES1_
mem_ptr_test(MemPtrT x,MemPtrT y)77 auto mem_ptr_test(MemPtrT x, MemPtrT y) {
78   // CHECK: %[[DEST:retval|agg.result]]
79   // CHECK: %cmp.ptr = icmp eq [[TY:i[0-9]+]] %lhs.memptr.ptr, %rhs.memptr.ptr
80   // CHECK: %cmp.ptr.null = icmp eq [[TY]] %lhs.memptr.ptr, 0
81   // CHECK: %cmp.adj = icmp eq [[TY]] %lhs.memptr.adj, %rhs.memptr.adj
82   // CHECK: %[[OR:.*]] = or i1
83   // CHECK-SAME %cmp.adj
84   // CHECK: %memptr.eq = and i1 %cmp.ptr, %[[OR]]
85   // CHECK: %sel.eq = select i1 %memptr.eq, i8 [[EQ]], i8 [[NE]]
86   // CHECK: %__value_ = getelementptr inbounds %[[SE]], %[[SE]]* %[[DEST]]
87   // CHECK: store i8 %sel.eq, i8* %__value_, align 1
88   // CHECK: ret
89   return x <=> y;
90 }
91 
92 // CHECK-LABEL: @_Z13mem_data_testM6MemPtriS0_
mem_data_test(MemDataT x,MemDataT y)93 auto mem_data_test(MemDataT x, MemDataT y) {
94   // CHECK: %[[DEST:retval|agg.result]]
95   // CHECK: %[[CMP:.*]] = icmp eq i{{[0-9]+}} %0, %1
96   // CHECK: %sel.eq = select i1 %[[CMP]], i8 [[EQ]], i8 [[NE]]
97   // CHECK: %__value_ = getelementptr inbounds %[[SE]], %[[SE]]* %[[DEST]]
98   // CHECK: store i8 %sel.eq, i8* %__value_, align 1
99   return x <=> y;
100 }
101 
102 // CHECK-LABEL: @_Z13test_constantv
test_constant()103 auto test_constant() {
104   // CHECK: %[[DEST:retval|agg.result]]
105   // CHECK-NOT: icmp
106   // CHECK: %__value_ = getelementptr inbounds %[[SO]], %[[SO]]* %[[DEST]]
107   // CHECK-NEXT: store i8 -1, i8* %__value_
108   // CHECK: ret
109   const int x = 42;
110   const int y = 101;
111   return x <=> y;
112 }
113 
114 // CHECK-LABEL: @_Z16test_nullptr_objPiDn
test_nullptr_obj(int * x,decltype(nullptr)y)115 auto test_nullptr_obj(int* x, decltype(nullptr) y) {
116   // CHECK: %[[DEST:retval|agg.result]]
117   // CHECK: %cmp.eq = icmp eq i32* %0, null
118   // CHECK: %sel.eq = select i1 %cmp.eq, i8 [[EQ]], i8 [[NE]]
119   // CHECK: %__value_ = getelementptr inbounds %[[SE]], %[[SE]]* %[[DEST]]
120   // CHECK: store i8 %sel.eq, i8* %__value_, align 1
121   return x <=> y;
122 }
123 
124 // CHECK-LABEL: @_Z18unscoped_enum_testijxy
unscoped_enum_test(int i,unsigned u,long long l,unsigned long long ul)125 void unscoped_enum_test(int i, unsigned u, long long l, unsigned long long ul) {
126   enum EnumA : int { A };
127   enum EnumB : unsigned { B };
128   // CHECK: %[[I:.*]] = load {{.*}} %i.addr
129   // CHECK: icmp slt i32 {{.*}} %[[I]]
130   (void)(A <=> i);
131 
132   // CHECK: %[[U:.*]] = load {{.*}} %u.addr
133   // CHECK: icmp ult i32 {{.*}} %[[U]]
134   (void)(A <=> u);
135 
136   // CHECK: %[[L:.*]] = load {{.*}} %l.addr
137   // CHECK: icmp slt i64 {{.*}} %[[L]]
138   (void)(A <=> l);
139 
140   // CHECK: %[[U2:.*]] = load {{.*}} %u.addr
141   // CHECK: icmp ult i32 {{.*}} %[[U2]]
142   (void)(B <=> u);
143 
144   // CHECK: %[[UL:.*]] = load {{.*}} %ul.addr
145   // CHECK: icmp ult i64 {{.*}} %[[UL]]
146   (void)(B <=> ul);
147 }
148 
149 namespace NullptrTest {
150 using nullptr_t = decltype(nullptr);
151 
152 // CHECK-LABEL: @_ZN11NullptrTest4testEDnDn(
test(nullptr_t x,nullptr_t y)153 auto test(nullptr_t x, nullptr_t y) {
154   // CHECK: %[[DEST:retval|agg.result]]
155   // CHECK-NOT: select
156   // CHECK: %__value_ = getelementptr inbounds %[[SE]], %[[SE]]* %[[DEST]]
157   // CHECK-NEXT: store i8 [[EQ]], i8* %__value_
158   // CHECK: ret
159   return x <=> y;
160 }
161 } // namespace NullptrTest
162 
163 namespace ComplexTest {
164 
test_float(_Complex float x,_Complex float y)165 auto test_float(_Complex float x, _Complex float y) {
166   // CHECK: %[[DEST:retval|agg.result]]
167   // CHECK: %cmp.eq.r = fcmp oeq float %x.real, %y.real
168   // CHECK: %cmp.eq.i = fcmp oeq float %x.imag, %y.imag
169   // CHECK: %and.eq = and i1 %cmp.eq.r, %cmp.eq.i
170   // CHECK: %sel.eq = select i1 %and.eq, i8 [[EQ]], i8 [[NE]]
171   // CHECK: %__value_ = getelementptr inbounds %[[WE]], %[[WE]]* %[[DEST]]
172   // CHECK: store i8 %sel.eq, i8* %__value_, align 1
173   return x <=> y;
174 }
175 
176 // CHECK-LABEL: @_ZN11ComplexTest8test_intECiS0_(
test_int(_Complex int x,_Complex int y)177 auto test_int(_Complex int x, _Complex int y) {
178   // CHECK: %[[DEST:retval|agg.result]]
179   // CHECK: %cmp.eq.r = icmp eq i32 %x.real, %y.real
180   // CHECK: %cmp.eq.i = icmp eq i32 %x.imag, %y.imag
181   // CHECK: %and.eq = and i1 %cmp.eq.r, %cmp.eq.i
182   // CHECK: %sel.eq = select i1 %and.eq, i8 [[EQ]], i8 [[NE]]
183   // CHECK: %__value_ = getelementptr inbounds %[[SE]], %[[SE]]* %[[DEST]]
184   // CHECK: store i8 %sel.eq, i8* %__value_, align 1
185   return x <=> y;
186 }
187 
188 } // namespace ComplexTest
189