FreeCalypso > hg > gsm-codec-lib
view efrtest/etsi-enc.c @ 183:452c1d5a6268
libgsmefr BFI w/o data: emit zero output after decoder reset
In real-life usage, each EFR decoder session will most likely begin
with lots of BFI frames before the first real frame arrives. However,
because the spec-defined home state of the decoder is speech rather
than CN, our regular logic for BFI w/o data would have to feed
pseudorandom noise to the decoder (in the "fixed codebook excitation
pulses" part), which is silly to do at the beginning of the decoder
session right out of reset. Therefore, let's check reset_flag_old,
and if we are still in the reset state, simply emit zero output.
author | Mychaela Falconia <falcon@freecalypso.org> |
---|---|
date | Tue, 03 Jan 2023 00:12:18 +0000 |
parents | a13b1605142b |
children | d4f47d0962e7 |
line wrap: on
line source
/* * gsmefr-etsi-enc is a test program for our EFR encoder: it reads raw * 16-bit PCM (matching ETSI's *.inp) as input and writes ETSI's *.cod * format as output, allowing our decoder to be tested with ETSI's * official test sequences. * * ETSI input and output files are read and written in LE byte order, * matching the official test sequences in ts_100725v050200p0.zip. */ #include <stdio.h> #include <stdint.h> #include <stdlib.h> #include <string.h> #include <strings.h> #include <unistd.h> #include "../libgsmefr/gsm_efr.h" static int read_input(inf, pcm, filename_for_errs, big_endian) FILE *inf; int16_t *pcm; char *filename_for_errs; { uint8_t file_bytes[320], *sp; int cc; unsigned n; cc = fread(file_bytes, 2, 160, inf); if (cc == 0) return 0; if (cc != 160) { fprintf(stderr, "error: short read from %s\n", filename_for_errs); exit(1); } sp = file_bytes; for (n = 0; n < 160; n++) { if (big_endian) pcm[n] = (sp[0] << 8) | sp[1]; else pcm[n] = sp[0] | (sp[1] << 8); sp += 2; } return 1; } static void frame2bits(frame, bits) uint8_t *frame, *bits; { unsigned nb, byte, mask, bit; for (nb = 0; nb < EFR_RTP_FRAME_LEN; nb++) { byte = *frame++; for (mask = 0x80; mask; mask >>= 1) { if (byte & mask) bit = 1; else bit = 0; *bits++ = bit; } } } static void emit_output(outf, bits, nbits, big_endian) FILE *outf; uint8_t *bits; unsigned nbits; { unsigned n; for (n = 0; n < nbits; n++) { if (big_endian) { putc(0, outf); putc(bits[n], outf); } else { putc(bits[n], outf); putc(0, outf); } } } main(argc, argv) char **argv; { char *infname, *outfname; FILE *inf, *outf; struct EFR_encoder_state *state; int16_t pcm[160]; uint8_t frame[EFR_RTP_FRAME_LEN], bits[250]; int opt, dtx = 0, rc, sp, vad, big_endian = 0; extern int optind; while ((opt = getopt(argc, argv, "bd")) != EOF) { switch (opt) { case 'b': big_endian = 1; continue; case 'd': dtx = 1; continue; default: usage: fprintf(stderr, "usage: %s [-b] [-d] input.inp output.cod\n", argv[0]); exit(1); } } if (argc != optind + 2) goto usage; infname = argv[optind]; outfname = argv[optind+1]; inf = fopen(infname, "r"); if (!inf) { perror(infname); exit(1); } outf = fopen(outfname, "w"); if (!outf) { perror(outfname); exit(1); } state = EFR_encoder_create(dtx); if (!state) { perror("EFR_encoder_create()"); exit(1); } for (;;) { rc = read_input(inf, pcm, infname, big_endian); if (!rc) break; EFR_encode_frame(state, pcm, frame, &sp, &vad); frame2bits(frame, bits); bits[248] = vad; bits[249] = sp; emit_output(outf, bits + 4, 246, big_endian); } fclose(outf); exit(0); }