1*61046927SAndroid Build Coastguard Worker /*
2*61046927SAndroid Build Coastguard Worker * Copyright © 2022 Imagination Technologies Ltd.
3*61046927SAndroid Build Coastguard Worker *
4*61046927SAndroid Build Coastguard Worker * Permission is hereby granted, free of charge, to any person obtaining a copy
5*61046927SAndroid Build Coastguard Worker * of this software and associated documentation files (the "Software"), to deal
6*61046927SAndroid Build Coastguard Worker * in the Software without restriction, including without limitation the rights
7*61046927SAndroid Build Coastguard Worker * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
8*61046927SAndroid Build Coastguard Worker * copies of the Software, and to permit persons to whom the Software is
9*61046927SAndroid Build Coastguard Worker * furnished to do so, subject to the following conditions:
10*61046927SAndroid Build Coastguard Worker *
11*61046927SAndroid Build Coastguard Worker * The above copyright notice and this permission notice (including the next
12*61046927SAndroid Build Coastguard Worker * paragraph) shall be included in all copies or substantial portions of the
13*61046927SAndroid Build Coastguard Worker * Software.
14*61046927SAndroid Build Coastguard Worker *
15*61046927SAndroid Build Coastguard Worker * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16*61046927SAndroid Build Coastguard Worker * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17*61046927SAndroid Build Coastguard Worker * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18*61046927SAndroid Build Coastguard Worker * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19*61046927SAndroid Build Coastguard Worker * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20*61046927SAndroid Build Coastguard Worker * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21*61046927SAndroid Build Coastguard Worker * SOFTWARE.
22*61046927SAndroid Build Coastguard Worker */
23*61046927SAndroid Build Coastguard Worker
24*61046927SAndroid Build Coastguard Worker #ifndef PVR_UTIL_H
25*61046927SAndroid Build Coastguard Worker #define PVR_UTIL_H
26*61046927SAndroid Build Coastguard Worker
27*61046927SAndroid Build Coastguard Worker #include <assert.h>
28*61046927SAndroid Build Coastguard Worker #include <stdint.h>
29*61046927SAndroid Build Coastguard Worker
30*61046927SAndroid Build Coastguard Worker #include "pvr_types.h"
31*61046927SAndroid Build Coastguard Worker
32*61046927SAndroid Build Coastguard Worker #include "util/bitscan.h"
33*61046927SAndroid Build Coastguard Worker #include "util/macros.h"
34*61046927SAndroid Build Coastguard Worker
pvr_dev_addr_is_aligned(pvr_dev_addr_t addr,const uint32_t alignment)35*61046927SAndroid Build Coastguard Worker static inline bool pvr_dev_addr_is_aligned(pvr_dev_addr_t addr,
36*61046927SAndroid Build Coastguard Worker const uint32_t alignment)
37*61046927SAndroid Build Coastguard Worker {
38*61046927SAndroid Build Coastguard Worker assert(util_is_power_of_two_nonzero(alignment));
39*61046927SAndroid Build Coastguard Worker return ((uintptr_t)(addr.addr) & (alignment - 1)) == 0;
40*61046927SAndroid Build Coastguard Worker }
41*61046927SAndroid Build Coastguard Worker
ptr_is_aligned(const void * const ptr,const uint32_t alignment)42*61046927SAndroid Build Coastguard Worker static inline bool ptr_is_aligned(const void *const ptr,
43*61046927SAndroid Build Coastguard Worker const uint32_t alignment)
44*61046927SAndroid Build Coastguard Worker {
45*61046927SAndroid Build Coastguard Worker assert(util_is_power_of_two_nonzero(alignment));
46*61046927SAndroid Build Coastguard Worker return ((uintptr_t)(ptr) & (alignment - 1)) == 0;
47*61046927SAndroid Build Coastguard Worker }
48*61046927SAndroid Build Coastguard Worker
49*61046927SAndroid Build Coastguard Worker /*****************************************************************************
50*61046927SAndroid Build Coastguard Worker Math functions
51*61046927SAndroid Build Coastguard Worker *****************************************************************************/
52*61046927SAndroid Build Coastguard Worker
u32_bin_digits(uint32_t x)53*61046927SAndroid Build Coastguard Worker static inline uint32_t u32_bin_digits(uint32_t x)
54*61046927SAndroid Build Coastguard Worker {
55*61046927SAndroid Build Coastguard Worker return x == 0 ? 1 : util_last_bit(x);
56*61046927SAndroid Build Coastguard Worker }
57*61046927SAndroid Build Coastguard Worker
u64_bin_digits(uint64_t x)58*61046927SAndroid Build Coastguard Worker static inline uint32_t u64_bin_digits(uint64_t x)
59*61046927SAndroid Build Coastguard Worker {
60*61046927SAndroid Build Coastguard Worker return x == 0 ? 1 : util_last_bit64(x);
61*61046927SAndroid Build Coastguard Worker }
62*61046927SAndroid Build Coastguard Worker
63*61046927SAndroid Build Coastguard Worker extern const uint8_t est_log10_from_log2[64 + 1];
64*61046927SAndroid Build Coastguard Worker extern const uint32_t u32_powers_of_ten[10];
65*61046927SAndroid Build Coastguard Worker extern const uint64_t u64_powers_of_ten[20];
66*61046927SAndroid Build Coastguard Worker
67*61046927SAndroid Build Coastguard Worker /*
68*61046927SAndroid Build Coastguard Worker * This function includes parts of a public-domain log10 algorithm from
69*61046927SAndroid Build Coastguard Worker * "Bit Twiddling Hacks", an aggregate collection which is:
70*61046927SAndroid Build Coastguard Worker *
71*61046927SAndroid Build Coastguard Worker * © 1997-2005 Sean Eron Anderson.
72*61046927SAndroid Build Coastguard Worker *
73*61046927SAndroid Build Coastguard Worker * https://graphics.stanford.edu/~seander/bithacks.html#IntegerLog10
74*61046927SAndroid Build Coastguard Worker */
u32_dec_digits(uint32_t x)75*61046927SAndroid Build Coastguard Worker static inline uint32_t u32_dec_digits(uint32_t x)
76*61046927SAndroid Build Coastguard Worker {
77*61046927SAndroid Build Coastguard Worker /* The value of nr_digits is an estimation of ceil(log10(x)) with variance
78*61046927SAndroid Build Coastguard Worker * +0/-1 (as per the comment on the estimation lut).
79*61046927SAndroid Build Coastguard Worker */
80*61046927SAndroid Build Coastguard Worker const uint32_t nr_digits = est_log10_from_log2[util_last_bit(x)];
81*61046927SAndroid Build Coastguard Worker
82*61046927SAndroid Build Coastguard Worker /* We then convert this approximation to a real value by adding 1 iff
83*61046927SAndroid Build Coastguard Worker * x >= 10**nr_digits using another lut to quickly compute the
84*61046927SAndroid Build Coastguard Worker * exponentiation. See the comment on u32_powers_of_ten for details of this.
85*61046927SAndroid Build Coastguard Worker */
86*61046927SAndroid Build Coastguard Worker return nr_digits + (x >= u32_powers_of_ten[nr_digits]);
87*61046927SAndroid Build Coastguard Worker }
88*61046927SAndroid Build Coastguard Worker
89*61046927SAndroid Build Coastguard Worker /* This function is an extended form of u32_dec_digits(); see the comments on
90*61046927SAndroid Build Coastguard Worker * that function for details.
91*61046927SAndroid Build Coastguard Worker */
u64_dec_digits(uint64_t x)92*61046927SAndroid Build Coastguard Worker static inline uint32_t u64_dec_digits(uint64_t x)
93*61046927SAndroid Build Coastguard Worker {
94*61046927SAndroid Build Coastguard Worker const uint32_t nr_digits = est_log10_from_log2[util_last_bit64(x)];
95*61046927SAndroid Build Coastguard Worker return nr_digits + (x >= u64_powers_of_ten[nr_digits]);
96*61046927SAndroid Build Coastguard Worker }
97*61046927SAndroid Build Coastguard Worker
u32_hex_digits(uint32_t x)98*61046927SAndroid Build Coastguard Worker static inline uint32_t u32_hex_digits(uint32_t x)
99*61046927SAndroid Build Coastguard Worker {
100*61046927SAndroid Build Coastguard Worker const uint32_t binary_digits = u32_bin_digits(x);
101*61046927SAndroid Build Coastguard Worker return (binary_digits >> 2) + !!(binary_digits & 3);
102*61046927SAndroid Build Coastguard Worker }
103*61046927SAndroid Build Coastguard Worker
u64_hex_digits(uint64_t x)104*61046927SAndroid Build Coastguard Worker static inline uint32_t u64_hex_digits(uint64_t x)
105*61046927SAndroid Build Coastguard Worker {
106*61046927SAndroid Build Coastguard Worker const uint32_t binary_digits = u64_bin_digits(x);
107*61046927SAndroid Build Coastguard Worker return (binary_digits >> 2) + !!(binary_digits & 3);
108*61046927SAndroid Build Coastguard Worker }
109*61046927SAndroid Build Coastguard Worker
110*61046927SAndroid Build Coastguard Worker #endif /* PVR_UTIL_H */
111