Implement LWG 2534, Constrain rvalue stream operators.
* include/std/istream (__is_convertible_to_basic_istream): New. (__is_extractable): Likewise. (operator>>(basic_istream<_CharT, _Traits>&&, _Tp&&)): Turn the stream parameter into a template parameter and constrain. * include/std/ostream (__is_convertible_to_basic_ostream): New. (__is_insertable): Likewise. (operator<<(basic_ostream<_CharT, _Traits>&&, const _Tp&)): Turn the stream parameter into a template parameter and constrain. * testsuite/27_io/basic_istream/extractors_other/char/4.cc: New. * testsuite/27_io/basic_istream/extractors_other/wchar_t/4.cc: Likewise. * testsuite/27_io/basic_ostream/inserters_other/char/6.cc: Likewise. * testsuite/27_io/basic_ostream/inserters_other/wchar_t/6.cc: Likewise. From-SVN: r243006
This commit is contained in:
parent
4010958106
commit
a7da488130
@ -1,3 +1,22 @@
|
||||
2016-11-30 Ville Voutilainen <ville.voutilainen@gmail.com>
|
||||
|
||||
Implement LWG 2534, Constrain rvalue stream operators.
|
||||
* include/std/istream (__is_convertible_to_basic_istream): New.
|
||||
(__is_extractable): Likewise.
|
||||
(operator>>(basic_istream<_CharT, _Traits>&&, _Tp&&)):
|
||||
Turn the stream parameter into a template parameter
|
||||
and constrain.
|
||||
* include/std/ostream (__is_convertible_to_basic_ostream): New.
|
||||
(__is_insertable): Likewise.
|
||||
(operator<<(basic_ostream<_CharT, _Traits>&&, const _Tp&)):
|
||||
Turn the stream parameter into a template parameter
|
||||
and constrain.
|
||||
* testsuite/27_io/basic_istream/extractors_other/char/4.cc: New.
|
||||
* testsuite/27_io/basic_istream/extractors_other/wchar_t/4.cc:
|
||||
Likewise.
|
||||
* testsuite/27_io/basic_ostream/inserters_other/char/6.cc: Likewise.
|
||||
* testsuite/27_io/basic_ostream/inserters_other/wchar_t/6.cc: Likewise.
|
||||
|
||||
2016-11-30 Christophe Lyon <christophe.lyon@linaro.org>
|
||||
|
||||
* testsuite/experimental/type_erased_allocator/2.cc: Add
|
||||
|
@ -908,6 +908,28 @@ _GLIBCXX_BEGIN_NAMESPACE_VERSION
|
||||
ws(basic_istream<_CharT, _Traits>& __is);
|
||||
|
||||
#if __cplusplus >= 201103L
|
||||
|
||||
template<typename _Tp>
|
||||
struct __is_convertible_to_basic_istream
|
||||
{
|
||||
template<typename _Ch, typename _Up>
|
||||
static true_type __check(basic_istream<_Ch, _Up>*);
|
||||
|
||||
static false_type __check(void*);
|
||||
public:
|
||||
using type = decltype(__check(declval<_Tp*>()));
|
||||
constexpr static bool value = type::value;
|
||||
};
|
||||
|
||||
template<typename _Istream, typename _Tp, typename = void>
|
||||
struct __is_extractable : false_type {};
|
||||
|
||||
template<typename _Istream, typename _Tp>
|
||||
struct __is_extractable<_Istream, _Tp,
|
||||
__void_t<decltype(declval<_Istream&>()
|
||||
>> declval<_Tp>())>>
|
||||
: true_type {};
|
||||
|
||||
// [27.7.1.6] Rvalue stream extraction
|
||||
// _GLIBCXX_RESOLVE_LIB_DEFECTS
|
||||
// 2328. Rvalue stream extraction should use perfect forwarding
|
||||
@ -921,9 +943,14 @@ _GLIBCXX_BEGIN_NAMESPACE_VERSION
|
||||
* rvalue streams since they won't bind to the extractor functions
|
||||
* that take an lvalue reference.
|
||||
*/
|
||||
template<typename _CharT, typename _Traits, typename _Tp>
|
||||
inline basic_istream<_CharT, _Traits>&
|
||||
operator>>(basic_istream<_CharT, _Traits>&& __is, _Tp&& __x)
|
||||
template<typename _Istream, typename _Tp>
|
||||
inline
|
||||
typename enable_if<__and_<__not_<is_lvalue_reference<_Istream>>,
|
||||
__is_convertible_to_basic_istream<
|
||||
typename remove_reference<_Istream>::type>,
|
||||
__is_extractable<_Istream&, _Tp&&>>::value,
|
||||
_Istream&>::type
|
||||
operator>>(_Istream&& __is, _Tp&& __x)
|
||||
{
|
||||
__is >> std::forward<_Tp>(__x);
|
||||
return __is;
|
||||
|
@ -613,6 +613,27 @@ _GLIBCXX_BEGIN_NAMESPACE_VERSION
|
||||
{ return __os.flush(); }
|
||||
|
||||
#if __cplusplus >= 201103L
|
||||
template<typename _Tp>
|
||||
struct __is_convertible_to_basic_ostream
|
||||
{
|
||||
template<typename _Ch, typename _Up>
|
||||
static true_type __check(basic_ostream<_Ch, _Up>*);
|
||||
|
||||
static false_type __check(void*);
|
||||
public:
|
||||
using type = decltype(__check(declval<_Tp*>()));
|
||||
constexpr static bool value = type::value;
|
||||
};
|
||||
|
||||
template<typename _Ostream, typename _Tp, typename = void>
|
||||
struct __is_insertable : false_type {};
|
||||
|
||||
template<typename _Ostream, typename _Tp>
|
||||
struct __is_insertable<_Ostream, _Tp,
|
||||
__void_t<decltype(declval<_Ostream&>()
|
||||
<< declval<const _Tp&>())>>
|
||||
: true_type {};
|
||||
|
||||
/**
|
||||
* @brief Generic inserter for rvalue stream
|
||||
* @param __os An input stream.
|
||||
@ -623,9 +644,15 @@ _GLIBCXX_BEGIN_NAMESPACE_VERSION
|
||||
* rvalue streams since they won't bind to the inserter functions
|
||||
* that take an lvalue reference.
|
||||
*/
|
||||
template<typename _CharT, typename _Traits, typename _Tp>
|
||||
inline basic_ostream<_CharT, _Traits>&
|
||||
operator<<(basic_ostream<_CharT, _Traits>&& __os, const _Tp& __x)
|
||||
template<typename _Ostream, typename _Tp>
|
||||
inline
|
||||
typename enable_if<__and_<__not_<is_lvalue_reference<_Ostream>>,
|
||||
__is_convertible_to_basic_ostream<
|
||||
typename remove_reference<_Ostream>::type>,
|
||||
__is_insertable<_Ostream&, const _Tp&>>::value,
|
||||
_Ostream&>::type
|
||||
//basic_ostream<_CharT, _Traits>&
|
||||
operator<<(_Ostream&& __os, const _Tp& __x)
|
||||
{
|
||||
__os << __x;
|
||||
return __os;
|
||||
|
@ -0,0 +1,96 @@
|
||||
// { dg-do run { target c++11 } }
|
||||
|
||||
// Copyright (C) 2016 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/>.
|
||||
|
||||
// 27.6.2.5.3 basic_ostream manipulator inserters
|
||||
|
||||
#include <sstream>
|
||||
|
||||
struct X {};
|
||||
std::istream& operator>>(std::istream&, X&) = delete;
|
||||
|
||||
struct Y {};
|
||||
std::istream& operator>>(std::istream& is, Y&) {return is;}
|
||||
std::istream& operator>>(std::istream& is, Y&&) {return is;}
|
||||
|
||||
struct Z{};
|
||||
|
||||
template <class T>
|
||||
auto f(T&&) -> decltype(void(std::declval<std::istream&>()
|
||||
>> std::declval<T&&>()),
|
||||
std::true_type());
|
||||
|
||||
std::false_type f(...);
|
||||
|
||||
template <class T>
|
||||
auto g(T&&) -> decltype(void(std::declval<std::istream&&>()
|
||||
>> std::declval<T&&>()),
|
||||
std::true_type());
|
||||
|
||||
std::false_type g(...);
|
||||
|
||||
void test01()
|
||||
{
|
||||
Y y;
|
||||
std::istringstream is;
|
||||
is >> y;
|
||||
is >> Y();
|
||||
std::istringstream() >> y;
|
||||
std::istringstream() >> Y();
|
||||
static_assert(!std::__is_extractable<std::istream&, X&>::value, "");
|
||||
static_assert(!std::__is_extractable<std::istream&&, X&>::value, "");
|
||||
static_assert(!std::__is_extractable<std::istream&, X&&>::value, "");
|
||||
static_assert(!std::__is_extractable<std::istream&&, X&&>::value, "");
|
||||
static_assert(std::__is_extractable<std::istream&, Y&>::value, "");
|
||||
static_assert(std::__is_extractable<std::istream&&, Y&>::value, "");
|
||||
static_assert(std::__is_extractable<std::istream&, Y&&>::value, "");
|
||||
static_assert(std::__is_extractable<std::istream&&, Y&&>::value, "");
|
||||
static_assert(!std::__is_extractable<std::istream&, Z&>::value, "");
|
||||
static_assert(!std::__is_extractable<std::istream&&, Z&>::value, "");
|
||||
static_assert(!std::__is_extractable<std::istream&, Z&&>::value, "");
|
||||
static_assert(!std::__is_extractable<std::istream&&, Z&&>::value, "");
|
||||
static_assert(std::is_same<decltype(f(std::declval<X&>())),
|
||||
std::false_type>::value, "");
|
||||
static_assert(std::is_same<decltype(f(std::declval<X&&>())),
|
||||
std::false_type>::value, "");
|
||||
static_assert(std::is_same<decltype(f(std::declval<Y&>())),
|
||||
std::true_type>::value, "");
|
||||
static_assert(std::is_same<decltype(f(std::declval<Y&&>())),
|
||||
std::true_type>::value, "");
|
||||
static_assert(std::is_same<decltype(f(std::declval<Z&>())),
|
||||
std::false_type>::value, "");
|
||||
static_assert(std::is_same<decltype(f(std::declval<Z&&>())),
|
||||
std::false_type>::value, "");
|
||||
static_assert(std::is_same<decltype(g(std::declval<X&>())),
|
||||
std::false_type>::value, "");
|
||||
static_assert(std::is_same<decltype(g(std::declval<X&&>())),
|
||||
std::false_type>::value, "");
|
||||
static_assert(std::is_same<decltype(g(std::declval<Y&>())),
|
||||
std::true_type>::value, "");
|
||||
static_assert(std::is_same<decltype(g(std::declval<Y&&>())),
|
||||
std::true_type>::value, "");
|
||||
static_assert(std::is_same<decltype(g(std::declval<Z&>())),
|
||||
std::false_type>::value, "");
|
||||
static_assert(std::is_same<decltype(g(std::declval<Z&&>())),
|
||||
std::false_type>::value, "");
|
||||
}
|
||||
|
||||
int main()
|
||||
{
|
||||
test01();
|
||||
}
|
@ -0,0 +1,96 @@
|
||||
// { dg-do run { target c++11 } }
|
||||
|
||||
// Copyright (C) 2016 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/>.
|
||||
|
||||
// 27.6.2.5.3 basic_ostream manipulator inserters
|
||||
|
||||
#include <sstream>
|
||||
|
||||
struct X {};
|
||||
std::wistream& operator>>(std::wistream&, X&) = delete;
|
||||
|
||||
struct Y {};
|
||||
std::wistream& operator>>(std::wistream& is, Y&) {return is;}
|
||||
std::wistream& operator>>(std::wistream& is, Y&&) {return is;}
|
||||
|
||||
struct Z{};
|
||||
|
||||
template <class T>
|
||||
auto f(T&&) -> decltype(void(std::declval<std::wistream&>()
|
||||
>> std::declval<T&&>()),
|
||||
std::true_type());
|
||||
|
||||
std::false_type f(...);
|
||||
|
||||
template <class T>
|
||||
auto g(T&&) -> decltype(void(std::declval<std::wistream&&>()
|
||||
>> std::declval<T&&>()),
|
||||
std::true_type());
|
||||
|
||||
std::false_type g(...);
|
||||
|
||||
void test01()
|
||||
{
|
||||
Y y;
|
||||
std::wistringstream is;
|
||||
is >> y;
|
||||
is >> Y();
|
||||
std::wistringstream() >> y;
|
||||
std::wistringstream() >> Y();
|
||||
static_assert(!std::__is_extractable<std::wistream&, X&>::value, "");
|
||||
static_assert(!std::__is_extractable<std::wistream&&, X&>::value, "");
|
||||
static_assert(!std::__is_extractable<std::wistream&, X&&>::value, "");
|
||||
static_assert(!std::__is_extractable<std::wistream&&, X&&>::value, "");
|
||||
static_assert(std::__is_extractable<std::wistream&, Y&>::value, "");
|
||||
static_assert(std::__is_extractable<std::wistream&&, Y&>::value, "");
|
||||
static_assert(std::__is_extractable<std::wistream&, Y&&>::value, "");
|
||||
static_assert(std::__is_extractable<std::wistream&&, Y&&>::value, "");
|
||||
static_assert(!std::__is_extractable<std::wistream&, Z&>::value, "");
|
||||
static_assert(!std::__is_extractable<std::wistream&&, Z&>::value, "");
|
||||
static_assert(!std::__is_extractable<std::wistream&, Z&&>::value, "");
|
||||
static_assert(!std::__is_extractable<std::wistream&&, Z&&>::value, "");
|
||||
static_assert(std::is_same<decltype(f(std::declval<X&>())),
|
||||
std::false_type>::value, "");
|
||||
static_assert(std::is_same<decltype(f(std::declval<X&&>())),
|
||||
std::false_type>::value, "");
|
||||
static_assert(std::is_same<decltype(f(std::declval<Y&>())),
|
||||
std::true_type>::value, "");
|
||||
static_assert(std::is_same<decltype(f(std::declval<Y&&>())),
|
||||
std::true_type>::value, "");
|
||||
static_assert(std::is_same<decltype(f(std::declval<Z&>())),
|
||||
std::false_type>::value, "");
|
||||
static_assert(std::is_same<decltype(f(std::declval<Z&&>())),
|
||||
std::false_type>::value, "");
|
||||
static_assert(std::is_same<decltype(g(std::declval<X&>())),
|
||||
std::false_type>::value, "");
|
||||
static_assert(std::is_same<decltype(g(std::declval<X&&>())),
|
||||
std::false_type>::value, "");
|
||||
static_assert(std::is_same<decltype(g(std::declval<Y&>())),
|
||||
std::true_type>::value, "");
|
||||
static_assert(std::is_same<decltype(g(std::declval<Y&&>())),
|
||||
std::true_type>::value, "");
|
||||
static_assert(std::is_same<decltype(g(std::declval<Z&>())),
|
||||
std::false_type>::value, "");
|
||||
static_assert(std::is_same<decltype(g(std::declval<Z&&>())),
|
||||
std::false_type>::value, "");
|
||||
}
|
||||
|
||||
int main()
|
||||
{
|
||||
test01();
|
||||
}
|
@ -0,0 +1,96 @@
|
||||
// { dg-do run { target c++11 } }
|
||||
|
||||
// Copyright (C) 2016 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/>.
|
||||
|
||||
// 27.6.2.5.3 basic_ostream manipulator inserters
|
||||
|
||||
#include <sstream>
|
||||
|
||||
struct X {};
|
||||
std::ostream& operator<<(std::ostream&, const X&) = delete;
|
||||
|
||||
struct Y {};
|
||||
std::ostream& operator<<(std::ostream& os, const Y&) {return os;}
|
||||
std::ostream& operator<<(std::ostream&& os, const Y&) {return os;}
|
||||
|
||||
struct Z{};
|
||||
|
||||
template <class T>
|
||||
auto f(T&&) -> decltype(void(std::declval<std::ostream&>()
|
||||
<< std::declval<T&&>()),
|
||||
std::true_type());
|
||||
|
||||
std::false_type f(...);
|
||||
|
||||
template <class T>
|
||||
auto g(T&&) -> decltype(void(std::declval<std::ostream&&>()
|
||||
<< std::declval<T&&>()),
|
||||
std::true_type());
|
||||
|
||||
std::false_type g(...);
|
||||
|
||||
void test01()
|
||||
{
|
||||
Y y;
|
||||
std::ostringstream os;
|
||||
os << y;
|
||||
os << Y();
|
||||
std::ostringstream() << y;
|
||||
std::ostringstream() << Y();
|
||||
static_assert(!std::__is_insertable<std::ostream&, X&>::value, "");
|
||||
static_assert(!std::__is_insertable<std::ostream&&, X&>::value, "");
|
||||
static_assert(!std::__is_insertable<std::ostream&, X&&>::value, "");
|
||||
static_assert(!std::__is_insertable<std::ostream&&, X&&>::value, "");
|
||||
static_assert(std::__is_insertable<std::ostream&, Y&>::value, "");
|
||||
static_assert(std::__is_insertable<std::ostream&&, Y&&>::value, "");
|
||||
static_assert(std::__is_insertable<std::ostream&, Y&>::value, "");
|
||||
static_assert(std::__is_insertable<std::ostream&&, Y&&>::value, "");
|
||||
static_assert(!std::__is_insertable<std::ostream&, Z&>::value, "");
|
||||
static_assert(!std::__is_insertable<std::ostream&&, Z&>::value, "");
|
||||
static_assert(!std::__is_insertable<std::ostream&, Z&&>::value, "");
|
||||
static_assert(!std::__is_insertable<std::ostream&&, Z&&>::value, "");
|
||||
static_assert(std::is_same<decltype(f(std::declval<X&>())),
|
||||
std::false_type>::value, "");
|
||||
static_assert(std::is_same<decltype(f(std::declval<X&&>())),
|
||||
std::false_type>::value, "");
|
||||
static_assert(std::is_same<decltype(f(std::declval<Y&>())),
|
||||
std::true_type>::value, "");
|
||||
static_assert(std::is_same<decltype(f(std::declval<Y&&>())),
|
||||
std::true_type>::value, "");
|
||||
static_assert(std::is_same<decltype(f(std::declval<Z&>())),
|
||||
std::false_type>::value, "");
|
||||
static_assert(std::is_same<decltype(f(std::declval<Z&&>())),
|
||||
std::false_type>::value, "");
|
||||
static_assert(std::is_same<decltype(g(std::declval<X&>())),
|
||||
std::false_type>::value, "");
|
||||
static_assert(std::is_same<decltype(g(std::declval<X&&>())),
|
||||
std::false_type>::value, "");
|
||||
static_assert(std::is_same<decltype(g(std::declval<Y&>())),
|
||||
std::true_type>::value, "");
|
||||
static_assert(std::is_same<decltype(g(std::declval<Y&&>())),
|
||||
std::true_type>::value, "");
|
||||
static_assert(std::is_same<decltype(g(std::declval<Z&>())),
|
||||
std::false_type>::value, "");
|
||||
static_assert(std::is_same<decltype(g(std::declval<Z&&>())),
|
||||
std::false_type>::value, "");
|
||||
}
|
||||
|
||||
int main()
|
||||
{
|
||||
test01();
|
||||
}
|
@ -0,0 +1,96 @@
|
||||
// { dg-do run { target c++11 } }
|
||||
|
||||
// Copyright (C) 2016 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/>.
|
||||
|
||||
// 27.6.2.5.3 basic_ostream manipulator inserters
|
||||
|
||||
#include <sstream>
|
||||
|
||||
struct X {};
|
||||
std::wostream& operator<<(std::wostream&, const X&) = delete;
|
||||
|
||||
struct Y {};
|
||||
std::wostream& operator<<(std::wostream& os, const Y&) {return os;}
|
||||
std::wostream& operator<<(std::wostream&& os, const Y&) {return os;}
|
||||
|
||||
struct Z{};
|
||||
|
||||
template <class T>
|
||||
auto f(T&&) -> decltype(void(std::declval<std::wostream&>()
|
||||
<< std::declval<T&&>()),
|
||||
std::true_type());
|
||||
|
||||
std::false_type f(...);
|
||||
|
||||
template <class T>
|
||||
auto g(T&&) -> decltype(void(std::declval<std::wostream&&>()
|
||||
<< std::declval<T&&>()),
|
||||
std::true_type());
|
||||
|
||||
std::false_type g(...);
|
||||
|
||||
void test01()
|
||||
{
|
||||
Y y;
|
||||
std::wostringstream os;
|
||||
os << y;
|
||||
os << Y();
|
||||
std::wostringstream() << y;
|
||||
std::wostringstream() << Y();
|
||||
static_assert(!std::__is_insertable<std::wostream&, X&>::value, "");
|
||||
static_assert(!std::__is_insertable<std::wostream&&, X&>::value, "");
|
||||
static_assert(!std::__is_insertable<std::wostream&, X&&>::value, "");
|
||||
static_assert(!std::__is_insertable<std::wostream&&, X&&>::value, "");
|
||||
static_assert(std::__is_insertable<std::wostream&, Y&>::value, "");
|
||||
static_assert(std::__is_insertable<std::wostream&&, Y&&>::value, "");
|
||||
static_assert(std::__is_insertable<std::wostream&, Y&>::value, "");
|
||||
static_assert(std::__is_insertable<std::wostream&&, Y&&>::value, "");
|
||||
static_assert(!std::__is_insertable<std::wostream&, Z&>::value, "");
|
||||
static_assert(!std::__is_insertable<std::wostream&&, Z&>::value, "");
|
||||
static_assert(!std::__is_insertable<std::wostream&, Z&&>::value, "");
|
||||
static_assert(!std::__is_insertable<std::wostream&&, Z&&>::value, "");
|
||||
static_assert(std::is_same<decltype(f(std::declval<X&>())),
|
||||
std::false_type>::value, "");
|
||||
static_assert(std::is_same<decltype(f(std::declval<X&&>())),
|
||||
std::false_type>::value, "");
|
||||
static_assert(std::is_same<decltype(f(std::declval<Y&>())),
|
||||
std::true_type>::value, "");
|
||||
static_assert(std::is_same<decltype(f(std::declval<Y&&>())),
|
||||
std::true_type>::value, "");
|
||||
static_assert(std::is_same<decltype(f(std::declval<Z&>())),
|
||||
std::false_type>::value, "");
|
||||
static_assert(std::is_same<decltype(f(std::declval<Z&&>())),
|
||||
std::false_type>::value, "");
|
||||
static_assert(std::is_same<decltype(g(std::declval<X&>())),
|
||||
std::false_type>::value, "");
|
||||
static_assert(std::is_same<decltype(g(std::declval<X&&>())),
|
||||
std::false_type>::value, "");
|
||||
static_assert(std::is_same<decltype(g(std::declval<Y&>())),
|
||||
std::true_type>::value, "");
|
||||
static_assert(std::is_same<decltype(g(std::declval<Y&&>())),
|
||||
std::true_type>::value, "");
|
||||
static_assert(std::is_same<decltype(g(std::declval<Z&>())),
|
||||
std::false_type>::value, "");
|
||||
static_assert(std::is_same<decltype(g(std::declval<Z&&>())),
|
||||
std::false_type>::value, "");
|
||||
}
|
||||
|
||||
int main()
|
||||
{
|
||||
test01();
|
||||
}
|
Loading…
x
Reference in New Issue
Block a user