Commit c9083d85 authored by Michael Niedermayer's avatar Michael Niedermayer

Merge commit '38d04442'

* commit '38d04442':
  output example: use the new AVFrame API for allocating the video frame

Conflicts:
	doc/examples/muxing.c
Merged-by: 's avatarMichael Niedermayer <michaelni@gmx.at>
parents af75a855 38d04442
...@@ -335,10 +335,32 @@ static void close_audio(AVFormatContext *oc, AVStream *st) ...@@ -335,10 +335,32 @@ static void close_audio(AVFormatContext *oc, AVStream *st)
/**************************************************************/ /**************************************************************/
/* video output */ /* video output */
static AVFrame *frame; static AVFrame *picture, *tmp_picture;
static AVPicture src_picture, dst_picture;
static int frame_count; static int frame_count;
static AVFrame *alloc_picture(enum AVPixelFormat pix_fmt, int width, int height)
{
AVFrame *picture;
int ret;
picture = av_frame_alloc();
if (!picture)
return NULL;
picture->format = pix_fmt;
picture->width = width;
picture->height = height;
/* allocate the buffers for the frame data */
ret = av_frame_get_buffer(picture, 32);
if (ret < 0) {
fprintf(stderr, "Could not allocate frame data.\n");
exit(1);
}
return picture;
}
static void open_video(AVFormatContext *oc, AVCodec *codec, AVStream *st) static void open_video(AVFormatContext *oc, AVCodec *codec, AVStream *st)
{ {
int ret; int ret;
...@@ -352,43 +374,38 @@ static void open_video(AVFormatContext *oc, AVCodec *codec, AVStream *st) ...@@ -352,43 +374,38 @@ static void open_video(AVFormatContext *oc, AVCodec *codec, AVStream *st)
} }
/* allocate and init a re-usable frame */ /* allocate and init a re-usable frame */
frame = av_frame_alloc(); picture = alloc_picture(c->pix_fmt, c->width, c->height);
if (!frame) { if (!picture) {
fprintf(stderr, "Could not allocate video frame\n"); fprintf(stderr, "Could not allocate video frame\n");
exit(1); exit(1);
} }
frame->format = c->pix_fmt;
frame->width = c->width;
frame->height = c->height;
/* Allocate the encoded raw picture. */
ret = avpicture_alloc(&dst_picture, c->pix_fmt, c->width, c->height);
if (ret < 0) {
fprintf(stderr, "Could not allocate picture: %s\n", av_err2str(ret));
exit(1);
}
/* If the output format is not YUV420P, then a temporary YUV420P /* If the output format is not YUV420P, then a temporary YUV420P
* picture is needed too. It is then converted to the required * picture is needed too. It is then converted to the required
* output format. */ * output format. */
tmp_picture = NULL;
if (c->pix_fmt != AV_PIX_FMT_YUV420P) { if (c->pix_fmt != AV_PIX_FMT_YUV420P) {
ret = avpicture_alloc(&src_picture, AV_PIX_FMT_YUV420P, c->width, c->height); tmp_picture = alloc_picture(AV_PIX_FMT_YUV420P, c->width, c->height);
if (ret < 0) { if (!tmp_picture) {
fprintf(stderr, "Could not allocate temporary picture: %s\n", fprintf(stderr, "Could not allocate temporary picture\n");
av_err2str(ret));
exit(1); exit(1);
} }
} }
/* copy data and linesize picture pointers to frame */
*((AVPicture *)frame) = dst_picture;
} }
/* Prepare a dummy image. */ /* Prepare a dummy image. */
static void fill_yuv_image(AVPicture *pict, int frame_index, static void fill_yuv_image(AVFrame *pict, int frame_index,
int width, int height) int width, int height)
{ {
int x, y, i; int x, y, i, ret;
/* when we pass a frame to the encoder, it may keep a reference to it
* internally;
* make sure we do not overwrite it here
*/
ret = av_frame_make_writable(pict);
if (ret < 0)
exit(1);
i = frame_index; i = frame_index;
...@@ -426,12 +443,12 @@ static void write_video_frame(AVFormatContext *oc, AVStream *st, int flush) ...@@ -426,12 +443,12 @@ static void write_video_frame(AVFormatContext *oc, AVStream *st, int flush)
exit(1); exit(1);
} }
} }
fill_yuv_image(&src_picture, frame_count, c->width, c->height); fill_yuv_image(tmp_picture, frame_count, c->width, c->height);
sws_scale(sws_ctx, sws_scale(sws_ctx,
(const uint8_t * const *)src_picture.data, src_picture.linesize, (const uint8_t * const *)tmp_picture->data, tmp_picture->linesize,
0, c->height, dst_picture.data, dst_picture.linesize); 0, c->height, picture->data, picture->linesize);
} else { } else {
fill_yuv_image(&dst_picture, frame_count, c->width, c->height); fill_yuv_image(picture, frame_count, c->width, c->height);
} }
} }
...@@ -442,7 +459,7 @@ static void write_video_frame(AVFormatContext *oc, AVStream *st, int flush) ...@@ -442,7 +459,7 @@ static void write_video_frame(AVFormatContext *oc, AVStream *st, int flush)
pkt.flags |= AV_PKT_FLAG_KEY; pkt.flags |= AV_PKT_FLAG_KEY;
pkt.stream_index = st->index; pkt.stream_index = st->index;
pkt.data = dst_picture.data[0]; pkt.data = (uint8_t*)picture;
pkt.size = sizeof(AVPicture); pkt.size = sizeof(AVPicture);
ret = av_interleaved_write_frame(oc, &pkt); ret = av_interleaved_write_frame(oc, &pkt);
...@@ -452,8 +469,8 @@ static void write_video_frame(AVFormatContext *oc, AVStream *st, int flush) ...@@ -452,8 +469,8 @@ static void write_video_frame(AVFormatContext *oc, AVStream *st, int flush)
av_init_packet(&pkt); av_init_packet(&pkt);
/* encode the image */ /* encode the image */
frame->pts = frame_count; picture->pts = frame_count;
ret = avcodec_encode_video2(c, &pkt, flush ? NULL : frame, &got_packet); ret = avcodec_encode_video2(c, &pkt, flush ? NULL : picture, &got_packet);
if (ret < 0) { if (ret < 0) {
fprintf(stderr, "Error encoding video frame: %s\n", av_err2str(ret)); fprintf(stderr, "Error encoding video frame: %s\n", av_err2str(ret));
exit(1); exit(1);
...@@ -479,9 +496,8 @@ static void write_video_frame(AVFormatContext *oc, AVStream *st, int flush) ...@@ -479,9 +496,8 @@ static void write_video_frame(AVFormatContext *oc, AVStream *st, int flush)
static void close_video(AVFormatContext *oc, AVStream *st) static void close_video(AVFormatContext *oc, AVStream *st)
{ {
avcodec_close(st->codec); avcodec_close(st->codec);
av_free(src_picture.data[0]); av_frame_free(&picture);
av_free(dst_picture.data[0]); av_frame_free(&tmp_picture);
av_frame_free(&frame);
} }
/**************************************************************/ /**************************************************************/
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment