From 199b20e3cbf77757bb2e03649c98fc73b19c611e Mon Sep 17 00:00:00 2001 From: Jonathan Wakely Date: Fri, 5 Apr 2019 19:06:02 +0100 Subject: [PATCH] Implement std::visit for C++2a (P0655R1) * doc/xml/manual/status_cxx2020.xml: Update status. * include/std/variant (visit): Define for C++2a (P0655R1). * testsuite/20_util/variant/visit_r.cc: New test. From-SVN: r270176 --- libstdc++-v3/ChangeLog | 4 ++ .../doc/xml/manual/status_cxx2020.xml | 3 +- libstdc++-v3/include/std/variant | 17 +++++++ .../testsuite/20_util/variant/visit_r.cc | 49 +++++++++++++++++++ 4 files changed, 71 insertions(+), 2 deletions(-) create mode 100644 libstdc++-v3/testsuite/20_util/variant/visit_r.cc diff --git a/libstdc++-v3/ChangeLog b/libstdc++-v3/ChangeLog index 65b16d8e68b..ef4a79214aa 100644 --- a/libstdc++-v3/ChangeLog +++ b/libstdc++-v3/ChangeLog @@ -1,5 +1,9 @@ 2019-04-05 Jonathan Wakely + * doc/xml/manual/status_cxx2020.xml: Update status. + * include/std/variant (visit): Define for C++2a (P0655R1). + * testsuite/20_util/variant/visit_r.cc: New test. + * include/bits/fs_dir.h (directory_iterator::operator*) (directory_iterator::operator->): Add noexcept. (operator==, operator!=): Replace namespace-scope equality operators diff --git a/libstdc++-v3/doc/xml/manual/status_cxx2020.xml b/libstdc++-v3/doc/xml/manual/status_cxx2020.xml index d40185c5db6..cedb3d03066 100644 --- a/libstdc++-v3/doc/xml/manual/status_cxx2020.xml +++ b/libstdc++-v3/doc/xml/manual/status_cxx2020.xml @@ -712,14 +712,13 @@ Feature-testing recommendations for C++. - visit<R>: Explicit Return Type for visit P0655R1 - + 9.1 diff --git a/libstdc++-v3/include/std/variant b/libstdc++-v3/include/std/variant index e52aa403009..fdf04cf624a 100644 --- a/libstdc++-v3/include/std/variant +++ b/libstdc++-v3/include/std/variant @@ -1610,6 +1610,23 @@ namespace __variant std::forward<_Variants>(__variants)...); } +#if __cplusplus > 201703L + template + constexpr _Res + visit(_Visitor&& __visitor, _Variants&&... __variants) + { + if ((__variants.valueless_by_exception() || ...)) + __throw_bad_variant_access("Unexpected index"); + + if constexpr (std::is_void_v<_Res>) + (void) __do_visit(std::forward<_Visitor>(__visitor), + std::forward<_Variants>(__variants)...); + else + return __do_visit(std::forward<_Visitor>(__visitor), + std::forward<_Variants>(__variants)...); + } +#endif + template struct __variant_hash_call_base_impl { diff --git a/libstdc++-v3/testsuite/20_util/variant/visit_r.cc b/libstdc++-v3/testsuite/20_util/variant/visit_r.cc new file mode 100644 index 00000000000..5eed0cf1a49 --- /dev/null +++ b/libstdc++-v3/testsuite/20_util/variant/visit_r.cc @@ -0,0 +1,49 @@ +// Copyright (C) 2019 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-options "-std=gnu++2a" } +// { dg-do run { target c++2a } } + +#include +#include + +void +test01() +{ + struct Visitor + { + int operator()(int, void*) const { return 0; } + int operator()(char& c, void* p) const { return &c == p; } + int operator()(int i, const char* s) const { return s[i] == '\0'; } + int operator()(char c, const char* s) const { return c == *s; } + }; + + std::variant v1{'c'}; + std::variant v2{"chars"}; + + auto res = std::visit(Visitor{}, v1, v2); + static_assert(std::is_same_v); + VERIFY( res == true ); + + static_assert(std::is_void_v(Visitor{}, v1, v2))>); +} + +int +main() +{ + test01(); +} -- 2.30.2