mirror of
https://github.com/romanz/amodem.git
synced 2026-03-17 23:26:01 +08:00
236 lines
8.5 KiB
Python
236 lines
8.5 KiB
Python
import numpy as np
|
|
import logging
|
|
import itertools
|
|
import functools
|
|
import time
|
|
|
|
log = logging.getLogger(__name__)
|
|
|
|
from . import stream
|
|
from . import dsp
|
|
from . import sampling
|
|
from . import train
|
|
from . import common
|
|
from . import framing
|
|
from . import equalizer
|
|
from . import detect
|
|
|
|
|
|
class Receiver(object):
|
|
|
|
def __init__(self, config, pylab=None):
|
|
self.stats = {}
|
|
self.plt = pylab or common.Dummy()
|
|
self.modem = dsp.MODEM(config.symbols)
|
|
self.frequencies = np.array(config.frequencies)
|
|
self.omegas = 2 * np.pi * self.frequencies / config.Fs
|
|
self.Nsym = config.Nsym
|
|
self.Tsym = config.Tsym
|
|
self.iters_per_report = config.baud # report once per second
|
|
self.modem_bitrate = config.modem_bps
|
|
self.equalizer = equalizer.Equalizer(config)
|
|
self.carrier_index = config.carrier_index
|
|
|
|
def _prefix(self, symbols, gain=1.0, skip=5):
|
|
S = common.take(symbols, len(train.prefix))
|
|
S = S[:, self.carrier_index] * gain
|
|
sliced = np.round(np.abs(S))
|
|
self.plt.figure()
|
|
self.plt.subplot(121)
|
|
self._constellation(S, sliced, 'Prefix')
|
|
|
|
bits = np.array(sliced, dtype=int)
|
|
self.plt.subplot(122)
|
|
self.plt.plot(np.abs(S))
|
|
self.plt.plot(train.prefix)
|
|
if any(bits != train.prefix):
|
|
raise ValueError('Incorrect prefix')
|
|
|
|
log.debug('Prefix OK')
|
|
|
|
nonzeros = np.array(train.prefix, dtype=bool)
|
|
pilot_tone = S[nonzeros]
|
|
phase = np.unwrap(np.angle(pilot_tone)) / (2 * np.pi)
|
|
indices = np.arange(len(phase))
|
|
a, b = dsp.linear_regression(indices[skip:-skip], phase[skip:-skip])
|
|
self.plt.figure()
|
|
self.plt.plot(indices, phase, ':')
|
|
self.plt.plot(indices, a * indices + b)
|
|
|
|
freq_err = a / (self.Tsym * self.frequencies[self.carrier_index])
|
|
last_phase = a * indices[-1] + b
|
|
log.debug('Current phase on carrier: %.3f', last_phase)
|
|
|
|
log.debug('Frequency error: %.2f ppm', freq_err * 1e6)
|
|
self.plt.title('Frequency drift: {0:.3f} ppm'.format(freq_err * 1e6))
|
|
return freq_err
|
|
|
|
def _train(self, sampler, order, lookahead):
|
|
Nfreq = len(self.frequencies)
|
|
train_symbols = self.equalizer.train_symbols(train.equalizer_length)
|
|
train_signal = self.equalizer.modulator(train_symbols) * Nfreq
|
|
|
|
prefix = postfix = train.silence_length * self.Nsym
|
|
signal_length = train.equalizer_length * self.Nsym + prefix + postfix
|
|
|
|
signal = sampler.take(signal_length + lookahead)
|
|
|
|
coeffs = self.equalizer.equalize_signal(
|
|
signal=signal[prefix:-postfix],
|
|
expected=train_signal,
|
|
order=order, lookahead=lookahead
|
|
)
|
|
|
|
self.plt.figure()
|
|
self.plt.plot(np.arange(order+lookahead), coeffs)
|
|
|
|
equalization_filter = dsp.FIR(h=coeffs)
|
|
equalized = list(equalization_filter(signal))
|
|
equalized = equalized[prefix+lookahead:-postfix+lookahead]
|
|
|
|
symbols = self.equalizer.demodulator(equalized, train.equalizer_length)
|
|
sliced = np.array(symbols).round()
|
|
errors = np.array(sliced - train_symbols, dtype=np.bool)
|
|
error_rate = errors.sum() / errors.size
|
|
|
|
errors = np.array(symbols - train_symbols)
|
|
rms = lambda x: (np.mean(np.abs(x) ** 2, axis=0) ** 0.5)
|
|
|
|
noise_rms = rms(errors)
|
|
signal_rms = rms(train_symbols)
|
|
SNRs = 20.0 * np.log10(signal_rms / noise_rms)
|
|
|
|
self.plt.figure()
|
|
for (i, freq), snr in zip(enumerate(self.frequencies), SNRs):
|
|
log.debug('%5.1f kHz: SNR = %5.2f dB', freq / 1e3, snr)
|
|
self._constellation(symbols[:, i], train_symbols[:, i],
|
|
'$F_c = {0} Hz$'.format(freq), index=i)
|
|
assert error_rate == 0, error_rate
|
|
|
|
return equalization_filter
|
|
|
|
def _demodulate(self, sampler, symbols):
|
|
streams = []
|
|
symbol_list = []
|
|
errors = {}
|
|
|
|
def error_handler(received, decoded, freq):
|
|
errors.setdefault(freq, []).append(received / decoded)
|
|
|
|
generators = common.split(symbols, n=len(self.omegas))
|
|
for freq, S in zip(self.frequencies, generators):
|
|
equalized = []
|
|
S = common.icapture(S, result=equalized)
|
|
symbol_list.append(equalized)
|
|
|
|
freq_handler = functools.partial(error_handler, freq=freq)
|
|
bits = self.modem.decode(S, freq_handler) # list of bit tuples
|
|
streams.append(bits) # stream per frequency
|
|
|
|
self.stats['symbol_list'] = symbol_list
|
|
self.stats['rx_bits'] = 0
|
|
self.stats['rx_start'] = time.time()
|
|
|
|
log.info('Starting demodulation')
|
|
for i, block in enumerate(common.izip(streams), 1):
|
|
for bits in block:
|
|
self.stats['rx_bits'] = self.stats['rx_bits'] + len(bits)
|
|
yield bits
|
|
|
|
if i % self.iters_per_report == 0:
|
|
err = np.array([e for v in errors.values() for e in v])
|
|
err = np.mean(np.angle(err))/(2*np.pi) if len(err) else 0
|
|
errors.clear()
|
|
|
|
duration = time.time() - self.stats['rx_start']
|
|
sampler.freq -= 0.01 * err * self.Tsym
|
|
sampler.offset -= err
|
|
log.debug(
|
|
'Got %8.1f kB, realtime: %6.2f%%, drift: %+5.2f ppm',
|
|
self.stats['rx_bits'] / 8e3,
|
|
duration * 100.0 / (i*self.Tsym),
|
|
(1.0 - sampler.freq) * 1e6
|
|
)
|
|
|
|
def start(self, signal, gain=1.0):
|
|
sampler = sampling.Sampler(signal, sampling.Interpolator())
|
|
symbols = dsp.Demux(sampler, omegas=self.omegas, Nsym=self.Nsym)
|
|
freq_err = self._prefix(symbols, gain=gain)
|
|
sampler.freq -= freq_err
|
|
|
|
filt = self._train(sampler, order=11, lookahead=6)
|
|
sampler.equalizer = lambda x: list(filt(x))
|
|
|
|
bitstream = self._demodulate(sampler, symbols)
|
|
self.bitstream = itertools.chain.from_iterable(bitstream)
|
|
|
|
def run(self, output):
|
|
data = framing.decode(self.bitstream)
|
|
self.size = 0
|
|
for chunk in common.iterate(data=data, size=256,
|
|
truncate=False, func=bytearray):
|
|
output.write(chunk)
|
|
self.size += len(chunk)
|
|
|
|
def report(self):
|
|
if self.stats:
|
|
duration = time.time() - self.stats['rx_start']
|
|
audio_time = self.stats['rx_bits'] / float(self.modem_bitrate)
|
|
log.debug('Demodulated %.3f kB @ %.3f seconds (%.1f%% realtime)',
|
|
self.stats['rx_bits'] / 8e3, duration,
|
|
100 * duration / audio_time if audio_time else 0)
|
|
|
|
log.info('Received %.3f kB @ %.3f seconds = %.3f kB/s',
|
|
self.size * 1e-3, duration, self.size * 1e-3 / duration)
|
|
|
|
self.plt.figure()
|
|
symbol_list = np.array(self.stats['symbol_list'])
|
|
for i, freq in enumerate(self.frequencies):
|
|
self._constellation(symbol_list[i], self.modem.symbols,
|
|
'$F_c = {0} Hz$'.format(freq), index=i)
|
|
self.plt.show()
|
|
|
|
def _constellation(self, y, symbols, title, index=None):
|
|
if index is not None:
|
|
Nfreq = len(self.frequencies)
|
|
height = np.floor(np.sqrt(Nfreq))
|
|
width = np.ceil(Nfreq / float(height))
|
|
self.plt.subplot(height, width, index + 1)
|
|
|
|
theta = np.linspace(0, 2*np.pi, 1000)
|
|
y = np.array(y)
|
|
self.plt.plot(y.real, y.imag, '.')
|
|
self.plt.plot(np.cos(theta), np.sin(theta), ':')
|
|
points = np.array(symbols)
|
|
self.plt.plot(points.real, points.imag, '+')
|
|
self.plt.grid('on')
|
|
self.plt.axis('equal')
|
|
self.plt.axis(np.array([-1, 1, -1, 1])*1.1)
|
|
self.plt.title(title)
|
|
|
|
|
|
def main(config, src, dst, pylab=None):
|
|
reader = stream.Reader(src, data_type=common.loads)
|
|
signal = itertools.chain.from_iterable(reader)
|
|
|
|
to_skip = int(config.skip_start * config.Fs)
|
|
log.debug('Skipping %.3f seconds', config.skip_start)
|
|
common.take(signal, to_skip)
|
|
|
|
reader.check = common.check_saturation
|
|
|
|
detector = detect.Detector(config=config)
|
|
receiver = Receiver(config=config, pylab=pylab)
|
|
success = False
|
|
try:
|
|
log.info('Waiting for carrier tone: %.1f kHz', config.Fc / 1e3)
|
|
signal, amplitude = detector.run(signal)
|
|
receiver.start(signal, gain=1.0/amplitude)
|
|
receiver.run(dst)
|
|
success = True
|
|
except Exception:
|
|
log.exception('Decoding failed')
|
|
|
|
receiver.report()
|
|
return success
|