// -*- 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 // // template // constexpr see below visit(Visitor&& vis, Variants&&... vars); #include #include #include #include #include #include #include "test_macros.h" #include "type_id.h" #include "variant_test_helpers.hpp" enum CallType : unsigned { CT_None, CT_NonConst = 1, CT_Const = 2, CT_LValue = 4, CT_RValue = 8 }; inline constexpr CallType operator|(CallType LHS, CallType RHS) { return static_cast(static_cast(LHS) | static_cast(RHS)); } struct ForwardingCallObject { template bool operator()(Args &&...) & { set_call(CT_NonConst | CT_LValue); return true; } template bool operator()(Args &&...) const & { set_call(CT_Const | CT_LValue); return true; } // Don't allow the call operator to be invoked as an rvalue. template bool operator()(Args &&...) && { set_call(CT_NonConst | CT_RValue); return true; } template bool operator()(Args &&...) const && { set_call(CT_Const | CT_RValue); return true; } template static void set_call(CallType type) { assert(last_call_type == CT_None); assert(last_call_args == nullptr); last_call_type = type; last_call_args = std::addressof(makeArgumentID()); } template static bool check_call(CallType type) { bool result = last_call_type == type && last_call_args && *last_call_args == makeArgumentID(); last_call_type = CT_None; last_call_args = nullptr; return result; } static CallType last_call_type; static const TypeID *last_call_args; }; CallType ForwardingCallObject::last_call_type = CT_None; const TypeID *ForwardingCallObject::last_call_args = nullptr; void test_call_operator_forwarding() { using Fn = ForwardingCallObject; Fn obj{}; const Fn &cobj = obj; { // test call operator forwarding - single variant, single arg using V = std::variant; V v(42); std::visit(obj, v); assert(Fn::check_call(CT_NonConst | CT_LValue)); std::visit(cobj, v); assert(Fn::check_call(CT_Const | CT_LValue)); std::visit(std::move(obj), v); assert(Fn::check_call(CT_NonConst | CT_RValue)); std::visit(std::move(cobj), v); assert(Fn::check_call(CT_Const | CT_RValue)); } { // test call operator forwarding - single variant, multi arg using V = std::variant; V v(42l); std::visit(obj, v); assert(Fn::check_call(CT_NonConst | CT_LValue)); std::visit(cobj, v); assert(Fn::check_call(CT_Const | CT_LValue)); std::visit(std::move(obj), v); assert(Fn::check_call(CT_NonConst | CT_RValue)); std::visit(std::move(cobj), v); assert(Fn::check_call(CT_Const | CT_RValue)); } { // test call operator forwarding - multi variant, multi arg using V = std::variant; using V2 = std::variant; V v(42l); V2 v2("hello"); std::visit(obj, v, v2); assert((Fn::check_call(CT_NonConst | CT_LValue))); std::visit(cobj, v, v2); assert((Fn::check_call(CT_Const | CT_LValue))); std::visit(std::move(obj), v, v2); assert((Fn::check_call(CT_NonConst | CT_RValue))); std::visit(std::move(cobj), v, v2); assert((Fn::check_call(CT_Const | CT_RValue))); } } void test_argument_forwarding() { using Fn = ForwardingCallObject; Fn obj{}; const auto Val = CT_LValue | CT_NonConst; { // single argument - value type using V = std::variant; V v(42); const V &cv = v; std::visit(obj, v); assert(Fn::check_call(Val)); std::visit(obj, cv); assert(Fn::check_call(Val)); std::visit(obj, std::move(v)); assert(Fn::check_call(Val)); std::visit(obj, std::move(cv)); assert(Fn::check_call(Val)); } #if !defined(TEST_VARIANT_HAS_NO_REFERENCES) { // single argument - lvalue reference using V = std::variant; int x = 42; V v(x); const V &cv = v; std::visit(obj, v); assert(Fn::check_call(Val)); std::visit(obj, cv); assert(Fn::check_call(Val)); std::visit(obj, std::move(v)); assert(Fn::check_call(Val)); std::visit(obj, std::move(cv)); assert(Fn::check_call(Val)); } { // single argument - rvalue reference using V = std::variant; int x = 42; V v(std::move(x)); const V &cv = v; std::visit(obj, v); assert(Fn::check_call(Val)); std::visit(obj, cv); assert(Fn::check_call(Val)); std::visit(obj, std::move(v)); assert(Fn::check_call(Val)); std::visit(obj, std::move(cv)); assert(Fn::check_call(Val)); } { // multi argument - multi variant using S = const std::string &; using V = std::variant; const std::string str = "hello"; long l = 43; V v1(42); const V &cv1 = v1; V v2(str); const V &cv2 = v2; V v3(std::move(l)); const V &cv3 = v3; std::visit(obj, v1, v2, v3); assert((Fn::check_call(Val))); std::visit(obj, cv1, cv2, std::move(v3)); assert((Fn::check_call(Val))); } #endif } struct ReturnFirst { template constexpr int operator()(int f, Args &&...) const { return f; } }; struct ReturnArity { template constexpr int operator()(Args &&...) const { return sizeof...(Args); } }; void test_constexpr() { constexpr ReturnFirst obj{}; constexpr ReturnArity aobj{}; { using V = std::variant; constexpr V v(42); static_assert(std::visit(obj, v) == 42, ""); } { using V = std::variant; constexpr V v(42l); static_assert(std::visit(obj, v) == 42, ""); } { using V1 = std::variant; using V2 = std::variant; using V3 = std::variant; constexpr V1 v1; constexpr V2 v2(nullptr); constexpr V3 v3; static_assert(std::visit(aobj, v1, v2, v3) == 3, ""); } { using V1 = std::variant; using V2 = std::variant; using V3 = std::variant; constexpr V1 v1; constexpr V2 v2(nullptr); constexpr V3 v3; static_assert(std::visit(aobj, v1, v2, v3) == 3, ""); } } void test_exceptions() { #ifndef TEST_HAS_NO_EXCEPTIONS ReturnArity obj{}; auto test = [&](auto &&... args) { try { std::visit(obj, args...); } catch (const std::bad_variant_access &) { return true; } catch (...) { } return false; }; { using V = std::variant; V v; makeEmpty(v); assert(test(v)); } { using V = std::variant; using V2 = std::variant; V v; makeEmpty(v); V2 v2("hello"); assert(test(v, v2)); } { using V = std::variant; using V2 = std::variant; V v; makeEmpty(v); V2 v2("hello"); assert(test(v2, v)); } { using V = std::variant; using V2 = std::variant; V v; makeEmpty(v); V2 v2; makeEmpty(v2); assert(test(v, v2)); } #endif } // See http://llvm.org/PR31916 void test_caller_accepts_nonconst() { struct A {}; struct Visitor { void operator()(A&) {} }; std::variant v; std::visit(Visitor{}, v); } int main() { test_call_operator_forwarding(); test_argument_forwarding(); test_constexpr(); test_exceptions(); test_caller_accepts_nonconst(); }