Unnamed Fossil Project

Artifact [30b92f876e]
Login

Artifact 30b92f876e91c1e719923c43c24930798e55e702:


#include <stdio.h>
#include "all.h"

#include <libavdevice/avdevice.h>
#include <libavcodec/avcodec.h>
#include <libavformat/avformat.h>
#include <speex/speex.h>

static AVFormatContext *fctx = NULL;
static AVCodecContext *dctx = NULL;

static void *estate;
static SpeexBits ebits;
static int speex_frame_samples;

SpeexEchoState *echo_state;

void error2(const char *str, int i) {
    fprintf(stderr, "%s: %i\n", str, i);
    abort();
}

void new_for_speex(const short *data, int samples) {
    static short storage[1000];
    static int nread = 0;
    static int in_packet = 0;

    memcpy(&storage[nread], data, samples*sizeof(data[0]));
    nread += samples;

    while(nread >= speex_frame_samples)
    {
        static short dest[1000];
        char buffer[2000];
        int nb;
        
        recv_output();

        emit_frame();

        speex_echo_cancellation(echo_state, storage, obuffer, dest);

        printf("Encoding one frame\n");
        speex_encode_int(estate, dest, &ebits);
        in_packet++;

        if (in_packet >= 1) {
            nb = speex_bits_nbytes(&ebits);

            /* Sending */
            nb = speex_bits_write(&ebits, buffer, sizeof buffer);

            printf("Sending %d bytes\n", nb);
            send_frame(buffer, nb);
            in_packet = 0;

            /* Reset encoding bits */
            speex_bits_reset(&ebits);
        }

        nread -= speex_frame_samples;
        memmove(storage, &storage[speex_frame_samples], nread*sizeof(storage[0]));
    }
}

void new_input_packet(const AVPacket *p) {
    AVFrame *f = avcodec_alloc_frame();
    int decoded;
    int err;

    f->nb_samples = speex_frame_samples;
    
    err = avcodec_decode_audio4(dctx, f, &decoded, p);
    if (err < 0)
        error2("Cannot decode", err);

    printf("Read packet of %d samples, %s format, %i rate\n", f->nb_samples,
            av_get_sample_fmt_name(f->format), f->sample_rate);
    new_for_speex((const short *) f->data[0], f->nb_samples);

    avcodec_free_frame(&f);
}

void init_input() {
    AVInputFormat *f = av_find_input_format("ALSA");

    if (!f)
        error("failed find ALSA");

    estate = speex_encoder_init(speex_lib_get_mode(SPEEX_MODEID_NB));
    speex_bits_init(&ebits);

    /* Prepare sample rate request */
    AVDictionary *options = NULL;
    av_dict_set(&options, "channels", "1", 0);
    char *rate = (char *) av_malloc(100);
    int val;
    speex_encoder_ctl(estate, SPEEX_GET_SAMPLING_RATE, &val);
    snprintf(rate, 100, "%d", val);
    av_dict_set(&options, "sample_rate", rate, AV_DICT_DONT_STRDUP_VAL);

    printf("Sample rate: %d Hz\n", val);

    speex_encoder_ctl(estate, SPEEX_GET_FRAME_SIZE, &speex_frame_samples);
    printf("Samples wanted by speex: %d\n", speex_frame_samples);

    echo_state = speex_echo_state_init(speex_frame_samples, 1024);

    int err;
    err = avformat_open_input(&fctx, "default", f, &options);
    if (err != 0)
        error("failed open");

    av_dict_free(&options);

    AVCodec *c = NULL;
    int nstream = av_find_best_stream(fctx, AVMEDIA_TYPE_AUDIO,
            -1, -1, &c, 0);
    if (nstream < 0)
        error("failed best stream");

    dctx = fctx->streams[nstream]->codec;

    err = avcodec_open2(dctx, c, NULL);
    if (err != 0)
        error("failed avcodec_open2");
}

void process_input()
{
    int err;
    do {
        AVPacket p;
        av_init_packet(&p);
        err = av_read_frame(fctx, &p);
        if (err == 0) {
            new_input_packet(&p);
            av_free_packet(&p);
        }
    } while(err == 0);

    speex_encoder_destroy(estate);

    avcodec_close(dctx);
    av_free(dctx);
    avformat_close_input(&fctx);
}