aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorClément Bœsch <u@pkh.me>2013-10-19 19:39:28 +0200
committerCarl Eugen Hoyos <cehoyos@ag.or.at>2013-11-18 14:23:52 +0100
commit7ce0f4ea3b206a4f7ce0ceae8751e096b1ebbc9b (patch)
tree4246cc88d0e88cfc02b5eb03f42454798272f7a3
parent3193b85be33347f5cd48f0978092c9b53181defa (diff)
downloadffmpeg-7ce0f4ea3b206a4f7ce0ceae8751e096b1ebbc9b.tar.gz
avformat/image2: allow muxing gif files.
Fixes Ticket #2936. (cherry picked from commit f70db22999d713da3306bf29ec763d670b9bf1ea)
-rw-r--r--libavformat/img2enc.c42
1 files changed, 39 insertions, 3 deletions
diff --git a/libavformat/img2enc.c b/libavformat/img2enc.c
index 8adf3526a6..56aa5fca22 100644
--- a/libavformat/img2enc.c
+++ b/libavformat/img2enc.c
@@ -21,6 +21,7 @@
*/
#include "libavutil/intreadwrite.h"
+#include "libavutil/avassert.h"
#include "libavutil/avstring.h"
#include "libavutil/log.h"
#include "libavutil/opt.h"
@@ -37,6 +38,7 @@ typedef struct {
char path[1024];
int update;
int use_strftime;
+ const char *muxer;
} VideoMuxData;
static int write_header(AVFormatContext *s)
@@ -44,7 +46,6 @@ static int write_header(AVFormatContext *s)
VideoMuxData *img = s->priv_data;
AVStream *st = s->streams[0];
const AVPixFmtDescriptor *desc = av_pix_fmt_desc_get(st->codec->pix_fmt);
- const char *str;
av_strlcpy(img->path, s->filename, sizeof(img->path));
@@ -54,14 +55,18 @@ static int write_header(AVFormatContext *s)
else
img->is_pipe = 1;
- str = strrchr(img->path, '.');
+ if (st->codec->codec_id == AV_CODEC_ID_GIF) {
+ img->muxer = "gif";
+ } else if (st->codec->codec_id == AV_CODEC_ID_RAWVIDEO) {
+ const char *str = strrchr(img->path, '.');
+ /* TODO: reindent */
img->split_planes = str
&& !av_strcasecmp(str + 1, "y")
&& s->nb_streams == 1
- && st->codec->codec_id == AV_CODEC_ID_RAWVIDEO
&& desc
&&(desc->flags & AV_PIX_FMT_FLAG_PLANAR)
&& desc->nb_components >= 3;
+ }
return 0;
}
@@ -124,6 +129,37 @@ static int write_packet(AVFormatContext *s, AVPacket *pkt)
avio_write(pb[3], pkt->data + ysize + 2*usize, ysize);
avio_close(pb[3]);
}
+ } else if (img->muxer) {
+ int ret;
+ AVStream *st;
+ AVPacket pkt2 = {0};
+ AVFormatContext *fmt = NULL;
+
+ av_assert0(!img->split_planes);
+
+ ret = avformat_alloc_output_context2(&fmt, NULL, img->muxer, s->filename);
+ if (ret < 0)
+ return ret;
+ st = avformat_new_stream(fmt, NULL);
+ if (!st) {
+ avformat_free_context(fmt);
+ return AVERROR(ENOMEM);
+ }
+ st->id = pkt->stream_index;
+
+ fmt->pb = pb[0];
+ if ((ret = av_copy_packet(&pkt2, pkt)) < 0 ||
+ (ret = av_dup_packet(&pkt2)) < 0 ||
+ (ret = avcodec_copy_context(st->codec, s->streams[0]->codec)) < 0 ||
+ (ret = avformat_write_header(fmt, NULL)) < 0 ||
+ (ret = av_interleaved_write_frame(fmt, &pkt2)) < 0 ||
+ (ret = av_write_trailer(fmt)) < 0) {
+ av_free_packet(&pkt2);
+ avformat_free_context(fmt);
+ return ret;
+ }
+ av_free_packet(&pkt2);
+ avformat_free_context(fmt);
} else {
avio_write(pb[0], pkt->data, pkt->size);
}