mirror of git://gcc.gnu.org/git/gcc.git
				
				
				
			
		
			
				
	
	
		
			339 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			339 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C++
		
	
	
	
| // { dg-options "-std=gnu++17" }
 | |
| // { dg-do run { target c++17 }  }
 | |
| 
 | |
| // Copyright (C) 2013-2018 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 <optional>
 | |
| #include <testsuite_hooks.h>
 | |
| 
 | |
| #include <tuple>
 | |
| 
 | |
| using std::bad_optional_access;
 | |
| static_assert( std::is_default_constructible<bad_optional_access>::value, "" );
 | |
| 
 | |
| struct trivially_destructible
 | |
| {
 | |
|   trivially_destructible() = delete;
 | |
|   trivially_destructible(trivially_destructible const&) = delete;
 | |
|   trivially_destructible& operator=(trivially_destructible const&) = delete;
 | |
|   trivially_destructible(trivially_destructible&&) = delete;
 | |
|   trivially_destructible& operator=(trivially_destructible&&) = delete;
 | |
|   ~trivially_destructible() noexcept = default;
 | |
| };
 | |
| 
 | |
| static_assert( std::is_trivially_destructible<trivially_destructible>(), "" );
 | |
| 
 | |
| struct no_default_constructor
 | |
| {
 | |
|   no_default_constructor() = delete;
 | |
| };
 | |
| 
 | |
| struct no_copy_constructor
 | |
| {
 | |
|   no_copy_constructor() = default;
 | |
|   no_copy_constructor(no_copy_constructor const&) = delete;
 | |
|   no_copy_constructor& operator=(no_copy_constructor const&) = default;
 | |
|   no_copy_constructor(no_copy_constructor&&) = default;
 | |
|   no_copy_constructor& operator=(no_copy_constructor&&) = default;
 | |
| };
 | |
| 
 | |
| struct no_copy_assignment
 | |
| {
 | |
|   no_copy_assignment() = default;
 | |
|   no_copy_assignment(no_copy_assignment const&) = default;
 | |
|   no_copy_assignment(no_copy_assignment&&) = default;
 | |
|   no_copy_assignment& operator=(no_copy_assignment&&) = default;
 | |
| };
 | |
| 
 | |
| struct no_move_constructor
 | |
| {
 | |
|   no_move_constructor() = default;
 | |
|   no_move_constructor(no_move_constructor const&) = default;
 | |
|   no_move_constructor& operator=(no_move_constructor const&) = default;
 | |
|   no_move_constructor(no_move_constructor&&) = delete;
 | |
|   no_move_constructor& operator=(no_move_constructor&&) = default;
 | |
| };
 | |
| 
 | |
| struct no_move_assignment
 | |
| {
 | |
|   no_move_assignment() = default;
 | |
|   no_move_assignment(no_move_assignment const&) = default;
 | |
|   no_move_assignment& operator=(no_move_assignment const&) = default;
 | |
|   no_move_assignment(no_move_assignment&&) = default;
 | |
|   no_move_assignment& operator=(no_move_assignment&&) = delete;
 | |
| };
 | |
| 
 | |
| struct no_copy : no_copy_constructor, no_copy_assignment { };
 | |
| struct no_move : no_move_constructor, no_move_assignment { };
 | |
| 
 | |
| // Laxest possible model of a value type for optional
 | |
| struct only_destructible
 | |
| {
 | |
|   only_destructible(only_destructible&&) = delete;
 | |
| };
 | |
| 
 | |
| int main()
 | |
