summaryrefslogtreecommitdiff
path: root/libavformat/img2enc.c
diff options
context:
space:
mode:
authorClément Bœsch <u@pkh.me>2013-10-19 19:39:28 +0200
committerClément Bœsch <u@pkh.me>2013-11-07 20:30:43 +0100
commitf70db22999d713da3306bf29ec763d670b9bf1ea (patch)
tree015928b83401f10a8f43518d7f1183a1f39fd7d9 /libavformat/img2enc.c
parentd9d752cfb55c445331fd9947100b15e87f65a42d (diff)
avformat/image2: allow muxing gif files.
Fixes Ticket #2936.
Diffstat (limited to 'libavformat/img2enc.c')
-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);
}