1*b1fb3d75SLouis Dionne //===----------------------------------------------------------------------===// 2*b1fb3d75SLouis Dionne // 3*b1fb3d75SLouis Dionne // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 4*b1fb3d75SLouis Dionne // See https://llvm.org/LICENSE.txt for license information. 5*b1fb3d75SLouis Dionne // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 6*b1fb3d75SLouis Dionne // 7*b1fb3d75SLouis Dionne //===----------------------------------------------------------------------===// 8*b1fb3d75SLouis Dionne 9*b1fb3d75SLouis Dionne // UNSUPPORTED: c++03, c++11, c++14, c++17 10*b1fb3d75SLouis Dionne 11*b1fb3d75SLouis Dionne // <bit> 12*b1fb3d75SLouis Dionne // 13*b1fb3d75SLouis Dionne // template<class To, class From> 14*b1fb3d75SLouis Dionne // constexpr To bit_cast(const From& from) noexcept; // C++20 15*b1fb3d75SLouis Dionne 16*b1fb3d75SLouis Dionne // This test makes sure that std::bit_cast fails when any of the following 17*b1fb3d75SLouis Dionne // constraints are violated: 18*b1fb3d75SLouis Dionne // 19*b1fb3d75SLouis Dionne // (1.1) sizeof(To) == sizeof(From) is true; 20*b1fb3d75SLouis Dionne // (1.2) is_trivially_copyable_v<To> is true; 21*b1fb3d75SLouis Dionne // (1.3) is_trivially_copyable_v<From> is true. 22*b1fb3d75SLouis Dionne // 23*b1fb3d75SLouis Dionne // Also check that it's ill-formed when the return type would be 24*b1fb3d75SLouis Dionne // ill-formed, even though that is not explicitly mentioned in the 25*b1fb3d75SLouis Dionne // specification (but it can be inferred from the synopsis). 26*b1fb3d75SLouis Dionne 27*b1fb3d75SLouis Dionne #include <bit> 28*b1fb3d75SLouis Dionne #include <concepts> 29*b1fb3d75SLouis Dionne 30*b1fb3d75SLouis Dionne template<class To, class From> 31*b1fb3d75SLouis Dionne concept bit_cast_is_valid = requires(From from) { 32*b1fb3d75SLouis Dionne { std::bit_cast<To>(from) } -> std::same_as<To>; 33*b1fb3d75SLouis Dionne }; 34*b1fb3d75SLouis Dionne 35*b1fb3d75SLouis Dionne // Types are not the same size 36*b1fb3d75SLouis Dionne namespace ns1 { 37*b1fb3d75SLouis Dionne struct To { char a; }; 38*b1fb3d75SLouis Dionne struct From { char a; char b; }; 39*b1fb3d75SLouis Dionne static_assert(!bit_cast_is_valid<To, From>); 40*b1fb3d75SLouis Dionne static_assert(!bit_cast_is_valid<From&, From>); 41*b1fb3d75SLouis Dionne } 42*b1fb3d75SLouis Dionne 43*b1fb3d75SLouis Dionne // To is not trivially copyable 44*b1fb3d75SLouis Dionne namespace ns2 { 45*b1fb3d75SLouis Dionne struct To { char a; To(To const&); }; 46*b1fb3d75SLouis Dionne struct From { char a; }; 47*b1fb3d75SLouis Dionne static_assert(!bit_cast_is_valid<To, From>); 48*b1fb3d75SLouis Dionne } 49*b1fb3d75SLouis Dionne 50*b1fb3d75SLouis Dionne // From is not trivially copyable 51*b1fb3d75SLouis Dionne namespace ns3 { 52*b1fb3d75SLouis Dionne struct To { char a; }; 53*b1fb3d75SLouis Dionne struct From { char a; From(From const&); }; 54*b1fb3d75SLouis Dionne static_assert(!bit_cast_is_valid<To, From>); 55*b1fb3d75SLouis Dionne } 56*b1fb3d75SLouis Dionne 57*b1fb3d75SLouis Dionne // The return type is ill-formed 58*b1fb3d75SLouis Dionne namespace ns4 { 59*b1fb3d75SLouis Dionne struct From { char a; char b; }; 60*b1fb3d75SLouis Dionne static_assert(!bit_cast_is_valid<char[2], From>); 61*b1fb3d75SLouis Dionne static_assert(!bit_cast_is_valid<int(), From>); 62*b1fb3d75SLouis Dionne } 63