Mercurial > hg > audiostuff
view intercom/gsm/rpedemo.c @ 6:22a74b01a099 default tip
implement more meaningful test program
author | Peter Meerwald <pmeerw@cosy.sbg.ac.at> |
---|---|
date | Fri, 25 Jun 2010 16:14:50 +0200 (2010-06-25) |
parents | 13be24d74cd2 |
children |
line wrap: on
line source
/* v1.1 - 22/Feb/1996 ============================================================================ RPEDEMO.C ~~~~~~~~~~ Description: ~~~~~~~~~~~~ Demonstration program for UGST/ITU-T RPE-LTP module. Takes the input file and processes by the GSM 06.10 Full Rate speech codec, depending on user's option: for encoding, input must be in either linear, A, mu law (G711); for decoding, in GSM 06.10 76-word format. The modules called have been written in Unix-C by Jutta Deneger and Carsten Borman, within the Communications and Operating Systems Research Group (KBS) of the Technishe Universitaet Berlin. This demo program has been written by Simao F.Campos Neto, from CPqD/Telebras based on previous UGST demo programs, as well on a driving program by the RPE-LTP program. Input data is supposed to be aligned at word boundaries, i.e., organized in 16-bit words, following the operating system normal organization (low-byte first for VMS and DOS; high byte first for most Unix systems). G711 compressed data is supposed to be in the 8 LEAST significant bits of the word and the RPE-LTP data will use a varied number of least significant bits. Both are without sign extension. Linear format, on its hand, expect data to be 16-bit aligned to the left, and the algorith will take only the 13 most significant bits. This complies to the general UGST integer data format. Output data will be generated in the same format as decribed above for the input data. Usage: ~~~~~~ $ rpedemo [-l|-u|-A] [-enc|-dec] InpFile OutFile BlockSize 1stBlock NoOfBlocks where: -l .......... input data for encoding and output data for decoding are in linear format (DEFAULT). -A .......... input data for encoding and output data for decoding are in A-law (G.711) format. -u .......... input data for encoding and output data for decoding are in u-law (G.711) format. -enc ........ run the only the decoder (default: run enc+dec) -dec ........ run the only the encoder (default: run enc+dec) InpFile ..... is the name of the file to be processed; OutFile ..... is the name with the processed data; BlockSize ... is the block size, in number of samples (default = 160) 1stBlock .... is the number of the first block of the input file to be processed; NoOfBlocks .. is the number of blocks to be processed, starting on block "1stBlock" Exit values: ~~~~~~~~~~~~ 0 success (all but VMS); 1 success (only in VMS); 2 error opening input file; 3 error creating output file; 4 error moving pointer to desired start of conversion; 5 error creating gsm object (i.e., state variable); 6 error reading input file; 7 error writing to file; Original authors: ~~~~~~~~~~~~~~~~~ Demo program: Simao Ferraz de Campos Neto EMail : simao@cpqd.ansp.br CPqD/Telebras Rd. Mogi Mirim-Campinas Km.118 DDS/Pr.11 13.088-061 - Campinas - SP (Brazil) Module routines: Jutta Degener (jutta@cs.tu-berlin.de) Carsten Bormann (cabo@cs.tu-berlin.de) History: ~~~~~~~~ 28/Oct/92 r.1.0 pl.0 by Carsten Bormann (cabo at kubus) Copyright 1992 by Jutta Degener and Carsten Bormann, Technische Universitaet Berlin. See the accompanying file "COPYRIGHT" for details. THERE IS ABSOLUTELY NO WARRANTY FOR THIS SOFTWARE. 30/Oct/92 r.1.0 pl.1 by Jutta Degener (jutta at kraftbus) Switched uid/gid in toast's [f]chown calls. 29/Jan/93 r.1.0 pl.2 by Jutta Degener (jutta at kraftbus) fixed L_add(0,-1) in src/add.c and inc/private.h, thanks to Raphael Trommer at AT&T Bell Laboratories; * ANSI C compatibility details 20/Mar/94 v1.0 Release of 1st version of demo program 22/Feb/95 v1.1 Cleaned compilation warnings, modified for Alpha VMX/AXP operation (as suggested by Kirchherr, FI/DBP Telekom) <simao@ctd.comsat.com> ============================================================================ */ /* ..... General definitions for UGST demo programs ..... */ #include "ugstdemo.h" /* ..... General include ..... */ #include <stdio.h> #include <stdlib.h> #include <string.h> #if defined(VMS) #include <stat.h> #else /* Unix/DOS */ #include <sys/stat.h> #endif /* ..... RPE-LTP module definitions ..... */ #include "private.h" #include "gsm.h" #include "rpeltp.h" /* ..... G.711 module definitions ..... */ #include "g711.h" /* ..... Local definitions ..... */ #define LINEAR 0 /* binary: 00 */ #define U_LAW 1 /* binary: 01 */ #define A_LAW 3 /* binary: 11 */ /* ..... Local function prototypes ..... */ void display_usage ARGS((void)); int main ARGS((int argc, char *argv[])); /* ------------------------------------------------------------------------- void display_usage(void); ~~~~~~~~~~~~~~~~~~ Display proper usage for the demo program. Generated automatically from program documentation. History: ~~~~~~~~ 20.Mar.94 v1.0 Created. ------------------------------------------------------------------------- */ #define P(x) printf x void display_usage() { P(("RPEDEMO: Version 1.0 of 17/Mar/1994 \n\n")); P((" Demonstration program for UGST/ITU-T RPE-LTP based on \n")); P((" module implemented in Unix-C by Jutta Deneger and Carsten \n")); P((" Borman, within the Communications and Operating Systems \n")); P((" Research Group (KBS) of the Technishe Universitaet Berlin.\n")); P((" This demo program has been written by Simao F.Campos Neto\n")); P(("\n")); P((" Usage:\n")); P((" $ rpedemo [-l|-u|-A] [-enc|-dec] InpFile OutFile BlockSize 1stBlock\n")); P((" NoOfBlocks \n")); P((" where:\n")); P((" -l .......... input data for encoding and output data for decoding\n")); P((" are in linear format (DEFAULT).\n")); P((" -A .......... input data for encoding and output data for decoding\n")); P((" are in A-law (G.711) format.\n")); P((" -u .......... input data for encoding and output data for decoding\n")); P((" are in u-law (G.711) format.\n")); P((" -enc ........ run the only the decoder (default: run enc+dec)\n")); P((" -dec ........ run the only the encoder (default: run enc+dec)\n")); P(("\n")); P((" InpFile ..... is the name of the file to be processed;\n")); P((" OutFile ..... is the name with the processed data;\n")); P((" BlockSize ... is the block size, in number of samples (default = 160)\n")); P((" 1stBlock .... is the number of the first block of the input file\n")); P((" to be processed;\n")); P((" NoOfBlocks .. is the number of blocks to be processed, starting on\n")); P((" block \"1stBlock\"\n")); /* Quit program */ exit(-128); } #undef P /* ************************************************************************** *** *** *** Demo-Program for testing the correct implementation *** *** and to show how to use the programs *** *** *** ************************************************************************** */ main(argc, argv) int argc; char *argv[]; { gsm rpe_enc_state, rpe_dec_state; gsm_signal rpe_frame[RPE_FRAME_SIZE]; Byte gsm_frame[33]; long N = 256, N1 = 1, N2 = 0, cur_blk, smpno, count = 0; #ifdef STATIC_ALLOCATION short tmp_buf[256], inp_buf[256], out_buf[256]; #else short *tmp_buf, *inp_buf, *out_buf; #endif /* G.711 Compression/expansion function pointers */ void (*compress) (), (*expand) (); /* File variables */ char FileIn[80], FileOut[80]; FILE *Fi, *Fo; long start_byte; char format, run_encoder, run_decoder; #ifdef VMS char mrs[15]; #endif /* SETTING DEFAULT OPTIONS */ format = LINEAR; run_encoder = run_decoder = 1; /* GETTING OPTIONS */ if (argc < 2) display_usage(); else { while (argc > 1 && argv[1][0] == '-') if (strcmp(argv[1], "-") == 0) { /* Oops, stop processing options! */ break; } else if (strcmp(argv[1], "-l") == 0) { /* Input/output (uncoded) data format is linear */ format = LINEAR; /* Move arg[cv] over the next valid option */ argv++; argc--; } else if (strcmp(argv[1], "-A") == 0 || strcmp(argv[1], "-a") == 0) { /* Input/output (uncoded) data format is A-law (G.711) */ format = A_LAW; /* Move arg[cv] over the next valid option */ argv++; argc--; } else if (strcmp(argv[1], "-u") == 0) { /* Input/output (uncoded) data format is u-law (G.711) */ format = U_LAW; /* Move arg[cv] over the next valid option */ argv++; argc--; } else if (strcmp(argv[1], "-enc") == 0) { /* Run only the encoder */ run_encoder = 1; run_decoder = 0; /* Move arg[cv] over the next valid option */ argv++; argc--; } else if (strcmp(argv[1], "-dec") == 0) { /* Run only the encoder */ run_encoder = 0; run_decoder = 1; /* Move arg[cv] over the next valid option */ argv++; argc--; } else { fprintf(stderr, "ERROR! Invalid option \"%s\" in command line\n\n", argv[1]); display_usage(); } } GET_PAR_S(1, "_Input File: .................. ", FileIn); GET_PAR_S(2, "_Output File: ................. ", FileOut); FIND_PAR_L(3, "_Block Size: .................. ", N, RPE_WIND_SIZE); FIND_PAR_L(4, "_Starting Block: .............. ", N1, 1); FIND_PAR_L(5, "_No. of Blocks: ............... ", N2, 0); /* Find staring byte in file; all are 16-bit word-aligned =>short data type */ start_byte = sizeof(short) * (long) (--N1) * (long) N; /* Check if is to process the whole file */ if (N2 == 0) { struct stat st; /* ... find the input file size ... */ stat(FileIn, &st); /* convert to block count, depending on whether the input file * is a uncoded or coded file */ if (run_encoder) N2 = (st.st_size - start_byte) / (N * sizeof(short)); else N2 = (st.st_size - start_byte) / (33 * sizeof(Byte)); } /* Choose A/u law */ if (format == A_LAW) { expand = alaw_expand; compress = alaw_compress; } else if (format == U_LAW) { expand = ulaw_expand; compress = ulaw_compress; } /* * ...... MEMORY ALLOCATION ......... */ #ifndef STATIC_ALLOCATION if ((inp_buf = (short *) calloc(N, sizeof(short))) == NULL) HARAKIRI("Error in memory allocation!\n", 1); if ((out_buf = (short *) calloc(N, sizeof(short))) == NULL) HARAKIRI("Error in memory allocation!\n", 1); if ((tmp_buf = (short *) calloc(N, sizeof(short))) == NULL) HARAKIRI("Error in memory allocation!\n", 1); #endif /* * ......... FILE PREPARATION ......... */ /* Define stuff for VMS binary, fixed-record files */ #ifdef VMS sprintf(mrs, "mrs=%d", 512); #endif /* Opening input/output files; abort if there's any problem */ if ((Fi = fopen(FileIn, RB)) == NULL) KILL(FileIn, 2); if ((Fo = fopen(FileOut, WB)) == NULL) KILL(FileOut, 3); /* Move pointer to 1st block of interest */ if (fseek(Fi, start_byte, 0) < 0l) KILL(FileIn, 4); /* ......... CREATE AND INIT GSM OBJECT (STATE VARIABLE) ......... */ if (!(rpe_enc_state = rpeltp_init())) HARAKIRI("Error creating state variable for encoder\n", 5); if (!(rpe_dec_state = rpeltp_init())) HARAKIRI("Error creating state variable for encoder\n", 5); /* ......... PROCESSING ACCORDING TO GSM 06.10 RPE-LTP CODEC ......... */ for (cur_blk = 0; cur_blk < N2; cur_blk++) { /* Reset output sample vector */ memset(out_buf, (int) 0, N); /* Read a block of samples */ if (run_encoder) { /* Reset sample vector */ memset(inp_buf, (int) 0, N); /* Read a block of uncoded samples */ if ((smpno = fread(inp_buf, sizeof(short), (long) N, Fi)) <= 0) break; } else { /* Reset frame vector */ memset(rpe_frame, (int) 0, (long) RPE_FRAME_SIZE); /* Read a packed frame */ if ((smpno = fread(gsm_frame, sizeof(Byte), 33, Fi)) <= 0) { fprintf(stderr, "fread gsm_frame\n"); break; } } /* Carry out the desired operation */ /* CODEC OPERATION */ if (run_encoder && run_decoder) { /* Run both and save decoded samples */ /* First, expand samples, if needed */ if (format) { memcpy(tmp_buf, inp_buf, (long) (sizeof(short) * smpno)); expand(smpno, tmp_buf, inp_buf); } /* Encode & decode ... */ rpeltp_encode(rpe_enc_state, inp_buf, rpe_frame); rpeltp_decode(rpe_dec_state, rpe_frame, out_buf); /* Compress samples, if requested */ if (format) { memcpy(tmp_buf, out_buf, (long) (sizeof(short) * N)); compress(N, tmp_buf, out_buf); } /* Save samples to file */ if (!(smpno = fwrite(out_buf, sizeof(short), (long) smpno, Fo))) break; } /* ENCODER-ONLY OPERATION */ else if (run_encoder) { /* First, expand samples, if needed */ if (format) { memcpy(tmp_buf, inp_buf, (long) (sizeof(short) * smpno)); expand(smpno, tmp_buf, inp_buf); } /* Run only the encoder and save rpe-ltp frame */ gsm_encode(rpe_enc_state, inp_buf, gsm_frame); if (!(smpno = fwrite(gsm_frame, sizeof(Byte), 33, Fo))) break; } /* DECODER-ONLY OPERATION */ else { /* Decode frame */ gsm_decode(rpe_dec_state, gsm_frame, out_buf); /* Compress samples, if requested */ if (format) { memcpy(tmp_buf, out_buf, (long) (sizeof(short) * N)); compress(N, tmp_buf, out_buf); } /* Save the decoded samples */ if (!(smpno = fwrite(out_buf, sizeof(short), (long) N, Fo))) break; } count += smpno; } /* Check for errors */ if (ferror(Fi)) KILL(FileIn, 6); else if (ferror(Fo)) KILL(FileOut, 7); /* ......... FINALIZATIONS ......... */ /* Close input and output files and state */ fclose(Fi); fclose(Fo); rpeltp_delete(rpe_enc_state); rpeltp_delete(rpe_dec_state); /* Exit with success for non-vms systems */ #ifndef VMS return (0); #endif } /* ............................. end of main() ............................. */