diff options
| author | Eric Fiselier <eric@efcs.ca> | 2016-07-23 22:19:19 +0000 | 
|---|---|---|
| committer | Eric Fiselier <eric@efcs.ca> | 2016-07-23 22:19:19 +0000 | 
| commit | 58ad17df0f023463f4b4737325ca445ab8581e09 (patch) | |
| tree | c7891b7117d1a451c7050c7ffc69616941320036 | |
| parent | 8ab4108df201eaba5853b1e0c50b25afdc8f8cb1 (diff) | |
| download | bcm5719-llvm-58ad17df0f023463f4b4737325ca445ab8581e09.tar.gz bcm5719-llvm-58ad17df0f023463f4b4737325ca445ab8581e09.zip | |
Implement the in_place tags from p0032r3.
That paper also has changes to any/optional but those will
be implemented later.
llvm-svn: 276537
| -rw-r--r-- | libcxx/include/utility | 58 | ||||
| -rw-r--r-- | libcxx/test/std/utilities/utility/utility.inplace/inplace.pass.cpp | 99 | 
2 files changed, 157 insertions, 0 deletions
| diff --git a/libcxx/include/utility b/libcxx/include/utility index 7b978ad3096..fd8a7f2fc52 100644 --- a/libcxx/include/utility +++ b/libcxx/include/utility @@ -171,6 +171,20 @@ template<class... T>  template<class T, class U=T>      T exchange(T& obj, U&& new_value); + +// 20.2.7, in-place construction // C++17 +struct in_place_tag { in_place_tag() = delete; }; // C++17 +using in_place_t = in_place_tag(&)(unspecified ); +template <class T> +    using in_place_type_t = in_place_tag(&)(unspecified <T>); +template <size_t I> +    using in_place_index_t = in_place_tag(&)(unspecified <I>); +in_place_tag in_place(unspecified ); +template <class T> +    in_place_tag in_place(unspecified <T>); +template <size_t I> +    in_place_tag in_place(unspecified <I>); +  }  // std  */ @@ -179,6 +193,7 @@ template<class T, class U=T>  #include <__tuple>  #include <type_traits>  #include <initializer_list> +#include <__debug>  #if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER)  #pragma GCC system_header @@ -782,6 +797,49 @@ _T1 exchange(_T1& __obj, _T2 && __new_value)  }  #endif  // _LIBCPP_STD_VER > 11 +#if _LIBCPP_STD_VER > 14 + +struct _LIBCPP_TYPE_VIS_ONLY __in_place_tag {}; +template <class> struct _LIBCPP_TYPE_VIS_ONLY __in_place_type_tag {}; +template <size_t> struct _LIBCPP_TYPE_VIS_ONLY __in_place_index_tag {}; + +struct _LIBCPP_TYPE_VIS_ONLY in_place_tag; + +using in_place_t = in_place_tag(&)(__in_place_tag); +template <class _Tp> +using in_place_type_t = in_place_tag(&)(__in_place_type_tag<_Tp>); +template <size_t _Nx> +using in_place_index_t = in_place_tag(&)(__in_place_index_tag<_Nx>); + +struct in_place_tag { +  in_place_tag() = delete; +private: +  explicit in_place_tag(__in_place_tag) {} + +  friend inline in_place_tag in_place(__in_place_tag __t); +  template <class _Tp> +  friend inline in_place_tag in_place(__in_place_type_tag<_Tp>); +  template <size_t _Nx> +  friend inline in_place_tag in_place(__in_place_index_tag<_Nx>); +}; + +inline in_place_tag in_place(__in_place_tag __t) { +    _LIBCPP_ASSERT(false, "The in_place function cannot be invoked"); +    return in_place_tag(__t); +} +template <class _Tp> +inline in_place_tag in_place(__in_place_type_tag<_Tp>) { +    _LIBCPP_ASSERT(false, "The in_place function cannot be invoked"); +    return in_place_tag(__in_place_tag{}); +} +template <size_t _Nx> +inline in_place_tag in_place(__in_place_index_tag<_Nx>) { +    _LIBCPP_ASSERT(false, "The in_place function cannot be invoked"); +    return in_place_tag(__in_place_tag{}); +} + +#endif // _LIBCPP_STD_VER > 14 +  _LIBCPP_END_NAMESPACE_STD  #endif  // _LIBCPP_UTILITY diff --git a/libcxx/test/std/utilities/utility/utility.inplace/inplace.pass.cpp b/libcxx/test/std/utilities/utility/utility.inplace/inplace.pass.cpp new file mode 100644 index 00000000000..0cf8e38f5ab --- /dev/null +++ b/libcxx/test/std/utilities/utility/utility.inplace/inplace.pass.cpp @@ -0,0 +1,99 @@ +//===----------------------------------------------------------------------===// +// +//                     The LLVM Compiler Infrastructure +// +// This file is dual licensed under the MIT and the University of Illinois Open +// Source Licenses. See LICENSE.TXT for details. +// +//===----------------------------------------------------------------------===// + +// UNSUPPORTED: c++98, c++03, c++11, c++14 + +// <utility> + +// struct in_place_tag { in_place_tag() = delete; }; +// +// using in_place_t = in_place_tag(&)(unspecified); +// template <class T> +//     using in_place_type_t = in_place_tag(&)(unspecified<T>); +// template <size_t N> +//     using in_place_index_t = in_place_tag(&)(unspecified<N>); +// +// in_place_tag in_place(unspecified); +// +// template <class T>; +// in_place_tag in_place(unspecified<T>); +// +// template <size_t N> +// in_place_tag in_place(unspecified<N>); + +#include <utility> +#include <cassert> +#include <memory> + +#include "test_macros.h" +#include "type_id.h" + +template <class Tp> +struct CheckRet : std::false_type {}; +template <class Arg> +struct CheckRet<std::in_place_tag(Arg)> : std::true_type {}; + +TypeID const* test_fn(std::in_place_t) { return &makeTypeID<std::in_place_t>(); } +template <class T> +TypeID const* test_fn(std::in_place_type_t<T>) +{ return &makeTypeID<std::in_place_type_t<T>>(); } + +template <size_t I> +TypeID const* test_fn(std::in_place_index_t<I>) +{ return &makeTypeID<std::in_place_index_t<I>>(); } + +// Concrete test overloads that don't have to be deduced. +template <class Tag> +TypeID const* concrete_test_fn(Tag) {  return &makeTypeID<Tag>(); } + +template <class Tp> +bool check_tag_basic() { +  using RawTp = typename std::remove_reference<Tp>::type; +  static_assert(std::is_lvalue_reference<Tp>::value, ""); +  static_assert(std::is_function<RawTp>::value, ""); +  static_assert(CheckRet<RawTp>::value, ""); +  auto concrete_fn = concrete_test_fn<Tp>; +  return test_fn((Tp)std::in_place) == &makeTypeID<Tp>() +      && concrete_fn(std::in_place) == &makeTypeID<Tp>(); +} + +int main() { +    // test in_place_tag +    { +        static_assert(!std::is_default_constructible<std::in_place_tag>::value, ""); +    } +    // test in_place_t +    { +        using T = std::in_place_t; +        assert(check_tag_basic<std::in_place_t>()); +        assert(test_fn((T)std::in_place) == &makeTypeID<T>()); +    } +    // test in_place_type_t +    { +        using T1 = std::in_place_type_t<void>; +        using T2 = std::in_place_type_t<int>; +        using T3 = std::in_place_type_t<const int>; +        assert(check_tag_basic<T1>()); +        assert(check_tag_basic<T2>()); +        assert(check_tag_basic<T3>()); +        static_assert(!std::is_same<T1, T2>::value && !std::is_same<T1, T3>::value, ""); +        static_assert(!std::is_same<T2, T3>::value, ""); +    } +    // test in_place_index_t +    { +        using T1 = std::in_place_index_t<0>; +        using T2 = std::in_place_index_t<1>; +        using T3 = std::in_place_index_t<static_cast<size_t>(-1)>; +        assert(check_tag_basic<T1>()); +        assert(check_tag_basic<T2>()); +        assert(check_tag_basic<T3>()); +        static_assert(!std::is_same<T1, T2>::value && !std::is_same<T1, T3>::value, ""); +        static_assert(!std::is_same<T2, T3>::value, ""); +    } +}
\ No newline at end of file | 

