11dc62f26SNikolas Klauser //===----------------------------------------------------------------------===//
21dc62f26SNikolas Klauser //
31dc62f26SNikolas Klauser // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
41dc62f26SNikolas Klauser // See https://llvm.org/LICENSE.txt for license information.
51dc62f26SNikolas Klauser // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
61dc62f26SNikolas Klauser //
71dc62f26SNikolas Klauser //===----------------------------------------------------------------------===//
81dc62f26SNikolas Klauser 
91dc62f26SNikolas Klauser // UNSUPPORTED: c++03, c++11, c++14, c++17, c++20
101dc62f26SNikolas Klauser 
111dc62f26SNikolas Klauser #include <bit>
121dc62f26SNikolas Klauser #include <cassert>
131dc62f26SNikolas Klauser #include <cstdint>
141dc62f26SNikolas Klauser #include <utility>
151dc62f26SNikolas Klauser 
161dc62f26SNikolas Klauser #include "test_macros.h"
171dc62f26SNikolas Klauser 
181dc62f26SNikolas Klauser template <class T>
191dc62f26SNikolas Klauser concept has_byteswap = requires(T t) {
201dc62f26SNikolas Klauser   std::byteswap(t);
211dc62f26SNikolas Klauser };
221dc62f26SNikolas Klauser 
231dc62f26SNikolas Klauser static_assert(!has_byteswap<void*>);
241dc62f26SNikolas Klauser static_assert(!has_byteswap<float>);
251dc62f26SNikolas Klauser static_assert(!has_byteswap<char[2]>);
261dc62f26SNikolas Klauser static_assert(!has_byteswap<std::byte>);
271dc62f26SNikolas Klauser 
281dc62f26SNikolas Klauser template <class T>
test_num(T in,T expected)291dc62f26SNikolas Klauser constexpr void test_num(T in, T expected) {
301dc62f26SNikolas Klauser   assert(std::byteswap(in) == expected);
311dc62f26SNikolas Klauser   ASSERT_SAME_TYPE(decltype(std::byteswap(in)), decltype(in));
321dc62f26SNikolas Klauser   ASSERT_NOEXCEPT(std::byteswap(in));
331dc62f26SNikolas Klauser }
341dc62f26SNikolas Klauser 
351dc62f26SNikolas Klauser template <class T>
get_test_data()361dc62f26SNikolas Klauser constexpr std::pair<T, T> get_test_data() {
371dc62f26SNikolas Klauser   switch (sizeof(T)) {
381dc62f26SNikolas Klauser   case 2:
391dc62f26SNikolas Klauser     return {static_cast<T>(0x1234), static_cast<T>(0x3412)};
401dc62f26SNikolas Klauser   case 4:
411dc62f26SNikolas Klauser     return {static_cast<T>(0x60AF8503), static_cast<T>(0x0385AF60)};
421dc62f26SNikolas Klauser   case 8:
431dc62f26SNikolas Klauser     return {static_cast<T>(0xABCDFE9477936406), static_cast<T>(0x0664937794FECDAB)};
44cb71d77cSCasey Carter   default:
451dc62f26SNikolas Klauser     assert(false);
46cb71d77cSCasey Carter     return {}; // for MSVC, whose `assert` is tragically not [[noreturn]]
47cb71d77cSCasey Carter   }
481dc62f26SNikolas Klauser }
491dc62f26SNikolas Klauser 
501dc62f26SNikolas Klauser template <class T>
test_implementation_defined_size()511dc62f26SNikolas Klauser constexpr void test_implementation_defined_size() {
521dc62f26SNikolas Klauser   const auto [in, expected] = get_test_data<T>();
531dc62f26SNikolas Klauser   test_num<T>(in, expected);
541dc62f26SNikolas Klauser }
551dc62f26SNikolas Klauser 
test()561dc62f26SNikolas Klauser constexpr bool test() {
571dc62f26SNikolas Klauser   test_num<uint8_t>(0xAB, 0xAB);
581dc62f26SNikolas Klauser   test_num<uint16_t>(0xCDEF, 0xEFCD);
591dc62f26SNikolas Klauser   test_num<uint32_t>(0x01234567, 0x67452301);
601dc62f26SNikolas Klauser   test_num<uint64_t>(0x0123456789ABCDEF, 0xEFCDAB8967452301);
611dc62f26SNikolas Klauser 
62cb71d77cSCasey Carter   test_num<int8_t>(static_cast<int8_t>(0xAB), static_cast<int8_t>(0xAB));
63cb71d77cSCasey Carter   test_num<int16_t>(static_cast<int16_t>(0xCDEF), static_cast<int16_t>(0xEFCD));
641dc62f26SNikolas Klauser   test_num<int32_t>(0x01234567, 0x67452301);
651dc62f26SNikolas Klauser   test_num<int64_t>(0x0123456789ABCDEF, 0xEFCDAB8967452301);
661dc62f26SNikolas Klauser 
67*8f972cb0SMark de Wever #ifndef TEST_HAS_NO_INT128
681dc62f26SNikolas Klauser   const auto in = static_cast<__uint128_t>(0x0123456789ABCDEF) << 64 | 0x13579BDF02468ACE;
691dc62f26SNikolas Klauser   const auto expected = static_cast<__uint128_t>(0xCE8A4602DF9B5713) << 64 | 0xEFCDAB8967452301;
701dc62f26SNikolas Klauser   test_num<__uint128_t>(in, expected);
711dc62f26SNikolas Klauser   test_num<__int128_t>(in, expected);
721dc62f26SNikolas Klauser #endif
731dc62f26SNikolas Klauser 
741dc62f26SNikolas Klauser   test_num<bool>(true, true);
751dc62f26SNikolas Klauser   test_num<bool>(false, false);
76cb71d77cSCasey Carter   test_num<char>(static_cast<char>(0xCD), static_cast<char>(0xCD));
771dc62f26SNikolas Klauser   test_num<unsigned char>(0xEF, 0xEF);
781dc62f26SNikolas Klauser   test_num<signed char>(0x45, 0x45);
791dc62f26SNikolas Klauser   test_num<char8_t>(0xAB, 0xAB);
801dc62f26SNikolas Klauser   test_num<char16_t>(0xABCD, 0xCDAB);
811dc62f26SNikolas Klauser   test_num<char32_t>(0xABCDEF01, 0x01EFCDAB);
821dc62f26SNikolas Klauser #ifndef TEST_HAS_NO_WIDE_CHARACTERS
831dc62f26SNikolas Klauser   test_implementation_defined_size<wchar_t>();
841dc62f26SNikolas Klauser #endif
851dc62f26SNikolas Klauser 
861dc62f26SNikolas Klauser   test_implementation_defined_size<short>();
871dc62f26SNikolas Klauser   test_implementation_defined_size<unsigned short>();
881dc62f26SNikolas Klauser   test_implementation_defined_size<int>();
891dc62f26SNikolas Klauser   test_implementation_defined_size<unsigned int>();
901dc62f26SNikolas Klauser   test_implementation_defined_size<long>();
911dc62f26SNikolas Klauser   test_implementation_defined_size<unsigned long>();
921dc62f26SNikolas Klauser   test_implementation_defined_size<long long>();
931dc62f26SNikolas Klauser   test_implementation_defined_size<unsigned long long>();
941dc62f26SNikolas Klauser   return true;
951dc62f26SNikolas Klauser }
961dc62f26SNikolas Klauser 
main(int,char **)971dc62f26SNikolas Klauser int main(int, char**) {
981dc62f26SNikolas Klauser   test();
991dc62f26SNikolas Klauser   static_assert(test());
1001dc62f26SNikolas Klauser 
1011dc62f26SNikolas Klauser   return 0;
1021dc62f26SNikolas Klauser }
103