view libgsmfr2/pp_bad.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 96c4ed5529bf
children
line wrap: on
line source

/*
 * In this module we implement our handling of BFI frame gaps
 * and invalid SID frames.
 */

#include <stdint.h>
#include <string.h>
#include "tw_gsmfr.h"
#include "pp_state.h"
#include "pp_internal.h"

static int reduce_xmaxc(uint8_t *frame)
{
	int mute_flag = 1;
	unsigned sub, xmaxc;

	for (sub = 0; sub < 4; sub++) {
		xmaxc = ((frame[sub*7+6] & 0x1F) << 1) | (frame[sub*7+7] >> 7);
		if (xmaxc > 4) {
			xmaxc -= 4;
			mute_flag = 0;
		} else
			xmaxc = 0;
		frame[sub*7+6] &= 0xE0;
		frame[sub*7+6] |= xmaxc >> 1;
		frame[sub*7+7] &= 0x7F;
		frame[sub*7+7] |= (xmaxc & 1) << 7;
	}
	return mute_flag;
}

static void random_grid_pos(struct gsmfr_preproc_state *st, uint8_t *frame)
{
	unsigned sub, Mc;

	for (sub = 0; sub < 4; sub++) {
		Mc = gsmfr_preproc_prng(st, 2);
		frame[sub*7+6] &= 0x9F;
		frame[sub*7+6] |= Mc << 5;
	}
}

static int reduce_xmaxc_sid(struct gsmfr_preproc_state *st)
{
	if (st->sid_xmaxc > 4) {
		st->sid_xmaxc -= 4;
		return 0;
	} else {
		st->sid_xmaxc = 0;
		return 1;
	}
}

void gsmfr_preproc_bfi(struct gsmfr_preproc_state *st, int taf, uint8_t *frame)
{
	int mute;

	switch (st->rx_state) {
	case NO_DATA:
		memcpy(frame, &gsmfr_preproc_silence_frame,
			GSMFR_RTP_FRAME_LEN);
		if (st->dtxd_nodata_count) {
			st->dtxd_sid_flag = 0;
			st->dtxd_nodata_count--;
		} else
			st->dtxd_sid_flag = 1;
		return;
	case SPEECH:
		memcpy(frame, &st->speech_frame, GSMFR_RTP_FRAME_LEN);
		st->rx_state = SPEECH_MUTING;
		st->dtxd_sid_flag = 0;
		return;
	case SPEECH_MUTING:
		mute = reduce_xmaxc(st->speech_frame);
		memcpy(frame, &st->speech_frame, GSMFR_RTP_FRAME_LEN);
		random_grid_pos(st, frame);
		if (mute) {
			st->rx_state = NO_DATA;
			st->dtxd_nodata_count = 0;
		}
		st->dtxd_sid_flag = 0;
		return;
	case COMFORT_NOISE:
		st->dtxd_sid_flag = 1;
		if (taf)
			st->rx_state = LOST_SID;
		gsmfr_preproc_gen_cn(st, frame);
		return;
	case LOST_SID:
		st->dtxd_sid_flag = 1;
		if (taf) {
			st->rx_state = CN_MUTING;
			reduce_xmaxc_sid(st);
			st->dtxd_sid_flag = 0;
		}
		gsmfr_preproc_gen_cn(st, frame);
		return;
	case CN_MUTING:
		st->dtxd_sid_flag = 0;
		if (reduce_xmaxc_sid(st)) {
			st->rx_state = NO_DATA;
			memcpy(frame, &gsmfr_preproc_silence_frame,
				GSMFR_RTP_FRAME_LEN);
			st->dtxd_nodata_count = 3;
		} else
			gsmfr_preproc_gen_cn(st, frame);
		return;
	}
}

void gsmfr_preproc_invalid_sid(struct gsmfr_preproc_state *st, uint8_t *frame)
{
	int mute;

	switch (st->rx_state) {
	case NO_DATA:
		memcpy(frame, &gsmfr_preproc_silence_frame,
			GSMFR_RTP_FRAME_LEN);
		if (st->dtxd_nodata_count) {
			st->dtxd_sid_flag = 0;
			st->dtxd_nodata_count--;
		} else
			st->dtxd_sid_flag = 1;
		return;
	case SPEECH:
		/*
		 * Make CN out of the last good speech frame, following the
		 * "NOTE" at the end of section 6.1.2 in TS 46.031.
		 */
		st->rx_state = COMFORT_NOISE;
		st->dtxd_sid_flag = 1;
		memcpy(st->sid_prefix, &st->speech_frame, 5);
		st->sid_xmaxc = gsmfr_preproc_xmaxc_mean(st->speech_frame);
		gsmfr_preproc_gen_cn(st, frame);
		return;
	case SPEECH_MUTING:
		/* ignore invalid SID in this state and act as if we got BFI */
		mute = reduce_xmaxc(st->speech_frame);
		memcpy(frame, &st->speech_frame, GSMFR_RTP_FRAME_LEN);
		random_grid_pos(st, frame);
		if (mute) {
			st->rx_state = NO_DATA;
			st->dtxd_nodata_count = 0;
		}
		st->dtxd_sid_flag = 0;
		return;
	case COMFORT_NOISE:
	case LOST_SID:
		st->rx_state = COMFORT_NOISE;
		st->dtxd_sid_flag = 1;
		gsmfr_preproc_gen_cn(st, frame);
		return;
	case CN_MUTING:
		st->dtxd_sid_flag = 0;
		if (reduce_xmaxc_sid(st)) {
			st->rx_state = NO_DATA;
			memcpy(frame, &gsmfr_preproc_silence_frame,
				GSMFR_RTP_FRAME_LEN);
			st->dtxd_nodata_count = 3;
		} else
			gsmfr_preproc_gen_cn(st, frame);
		return;
	}
}

void gsmfr_preproc_bfi_bits(struct gsmfr_preproc_state *st,
			    const uint8_t *bad_frame, int taf,
			    uint8_t *frame_out)
{
	if (gsmfr_preproc_sid_classify(bad_frame) == 0)
		gsmfr_preproc_bfi(st, taf, frame_out);
	else
		gsmfr_preproc_invalid_sid(st, frame_out);
}