2020-01-19 20:09:11 -05:00
|
|
|
//
|
|
|
|
// LFSR.hpp
|
|
|
|
// Clock Signal
|
|
|
|
//
|
|
|
|
// Created by Thomas Harte on 19/01/2020.
|
|
|
|
// Copyright © 2020 Thomas Harte. All rights reserved.
|
|
|
|
//
|
|
|
|
|
|
|
|
#ifndef LFSR_h
|
|
|
|
#define LFSR_h
|
|
|
|
|
2020-04-25 22:21:10 -04:00
|
|
|
#include <cstdint>
|
|
|
|
#include <cstdlib>
|
|
|
|
|
2021-06-22 19:33:29 -04:00
|
|
|
#include "Sizes.hpp"
|
|
|
|
|
2020-01-19 23:14:35 -05:00
|
|
|
namespace Numeric {
|
|
|
|
|
|
|
|
template <typename IntType> struct LSFRPolynomial {};
|
2020-01-19 20:09:11 -05:00
|
|
|
|
|
|
|
// The following were taken 'at random' from https://users.ece.cmu.edu/~koopman/lfsr/index.html
|
|
|
|
template <> struct LSFRPolynomial<uint64_t> {
|
|
|
|
static constexpr uint64_t value = 0x80000000000019E2;
|
|
|
|
};
|
|
|
|
|
|
|
|
template <> struct LSFRPolynomial<uint32_t> {
|
|
|
|
static constexpr uint32_t value = 0x80000C34;
|
|
|
|
};
|
|
|
|
|
|
|
|
template <> struct LSFRPolynomial<uint16_t> {
|
|
|
|
static constexpr uint16_t value = 0x853E;
|
|
|
|
};
|
|
|
|
|
|
|
|
template <> struct LSFRPolynomial<uint8_t> {
|
|
|
|
static constexpr uint8_t value = 0xAF;
|
|
|
|
};
|
|
|
|
|
|
|
|
/*!
|
|
|
|
Provides a linear-feedback shift register with a random initial state; if no polynomial is supplied
|
|
|
|
then one will be picked that is guaranteed to give the maximal number of LFSR states that can fit
|
|
|
|
in the specified int type.
|
|
|
|
*/
|
|
|
|
template <typename IntType = uint64_t, IntType polynomial = LSFRPolynomial<IntType>::value> class LFSR {
|
|
|
|
public:
|
2020-04-05 22:58:09 -04:00
|
|
|
/*!
|
|
|
|
Constructs an LFSR with a random initial value.
|
|
|
|
*/
|
2020-05-20 23:34:26 -04:00
|
|
|
constexpr LFSR() noexcept {
|
2020-04-25 22:21:10 -04:00
|
|
|
// Randomise the value, ensuring it doesn't end up being 0;
|
|
|
|
// don't set any top bits, in case this is a signed type.
|
2020-01-19 20:09:11 -05:00
|
|
|
while(!value_) {
|
|
|
|
uint8_t *value_byte = reinterpret_cast<uint8_t *>(&value_);
|
|
|
|
for(size_t c = 0; c < sizeof(IntType); ++c) {
|
2020-04-25 22:21:10 -04:00
|
|
|
*value_byte = uint8_t(uint64_t(rand()) * 127 / RAND_MAX);
|
2020-01-19 20:09:11 -05:00
|
|
|
++value_byte;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-05 22:58:09 -04:00
|
|
|
/*!
|
|
|
|
Constructs an LFSR with the specified initial value.
|
|
|
|
|
|
|
|
An initial value of 0 is invalid.
|
|
|
|
*/
|
|
|
|
LFSR(IntType initial_value) : value_(initial_value) {}
|
|
|
|
|
2020-01-19 20:32:58 -05:00
|
|
|
/*!
|
|
|
|
Advances the LSFR, returning either an @c IntType of value @c 1 or @c 0,
|
|
|
|
determining the bit that was just shifted out.
|
|
|
|
*/
|
2020-01-19 20:09:11 -05:00
|
|
|
IntType next() {
|
2021-06-26 23:39:37 -04:00
|
|
|
const auto result = IntType(value_ & 1);
|
|
|
|
value_ = IntType((value_ >> 1) ^ (result * polynomial));
|
2020-01-19 20:09:11 -05:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
IntType value_ = 0;
|
|
|
|
};
|
|
|
|
|
2021-06-22 20:50:03 -04:00
|
|
|
template <uint64_t polynomial> class LFSRv: public LFSR<typename MinIntTypeValue<polynomial>::type, polynomial> {};
|
2021-06-22 19:33:29 -04:00
|
|
|
|
2020-01-19 23:14:35 -05:00
|
|
|
}
|
|
|
|
|
2020-01-19 20:09:11 -05:00
|
|
|
#endif /* LFSR_h */
|