Files
amodem/amodem/calib.py
2014-12-29 17:54:42 +02:00

86 lines
2.4 KiB
Python

import numpy as np
import logging
log = logging.getLogger(__name__)
from . import common
ALLOWED_EXCEPTIONS = (IOError, KeyboardInterrupt)
def send(config, dst, src=None, verbose=False):
calibration_symbols = int(1.0 * config.Fs)
t = np.arange(0, calibration_symbols) * config.Ts
signal = [np.sin(2 * np.pi * f * t) for f in config.frequencies]
signal = common.dumps(np.concatenate(signal))
try:
while True:
dst.write(signal)
except ALLOWED_EXCEPTIONS:
pass
def run_recorder(config, recorder):
frame_length = 200 * config.Nsym
frame_size = frame_length * config.sample_size
t = np.arange(0, frame_length) * config.Ts
scaling_factor = 0.5 * len(t)
carriers = [np.exp(2j * np.pi * f * t) for f in config.frequencies]
carriers = np.array(carriers) / scaling_factor
states = [True]
errors = ['weak', 'strong', 'noisy']
try:
while True:
data = recorder.read(frame_size)
if len(data) < frame_size:
return
data = common.loads(data)
frame = data - np.mean(data)
coeffs = np.dot(carriers, frame)
peak = np.max(np.abs(frame))
total = np.sqrt(np.dot(frame, frame) / scaling_factor)
max_index = np.argmax(np.abs(coeffs))
freq = config.frequencies[max_index]
rms = abs(coeffs[max_index])
coherency = rms / total
flags = [rms > 0.1, peak < 1.0, coherency > 0.99]
states.append(all(flags))
states = states[-2:]
message = 'good signal'
error = not any(states)
if error:
error_index = flags.index(False)
message = 'too {0} signal'.format(errors[error_index])
yield dict(
freq=freq, rms=rms, peak=peak, coherency=coherency,
total=total, error=error, message=message
)
except ALLOWED_EXCEPTIONS:
pass
fmt = '{freq:6.0f} Hz: {message:s}{extra:s}'
fields = ['peak', 'total', 'rms', 'coherency']
def recv(config, src, dst=None, verbose=False):
extra = ''
if verbose:
extra = ''.join(', {0}={{{0}:.4f}}'.format(f) for f in fields)
for result in run_recorder(config=config, recorder=src):
msg = fmt.format(extra=extra.format(**result), **result)
if not result['error']:
log.info(msg)
else:
log.error(msg)