//===----------------------------------------------------------------------===// // // 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++98, c++03, c++11, c++14 // // constexpr const T* optional::operator->() const; #ifdef _LIBCPP_DEBUG #define _LIBCPP_ASSERT(x, m) ((x) ? (void)0 : std::exit(0)) #endif #include #include #include #include "test_macros.h" using std::optional; struct X { constexpr int test() const {return 3;} }; struct Y { int test() const noexcept {return 2;} }; struct Z { const Z* operator&() const; constexpr int test() const {return 1;} }; int main(int, char**) { { const std::optional opt; ((void)opt); ASSERT_SAME_TYPE(decltype(opt.operator->()), X const*); // ASSERT_NOT_NOEXCEPT(opt.operator->()); // 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. } { constexpr optional opt(X{}); static_assert(opt->test() == 3, ""); } { constexpr optional opt(Y{}); assert(opt->test() == 2); } { constexpr optional opt(Z{}); static_assert(opt->test() == 1, ""); } #ifdef _LIBCPP_DEBUG { const optional opt; assert(opt->test() == 3); assert(false); } #endif // _LIBCPP_DEBUG return 0; }