mirror of
https://github.com/llvm-mirror/libcxx.git
synced 2025-10-24 12:02:36 +08:00

Libc++ is used as a system library on macOS and iOS (amongst others). In order for users to be able to compile a binary that is intended to be deployed to an older version of the platform, clang provides the availability attribute <https://clang.llvm.org/docs/AttributeReference.html#availability>_ that can be placed on declarations to describe the lifecycle of a symbol in the library. See docs/DesignDocs/AvailabilityMarkup.rst for more information. Differential Revision: https://reviews.llvm.org/D31739 git-svn-id: https://llvm.org/svn/llvm-project/libcxx/trunk@302172 91177308-0d34-0410-b5e6-96231b3b80d8
240 lines
5.5 KiB
C++
240 lines
5.5 KiB
C++
// -*- C++ -*-
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// The LLVM Compiler Infrastructure
|
|
//
|
|
// This file is dual licensed under the MIT and the University of Illinois Open
|
|
// Source Licenses. See LICENSE.TXT for details.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// UNSUPPORTED: c++98, c++03, c++11, c++14
|
|
|
|
// XFAIL: with_system_cxx_lib=macosx10.12
|
|
// XFAIL: with_system_cxx_lib=macosx10.11
|
|
// XFAIL: with_system_cxx_lib=macosx10.10
|
|
// XFAIL: with_system_cxx_lib=macosx10.9
|
|
// XFAIL: with_system_cxx_lib=macosx10.7
|
|
// XFAIL: with_system_cxx_lib=macosx10.8
|
|
|
|
// <variant>
|
|
|
|
// template <class ...Types> class variant;
|
|
|
|
// template <class T>
|
|
// variant& operator=(T&&) noexcept(see below);
|
|
|
|
#include <cassert>
|
|
#include <string>
|
|
#include <type_traits>
|
|
#include <variant>
|
|
|
|
#include "test_macros.h"
|
|
#include "variant_test_helpers.hpp"
|
|
|
|
namespace MetaHelpers {
|
|
|
|
struct Dummy {
|
|
Dummy() = default;
|
|
};
|
|
|
|
struct ThrowsCtorT {
|
|
ThrowsCtorT(int) noexcept(false) {}
|
|
ThrowsCtorT &operator=(int) noexcept { return *this; }
|
|
};
|
|
|
|
struct ThrowsAssignT {
|
|
ThrowsAssignT(int) noexcept {}
|
|
ThrowsAssignT &operator=(int) noexcept(false) { return *this; }
|
|
};
|
|
|
|
struct NoThrowT {
|
|
NoThrowT(int) noexcept {}
|
|
NoThrowT &operator=(int) noexcept { return *this; }
|
|
};
|
|
|
|
} // namespace MetaHelpers
|
|
|
|
namespace RuntimeHelpers {
|
|
#ifndef TEST_HAS_NO_EXCEPTIONS
|
|
|
|
struct ThrowsCtorT {
|
|
int value;
|
|
ThrowsCtorT() : value(0) {}
|
|
ThrowsCtorT(int) noexcept(false) { throw 42; }
|
|
ThrowsCtorT &operator=(int v) noexcept {
|
|
value = v;
|
|
return *this;
|
|
}
|
|
};
|
|
|
|
struct ThrowsAssignT {
|
|
int value;
|
|
ThrowsAssignT() : value(0) {}
|
|
ThrowsAssignT(int v) noexcept : value(v) {}
|
|
ThrowsAssignT &operator=(int) noexcept(false) { throw 42; }
|
|
};
|
|
|
|
struct NoThrowT {
|
|
int value;
|
|
NoThrowT() : value(0) {}
|
|
NoThrowT(int v) noexcept : value(v) {}
|
|
NoThrowT &operator=(int v) noexcept {
|
|
value = v;
|
|
return *this;
|
|
}
|
|
};
|
|
|
|
#endif // !defined(TEST_HAS_NO_EXCEPTIONS)
|
|
} // namespace RuntimeHelpers
|
|
|
|
void test_T_assignment_noexcept() {
|
|
using namespace MetaHelpers;
|
|
{
|
|
using V = std::variant<Dummy, NoThrowT>;
|
|
static_assert(std::is_nothrow_assignable<V, int>::value, "");
|
|
}
|
|
{
|
|
using V = std::variant<Dummy, ThrowsCtorT>;
|
|
static_assert(!std::is_nothrow_assignable<V, int>::value, "");
|
|
}
|
|
{
|
|
using V = std::variant<Dummy, ThrowsAssignT>;
|
|
static_assert(!std::is_nothrow_assignable<V, int>::value, "");
|
|
}
|
|
}
|
|
|
|
void test_T_assignment_sfinae() {
|
|
{
|
|
using V = std::variant<long, unsigned>;
|
|
static_assert(!std::is_assignable<V, int>::value, "ambiguous");
|
|
}
|
|
{
|
|
using V = std::variant<std::string, std::string>;
|
|
static_assert(!std::is_assignable<V, const char *>::value, "ambiguous");
|
|
}
|
|
{
|
|
using V = std::variant<std::string, void *>;
|
|
static_assert(!std::is_assignable<V, int>::value, "no matching operator=");
|
|
}
|
|
#if !defined(TEST_VARIANT_HAS_NO_REFERENCES)
|
|
{
|
|
using V = std::variant<int, int &&>;
|
|
static_assert(!std::is_assignable<V, int>::value, "ambiguous");
|
|
}
|
|
{
|
|
using V = std::variant<int, const int &>;
|
|
static_assert(!std::is_assignable<V, int>::value, "ambiguous");
|
|
}
|
|
#endif
|
|
}
|
|
|
|
void test_T_assignment_basic() {
|
|
{
|
|
std::variant<int> v(43);
|
|
v = 42;
|
|
assert(v.index() == 0);
|
|
assert(std::get<0>(v) == 42);
|
|
}
|
|
{
|
|
std::variant<int, long> v(43l);
|
|
v = 42;
|
|
assert(v.index() == 0);
|
|
assert(std::get<0>(v) == 42);
|
|
v = 43l;
|
|
assert(v.index() == 1);
|
|
assert(std::get<1>(v) == 43);
|
|
}
|
|
#if !defined(TEST_VARIANT_HAS_NO_REFERENCES)
|
|
{
|
|
using V = std::variant<int &, int &&, long>;
|
|
int x = 42;
|
|
V v(43l);
|
|
v = x;
|
|
assert(v.index() == 0);
|
|
assert(&std::get<0>(v) == &x);
|
|
v = std::move(x);
|
|
assert(v.index() == 1);
|
|
assert(&std::get<1>(v) == &x);
|
|
// 'long' is selected by FUN(const int &) since 'const int &' cannot bind
|
|
// to 'int&'.
|
|
const int &cx = x;
|
|
v = cx;
|
|
assert(v.index() == 2);
|
|
assert(std::get<2>(v) == 42);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
void test_T_assignment_performs_construction() {
|
|
using namespace RuntimeHelpers;
|
|
#ifndef TEST_HAS_NO_EXCEPTIONS
|
|
{
|
|
using V = std::variant<std::string, ThrowsCtorT>;
|
|
V v(std::in_place_type<std::string>, "hello");
|
|
try {
|
|
v = 42;
|
|
} catch (...) { /* ... */
|
|
}
|
|
assert(v.valueless_by_exception());
|
|
}
|
|
{
|
|
using V = std::variant<ThrowsAssignT, std::string>;
|
|
V v(std::in_place_type<std::string>, "hello");
|
|
v = 42;
|
|
assert(v.index() == 0);
|
|
assert(std::get<0>(v).value == 42);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
void test_T_assignment_performs_assignment() {
|
|
using namespace RuntimeHelpers;
|
|
#ifndef TEST_HAS_NO_EXCEPTIONS
|
|
{
|
|
using V = std::variant<ThrowsCtorT>;
|
|
V v;
|
|
v = 42;
|
|
assert(v.index() == 0);
|
|
assert(std::get<0>(v).value == 42);
|
|
}
|
|
{
|
|
using V = std::variant<ThrowsCtorT, std::string>;
|
|
V v;
|
|
v = 42;
|
|
assert(v.index() == 0);
|
|
assert(std::get<0>(v).value == 42);
|
|
}
|
|
{
|
|
using V = std::variant<ThrowsAssignT>;
|
|
V v(100);
|
|
try {
|
|
v = 42;
|
|
assert(false);
|
|
} catch (...) { /* ... */
|
|
}
|
|
assert(v.index() == 0);
|
|
assert(std::get<0>(v).value == 100);
|
|
}
|
|
{
|
|
using V = std::variant<std::string, ThrowsAssignT>;
|
|
V v(100);
|
|
try {
|
|
v = 42;
|
|
assert(false);
|
|
} catch (...) { /* ... */
|
|
}
|
|
assert(v.index() == 1);
|
|
assert(std::get<1>(v).value == 100);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
int main() {
|
|
test_T_assignment_basic();
|
|
test_T_assignment_performs_construction();
|
|
test_T_assignment_performs_assignment();
|
|
test_T_assignment_noexcept();
|
|
test_T_assignment_sfinae();
|
|
}
|