From 1e4c050d6ceb1198032fdcfcf12c44406cea2d60 Mon Sep 17 00:00:00 2001 From: Peter Dimov Date: Tue, 18 Oct 2022 00:46:59 +0300 Subject: [PATCH] Add result<>::emplace (refs #85) --- include/boost/system/result.hpp | 15 +++ test/Jamfile.v2 | 1 + test/result_emplace.cpp | 176 ++++++++++++++++++++++++++++++++ 3 files changed, 192 insertions(+) create mode 100644 test/result_emplace.cpp diff --git a/include/boost/system/result.hpp b/include/boost/system/result.hpp index 4db8cde..fb2f9bd 100644 --- a/include/boost/system/result.hpp +++ b/include/boost/system/result.hpp @@ -401,6 +401,14 @@ public: return has_error()? variant2::unsafe_get<1>( v_ ): E(); } + // emplace + + template + BOOST_CXX14_CONSTEXPR T& emplace( A&&... a ) + { + return v_.template emplace<0>( std::forward(a)... ); + } + // swap BOOST_CXX14_CONSTEXPR void swap( result& r ) @@ -573,6 +581,13 @@ public: return has_error()? variant2::unsafe_get<1>( v_ ): E(); } + // emplace + + BOOST_CXX14_CONSTEXPR void emplace() + { + v_.template emplace<0>(); + } + // swap BOOST_CXX14_CONSTEXPR void swap( result& r ) diff --git a/test/Jamfile.v2 b/test/Jamfile.v2 index 7a8d185..9d9f146 100644 --- a/test/Jamfile.v2 +++ b/test/Jamfile.v2 @@ -183,3 +183,4 @@ run result_convert_construct.cpp : : : $(CPP11) ; run result_typedefs.cpp : : : $(CPP11) ; run result_value_construct3.cpp : : : $(CPP11) ; run result_error_construct3.cpp : : : $(CPP11) ; +run result_emplace.cpp : : : $(CPP11) ; diff --git a/test/result_emplace.cpp b/test/result_emplace.cpp new file mode 100644 index 0000000..53a0e31 --- /dev/null +++ b/test/result_emplace.cpp @@ -0,0 +1,176 @@ +// Copyright 2017, 2021, 2022 Peter Dimov. +// Distributed under the Boost Software License, Version 1.0. +// https://www.boost.org/LICENSE_1_0.txt + +#include +#include +#include + +using namespace boost::system; + +struct X +{ + static int instances; + + int v_; + + explicit X( int v ): v_( v ) { ++instances; } + + X( int v1, int v2 ): v_( v1+v2 ) { ++instances; } + X( int v1, int v2, int v3 ): v_( v1+v2+v3 ) { ++instances; } + + X( X const& ) = delete; + X& operator=( X const& ) = delete; + + ~X() { --instances; } +}; + +int X::instances = 0; + +struct Y +{ + static int instances; + + Y() noexcept { ++instances; } + + Y( Y const& ) noexcept { ++instances; } + + Y& operator=( Y const& ) = default; + + ~Y() { --instances; } +}; + +int Y::instances = 0; + +int main() +{ + { + result r; + + BOOST_TEST( r.has_value() ); + + r.emplace( 1 ); + + BOOST_TEST( r.has_value() ); + BOOST_TEST_EQ( r.value(), 1 ); + } + + { + result r( ENOENT, generic_category() ); + + BOOST_TEST( !r.has_value() ); + + r.emplace( 1 ); + + BOOST_TEST( r.has_value() ); + BOOST_TEST_EQ( r.value(), 1 ); + } + + BOOST_TEST_EQ( X::instances, 0 ); + + { + result r( 0 ); + + BOOST_TEST( r.has_value() ); + BOOST_TEST_EQ( r.value().v_, 0 ); + BOOST_TEST_EQ( X::instances, 1 ); + + r.emplace( 1 ); + BOOST_TEST( r.has_value() ); + BOOST_TEST_EQ( r.value().v_, 1 ); + BOOST_TEST_EQ( X::instances, 1 ); + + r.emplace( 1, 2 ); + BOOST_TEST( r.has_value() ); + BOOST_TEST_EQ( r.value().v_, 1+2 ); + BOOST_TEST_EQ( X::instances, 1 ); + + r.emplace( 1, 2, 3 ); + BOOST_TEST( r.has_value() ); + BOOST_TEST_EQ( r.value().v_, 1+2+3 ); + BOOST_TEST_EQ( X::instances, 1 ); + } + + BOOST_TEST_EQ( X::instances, 0 ); + + { + result r( ENOENT, generic_category() ); + + BOOST_TEST( !r.has_value() ); + BOOST_TEST_EQ( X::instances, 0 ); + + r.emplace( 1, 2 ); + BOOST_TEST( r.has_value() ); + BOOST_TEST_EQ( r.value().v_, 1+2 ); + BOOST_TEST_EQ( X::instances, 1 ); + } + + BOOST_TEST_EQ( X::instances, 0 ); + BOOST_TEST_EQ( Y::instances, 0 ); + + { + result r( Y{} ); + + BOOST_TEST( !r.has_value() ); + BOOST_TEST_EQ( Y::instances, 1 ); + + r.emplace( 1 ); + BOOST_TEST( r.has_value() ); + BOOST_TEST_EQ( *r, 1 ); + BOOST_TEST_EQ( Y::instances, 0 ); + } + + BOOST_TEST_EQ( X::instances, 0 ); + BOOST_TEST_EQ( Y::instances, 0 ); + + { + result r( in_place_error ); + + BOOST_TEST( !r.has_value() ); + BOOST_TEST_EQ( X::instances, 0 ); + BOOST_TEST_EQ( Y::instances, 1 ); + + r.emplace( 1, 2, 3 ); + + BOOST_TEST( r.has_value() ); + BOOST_TEST_EQ( r->v_, 1+2+3 ); + BOOST_TEST_EQ( X::instances, 1 ); + BOOST_TEST_EQ( Y::instances, 0 ); + } + + BOOST_TEST_EQ( X::instances, 0 ); + BOOST_TEST_EQ( Y::instances, 0 ); + + { + result r; + + BOOST_TEST( r.has_value() ); + + r.emplace(); + BOOST_TEST( r.has_value() ); + } + + { + result r( ENOENT, generic_category() ); + + BOOST_TEST( !r.has_value() ); + + r.emplace(); + BOOST_TEST( r.has_value() ); + } + + BOOST_TEST_EQ( Y::instances, 0 ); + + { + result r( in_place_error ); + + BOOST_TEST( !r.has_value() ); + BOOST_TEST_EQ( Y::instances, 1 ); + + r.emplace(); + BOOST_TEST( r.has_value() ); + BOOST_TEST_EQ( Y::instances, 0 ); + } + + return boost::report_errors(); +}