mirror of
https://github.com/f4exb/sdrangel.git
synced 2025-04-06 11:39:02 -04:00
SDRDaemon: make Rx side truly 24/16 bit compatible in all configurations
This commit is contained in:
parent
22746ff813
commit
404c73fb80
@ -52,7 +52,7 @@ DaemonSink::DaemonSink(DeviceSourceAPI *deviceAPI) :
|
||||
m_dataBlock(0),
|
||||
m_centerFrequency(0),
|
||||
m_sampleRate(48000),
|
||||
m_sampleBytes(SDR_RX_SAMP_SZ == 24 ? 4 : 2),
|
||||
m_sampleBytes(SDR_RX_SAMP_SZ <= 16 ? 2 : 4),
|
||||
m_nbBlocksFEC(0),
|
||||
m_txDelay(35),
|
||||
m_dataAddress("127.0.0.1"),
|
||||
@ -82,7 +82,7 @@ DaemonSink::~DaemonSink()
|
||||
void DaemonSink::setTxDelay(int txDelay, int nbBlocksFEC)
|
||||
{
|
||||
double txDelayRatio = txDelay / 100.0;
|
||||
double delay = m_sampleRate == 0 ? 1.0 : (127*127*txDelayRatio) / m_sampleRate;
|
||||
double delay = m_sampleRate == 0 ? 1.0 : (127*SDRDaemonSamplesPerBlock*txDelayRatio) / m_sampleRate;
|
||||
delay /= 128 + nbBlocksFEC;
|
||||
m_txDelay = roundf(delay*1e6); // microseconds
|
||||
qDebug() << "DaemonSink::setTxDelay:"
|
||||
|
@ -287,7 +287,7 @@ void DaemonSinkGUI::on_nbFECBlocks_valueChanged(int value)
|
||||
void DaemonSinkGUI::updateTxDelayTime()
|
||||
{
|
||||
double txDelayRatio = m_settings.m_txDelay / 100.0;
|
||||
double delay = m_sampleRate == 0 ? 0.0 : (127*127*txDelayRatio) / m_sampleRate;
|
||||
double delay = m_sampleRate == 0 ? 0.0 : (127*SDRDaemonSamplesPerBlock*txDelayRatio) / m_sampleRate;
|
||||
delay /= 128 + m_settings.m_nbFECBlocks;
|
||||
ui->txDelayTime->setText(tr("%1µs").arg(QString::number(delay*1e6, 'f', 0)));
|
||||
}
|
||||
|
@ -50,9 +50,9 @@ The value is a percentage of the nominal time it takes to process a block of sam
|
||||
- Sample rate on the network: _SR_
|
||||
- Delay percentage: _d_
|
||||
- Number of FEC blocks: _F_
|
||||
- There are 127 blocks of I/Q data per frame (1 meta block for 128 blocks) and each I/Q data block of 512 bytes (128 samples) has a 4 bytes header (1 sample) thus there are 127 samples remaining effectively. This gives the constant 127*127 = 16219 samples per frame in the formula
|
||||
- There are 127 blocks of I/Q data per frame (1 meta block for 128 blocks) and each I/Q data block of 512 bytes (128 samples) has a 8 bytes header (2 samples) thus there are 126 samples remaining effectively. This gives the constant 127*126 = 16002 samples per frame in the formula
|
||||
|
||||
Formula: ((127 ✕ 127 ✕ _d_) / _SR_) / (128 + _F_)
|
||||
Formula: ((127 ✕ 126 ✕ _d_) / _SR_) / (128 + _F_)
|
||||
|
||||
<h3>6: Forward Error Correction setting and status</h3>
|
||||
|
||||
@ -64,7 +64,7 @@ This sets the number of FEC blocks per frame. A frame consists of 128 data block
|
||||
|
||||
<h4>6.2: Distant transmitter queue length</h4>
|
||||
|
||||
This is the samples queue length reported from the distant transmitter. This is a number of vectors of 127 ✕ 127 ✕ _I_ samples where _I_ is the interpolation factor. This corresponds to a block of 127 ✕ 127 samples sent over the network. This numbers serves to throttle the sample generator so that the queue length is close to 8 vectors.
|
||||
This is the samples queue length reported from the distant transmitter. This is a number of vectors of 127 ✕ 127 ✕ _I_ samples where _I_ is the interpolation factor. This corresponds to a block of 127 ✕ 126 samples sent over the network. This numbers serves to throttle the sample generator so that the queue length is close to 8 vectors.
|
||||
|
||||
<h4>6.3: Stream status</h4>
|
||||
|
||||
|
@ -214,7 +214,7 @@ void SDRdaemonSinkGui::updateSampleRate()
|
||||
|
||||
void SDRdaemonSinkGui::updateTxDelayTooltip()
|
||||
{
|
||||
double delay = ((127*127*m_settings.m_txDelay) / m_settings.m_sampleRate)/(128 + m_settings.m_nbFECBlocks);
|
||||
double delay = ((127*126*m_settings.m_txDelay) / m_settings.m_sampleRate)/(128 + m_settings.m_nbFECBlocks);
|
||||
ui->txDelayText->setToolTip(tr("%1 us").arg(QString::number(delay*1e6, 'f', 0)));
|
||||
}
|
||||
|
||||
|
@ -96,7 +96,7 @@ bool SDRdaemonSinkOutput::start()
|
||||
m_lastTimestampRateCorrection = 0;
|
||||
m_lastQueueLength = -2; // set first value out of bounds
|
||||
|
||||
double delay = ((127*127*m_settings.m_txDelay) / m_settings.m_sampleRate)/(128 + m_settings.m_nbFECBlocks);
|
||||
double delay = ((127*126*m_settings.m_txDelay) / m_settings.m_sampleRate)/(128 + m_settings.m_nbFECBlocks);
|
||||
m_sdrDaemonSinkThread->setTxDelay((int) (delay*1e6));
|
||||
|
||||
mutexLocker.unlock();
|
||||
@ -278,13 +278,13 @@ void SDRdaemonSinkOutput::applySettings(const SDRdaemonSinkSettings& settings, b
|
||||
|
||||
if (changeTxDelay)
|
||||
{
|
||||
double delay = ((127*127*settings.m_txDelay) / settings.m_sampleRate)/(128 + settings.m_nbFECBlocks);
|
||||
double delay = ((127*126*settings.m_txDelay) / settings.m_sampleRate)/(128 + settings.m_nbFECBlocks);
|
||||
qDebug("SDRdaemonSinkOutput::applySettings: Tx delay: %f us", delay*1e6);
|
||||
|
||||
if (m_sdrDaemonSinkThread != 0)
|
||||
{
|
||||
// delay is calculated as a fraction of the nominal UDP block process time
|
||||
// frame size: 127 * 127 samples
|
||||
// frame size: 127 * 126 samples
|
||||
// divided by sample rate gives the frame process time
|
||||
// divided by the number of actual blocks including FEC blocks gives the block (i.e. UDP block) process time
|
||||
m_sdrDaemonSinkThread->setTxDelay((int) (delay*1e6));
|
||||
|
@ -29,7 +29,7 @@ MESSAGE_CLASS_DEFINITION(UDPSinkFECWorker::MsgConfigureRemoteAddress, Message)
|
||||
|
||||
UDPSinkFEC::UDPSinkFEC() :
|
||||
m_sampleRate(48000),
|
||||
m_sampleBytes(SDR_TX_SAMP_SZ == 24 ? 4 : 2),
|
||||
m_sampleBytes(SDR_TX_SAMP_SZ <= 16 ? 2 : 4),
|
||||
m_sampleBits(SDR_TX_SAMP_SZ),
|
||||
m_nbSamples(0),
|
||||
m_nbBlocksFEC(0),
|
||||
@ -101,7 +101,6 @@ void UDPSinkFEC::write(const SampleVector::iterator& begin, uint32_t sampleChunk
|
||||
|
||||
gettimeofday(&tv, 0);
|
||||
|
||||
// create meta data TODO: semaphore
|
||||
metaData.m_centerFrequency = 0; // frequency not set by stream
|
||||
metaData.m_sampleRate = m_sampleRate;
|
||||
metaData.m_sampleBytes = m_sampleBytes & 0xF;
|
||||
@ -142,6 +141,7 @@ void UDPSinkFEC::write(const SampleVector::iterator& begin, uint32_t sampleChunk
|
||||
m_txBlockIndex = 1; // next Tx block with data
|
||||
}
|
||||
|
||||
// TODO: memcpy is valid for 4 bytes samples only (16 bits) else conversion must take place to take only LSB assuming Tx is 16 bit only
|
||||
if (m_sampleIndex + inRemainingSamples < samplesPerBlock) // there is still room in the current super block
|
||||
{
|
||||
memcpy((char *) &m_superBlock.protectedBlock.m_samples[m_sampleIndex],
|
||||
@ -167,7 +167,6 @@ void UDPSinkFEC::write(const SampleVector::iterator& begin, uint32_t sampleChunk
|
||||
int nbBlocksFEC = m_nbBlocksFEC;
|
||||
int txDelay = m_txDelay;
|
||||
|
||||
// TODO: send blocks
|
||||
//qDebug("UDPSinkFEC::write: push frame to worker: %u", m_frameCount);
|
||||
m_udpWorker->pushTxFrame(m_txBlocks[m_txBlocksIndex], nbBlocksFEC, txDelay, m_frameCount);
|
||||
//m_txThread = new std::thread(transmitUDP, this, m_txBlocks[m_txBlocksIndex], m_frameCount, nbBlocksFEC, txDelay, m_cm256Valid);
|
||||
|
@ -72,9 +72,10 @@ public:
|
||||
uint16_t frameIndex;
|
||||
uint8_t blockIndex;
|
||||
uint8_t filler;
|
||||
uint32_t filler2;
|
||||
};
|
||||
|
||||
static const int samplesPerBlock = (m_udpSize - sizeof(Header)) / sizeof(Sample);
|
||||
static const int samplesPerBlock = (m_udpSize - sizeof(Header)) / (2 * SDR_TX_SAMP_SZ);
|
||||
|
||||
struct ProtectedBlock
|
||||
{
|
||||
|
@ -159,9 +159,10 @@ The value is a percentage of the nominal time it takes to process a block of sam
|
||||
- Sample rate on the network: _SR_
|
||||
- Delay percentage: _d_
|
||||
- Number of FEC blocks: _F_
|
||||
- There are 127 blocks of I/Q data per frame (1 meta block for 128 blocks) and each I/Q data block of 512 bytes (128 samples) has a 4 bytes header (1 sample) thus there are 127 samples remaining effectively. This gives the constant 127*127 = 16219 samples per frame in the formula
|
||||
- There are 127 blocks of I/Q data per frame (1 meta block for 128 blocks) and each I/Q data block of 512 bytes (128 samples) has a 8 bytes header (1 sample for 24 bit and 2 samples for 16 bit) thus there are 126 (16 bit) or 63 (24 bit) samples remaining effectively.
|
||||
|
||||
Formula: ((127 ✕ 127 ✕ _d_) / _SR_) / (128 + _F_)
|
||||
Formula (16 bit): ((127 ✕ 126 ✕ _d_) / _SR_) / (128 + _F_)
|
||||
Formula (24 bit): ((127 ✕ 63 ✕ _d_) / _SR_) / (128 + _F_) thus half the above
|
||||
|
||||
<h3>8: Desired distant device sample rate</h3>
|
||||
|
||||
|
@ -62,6 +62,7 @@ public:
|
||||
uint16_t frameIndex;
|
||||
uint8_t blockIndex;
|
||||
uint8_t filler;
|
||||
uint32_t filler2;
|
||||
};
|
||||
|
||||
static const int framesSize = SDRDAEMONSOURCE_NBDECODERSLOTS * (SDRDAEMONSOURCE_NBORIGINALBLOCKS - 1) * (SDRDAEMONSOURCE_UDPSIZE - sizeof(Header));
|
||||
|
@ -56,7 +56,7 @@ SDRdaemonSourceGui::SDRdaemonSourceGui(DeviceUISet *deviceUISet, QWidget* parent
|
||||
m_bufferGauge(-50),
|
||||
m_nbOriginalBlocks(128),
|
||||
m_nbFECBlocks(0),
|
||||
m_sampleBits(16),
|
||||
m_sampleBits(16), // assume 16 bits to start with
|
||||
m_sampleBytes(2),
|
||||
m_samplesCount(0),
|
||||
m_tickCount(0),
|
||||
|
@ -264,7 +264,7 @@ void SDRdaemonSourceUDPHandler::tick()
|
||||
}
|
||||
|
||||
const SDRdaemonSourceBuffer::MetaDataFEC& metaData = m_sdrDaemonBuffer.getCurrentMeta();
|
||||
m_readLength = m_readLengthSamples * metaData.m_sampleBytes * 2;
|
||||
m_readLength = m_readLengthSamples * (metaData.m_sampleBytes & 0xF) * 2;
|
||||
|
||||
if (SDR_RX_SAMP_SZ == metaData.m_sampleBits) // same sample size
|
||||
{
|
||||
@ -304,9 +304,9 @@ void SDRdaemonSourceUDPHandler::tick()
|
||||
|
||||
for (unsigned int is = 0; is < m_readLengthSamples; is++)
|
||||
{
|
||||
m_converterBuffer[is] = ((int32_t *)buf)[2*is]>>8; // I -> MSB
|
||||
m_converterBuffer[is] = ((int32_t *)buf)[2*is+1]>>8; // Q -> MSB
|
||||
m_converterBuffer[is] <<=16;
|
||||
m_converterBuffer[is] += ((int32_t *)buf)[2*is+1]>>8; // Q -> LSB
|
||||
m_converterBuffer[is] += ((int32_t *)buf)[2*is]>>8; // I -> LSB
|
||||
}
|
||||
|
||||
m_sampleFifo->write(reinterpret_cast<quint8*>(m_converterBuffer), m_readLengthSamples*sizeof(Sample));
|
||||
|
@ -70,18 +70,20 @@ struct SDRDaemonHeader
|
||||
uint16_t m_frameIndex;
|
||||
uint8_t m_blockIndex;
|
||||
uint8_t m_filler;
|
||||
uint32_t m_filler2;
|
||||
|
||||
void init()
|
||||
{
|
||||
m_frameIndex = 0;
|
||||
m_blockIndex = 0;
|
||||
m_filler = 0;
|
||||
m_filler2 = 0;
|
||||
}
|
||||
};
|
||||
|
||||
static const int SDRDaemonUdpSize = UDPSINKFEC_UDPSIZE;
|
||||
static const int SDRDaemonNbOrginalBlocks = UDPSINKFEC_NBORIGINALBLOCKS;
|
||||
static const int SDRDaemonSamplesPerBlock = (UDPSINKFEC_UDPSIZE - sizeof(SDRDaemonHeader)) / (SDR_RX_SAMP_SZ/4);
|
||||
static const int SDRDaemonSamplesPerBlock = (UDPSINKFEC_UDPSIZE - sizeof(SDRDaemonHeader)) / sizeof(Sample);
|
||||
|
||||
struct SDRDaemonProtectedBlock
|
||||
{
|
||||
|
Loading…
Reference in New Issue
Block a user