74 lines
2.2 KiB
C++
74 lines
2.2 KiB
C++
//===----------------------------------------------------------------------===//
|
|
//
|
|
// 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, c++17, c++20
|
|
// <optional>
|
|
|
|
// template<class F> constexpr optional or_else(F&&) &&;
|
|
// template<class F> constexpr optional or_else(F&&) const&;
|
|
|
|
#include "MoveOnly.h"
|
|
|
|
#include <cassert>
|
|
#include <optional>
|
|
|
|
struct NonMovable {
|
|
NonMovable() = default;
|
|
NonMovable(NonMovable&&) = delete;
|
|
};
|
|
|
|
template <class Opt, class F>
|
|
concept has_or_else = requires(Opt&& opt, F&& f) {
|
|
{std::forward<Opt>(opt).or_else(std::forward<F>(f))};
|
|
};
|
|
|
|
template <class T>
|
|
std::optional<T> return_optional() {}
|
|
|
|
static_assert(has_or_else<std::optional<int>&, decltype(return_optional<int>)>);
|
|
static_assert(has_or_else<std::optional<int>&&, decltype(return_optional<int>)>);
|
|
static_assert(!has_or_else<std::optional<MoveOnly>&, decltype(return_optional<MoveOnly>)>);
|
|
static_assert(has_or_else<std::optional<MoveOnly>&&, decltype(return_optional<MoveOnly>)>);
|
|
static_assert(!has_or_else<std::optional<NonMovable>&, decltype(return_optional<NonMovable>)>);
|
|
static_assert(!has_or_else<std::optional<NonMovable>&&, decltype(return_optional<NonMovable>)>);
|
|
|
|
std::optional<int> take_int(int) { return 0; }
|
|
void take_int_return_void(int) {}
|
|
|
|
static_assert(!has_or_else<std::optional<int>, decltype(take_int)>);
|
|
static_assert(!has_or_else<std::optional<int>, decltype(take_int_return_void)>);
|
|
static_assert(!has_or_else<std::optional<int>, int>);
|
|
|
|
constexpr bool test() {
|
|
{
|
|
std::optional<int> opt;
|
|
assert(opt.or_else([] { return std::optional<int>{0}; }) == 0);
|
|
opt = 1;
|
|
opt.or_else([] {
|
|
assert(false);
|
|
return std::optional<int>{};
|
|
});
|
|
}
|
|
{
|
|
std::optional<MoveOnly> opt;
|
|
opt = std::move(opt).or_else([] { return std::optional<MoveOnly>{MoveOnly{}}; });
|
|
std::move(opt).or_else([] {
|
|
assert(false);
|
|
return std::optional<MoveOnly>{};
|
|
});
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
int main(int, char**) {
|
|
test();
|
|
static_assert(test());
|
|
return 0;
|
|
}
|