Files
amodem/loop.py
2014-07-22 14:19:51 +03:00

52 lines
1.4 KiB
Python

import numpy as np
import itertools
import recv
import sampling
import sigproc
class Filter(object):
def __init__(self, b, a=()):
self.b = b
self.a = a
self.x = [0] * len(b)
self.y = [0] * len(a)
def __call__(self, x):
self.x = [x] + self.x
self.x = self.x[:len(self.b)]
self.y = self.y[:len(self.a)]
y = np.dot(self.x, self.b) + np.dot(self.y, self.a)
self.y = [y] + self.y
return y
class FreqLoop(object):
def __init__(self, x, freqs, prefix=None):
interp = sampling.Interpolator()
if prefix is None:
prefix = []
else:
prefix = [prefix] * interp.width
src = itertools.chain(prefix, x)
self.sampler = sampling.Sampler(src, interp)
self.gens = []
samplers = itertools.tee(self.sampler, len(freqs))
for freq, generator in zip(freqs, samplers):
self.gens.append( recv.extract_symbols(generator, freq) )
Kp, Ki = 0.2, 0.01
b = np.array([1, -1])*Kp + np.array([0.5, 0.5])*Ki
self.filt = Filter(b=b, a=[1])
self.correction = 0.0
def correct(self, actual, expected):
self.err = np.angle(expected / actual) / np.pi
self.err = sigproc.clip(self.err, [-0.1, 0.1])
self.correction = self.filt(self.err)
self.sampler.correct(offset=self.correction)
def __iter__(self):
return itertools.izip(*self.gens)