500 lines
16 KiB
C
500 lines
16 KiB
C
//#define ADD_MAINS_INTERFERENCE
|
|
/*
|
|
* SpanDSP - a series of DSP components for telephony
|
|
*
|
|
* v17_tests.c
|
|
*
|
|
* Written by Steve Underwood <steveu@coppice.org>
|
|
*
|
|
* Copyright (C) 2004 Steve Underwood
|
|
*
|
|
* All rights reserved.
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License version 2, as
|
|
* published by the Free Software Foundation.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
|
*
|
|
* $Id: v17_tests.c,v 1.98 2009/01/12 17:20:59 steveu Exp $
|
|
*/
|
|
|
|
/*! \page v17_tests_page V.17 modem tests
|
|
\section v17_tests_page_sec_1 What does it do?
|
|
These tests test one way paths, as V.17 is a half-duplex modem. They allow either:
|
|
|
|
- A V.17 transmit modem to feed a V.17 receive modem through a telephone line
|
|
model. BER testing is then used to evaluate performance under various line
|
|
conditions. This is effective for testing the basic performance of the
|
|
receive modem. It is also the only test mode provided for evaluating the
|
|
transmit modem.
|
|
|
|
- A V.17 receive modem is used to decode V.17 audio, stored in a wave file.
|
|
This is good way to evaluate performance with audio recorded from other
|
|
models of modem, and with real world problematic telephone lines.
|
|
|
|
If the appropriate GUI environment exists, the tests are built such that a visual
|
|
display of modem status is maintained.
|
|
|
|
\section v17_tests_page_sec_2 How is it used?
|
|
*/
|
|
|
|
/* Enable the following definition to enable direct probing into the FAX structures */
|
|
#define WITH_SPANDSP_INTERNALS
|
|
|
|
#if defined(HAVE_CONFIG_H)
|
|
#include "config.h"
|
|
#endif
|
|
|
|
#if defined(HAVE_FL_FL_H) && defined(HAVE_FL_FL_CARTESIAN_H) && defined(HAVE_FL_FL_AUDIO_METER_H)
|
|
#define ENABLE_GUI
|
|
#endif
|
|
|
|
#include <stdlib.h>
|
|
#include <stdio.h>
|
|
#include <fcntl.h>
|
|
#include <unistd.h>
|
|
#include <string.h>
|
|
#include <audiofile.h>
|
|
|
|
//#if defined(WITH_SPANDSP_INTERNALS)
|
|
#define SPANDSP_EXPOSE_INTERNAL_STRUCTURES
|
|
//#endif
|
|
|
|
#include "spandsp.h"
|
|
#include "spandsp-sim.h"
|
|
|
|
#if defined(ENABLE_GUI)
|
|
#include "modem_monitor.h"
|
|
#include "line_model_monitor.h"
|
|
#endif
|
|
|
|
#define BLOCK_LEN 160
|
|
|
|
#define OUT_FILE_NAME "v17.wav"
|
|
|
|
char *decode_test_file = NULL;
|
|
int use_gui = FALSE;
|
|
|
|
int symbol_no = 0;
|
|
|
|
int rx_bits = 0;
|
|
int tx_bits = 0;
|
|
|
|
int test_bps;
|
|
|
|
bert_state_t bert;
|
|
one_way_line_model_state_t *line_model;
|
|
|
|
#if defined(ENABLE_GUI)
|
|
qam_monitor_t *qam_monitor;
|
|
#endif
|
|
|
|
bert_results_t latest_results;
|
|
|
|
static void reporter(void *user_data, int reason, bert_results_t *results)
|
|
{
|
|
switch (reason)
|
|
{
|
|
case BERT_REPORT_REGULAR:
|
|
printf("BERT report regular - %d bits, %d bad bits, %d resyncs\n", results->total_bits, results->bad_bits, results->resyncs);
|
|
memcpy(&latest_results, results, sizeof(latest_results));
|
|
break;
|
|
default:
|
|
printf("BERT report %s\n", bert_event_to_str(reason));
|
|
break;
|
|
}
|
|
}
|
|
/*- End of function --------------------------------------------------------*/
|
|
|
|
static int v17_rx_status(void *user_data, int status)
|
|
{
|
|
v17_rx_state_t *rx;
|
|
int i;
|
|
int len;
|
|
complexf_t *coeffs;
|
|
|
|
printf("V.17 rx status is %s (%d)\n", signal_status_to_str(status), status);
|
|
rx = (v17_rx_state_t *) user_data;
|
|
switch (status)
|
|
{
|
|
case SIG_STATUS_TRAINING_SUCCEEDED:
|
|
len = v17_rx_equalizer_state(rx, &coeffs);
|
|
printf("Equalizer:\n");
|
|
for (i = 0; i < len; i++)
|
|
printf("%3d (%15.5f, %15.5f) -> %15.5f\n", i, coeffs[i].re, coeffs[i].im, powerf(&coeffs[i]));
|
|
break;
|
|
}
|
|
return 0;
|
|
}
|
|
/*- End of function --------------------------------------------------------*/
|
|
|
|
static void v17putbit(void *user_data, int bit)
|
|
{
|
|
v17_rx_state_t *rx;
|
|
|
|
if (bit < 0)
|
|
{
|
|
v17_rx_status(user_data, bit);
|
|
return;
|
|
}
|
|
|
|
rx = (v17_rx_state_t *) user_data;
|
|
if (decode_test_file)
|
|
printf("Rx bit %d - %d\n", rx_bits++, bit);
|
|
else
|
|
bert_put_bit(&bert, bit);
|
|
}
|
|
/*- End of function --------------------------------------------------------*/
|
|
|
|
static int v17_tx_status(void *user_data, int status)
|
|
{
|
|
printf("V.17 tx status is %s (%d)\n", signal_status_to_str(status), status);
|
|
return 0;
|
|
}
|
|
/*- End of function --------------------------------------------------------*/
|
|
|
|
static int v17getbit(void *user_data)
|
|
{
|
|
return bert_get_bit(&bert);
|
|
}
|
|
/*- End of function --------------------------------------------------------*/
|
|
|
|
static void qam_report(void *user_data, const complexf_t *constel, const complexf_t *target, int symbol)
|
|
{
|
|
int i;
|
|
int len;
|
|
complexf_t *coeffs;
|
|
float fpower;
|
|
v17_rx_state_t *rx;
|
|
static float smooth_power = 0.0f;
|
|
static int update_interval = 100;
|
|
|
|
rx = (v17_rx_state_t *) user_data;
|
|
if (constel)
|
|
{
|
|
#if defined(ENABLE_GUI)
|
|
if (use_gui)
|
|
{
|
|
qam_monitor_update_constel(qam_monitor, constel);
|
|
qam_monitor_update_carrier_tracking(qam_monitor, v17_rx_carrier_frequency(rx));
|
|
qam_monitor_update_symbol_tracking(qam_monitor, v17_rx_symbol_timing_correction(rx));
|
|
}
|
|
#endif
|
|
fpower = (constel->re - target->re)*(constel->re - target->re)
|
|
+ (constel->im - target->im)*(constel->im - target->im);
|
|
smooth_power = 0.95f*smooth_power + 0.05f*fpower;
|
|
printf("%8d [%8.4f, %8.4f] [%8.4f, %8.4f] %2x %8.4f %8.4f %9.4f %7.3f %7.2f\n",
|
|
symbol_no,
|
|
constel->re,
|
|
constel->im,
|
|
target->re,
|
|
target->im,
|
|
symbol,
|
|
fpower,
|
|
smooth_power,
|
|
v17_rx_carrier_frequency(rx),
|
|
v17_rx_signal_power(rx),
|
|
v17_rx_symbol_timing_correction(rx));
|
|
printf("Carrier %d %f %f\n", symbol_no, v17_rx_carrier_frequency(rx), v17_rx_symbol_timing_correction(rx));
|
|
symbol_no++;
|
|
if (--update_interval <= 0)
|
|
{
|
|
len = v17_rx_equalizer_state(rx, &coeffs);
|
|
printf("Equalizer A:\n");
|
|
for (i = 0; i < len; i++)
|
|
printf("%3d (%15.5f, %15.5f) -> %15.5f\n", i, coeffs[i].re, coeffs[i].im, powerf(&coeffs[i]));
|
|
#if defined(ENABLE_GUI)
|
|
if (use_gui)
|
|
qam_monitor_update_equalizer(qam_monitor, coeffs, len);
|
|
#endif
|
|
update_interval = 100;
|
|
}
|
|
}
|
|
}
|
|
/*- End of function --------------------------------------------------------*/
|
|
|
|
int main(int argc, char *argv[])
|
|
{
|
|
v17_rx_state_t *rx;
|
|
v17_tx_state_t *tx;
|
|
bert_results_t bert_results;
|
|
int16_t gen_amp[BLOCK_LEN];
|
|
int16_t amp[BLOCK_LEN];
|
|
AFfilehandle inhandle;
|
|
AFfilehandle outhandle;
|
|
int outframes;
|
|
int samples;
|
|
int tep;
|
|
int block_no;
|
|
int noise_level;
|
|
int signal_level;
|
|
int bits_per_test;
|
|
int line_model_no;
|
|
int log_audio;
|
|
int channel_codec;
|
|
int rbs_pattern;
|
|
int opt;
|
|
logging_state_t *logging;
|
|
|
|
channel_codec = MUNGE_CODEC_NONE;
|
|
rbs_pattern = 0;
|
|
test_bps = 14400;
|
|
tep = FALSE;
|
|
line_model_no = 0;
|
|
decode_test_file = NULL;
|
|
use_gui = FALSE;
|
|
noise_level = -70;
|
|
signal_level = -13;
|
|
bits_per_test = 50000;
|
|
log_audio = FALSE;
|
|
while ((opt = getopt(argc, argv, "b:B:c:d:glm:n:r:s:t")) != -1)
|
|
{
|
|
switch (opt)
|
|
{
|
|
case 'b':
|
|
test_bps = atoi(optarg);
|
|
if (test_bps != 14400 && test_bps != 12000 && test_bps != 9600 && test_bps != 7200)
|
|
{
|
|
fprintf(stderr, "Invalid bit rate specified\n");
|
|
exit(2);
|
|
}
|
|
break;
|
|
case 'B':
|
|
bits_per_test = atoi(optarg);
|
|
break;
|
|
case 'c':
|
|
channel_codec = atoi(optarg);
|
|
break;
|
|
case 'd':
|
|
decode_test_file = optarg;
|
|
break;
|
|
case 'g':
|
|
#if defined(ENABLE_GUI)
|
|
use_gui = TRUE;
|
|
#else
|
|
fprintf(stderr, "Graphical monitoring not available\n");
|
|
exit(2);
|
|
#endif
|
|
break;
|
|
case 'l':
|
|
log_audio = TRUE;
|
|
break;
|
|
case 'm':
|
|
line_model_no = atoi(optarg);
|
|
break;
|
|
case 'n':
|
|
noise_level = atoi(optarg);
|
|
break;
|
|
case 'r':
|
|
rbs_pattern = atoi(optarg);
|
|
break;
|
|
case 's':
|
|
signal_level = atoi(optarg);
|
|
break;
|
|
case 't':
|
|
tep = TRUE;
|
|
break;
|
|
default:
|
|
//usage();
|
|
exit(2);
|
|
break;
|
|
}
|
|
}
|
|
inhandle = NULL;
|
|
outhandle = NULL;
|
|
|
|
if (log_audio)
|
|
{
|
|
if ((outhandle = afOpenFile_telephony_write(OUT_FILE_NAME, 1)) == AF_NULL_FILEHANDLE)
|
|
{
|
|
fprintf(stderr, " Cannot create wave file '%s'\n", OUT_FILE_NAME);
|
|
exit(2);
|
|
}
|
|
}
|
|
|
|
if (decode_test_file)
|
|
{
|
|
/* We will decode the audio from a wave file. */
|
|
tx = NULL;
|
|
if ((inhandle = afOpenFile_telephony_read(decode_test_file, 1)) == AF_NULL_FILEHANDLE)
|
|
{
|
|
fprintf(stderr, " Cannot open wave file '%s'\n", decode_test_file);
|
|
exit(2);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
/* We will generate V.17 audio, and add some noise to it. */
|
|
tx = v17_tx_init(NULL, test_bps, tep, v17getbit, NULL);
|
|
logging = v17_tx_get_logging_state(tx);
|
|
span_log_set_level(logging, SPAN_LOG_SHOW_SEVERITY | SPAN_LOG_SHOW_PROTOCOL | SPAN_LOG_FLOW);
|
|
span_log_set_tag(logging, "V.17 tx");
|
|
v17_tx_power(tx, signal_level);
|
|
v17_tx_set_modem_status_handler(tx, v17_tx_status, (void *) tx);
|
|
#if defined(WITH_SPANDSP_INTERNALS)
|
|
/* Move the carrier off a bit */
|
|
tx->carrier_phase_rate = dds_phase_ratef(1792.0f);
|
|
tx->carrier_phase = 0x40000000;
|
|
#endif
|
|
|
|
bert_init(&bert, bits_per_test, BERT_PATTERN_ITU_O152_11, test_bps, 20);
|
|
bert_set_report(&bert, 10000, reporter, NULL);
|
|
|
|
if ((line_model = one_way_line_model_init(line_model_no, (float) noise_level, channel_codec, rbs_pattern)) == NULL)
|
|
{
|
|
fprintf(stderr, " Failed to create line model\n");
|
|
exit(2);
|
|
}
|
|
one_way_line_model_set_dc(line_model, 0.0f);
|
|
#if defined(ADD_MAINS_INTERFERENCE)
|
|
one_way_line_model_set_mains_pickup(line_model, 50, -40.0f);
|
|
#endif
|
|
}
|
|
|
|
rx = v17_rx_init(NULL, test_bps, v17putbit, NULL);
|
|
v17_rx_set_modem_status_handler(rx, v17_rx_status, (void *) rx);
|
|
v17_rx_set_qam_report_handler(rx, qam_report, (void *) rx);
|
|
logging = v17_rx_get_logging_state(rx);
|
|
span_log_set_level(logging, SPAN_LOG_SHOW_SEVERITY | SPAN_LOG_SHOW_PROTOCOL | SPAN_LOG_FLOW);
|
|
span_log_set_tag(logging, "V.17 rx");
|
|
|
|
#if defined(ENABLE_GUI)
|
|
if (use_gui)
|
|
{
|
|
qam_monitor = qam_monitor_init(10.0f, NULL);
|
|
if (!decode_test_file)
|
|
{
|
|
start_line_model_monitor(129);
|
|
line_model_monitor_line_model_update(line_model->near_filter, line_model->near_filter_len);
|
|
}
|
|
}
|
|
#endif
|
|
|
|
memset(&latest_results, 0, sizeof(latest_results));
|
|
for (block_no = 0; block_no < 100000000; block_no++)
|
|
{
|
|
if (decode_test_file)
|
|
{
|
|
samples = afReadFrames(inhandle,
|
|
AF_DEFAULT_TRACK,
|
|
amp,
|
|
BLOCK_LEN);
|
|
#if defined(ENABLE_GUI)
|
|
if (use_gui)
|
|
qam_monitor_update_audio_level(qam_monitor, amp, samples);
|
|
#endif
|
|
if (samples == 0)
|
|
break;
|
|
}
|
|
else
|
|
{
|
|
samples = v17_tx(tx, gen_amp, BLOCK_LEN);
|
|
#if defined(ENABLE_GUI)
|
|
if (use_gui)
|
|
qam_monitor_update_audio_level(qam_monitor, gen_amp, samples);
|
|
#endif
|
|
if (samples == 0)
|
|
{
|
|
printf("Restarting on zero output\n");
|
|
|
|
/* Push a little silence through, to ensure all the data bits get out of the buffers */
|
|
memset(amp, 0, BLOCK_LEN*sizeof(int16_t));
|
|
v17_rx(rx, amp, BLOCK_LEN);
|
|
|
|
/* Note that we might get a few bad bits as the carrier shuts down. */
|
|
bert_result(&bert, &bert_results);
|
|
fprintf(stderr, "Final result %ddBm0/%ddBm0, %d bits, %d bad bits, %d resyncs\n", signal_level, noise_level, bert_results.total_bits, bert_results.bad_bits, bert_results.resyncs);
|
|
fprintf(stderr, "Last report %ddBm0/%ddBm0, %d bits, %d bad bits, %d resyncs\n", signal_level, noise_level, latest_results.total_bits, latest_results.bad_bits, latest_results.resyncs);
|
|
/* See if bit errors are appearing yet. Also check we are getting enough bits out of the receiver. The last regular report
|
|
should be error free, though the final report will generally contain bits errors as the carrier was dying. The total
|
|
number of bits out of the receiver should be at least the number we sent. Also, since BERT sync should have occurred
|
|
rapidly at the start of transmission, the last report should have occurred at not much less than the total number of
|
|
bits we sent. */
|
|
if (bert_results.total_bits < bits_per_test
|
|
||
|
|
latest_results.total_bits < bits_per_test - 100
|
|
||
|
|
latest_results.bad_bits != 0)
|
|
{
|
|
break;
|
|
}
|
|
memset(&latest_results, 0, sizeof(latest_results));
|
|
#if defined(WITH_SPANDSP_INTERNALS)
|
|
signal_level--;
|
|
/* Bump the receiver AGC gain by 1dB, to compensate for the above */
|
|
rx->agc_scaling_save *= 1.122f;
|
|
#endif
|
|
v17_tx_restart(tx, test_bps, tep, TRUE);
|
|
v17_tx_power(tx, signal_level);
|
|
v17_rx_restart(rx, test_bps, TRUE);
|
|
//rx.eq_put_step = rand()%(192*10/3);
|
|
bert_init(&bert, bits_per_test, BERT_PATTERN_ITU_O152_11, test_bps, 20);
|
|
bert_set_report(&bert, 10000, reporter, NULL);
|
|
one_way_line_model_release(line_model);
|
|
if ((line_model = one_way_line_model_init(line_model_no, (float) noise_level, channel_codec, 0)) == NULL)
|
|
{
|
|
fprintf(stderr, " Failed to create line model\n");
|
|
exit(2);
|
|
}
|
|
}
|
|
if (log_audio)
|
|
{
|
|
outframes = afWriteFrames(outhandle,
|
|
AF_DEFAULT_TRACK,
|
|
gen_amp,
|
|
samples);
|
|
if (outframes != samples)
|
|
{
|
|
fprintf(stderr, " Error writing wave file\n");
|
|
exit(2);
|
|
}
|
|
}
|
|
one_way_line_model(line_model, amp, gen_amp, samples);
|
|
}
|
|
#if defined(ENABLE_GUI)
|
|
if (use_gui && !decode_test_file)
|
|
line_model_monitor_line_spectrum_update(amp, samples);
|
|
#endif
|
|
v17_rx(rx, amp, samples);
|
|
}
|
|
if (!decode_test_file)
|
|
{
|
|
bert_result(&bert, &bert_results);
|
|
fprintf(stderr, "At completion:\n");
|
|
fprintf(stderr, "Final result %ddBm0/%ddBm0, %d bits, %d bad bits, %d resyncs\n", signal_level, noise_level, bert_results.total_bits, bert_results.bad_bits, bert_results.resyncs);
|
|
fprintf(stderr, "Last report %ddBm0/%ddBm0, %d bits, %d bad bits, %d resyncs\n", signal_level, noise_level, latest_results.total_bits, latest_results.bad_bits, latest_results.resyncs);
|
|
one_way_line_model_release(line_model);
|
|
|
|
if (signal_level > -43)
|
|
{
|
|
printf("Tests failed.\n");
|
|
exit(2);
|
|
}
|
|
|
|
printf("Tests passed.\n");
|
|
}
|
|
#if defined(ENABLE_GUI)
|
|
if (use_gui)
|
|
qam_wait_to_end(qam_monitor);
|
|
#endif
|
|
if (log_audio)
|
|
{
|
|
if (afCloseFile(outhandle))
|
|
{
|
|
fprintf(stderr, " Cannot close wave file '%s'\n", OUT_FILE_NAME);
|
|
exit(2);
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
/*- End of function --------------------------------------------------------*/
|
|
/*- End of file ------------------------------------------------------------*/
|