2008-09-03 19:02:00 +00:00
//#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 0213 9 , USA .
*
2008-09-09 17:04:42 +00:00
* $ Id : v17_tests . c , v 1.90 2008 / 09 / 07 12 : 45 : 17 steveu Exp $
2008-09-03 19:02:00 +00:00
*/
/*! \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 ;
2008-09-09 17:04:42 +00:00
printf ( " V.17 rx status is %s (%d) \n " , signal_status_to_str ( status ) , status ) ;
2008-09-03 19:02:00 +00:00
rx = ( v17_rx_state_t * ) user_data ;
switch ( status )
{
2008-09-09 17:04:42 +00:00
case SIG_STATUS_TRAINING_SUCCEEDED :
2008-09-03 19:02:00 +00:00
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 )
{
2008-09-09 17:04:42 +00:00
printf ( " V.17 tx status is %s (%d) \n " , signal_status_to_str ( status ) , status ) ;
2008-09-03 19:02:00 +00:00
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 ) ;
2008-09-09 17:04:42 +00:00
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 ) ;
2008-09-03 19:02:00 +00:00
/* 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 " ) ;
2008-09-09 17:04:42 +00:00
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 ) ;
2008-09-03 19:02:00 +00:00
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 ------------------------------------------------------------*/