1*879cbac0Szoecarver //===----------------------------------------------------------------------===//
2*879cbac0Szoecarver //
3*879cbac0Szoecarver // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4*879cbac0Szoecarver // See https://llvm.org/LICENSE.txt for license information.
5*879cbac0Szoecarver // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6*879cbac0Szoecarver //
7*879cbac0Szoecarver //===----------------------------------------------------------------------===//
8*879cbac0Szoecarver
9*879cbac0Szoecarver // UNSUPPORTED: c++03, c++11, c++14, c++17
10*879cbac0Szoecarver
11*879cbac0Szoecarver // <functional>
12*879cbac0Szoecarver
13*879cbac0Szoecarver // ranges::not_equal_to
14*879cbac0Szoecarver
15*879cbac0Szoecarver #include <functional>
16*879cbac0Szoecarver #include <type_traits>
17*879cbac0Szoecarver #include <cassert>
18*879cbac0Szoecarver
19*879cbac0Szoecarver #include "test_macros.h"
20*879cbac0Szoecarver #include "compare_types.h"
21*879cbac0Szoecarver #include "MoveOnly.h"
22*879cbac0Szoecarver #include "pointer_comparison_test_helper.h"
23*879cbac0Szoecarver
24*879cbac0Szoecarver struct NotEqualityComparable {
25*879cbac0Szoecarver friend bool operator==(const NotEqualityComparable&, const NotEqualityComparable&);
26*879cbac0Szoecarver friend bool operator!=(const NotEqualityComparable&, const NotEqualityComparable&) = delete;
27*879cbac0Szoecarver };
28*879cbac0Szoecarver
29*879cbac0Szoecarver static_assert(!std::is_invocable_v<std::ranges::equal_to, NotEqualityComparable, NotEqualityComparable>);
30*879cbac0Szoecarver static_assert(!std::is_invocable_v<std::ranges::equal_to, int, MoveOnly>);
31*879cbac0Szoecarver static_assert(std::is_invocable_v<std::ranges::equal_to, explicit_operators, explicit_operators>);
32*879cbac0Szoecarver
33*879cbac0Szoecarver static_assert(requires { typename std::ranges::not_equal_to::is_transparent; });
34*879cbac0Szoecarver
35*879cbac0Szoecarver struct PtrAndNotEqOperator {
operator void*PtrAndNotEqOperator36*879cbac0Szoecarver constexpr operator void*() const { return nullptr; }
37*879cbac0Szoecarver // We *don't* want operator!= to be picked here.
operator !=(PtrAndNotEqOperator,PtrAndNotEqOperator)38*879cbac0Szoecarver friend constexpr bool operator!=(PtrAndNotEqOperator, PtrAndNotEqOperator) { return true; }
39*879cbac0Szoecarver };
40*879cbac0Szoecarver
test()41*879cbac0Szoecarver constexpr bool test() {
42*879cbac0Szoecarver auto fn = std::ranges::not_equal_to();
43*879cbac0Szoecarver
44*879cbac0Szoecarver assert(fn(MoveOnly(41), MoveOnly(42)));
45*879cbac0Szoecarver
46*879cbac0Szoecarver // These are the opposite of other tests.
47*879cbac0Szoecarver ForwardingTestObject a;
48*879cbac0Szoecarver ForwardingTestObject b;
49*879cbac0Szoecarver assert(fn(a, b));
50*879cbac0Szoecarver assert(!fn(std::move(a), std::move(b)));
51*879cbac0Szoecarver
52*879cbac0Szoecarver assert(fn(1, 2));
53*879cbac0Szoecarver assert(!fn(2, 2));
54*879cbac0Szoecarver assert(fn(2, 1));
55*879cbac0Szoecarver
56*879cbac0Szoecarver assert(fn(2, 1L));
57*879cbac0Szoecarver
58*879cbac0Szoecarver // Make sure that "ranges::equal_to(x, y) == !ranges::not_equal_to(x, y)", even here.
59*879cbac0Szoecarver assert(!fn(PtrAndNotEqOperator(), PtrAndNotEqOperator()));
60*879cbac0Szoecarver assert(std::ranges::equal_to()(PtrAndNotEqOperator(), PtrAndNotEqOperator()));
61*879cbac0Szoecarver
62*879cbac0Szoecarver return true;
63*879cbac0Szoecarver }
64*879cbac0Szoecarver
main(int,char **)65*879cbac0Szoecarver int main(int, char**) {
66*879cbac0Szoecarver
67*879cbac0Szoecarver test();
68*879cbac0Szoecarver static_assert(test());
69*879cbac0Szoecarver
70*879cbac0Szoecarver // test total ordering of int* for not_equal_to<int*> and not_equal_to<void>.
71*879cbac0Szoecarver do_pointer_comparison_test(std::ranges::not_equal_to());
72*879cbac0Szoecarver
73*879cbac0Szoecarver return 0;
74*879cbac0Szoecarver }
75