-
Notifications
You must be signed in to change notification settings - Fork 51
/
Copy pathchoc_MathHelpers.h
125 lines (109 loc) · 4.51 KB
/
choc_MathHelpers.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
//
// ██████ ██ ██ ██████ ██████
// ██ ██ ██ ██ ██ ██ ** Classy Header-Only Classes **
// ██ ███████ ██ ██ ██
// ██ ██ ██ ██ ██ ██ https://github.com/Tracktion/choc
// ██████ ██ ██ ██████ ██████
//
// CHOC is (C)2022 Tracktion Corporation, and is offered under the terms of the ISC license:
//
// Permission to use, copy, modify, and/or distribute this software for any purpose with or
// without fee is hereby granted, provided that the above copyright notice and this permission
// notice appear in all copies. THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL
// WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
// AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR
// CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS,
// WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
// CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
#ifndef CHOC_MATH_HELPERS_HEADER_INCLUDED
#define CHOC_MATH_HELPERS_HEADER_INCLUDED
#include <cstdlib>
#include <cstdint>
#ifdef _MSC_VER
#include <intrin.h>
#pragma intrinsic (_BitScanReverse)
#ifdef _WIN64
#pragma intrinsic (_BitScanReverse64)
#endif
#if defined (_M_X64) && ! defined (_M_ARM64EC)
#pragma intrinsic (_umul128)
#define CHOC_HAS_UMUL128 1
#endif
#endif
namespace choc::math
{
//==============================================================================
/// Returns true if the given value is 2^something
template <typename Integer>
constexpr bool isPowerOf2 (Integer n) { return n > 0 && (n & (n - 1)) == 0; }
// Returns the number of contiguously-clear upper bits in a 32-bit value
/// Note this operation is undefined for value == 0!
inline uint32_t countUpperClearBits (uint32_t value)
{
#ifdef _MSC_VER
unsigned long result = 0;
_BitScanReverse (&result, static_cast<unsigned long> (value));
return static_cast<uint32_t> (31u - result);
#else
return static_cast<uint32_t> (__builtin_clz (value));
#endif
}
/// Returns the number of contiguously-clear upper bits in a 64-bit value.
/// Note this operation is undefined for value == 0!
inline uint32_t countUpperClearBits (uint64_t value)
{
#ifdef _MSC_VER
unsigned long result = 0;
#ifdef _WIN64
_BitScanReverse64 (&result, value);
#else
if (_BitScanReverse (&result, static_cast<unsigned long> (value >> 32))) return static_cast<uint32_t> (31u - result);
_BitScanReverse (&result, static_cast<unsigned long> (value));
#endif
return static_cast<uint32_t> (63u - result);
#else
return static_cast<uint32_t> (__builtin_clzll (value));
#endif
}
/// Returns the number of decimal digits required to print a given unsigned number
inline int getNumDecimalDigits (uint32_t n)
{
return n < 1000 ? (n < 10 ? 1 : (n < 100 ? 2 : 3))
: n < 1000000 ? (n < 10000 ? 4 : (n < 100000 ? 5 : 6))
: n < 100000000 ? (n < 10000000 ? 7 : 8)
: n < 1000000000 ? 9 : 10;
}
//==============================================================================
/// Used as a return type for multiply128()
struct Int128
{
uint64_t high, low;
};
/// A cross-platform function to multiply two 64-bit numbers and return a 128-bit result
inline Int128 multiply128 (uint64_t a, uint64_t b)
{
#if CHOC_HAS_UMUL128
Int128 result;
result.low = _umul128 (a, b, &result.high);
return result;
#elif __LP64__
#if __GNUC__
#pragma GCC diagnostic push
#pragma GCC diagnostic ignored "-Wpedantic"
#endif
auto total = static_cast<unsigned __int128> (a) * static_cast<unsigned __int128> (b);
#if __GNUC__
#pragma GCC diagnostic pop
#endif
return { static_cast<uint64_t> (total >> 64), static_cast<uint64_t> (total) };
#else
uint64_t a0 = static_cast<uint32_t> (a), a1 = a >> 32,
b0 = static_cast<uint32_t> (b), b1 = b >> 32;
auto p10 = a1 * b0, p00 = a0 * b0,
p11 = a1 * b1, p01 = a0 * b1;
auto middleBits = p10 + static_cast<uint32_t> (p01) + (p00 >> 32);
return { p11 + (middleBits >> 32) + (p01 >> 32), (middleBits << 32) | static_cast<uint32_t> (p00) };
#endif
}
} // namespace choc::math
#endif