blob: 85733b31fb6d24919923c8297db226ea1702671c [file] [log] [blame]
// Copyright (C) 2008-2023 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();
}