//===----------------------------------------------------------------------===// // // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. // See https://llvm.org/LICENSE.txt for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception // //===----------------------------------------------------------------------===// // UNSUPPORTED: c++03, c++11, c++14 // // constexpr T&& optional::operator*() &&; #include #include #include #include "test_macros.h" using std::optional; struct X { constexpr int test() const& {return 3;} int test() & {return 4;} constexpr int test() const&& {return 5;} int test() && {return 6;} }; struct Y { constexpr int test() && {return 7;} }; constexpr int test() { optional opt{Y{}}; return (*std::move(opt)).test(); } int main(int, char**) { { optional opt; ((void)opt); ASSERT_SAME_TYPE(decltype(*std::move(opt)), X&&); // ASSERT_NOT_NOEXCEPT(*std::move(opt)); // FIXME: This assertion fails with GCC because it can see that // (A) operator*() is constexpr, and // (B) there is no path through the function that throws. // It's arguable if this is the correct behavior for the noexcept // operator. // Regardless this function should still be noexcept(false) because // it has a narrow contract. } { optional opt(X{}); assert((*std::move(opt)).test() == 6); } static_assert(test() == 7, ""); return 0; }