xref: /llvm-project/libcxx/test/std/numerics/bit/bit.cast/bit_cast.compile.pass.cpp (revision d2baefae6846765eef6a6dd69d4fdf1082ce29ad)
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