Commit 5b7e7544 authored by Anton Khirnov's avatar Anton Khirnov Committed by Michael Niedermayer

mp3enc: support for id3v2.3 tags using a per-muxer AVOption

fixes issue2562.
Signed-off-by: 's avatarJanne Grunau <janne-ffmpeg@jannau.net>
(cherry picked from commit 22272f61)
parent fe01dd8d
...@@ -24,6 +24,7 @@ ...@@ -24,6 +24,7 @@
#include "id3v1.h" #include "id3v1.h"
#include "id3v2.h" #include "id3v2.h"
#include "libavutil/intreadwrite.h" #include "libavutil/intreadwrite.h"
#include "libavutil/opt.h"
static int id3v1_set_string(AVFormatContext *s, const char *key, static int id3v1_set_string(AVFormatContext *s, const char *key,
uint8_t *buf, int buf_size) uint8_t *buf, int buf_size)
...@@ -148,7 +149,26 @@ AVOutputFormat mp2_muxer = { ...@@ -148,7 +149,26 @@ AVOutputFormat mp2_muxer = {
#endif #endif
#if CONFIG_MP3_MUXER #if CONFIG_MP3_MUXER
static int id3v2_check_write_tag(AVFormatContext *s, AVMetadataTag *t, const char table[][4]) typedef struct MP3Context {
const AVClass *class;
int id3v2_version;
} MP3Context;
static const AVOption options[] = {
{ "id3v2_version", "Select ID3v2 version to write. Currently 3 and 4 are supported.",
offsetof(MP3Context, id3v2_version), FF_OPT_TYPE_INT, 4, 3, 4, AV_OPT_FLAG_ENCODING_PARAM},
{ NULL },
};
static const AVClass mp3_muxer_class = {
"MP3 muxer",
av_default_item_name,
options,
LIBAVUTIL_VERSION_INT,
};
static int id3v2_check_write_tag(AVFormatContext *s, AVMetadataTag *t, const char table[][4],
enum ID3v2Encoding enc)
{ {
uint32_t tag; uint32_t tag;
int i; int i;
...@@ -158,21 +178,23 @@ static int id3v2_check_write_tag(AVFormatContext *s, AVMetadataTag *t, const cha ...@@ -158,21 +178,23 @@ static int id3v2_check_write_tag(AVFormatContext *s, AVMetadataTag *t, const cha
tag = AV_RB32(t->key); tag = AV_RB32(t->key);
for (i = 0; *table[i]; i++) for (i = 0; *table[i]; i++)
if (tag == AV_RB32(table[i])) if (tag == AV_RB32(table[i]))
return id3v2_put_ttag(s, t->value, NULL, tag, ID3v2_ENCODING_UTF8); return id3v2_put_ttag(s, t->value, NULL, tag, enc);
return -1; return -1;
} }
/** /**
* Write an ID3v2.4 header at beginning of stream * Write an ID3v2 header at beginning of stream
*/ */
static int mp3_write_header(struct AVFormatContext *s) static int mp3_write_header(struct AVFormatContext *s)
{ {
MP3Context *mp3 = s->priv_data;
AVMetadataTag *t = NULL; AVMetadataTag *t = NULL;
int totlen = 0; int totlen = 0, enc = mp3->id3v2_version == 3 ? ID3v2_ENCODING_UTF16BOM :
ID3v2_ENCODING_UTF8;
int64_t size_pos, cur_pos; int64_t size_pos, cur_pos;
put_be32(s->pb, MKBETAG('I', 'D', '3', 0x04)); /* ID3v2.4 */ put_be32(s->pb, MKBETAG('I', 'D', '3', mp3->id3v2_version));
put_byte(s->pb, 0); put_byte(s->pb, 0);
put_byte(s->pb, 0); /* flags */ put_byte(s->pb, 0); /* flags */
...@@ -181,22 +203,24 @@ static int mp3_write_header(struct AVFormatContext *s) ...@@ -181,22 +203,24 @@ static int mp3_write_header(struct AVFormatContext *s)
put_be32(s->pb, 0); put_be32(s->pb, 0);
ff_metadata_conv(&s->metadata, ff_id3v2_34_metadata_conv, NULL); ff_metadata_conv(&s->metadata, ff_id3v2_34_metadata_conv, NULL);
ff_metadata_conv(&s->metadata, ff_id3v2_4_metadata_conv, NULL); if (mp3->id3v2_version == 4)
ff_metadata_conv(&s->metadata, ff_id3v2_4_metadata_conv, NULL);
while ((t = av_metadata_get(s->metadata, "", t, AV_METADATA_IGNORE_SUFFIX))) { while ((t = av_metadata_get(s->metadata, "", t, AV_METADATA_IGNORE_SUFFIX))) {
int ret; int ret;
if ((ret = id3v2_check_write_tag(s, t, ff_id3v2_tags)) > 0) { if ((ret = id3v2_check_write_tag(s, t, ff_id3v2_tags, enc)) > 0) {
totlen += ret; totlen += ret;
continue; continue;
} }
if ((ret = id3v2_check_write_tag(s, t, ff_id3v2_4_tags)) > 0) { if ((ret = id3v2_check_write_tag(s, t, mp3->id3v2_version == 3 ?
ff_id3v2_3_tags : ff_id3v2_4_tags, enc)) > 0) {
totlen += ret; totlen += ret;
continue; continue;
} }
/* unknown tag, write as TXXX frame */ /* unknown tag, write as TXXX frame */
if ((ret = id3v2_put_ttag(s, t->key, t->value, MKBETAG('T', 'X', 'X', 'X'), if ((ret = id3v2_put_ttag(s, t->key, t->value, MKBETAG('T', 'X', 'X', 'X'), enc)) < 0)
ID3v2_ENCODING_UTF8)) < 0)
return ret; return ret;
totlen += ret; totlen += ret;
} }
...@@ -214,12 +238,13 @@ AVOutputFormat mp3_muxer = { ...@@ -214,12 +238,13 @@ AVOutputFormat mp3_muxer = {
NULL_IF_CONFIG_SMALL("MPEG audio layer 3"), NULL_IF_CONFIG_SMALL("MPEG audio layer 3"),
"audio/x-mpeg", "audio/x-mpeg",
"mp3", "mp3",
0, sizeof(MP3Context),
CODEC_ID_MP3, CODEC_ID_MP3,
CODEC_ID_NONE, CODEC_ID_NONE,
mp3_write_header, mp3_write_header,
mp3_write_packet, mp3_write_packet,
mp3_write_trailer, mp3_write_trailer,
AVFMT_NOTIMESTAMPS, AVFMT_NOTIMESTAMPS,
.priv_class = &mp3_muxer_class,
}; };
#endif #endif
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