Merge commit 'f78d360bba6dcfb585847a49a84e89c25950fbdb'

* commit 'f78d360bba6dcfb585847a49a84e89c25950fbdb':
  examples/decode_video: use a parser for splitting the input

Merged-by: James Almer <jamrial@gmail.com>
This commit is contained in:
James Almer 2017-04-04 14:34:52 -03:00
commit fddd6af45c
1 changed files with 54 additions and 52 deletions

View File

@ -48,43 +48,47 @@ static void pgm_save(unsigned char *buf, int wrap, int xsize, int ysize,
fclose(f); fclose(f);
} }
static int decode_write_frame(const char *outfilename, AVCodecContext *avctx, static void decode(AVCodecContext *dec_ctx, AVFrame *frame, AVPacket *pkt,
AVFrame *frame, int *frame_count, AVPacket *pkt, int last) const char *filename)
{ {
int len, got_frame;
char buf[1024]; char buf[1024];
int ret, got_picture;
len = avcodec_decode_video2(avctx, frame, &got_frame, pkt); while (pkt->size > 0) {
if (len < 0) { ret = avcodec_decode_video2(dec_ctx, frame, &got_picture, pkt);
fprintf(stderr, "Error while decoding frame %d\n", *frame_count); if (ret < 0) {
return len; fprintf(stderr, "Error while decoding frame %d\n", dec_ctx->frame_number);
exit(1);
} }
if (got_frame) { if (got_picture) {
printf("Saving %sframe %3d\n", last ? "last " : "", *frame_count); printf("saving frame %3d\n", dec_ctx->frame_number);
fflush(stdout); fflush(stdout);
/* the picture is allocated by the decoder, no need to free it */ /* the picture is allocated by the decoder. no need to
snprintf(buf, sizeof(buf), outfilename, *frame_count); free it */
snprintf(buf, sizeof(buf), filename, dec_ctx->frame_number);
pgm_save(frame->data[0], frame->linesize[0], pgm_save(frame->data[0], frame->linesize[0],
frame->width, frame->height, buf); frame->width, frame->height, buf);
(*frame_count)++;
} }
if (pkt->data) { if (pkt->data) {
pkt->size -= len; pkt->size -= ret;
pkt->data += len; pkt->data += ret;
}
} }
return 0;
} }
int main(int argc, char **argv) int main(int argc, char **argv)
{ {
const char *filename, *outfilename; const char *filename, *outfilename;
const AVCodec *codec; const AVCodec *codec;
AVCodecParserContext *parser;
AVCodecContext *c= NULL; AVCodecContext *c= NULL;
int frame_count;
FILE *f; FILE *f;
AVFrame *frame; AVFrame *frame;
uint8_t inbuf[INBUF_SIZE + AV_INPUT_BUFFER_PADDING_SIZE]; uint8_t inbuf[INBUF_SIZE + AV_INPUT_BUFFER_PADDING_SIZE];
uint8_t *data;
size_t data_size;
int ret;
AVPacket avpkt; AVPacket avpkt;
if (argc <= 2) { if (argc <= 2) {
@ -108,15 +112,18 @@ int main(int argc, char **argv)
exit(1); exit(1);
} }
parser = av_parser_init(codec->id);
if (!parser) {
fprintf(stderr, "parser not found\n");
exit(1);
}
c = avcodec_alloc_context3(codec); c = avcodec_alloc_context3(codec);
if (!c) { if (!c) {
fprintf(stderr, "Could not allocate video codec context\n"); fprintf(stderr, "Could not allocate video codec context\n");
exit(1); exit(1);
} }
if (codec->capabilities & AV_CODEC_CAP_TRUNCATED)
c->flags |= AV_CODEC_FLAG_TRUNCATED; // we do not send complete frames
/* For some codecs, such as msmpeg4 and mpeg4, width and height /* For some codecs, such as msmpeg4 and mpeg4, width and height
MUST be initialized there because this information is not MUST be initialized there because this information is not
available in the bitstream. */ available in the bitstream. */
@ -139,42 +146,37 @@ int main(int argc, char **argv)
exit(1); exit(1);
} }
frame_count = 0; while (!feof(f)) {
for (;;) { /* read raw data from the input file */
avpkt.size = fread(inbuf, 1, INBUF_SIZE, f); data_size = fread(inbuf, 1, INBUF_SIZE, f);
if (avpkt.size == 0) if (!data_size)
break; break;
/* NOTE1: some codecs are stream based (mpegvideo, mpegaudio) /* use the parser to split the data into frames */
and this is the only method to use them because you cannot data = inbuf;
know the compressed data size before analysing it. while (data_size > 0) {
ret = av_parser_parse2(parser, c, &avpkt.data, &avpkt.size,
BUT some other codecs (msmpeg4, mpeg4) are inherently frame data, data_size, AV_NOPTS_VALUE, AV_NOPTS_VALUE, 0);
based, so you must call them with all the data for one if (ret < 0) {
frame exactly. You must also initialize 'width' and fprintf(stderr, "Error while parsing\n");
'height' before initializing them. */
/* NOTE2: some codecs allow the raw parameters (frame size,
sample rate) to be changed at any frame. We handle this, so
you should also take care of it */
/* here, we use a stream based decoder (mpeg1video), so we
feed decoder and see if it could decode a frame */
avpkt.data = inbuf;
while (avpkt.size > 0)
if (decode_write_frame(outfilename, c, frame, &frame_count, &avpkt, 0) < 0)
exit(1); exit(1);
} }
data += ret;
data_size -= ret;
/* Some codecs, such as MPEG, transmit the I- and P-frame with a if (avpkt.size)
latency of one frame. You must do the following to have a decode(c, frame, &avpkt, outfilename);
chance to get the last frame of the video. */ }
}
/* flush the decoder */
avpkt.data = NULL; avpkt.data = NULL;
avpkt.size = 0; avpkt.size = 0;
decode_write_frame(outfilename, c, frame, &frame_count, &avpkt, 1); decode(c, frame, &avpkt, outfilename);
fclose(f); fclose(f);
av_parser_close(parser);
avcodec_free_context(&c); avcodec_free_context(&c);
av_frame_free(&frame); av_frame_free(&frame);