mirror of https://github.com/FFmpeg/FFmpeg.git
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
133 lines
3.5 KiB
133 lines
3.5 KiB
/** |
|
Copyright (C) 2005 Måns Rullgård |
|
|
|
Permission is hereby granted, free of charge, to any person |
|
obtaining a copy of this software and associated documentation |
|
files (the "Software"), to deal in the Software without |
|
restriction, including without limitation the rights to use, copy, |
|
modify, merge, publish, distribute, sublicense, and/or sell copies |
|
of the Software, and to permit persons to whom the Software is |
|
furnished to do so, subject to the following conditions: |
|
|
|
The above copyright notice and this permission notice shall be |
|
included in all copies or substantial portions of the Software. |
|
|
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, |
|
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF |
|
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND |
|
NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT |
|
HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, |
|
WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, |
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER |
|
DEALINGS IN THE SOFTWARE. |
|
**/ |
|
|
|
#include <stdlib.h> |
|
#include <theora/theora.h> |
|
#include "avcodec.h" |
|
|
|
typedef struct TheoraContext { |
|
theora_info info; |
|
theora_state state; |
|
theora_comment comment; |
|
ogg_packet op; |
|
} TheoraContext; |
|
|
|
static int |
|
Theora_decode_frame(AVCodecContext *ctx, void *outdata, int *outdata_size, |
|
uint8_t *buf, int buf_size) |
|
{ |
|
TheoraContext *thc = ctx->priv_data; |
|
AVFrame *frame = outdata; |
|
yuv_buffer yuv; |
|
|
|
thc->op.packet = buf; |
|
thc->op.bytes = buf_size; |
|
|
|
if(theora_decode_packetin(&thc->state, &thc->op)) |
|
return -1; |
|
|
|
theora_decode_YUVout(&thc->state, &yuv); |
|
|
|
frame->data[0] = yuv.y; |
|
frame->data[1] = yuv.u; |
|
frame->data[2] = yuv.v; |
|
frame->linesize[0] = yuv.y_stride; |
|
frame->linesize[1] = yuv.uv_stride; |
|
frame->linesize[2] = yuv.uv_stride; |
|
|
|
*outdata_size = sizeof(*frame); |
|
return buf_size; |
|
} |
|
|
|
static int |
|
Theora_decode_end(AVCodecContext *ctx) |
|
{ |
|
TheoraContext *thc = ctx->priv_data; |
|
theora_info_clear(&thc->info); |
|
theora_comment_clear(&thc->comment); |
|
return 0; |
|
} |
|
|
|
static int |
|
Theora_decode_init(AVCodecContext *ctx) |
|
{ |
|
TheoraContext *thc = ctx->priv_data; |
|
int size, hs, i; |
|
ogg_packet op; |
|
uint8_t *cdp; |
|
|
|
if(ctx->extradata_size < 6) |
|
return -1; |
|
|
|
theora_info_init(&thc->info); |
|
|
|
memset(&op, 0, sizeof(op)); |
|
cdp = ctx->extradata; |
|
size = ctx->extradata_size; |
|
|
|
for(i = 0; i < 3; i++){ |
|
hs = *cdp++ << 8; |
|
hs += *cdp++; |
|
size -= 2; |
|
|
|
if(hs > size){ |
|
av_log(ctx, AV_LOG_ERROR, "extradata too small: %i > %i\n", |
|
hs, size); |
|
return -1; |
|
} |
|
|
|
op.packet = cdp; |
|
op.bytes = hs; |
|
op.b_o_s = !i; |
|
if(theora_decode_header(&thc->info, &thc->comment, &op)) |
|
return -1; |
|
op.packetno++; |
|
|
|
cdp += hs; |
|
size -= hs; |
|
} |
|
|
|
theora_decode_init(&thc->state, &thc->info); |
|
|
|
ctx->width = thc->info.width; |
|
ctx->height = thc->info.height; |
|
ctx->time_base.num = thc->info.fps_denominator; |
|
ctx->time_base.den = thc->info.fps_numerator; |
|
ctx->pix_fmt = PIX_FMT_YUV420P; /* FIXME: others are possible */ |
|
|
|
return 0; |
|
} |
|
|
|
AVCodec oggtheora_decoder = { |
|
"theora", |
|
CODEC_TYPE_VIDEO, |
|
CODEC_ID_THEORA, |
|
sizeof(TheoraContext), |
|
Theora_decode_init, |
|
NULL, |
|
Theora_decode_end, |
|
Theora_decode_frame, |
|
0, |
|
NULL |
|
};
|
|
|