freeswitch/libs/spandsp/tests/v17_tests.c

547 lines
17 KiB
C
Raw Normal View History

//#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.87 2008/08/29 09:28:13 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?
*/
#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>
#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_SYNCED:
printf("BERT report synced\n");
break;
case BERT_REPORT_UNSYNCED:
printf("BERT report unsync'ed\n");
break;
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;
case BERT_REPORT_GT_10_2:
printf("BERT report > 1 in 10^2\n");
break;
case BERT_REPORT_LT_10_2:
printf("BERT report < 1 in 10^2\n");
break;
case BERT_REPORT_LT_10_3:
printf("BERT report < 1 in 10^3\n");
break;
case BERT_REPORT_LT_10_4:
printf("BERT report < 1 in 10^4\n");
break;
case BERT_REPORT_LT_10_5:
printf("BERT report < 1 in 10^5\n");
break;
case BERT_REPORT_LT_10_6:
printf("BERT report < 1 in 10^6\n");
break;
case BERT_REPORT_LT_10_7:
printf("BERT report < 1 in 10^7\n");
break;
default:
printf("BERT report reason %d\n", 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 %d\n", status);
rx = (v17_rx_state_t *) user_data;
switch (status)
{
case PUTBIT_TRAINING_FAILED:
printf("Training failed\n");
break;
case PUTBIT_TRAINING_IN_PROGRESS:
printf("Training in progress\n");
break;
case PUTBIT_TRAINING_SUCCEEDED:
printf("Training succeeded\n");
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;
case PUTBIT_CARRIER_UP:
printf("Carrier up\n");
break;
case PUTBIT_CARRIER_DOWN:
printf("Carrier down\n");
break;
default:
printf("Eh! - %d\n", status);
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)
{
switch (status)
{
case MODEM_TX_STATUS_DATA_EXHAUSTED:
printf("V.17 tx data exhausted\n");
break;
case MODEM_TX_STATUS_SHUTDOWN_COMPLETE:
printf("V.17 tx shutdown complete\n");
break;
default:
printf("V.17 tx status is %d\n", status);
break;
}
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;
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:c:d:glm:n:r:s:t")) != -1)
{
switch (opt)
{
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;
}
}
argc -= optind;
argv += optind;
if (argc > 0)
{
if (strcmp(argv[0], "14400") == 0)
test_bps = 14400;
else if (strcmp(argv[0], "12000") == 0)
test_bps = 12000;
else if (strcmp(argv[0], "9600") == 0)
test_bps = 9600;
else if (strcmp(argv[0], "7200") == 0)
test_bps = 7200;
else
{
fprintf(stderr, "Invalid bit rate\n");
exit(2);
}
}
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. */
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. */
v17_tx_init(&tx, test_bps, tep, v17getbit, NULL);
v17_tx_power(&tx, signal_level);
v17_tx_set_modem_status_handler(&tx, v17_tx_status, (void *) &tx);
/* Move the carrier off a bit */
tx.carrier_phase_rate = dds_phase_ratef(1792.0f);
tx.carrier_phase = 0x40000000;
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
}
v17_rx_init(&rx, test_bps, v17putbit, &rx);
v17_rx_set_modem_status_handler(&rx, v17_rx_status, (void *) &rx);
v17_rx_set_qam_report_handler(&rx, qam_report, (void *) &rx);
span_log_set_level(&rx.logging, SPAN_LOG_SHOW_SEVERITY | SPAN_LOG_SHOW_PROTOCOL | SPAN_LOG_FLOW);
span_log_set_tag(&rx.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, %d bits, %d bad bits, %d resyncs\n", signal_level, bert_results.total_bits, bert_results.bad_bits, bert_results.resyncs);
fprintf(stderr, "Last report %ddBm0, %d bits, %d bad bits, %d resyncs\n", signal_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));
signal_level--;
/* Bump the receiver AGC gain by 1dB, to compensate for the above */
rx.agc_scaling_save *= 1.122f;
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,
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, %d bits, %d bad bits, %d resyncs\n", signal_level, bert_results.total_bits, bert_results.bad_bits, bert_results.resyncs);
fprintf(stderr, "Last report %ddBm0, %d bits, %d bad bits, %d resyncs\n", signal_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 ------------------------------------------------------------*/