forked from boostorg/variant2
Require valueless to be first, add test
This commit is contained in:
@ -407,7 +407,7 @@ template<class T1, class... T> union variant_storage_impl<mp_true, T1, T...>
|
|||||||
rest_.emplace( mp_size_t<I-1>(), std::forward<A>(a)... );
|
rest_.emplace( mp_size_t<I-1>(), std::forward<A>(a)... );
|
||||||
}
|
}
|
||||||
|
|
||||||
template<std::size_t I, class... A> constexpr void emplace_impl( mp_true, mp_size_t<I>, A&&... a ) noexcept
|
template<std::size_t I, class... A> constexpr void emplace_impl( mp_true, mp_size_t<I>, A&&... a )
|
||||||
{
|
{
|
||||||
*this = variant_storage_impl( mp_size_t<I>(), std::forward<A>(a)... );
|
*this = variant_storage_impl( mp_size_t<I>(), std::forward<A>(a)... );
|
||||||
}
|
}
|
||||||
@ -459,8 +459,10 @@ template<class U, class... T> using resolve_overload_index = mp_find<mp_list<T..
|
|||||||
|
|
||||||
// variant_base
|
// variant_base
|
||||||
|
|
||||||
|
template<class... T> using can_be_valueless = std::is_same<mp_first<mp_list<T...>>, valueless>;
|
||||||
|
|
||||||
template<bool is_trivially_destructible, bool is_single_buffered, class... T> struct variant_base_impl; // trivially destructible, single buffered
|
template<bool is_trivially_destructible, bool is_single_buffered, class... T> struct variant_base_impl; // trivially destructible, single buffered
|
||||||
template<class... T> using variant_base = variant_base_impl<mp_all<std::is_trivially_destructible<T>...>::value, mp_any<mp_all<std::is_nothrow_move_constructible<T>...>, std::is_same<T, valueless>...>::value, T...>;
|
template<class... T> using variant_base = variant_base_impl<mp_all<std::is_trivially_destructible<T>...>::value, mp_any<mp_all<std::is_nothrow_move_constructible<T>...>, can_be_valueless<T...>>::value, T...>;
|
||||||
|
|
||||||
struct none {};
|
struct none {};
|
||||||
|
|
||||||
@ -517,10 +519,10 @@ template<class... T> struct variant_base_impl<true, true, T...>
|
|||||||
|
|
||||||
template<std::size_t J, class U, class... A> void emplace_impl( mp_false, mp_false, A&&... a )
|
template<std::size_t J, class U, class... A> void emplace_impl( mp_false, mp_false, A&&... a )
|
||||||
{
|
{
|
||||||
std::size_t const K = mp_find<mp_list<T...>, valueless>::value;
|
if( can_be_valueless<T...>::value ) // T0 == valueless
|
||||||
|
|
||||||
if( K < sizeof...(T) ) // have valueless
|
|
||||||
{
|
{
|
||||||
|
std::size_t const K = 0;
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
st1_.emplace( mp_size_t<J>(), std::forward<A>(a)... );
|
st1_.emplace( mp_size_t<J>(), std::forward<A>(a)... );
|
||||||
@ -550,7 +552,7 @@ template<class... T> struct variant_base_impl<true, true, T...>
|
|||||||
std::size_t const J = I+1;
|
std::size_t const J = I+1;
|
||||||
using U = mp_at_c<variant<T...>, I>;
|
using U = mp_at_c<variant<T...>, I>;
|
||||||
|
|
||||||
this->emplace_impl<J, U>( std::is_nothrow_constructible<U, A...>(), mp_all<variant2::detail::is_trivially_move_constructible<U>, variant2::detail::is_trivially_move_assignable<T>...>(), std::forward<A>(a)... );
|
this->emplace_impl<J, U>( std::is_nothrow_constructible<U, A&&...>(), mp_all<variant2::detail::is_trivially_move_constructible<U>, variant2::detail::is_trivially_move_assignable<T>...>(), std::forward<A>(a)... );
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -670,8 +672,6 @@ template<class... T> struct variant_base_impl<false, true, T...>
|
|||||||
{
|
{
|
||||||
size_t const J = I+1;
|
size_t const J = I+1;
|
||||||
|
|
||||||
std::size_t const K = mp_find<mp_list<T...>, valueless>::value;
|
|
||||||
|
|
||||||
using U = mp_at_c<variant<T...>, I>;
|
using U = mp_at_c<variant<T...>, I>;
|
||||||
|
|
||||||
if( std::is_nothrow_constructible<U, A...>::value )
|
if( std::is_nothrow_constructible<U, A...>::value )
|
||||||
@ -681,8 +681,10 @@ template<class... T> struct variant_base_impl<false, true, T...>
|
|||||||
st1_.emplace( mp_size_t<J>(), std::forward<A>(a)... );
|
st1_.emplace( mp_size_t<J>(), std::forward<A>(a)... );
|
||||||
ix_ = J;
|
ix_ = J;
|
||||||
}
|
}
|
||||||
else if( K < sizeof...(T) ) // have valueless
|
else if( can_be_valueless<T...>::value ) // T0 == valueless
|
||||||
{
|
{
|
||||||
|
std::size_t const K = 0;
|
||||||
|
|
||||||
_destroy();
|
_destroy();
|
||||||
|
|
||||||
try
|
try
|
||||||
@ -692,8 +694,8 @@ template<class... T> struct variant_base_impl<false, true, T...>
|
|||||||
}
|
}
|
||||||
catch( ... )
|
catch( ... )
|
||||||
{
|
{
|
||||||
st1_.emplace( mp_size_t<K>() );
|
st1_.emplace( mp_size_t<K+1>() );
|
||||||
ix_ = K;
|
ix_ = K+1;
|
||||||
|
|
||||||
throw;
|
throw;
|
||||||
}
|
}
|
||||||
|
@ -63,3 +63,4 @@ run variant_visit.cpp : : : $(REQ) ;
|
|||||||
run variant_lt_gt.cpp : : : $(REQ) ;
|
run variant_lt_gt.cpp : : : $(REQ) ;
|
||||||
run variant_convert_construct.cpp : : : $(REQ) ;
|
run variant_convert_construct.cpp : : : $(REQ) ;
|
||||||
run variant_subset.cpp : : : $(REQ) ;
|
run variant_subset.cpp : : : $(REQ) ;
|
||||||
|
run variant_valueless.cpp : : : $(REQ) ;
|
||||||
|
235
test/variant_valueless.cpp
Normal file
235
test/variant_valueless.cpp
Normal file
@ -0,0 +1,235 @@
|
|||||||
|
|
||||||
|
// Copyright 2017 Peter Dimov.
|
||||||
|
//
|
||||||
|
// Distributed under the Boost Software License, Version 1.0.
|
||||||
|
//
|
||||||
|
// See accompanying file LICENSE_1_0.txt or copy at
|
||||||
|
// http://www.boost.org/LICENSE_1_0.txt
|
||||||
|
|
||||||
|
#include <boost/variant2/variant.hpp>
|
||||||
|
#include <boost/core/lightweight_test.hpp>
|
||||||
|
#include <type_traits>
|
||||||
|
#include <utility>
|
||||||
|
#include <string>
|
||||||
|
#include <stdexcept>
|
||||||
|
|
||||||
|
using namespace boost::variant2;
|
||||||
|
|
||||||
|
#define STATIC_ASSERT(...) static_assert(__VA_ARGS__, #__VA_ARGS__)
|
||||||
|
|
||||||
|
enum E1
|
||||||
|
{
|
||||||
|
e1
|
||||||
|
};
|
||||||
|
|
||||||
|
enum E2
|
||||||
|
{
|
||||||
|
e2
|
||||||
|
};
|
||||||
|
|
||||||
|
struct X1
|
||||||
|
{
|
||||||
|
X1() = default;
|
||||||
|
X1( E2 ) { throw std::runtime_error( "X1(E2)" ); }
|
||||||
|
};
|
||||||
|
|
||||||
|
STATIC_ASSERT( std::is_nothrow_copy_constructible<X1>::value );
|
||||||
|
STATIC_ASSERT( std::is_nothrow_move_constructible<X1>::value );
|
||||||
|
STATIC_ASSERT( !std::is_nothrow_constructible<X1, E2>::value );
|
||||||
|
STATIC_ASSERT( std::is_trivially_destructible<X1>::value );
|
||||||
|
|
||||||
|
struct X2
|
||||||
|
{
|
||||||
|
X2() = default;
|
||||||
|
~X2();
|
||||||
|
X2( E2 ) { throw std::runtime_error( "X1(E2)" ); }
|
||||||
|
};
|
||||||
|
|
||||||
|
X2::~X2() {}
|
||||||
|
|
||||||
|
STATIC_ASSERT( std::is_nothrow_copy_constructible<X2>::value );
|
||||||
|
STATIC_ASSERT( std::is_nothrow_move_constructible<X2>::value );
|
||||||
|
STATIC_ASSERT( !std::is_nothrow_constructible<X2, E2>::value );
|
||||||
|
STATIC_ASSERT( !std::is_trivially_destructible<X2>::value );
|
||||||
|
|
||||||
|
struct X3
|
||||||
|
{
|
||||||
|
X3() = default;
|
||||||
|
X3( X3 const& ) {}
|
||||||
|
X3( X3&& ) {}
|
||||||
|
};
|
||||||
|
|
||||||
|
STATIC_ASSERT( !std::is_nothrow_copy_constructible<X3>::value );
|
||||||
|
STATIC_ASSERT( !std::is_nothrow_move_constructible<X3>::value );
|
||||||
|
STATIC_ASSERT( std::is_trivially_destructible<X3>::value );
|
||||||
|
|
||||||
|
int main()
|
||||||
|
{
|
||||||
|
{
|
||||||
|
variant<monostate, E1, X1> v;
|
||||||
|
|
||||||
|
v = e1;
|
||||||
|
|
||||||
|
BOOST_TEST_EQ( v.index(), 1 );
|
||||||
|
BOOST_TEST_EQ( get<1>(v), e1 );
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
v = e2;
|
||||||
|
BOOST_ERROR( "`v = e2;` failed to throw" );
|
||||||
|
}
|
||||||
|
catch( std::exception const& )
|
||||||
|
{
|
||||||
|
BOOST_TEST_EQ( v.index(), 1 );
|
||||||
|
BOOST_TEST_EQ( get<1>(v), e1 );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
variant<X3, E1, X1> v;
|
||||||
|
|
||||||
|
v = e1;
|
||||||
|
|
||||||
|
BOOST_TEST_EQ( v.index(), 1 );
|
||||||
|
BOOST_TEST_EQ( get<1>(v), e1 );
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
v = e2;
|
||||||
|
BOOST_ERROR( "`v = e2;` failed to throw" );
|
||||||
|
}
|
||||||
|
catch( std::exception const& )
|
||||||
|
{
|
||||||
|
BOOST_TEST_EQ( v.index(), 1 );
|
||||||
|
BOOST_TEST_EQ( get<1>(v), e1 );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
variant<valueless, E1, X1> v;
|
||||||
|
|
||||||
|
v = e1;
|
||||||
|
|
||||||
|
BOOST_TEST_EQ( v.index(), 1 );
|
||||||
|
BOOST_TEST_EQ( get<1>(v), e1 );
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
v = e2;
|
||||||
|
BOOST_ERROR( "`v = e2;` failed to throw" );
|
||||||
|
}
|
||||||
|
catch( std::exception const& )
|
||||||
|
{
|
||||||
|
// all types are trivial, so we get a constexpr path
|
||||||
|
// and v.index() stays 1
|
||||||
|
|
||||||
|
BOOST_TEST( v.index() == 0 || v.index() == 1 );
|
||||||
|
|
||||||
|
if( v.index() == 1 )
|
||||||
|
{
|
||||||
|
BOOST_TEST_EQ( get<1>(v), e1 );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
variant<monostate, E1, X1, valueless> v;
|
||||||
|
|
||||||
|
v = e1;
|
||||||
|
|
||||||
|
BOOST_TEST_EQ( v.index(), 1 );
|
||||||
|
BOOST_TEST_EQ( get<1>(v), e1 );
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
v = e2;
|
||||||
|
BOOST_ERROR( "`v = e2` failed to throw" );
|
||||||
|
}
|
||||||
|
catch( std::exception const& )
|
||||||
|
{
|
||||||
|
BOOST_TEST_EQ( v.index(), 1 );
|
||||||
|
BOOST_TEST_EQ( get<1>(v), e1 );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
variant<monostate, E1, X2> v;
|
||||||
|
|
||||||
|
v = e1;
|
||||||
|
|
||||||
|
BOOST_TEST_EQ( v.index(), 1 );
|
||||||
|
BOOST_TEST_EQ( get<1>(v), e1 );
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
v = e2;
|
||||||
|
BOOST_ERROR( "`v = e2;` failed to throw" );
|
||||||
|
}
|
||||||
|
catch( std::exception const& )
|
||||||
|
{
|
||||||
|
BOOST_TEST_EQ( v.index(), 1 );
|
||||||
|
BOOST_TEST_EQ( get<1>(v), e1 );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
variant<X3, E1, X2> v;
|
||||||
|
|
||||||
|
v = e1;
|
||||||
|
|
||||||
|
BOOST_TEST_EQ( v.index(), 1 );
|
||||||
|
BOOST_TEST_EQ( get<1>(v), e1 );
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
v = e2;
|
||||||
|
BOOST_ERROR( "`v = e2;` failed to throw" );
|
||||||
|
}
|
||||||
|
catch( std::exception const& )
|
||||||
|
{
|
||||||
|
BOOST_TEST_EQ( v.index(), 1 );
|
||||||
|
BOOST_TEST_EQ( get<1>(v), e1 );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
variant<valueless, E1, X2> v;
|
||||||
|
|
||||||
|
v = e1;
|
||||||
|
|
||||||
|
BOOST_TEST_EQ( v.index(), 1 );
|
||||||
|
BOOST_TEST_EQ( get<1>(v), e1 );
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
v = e2;
|
||||||
|
BOOST_ERROR( "`v = e2;` failed to throw" );
|
||||||
|
}
|
||||||
|
catch( std::exception const& )
|
||||||
|
{
|
||||||
|
BOOST_TEST_EQ( v.index(), 0 );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
variant<monostate, E1, X2, valueless> v;
|
||||||
|
|
||||||
|
v = e1;
|
||||||
|
|
||||||
|
BOOST_TEST_EQ( v.index(), 1 );
|
||||||
|
BOOST_TEST_EQ( get<1>(v), e1 );
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
v = e2;
|
||||||
|
BOOST_ERROR( "`v = e2;` failed to throw" );
|
||||||
|
}
|
||||||
|
catch( std::exception const& )
|
||||||
|
{
|
||||||
|
BOOST_TEST_EQ( v.index(), 1 );
|
||||||
|
BOOST_TEST_EQ( get<1>(v), e1 );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return boost::report_errors();
|
||||||
|
}
|
Reference in New Issue
Block a user