]> CyberLeo.Net >> Repos - FreeBSD/releng/7.2.git/blob - sys/cddl/compat/opensolaris/sys/sysmacros.h
Create releng/7.2 from stable/7 in preparation for 7.2-RELEASE.
[FreeBSD/releng/7.2.git] / sys / cddl / compat / opensolaris / sys / sysmacros.h
1 /*
2  * CDDL HEADER START
3  *
4  * The contents of this file are subject to the terms of the
5  * Common Development and Distribution License, Version 1.0 only
6  * (the "License").  You may not use this file except in compliance
7  * with the License.
8  *
9  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
10  * or http://www.opensolaris.org/os/licensing.
11  * See the License for the specific language governing permissions
12  * and limitations under the License.
13  *
14  * When distributing Covered Code, include this CDDL HEADER in each
15  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
16  * If applicable, add the following below this CDDL HEADER, with the
17  * fields enclosed by brackets "[]" replaced with your own identifying
18  * information: Portions Copyright [yyyy] [name of copyright owner]
19  *
20  * CDDL HEADER END
21  *
22  * $FreeBSD$
23  */
24 /*      Copyright (c) 1984, 1986, 1987, 1988, 1989 AT&T */
25 /*        All Rights Reserved   */
26
27
28 /*
29  * Copyright 2004 Sun Microsystems, Inc.  All rights reserved.
30  * Use is subject to license terms.
31  */
32
33 #ifndef _OPENSOLARIS_SYS_SYSMACROS_H_
34 #define _OPENSOLARIS_SYS_SYSMACROS_H_
35
36 #include <sys/param.h>
37
38 #ifdef  __cplusplus
39 extern "C" {
40 #endif
41
42 /*
43  * Macro for checking power of 2 address alignment.
44  */
45 #define IS_P2ALIGNED(v, a) ((((uintptr_t)(v)) & ((uintptr_t)(a) - 1)) == 0)
46
47 /*
48  * Macro to determine if value is a power of 2
49  */
50 #define ISP2(x)         (((x) & ((x) - 1)) == 0)
51
52 /*
53  * Macros for various sorts of alignment and rounding when the alignment
54  * is known to be a power of 2.
55  */
56 #define P2ALIGN(x, align)               ((x) & -(align))
57 #define P2PHASE(x, align)               ((x) & ((align) - 1))
58 #define P2NPHASE(x, align)              (-(x) & ((align) - 1))
59 #define P2ROUNDUP(x, align)             (-(-(x) & -(align)))
60 #define P2END(x, align)                 (-(~(x) & -(align)))
61 #define P2PHASEUP(x, align, phase)      ((phase) - (((phase) - (x)) & -(align)))
62 #define P2CROSS(x, y, align)            (((x) ^ (y)) > (align) - 1)
63 /*
64  * Determine whether two numbers have the same high-order bit.
65  */
66 #define P2SAMEHIGHBIT(x, y)             (((x) ^ (y)) < ((x) & (y)))
67
68 /*
69  * Typed version of the P2* macros.  These macros should be used to ensure
70  * that the result is correctly calculated based on the data type of (x),
71  * which is passed in as the last argument, regardless of the data
72  * type of the alignment.  For example, if (x) is of type uint64_t,
73  * and we want to round it up to a page boundary using "PAGESIZE" as
74  * the alignment, we can do either
75  *      P2ROUNDUP(x, (uint64_t)PAGESIZE)
76  * or
77  *      P2ROUNDUP_TYPED(x, PAGESIZE, uint64_t)
78  */
79 #define P2ALIGN_TYPED(x, align, type)   \
80         ((type)(x) & -(type)(align))
81 #define P2PHASE_TYPED(x, align, type)   \
82         ((type)(x) & ((type)(align) - 1))
83 #define P2NPHASE_TYPED(x, align, type)  \
84         (-(type)(x) & ((type)(align) - 1))
85 #define P2ROUNDUP_TYPED(x, align, type) \
86         (-(-(type)(x) & -(type)(align)))
87 #define P2END_TYPED(x, align, type)     \
88         (-(~(type)(x) & -(type)(align)))
89 #define P2PHASEUP_TYPED(x, align, phase, type)  \
90         ((type)(phase) - (((type)(phase) - (type)(x)) & -(type)(align)))
91 #define P2CROSS_TYPED(x, y, align, type)        \
92         (((type)(x) ^ (type)(y)) > (type)(align) - 1)
93 #define P2SAMEHIGHBIT_TYPED(x, y, type) \
94         (((type)(x) ^ (type)(y)) < ((type)(x) & (type)(y)))
95
96 #ifdef _KERNEL
97 #define memmove(dst, src, size) bcopy((src), (dst), (size))
98 #endif
99
100 /*
101  * Find highest one bit set.
102  *      Returns bit number + 1 of highest bit that is set, otherwise returns 0.
103  * High order bit is 31 (or 63 in _LP64 kernel).
104  */
105 static __inline int
106 highbit(ulong_t i)
107 {
108         register int h = 1;
109
110         if (i == 0)
111                 return (0);
112 #ifdef _LP64
113         if (i & 0xffffffff00000000ul) {
114                 h += 32; i >>= 32;
115         }
116 #endif
117         if (i & 0xffff0000) {
118                 h += 16; i >>= 16;
119         }
120         if (i & 0xff00) {
121                 h += 8; i >>= 8;
122         }
123         if (i & 0xf0) {
124                 h += 4; i >>= 4;
125         }
126         if (i & 0xc) {
127                 h += 2; i >>= 2;
128         }
129         if (i & 0x2) {
130                 h += 1;
131         }
132         return (h);
133 }
134
135 #ifdef  __cplusplus
136 }
137 #endif
138
139 #endif  /* _OPENSOLARIS_SYS_SYSMACROS_H_ */