mirror of
https://github.com/AlexandreRouma/SDRPlusPlus.git
synced 2025-11-05 03:09:11 +01:00
even more stuff
This commit is contained in:
9
core/src/dsp/compression/pcm_type.h
Normal file
9
core/src/dsp/compression/pcm_type.h
Normal file
@@ -0,0 +1,9 @@
|
||||
#pragma once
|
||||
|
||||
namespace dsp::compression {
|
||||
enum PCMType {
|
||||
PCM_TYPE_I8,
|
||||
PCM_TYPE_I16,
|
||||
PCM_TYPE_F32
|
||||
};
|
||||
}
|
||||
79
core/src/dsp/compression/sample_stream_compressor.h
Normal file
79
core/src/dsp/compression/sample_stream_compressor.h
Normal file
@@ -0,0 +1,79 @@
|
||||
#pragma once
|
||||
#include "../processor.h"
|
||||
#include "pcm_type.h"
|
||||
|
||||
namespace dsp::compression {
|
||||
class SampleStreamCompressor : public Processor<complex_t, uint8_t> {
|
||||
using base_type = Processor<complex_t, uint8_t>;
|
||||
public:
|
||||
SampleStreamCompressor() {}
|
||||
|
||||
SampleStreamCompressor(stream<complex_t>* in, PCMType pcmType) { init(in, pcmType); }
|
||||
|
||||
void init(stream<complex_t>* in, PCMType pcmType) {
|
||||
_pcmType = pcmType;
|
||||
base_type::init(in);
|
||||
}
|
||||
|
||||
void setPCMType(PCMType pcmType) {
|
||||
assert(base_type::_block_init);
|
||||
std::lock_guard<std::recursive_mutex> lck(base_type::ctrlMtx);
|
||||
base_type::tempStop();
|
||||
_pcmType = pcmType;
|
||||
base_type::tempStart();
|
||||
}
|
||||
|
||||
inline static int process(int count, PCMType pcmType, const complex_t* in, uint8_t* out) {
|
||||
uint16_t* compressionType = (uint16_t*)out;
|
||||
uint16_t* sampleType = (uint16_t*)&out[2];
|
||||
float* scaler = (float*)&out[4];
|
||||
void* dataBuf = &out[8];
|
||||
|
||||
// Write options and leave blank space for compression
|
||||
*compressionType = 0;
|
||||
*sampleType = pcmType;
|
||||
|
||||
// If type is float32, no compression is needed
|
||||
if (pcmType == PCMType::PCM_TYPE_F32) {
|
||||
*scaler = 0;
|
||||
memcpy(dataBuf, in, count * sizeof(complex_t));
|
||||
return count;
|
||||
}
|
||||
|
||||
// Find maximum value
|
||||
uint32_t maxIdx;
|
||||
volk_32f_index_max_32u(&maxIdx, (float*)in, count * 2);
|
||||
float maxVal = ((float*)in)[maxIdx];
|
||||
*scaler = maxVal;
|
||||
|
||||
// Convert to the right type and send it out (sign bit determines pcm type)
|
||||
if (pcmType == PCMType::PCM_TYPE_I8) {
|
||||
volk_32f_s32f_convert_8i((int8_t*)dataBuf, (float*)in, 128.0f / maxVal, count * 2);
|
||||
return 8 + (count * sizeof(int8_t) * 2);
|
||||
}
|
||||
else if (pcmType == PCMType::PCM_TYPE_I16) {
|
||||
volk_32f_s32f_convert_16i((int16_t*)dataBuf, (float*)in, 32768.0f / maxVal, count * 2);
|
||||
return 8 + (count * sizeof(int16_t) * 2);
|
||||
}
|
||||
|
||||
return count;
|
||||
}
|
||||
|
||||
int run() {
|
||||
int count = base_type::_in->read();
|
||||
if (count < 0) { return -1; }
|
||||
|
||||
int outCount = process(count, _pcmType, base_type::_in->readBuf, base_type::out.writeBuf);
|
||||
|
||||
// Swap if some data was generated
|
||||
base_type::_in->flush();
|
||||
if (outCount) {
|
||||
if (!base_type::out.swap(outCount)) { return -1; }
|
||||
}
|
||||
return outCount;
|
||||
}
|
||||
|
||||
protected:
|
||||
PCMType _pcmType;
|
||||
};
|
||||
}
|
||||
50
core/src/dsp/compression/sample_stream_decompressor.h
Normal file
50
core/src/dsp/compression/sample_stream_decompressor.h
Normal file
@@ -0,0 +1,50 @@
|
||||
#pragma once
|
||||
#include "../processor.h"
|
||||
#include "pcm_type.h"
|
||||
|
||||
namespace dsp::compression {
|
||||
class SampleStreamDecompressor : public Processor<uint8_t, complex_t> {
|
||||
using base_type = Processor<uint8_t, complex_t>;
|
||||
public:
|
||||
SampleStreamDecompressor() {}
|
||||
|
||||
SampleStreamDecompressor(stream<uint8_t>* in) { base_type::init(in); }
|
||||
|
||||
inline int process(int count, const uint8_t* in, complex_t* out) {
|
||||
uint16_t sampleType = *(uint16_t*)&in[2];
|
||||
float scaler = *(float*)&in[4];
|
||||
const void* dataBuf = &in[8];
|
||||
|
||||
if (sampleType == PCMType::PCM_TYPE_F32) {
|
||||
memcpy(out, dataBuf, count - 8);
|
||||
return (count - 8) / sizeof(complex_t);
|
||||
}
|
||||
else if (sampleType == PCMType::PCM_TYPE_I16) {
|
||||
int outCount = (count - 8) / (sizeof(int16_t) * 2);
|
||||
volk_16i_s32f_convert_32f((float*)out, (int16_t*)dataBuf, 32768.0f / scaler, outCount * 2);
|
||||
return outCount;
|
||||
}
|
||||
else if (sampleType == PCMType::PCM_TYPE_I8) {
|
||||
int outCount = (count - 8) / (sizeof(int8_t) * 2);
|
||||
volk_8i_s32f_convert_32f((float*)out, (int8_t*)dataBuf, 128.0f / scaler, outCount * 2);
|
||||
return outCount;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
int run() {
|
||||
int count = base_type::_in->read();
|
||||
if (count < 0) { return -1; }
|
||||
|
||||
int outCount = process(count, base_type::_in->readBuf, base_type::out.writeBuf);
|
||||
|
||||
// Swap if some data was generated
|
||||
base_type::_in->flush();
|
||||
if (outCount) {
|
||||
if (!base_type::out.swap(outCount)) { return -1; }
|
||||
}
|
||||
return outCount;
|
||||
}
|
||||
};
|
||||
}
|
||||
Reference in New Issue
Block a user