I want to implement a Linear-feedback shift register for the following polynomial x^24 + x^23 + x^22 + x^20 + x^19 + x^18 + x^17 + x^16 + x^15 + x^13 + x^12 + x^8 + x^7 + x^6 + 1
, relying on what can be found here with the associated C code:
https://en.wikipedia.org/wiki/Linear-feedback_shift_register#Fibonacci_LFSRs
I have used uint32_t for all my data types and lfsr is set to start_state
when the function is called. The central lines in the function are:
bit = ((lfsr >> 7) ^ (lfsr >> 8) ^ (lfsr >> 9) ^ (lfsr >> 11) ^ (lfsr >> 12) ^ (lfsr >> 13) ^ (lfsr >> 14) ^ (lfsr >> 15) ^ (lfsr >> 16) ^ (lfsr >> 18) ^ (lfsr >> 19) ^ (lfsr >> 23) ^ (lfsr >> 24) ^ (lfsr >> 25))& 1u;
lfsr = (lfsr >> 1) | (bit << 31);
But this can't fit, because I have as condition do{...} while (lfsr != start_state );
so that correspondingly after about 17 million runs the function should be finished, but it just keeps running permanently, which is why my mapping of the polynomial to the 32 bit sequence can't fit. What am I doing wrong here?
I added the minimal example.
void lfsr_fib(uint32_t lfsr)
{
uint32_t bit;
uint32_t compareNr = 0;
do
{
/*
polynomial: x^24 + x^23 + x^22 + x^20 + x^19 + x^18 + x^17 + x^16 + x^15 + x^13 + x^12 + x^8 + x^7 + x^6 + 1
*/
bit = ((lfsr >> 7) ^ (lfsr >> 8) ^ (lfsr >> 9) ^ (lfsr >> 11) ^ (lfsr >> 12) ^ (lfsr >> 13) ^ (lfsr >> 14) ^ (lfsr >> 15) ^ (lfsr >> 16) ^ (lfsr >> 18) ^ (lfsr >> 19) ^ (lfsr >> 23) ^ (lfsr >> 24) ^ (lfsr >> 25))& 1u;
lfsr = (lfsr >> 1) | (bit << 31);
++period;
printf("%d\n", period);
} while (lfsr != start_state );
}
Update:
My teacher announced that he made a mistake. The start_value has 32 bit which he actually wanted to be 24 bit.
Now he gave us the tip to use the 32 bit start value and replace lfsr = (lfsr >> 1) | (bit << 31);
by lfsr = (lfsr >> 1) | (bit << 24);
.
I did the polynomial in your comment, but it did not work (seemed to loop infinitely).
I looked at the wikipedia page, and changed the rightmost term from
+ 1
to+ lfsr
per the wiki.That did work (I'm guessing it's correct because it stopped).
Here is the restructured code:
In the code above, I've used
cpp
conditionals to denote old vs. new code:Note: this can be cleaned up by running the file through
unifdef -k
Here is the program output:
UPDATE:
A single bit add is the same as xor (xor is how ALUs do addition). So, they are the same in this context.
I'm not sure how you get 16777215 as the period. The period changes a bit, based on the initial value. The code below produces two different periods: 7912905 and 15825810
Here is an enhanced version of the code that allows more experimentation:
Compiled with
-DY=3
(others fail), and run with./test -R -C100
, here is the program output: