1 /***************************************************************************
2 * Copyright 1995, Technion, Israel Institute of Technology
3 * Electrical Eng, Software Lab.
4 * Author: Michael Veksler.
5 ***************************************************************************
7 * Purpose : manipulate array of bits
8 * Portability: This is not completely portable, non CISC arcitectures
9 * Might not have atomic Clear/Set/Toggle bit. On those
10 * architectures semaphores should be used.
11 * Big Endian Concerns: This code is big endian compatible,
12 * but the byte order will be different (i.e. bit 0 will be
14 ***************************************************************************
20 ** uncoment the following line to disable assertions,
21 ** this may boost performance by up to 50%
25 #if defined(linux) && !defined(NO_ASM)
33 #include "bit_array.h"
35 #define inline __inline__ /* So we can compile with -ansi */
36 #include <asm/bitops.h>
38 static __inline__
int clear_bit(int bit
, int *mem
);
39 static __inline__
int set_bit(int bit
, int *mem
);
40 #endif /* HAS_BITOPS */
43 #define INT_NR(bit_nr) ((bit_nr) >> INT_LOG2)
44 #define INT_COUNT(bit_count) INT_NR( bit_count + BITS_PER_INT - 1 )
45 #define BIT_IN_INT(bit_nr) ((bit_nr) & (BITS_PER_INT - 1))
47 #if !defined(HAS_BITOPS)
49 /* first_zero maps bytes value to the index of first zero bit */
50 static char first_zero
[256];
51 static int arrays_initialized
=0;
55 ** initialize static arrays used for bit operations speedup.
56 ** Currently initialized: first_zero[256]
57 ** set "arrays_initialized" to inidate that arrays where initialized
60 static void initialize_arrays()
65 for (i
=0 ; i
<256 ; i
++) {
66 /* find the first zero bit in `i' */
67 for (bit
=0 ; bit
< BITS_PER_BYTE
; bit
++)
68 /* break if the bit is zero */
69 if ( ( (1 << bit
) & i
)
78 ** Find first zero bit in the integer.
79 ** Assume there is at least one zero.
81 static __inline__
int find_zbit_in_integer(unsigned int integer
)
85 /* find the zero bit */
86 for (i
=0 ; i
< sizeof(int) ; i
++, integer
>>=8) {
87 int byte
= integer
& 0xff;
90 return ( first_zero
[ byte
]
93 assert(0); /* never reached */
97 /* return -1 on failure */
98 static __inline__
int find_first_zero_bit(unsigned *array
, int bits
)
100 unsigned int integer
;
102 int bytes
=INT_COUNT(bits
);
104 if (!arrays_initialized
)
107 for ( i
=bytes
; i
; i
--, array
++) {
110 /* test if integer contains a zero bit */
112 return ( find_zbit_in_integer(integer
)
113 + ((bytes
-i
) << INT_LOG2
) );
116 /* indicate failure */
120 static __inline__
int test_bit(int pos
, unsigned *array
)
122 unsigned int integer
;
123 int bit
= BIT_IN_INT(pos
);
125 integer
= array
[ pos
>> INT_LOG2
];
127 return ( (integer
& (1 << bit
)) != 0
133 ** The following two functions are x86 specific ,
134 ** other processors will need porting
137 /* inputs: bit number and memory address (32 bit) */
138 /* output: Value of the bit before modification */
139 static __inline__
int clear_bit(int bit
, int *mem
)
146 :"=m" (*mem
), "=&r" (ret
)
151 static __inline__
int set_bit(int bit
, int *mem
)
157 :"=m" (*mem
), "=&r" (ret
)
162 #endif /* !deined(HAS_BITOPS) */
165 /* AssembleArray: assemble an array object using existing data */
166 bit_array
*AssembleArray(bit_array
*new_array
, unsigned int *buff
, int bits
)
168 assert(new_array
!=NULL
);
171 assert((1 << INT_LOG2
) == BITS_PER_INT
); /* if fails, redefine INT_LOG2 */
173 new_array
->bits
=bits
;
174 new_array
->array
=buff
;
178 /* ResetArray: reset the bit array to zeros */
179 int ResetArray(bit_array
*bits
)
185 assert(bits
->array
!=NULL
);
187 for(i
= INT_COUNT(bits
->bits
), p
=bits
->array
; i
; p
++, i
--)
193 /* VacantBit: find a vacant (zero) bit in the array,
194 * Return: Bit index on success, -1 on failure.
196 int VacantBit(bit_array
*bits
)
201 assert(bits
->array
!=NULL
);
203 bit
= find_first_zero_bit(bits
->array
, bits
->bits
);
205 if (bit
>= bits
->bits
) /* failed? */
211 int SampleBit(bit_array
*bits
, int i
)
213 assert(bits
!= NULL
);
214 assert(bits
->array
!= NULL
);
215 assert(i
>= 0 && i
< bits
->bits
);
217 return ( test_bit(i
,bits
->array
) != 0
225 ** Use "compare and exchange" mechanism to make sure
226 ** that bits are not modified while "integer" value
229 ** This may be the slowest technique, but it is the most portable
230 ** (Since most architectures have compare and exchange command)
232 int AssignBit(bit_array
*bits
, int bit_nr
, int val
)
236 assert(bits
!= NULL
);
237 assert(bits
->array
!= NULL
);
238 assert(val
==0 || val
==1);
239 assert(bit_nr
>= 0 && bit_nr
< bits
->bits
);
242 ret
= clear_bit(BIT_IN_INT(bit_nr
), &bits
->array
[ INT_NR(bit_nr
) ]);
244 ret
= set_bit(BIT_IN_INT(bit_nr
), &bits
->array
[ INT_NR(bit_nr
) ]);
246 return ( (ret
!=0) ? 1 : 0);
250 ** Allocate a free bit (==0) and make it used (==1).
251 ** This operation is guaranteed to resemble an atomic instruction.
253 ** Return: allocated bit index, or -1 on failure.
255 ** There is a crack between locating free bit, and allocating it.
256 ** We assign 1 to the bit, test it was not '1' before the assignment.
257 ** If it was, restart the seek and assign cycle.
261 int AllocateBit(bit_array
*bits
)
266 assert(bits
!= NULL
);
267 assert(bits
->array
!= NULL
);
270 bit_nr
= VacantBit(bits
);
272 if (bit_nr
== -1) /* No vacant bit ? */
275 orig_bit
= AssignBit(bits
, bit_nr
, 1);
276 } while (orig_bit
!= 0); /* it got assigned before we tried */
281 #endif /* CONFIG_IPC */