This is needed for upcoming ID3v2.3 muxing support. Signed-off-by: Janne Grunau <janne-ffmpeg@jannau.net>tags/n0.8
@@ -273,46 +273,67 @@ void ff_id3v2_read(AVFormatContext *s, const char *magic) | |||
url_fseek(s->pb, off, SEEK_SET); | |||
} | |||
} while (found_header); | |||
ff_metadata_conv(&s->metadata, NULL, ff_id3v2_metadata_conv); | |||
ff_metadata_conv(&s->metadata, NULL, ff_id3v2_34_metadata_conv); | |||
ff_metadata_conv(&s->metadata, NULL, ff_id3v2_2_metadata_conv); | |||
ff_metadata_conv(&s->metadata, NULL, ff_id3v2_4_metadata_conv); | |||
} | |||
const AVMetadataConv ff_id3v2_metadata_conv[] = { | |||
const AVMetadataConv ff_id3v2_34_metadata_conv[] = { | |||
{ "TALB", "album"}, | |||
{ "TAL", "album"}, | |||
{ "TCOM", "composer"}, | |||
{ "TCON", "genre"}, | |||
{ "TCO", "genre"}, | |||
{ "TCOP", "copyright"}, | |||
{ "TDRL", "date"}, | |||
{ "TDRC", "date"}, | |||
{ "TDEN", "creation_time"}, | |||
{ "TENC", "encoded_by"}, | |||
{ "TEN", "encoded_by"}, | |||
{ "TIT2", "title"}, | |||
{ "TT2", "title"}, | |||
{ "TLAN", "language"}, | |||
{ "TPE1", "artist"}, | |||
{ "TP1", "artist"}, | |||
{ "TPE2", "album_artist"}, | |||
{ "TP2", "album_artist"}, | |||
{ "TPE3", "performer"}, | |||
{ "TP3", "performer"}, | |||
{ "TPOS", "disc"}, | |||
{ "TPUB", "publisher"}, | |||
{ "TRCK", "track"}, | |||
{ "TRK", "track"}, | |||
{ "TSSE", "encoder"}, | |||
{ 0 } | |||
}; | |||
const AVMetadataConv ff_id3v2_4_metadata_conv[] = { | |||
{ "TDRL", "date"}, | |||
{ "TDRC", "date"}, | |||
{ "TDEN", "creation_time"}, | |||
{ "TSOA", "album-sort"}, | |||
{ "TSOP", "artist-sort"}, | |||
{ "TSOT", "title-sort"}, | |||
{ "TSSE", "encoder"}, | |||
{ 0 } | |||
}; | |||
const AVMetadataConv ff_id3v2_2_metadata_conv[] = { | |||
{ "TAL", "album"}, | |||
{ "TCO", "genre"}, | |||
{ "TT2", "title"}, | |||
{ "TEN", "encoded_by"}, | |||
{ "TP1", "artist"}, | |||
{ "TP2", "album_artist"}, | |||
{ "TP3", "performer"}, | |||
{ "TRK", "track"}, | |||
{ 0 } | |||
}; | |||
const char ff_id3v2_tags[][4] = { | |||
"TALB", "TBPM", "TCOM", "TCON", "TCOP", "TDEN", "TDLY", "TDOR", "TDRC", | |||
"TDRL", "TDTG", "TENC", "TEXT", "TFLT", "TIPL", "TIT1", "TIT2", "TIT3", | |||
"TKEY", "TLAN", "TLEN", "TMCL", "TMED", "TMOO", "TOAL", "TOFN", "TOLY", | |||
"TOPE", "TOWN", "TPE1", "TPE2", "TPE3", "TPE4", "TPOS", "TPRO", "TPUB", | |||
"TRCK", "TRSN", "TRSO", "TSOA", "TSOP", "TSOT", "TSRC", "TSSE", "TSST", | |||
"TALB", "TBPM", "TCOM", "TCON", "TCOP", "TDLY", "TENC", "TEXT", | |||
"TFLT", "TIT1", "TIT2", "TIT3", "TKEY", "TLAN", "TLEN", "TMED", | |||
"TOAL", "TOFN", "TOLY", "TOPE", "TOWN", "TPE1", "TPE2", "TPE3", | |||
"TPE4", "TPOS", "TPUB", "TRCK", "TRSN", "TRSO", "TSRC", "TSSE", | |||
{ 0 }, | |||
}; | |||
const char ff_id3v2_4_tags[][4] = { | |||
"TDEN", "TDOR", "TDRC", "TDRL", "TDTG", "TIPL", "TMCL", "TMOO", | |||
"TPRO", "TSOA", "TSOP", "TSOT", "TSST", | |||
{ 0 }, | |||
}; | |||
const char ff_id3v2_3_tags[][4] = { | |||
"TDAT", "TIME", "TORY", "TRDA", "TSIZ", "TYER", | |||
{ 0 }, | |||
}; |
@@ -65,12 +65,25 @@ int ff_id3v2_tag_len(const uint8_t *buf); | |||
*/ | |||
void ff_id3v2_read(AVFormatContext *s, const char *magic); | |||
extern const AVMetadataConv ff_id3v2_metadata_conv[]; | |||
extern const AVMetadataConv ff_id3v2_34_metadata_conv[]; | |||
extern const AVMetadataConv ff_id3v2_4_metadata_conv[]; | |||
extern const AVMetadataConv ff_id3v2_2_metadata_conv[]; | |||
/** | |||
* A list of ID3v2.4 text information frames. | |||
* A list of text information frames allowed in both ID3 v2.3 and v2.4 | |||
* http://www.id3.org/id3v2.4.0-frames | |||
* http://www.id3.org/id3v2.4.0-changes | |||
*/ | |||
extern const char ff_id3v2_tags[][4]; | |||
/** | |||
* ID3v2.4-only text information frames. | |||
*/ | |||
extern const char ff_id3v2_4_tags[][4]; | |||
/** | |||
* ID3v2.3-only text information frames. | |||
*/ | |||
extern const char ff_id3v2_3_tags[][4]; | |||
#endif /* AVFORMAT_ID3V2_H */ |
@@ -148,6 +148,20 @@ AVOutputFormat mp2_muxer = { | |||
#endif | |||
#if CONFIG_MP3_MUXER | |||
static int id3v2_check_write_tag(AVFormatContext *s, AVMetadataTag *t, const char table[][4]) | |||
{ | |||
uint32_t tag; | |||
int i; | |||
if (t->key[0] != 'T' || strlen(t->key) != 4) | |||
return -1; | |||
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 -1; | |||
} | |||
/** | |||
* Write an ID3v2.4 header at beginning of stream | |||
*/ | |||
@@ -166,29 +180,25 @@ static int mp3_write_header(struct AVFormatContext *s) | |||
size_pos = url_ftell(s->pb); | |||
put_be32(s->pb, 0); | |||
ff_metadata_conv(&s->metadata, ff_id3v2_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); | |||
while ((t = av_metadata_get(s->metadata, "", t, AV_METADATA_IGNORE_SUFFIX))) { | |||
uint32_t tag = 0; | |||
int ret; | |||
if (t->key[0] == 'T' && strlen(t->key) == 4) { | |||
int i; | |||
for (i = 0; *ff_id3v2_tags[i]; i++) | |||
if (AV_RB32(t->key) == AV_RB32(ff_id3v2_tags[i])) { | |||
tag = AV_RB32(t->key); | |||
if ((ret = id3v2_put_ttag(s, t->value, NULL, tag, ID3v2_ENCODING_UTF8)) < 0) | |||
return ret; | |||
totlen += ret; | |||
break; | |||
} | |||
if ((ret = id3v2_check_write_tag(s, t, ff_id3v2_tags)) > 0) { | |||
totlen += ret; | |||
continue; | |||
} | |||
if (!tag) { /* unknown tag, write as TXXX frame */ | |||
tag = MKBETAG('T', 'X', 'X', 'X'); | |||
if ((ret = id3v2_put_ttag(s, t->key, t->value, tag, ID3v2_ENCODING_UTF8)) < 0) | |||
return ret; | |||
if ((ret = id3v2_check_write_tag(s, t, ff_id3v2_4_tags)) > 0) { | |||
totlen += ret; | |||
continue; | |||
} | |||
/* unknown tag, write as TXXX frame */ | |||
if ((ret = id3v2_put_ttag(s, t->key, t->value, MKBETAG('T', 'X', 'X', 'X'), | |||
ID3v2_ENCODING_UTF8)) < 0) | |||
return ret; | |||
totlen += ret; | |||
} | |||
cur_pos = url_ftell(s->pb); | |||