Mercurial > libavcodec.hg
changeset 636:57b9a37546a0 libavcodec
oggvorbis support patch by (Mark Hills <mark at pogo dot org dot uk>)
author | michaelni |
---|---|
date | Sun, 01 Sep 2002 18:07:56 +0000 |
parents | 3e0f62e5eed6 |
children | de12d5b9c9ad |
files | Makefile allcodecs.c avcodec.h oggvorbis.c oggvorbis.h |
diffstat | 5 files changed, 160 insertions(+), 0 deletions(-) [+] |
line wrap: on
line diff
--- a/Makefile Sun Sep 01 16:52:33 2002 +0000 +++ b/Makefile Sun Sep 01 18:07:56 2002 +0000 @@ -34,6 +34,11 @@ EXTRALIBS += -lmp3lame endif +ifeq ($(CONFIG_VORBIS),yes) +OBJS += oggvorbis.o +EXTRALIBS += -lvorbis -lvorbisenc +endif + ifeq ($(TARGET_GPROF),yes) CFLAGS+=-p LDFLAGS+=-p
--- a/allcodecs.c Sun Sep 01 16:52:33 2002 +0000 +++ b/allcodecs.c Sun Sep 01 18:07:56 2002 +0000 @@ -39,6 +39,9 @@ #ifdef CONFIG_MP3LAME register_avcodec(&mp3lame_encoder); #endif +#ifdef CONFIG_VORBIS + register_avcodec(&oggvorbis_encoder); +#endif register_avcodec(&mpeg1video_encoder); register_avcodec(&h263_encoder); register_avcodec(&h263p_encoder);
--- a/avcodec.h Sun Sep 01 16:52:33 2002 +0000 +++ b/avcodec.h Sun Sep 01 18:07:56 2002 +0000 @@ -15,6 +15,7 @@ CODEC_ID_RV10, CODEC_ID_MP2, CODEC_ID_MP3LAME, + CODEC_ID_VORBIS, CODEC_ID_AC3, CODEC_ID_MJPEG, CODEC_ID_MPEG4, @@ -389,6 +390,7 @@ extern AVCodec ac3_encoder; extern AVCodec mp2_encoder; extern AVCodec mp3lame_encoder; +extern AVCodec oggvorbis_encoder; extern AVCodec mpeg1video_encoder; extern AVCodec h263_encoder; extern AVCodec h263p_encoder;
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/oggvorbis.c Sun Sep 01 18:07:56 2002 +0000 @@ -0,0 +1,140 @@ +/* + * Ogg Vorbis codec support via libvorbisenc + * Mark Hills <mark@pogo.org.uk> + */ + +#include <time.h> + +#include <vorbis/vorbisenc.h> + +#include "avcodec.h" +#include "oggvorbis.h" + +#define OGGVORBIS_FRAME_SIZE 1024 + + +typedef struct OggVorbisContext { + vorbis_info vi ; + vorbis_dsp_state vd ; + vorbis_block vb ; +} OggVorbisContext ; + + +int oggvorbis_init_encoder(vorbis_info *vi, AVCodecContext *avccontext) { + if(avccontext->quality) { /* VBR requested */ + + fprintf(stderr, "init_encode: channels=%d quality=%d\n", + avccontext->channels, avccontext->quality) ; + + return vorbis_encode_init_vbr(vi, avccontext->channels, + avccontext->sample_rate, (float)avccontext->quality / 1000) ; + } + + fprintf(stderr, "init_encoder: channels=%d bitrate=%d tolerance=%d\n", + avccontext->channels, avccontext->bit_rate, + avccontext->bit_rate_tolerance) ; + + return vorbis_encode_init(vi, avccontext->channels, + avccontext->sample_rate, -1, avccontext->bit_rate, -1) ; +} + + +static int oggvorbis_encode_init(AVCodecContext *avccontext) { + OggVorbisContext *context = avccontext->priv_data ; + + fprintf(stderr, "oggvorbis_encode_init\n") ; + + vorbis_info_init(&context->vi) ; + + if(oggvorbis_init_encoder(&context->vi, avccontext) < 0) { + fprintf(stderr, "oggvorbis_encode_init: init_encoder failed") ; + return -1 ; + } + + vorbis_analysis_init(&context->vd, &context->vi) ; + vorbis_block_init(&context->vd, &context->vb) ; + + avccontext->frame_size = OGGVORBIS_FRAME_SIZE ; + + return 0 ; +} + + +int oggvorbis_encode_frame(AVCodecContext *avccontext, unsigned char *packets, + int buf_size, void *data) +{ + OggVorbisContext *context = avccontext->priv_data ; + float **buffer ; + ogg_packet op ; + signed char *audio = data ; + int l, samples = buf_size / 16 ; /* samples = OGGVORBIS_FRAME_SIZE */ ; + + buffer = vorbis_analysis_buffer(&context->vd, samples) ; + + if(context->vi.channels == 1) { + for(l = 0 ; l < samples ; l++) + buffer[0][l]=((audio[l*2+1]<<8)|(0x00ff&(int)audio[l*2]))/32768.f; + } else { + for(l = 0 ; l < samples ; l++){ + buffer[0][l]=((audio[l*4+1]<<8)|(0x00ff&(int)audio[l*4]))/32768.f; + buffer[1][l]=((audio[l*4+3]<<8)|(0x00ff&(int)audio[l*4+2]))/32768.f; + } + } + + vorbis_analysis_wrote(&context->vd, samples) ; + + l = 0 ; + + while(vorbis_analysis_blockout(&context->vd, &context->vb) == 1) { + vorbis_analysis(&context->vb, NULL); + vorbis_bitrate_addblock(&context->vb) ; + + while(vorbis_bitrate_flushpacket(&context->vd, &op)) { + memcpy(packets + l, &op, sizeof(ogg_packet)) ; + memcpy(packets + l + sizeof(ogg_packet), op.packet, op.bytes) ; + l += sizeof(ogg_packet) + op.bytes ; + } + } + + return l ; +} + + +int oggvorbis_encode_close(AVCodecContext *avccontext) { + OggVorbisContext *context = avccontext->priv_data ; +/* ogg_packet op ; */ + + fprintf(stderr, "oggvorbis_encode_close\n") ; + + vorbis_analysis_wrote(&context->vd, 0) ; /* notify vorbisenc this is EOF */ + + /* We need to write all the remaining packets into the stream + * on closing */ + +/* + while(vorbis_bitrate_flushpacket(&context->vd, &op)) { + memcpy(packets + l, &op, sizeof(ogg_packet)) ; + memcpy(packets + l + sizeof(ogg_packet), op.packet, op.bytes) ; + l += sizeof(ogg_packet) + op.bytes ; + } +*/ + + vorbis_block_clear(&context->vb); + vorbis_dsp_clear(&context->vd); + vorbis_info_clear(&context->vi); + + return 0 ; +} + + +AVCodec oggvorbis_encoder = { + "vorbis", + CODEC_TYPE_AUDIO, + CODEC_ID_VORBIS, + sizeof(OggVorbisContext), + oggvorbis_encode_init, + oggvorbis_encode_frame, + oggvorbis_encode_close +}; + +
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/oggvorbis.h Sun Sep 01 18:07:56 2002 +0000 @@ -0,0 +1,10 @@ +#ifndef AVCODEC_OGGVORBIS_H +#define AVCODEC_OGGVORBIS_H + +#include <vorbis/vorbisenc.h> + +#include "avcodec.h" + +int oggvorbis_init_encoder(vorbis_info *vi, AVCodecContext *avccontext) ; + +#endif