mirror of
https://github.com/TartanLlama/optional.git
synced 2025-10-04 00:41:02 +02:00
93 lines
2.6 KiB
C++
93 lines
2.6 KiB
C++
#include "catch.hpp"
|
|
#include "optional.hpp"
|
|
|
|
TEST_CASE("Noexcept", "[noexcept]") {
|
|
tl::optional<int> o1{4};
|
|
tl::optional<int> o2{42};
|
|
|
|
SECTION("comparison with nullopt") {
|
|
REQUIRE(noexcept(o1 == tl::nullopt));
|
|
REQUIRE(noexcept(tl::nullopt == o1));
|
|
REQUIRE(noexcept(o1 != tl::nullopt));
|
|
REQUIRE(noexcept(tl::nullopt != o1));
|
|
REQUIRE(noexcept(o1 < tl::nullopt));
|
|
REQUIRE(noexcept(tl::nullopt < o1));
|
|
REQUIRE(noexcept(o1 <= tl::nullopt));
|
|
REQUIRE(noexcept(tl::nullopt <= o1));
|
|
REQUIRE(noexcept(o1 > tl::nullopt));
|
|
REQUIRE(noexcept(tl::nullopt > o1));
|
|
REQUIRE(noexcept(o1 >= tl::nullopt));
|
|
REQUIRE(noexcept(tl::nullopt >= o1));
|
|
}
|
|
|
|
SECTION("swap") {
|
|
REQUIRE(noexcept(swap(o1, o2)) == noexcept(o1.swap(o2)));
|
|
|
|
struct nothrow_swappable {
|
|
nothrow_swappable &swap(const nothrow_swappable &) noexcept {
|
|
return *this;
|
|
}
|
|
};
|
|
|
|
struct throw_swappable {
|
|
throw_swappable(throw_swappable &&) {}
|
|
throw_swappable &swap(const throw_swappable &) { return *this; }
|
|
};
|
|
|
|
tl::optional<nothrow_swappable> ont;
|
|
tl::optional<throw_swappable> ot;
|
|
|
|
REQUIRE(noexcept(ont.swap(ont)));
|
|
REQUIRE(!noexcept(ot.swap(ot)));
|
|
}
|
|
|
|
SECTION("constructors") {
|
|
REQUIRE(noexcept(tl::optional<int>{}));
|
|
REQUIRE(noexcept(tl::optional<int>{tl::nullopt}));
|
|
|
|
struct nothrow_move {
|
|
nothrow_move(nothrow_move &&) noexcept = default;
|
|
};
|
|
|
|
struct throw_move {
|
|
throw_move(throw_move &&){};
|
|
};
|
|
|
|
using nothrow_opt = tl::optional<nothrow_move>;
|
|
using throw_opt = tl::optional<throw_move>;
|
|
|
|
REQUIRE(noexcept(nothrow_opt{std::declval<nothrow_opt>()}));
|
|
REQUIRE(!noexcept(throw_opt{std::declval<throw_opt>()}));
|
|
}
|
|
|
|
SECTION("assignment") {
|
|
REQUIRE(noexcept(o1 = tl::nullopt));
|
|
|
|
struct nothrow_move_assign {
|
|
nothrow_move_assign() = default;
|
|
nothrow_move_assign(nothrow_move_assign &&) noexcept = default;
|
|
nothrow_move_assign &operator=(const nothrow_move_assign &) = default;
|
|
};
|
|
|
|
struct throw_move_assign {
|
|
throw_move_assign() = default;
|
|
throw_move_assign(throw_move_assign &&){};
|
|
throw_move_assign &operator=(const throw_move_assign &) {}
|
|
};
|
|
|
|
using nothrow_opt = tl::optional<nothrow_move_assign>;
|
|
using throw_opt = tl::optional<throw_move_assign>;
|
|
|
|
REQUIRE(
|
|
noexcept(std::declval<nothrow_opt>() = std::declval<nothrow_opt>()));
|
|
REQUIRE(!noexcept(std::declval<throw_opt>() = std::declval<throw_opt>()));
|
|
}
|
|
|
|
SECTION("observers") {
|
|
REQUIRE(noexcept(static_cast<bool>(o1)));
|
|
REQUIRE(noexcept(o1.has_value()));
|
|
}
|
|
|
|
SECTION("modifiers") { REQUIRE(noexcept(o1.reset())); }
|
|
}
|