openwebrx-clone/csdr/chain/fft.py

97 lines
3.2 KiB
Python
Raw Permalink Normal View History

2020-12-16 17:52:00 +00:00
from csdr.chain import Chain
2021-07-16 14:12:16 +00:00
from pycsdr.modules import Fft, LogPower, LogAveragePower, FftSwap, FftAdpcm
2020-12-16 17:52:00 +00:00
class FftAverager(Chain):
def __init__(self, fft_size, fft_averages):
self.fftSize = fft_size
self.fftAverages = fft_averages
workers = [self._getWorker()]
2021-08-23 12:25:28 +00:00
super().__init__(workers)
def setFftAverages(self, fft_averages):
if self.fftAverages == fft_averages:
return
self.fftAverages = fft_averages
self.replace(0, self._getWorker())
def _getWorker(self):
if self.fftAverages == 0:
return LogPower(add_db=-70)
else:
return LogAveragePower(add_db=-70, fft_size=self.fftSize, avg_number=self.fftAverages)
2020-12-16 17:52:00 +00:00
class FftChain(Chain):
2020-12-20 21:55:10 +00:00
def __init__(self, samp_rate, fft_size, fft_v_overlap_factor, fft_fps, fft_compression):
self.sampleRate = samp_rate
self.vOverlapFactor = fft_v_overlap_factor
self.fps = fft_fps
self.size = fft_size
self.blockSize = 0
self.fft = Fft(size=self.size, every_n_samples=self.blockSize)
self.averager = FftAverager(fft_size=self.size, fft_averages=10)
2021-07-16 14:12:16 +00:00
self.fftExchangeSides = FftSwap(fft_size=self.size)
2020-12-16 17:52:00 +00:00
workers = [
2020-12-19 15:41:48 +00:00
self.fft,
self.averager,
2020-12-19 15:41:48 +00:00
self.fftExchangeSides,
2020-12-16 17:52:00 +00:00
]
2020-12-19 15:41:48 +00:00
self.compressFftAdpcm = None
2020-12-16 17:52:00 +00:00
if fft_compression == "adpcm":
2021-07-16 14:12:16 +00:00
self.compressFftAdpcm = FftAdpcm(fft_size=self.size)
2020-12-19 15:41:48 +00:00
workers += [self.compressFftAdpcm]
2020-12-20 21:55:10 +00:00
self._updateParameters()
2021-08-23 12:25:28 +00:00
super().__init__(workers)
2020-12-20 21:55:10 +00:00
def _setBlockSize(self, fft_block_size):
if self.blockSize == int(fft_block_size):
return
self.blockSize = int(fft_block_size)
self.fft.setEveryNSamples(self.blockSize)
def setVOverlapFactor(self, fft_v_overlap_factor):
if self.vOverlapFactor == fft_v_overlap_factor:
return
self.vOverlapFactor = fft_v_overlap_factor
self._updateParameters()
def setFps(self, fft_fps):
if self.fps == fft_fps:
return
self.fps = fft_fps
self._updateParameters()
def setSampleRate(self, samp_rate):
if self.sampleRate == samp_rate:
return
self.sampleRate = samp_rate
self._updateParameters()
def _updateParameters(self):
fftAverages = 0
2020-12-20 21:55:10 +00:00
if self.vOverlapFactor > 0:
fftAverages = int(round(1.0 * self.sampleRate / self.size / self.fps / (1.0 - self.vOverlapFactor)))
self.averager.setFftAverages(fftAverages)
if fftAverages == 0:
2020-12-20 21:55:10 +00:00
self._setBlockSize(self.sampleRate / self.fps)
else:
self._setBlockSize(self.sampleRate / self.fps / fftAverages)
2021-09-27 15:29:51 +00:00
def setCompression(self, compression: str) -> None:
if compression == "adpcm" and not self.compressFftAdpcm:
self.compressFftAdpcm = FftAdpcm(self.size)
# should always be at the end
self.append(self.compressFftAdpcm)
elif compression == "none" and self.compressFftAdpcm:
self.compressFftAdpcm.stop()
self.compressFftAdpcm = None
# should always be at that position (right?)
self.remove(3)