blob: 8a6d5814199824341e3ce3deab937f0d121d1a1e (
plain) (
blame)
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
85
86
87
|
#pragma once
#include <concepts>
#include <utility>
namespace NYT {
////////////////////////////////////////////////////////////////////////////////
//! Default implementation of wrapper traits you can specialize for your type
template <class T>
requires std::is_object_v<T>
struct TBasicWrapperTraits
{
static constexpr bool IsTrivialWrapper = true;
using TUnwrapped = T;
//! Default implementations just forward the argument
template <class U>
requires std::same_as<std::remove_cvref_t<U>, T>
static constexpr decltype(auto) Unwrap(U&& wrapper) noexcept;
static constexpr bool HasValue(const T& wrapper) noexcept;
};
////////////////////////////////////////////////////////////////////////////////
//! Represents common denominator of every single value wrapper
template <class T>
requires std::is_object_v<T>
struct TWrapperTraits
{
public:
static constexpr bool IsTrivialWrapper = TBasicWrapperTraits<T>::IsTrivialWrapper;
using TWrapped = T;
using TUnwrapped = typename TBasicWrapperTraits<T>::TUnwrapped;
static constexpr bool HasValue(const T& wrapper) noexcept;
static constexpr bool RecursiveHasValue(const T& wrapper) noexcept;
template <class U>
requires std::same_as<std::remove_cvref_t<U>, T>
static constexpr decltype(auto) Unwrap(U&& wrapper);
template <class U>
requires std::same_as<std::remove_cvref_t<U>, T>
static constexpr decltype(auto) RecursiveUnwrap(U&& wrapper);
using TRecursiveUnwrapped = std::remove_cvref_t<decltype(TWrapperTraits<T>::RecursiveUnwrap(std::declval<T>()))>;
//! Unfortunatelly, clang is incapable of processing associated constraints if they depend
//! on class information (e.g. aliases and static varibles) and written out-of-line.
//! TODO(arkady-e1ppa): Add proper constraints when clang supports them:
//! Wrap: std::same_as<std::remove_cvref_t<U>, TUnwrapped>
//! RecursiveWrap: std::same_as<std::remove_cvref_t<U>, TRecursiveUnwrapped>
//! Proper constructible_from checks? Easy for wrap, hard for recursive wrap.
template <class U>
static constexpr T Wrap(U&& unwrapped) noexcept;
template <class U>
static constexpr T RecursiveWrap(U&& unwrapped) noexcept;
};
////////////////////////////////////////////////////////////////////////////////
template <class T>
concept CNonTrivialWrapper =
(!TBasicWrapperTraits<T>::IsTrivialWrapper) &&
requires (T& wrapper, const T& const_wrapper) {
typename TBasicWrapperTraits<T>::TUnwrapped;
{ TBasicWrapperTraits<T>::Unwrap(wrapper) } -> std::same_as<typename TBasicWrapperTraits<T>::TUnwrapped&>;
{ TBasicWrapperTraits<T>::Unwrap(std::move(wrapper)) } -> std::same_as<typename TBasicWrapperTraits<T>::TUnwrapped&&>;
{ TBasicWrapperTraits<T>::Unwrap(const_wrapper) } -> std::same_as<const typename TBasicWrapperTraits<T>::TUnwrapped&>;
{ TBasicWrapperTraits<T>::HasValue(const_wrapper) } -> std::same_as<bool>;
};
////////////////////////////////////////////////////////////////////////////////
} // namespace NYT
#define WRAPPER_TRAITS_INL_H_
#include "wrapper_traits-inl.h"
#undef WRAPPER_TRAITS_INL_H_
|