xref: /aosp_15_r20/external/mesa3d/src/imagination/common/pvr_util.h (revision 6104692788411f58d303aa86923a9ff6ecaded22)
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