// -*- 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++98, c++03, c++11, c++14 // XFAIL: dylib-has-no-bad_variant_access && !libcpp-no-exceptions // // template class variant; // template // variant_alternative_t>& emplace(Args&&... args); #include #include #include #include #include "archetypes.h" #include "test_convertible.h" #include "test_macros.h" #include "variant_test_helpers.h" template constexpr auto test_emplace_exists_imp(int) -> decltype( std::declval().template emplace(std::declval()...), true) { return true; } template constexpr auto test_emplace_exists_imp(long) -> bool { return false; } template constexpr bool emplace_exists() { return test_emplace_exists_imp(0); } void test_emplace_sfinae() { { using V = std::variant; static_assert(emplace_exists(), ""); static_assert(emplace_exists(), ""); static_assert(!emplace_exists(), "cannot construct"); static_assert(emplace_exists(), ""); static_assert(emplace_exists(), ""); static_assert(!emplace_exists(), ""); static_assert(!emplace_exists(), "cannot construct"); static_assert(emplace_exists(), ""); static_assert(emplace_exists(), ""); static_assert(!emplace_exists(), "cannot construct"); } #if !defined(TEST_VARIANT_HAS_NO_REFERENCES) { using V = std::variant; static_assert(emplace_exists(), ""); static_assert(emplace_exists(), ""); static_assert(emplace_exists(), ""); static_assert(!emplace_exists(), "too many args"); static_assert(emplace_exists(), ""); static_assert(!emplace_exists(), "cannot default construct ref"); static_assert(!emplace_exists(), "cannot bind ref"); static_assert(!emplace_exists(), "cannot bind ref"); static_assert(emplace_exists(), ""); static_assert(emplace_exists(), ""); static_assert(emplace_exists(), ""); static_assert(!emplace_exists(), "not constructible from void*"); static_assert(emplace_exists(), ""); static_assert(!emplace_exists(), "cannot bind ref"); static_assert(!emplace_exists(), "cannot bind ref"); static_assert(!emplace_exists(), "cannot bind ref"); static_assert(!emplace_exists(), "no ctors"); } #endif } void test_basic() { { using V = std::variant; V v(42); auto& ref1 = v.emplace<0>(); static_assert(std::is_same_v, ""); assert(std::get<0>(v) == 0); assert(&ref1 == &std::get<0>(v)); auto& ref2 = v.emplace<0>(42); static_assert(std::is_same_v, ""); assert(std::get<0>(v) == 42); assert(&ref2 == &std::get<0>(v)); } { using V = std::variant; const int x = 100; V v(std::in_place_index<0>, -1); // default emplace a value auto& ref1 = v.emplace<1>(); static_assert(std::is_same_v, ""); assert(std::get<1>(v) == 0); assert(&ref1 == &std::get<1>(v)); auto& ref2 = v.emplace<2>(&x); static_assert(std::is_same_v, ""); assert(std::get<2>(v) == &x); assert(&ref2 == &std::get<2>(v)); // emplace with multiple args auto& ref3 = v.emplace<4>(3, 'a'); static_assert(std::is_same_v, ""); assert(std::get<4>(v) == "aaa"); assert(&ref3 == &std::get<4>(v)); } #if !defined(TEST_VARIANT_HAS_NO_REFERENCES) { using V = std::variant; const int x = 100; int y = 42; int z = 43; V v(std::in_place_index<0>, -1); // default emplace a value auto& ref1 = v.emplace<1>(); static_assert(std::is_same_v, ""); assert(std::get<1>(v) == 0); assert(&ref1 == &std::get<1>(v)); // emplace a reference auto& ref2 = v.emplace<2>(x); static_assert(std::is_same_v<&, decltype(ref)>, ""); assert(&std::get<2>(v) == &x); assert(&ref2 == &std::get<2>(v)); // emplace an rvalue reference auto& ref3 = v.emplace<3>(std::move(y)); static_assert(std::is_same_v<&, decltype(ref)>, ""); assert(&std::get<3>(v) == &y); assert(&ref3 == &std::get<3>(v)); // re-emplace a new reference over the active member auto& ref4 = v.emplace<3>(std::move(z)); static_assert(std::is_same_v<&, decltype(ref)>, ""); assert(&std::get<3>(v) == &z); assert(&ref4 == &std::get<3>(v)); // emplace with multiple args auto& ref5 = v.emplace<5>(3, 'a'); static_assert(std::is_same_v, ""); assert(std::get<5>(v) == "aaa"); assert(&ref5 == &std::get<5>(v)); } #endif } int main(int, char**) { test_basic(); test_emplace_sfinae(); return 0; }