| {
 | |
|   {
 | |
|     static_assert( std::is_trivially_destructible<std::optional<trivially_destructible>>(), "" );
 | |
|   }
 | |
| 
 | |
|   {
 | |
|     using T = no_default_constructor;
 | |
|     using O = std::optional<T>;
 | |
|     static_assert( std::is_same<O::value_type, T>(), "" );
 | |
|     static_assert( std::is_default_constructible<O>(), "" );
 | |
|     { O o; }
 | |
|     static_assert( std::is_copy_constructible<O>(), "" );
 | |
|     { O o; auto copy = o; }
 | |
|     static_assert( std::is_copy_assignable<O>(), "" );
 | |
|     { O o, p; p = o; }
 | |
|     static_assert( std::is_move_constructible<O>(), "" );
 | |
|     { O o; auto moved_to = std::move(o); }
 | |
|     static_assert( std::is_move_assignable<O>(), "" );
 | |
|     { O o, p; p = std::move(o); }
 | |
|   }
 | |
| 
 | |
|   {
 | |
|     using T = no_copy_constructor;
 | |
|     using O = std::optional<T>;
 | |
|     static_assert( std::is_same<O::value_type, T>(), "" );
 | |
|     static_assert( std::is_default_constructible<O>(), "" );
 | |
|     { O o; }
 | |
|     static_assert( !std::is_copy_constructible<O>(), "" );
 | |
|     static_assert( !std::is_copy_assignable<O>(), "" );
 | |
|     static_assert( std::is_move_constructible<O>(), "" );
 | |
|     { O o; auto moved_to = std::move(o); }
 | |
|     static_assert( std::is_move_assignable<O>(), "" );
 | |
|     { O o, p; p = std::move(o); }
 | |
|   }
 | |
| 
 | |
|   {
 | |
|     using T = no_copy_assignment;
 | |
|     using O = std::optional<T>;
 | |
|     static_assert( std::is_default_constructible<O>(), "" );
 | |
|     { O o; }
 | |
|     static_assert( std::is_copy_constructible<O>(), "" );
 | |
|     { O o; auto copy = o; }
 | |
|     static_assert( !std::is_copy_assignable<O>(), "" );
 | |
|     static_assert( std::is_move_constructible<O>(), "" );
 | |
|     { O o; auto moved_to = std::move(o); }
 | |
|     static_assert( std::is_move_assignable<O>(), "" );
 | |
|     { O o, p; p = std::move(o); }
 | |
|   }
 | |
| 
 | |
|   {
 | |
|     using T = no_copy;
 | |
|     using O = std::optional<T>;
 | |
|     static_assert( std::is_same<O::value_type, T>(), "" );
 | |
|     static_assert( std::is_default_constructible<O>(), "" );
 | |
|     { O o; }
 | |
|     static_assert( !std::is_copy_constructible<O>(), "" );
 | |
|     static_assert( !std::is_copy_assignable<O>(), "" );
 | |
|     static_assert( std::is_move_constructible<O>(), "" );
 | |
|     { O o; auto moved_to = std::move(o); }
 | |
|     static_assert( std::is_move_assignable<O>(), "" );
 | |
|     { O o, p; p = std::move(o); }
 | |
|   }
 | |
| 
 | |
|   {
 | |
|     using T = no_move_constructor;
 | |
|     using O = std::optional<T>;
 | |
|     static_assert( std::is_same<O::value_type, T>(), "" );
 | |
|     static_assert( std::is_default_constructible<O>(), "" );
 | |
|     { O o; }
 | |
|     static_assert( std::is_copy_constructible<O>(), "" );
 | |
|     { O o; auto copy = o; }
 | |
|     static_assert( std::is_copy_assignable<O>(), "" );
 | |
|     /*
 | |
|      * T should be move constructible due to [12.8/11], which is a new rule in C++1y
 | |
|      * not yet implemented by GCC. Because there is already a special exception in C++11
 | |
|      * for the generation of the special members that GCC implements (at least some of the
 | |
|      * time), this does not affect the std::optional implementation however. So the assertion
 | |
|      * for T should be changed (or removed altogether) when the time comes, but the rest
 | |
|      * should however remain correct and unchanged.
 | |
|      */
 | |
|     static_assert( !std::is_move_constructible<T>(), "" );
 | |
|     static_assert( std::is_move_constructible<O>(), "" );
 | |
|     { O o; auto moved_to = std::move(o); }
 | |
|     static_assert( std::is_move_assignable<O>(), "" );
 | |
|     { O o, p; p = std::move(o); }
 | |
|   }
 | |
| 
 | |
|   {
 | |
|     using T = no_move_assignment;
 | |
|     using O = std::optional<T>;
 | |
|     static_assert( std::is_same<O::value_type, T>(), "" );
 | |
|     static_assert( std::is_default_constructible<O>(), "" );
 | |
|     { O o; }
 | |
|     static_assert( std::is_copy_constructible<O>(), "" );
 | |
|     { O o; auto copy = o; }
 | |
|     static_assert( std::is_copy_assignable<O>(), "" );
 | |
|     { O o, p; p = o; }
 | |
|     static_assert( std::is_move_constructible<O>(), "" );
 | |
|     { O o; auto moved_to = std::move(o); }
 | |
|     /*
 | |
|      * Paragraph 23 of same leads to a similar situation but with respect to move
 | |
|      * assignment.
 | |
|      */
 | |
|     static_assert( !std::is_move_assignable<T>(), "" );
 | |
|     static_assert( std::is_move_assignable<O>(), "" );
 | |
|     { O o, p; p = std::move(o); }
 | |
|   }
 | |
| 
 | |
|   {
 | |
|     using T = no_move;
 | |
|     using O = std::optional<T>;
 | |
|     static_assert( std::is_same<O::value_type, T>(), "" );
 | |
|     static_assert( std::is_default_constructible<O>(), "" );
 | |
|     { O o; }
 | |
|     static_assert( std::is_copy_constructible<O>(), "" );
 | |
|     { O o; auto copy = o; }
 | |
|     static_assert( std::is_copy_assignable<O>(), "" );
 | |
|     { O o, p; p = o; }
 | |
|     static_assert( std::is_move_constructible<O>(), "" );
 | |
|     { O o; auto moved_to = std::move(o); }
 | |
|     static_assert( std::is_move_assignable<O>(), "" );
 | |
|     { O o, p; p = std::move(o); }
 | |
|   }
 | |
| 
 | |
|   {
 | |
|     using T = only_destructible;
 | |
|     using O = std::optional<T>;
 | |
|     static_assert( std::is_same<O::value_type, T>(), "" );
 | |
|     static_assert( std::is_default_constructible<O>(), "" );
 | |
|     { O o; }
 | |
|     static_assert( !std::is_copy_constructible<O>(), "" );
 | |
|     static_assert( !std::is_copy_assignable<O>(), "" );
 | |
|     static_assert( !std::is_move_constructible<O>(), "" );
 | |
|     static_assert( !std::is_move_assignable<O>(), "" );
 | |
|   }
 | |
| 
 | |
|   {
 | |
|     /*
 | |
|      * Should not complain about 'invalid' specializations as long as
 | |
|      * they're not instantiated.
 | |
|      */
 | |
|     using A = std::optional<int&>;
 | |
|     using B = std::optional<int&&>;
 | |
|     using C1 = std::optional<std::in_place_t>;
 | |
|     using C2 = std::optional<std::in_place_t const>;
 | |
|     using C3 = std::optional<std::in_place_t volatile>;
 | |
|     using C4 = std::optional<std::in_place_t const volatile>;
 | |
|     using D1 = std::optional<std::nullopt_t>;
 | |
|     using D2 = std::optional<std::nullopt_t const>;
 | |
|     using D3 = std::optional<std::nullopt_t volatile>;
 | |
|     using D4 = std::optional<std::nullopt_t const volatile>;
 | |
| 
 | |
|     using X = std::tuple<A, B, C1, C2, C3, C4, D1, D2, D3, D4>;
 | |
|   }
 | |
| 
 | |
|   {
 | |
|     std::optional<const int> o { 42 };
 | |
|     static_assert( std::is_same<decltype(o)::value_type, const int>(), "" );
 | |
|     VERIFY( o );
 | |
|     VERIFY( *o == 42 );
 | |
|   }
 | |
| 
 | |
|   {
 | |
|     constexpr std::optional<const int> o { 33 };
 | |
|     static_assert( std::is_same<decltype(o)::value_type, const int>(), "" );
 | |
|     static_assert( o, "" );
 | |
|     static_assert( *o == 33, "" );
 | |
|   }
 | |
| }
 | |
