|
@@ -24,6 +24,7 @@
|
|
|
#include "id3v1.h"
|
|
|
#include "id3v2.h"
|
|
|
#include "libavutil/intreadwrite.h"
|
|
|
+#include "libavutil/opt.h"
|
|
|
|
|
|
static int id3v1_set_string(AVFormatContext *s, const char *key,
|
|
|
uint8_t *buf, int buf_size)
|
|
@@ -148,7 +149,26 @@ AVOutputFormat mp2_muxer = {
|
|
|
#endif
|
|
|
|
|
|
#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;
|
|
|
int i;
|
|
@@ -158,21 +178,23 @@ static int id3v2_check_write_tag(AVFormatContext *s, AVMetadataTag *t, const cha
|
|
|
tag = AV_RB32(t->key);
|
|
|
for (i = 0; *table[i]; 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;
|
|
|
}
|
|
|
|
|
|
|
|
|
- * 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)
|
|
|
{
|
|
|
+ MP3Context *mp3 = s->priv_data;
|
|
|
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;
|
|
|
|
|
|
- put_be32(s->pb, MKBETAG('I', 'D', '3', 0x04));
|
|
|
+ put_be32(s->pb, MKBETAG('I', 'D', '3', mp3->id3v2_version));
|
|
|
put_byte(s->pb, 0);
|
|
|
put_byte(s->pb, 0);
|
|
|
|
|
@@ -181,22 +203,24 @@ static int mp3_write_header(struct AVFormatContext *s)
|
|
|
put_be32(s->pb, 0);
|
|
|
|
|
|
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))) {
|
|
|
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;
|
|
|
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;
|
|
|
continue;
|
|
|
}
|
|
|
|
|
|
|
|
|
- if ((ret = id3v2_put_ttag(s, t->key, t->value, MKBETAG('T', 'X', 'X', 'X'),
|
|
|
- ID3v2_ENCODING_UTF8)) < 0)
|
|
|
+ if ((ret = id3v2_put_ttag(s, t->key, t->value, MKBETAG('T', 'X', 'X', 'X'), enc)) < 0)
|
|
|
return ret;
|
|
|
totlen += ret;
|
|
|
}
|
|
@@ -214,12 +238,13 @@ AVOutputFormat mp3_muxer = {
|
|
|
NULL_IF_CONFIG_SMALL("MPEG audio layer 3"),
|
|
|
"audio/x-mpeg",
|
|
|
"mp3",
|
|
|
- 0,
|
|
|
+ sizeof(MP3Context),
|
|
|
CODEC_ID_MP3,
|
|
|
CODEC_ID_NONE,
|
|
|
mp3_write_header,
|
|
|
mp3_write_packet,
|
|
|
mp3_write_trailer,
|
|
|
AVFMT_NOTIMESTAMPS,
|
|
|
+ .priv_class = &mp3_muxer_class,
|
|
|
};
|
|
|
#endif
|