mirror of
https://github.com/romanz/amodem.git
synced 2026-04-04 11:36:25 +08:00
refactor equalizers and its tests
This commit is contained in:
63
amodem/equalizer.py
Normal file
63
amodem/equalizer.py
Normal file
@@ -0,0 +1,63 @@
|
||||
import numpy as np
|
||||
from numpy.linalg import lstsq
|
||||
|
||||
from amodem import dsp, config, send
|
||||
|
||||
import itertools
|
||||
import random
|
||||
|
||||
_constellation = [1, 1j, -1, -1j]
|
||||
|
||||
|
||||
def train_symbols(length, seed=0, Nfreq=config.Nfreq):
|
||||
r = random.Random(seed)
|
||||
choose = lambda: [r.choice(_constellation) for j in range(Nfreq)]
|
||||
return np.array([choose() for i in range(length)])
|
||||
|
||||
|
||||
def modulator(symbols, carriers=send.sym.carrier):
|
||||
gain = 1.0 / len(carriers)
|
||||
result = []
|
||||
for s in symbols:
|
||||
result.append(np.dot(s, carriers))
|
||||
result = np.concatenate(result).real * gain
|
||||
assert np.max(np.abs(result)) <= 1
|
||||
return result
|
||||
|
||||
|
||||
def demodulator(signal, size):
|
||||
signal = itertools.chain(signal, itertools.repeat(0))
|
||||
symbols = dsp.Demux(signal, config.frequencies)
|
||||
return np.array(list(itertools.islice(symbols, size)))
|
||||
|
||||
|
||||
def equalize(signal, symbols, order):
|
||||
Nsym = config.Nsym
|
||||
Nfreq = config.Nfreq
|
||||
carriers = send.sym.carrier
|
||||
|
||||
assert symbols.shape[1] == Nfreq
|
||||
length = symbols.shape[0]
|
||||
|
||||
matched = np.array(carriers) * Nfreq / (0.5*Nsym)
|
||||
matched = matched[:, ::-1].transpose().conj()
|
||||
y = dsp.lfilter(x=signal, b=matched, a=[1])
|
||||
|
||||
A = np.zeros([symbols.size, order], dtype=np.complex128)
|
||||
b = np.zeros([symbols.size], dtype=np.complex128)
|
||||
|
||||
index = 0
|
||||
for j in range(Nfreq):
|
||||
for i in range(length):
|
||||
offset = (i+1)*Nsym
|
||||
row = y[offset-order:offset, j]
|
||||
A[index, :] = row
|
||||
b[index] = symbols[i, j]
|
||||
index += 1
|
||||
|
||||
A = np.array(A)
|
||||
b = np.array(b)
|
||||
h, residuals, rank, sv = lstsq(A, b)
|
||||
h = h[::-1].real
|
||||
|
||||
return h
|
||||
Reference in New Issue
Block a user