2020-12-16 18:52:00 +01:00
|
|
|
from csdr.chain import Chain
|
2021-07-16 16:12:16 +02:00
|
|
|
from pycsdr.modules import Fft, LogPower, LogAveragePower, FftSwap, FftAdpcm
|
2020-12-16 18:52:00 +01:00
|
|
|
|
|
|
|
|
2020-12-21 00:33:48 +01:00
|
|
|
class FftAverager(Chain):
|
|
|
|
def __init__(self, fft_size, fft_averages):
|
|
|
|
self.fftSize = fft_size
|
2021-07-20 00:44:41 +02:00
|
|
|
self.fftAverages = fft_averages
|
|
|
|
workers = [self._getWorker()]
|
2021-08-23 14:25:28 +02:00
|
|
|
super().__init__(workers)
|
2020-12-21 00:33:48 +01:00
|
|
|
|
|
|
|
def setFftAverages(self, fft_averages):
|
|
|
|
if self.fftAverages == fft_averages:
|
|
|
|
return
|
|
|
|
self.fftAverages = fft_averages
|
2021-07-20 00:44:41 +02:00
|
|
|
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-21 00:33:48 +01:00
|
|
|
|
|
|
|
|
2020-12-16 18:52:00 +01:00
|
|
|
class FftChain(Chain):
|
2020-12-20 22:55:10 +01: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)
|
2020-12-25 20:27:30 +01:00
|
|
|
self.averager = FftAverager(fft_size=self.size, fft_averages=10)
|
2021-07-16 16:12:16 +02:00
|
|
|
self.fftExchangeSides = FftSwap(fft_size=self.size)
|
2020-12-16 18:52:00 +01:00
|
|
|
workers = [
|
2020-12-19 16:41:48 +01:00
|
|
|
self.fft,
|
2020-12-21 00:33:48 +01:00
|
|
|
self.averager,
|
2020-12-19 16:41:48 +01:00
|
|
|
self.fftExchangeSides,
|
2020-12-16 18:52:00 +01:00
|
|
|
]
|
2020-12-19 16:41:48 +01:00
|
|
|
self.compressFftAdpcm = None
|
2020-12-16 18:52:00 +01:00
|
|
|
if fft_compression == "adpcm":
|
2021-07-16 16:12:16 +02:00
|
|
|
self.compressFftAdpcm = FftAdpcm(fft_size=self.size)
|
2020-12-19 16:41:48 +01:00
|
|
|
workers += [self.compressFftAdpcm]
|
2020-12-20 22:55:10 +01:00
|
|
|
|
|
|
|
self._updateParameters()
|
|
|
|
|
2021-08-23 14:25:28 +02:00
|
|
|
super().__init__(workers)
|
2020-12-19 17:13:36 +01:00
|
|
|
|
2020-12-20 22:55:10 +01: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):
|
2020-12-21 00:33:48 +01:00
|
|
|
fftAverages = 0
|
|
|
|
|
2020-12-20 22:55:10 +01:00
|
|
|
if self.vOverlapFactor > 0:
|
2020-12-21 00:33:48 +01:00
|
|
|
fftAverages = int(round(1.0 * self.sampleRate / self.size / self.fps / (1.0 - self.vOverlapFactor)))
|
|
|
|
self.averager.setFftAverages(fftAverages)
|
2020-12-19 17:13:36 +01:00
|
|
|
|
2020-12-21 00:33:48 +01:00
|
|
|
if fftAverages == 0:
|
2020-12-20 22:55:10 +01:00
|
|
|
self._setBlockSize(self.sampleRate / self.fps)
|
|
|
|
else:
|
2020-12-21 00:33:48 +01:00
|
|
|
self._setBlockSize(self.sampleRate / self.fps / fftAverages)
|
2021-09-27 17:29:51 +02: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)
|