/*
Copyright (C) 2005, 2006, 2007, 2008, 2009, 2010, 2011 Her Majesty
the Queen in Right of Canada (Communications Research Center Canada)
*/
/*
This file is part of ODR-DabMod.
ODR-DabMod is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as
published by the Free Software Foundation, either version 3 of the
License, or (at your option) any later version.
ODR-DabMod is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with ODR-DabMod. If not, see .
*/
#include "PrbsGenerator.h"
#include "PcDebug.h"
#include
#include
#include
PrbsGenerator::PrbsGenerator(size_t framesize, uint32_t polynomial,
uint32_t accum, size_t init) :
ModCodec(),
d_framesize(framesize),
d_polynomial(polynomial),
d_accum(accum),
d_accum_init(accum),
d_init(init)
{
PDEBUG("PrbsGenerator::PrbsGenerator(%zu, %u, %u, %zu) @ %p\n",
framesize, polynomial, accum, init, this);
gen_prbs_table();
gen_weight_table();
}
PrbsGenerator::~PrbsGenerator()
{
PDEBUG("PrbsGenerator::~PrbsGenerator() @ %p\n", this);
}
/*
* Generate a table of matrix products to update a 32-bit PRBS generator.
*/
void PrbsGenerator::gen_prbs_table()
{
int i;
for (i = 0; i < 4; ++i) {
int j;
for (j = 0; j < 256; ++j) {
uint32_t prbs_accum = ((uint32_t)j << (i * 8));
int k;
for (k = 0; k < 8; ++k) {
prbs_accum = (prbs_accum << 1)
^ parity_check(prbs_accum & d_polynomial);
}
d_prbs_table[i][j] = (prbs_accum & 0xff);
}
}
}
/*
* Generate the weight table.
*/
void PrbsGenerator::gen_weight_table()
{
int i;
for (i = 0; i < 256; ++i) {
unsigned char mask=1U, ones_count = 0U;
int j;
for (j = 0; j < 8; ++j) {
ones_count += ((i & mask) != 0U);
mask = mask << 1;
}
d_weight[i] = ones_count;
}
}
/*
* Generate a parity check for a 32-bit word.
*/
uint32_t PrbsGenerator::parity_check(uint32_t prbs_accum)
{
uint32_t mask=1UL, parity=0UL;
int i;
for (i = 0; i < 32; ++i) {
parity ^= ((prbs_accum & mask) != 0UL);
mask <<= 1;
}
return parity;
}
/*
* Update a 32-bit PRBS generator eight bits at a time.
*/
uint32_t PrbsGenerator::update_prbs()
{
unsigned char acc_lsb = 0;
int i;
for (i = 0; i < 4; ++i) {
// PDEBUG("0x%x = 0x%x ^ 0x%x\n",
// acc_lsb ^ d_prbs_table [i][(d_accum >> (i * 8)) & 0xff],
// acc_lsb, d_prbs_table [i][(d_accum >> (i * 8)) & 0xff]);
acc_lsb ^= d_prbs_table[i][(d_accum >> (i * 8)) & 0xff];
}
return (d_accum << 8) ^ ((uint32_t)acc_lsb);
}
int PrbsGenerator::process(Buffer* const dataIn, Buffer* dataOut)
{
PDEBUG("PrbsGenerator::process(dataIn: %p, dataOut: %p)\n",
dataIn, dataOut);
dataOut->setLength(d_framesize);
unsigned char* out = reinterpret_cast(dataOut->getData());
// Initialization
if (d_accum_init) {
d_accum = d_accum_init;
} else {
d_accum = 0;
while (d_accum < d_polynomial) {
d_accum <<= 1;
d_accum |= 1;
}
}
//PDEBUG("Polynomial: 0x%x\n", d_polynomial);
//PDEBUG("Init accum: 0x%x\n", d_accum);
size_t i = 0;
while (i < d_init) {
out[i++] = 0xff;
}
for (; i < d_framesize; ++i) {
// Writting data
d_accum = update_prbs();
if ((d_accum_init == 0xa9) && (i % 188 == 0)) { // DVB energy dispersal
out[i] = 0;
} else {
out[i] = (unsigned char)(d_accum & 0xff);
}
//PDEBUG("accum: 0x%x\n", d_accum);
}
if ((dataIn != NULL) && dataIn->getLength()) {
PDEBUG(" mixing input\n");
const unsigned char* in = reinterpret_cast(dataIn->getData());
if (dataIn->getLength() != dataOut->getLength()) {
PDEBUG("%zu != %zu\n", dataIn->getLength(), dataOut->getLength());
throw std::runtime_error("PrbsGenerator::process "
"input size is not equal to output size!\n");
}
for (size_t i = 0; i < dataOut->getLength(); ++i) {
out[i] ^= in[i];
}
}
return dataOut->getLength();
}