| 
 | |
| using std::void_t;
 | |
| using std::declval;
 | |
| using std::true_type;
 | |
| using std::false_type;
 | |
| 
 | |
| template <class T, class = void>
 | |
| struct is_eq_comparable : false_type {};
 | |
| template <class T>
 | |
| struct is_eq_comparable<T, void_t<decltype(declval<T>() == declval<T>())>>
 | |
| : true_type {};
 | |
| 
 | |
| template <class T, class = void>
 | |
| struct is_neq_comparable : false_type {};
 | |
| template <class T>
 | |
| struct is_neq_comparable<T, void_t<decltype(declval<T>() != declval<T>())>>
 | |
| : true_type {};
 | |
| 
 | |
| template <class T, class = void>
 | |
| struct is_lt_comparable : false_type {};
 | |
| template <class T>
 | |
| struct is_lt_comparable<T, void_t<decltype(declval<T>() < declval<T>())>>
 | |
| : true_type {};
 | |
| 
 | |
| template <class T, class = void>
 | |
| struct is_gt_comparable : false_type {};
 | |
| template <class T>
 | |
| struct is_gt_comparable<T, void_t<decltype(declval<T>() > declval<T>())>>
 | |
| : true_type {};
 | |
| 
 | |
| template <class T, class = void>
 | |
