Here is the code that reports the bit parity
of a given integer:
01: bool parity(unsigned int x)
02: {
03: x ^= x >> 16;
04: x ^= x >> 8;
05: x ^= x >> 4;
06: x &= 0x0F;
07: return ((0x6996 >> x) & 1) != 0;
08: }
I found this here.. while there seems to be explanation in the link, I do not understand.
The first explanation that start with The code first "merges" bits 0 − 15 with bits 16 − 31 using a right shift and XOR (line 3).
is making it hard for me to understand as to what is going on. I tried to play around them but that did not help. if a clarity on how this work is given, it will be useful for beginners like me
Thanks
EDIT: from post below:
value : 1101 1110 1010 1101 1011 1110 1110 1111
value >> 16: 0000 0000 0000 0000 1101 1110 1010 1101
----------------------------------------------------
xor : 1101 1110 1010 1101 0110 0001 0100 0010
now right shift this again by 8 bits:
value : 1101 1110 1010 1101 0110 0001 0100 0010
value >>8 : 0000 0000 1101 1110 1010 1101 0110 0001
----------------------------------------------------
xor : 1101 1110 1110 0001 0100 1100 0010 0011
so where is the merging of parity happening here?
Let's start first with a 2-bit example so you can see what's going on. The four possibilities are:
You can see that
a^b (xor)
gives 0 for an even number of one-bits and 1 for an odd number. This woks for 3-bit values as well:The same trick is being used in lines 3 through 6 to merge all 32 bits into a single 4-bit value. Line 3 merges
b31-16
withb15-0
to give a 16-bit value, then line 4 merges the resultantb15-b8
withb7-b0
, then line 5 merges the resultantb7-b4
withb3-b0
. Sinceb31-b4
(the upper half of each xor operation) aren't cleared by that operations, line 6 takes care of that by clearing them out (anding with binary0000...1111
to clear all but the lower 4 bits).The merging here is achieved in a chunking mode. By "chunking", I mean that it treats the value in reducing chunks rather than as individual bits, which allows it to efficiently reduce the value to a 4-bit size (it can do this because the xor operation is both associative and commutative). The alternative would be to perform seven xor operations on the nybbles rather than three. Or, in complexity analysis terms, O(log n) instead of O(n).
Say you have the value
0xdeadbeef
, which is binary1101 1110 1010 1101 1011 1110 1110 1111
. The merging happens thus:(with the irrelevant bits, those which will not be used in future, left as
.
characters).For the complete operation:
And, looking up
0011
in the table below, we see that it gives even parity (there are 24 1-bits in the original value). Changing just one bit in that original value (any bit, I've chosen the righmost bit) will result in the opposite case:And
0010
in the below table is odd parity.The only "magic" there is the
0x6996
value which is shifted by the four-bit value to ensure the lower bit is set appropriately, then that bit is used to decide the parity. The reason0x6996
(binary0110 1001 1001 0110
) is used is because of the nature of parity for binary values as shown in the lined page:Note that it's not necessary to do the final shift-of-a-constant. You could just as easily continue the merging operations until you get down to a single bit, then use that bit:
However, once you have the value
0...15
, a shift of a constant by that value is likely to be faster than two extra shift-and-xor operations.