// 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 // . // { dg-do run { target c++11 } } #include #include #include 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 struct is_insertable : std::false_type { }; template using void_t = void; template using insert_result = decltype(std::declval() << std::declval()); template struct is_insertable>> : std::true_type { }; // LWG 1203 negative tests void test04() { static_assert( is_insertable::value, "valid using the friend operator<<" ); static_assert( !is_insertable::value, "ill-formed because ios_base is not derived from ios_base" ); static_assert( is_insertable::value, "valid using the friend operator<<" ); static_assert( !is_insertable::value, "ill-formed because O is not publicly derived from ios_base" ); } int main() { test01(); test02(); test03(); test04(); }