]> git.sesse.net Git - ffmpeg/blob - libavformat/asf.c
avformat/avio: Add Metacube support
[ffmpeg] / libavformat / asf.c
1 /*
2  * Copyright (c) 2000, 2001 Fabrice Bellard
3  *
4  * This file is part of FFmpeg.
5  *
6  * FFmpeg is free software; you can redistribute it and/or
7  * modify it under the terms of the GNU Lesser General Public
8  * License as published by the Free Software Foundation; either
9  * version 2.1 of the License, or (at your option) any later version.
10  *
11  * FFmpeg is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14  * Lesser General Public License for more details.
15  *
16  * You should have received a copy of the GNU Lesser General Public
17  * License along with FFmpeg; if not, write to the Free Software
18  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
19  */
20
21 #include "asf.h"
22 #include "id3v2.h"
23 #include "internal.h"
24
25 const ff_asf_guid ff_asf_header = {
26     0x30, 0x26, 0xB2, 0x75, 0x8E, 0x66, 0xCF, 0x11, 0xA6, 0xD9, 0x00, 0xAA, 0x00, 0x62, 0xCE, 0x6C
27 };
28
29 const ff_asf_guid ff_asf_file_header = {
30     0xA1, 0xDC, 0xAB, 0x8C, 0x47, 0xA9, 0xCF, 0x11, 0x8E, 0xE4, 0x00, 0xC0, 0x0C, 0x20, 0x53, 0x65
31 };
32
33 const ff_asf_guid ff_asf_stream_header = {
34     0x91, 0x07, 0xDC, 0xB7, 0xB7, 0xA9, 0xCF, 0x11, 0x8E, 0xE6, 0x00, 0xC0, 0x0C, 0x20, 0x53, 0x65
35 };
36
37 const ff_asf_guid ff_asf_ext_stream_header = {
38     0xCB, 0xA5, 0xE6, 0x14, 0x72, 0xC6, 0x32, 0x43, 0x83, 0x99, 0xA9, 0x69, 0x52, 0x06, 0x5B, 0x5A
39 };
40
41 const ff_asf_guid ff_asf_audio_stream = {
42     0x40, 0x9E, 0x69, 0xF8, 0x4D, 0x5B, 0xCF, 0x11, 0xA8, 0xFD, 0x00, 0x80, 0x5F, 0x5C, 0x44, 0x2B
43 };
44
45 const ff_asf_guid ff_asf_audio_conceal_spread = {
46     0x50, 0xCD, 0xC3, 0xBF, 0x8F, 0x61, 0xCF, 0x11, 0x8B, 0xB2, 0x00, 0xAA, 0x00, 0xB4, 0xE2, 0x20
47 };
48
49 const ff_asf_guid ff_asf_video_stream = {
50     0xC0, 0xEF, 0x19, 0xBC, 0x4D, 0x5B, 0xCF, 0x11, 0xA8, 0xFD, 0x00, 0x80, 0x5F, 0x5C, 0x44, 0x2B
51 };
52
53 const ff_asf_guid ff_asf_jfif_media = {
54     0x00, 0xE1, 0x1B, 0xB6, 0x4E, 0x5B, 0xCF, 0x11, 0xA8, 0xFD, 0x00, 0x80, 0x5F, 0x5C, 0x44, 0x2B
55 };
56
57 const ff_asf_guid ff_asf_video_conceal_none = {
58     0x00, 0x57, 0xFB, 0x20, 0x55, 0x5B, 0xCF, 0x11, 0xA8, 0xFD, 0x00, 0x80, 0x5F, 0x5C, 0x44, 0x2B
59 };
60
61 const ff_asf_guid ff_asf_command_stream = {
62     0xC0, 0xCF, 0xDA, 0x59, 0xE6, 0x59, 0xD0, 0x11, 0xA3, 0xAC, 0x00, 0xA0, 0xC9, 0x03, 0x48, 0xF6
63 };
64
65 const ff_asf_guid ff_asf_comment_header = {
66     0x33, 0x26, 0xb2, 0x75, 0x8E, 0x66, 0xCF, 0x11, 0xa6, 0xd9, 0x00, 0xaa, 0x00, 0x62, 0xce, 0x6c
67 };
68
69 const ff_asf_guid ff_asf_codec_comment_header = {
70     0x40, 0x52, 0xD1, 0x86, 0x1D, 0x31, 0xD0, 0x11, 0xA3, 0xA4, 0x00, 0xA0, 0xC9, 0x03, 0x48, 0xF6
71 };
72 const ff_asf_guid ff_asf_codec_comment1_header = {
73     0x41, 0x52, 0xd1, 0x86, 0x1D, 0x31, 0xD0, 0x11, 0xa3, 0xa4, 0x00, 0xa0, 0xc9, 0x03, 0x48, 0xf6
74 };
75
76 const ff_asf_guid ff_asf_data_header = {
77     0x36, 0x26, 0xb2, 0x75, 0x8E, 0x66, 0xCF, 0x11, 0xa6, 0xd9, 0x00, 0xaa, 0x00, 0x62, 0xce, 0x6c
78 };
79
80 const ff_asf_guid ff_asf_head1_guid = {
81     0xb5, 0x03, 0xbf, 0x5f, 0x2E, 0xA9, 0xCF, 0x11, 0x8e, 0xe3, 0x00, 0xc0, 0x0c, 0x20, 0x53, 0x65
82 };
83
84 const ff_asf_guid ff_asf_head2_guid = {
85     0x11, 0xd2, 0xd3, 0xab, 0xBA, 0xA9, 0xCF, 0x11, 0x8e, 0xe6, 0x00, 0xc0, 0x0c, 0x20, 0x53, 0x65
86 };
87
88 const ff_asf_guid ff_asf_extended_content_header = {
89     0x40, 0xA4, 0xD0, 0xD2, 0x07, 0xE3, 0xD2, 0x11, 0x97, 0xF0, 0x00, 0xA0, 0xC9, 0x5E, 0xA8, 0x50
90 };
91
92 const ff_asf_guid ff_asf_simple_index_header = {
93     0x90, 0x08, 0x00, 0x33, 0xB1, 0xE5, 0xCF, 0x11, 0x89, 0xF4, 0x00, 0xA0, 0xC9, 0x03, 0x49, 0xCB
94 };
95
96 const ff_asf_guid ff_asf_ext_stream_embed_stream_header = {
97     0xe2, 0x65, 0xfb, 0x3a, 0xEF, 0x47, 0xF2, 0x40, 0xac, 0x2c, 0x70, 0xa9, 0x0d, 0x71, 0xd3, 0x43
98 };
99
100 const ff_asf_guid ff_asf_ext_stream_audio_stream = {
101     0x9d, 0x8c, 0x17, 0x31, 0xE1, 0x03, 0x28, 0x45, 0xb5, 0x82, 0x3d, 0xf9, 0xdb, 0x22, 0xf5, 0x03
102 };
103
104 const ff_asf_guid ff_asf_metadata_header = {
105     0xea, 0xcb, 0xf8, 0xc5, 0xaf, 0x5b, 0x77, 0x48, 0x84, 0x67, 0xaa, 0x8c, 0x44, 0xfa, 0x4c, 0xca
106 };
107
108 const ff_asf_guid ff_asf_metadata_library_header = {
109     0x94, 0x1c, 0x23, 0x44, 0x98, 0x94, 0xd1, 0x49, 0xa1, 0x41, 0x1d, 0x13, 0x4e, 0x45, 0x70, 0x54
110 };
111
112 const ff_asf_guid ff_asf_marker_header = {
113     0x01, 0xCD, 0x87, 0xF4, 0x51, 0xA9, 0xCF, 0x11, 0x8E, 0xE6, 0x00, 0xC0, 0x0C, 0x20, 0x53, 0x65
114 };
115
116 const ff_asf_guid ff_asf_reserved_4 = {
117         0x20, 0xdb, 0xfe, 0x4c, 0xf6, 0x75, 0xCF, 0x11, 0x9c, 0x0f, 0x00, 0xa0, 0xc9, 0x03, 0x49, 0xcb
118 };
119
120 /* I am not a number !!! This GUID is the one found on the PC used to
121  * generate the stream */
122 const ff_asf_guid ff_asf_my_guid = {
123     0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0
124 };
125
126 const ff_asf_guid ff_asf_language_guid = {
127     0xa9, 0x46, 0x43, 0x7c, 0xe0, 0xef, 0xfc, 0x4b, 0xb2, 0x29, 0x39, 0x3e, 0xde, 0x41, 0x5c, 0x85
128 };
129
130 const ff_asf_guid ff_asf_content_encryption = {
131     0xfb, 0xb3, 0x11, 0x22, 0x23, 0xbd, 0xd2, 0x11, 0xb4, 0xb7, 0x00, 0xa0, 0xc9, 0x55, 0xfc, 0x6e
132 };
133
134 const ff_asf_guid ff_asf_ext_content_encryption = {
135     0x14, 0xe6, 0x8a, 0x29, 0x22, 0x26, 0x17, 0x4c, 0xb9, 0x35, 0xda, 0xe0, 0x7e, 0xe9, 0x28, 0x9c
136 };
137
138 const ff_asf_guid ff_asf_digital_signature = {
139     0xfc, 0xb3, 0x11, 0x22, 0x23, 0xbd, 0xd2, 0x11, 0xb4, 0xb7, 0x00, 0xa0, 0xc9, 0x55, 0xfc, 0x6e
140 };
141
142 const ff_asf_guid ff_asf_extended_stream_properties_object = {
143     0xcb, 0xa5, 0xe6, 0x14, 0x72, 0xc6, 0x32, 0x43, 0x83, 0x99, 0xa9, 0x69, 0x52, 0x06, 0x5b, 0x5a
144 };
145
146 const ff_asf_guid ff_asf_group_mutual_exclusion_object = {
147     0x40, 0x5a, 0x46, 0xd1, 0x79, 0x5a, 0x38, 0x43, 0xb7, 0x1b, 0xe3, 0x6b, 0x8f, 0xd6, 0xc2, 0x49
148 };
149
150 const ff_asf_guid ff_asf_mutex_language = {
151     0x00, 0x2a, 0xe2, 0xd6, 0xda, 0x35, 0xd1, 0x11, 0x90, 0x34, 0x00, 0xa0, 0xc9, 0x03, 0x49, 0xbe
152 };
153
154 /* List of official tags at http://msdn.microsoft.com/en-us/library/dd743066(VS.85).aspx */
155 const AVMetadataConv ff_asf_metadata_conv[] = {
156     { "WM/AlbumArtist",          "album_artist"     },
157     { "WM/AlbumTitle",           "album"            },
158     { "Author",                  "artist"           },
159     { "Description",             "comment"          },
160     { "WM/Composer",             "composer"         },
161     { "WM/EncodedBy",            "encoded_by"       },
162     { "WM/EncodingSettings",     "encoder"          },
163     { "WM/Genre",                "genre"            },
164     { "WM/Language",             "language"         },
165     { "WM/OriginalFilename",     "filename"         },
166     { "WM/PartOfSet",            "disc"             },
167     { "WM/Publisher",            "publisher"        },
168     { "WM/Tool",                 "encoder"          },
169     { "WM/TrackNumber",          "track"            },
170     { "WM/MediaStationCallSign", "service_provider" },
171     { "WM/MediaStationName",     "service_name"     },
172 //  { "Year"               , "date"        }, TODO: conversion year<->date
173     { 0 }
174 };
175
176 /* MSDN claims that this should be "compatible with the ID3 frame, APIC",
177  * but in reality this is only loosely similar */
178 static int asf_read_picture(AVFormatContext *s, int len)
179 {
180     const CodecMime *mime = ff_id3v2_mime_tags;
181     enum  AVCodecID id    = AV_CODEC_ID_NONE;
182     char mimetype[64];
183     uint8_t  *desc = NULL;
184     AVStream   *st = NULL;
185     int ret, type, picsize, desc_len;
186
187     /* type + picsize + mime + desc */
188     if (len < 1 + 4 + 2 + 2) {
189         av_log(s, AV_LOG_ERROR, "Invalid attached picture size: %d.\n", len);
190         return AVERROR_INVALIDDATA;
191     }
192
193     /* picture type */
194     type = avio_r8(s->pb);
195     len--;
196     if (type >= FF_ARRAY_ELEMS(ff_id3v2_picture_types) || type < 0) {
197         av_log(s, AV_LOG_WARNING, "Unknown attached picture type: %d.\n", type);
198         type = 0;
199     }
200
201     /* picture data size */
202     picsize = avio_rl32(s->pb);
203     len    -= 4;
204
205     /* picture MIME type */
206     len -= avio_get_str16le(s->pb, len, mimetype, sizeof(mimetype));
207     while (mime->id != AV_CODEC_ID_NONE) {
208         if (!strncmp(mime->str, mimetype, sizeof(mimetype))) {
209             id = mime->id;
210             break;
211         }
212         mime++;
213     }
214     if (id == AV_CODEC_ID_NONE) {
215         av_log(s, AV_LOG_ERROR, "Unknown attached picture mimetype: %s.\n",
216                mimetype);
217         return 0;
218     }
219
220     if (picsize >= len) {
221         av_log(s, AV_LOG_ERROR, "Invalid attached picture data size: %d >= %d.\n",
222                picsize, len);
223         return AVERROR_INVALIDDATA;
224     }
225
226     /* picture description */
227     desc_len = (len - picsize) * 2 + 1;
228     desc     = av_malloc(desc_len);
229     if (!desc)
230         return AVERROR(ENOMEM);
231     len -= avio_get_str16le(s->pb, len - picsize, desc, desc_len);
232
233     ret = ff_add_attached_pic(s, NULL, s->pb, NULL, picsize);
234     if (ret < 0)
235         goto fail;
236     st = s->streams[s->nb_streams - 1];
237
238     st->codecpar->codec_id        = id;
239
240     if (*desc) {
241         if (av_dict_set(&st->metadata, "title", desc, AV_DICT_DONT_STRDUP_VAL) < 0)
242             av_log(s, AV_LOG_WARNING, "av_dict_set failed.\n");
243     } else
244         av_freep(&desc);
245
246     if (av_dict_set(&st->metadata, "comment", ff_id3v2_picture_types[type], 0) < 0)
247         av_log(s, AV_LOG_WARNING, "av_dict_set failed.\n");
248
249     return 0;
250
251 fail:
252     av_freep(&desc);
253     return ret;
254 }
255
256 static int get_id3_tag(AVFormatContext *s, int len)
257 {
258     ID3v2ExtraMeta *id3v2_extra_meta;
259
260     ff_id3v2_read(s, ID3v2_DEFAULT_MAGIC, &id3v2_extra_meta, len);
261     if (id3v2_extra_meta) {
262         ff_id3v2_parse_apic(s, id3v2_extra_meta);
263         ff_id3v2_parse_chapters(s, id3v2_extra_meta);
264         ff_id3v2_free_extra_meta(&id3v2_extra_meta);
265     }
266     return 0;
267 }
268
269 int ff_asf_handle_byte_array(AVFormatContext *s, const char *name,
270                              int val_len)
271 {
272     if (!strcmp(name, "WM/Picture")) // handle cover art
273         return asf_read_picture(s, val_len);
274     else if (!strcmp(name, "ID3")) // handle ID3 tag
275         return get_id3_tag(s, val_len);
276
277     return 1;
278 }