forked from OSchip/llvm-project
424 lines
17 KiB
C++
424 lines
17 KiB
C++
// Force x86-64 because some of our heuristics are actually based
|
|
// on integer sizes.
|
|
|
|
// RUN: %clang_cc1 -triple x86_64-apple-darwin -fcxx-exceptions -fsyntax-only -pedantic -verify -Wsign-compare -std=c++2a %s
|
|
|
|
#include "Inputs/std-compare.h"
|
|
|
|
#define ASSERT_TYPE(...) static_assert(__is_same(__VA_ARGS__))
|
|
#define ASSERT_EXPR_TYPE(Expr, Expect) static_assert(__is_same(decltype(Expr), Expect));
|
|
|
|
void self_compare() {
|
|
int a;
|
|
int *b = nullptr;
|
|
|
|
(void)(a <=> a); // expected-warning {{self-comparison always evaluates to 'std::strong_ordering::equal'}}
|
|
(void)(b <=> b); // expected-warning {{self-comparison always evaluates to 'std::strong_ordering::equal'}}
|
|
}
|
|
|
|
void test0(long a, unsigned long b) {
|
|
enum EnumA : int {A};
|
|
enum EnumB {B};
|
|
enum EnumC {C = 0x10000};
|
|
|
|
(void)((short)a <=> (unsigned short)b);
|
|
|
|
// (a,b)
|
|
(void)(a <=> (unsigned long)b); // expected-error {{argument to 'operator<=>' cannot be narrowed}}
|
|
(void)(a <=> (unsigned int) b);
|
|
(void)(a <=> (unsigned short) b);
|
|
(void)(a <=> (unsigned char) b);
|
|
(void)((long)a <=> b); // expected-error {{argument to 'operator<=>' cannot be narrowed}}
|
|
(void)((int)a <=> b); // expected-error {{argument to 'operator<=>' cannot be narrowed}}
|
|
(void)((short)a <=> b); // expected-error {{argument to 'operator<=>' cannot be narrowed}}
|
|
(void)((signed char)a <=> b); // expected-error {{argument to 'operator<=>' cannot be narrowed}}
|
|
(void)((long)a <=> (unsigned long)b); // expected-error {{argument to 'operator<=>' cannot be narrowed}}
|
|
(void)((int)a <=> (unsigned int)b); // expected-error {{argument to 'operator<=>' cannot be narrowed}}
|
|
(void)((short) a <=> (unsigned short) b);
|
|
(void)((signed char) a <=> (unsigned char) b);
|
|
|
|
// (A,b)
|
|
(void)(A <=> (unsigned long) b);
|
|
(void)(A <=> (unsigned int) b);
|
|
(void)(A <=> (unsigned short) b);
|
|
(void)(A <=> (unsigned char) b);
|
|
(void)((long) A <=> b);
|
|
(void)((int) A <=> b);
|
|
(void)((short) A <=> b);
|
|
(void)((signed char) A <=> b);
|
|
(void)((long) A <=> (unsigned long) b);
|
|
(void)((int) A <=> (unsigned int) b);
|
|
(void)((short) A <=> (unsigned short) b);
|
|
(void)((signed char) A <=> (unsigned char) b);
|
|
|
|
// (a,B)
|
|
(void)(a <=> (unsigned long) B); // expected-error {{argument to 'operator<=>' cannot be narrowed from type 'long' to 'unsigned long'}}
|
|
(void)(a <=> (unsigned int) B);
|
|
(void)(a <=> (unsigned short) B);
|
|
(void)(a <=> (unsigned char) B);
|
|
(void)((long) a <=> B);
|
|
(void)((int) a <=> B);
|
|
(void)((short) a <=> B);
|
|
(void)((signed char) a <=> B);
|
|
(void)((long) a <=> (unsigned long) B); // expected-error {{argument to 'operator<=>' cannot be narrowed from type 'long' to 'unsigned long'}}
|
|
(void)((int) a <=> (unsigned int) B); // expected-error {{argument to 'operator<=>' cannot be narrowed from type 'int' to 'unsigned int'}}
|
|
(void)((short) a <=> (unsigned short) B);
|
|
(void)((signed char) a <=> (unsigned char) B);
|
|
|
|
// (C,b)
|
|
(void)(C <=> (unsigned long) b);
|
|
(void)(C <=> (unsigned int) b);
|
|
(void)(C <=> (unsigned short) b); // expected-warning {{comparison of constant 'C' (65536) with expression of type 'unsigned short' is always 'std::strong_ordering::greater'}}
|
|
(void)(C <=> (unsigned char) b); // expected-warning {{comparison of constant 'C' (65536) with expression of type 'unsigned char' is always 'std::strong_ordering::greater'}}
|
|
(void)((long) C <=> b);
|
|
(void)((int) C <=> b);
|
|
(void)((short) C <=> b);
|
|
(void)((signed char) C <=> b);
|
|
(void)((long) C <=> (unsigned long) b);
|
|
(void)((int) C <=> (unsigned int) b);
|
|
(void)((short) C <=> (unsigned short) b);
|
|
(void)((signed char) C <=> (unsigned char) b);
|
|
|
|
// (a,C)
|
|
(void)(a <=> (unsigned long) C); // expected-error {{argument to 'operator<=>' cannot be narrowed from type 'long' to 'unsigned long'}}
|
|
(void)(a <=> (unsigned int) C);
|
|
(void)(a <=> (unsigned short) C);
|
|
(void)(a <=> (unsigned char) C);
|
|
(void)((long) a <=> C);
|
|
(void)((int) a <=> C);
|
|
(void)((short) a <=> C); // expected-warning {{comparison of constant 'C' (65536) with expression of type 'short' is always 'std::strong_ordering::less'}}
|
|
(void)((signed char) a <=> C); // expected-warning {{comparison of constant 'C' (65536) with expression of type 'signed char' is always 'std::strong_ordering::less'}}
|
|
(void)((long) a <=> (unsigned long) C); // expected-error {{argument to 'operator<=>' cannot be narrowed from type 'long' to 'unsigned long'}}
|
|
(void)((int) a <=> (unsigned int) C); // expected-error {{argument to 'operator<=>' cannot be narrowed from type 'int' to 'unsigned int'}}
|
|
(void)((short) a <=> (unsigned short) C);
|
|
(void)((signed char) a <=> (unsigned char) C);
|
|
|
|
// (0x80000,b)
|
|
(void)(0x80000 <=> (unsigned long) b);
|
|
(void)(0x80000 <=> (unsigned int) b);
|
|
(void)(0x80000 <=> (unsigned short) b); // expected-warning {{result of comparison of constant 524288 with expression of type 'unsigned short' is always 'std::strong_ordering::greater'}}
|
|
(void)(0x80000 <=> (unsigned char) b); // expected-warning {{result of comparison of constant 524288 with expression of type 'unsigned char' is always 'std::strong_ordering::greater'}}
|
|
(void)((long) 0x80000 <=> b);
|
|
(void)((int) 0x80000 <=> b);
|
|
(void)((short) 0x80000 <=> b);
|
|
(void)((signed char) 0x80000 <=> b);
|
|
(void)((long) 0x80000 <=> (unsigned long) b);
|
|
(void)((int) 0x80000 <=> (unsigned int) b);
|
|
(void)((short) 0x80000 <=> (unsigned short) b);
|
|
(void)((signed char) 0x80000 <=> (unsigned char) b);
|
|
|
|
// (a,0x80000)
|
|
(void)(a <=> (unsigned long)0x80000); // expected-error {{argument to 'operator<=>' cannot be narrowed}}
|
|
(void)(a <=> (unsigned int) 0x80000);
|
|
(void)(a <=> (unsigned short) 0x80000);
|
|
(void)(a <=> (unsigned char) 0x80000);
|
|
(void)((long) a <=> 0x80000);
|
|
(void)((int) a <=> 0x80000);
|
|
(void)((short) a <=> 0x80000); // expected-warning {{comparison of constant 524288 with expression of type 'short' is always 'std::strong_ordering::less'}}
|
|
(void)((signed char) a <=> 0x80000); // expected-warning {{comparison of constant 524288 with expression of type 'signed char' is always 'std::strong_ordering::less'}}
|
|
(void)((long)a <=> (unsigned long)0x80000); // expected-error {{argument to 'operator<=>' cannot be narrowed}}
|
|
(void)((int)a <=> (unsigned int)0x80000); // expected-error {{argument to 'operator<=>' cannot be narrowed}}
|
|
(void)((short) a <=> (unsigned short) 0x80000);
|
|
(void)((signed char) a <=> (unsigned char) 0x80000);
|
|
}
|
|
|
|
void test5(bool b, bool b2) {
|
|
enum EnumA { A };
|
|
(void)(b <=> b2); // OK
|
|
(void)(true <=> b); // OK
|
|
(void)(b <=> -10); // expected-error {{invalid operands to binary expression ('bool' and 'int')}}
|
|
(void)(b <=> char(1)); // expected-error {{invalid operands to binary expression ('bool' and 'char')}}
|
|
(void)(b <=> A); // expected-error {{invalid operands to binary expression ('bool' and 'EnumA')}}
|
|
|
|
// FIXME: Should this be accepted when narrowing doesn't occur?
|
|
(void)(b <=> 0); // expected-error {{invalid operands to binary expression ('bool' and 'int')}}
|
|
(void)(b <=> 1); // expected-error {{invalid operands to binary expression ('bool' and 'int')}}
|
|
}
|
|
|
|
void test6(signed char sc) {
|
|
(void)(sc <=> 200); // expected-warning{{comparison of constant 200 with expression of type 'signed char' is always 'std::strong_ordering::less'}}
|
|
(void)(200 <=> sc); // expected-warning{{comparison of constant 200 with expression of type 'signed char' is always 'std::strong_ordering::greater'}}
|
|
}
|
|
|
|
// Test many signedness combinations.
|
|
void test7(unsigned long other) {
|
|
// Common unsigned, other unsigned, constant unsigned
|
|
(void)((unsigned)other <=> (unsigned long)(0x1'ffff'ffff)); // expected-warning{{less}}
|
|
(void)((unsigned)other <=> (unsigned long)(0xffff'ffff));
|
|
(void)((unsigned long)other <=> (unsigned)(0x1'ffff'ffff));
|
|
(void)((unsigned long)other <=> (unsigned)(0xffff'ffff));
|
|
|
|
// Common unsigned, other signed, constant unsigned
|
|
(void)((int)other <=> (unsigned long)(0xffff'ffff'ffff'ffff)); // expected-error {{argument to 'operator<=>' cannot be narrowed}}
|
|
(void)((int)other <=> (unsigned long)(0x0000'0000'ffff'ffff)); // expected-error {{argument to 'operator<=>' cannot be narrowed}}
|
|
(void)((int)other <=> (unsigned long)(0x0000'0000'0fff'ffff)); // expected-error {{argument to 'operator<=>' cannot be narrowed}}
|
|
(void)((int)other <=> (unsigned)(0x8000'0000)); // expected-error {{argument to 'operator<=>' cannot be narrowed}}
|
|
|
|
// Common unsigned, other unsigned, constant signed
|
|
(void)((unsigned long)other <=> (int)(0xffff'ffff)); // expected-error {{argument to 'operator<=>' evaluates to -1, which cannot be narrowed to type 'unsigned long'}}
|
|
|
|
// Common unsigned, other signed, constant signed
|
|
// Should not be possible as the common type should also be signed.
|
|
|
|
// Common signed, other signed, constant signed
|
|
(void)((int)other <=> (long)(0xffff'ffff)); // expected-warning{{less}}
|
|
(void)((int)other <=> (long)(0xffff'ffff'0000'0000)); // expected-warning{{greater}}
|
|
(void)((int)other <=> (long)(0x0fff'ffff));
|
|
(void)((int)other <=> (long)(0xffff'ffff'f000'0000));
|
|
|
|
// Common signed, other signed, constant unsigned
|
|
(void)((int)other <=> (unsigned char)(0xffff));
|
|
(void)((int)other <=> (unsigned char)(0xff));
|
|
|
|
// Common signed, other unsigned, constant signed
|
|
(void)((unsigned char)other <=> (int)(0xff));
|
|
(void)((unsigned char)other <=> (int)(0xffff)); // expected-warning{{less}}
|
|
|
|
// Common signed, other unsigned, constant unsigned
|
|
(void)((unsigned char)other <=> (unsigned short)(0xff));
|
|
(void)((unsigned char)other <=> (unsigned short)(0x100)); // expected-warning{{less}}
|
|
(void)((unsigned short)other <=> (unsigned char)(0xff));
|
|
}
|
|
|
|
void test8(void *vp, const void *cvp, int *ip) {
|
|
(void)(vp <=> cvp); // OK, void* comparisons are allowed.
|
|
(void)(vp <=> ip);
|
|
(void)(ip <=> cvp);
|
|
}
|
|
|
|
void test9(long double ld, double d, float f, int i, long long ll) {
|
|
(void)(f <=> ll); // OK, floating-point to integer is OK
|
|
(void)(d <=> ld);
|
|
(void)(i <=> f);
|
|
}
|
|
|
|
typedef int *INTPTR;
|
|
void test_typedef_bug(int *x, INTPTR y) {
|
|
(void)(x <=> y);
|
|
}
|
|
|
|
using nullptr_t = decltype(nullptr);
|
|
|
|
struct Class {};
|
|
struct ClassB : Class {};
|
|
struct Class2 {};
|
|
using FnTy = void(int);
|
|
using FnTy2 = long(int);
|
|
using MemFnTy = void (Class::*)() const;
|
|
using MemFnTyB = void (ClassB::*)() const;
|
|
using MemFnTy2 = void (Class::*)();
|
|
using MemFnTy3 = void (Class2::*)() const;
|
|
using MemDataTy = long(Class::*);
|
|
|
|
void test_nullptr(int *x, FnTy *fp, MemFnTy memp, MemDataTy memdp) {
|
|
auto r1 = (nullptr <=> nullptr);
|
|
ASSERT_EXPR_TYPE(r1, std::strong_equality);
|
|
|
|
auto r2 = (nullptr <=> x);
|
|
ASSERT_EXPR_TYPE(r2, std::strong_equality);
|
|
|
|
auto r3 = (fp <=> nullptr);
|
|
ASSERT_EXPR_TYPE(r3, std::strong_equality);
|
|
|
|
auto r4 = (0 <=> fp);
|
|
ASSERT_EXPR_TYPE(r4, std::strong_equality);
|
|
|
|
auto r5 = (nullptr <=> memp);
|
|
ASSERT_EXPR_TYPE(r5, std::strong_equality);
|
|
|
|
auto r6 = (0 <=> memdp);
|
|
ASSERT_EXPR_TYPE(r6, std::strong_equality);
|
|
|
|
auto r7 = (0 <=> nullptr);
|
|
ASSERT_EXPR_TYPE(r7, std::strong_equality);
|
|
}
|
|
|
|
void test_compatible_pointer(FnTy *f1, FnTy2 *f2, MemFnTy mf1, MemFnTyB mfb,
|
|
MemFnTy2 mf2, MemFnTy3 mf3) {
|
|
(void)(f1 <=> f2); // expected-error {{distinct pointer types}}
|
|
|
|
auto r1 = (mf1 <=> mfb); // OK
|
|
ASSERT_EXPR_TYPE(r1, std::strong_equality);
|
|
ASSERT_EXPR_TYPE((mf1 <=> mfb), std::strong_equality);
|
|
|
|
(void)(mf1 <=> mf2); // expected-error {{distinct pointer types}}
|
|
(void)(mf3 <=> mf1); // expected-error {{distinct pointer types}}
|
|
}
|
|
|
|
// Test that variable narrowing is deferred for value dependent expressions
|
|
template <int Val>
|
|
auto test_template_overflow() {
|
|
// expected-error@+1 {{argument to 'operator<=>' evaluates to -1, which cannot be narrowed to type 'unsigned long'}}
|
|
return (Val <=> (unsigned long)0);
|
|
}
|
|
template auto test_template_overflow<0>();
|
|
template auto test_template_overflow<-1>(); // expected-note {{requested here}}
|
|
|
|
void test_enum_integral_compare() {
|
|
enum EnumA : int {A, ANeg = -1, AMax = __INT_MAX__};
|
|
enum EnumB : unsigned {B, BMax = __UINT32_MAX__ };
|
|
enum EnumC : int {C = -1, C0 = 0};
|
|
|
|
(void)(A <=> C); // expected-error {{invalid operands to binary expression ('EnumA' and 'EnumC')}}
|
|
|
|
(void)(A <=> (unsigned)0);
|
|
(void)((unsigned)0 <=> A);
|
|
(void)(ANeg <=> (unsigned)0); // expected-error {{argument to 'operator<=>' evaluates to -1, which cannot be narrowed to type 'unsigned int'}}
|
|
(void)((unsigned)0 <=> ANeg); // expected-error {{cannot be narrowed}}
|
|
|
|
(void)(B <=> 42);
|
|
(void)(42 <=> B);
|
|
(void)(B <=> (unsigned long long)42);
|
|
(void)(B <=> -1); // expected-error {{argument to 'operator<=>' evaluates to -1, which cannot be narrowed to type 'unsigned int'}}
|
|
(void)(BMax <=> (unsigned long)-1);
|
|
|
|
(void)(C0 <=> (unsigned)42);
|
|
(void)(C <=> (unsigned)42); // expected-error {{argument to 'operator<=>' evaluates to -1, which cannot be narrowed to type 'unsigned int'}}
|
|
}
|
|
|
|
namespace EnumCompareTests {
|
|
|
|
enum class EnumA { A, A2 };
|
|
enum class EnumB { B };
|
|
enum class EnumC : unsigned { C };
|
|
|
|
void test_enum_enum_compare_no_builtin() {
|
|
auto r1 = (EnumA::A <=> EnumA::A2); // OK
|
|
ASSERT_EXPR_TYPE(r1, std::strong_ordering);
|
|
(void)(EnumA::A <=> EnumA::A); // expected-warning {{self-comparison always evaluates to 'std::strong_ordering::equal'}}
|
|
(void)(EnumA::A <=> EnumB::B); // expected-error {{invalid operands to binary expression ('EnumCompareTests::EnumA' and 'EnumCompareTests::EnumB')}}
|
|
(void)(EnumB::B <=> EnumA::A); // expected-error {{invalid operands}}
|
|
}
|
|
|
|
template <int>
|
|
struct Tag {};
|
|
// expected-note@+1 {{candidate}}
|
|
Tag<0> operator<=>(EnumA, EnumA) {
|
|
return {};
|
|
}
|
|
Tag<1> operator<=>(EnumA, EnumB) {
|
|
return {};
|
|
}
|
|
|
|
void test_enum_ovl_provided() {
|
|
auto r1 = (EnumA::A <=> EnumA::A);
|
|
ASSERT_EXPR_TYPE(r1, Tag<0>);
|
|
auto r2 = (EnumA::A <=> EnumB::B);
|
|
ASSERT_EXPR_TYPE(r2, Tag<1>);
|
|
(void)(EnumB::B <=> EnumA::A); // expected-error {{invalid operands to binary expression ('EnumCompareTests::EnumB' and 'EnumCompareTests::EnumA')}}
|
|
}
|
|
|
|
void enum_float_test() {
|
|
enum EnumA { A };
|
|
(void)(A <=> (float)0); // expected-error {{invalid operands to binary expression ('EnumA' and 'float')}}
|
|
(void)((double)0 <=> A); // expected-error {{invalid operands to binary expression ('double' and 'EnumA')}}
|
|
(void)((long double)0 <=> A); // expected-error {{invalid operands to binary expression ('long double' and 'EnumA')}}
|
|
}
|
|
|
|
enum class Bool1 : bool { Zero,
|
|
One };
|
|
enum Bool2 : bool { B2_Zero,
|
|
B2_One };
|
|
|
|
void test_bool_enum(Bool1 A1, Bool1 A2, Bool2 B1, Bool2 B2) {
|
|
(void)(A1 <=> A2);
|
|
(void)(B1 <=> B2);
|
|
}
|
|
|
|
} // namespace EnumCompareTests
|
|
|
|
namespace TestUserDefinedConvSeq {
|
|
|
|
template <class T, T Val>
|
|
struct Conv {
|
|
constexpr operator T() const { return Val; }
|
|
operator T() { return Val; }
|
|
};
|
|
|
|
void test_user_conv() {
|
|
{
|
|
using C = Conv<int, 0>;
|
|
C c;
|
|
const C cc;
|
|
(void)(0 <=> c);
|
|
(void)(c <=> -1);
|
|
(void)((unsigned)0 <=> cc);
|
|
(void)((unsigned)0 <=> c); // expected-error {{argument to 'operator<=>' cannot be narrowed from type 'int' to 'unsigned int'}}
|
|
}
|
|
{
|
|
using C = Conv<int, -1>;
|
|
C c;
|
|
const C cc;
|
|
(void)(c <=> 0);
|
|
(void)(cc <=> (unsigned)0); // expected-error {{argument to 'operator<=>' evaluates to -1, which cannot be narrowed to type 'unsigned int'}}
|
|
(void)(c <=> (unsigned)0); // expected-error {{cannot be narrowed from type 'int' to 'unsigned int'}}
|
|
}
|
|
}
|
|
|
|
} // namespace TestUserDefinedConvSeq
|
|
|
|
void test_array_conv() {
|
|
int arr[5];
|
|
int *ap = arr + 2;
|
|
int arr2[3];
|
|
(void)(arr <=> arr); // expected-error {{invalid operands to binary expression ('int [5]' and 'int [5]')}}
|
|
(void)(+arr <=> arr);
|
|
}
|
|
|
|
void test_mixed_float_int(float f, double d, long double ld) {
|
|
extern int i;
|
|
extern unsigned u;
|
|
extern long l;
|
|
extern short s;
|
|
extern unsigned short us;
|
|
auto r1 = (f <=> i);
|
|
ASSERT_EXPR_TYPE(r1, std::partial_ordering);
|
|
|
|
auto r2 = (us <=> ld);
|
|
ASSERT_EXPR_TYPE(r2, std::partial_ordering);
|
|
|
|
auto r3 = (s <=> f);
|
|
ASSERT_EXPR_TYPE(r3, std::partial_ordering);
|
|
|
|
auto r4 = (0.0 <=> i);
|
|
ASSERT_EXPR_TYPE(r4, std::partial_ordering);
|
|
}
|
|
|
|
namespace NullptrTest {
|
|
using nullptr_t = decltype(nullptr);
|
|
void foo(nullptr_t x, nullptr_t y) {
|
|
auto r = x <=> y;
|
|
ASSERT_EXPR_TYPE(r, std::strong_equality);
|
|
}
|
|
} // namespace NullptrTest
|
|
|
|
namespace ComplexTest {
|
|
|
|
enum class StrongE {};
|
|
enum WeakE { E_One,
|
|
E_Two };
|
|
|
|
void test_diag(_Complex int ci, _Complex float cf, _Complex double cd, int i, float f, StrongE E1, WeakE E2, int *p) {
|
|
(void)(ci <=> (_Complex int &)ci);
|
|
(void)(ci <=> cf);
|
|
(void)(ci <=> i);
|
|
(void)(ci <=> f);
|
|
(void)(cf <=> i);
|
|
(void)(cf <=> f);
|
|
(void)(ci <=> p); // expected-error {{invalid operands}}
|
|
(void)(ci <=> E1); // expected-error {{invalid operands}}
|
|
(void)(E2 <=> cf); // expected-error {{invalid operands}}
|
|
}
|
|
|
|
void test_int(_Complex int x, _Complex int y) {
|
|
auto r = x <=> y;
|
|
ASSERT_EXPR_TYPE(r, std::strong_equality);
|
|
}
|
|
|
|
void test_double(_Complex double x, _Complex double y) {
|
|
auto r = x <=> y;
|
|
ASSERT_EXPR_TYPE(r, std::weak_equality);
|
|
}
|
|
|
|
} // namespace ComplexTest
|