| struct is_le_comparable : false_type {};
 | |
| template <class T>
 | |
| struct is_le_comparable<T, void_t<decltype(declval<T>() <= declval<T>())>>
 | |
| : true_type {};
 | |
| 
 | |
| template <class T, class = void>
 | |
| struct is_ge_comparable : false_type {};
 | |
| template <class T>
 | |
| struct is_ge_comparable<T, void_t<decltype(declval<T>() >= declval<T>())>>
 | |
| : true_type {};
 | |
| 
 | |
| using std::optional;
 | |
| 
 | |
| static_assert(is_eq_comparable<optional<int>>::value, "");
 | |
| static_assert(is_neq_comparable<optional<int>>::value, "");
 | |
| static_assert(is_lt_comparable<optional<int>>::value, "");
 | |
| static_assert(is_gt_comparable<optional<int>>::value, "");
 | |
| static_assert(is_le_comparable<optional<int>>::value, "");
 | |
| static_assert(is_ge_comparable<optional<int>>::value, "");
 | |
| 
 | |
| struct JustEq {};
 | |
| bool operator==(const JustEq&, const JustEq&);
 | |
| 
 | |
| static_assert(is_eq_comparable<optional<JustEq>>::value, "");
 | |
| static_assert(!is_neq_comparable<optional<JustEq>>::value, "");
 | |
| static_assert(!is_lt_comparable<optional<JustEq>>::value, "");
 | |
| static_assert(!is_gt_comparable<optional<JustEq>>::value, "");
 | |
| static_assert(!is_le_comparable<optional<JustEq>>::value, "");
 | |
| static_assert(!is_ge_comparable<optional<JustEq>>::value, "");
 | |
| 
 | |
| struct JustLt {};
 | |
| bool operator<(const JustLt&, const JustLt&);
 | |
| 
 | |
| static_assert(!is_eq_comparable<optional<JustLt>>::value, "");
 | |
| static_assert(!is_neq_comparable<optional<JustLt>>::value, "");
 | |
| static_assert(is_lt_comparable<optional<JustLt>>::value, "");
 | |
| static_assert(!is_gt_comparable<optional<JustLt>>::value, "");
 | |
| static_assert(!is_le_comparable<optional<JustLt>>::value, "");
 | |
| static_assert(!is_ge_comparable<optional<JustLt>>::value, "");
 | |
| 
 | |
| static_assert(!std::is_assignable<optional<JustEq>&,
 | |
| 	      optional<JustLt>>::value, "");
 | |
| static_assert(!std::is_assignable<optional<JustEq>&,
 | |
| 	      JustLt>::value, "");
 | |
| static_assert(!std::is_assignable<optional<JustEq>&,
 | |
| 	      optional<JustLt>&>::value, "");
 | |
| static_assert(!std::is_assignable<optional<JustEq>&,
 | |
| 	      JustLt&>::value, "");
 |