⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 roqaudioenc.c

📁 ffmpeg的完整源代码和作者自己写的文档。不但有在Linux的工程哦
💻 C
字号:
/* * RoQ audio encoder * * Copyright (c) 2005 Eric Lasota *    Based on RoQ specs (c)2001 Tim Ferguson * * This file is part of FFmpeg. * * FFmpeg is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * FFmpeg is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with FFmpeg; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA */#include "avcodec.h"#include "bytestream.h"#define ROQ_FIRST_FRAME_SIZE     (735*8)#define ROQ_FRAME_SIZE           735#define MAX_DPCM (127*127)static unsigned char dpcmValues[MAX_DPCM];typedef struct{    short lastSample[2];} ROQDPCMContext_t;static void roq_dpcm_table_init(void){    int i;    /* Create a table of quick DPCM values */    for (i=0; i<MAX_DPCM; i++) {        int s= ff_sqrt(i);        int mid= s*s + s;        dpcmValues[i]= s + (i>mid);    }}static int roq_dpcm_encode_init(AVCodecContext *avctx){    ROQDPCMContext_t *context = avctx->priv_data;    if (avctx->channels > 2) {        av_log(avctx, AV_LOG_ERROR, "Audio must be mono or stereo\n");        return -1;    }    if (avctx->sample_rate != 22050) {        av_log(avctx, AV_LOG_ERROR, "Audio must be 22050 Hz\n");        return -1;    }    if (avctx->sample_fmt != SAMPLE_FMT_S16) {        av_log(avctx, AV_LOG_ERROR, "Audio must be signed 16-bit\n");        return -1;    }    roq_dpcm_table_init();    avctx->frame_size = ROQ_FIRST_FRAME_SIZE;    context->lastSample[0] = context->lastSample[1] = 0;    avctx->coded_frame= avcodec_alloc_frame();    avctx->coded_frame->key_frame= 1;    return 0;}static unsigned char dpcm_predict(short *previous, short current){    int diff;    int negative;    int result;    int predicted;    diff = current - *previous;    negative = diff<0;    diff = FFABS(diff);    if (diff >= MAX_DPCM)        result = 127;    else        result = dpcmValues[diff];    /* See if this overflows */ retry:    diff = result*result;    if (negative)        diff = -diff;    predicted = *previous + diff;    /* If it overflows, back off a step */    if (predicted > 32767 || predicted < -32768) {        result--;        goto retry;    }    /* Add the sign bit */    result |= negative << 7;   //if (negative) result |= 128;    *previous = predicted;    return result;}static int roq_dpcm_encode_frame(AVCodecContext *avctx,                unsigned char *frame, int buf_size, void *data){    int i, samples, stereo, ch;    short *in;    unsigned char *out;    ROQDPCMContext_t *context = avctx->priv_data;    stereo = (avctx->channels == 2);    if (stereo) {        context->lastSample[0] &= 0xFF00;        context->lastSample[1] &= 0xFF00;    }    out = frame;    in = data;    bytestream_put_byte(&out, stereo ? 0x21 : 0x20);    bytestream_put_byte(&out, 0x10);    bytestream_put_le32(&out, avctx->frame_size*avctx->channels);    if (stereo) {        bytestream_put_byte(&out, (context->lastSample[1])>>8);        bytestream_put_byte(&out, (context->lastSample[0])>>8);    } else        bytestream_put_le16(&out, context->lastSample[0]);    /* Write the actual samples */    samples = avctx->frame_size;    for (i=0; i<samples; i++)        for (ch=0; ch<avctx->channels; ch++)            *out++ = dpcm_predict(&context->lastSample[ch], *in++);    /* Use smaller frames from now on */    avctx->frame_size = ROQ_FRAME_SIZE;    /* Return the result size */    return out - frame;}static int roq_dpcm_encode_close(AVCodecContext *avctx){    av_freep(&avctx->coded_frame);    return 0;}AVCodec roq_dpcm_encoder = {    "roq_dpcm",    CODEC_TYPE_AUDIO,    CODEC_ID_ROQ_DPCM,    sizeof(ROQDPCMContext_t),    roq_dpcm_encode_init,    roq_dpcm_encode_frame,    roq_dpcm_encode_close,    NULL,};

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -