1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
|
// { dg-do run { target c++17 } }
// Copyright (C) 2016-2025 Free Software Foundation, Inc.
//
// This file is part of the GNU ISO C++ Library. This library is free
// software; you can redistribute it and/or modify it under the
// terms of the GNU General Public License as published by the
// Free Software Foundation; either version 3, or (at your option)
// any later version.
// This library is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
// You should have received a copy of the GNU General Public License along
// with this library; see the file COPYING3. If not see
// <http://www.gnu.org/licenses/>.
#include <variant>
#include <testsuite_hooks.h>
class S{}; // No hash specialization
template<class T>
auto f(int) -> decltype(std::hash<std::variant<T>>(), std::true_type());
template<class T>
auto f(...) -> decltype(std::false_type());
static_assert(!decltype(f<S>(0))::value, "");
static_assert(!decltype(f<std::variant<S>>(0))::value, "");
static_assert(!decltype(f<std::variant<S, S>>(0))::value, "");
static_assert(decltype(f<std::variant<int>>(0))::value, "");
static_assert(decltype(f<std::variant<int, int>>(0))::value, "");
static_assert(!std::is_invocable_v<
std::hash<std::variant<S>>&, std::variant<S> const&> );
static_assert(!std::is_invocable_v<
std::hash<std::variant<S, int>>&, std::variant<S, int> const&> );
static_assert(std::is_invocable_v<
std::hash<std::variant<int>>&, std::variant<int> const&> );
static_assert(std::is_invocable_v<
std::hash<std::variant<int, int>>&, std::variant<int, int> const&> );
int main()
{
int x = 42;
std::variant<int> x2 = 42;
VERIFY(std::hash<int>()(x) == std::hash<std::variant<int>>()(x2));
}
// Check for presence/absence of nested types.
template<typename T> using res_type = typename std::hash<T>::result_type;
template<typename T> using arg_type = typename std::hash<T>::argument_type;
template<typename Variant, typename = void>
constexpr bool has_res_type = false;
template<typename Variant>
constexpr bool has_res_type<Variant, std::void_t<res_type<Variant>>> = true;
template<typename Variant, typename = void>
constexpr bool has_arg_type = false;
template<typename Variant>
constexpr bool has_arg_type<Variant, std::void_t<arg_type<Variant>>> = true;
template<typename... Ts>
constexpr bool has_no_types
= ! has_res_type<std::variant<Ts...>> && ! has_arg_type<std::variant<Ts...>>;
#if __cplusplus >= 202002L
// Nested types result_type and argument_type are not present in C++20
static_assert( has_no_types<int> );
static_assert( has_no_types<int, double> );
#else
// Nested types result_type and argument_type are deprecated in C++17.
using R1 = std::hash<std::variant<int>>::result_type; // { dg-warning "deprecated" "" { target c++17_only } }
using A1 = std::hash<std::variant<int>>::argument_type; // { dg-warning "deprecated" "" { target c++17_only } }
using R2 = std::hash<std::variant<char, int>>::result_type; // { dg-warning "deprecated" "" { target c++17_only } }
using A2 = std::hash<std::variant<char, int>>::argument_type; // { dg-warning "deprecated" "" { target c++17_only } }
#endif
// Disabled specializations do not have the nested types.
static_assert( has_no_types<S> );
static_assert( has_no_types<int, S> );
|