I have a long sequence of bits stored in an array of unsigned long integers, like this
struct bit_array
{
int size; /* nr of bits */
unsigned long *array; /* the container that stores bits */
}
I am trying to design an algorithm to reverse the order of bits in *array. Problems:
size
can be anything, i.e. not necessarily a multiple of 8 or 32 etc, so the first bit in the input array can end up at any position within the unsigned long in the output array;- the algorithm should be platform-independent, i.e. work for any
sizeof(unsigned long)
.
Code, pseudocode, algo description etc. -- anything better than bruteforce ("bit by bit") approach is welcome.
You must define what is the order of bits in an
unsigned long
. You might assume that bit n is corresponds toarray[x] & (1 << n)
but this needs to be specified. If so, you need to handle the byte ordering (little or big endian) if you are going to use access the array as bytes instead of unsigned long.I would definitely implement brute force first and measure whether the speed is an issue. No need to waste time trying to optimize this if it is not used a lot on large arrays. An optimized version can be tricky to implement correctly. If you end up trying anyway, the brute force version can be used to verify correctness on test values and benchmark the speed of the optimized version.
In a collection of related topics which can be found here, the bits of an individual array entry could be reversed as follows.
The reversal of the entire array could be done afterwards by rearranging the individual positions.
I like the idea of lookup table. Still it's also a typical task for log(n) group bit tricks that may be very fast. Like:
The underlying idea is that when we aim to reverse the order of some sequence we may swap the head and tail halves of this sequence and then separately reverse each of halves (which is done here by applying the same procedure recursively to each half).
Here is a more portable version supporting
unsigned long
widths of 4,8,16 or 32 bytes.My favorite solution is to fill a lookup-table that does bit-reversal on a single byte (hence 256 byte entries).
You apply the table to 1 to 4 bytes of the input operand, with a swap. If the size isn't a multiple of 8, you will need to adjust by a final right shift.
This scales well to larger integers.
Example:
To split the number into bytes portably, you need to use bitwise masking/shifts; mapping of a struct or array of bytes onto the integer can make it more efficient.
For brute performance, you can think of mapping up to 16 bits at a time, but this doesn't look quite reasonable.
I would split the problem into two parts.
First, I would ignore the fact that the number of used bits is not a multiple of 32. I would use one of the given methods to swap around the whole array like that.
pseudocode:
and then fix up the fact that the first few bits (call than number
n
) are actually garbage bits from the end of the longs:You could try to fold that into one pass over the whole array of course. Something like this:
I'm abstracting from the edge cases here (first and last longword), and you may need to reverse the shifting direction depending on how the bits are ordered inside each longword.
The fact that the size is not multiple of
sizeof(long)
is the hardest part of the problem. This can result in a lot of bit shifting.But, you don't have to do that if you can introduce new struct member:
Offset would tell you how many bits to ignore at the beginning of the array.
Then you only only have to do following steps: