11a036e9cSMark de Wever //===----------------------------------------------------------------------===// 21a036e9cSMark de Wever // 31ee41b41SNikolas Klauser // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 41ee41b41SNikolas Klauser // See https://llvm.org/LICENSE.txt for license information. 51ee41b41SNikolas Klauser // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 61a036e9cSMark de Wever // 71a036e9cSMark de Wever //===----------------------------------------------------------------------===// 81ee41b41SNikolas Klauser 91a036e9cSMark de Wever // UNSUPPORTED: c++03, c++11, c++14, c++17 101a036e9cSMark de Wever 111a036e9cSMark de Wever // template <class T> 123347e7d4SArthur O'Dwyer // constexpr int bit_width(T x) noexcept; 131a036e9cSMark de Wever 14e130fbe2SArthur O'Dwyer // Constraints: T is an unsigned integer type 15e130fbe2SArthur O'Dwyer // Returns: If x == 0, 0; otherwise one plus the base-2 logarithm of x, with any fractional part discarded. 161a036e9cSMark de Wever 171a036e9cSMark de Wever 181a036e9cSMark de Wever #include <bit> 191a036e9cSMark de Wever #include <cassert> 20*d6832a61SLouis Dionne #include <cstddef> 21e130fbe2SArthur O'Dwyer #include <cstdint> 2286aac87fSNikolas Klauser #include <limits> 23e130fbe2SArthur O'Dwyer #include <type_traits> 241a036e9cSMark de Wever 251a036e9cSMark de Wever #include "test_macros.h" 261a036e9cSMark de Wever 27e130fbe2SArthur O'Dwyer struct A {}; 281a036e9cSMark de Wever enum E1 : unsigned char { rEd }; 291a036e9cSMark de Wever enum class E2 : unsigned char { red }; 301a036e9cSMark de Wever 31e130fbe2SArthur O'Dwyer template <class T> 32e130fbe2SArthur O'Dwyer constexpr bool test() 331a036e9cSMark de Wever { 343347e7d4SArthur O'Dwyer ASSERT_SAME_TYPE(decltype(std::bit_width(T())), int); 35e130fbe2SArthur O'Dwyer ASSERT_NOEXCEPT(std::bit_width(T())); 36e130fbe2SArthur O'Dwyer T max = std::numeric_limits<T>::max(); 371a036e9cSMark de Wever 383347e7d4SArthur O'Dwyer assert(std::bit_width(T(0)) == 0); 393347e7d4SArthur O'Dwyer assert(std::bit_width(T(1)) == 1); 403347e7d4SArthur O'Dwyer assert(std::bit_width(T(2)) == 2); 413347e7d4SArthur O'Dwyer assert(std::bit_width(T(3)) == 2); 423347e7d4SArthur O'Dwyer assert(std::bit_width(T(4)) == 3); 433347e7d4SArthur O'Dwyer assert(std::bit_width(T(5)) == 3); 443347e7d4SArthur O'Dwyer assert(std::bit_width(T(6)) == 3); 453347e7d4SArthur O'Dwyer assert(std::bit_width(T(7)) == 3); 463347e7d4SArthur O'Dwyer assert(std::bit_width(T(8)) == 4); 473347e7d4SArthur O'Dwyer assert(std::bit_width(T(9)) == 4); 483347e7d4SArthur O'Dwyer assert(std::bit_width(T(125)) == 7); 493347e7d4SArthur O'Dwyer assert(std::bit_width(T(126)) == 7); 503347e7d4SArthur O'Dwyer assert(std::bit_width(T(127)) == 7); 513347e7d4SArthur O'Dwyer assert(std::bit_width(T(128)) == 8); 523347e7d4SArthur O'Dwyer assert(std::bit_width(T(129)) == 8); 533347e7d4SArthur O'Dwyer assert(std::bit_width(T(130)) == 8); 543347e7d4SArthur O'Dwyer assert(std::bit_width(T(max - 1)) == std::numeric_limits<T>::digits); 553347e7d4SArthur O'Dwyer assert(std::bit_width(max) == std::numeric_limits<T>::digits); 56e130fbe2SArthur O'Dwyer 578f972cb0SMark de Wever #ifndef TEST_HAS_NO_INT128 58e130fbe2SArthur O'Dwyer if constexpr (std::is_same_v<T, __uint128_t>) { 59e130fbe2SArthur O'Dwyer T val = 128; 60e130fbe2SArthur O'Dwyer val <<= 32; 61e130fbe2SArthur O'Dwyer assert(std::bit_width(val-1) == 39); 62e130fbe2SArthur O'Dwyer assert(std::bit_width(val) == 40); 63e130fbe2SArthur O'Dwyer assert(std::bit_width(val+1) == 40); 64e130fbe2SArthur O'Dwyer val <<= 60; 65e130fbe2SArthur O'Dwyer assert(std::bit_width(val-1) == 99); 66e130fbe2SArthur O'Dwyer assert(std::bit_width(val) == 100); 67e130fbe2SArthur O'Dwyer assert(std::bit_width(val+1) == 100); 68e130fbe2SArthur O'Dwyer } 69e130fbe2SArthur O'Dwyer #endif 70e130fbe2SArthur O'Dwyer 71e130fbe2SArthur O'Dwyer return true; 721a036e9cSMark de Wever } 731a036e9cSMark de Wever 741a036e9cSMark de Wever int main(int, char**) 751a036e9cSMark de Wever { 761a036e9cSMark de Wever 771a036e9cSMark de Wever { 781a036e9cSMark de Wever auto lambda = [](auto x) -> decltype(std::bit_width(x)) {}; 791a036e9cSMark de Wever using L = decltype(lambda); 801a036e9cSMark de Wever 81e130fbe2SArthur O'Dwyer static_assert(!std::is_invocable_v<L, signed char>); 82e130fbe2SArthur O'Dwyer static_assert(!std::is_invocable_v<L, short>); 83e130fbe2SArthur O'Dwyer static_assert(!std::is_invocable_v<L, int>); 84e130fbe2SArthur O'Dwyer static_assert(!std::is_invocable_v<L, long>); 85e130fbe2SArthur O'Dwyer static_assert(!std::is_invocable_v<L, long long>); 868f972cb0SMark de Wever #ifndef TEST_HAS_NO_INT128 87e130fbe2SArthur O'Dwyer static_assert(!std::is_invocable_v<L, __int128_t>); 881a036e9cSMark de Wever #endif 891a036e9cSMark de Wever 90bd5d0feeSMark de Wever static_assert(!std::is_invocable_v<L, std::int8_t>); 91bd5d0feeSMark de Wever static_assert(!std::is_invocable_v<L, std::int16_t>); 92bd5d0feeSMark de Wever static_assert(!std::is_invocable_v<L, std::int32_t>); 93bd5d0feeSMark de Wever static_assert(!std::is_invocable_v<L, std::int64_t>); 9492fc93e1SMark de Wever static_assert(!std::is_invocable_v<L, std::intmax_t>); 95d59a43feSMark de Wever static_assert(!std::is_invocable_v<L, std::intptr_t>); 96d8681356SMark de Wever static_assert(!std::is_invocable_v<L, std::ptrdiff_t>); 97e130fbe2SArthur O'Dwyer 98e130fbe2SArthur O'Dwyer static_assert(!std::is_invocable_v<L, bool>); 99e130fbe2SArthur O'Dwyer static_assert(!std::is_invocable_v<L, char>); 100e130fbe2SArthur O'Dwyer static_assert(!std::is_invocable_v<L, wchar_t>); 1019027887eSMark de Wever #ifndef TEST_HAS_NO_CHAR8_T 102e130fbe2SArthur O'Dwyer static_assert(!std::is_invocable_v<L, char8_t>); 103e130fbe2SArthur O'Dwyer #endif 104e130fbe2SArthur O'Dwyer static_assert(!std::is_invocable_v<L, char16_t>); 105e130fbe2SArthur O'Dwyer static_assert(!std::is_invocable_v<L, char32_t>); 106e130fbe2SArthur O'Dwyer 107e130fbe2SArthur O'Dwyer static_assert(!std::is_invocable_v<L, A>); 108e130fbe2SArthur O'Dwyer static_assert(!std::is_invocable_v<L, A*>); 109e130fbe2SArthur O'Dwyer static_assert(!std::is_invocable_v<L, E1>); 110e130fbe2SArthur O'Dwyer static_assert(!std::is_invocable_v<L, E2>); 1111a036e9cSMark de Wever } 1121a036e9cSMark de Wever 113e130fbe2SArthur O'Dwyer static_assert(test<unsigned char>()); 114e130fbe2SArthur O'Dwyer static_assert(test<unsigned short>()); 115e130fbe2SArthur O'Dwyer static_assert(test<unsigned int>()); 116e130fbe2SArthur O'Dwyer static_assert(test<unsigned long>()); 117e130fbe2SArthur O'Dwyer static_assert(test<unsigned long long>()); 1188f972cb0SMark de Wever #ifndef TEST_HAS_NO_INT128 119e130fbe2SArthur O'Dwyer static_assert(test<__uint128_t>()); 1201a036e9cSMark de Wever #endif 121bd5d0feeSMark de Wever static_assert(test<std::uint8_t>()); 122bd5d0feeSMark de Wever static_assert(test<std::uint16_t>()); 123da79d6e1SMark de Wever static_assert(test<std::uint32_t>()); 124bd5d0feeSMark de Wever static_assert(test<std::uint64_t>()); 12592fc93e1SMark de Wever static_assert(test<std::uintmax_t>()); 126d59a43feSMark de Wever static_assert(test<std::uintptr_t>()); 127fb855eb9SMark de Wever static_assert(test<std::size_t>()); 1281a036e9cSMark de Wever 129e130fbe2SArthur O'Dwyer test<unsigned char>(); 130e130fbe2SArthur O'Dwyer test<unsigned short>(); 131e130fbe2SArthur O'Dwyer test<unsigned int>(); 132e130fbe2SArthur O'Dwyer test<unsigned long>(); 133e130fbe2SArthur O'Dwyer test<unsigned long long>(); 1348f972cb0SMark de Wever #ifndef TEST_HAS_NO_INT128 135e130fbe2SArthur O'Dwyer test<__uint128_t>(); 1361a036e9cSMark de Wever #endif 137bd5d0feeSMark de Wever test<std::uint8_t>(); 138bd5d0feeSMark de Wever test<std::uint16_t>(); 139da79d6e1SMark de Wever test<std::uint32_t>(); 140bd5d0feeSMark de Wever test<std::uint64_t>(); 14192fc93e1SMark de Wever test<std::uintmax_t>(); 142d59a43feSMark de Wever test<std::uintptr_t>(); 143fb855eb9SMark de Wever test<std::size_t>(); 1441a036e9cSMark de Wever 1451a036e9cSMark de Wever return 0; 1461a036e9cSMark de Wever } 147