blob: 40eb011f195fdc8f1ed05cbec4b4a062ea4f0007 [file] [log] [blame]
Austin Schuh36244a12019-09-21 17:52:38 -07001// Copyright 2017 The Abseil Authors.
2//
3// Licensed under the Apache License, Version 2.0 (the "License");
4// you may not use this file except in compliance with the License.
5// You may obtain a copy of the License at
6//
7// https://www.apache.org/licenses/LICENSE-2.0
8//
9// Unless required by applicable law or agreed to in writing, software
10// distributed under the License is distributed on an "AS IS" BASIS,
11// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12// See the License for the specific language governing permissions and
13// limitations under the License.
14
15#ifndef ABSL_RANDOM_INTERNAL_TRAITS_H_
16#define ABSL_RANDOM_INTERNAL_TRAITS_H_
17
18#include <cstdint>
19#include <limits>
20#include <type_traits>
21
22#include "absl/base/config.h"
23
24namespace absl {
25namespace random_internal {
26
27// random_internal::is_widening_convertible<A, B>
28//
29// Returns whether a type A is widening-convertible to a type B.
30//
31// A is widening-convertible to B means:
32// A a = <any number>;
33// B b = a;
34// A c = b;
35// EXPECT_EQ(a, c);
36template <typename A, typename B>
37class is_widening_convertible {
38 // As long as there are enough bits in the exact part of a number:
39 // - unsigned can fit in float, signed, unsigned
40 // - signed can fit in float, signed
41 // - float can fit in float
42 // So we define rank to be:
43 // - rank(float) -> 2
44 // - rank(signed) -> 1
45 // - rank(unsigned) -> 0
46 template <class T>
47 static constexpr int rank() {
48 return !std::numeric_limits<T>::is_integer +
49 std::numeric_limits<T>::is_signed;
50 }
51
52 public:
53 // If an arithmetic-type B can represent at least as many digits as a type A,
54 // and B belongs to a rank no lower than A, then A can be safely represented
55 // by B through a widening-conversion.
56 static constexpr bool value =
57 std::numeric_limits<A>::digits <= std::numeric_limits<B>::digits &&
58 rank<A>() <= rank<B>();
59};
60
61// unsigned_bits<N>::type returns the unsigned int type with the indicated
62// number of bits.
63template <size_t N>
64struct unsigned_bits;
65
66template <>
67struct unsigned_bits<8> {
68 using type = uint8_t;
69};
70template <>
71struct unsigned_bits<16> {
72 using type = uint16_t;
73};
74template <>
75struct unsigned_bits<32> {
76 using type = uint32_t;
77};
78template <>
79struct unsigned_bits<64> {
80 using type = uint64_t;
81};
82
83#ifdef ABSL_HAVE_INTRINSIC_INT128
84template <>
85struct unsigned_bits<128> {
86 using type = __uint128_t;
87};
88#endif
89
90template <typename IntType>
91struct make_unsigned_bits {
92 using type = typename unsigned_bits<std::numeric_limits<
93 typename std::make_unsigned<IntType>::type>::digits>::type;
94};
95
96} // namespace random_internal
97} // namespace absl
98
99#endif // ABSL_RANDOM_INTERNAL_TRAITS_H_