view libgsmfr2/full_dec.c @ 553:ebcf414b7d99

doc/TFO-transform: describe details for FRv1, both modes
author Mychaela Falconia <falcon@freecalypso.org>
date Mon, 07 Oct 2024 08:24:24 +0000
parents 3a617e4e9b27
children
line wrap: on
line source

/*
 * This module implements the "full decoder" functionality of libgsmfr2:
 * first the Rx DTX handler, then the regular GSM 06.10 decoder.  This full
 * decoder also implements the optional homing feature, resetting both
 * components upon receiving DHF.
 */

#include <stdint.h>
#include <stdlib.h>
#include <string.h>
#include "tw_gsmfr.h"
#include "typedef.h"
#include "ed_state.h"
#include "pp_state.h"

struct gsmfr_fulldec_state {
	struct gsmfr_0610_state		dec_0610;
	struct gsmfr_preproc_state	rx_dtx;
	int	is_homed;
};

const unsigned gsmfr_fulldec_state_size = sizeof(struct gsmfr_fulldec_state);

struct gsmfr_fulldec_state *gsmfr_fulldec_create(void)
{
	struct gsmfr_fulldec_state *st;

	st = malloc(sizeof(struct gsmfr_fulldec_state));
	if (st)
		gsmfr_fulldec_reset(st);
	return st;
}

void gsmfr_fulldec_reset(struct gsmfr_fulldec_state *st)
{
	gsmfr_0610_reset(&st->dec_0610);
	gsmfr_preproc_reset(&st->rx_dtx);
	st->is_homed = 1;
}

static void emit_ehf_output(int16_t *pcm_out)
{
	unsigned n;

	for (n = 0; n < 160; n++)
		pcm_out[n] = 0x0008;
}

void gsmfr_fulldec_good_frame(struct gsmfr_fulldec_state *st,
			      const uint8_t *frame_in, int16_t *pcm_out)
{
	uint8_t frame_mod[GSMFR_RTP_FRAME_LEN];

	if (st->is_homed && !memcmp(frame_in, gsmfr_decoder_homing_frame, 12)) {
		emit_ehf_output(pcm_out);
		return;
	}
	memcpy(frame_mod, frame_in, GSMFR_RTP_FRAME_LEN);
	gsmfr_preproc_good_frame(&st->rx_dtx, frame_mod);
	gsmfr_0610_decode_frame(&st->dec_0610, frame_mod, pcm_out);
	if (!memcmp(frame_in, gsmfr_decoder_homing_frame, GSMFR_RTP_FRAME_LEN))
		gsmfr_fulldec_reset(st);
	else
		st->is_homed = 0;
}

void gsmfr_fulldec_bfi(struct gsmfr_fulldec_state *st, int taf,
			int16_t *pcm_out)
{
	uint8_t frame_mod[GSMFR_RTP_FRAME_LEN];

	if (st->is_homed) {
		memset(pcm_out, 0, sizeof(int16_t) * 160);
		return;
	}
	gsmfr_preproc_bfi(&st->rx_dtx, taf, frame_mod);
	gsmfr_0610_decode_frame(&st->dec_0610, frame_mod, pcm_out);
}

void gsmfr_fulldec_bfi_bits(struct gsmfr_fulldec_state *st,
			    const uint8_t *bad_frame, int taf, int16_t *pcm_out)
{
	uint8_t frame_mod[GSMFR_RTP_FRAME_LEN];

	if (st->is_homed) {
		memset(pcm_out, 0, sizeof(int16_t) * 160);
		return;
	}
	gsmfr_preproc_bfi_bits(&st->rx_dtx, bad_frame, taf, frame_mod);
	gsmfr_0610_decode_frame(&st->dec_0610, frame_mod, pcm_out);
}