gcc/libstdc++-v3/testsuite/27_io/rvalue_streams.cc
Jonathan Wakely aa475c4ac8 libstdc++: Implement LWG 1203 for rvalue iostreams
This implements the resolution of LWG 1203 so that the constraints for
rvalue stream insertion/extraction are simpler, and the return type is
the original rvalue stream type not its base class.

Signed-off-by: Jonathan Wakely <jwakely@redhat.com>

libstdc++-v3/ChangeLog:

	* include/std/istream (operator>>(Istream&&, x&)): Simplify, as
	per LWG 1203.
	* include/std/ostream (operator<<(Ostream&&, const x&)):
	Likewise.
	* testsuite/27_io/basic_istream/extractors_character/char/lwg2499_neg.cc:
	Adjust dg-error pattern.
	* testsuite/27_io/basic_istream/extractors_character/wchar_t/lwg2499_neg.cc:
	Likewise.
	* testsuite/27_io/basic_istream/extractors_other/char/4.cc: Define
	is_extractable trait to replace std::__is_extractable. Make it
	work with rvalue streams as well as lvalues, to replace f() and
	g() helper functions.
	* testsuite/27_io/basic_istream/extractors_other/wchar_t/4.cc:
	Likewise.
	* testsuite/27_io/basic_ostream/inserters_other/char/6.cc:
	Define is_insertable trait to replace std::__is_insertable. Make
	it work with rvalue streams as well as lvalues, to replace f()
	and g() helper functions.
	* testsuite/27_io/basic_ostream/inserters_other/wchar_t/6.cc:
	Likewise.
	* testsuite/27_io/filesystem/path/io/dr2989.cc: Prune additional
	errors from new constraints.
	* testsuite/27_io/rvalue_streams-2.cc: Remove PR 80675 checks,
	which are no longer expected to compile.
	* testsuite/27_io/rvalue_streams.cc: Adjust existing test.
	Verify LWG 1203 changes.
2021-04-30 14:33:19 +01:00

133 lines
3.5 KiB
C++

// Copyright (C) 2008-2021 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/>.
// { dg-do run { target c++11 } }
#include <sstream>
#include <string>
#include <testsuite_hooks.h>
void
test01()
{
int i = 1742;
std::ostringstream() << i;
std::string result ("1742");
int i2;
std::istringstream(result) >> i2;
VERIFY (i == i2);
}
struct X { bool as_rvalue; };
void operator>>(std::istream&, X& x) { x.as_rvalue = false; }
void operator>>(std::istream&, X&& x) { x.as_rvalue = true; }
// LWG 2328 Rvalue stream extraction should use perfect forwarding
void
test02()
{
X x;
std::istringstream is;
auto&& ref1 = (std::move(is) >> x);
VERIFY( &ref1 == &is );
VERIFY( x.as_rvalue == false );
auto&& ref2 = (std::move(is) >> std::move(x));
VERIFY( &ref2 == &is );
VERIFY( x.as_rvalue == true );
char arr[2];
#if __cplusplus <= 201703L
std::istringstream("x") >> &arr[0];
#endif
std::istringstream("x") >> arr;
VERIFY( std::string(arr) == "x" );
}
// LWG 1203 More useful rvalue stream insertion
void
test03()
{
int i = 1203;
std::string result = (std::ostringstream() << "i = " << i).str();
VERIFY( result == "i = 1203" );
std::ostringstream os;
std::ostringstream&& ros = std::move(os) << result;
VERIFY( &ros == &os );
VERIFY( ros.str() == result );
std::stringstream ss;
std::stringstream&& rss = std::move(ss) << result;
VERIFY( &rss == &ss );
VERIFY( rss.str() == result );
std::istringstream is("first second third");
std::istringstream&& ris = std::move(is) >> result;
VERIFY( &ris == &is );
VERIFY( result == "first" );
std::stringstream ss2("fourth fifth sixth");
std::stringstream&& rss2 = std::move(ss2) >> result;
VERIFY( &rss2 == &ss2 );
VERIFY( result == "fourth" );
}
struct A { friend void operator<<(std::ios_base&, A) { } };
struct O : private std::ios_base { friend void operator<<(O&, int) { } };
template<typename Ostream, typename T, typename = void>
struct is_insertable
: std::false_type
{ };
template<typename> using void_t = void;
template<typename Ostream, typename T>
using insert_result
= decltype(std::declval<Ostream>() << std::declval<const T&>());
template<typename Ostream, typename T>
struct is_insertable<Ostream, T, void_t<insert_result<Ostream, T>>>
: std::true_type
{ };
// LWG 1203 negative tests
void
test04()
{
static_assert( is_insertable<std::ios_base&, A>::value,
"valid using the friend operator<<" );
static_assert( !is_insertable<std::ios_base&&, A>::value,
"ill-formed because ios_base is not derived from ios_base" );
static_assert( is_insertable<O&, int>::value,
"valid using the friend operator<<" );
static_assert( !is_insertable<O&&, int>::value,
"ill-formed because O is not publicly derived from ios_base" );
}
int
main()
{
test01();
test02();
test03();
test04();
}