blob: 64dd60ef3da2ca8a39be1fa83f633c6e824e3108 [file] [log] [blame]
Eric Fiselier0683c0e2018-05-07 21:07:10 +00001// 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
19auto test_signed(int x, int y) {
20 // CHECK: %retval = alloca %[[SO]]
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]]* %retval, i32 0, i32 0
26 // CHECK: store i8 %sel.eq, i8* %__value_, align 1
27 // CHECK: %[[FINAL:.*]] = getelementptr inbounds %[[SO]], %[[SO]]* %retval
28 // CHECK: %[[RET:.*]] = load i8, i8* %[[FINAL]]
29 // CHECK: ret i8 %[[RET]]
30 return x <=> y;
31}
32
33// CHECK-LABEL: @_Z13test_unsignedjj
34auto test_unsigned(unsigned x, unsigned y) {
35 // CHECK: %cmp.lt = icmp ult i32 %0, %1
36 // CHECK: %sel.lt = select i1 %cmp.lt, i8 [[LT]], i8 [[GT]]
37 // CHECK: %cmp.eq = icmp eq i32 %0, %1
38 // CHECK: %sel.eq = select i1 %cmp.eq, i8 [[EQ]], i8 %sel.lt
39 // CHECK: %retval
40 // CHECK: %sel.eq
41 // CHECK: ret
42 return x <=> y;
43}
44
45// CHECK-LABEL: @_Z10float_testdd
46auto float_test(double x, double y) {
47 // CHECK: %retval = alloca %[[PO]]
48 // CHECK: %cmp.eq = fcmp oeq double %0, %1
49 // CHECK: %sel.eq = select i1 %cmp.eq, i8 [[EQ]], i8 [[UNORD]]
50 // CHECK: %cmp.gt = fcmp ogt double %0, %1
51 // CHECK: %sel.gt = select i1 %cmp.gt, i8 [[GT]], i8 %sel.eq
52 // CHECK: %cmp.lt = fcmp olt double %0, %1
53 // CHECK: %sel.lt = select i1 %cmp.lt, i8 [[LT]], i8 %sel.gt
54 // CHECK: %retval
55 // CHECK: %sel.lt
56 // CHECK: ret
57 return x <=> y;
58}
59
60// CHECK-LABEL: @_Z8ptr_testPiS_
61auto ptr_test(int *x, int *y) {
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: %retval
67 // CHECK: %sel.eq
68 // CHECK: ret
69 return x <=> y;
70}
71
72struct MemPtr {};
73using MemPtrT = void (MemPtr::*)();
74using MemDataT = int(MemPtr::*);
75
76// CHECK-LABEL: @_Z12mem_ptr_testM6MemPtrFvvES1_
77auto mem_ptr_test(MemPtrT x, MemPtrT y) {
78 // CHECK: %retval = alloca %[[SE]]
79 // CHECK: %cmp.ptr = icmp eq i64 %lhs.memptr.ptr, %rhs.memptr.ptr
80 // CHECK: %cmp.ptr.null = icmp eq i64 %lhs.memptr.ptr, 0
81 // CHECK: %cmp.adj = icmp eq i64 %lhs.memptr.adj, %rhs.memptr.adj
82 // CHECK: %6 = or i1 %cmp.ptr.null, %cmp.adj
83 // CHECK: %memptr.eq = and i1 %cmp.ptr, %6
84 // CHECK: %sel.eq = select i1 %memptr.eq, i8 [[EQ]], i8 [[NE]]
85 // CHECK: %retval
86 // CHECK: %sel.eq
87 // CHECK: ret
88 return x <=> y;
89}
90
91// CHECK-LABEL: @_Z13mem_data_testM6MemPtriS0_
92auto mem_data_test(MemDataT x, MemDataT y) {
93 // CHECK: %retval = alloca %[[SE]]
94 // CHECK: %[[CMP:.*]] = icmp eq i64 %0, %1
95 // CHECK: %sel.eq = select i1 %[[CMP]], i8 [[EQ]], i8 [[NE]]
96 // CHECK: %retval
97 // CHECK: %sel.eq
98 return x <=> y;
99}
100
101// CHECK-LABEL: @_Z13test_constantv
102auto test_constant() {
103 // CHECK: entry:
104 // CHECK-NOT: icmp
105 // CHECK: %__value_ = getelementptr inbounds %[[SO]], %[[SO]]* %retval
106 // CHECK-NEXT: store i8 [[LT]], i8* %__value_
107 // CHECK-NEXT: %[[TMP:.*]] = getelementptr inbounds %[[SO]], %[[SO]]* %retval
108 // CHECK-NEXT: %[[RET:.*]] = load i8, i8* %[[TMP]]
109 // CHECK-NEXT: ret i8 %[[RET]]
110 const int x = 42;
111 const int y = 101;
112 return x <=> y;
113}
114
115// CHECK-LABEL: @_Z16test_nullptr_objPiDn
116auto test_nullptr_obj(int* x, decltype(nullptr) y) {
117 // CHECK: %retval = alloca %[[SE]]
118 // CHECK: %cmp.eq = icmp eq i32* %0, null
119 // CHECK: %sel.eq = select i1 %cmp.eq, i8 [[EQ]], i8 [[NE]]
120 // CHECK: %retval
121 // CHECK: %sel.eq
122 return x <=> y;
123}
124
125// CHECK-LABEL: @_Z18unscoped_enum_testijlm
126void unscoped_enum_test(int i, unsigned u, long l, unsigned long ul) {
127 enum EnumA : int { A };
128 enum EnumB : unsigned { B };
129 // CHECK: %[[I:.*]] = load {{.*}} %i.addr
130 // CHECK: icmp slt i32 {{.*}} %[[I]]
131 (void)(A <=> i);
132
133 // CHECK: %[[U:.*]] = load {{.*}} %u.addr
134 // CHECK: icmp ult i32 {{.*}} %[[U]]
135 (void)(A <=> u);
136
137 // CHECK: %[[L:.*]] = load {{.*}} %l.addr
138 // CHECK: icmp slt i64 {{.*}} %[[L]]
139 (void)(A <=> l);
140
141 // CHECK: %[[U2:.*]] = load {{.*}} %u.addr
142 // CHECK: icmp ult i32 {{.*}} %[[U2]]
143 (void)(B <=> u);
144
145 // CHECK: %[[UL:.*]] = load {{.*}} %ul.addr
146 // CHECK: icmp ult i64 {{.*}} %[[UL]]
147 (void)(B <=> ul);
148}
149
150namespace NullptrTest {
151using nullptr_t = decltype(nullptr);
152
153// CHECK-LABEL: @_ZN11NullptrTest4testEDnDn(
154auto test(nullptr_t x, nullptr_t y) {
155 // CHECK-NOT: select
156 // CHECK: %__value_ = getelementptr inbounds %[[SE]], %[[SE]]* %retval
157 // CHECK-NEXT: store i8 [[EQ]], i8* %__value_
158 // CHECK: ret
159 return x <=> y;
160}
161} // namespace NullptrTest
162
163namespace ComplexTest {
164
165auto test_float(_Complex float x, _Complex float y) {
166 // CHECK: %cmp.eq.r = fcmp oeq float %x.real, %y.real
167 // CHECK: %cmp.eq.i = fcmp oeq float %x.imag, %y.imag
168 // CHECK: %and.eq = and i1 %cmp.eq.r, %cmp.eq.i
169 // CHECK: %sel.eq = select i1 %and.eq, i8 [[EQ]], i8 [[NE]]
170 // CHECK: %__value_ = getelementptr inbounds %[[WE]], %[[WE]]* %retval
171 // CHECK: store i8 %sel.eq, i8* %__value_, align 1
172 return x <=> y;
173}
174
175// CHECK-LABEL: @_ZN11ComplexTest8test_intECiS0_(
176auto test_int(_Complex int x, _Complex int y) {
177 // CHECK: %cmp.eq.r = icmp eq i32 %x.real, %y.real
178 // CHECK: %cmp.eq.i = icmp eq i32 %x.imag, %y.imag
179 // CHECK: %and.eq = and i1 %cmp.eq.r, %cmp.eq.i
180 // CHECK: %sel.eq = select i1 %and.eq, i8 [[EQ]], i8 [[NE]]
181 // CHECK: %__value_ = getelementptr inbounds %[[SE]], %[[SE]]* %retval
182 // CHECK: store i8 %sel.eq, i8* %__value_, align 1
183 return x <=> y;
184}
185
186} // namespace ComplexTest