//========================================================================= // FILENAME : tagutils-asf.c // DESCRIPTION : ASF (wma/wmv) metadata reader //========================================================================= // Copyright (c) 2008- NETGEAR, Inc. All Rights Reserved. //========================================================================= /* * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ static int _asf_read_file_properties(FILE *fp, asf_file_properties_t *p, __u32 size) { int len; len = sizeof(*p) - offsetof(asf_file_properties_t, FileID); if (size < len) return -1; memset(p, 0, sizeof(*p)); p->ID = ASF_FileProperties; p->Size = size; if (len != fread(&p->FileID, 1, len, fp)) return -1; return 0; } static int _asf_read_audio_stream(FILE *fp, struct song_metadata *psong, int size) { asf_audio_stream_t s; int len; len = sizeof(s) - sizeof(s.Hdr); if (len > size) len = size; if (len != fread(&s.wfx, 1, len, fp)) return -1; psong->channels = le16_to_cpu(s.wfx.nChannels); psong->bitrate = le32_to_cpu(s.wfx.nAvgBytesPerSec) * 8; psong->samplerate = le32_to_cpu(s.wfx.nSamplesPerSec); return 0; } static int _asf_read_media_stream(FILE *fp, struct song_metadata *psong, __u32 size) { asf_media_stream_t s; avi_audio_format_t wfx; int len; len = sizeof(s) - sizeof(s.Hdr); if (len > size) len = size; if (len != fread(&s.MajorType, 1, len, fp)) return -1; if (IsEqualGUID(&s.MajorType, &ASF_MediaTypeAudio) && IsEqualGUID(&s.FormatType, &ASF_FormatTypeWave) && s.FormatSize >= sizeof(wfx)) { if (sizeof(wfx) != fread(&wfx, 1, sizeof(wfx), fp)) return -1; psong->channels = le16_to_cpu(wfx.nChannels); psong->bitrate = le32_to_cpu(wfx.nAvgBytesPerSec) * 8; psong->samplerate = le32_to_cpu(wfx.nSamplesPerSec); } return 0; } static int _asf_read_stream_object(FILE *fp, struct song_metadata *psong, __u32 size) { asf_stream_object_t s; int len; len = sizeof(s) - sizeof(asf_object_t); if (size < len) return -1; if (len != fread(&s.StreamType, 1, len, fp)) return -1; if (IsEqualGUID(&s.StreamType, &ASF_AudioStream)) _asf_read_audio_stream(fp, psong, s.TypeSpecificSize); else if (IsEqualGUID(&s.StreamType, &ASF_StreamBufferStream)) _asf_read_media_stream(fp, psong, s.TypeSpecificSize); else { DPRINTF(E_ERROR, L_SCAN_SCANNER, "Unknown asf stream type.\n"); } return 0; } static int _asf_read_extended_stream_object(FILE *fp, struct song_metadata *psong, __u32 size) { int i, len; long off; asf_object_t tmp; asf_extended_stream_object_t xs; asf_stream_name_t nm; asf_payload_extension_t pe; if (size < sizeof(asf_extended_stream_object_t)) return -1; len = sizeof(xs) - offsetof(asf_extended_stream_object_t, StartTime); if (len != fread(&xs.StartTime, 1, len, fp)) return -1; off = sizeof(xs); for (i = 0; i < xs.StreamNameCount; i++) { if (off + sizeof(nm) > size) return -1; if (sizeof(nm) != fread(&nm, 1, sizeof(nm), fp)) return -1; off += sizeof(nm); if (off + nm.Length > sizeof(asf_extended_stream_object_t)) return -1; if (nm.Length > 0) fseek(fp, nm.Length, SEEK_CUR); off += nm.Length; } for (i = 0; i < xs.PayloadExtensionSystemCount; i++) { if (off + sizeof(pe) > size) return -1; if (sizeof(pe) != fread(&pe, 1, sizeof(pe), fp)) return -1; off += sizeof(pe); if (pe.InfoLength > 0) fseek(fp, pe.InfoLength, SEEK_CUR); off += pe.InfoLength; } if (off < size) { if (sizeof(tmp) != fread(&tmp, 1, sizeof(tmp), fp)) return -1; if (IsEqualGUID(&tmp.ID, &ASF_StreamHeader)) _asf_read_stream_object(fp, psong, tmp.Size); } return 0; } static int _asf_read_header_extension(FILE *fp, struct song_metadata *psong, __u32 size) { off_t pos; long off; asf_header_extension_t ext; asf_object_t tmp; if (size < sizeof(asf_header_extension_t)) return -1; fread(&ext.Reserved1, 1, sizeof(ext.Reserved1), fp); ext.Reserved2 = fget_le16(fp); ext.DataSize = fget_le32(fp); pos = ftell(fp); off = 0; while (off < ext.DataSize) { if (sizeof(asf_header_extension_t) + off > size) break; if (sizeof(tmp) != fread(&tmp, 1, sizeof(tmp), fp)) break; if (off + tmp.Size > ext.DataSize) break; if (IsEqualGUID(&tmp.ID, &ASF_ExtendedStreamPropertiesObject)) _asf_read_extended_stream_object(fp, psong, tmp.Size); off += tmp.Size; fseek(fp, pos + off, SEEK_SET); } return 0; } static int _asf_load_string(FILE *fp, int type, int size, char *buf, int len) { unsigned char data[2048]; __u16 wc; int i, j; i = 0; if (size && (size <= sizeof(data)) && (size == fread(data, 1, size, fp))) { switch (type) { case ASF_VT_UNICODE: for (j = 0; j < size; j += 2) { wc = *(__s16*)&data[j]; i += utf16le_to_utf8(&buf[i], len - i, wc); } break; case ASF_VT_BYTEARRAY: for (i = 0; i < size; i++) { if (i + 1 >= len) break; buf[i] = data[i]; } break; case ASF_VT_BOOL: case ASF_VT_DWORD: if (size >= 4) i = snprintf(buf, len, "%d", le32_to_cpu(*(__s32*)&data[0])); break; case ASF_VT_QWORD: if (size >= 8) { #if __WORDSIZE == 64 i = snprintf(buf, len, "%ld", le64_to_cpu(*(__s64*)&data[0])); #else i = snprintf(buf, len, "%lld", le64_to_cpu(*(__s64*)&data[0])); #endif } break; case ASF_VT_WORD: if (size >= 2) i = snprintf(buf, len, "%d", le16_to_cpu(*(__s16*)&data[0])); break; } size = 0; } else fseek(fp, size, SEEK_CUR); buf[i] = 0; return i; } static void* _asf_load_picture(FILE *fp, int size, void *bm, int *bm_size) { int i; char buf[256]; char pic_type; long pic_size; // // Picture type $xx // Data length $xx $xx $xx $xx // MIME type $00 // Description $00 // Picture data pic_type = fget_byte(fp); size -= 1; pic_size = fget_le32(fp); size -= 4; for (i=0; i 0) { if (!(bm = malloc(size))) { DPRINTF(E_ERROR, L_SCAN_SCANNER, "Couldn't allocate %d bytes\n", size); } else { *bm_size = size; if (size <= *bm_size) { fread(bm, 1, size, fp); } else { DPRINTF(E_ERROR, L_SCAN_SCANNER, "Overrun %d bytes required\n", size); free(bm); bm = NULL; } } } else { DPRINTF(E_ERROR, L_SCAN_SCANNER, "No binary data\n"); size = 0; bm = NULL; } } else { DPRINTF(E_ERROR, L_SCAN_SCANNER, "Invalid mime type <%s> in %s\n", buf); } *bm_size = size; return bm; } static int _get_asffileinfo(char *file, struct song_metadata *psong) { FILE *fp; asf_object_t hdr; asf_object_t tmp; unsigned long NumObjects; unsigned short Reserved; unsigned short TitleLength; unsigned short AuthorLength; unsigned short CopyrightLength; unsigned short DescriptionLength; unsigned short RatingLength; unsigned short NumEntries; unsigned short NameLength; unsigned short ValueType; unsigned short ValueLength; off_t pos; char buf[2048]; int mask; asf_file_properties_t FileProperties; psong->vbr_scale = -1; if (!(fp = fopen(file, "rb"))) { DPRINTF(E_ERROR, L_SCAN_SCANNER, "Could not open %s for reading\n",file); return -1; } if (sizeof(hdr) != fread(&hdr, 1, sizeof(hdr), fp)) { DPRINTF(E_ERROR, L_SCAN_SCANNER, "Error reading %s\n", file); fclose(fp); return -1; } hdr.Size = le64_to_cpu(hdr.Size); if (!IsEqualGUID(&hdr.ID, &ASF_HeaderObject)) { DPRINTF(E_ERROR, L_SCAN_SCANNER, "Not a valid header\n"); fclose(fp); return -1; } NumObjects = fget_le32(fp); Reserved = fget_le16(fp); pos = ftell(fp); mask = 0; while (NumObjects > 0) { if (sizeof(tmp) != fread(&tmp, 1, sizeof(tmp), fp)) break; tmp.Size = le64_to_cpu(tmp.Size); if (pos + tmp.Size > hdr.Size) { DPRINTF(E_ERROR, L_SCAN_SCANNER, "Size overrun reading header object %I64x\n", tmp.Size); break; } if (IsEqualGUID(&tmp.ID, &ASF_FileProperties)) { _asf_read_file_properties(fp, &FileProperties, tmp.Size); psong->song_length = le64_to_cpu(FileProperties.PlayDuration) / 10000; psong->bitrate = le64_to_cpu(FileProperties.MaxBitrate); } else if (IsEqualGUID(&tmp.ID, &ASF_ContentDescription)) { TitleLength = fget_le16(fp); AuthorLength = fget_le16(fp); CopyrightLength = fget_le16(fp); DescriptionLength = fget_le16(fp); RatingLength = fget_le16(fp); if (_asf_load_string(fp, ASF_VT_UNICODE, TitleLength, buf, sizeof(buf))) { if (buf[0]) psong->title = strdup(buf); } if (_asf_load_string(fp, ASF_VT_UNICODE, AuthorLength, buf, sizeof(buf))) { if (buf[0]) psong->contributor[ROLE_TRACKARTIST] = strdup(buf); } if (CopyrightLength) fseek(fp, CopyrightLength, SEEK_CUR); if (DescriptionLength) fseek(fp, DescriptionLength, SEEK_CUR); if (RatingLength) fseek(fp, RatingLength, SEEK_CUR); } else if (IsEqualGUID(&tmp.ID, &ASF_ExtendedContentDescription)) { NumEntries = fget_le16(fp); while (NumEntries > 0) { NameLength = fget_le16(fp); _asf_load_string(fp, ASF_VT_UNICODE, NameLength, buf, sizeof(buf)); ValueType = fget_le16(fp); ValueLength = fget_le16(fp); if (!strcasecmp(buf, "AlbumTitle") || !strcasecmp(buf, "WM/AlbumTitle")) { if (_asf_load_string(fp, ValueType, ValueLength, buf, sizeof(buf))) if (buf[0]) psong->album = strdup(buf); } else if (!strcasecmp(buf, "AlbumArtist") || !strcasecmp(buf, "WM/AlbumArtist")) { if (_asf_load_string(fp, ValueType, ValueLength, buf, sizeof(buf))) { if (buf[0]) psong->contributor[ROLE_ALBUMARTIST] = strdup(buf); } } else if (!strcasecmp(buf, "Description") || !strcasecmp(buf, "WM/Track")) { if (_asf_load_string(fp, ValueType, ValueLength, buf, sizeof(buf))) if (buf[0]) psong->track = atoi(buf); } else if (!strcasecmp(buf, "Genre") || !strcasecmp(buf, "WM/Genre")) { if (_asf_load_string(fp, ValueType, ValueLength, buf, sizeof(buf))) if (buf[0]) psong->genre = strdup(buf); } else if (!strcasecmp(buf, "Year") || !strcasecmp(buf, "WM/Year")) { if (_asf_load_string(fp, ValueType, ValueLength, buf, sizeof(buf))) if (buf[0]) psong->year = atoi(buf); } else if (!strcasecmp(buf, "WM/Director")) { if (_asf_load_string(fp, ValueType, ValueLength, buf, sizeof(buf))) if (buf[0]) psong->contributor[ROLE_CONDUCTOR] = strdup(buf); } else if (!strcasecmp(buf, "WM/Picture") && (ValueType == ASF_VT_BYTEARRAY)) { psong->image = _asf_load_picture(fp, ValueLength, psong->image, &psong->image_size); } else if (!strcasecmp(buf, "TrackNumber") || !strcasecmp(buf, "WM/TrackNumber")) { if (_asf_load_string(fp, ValueType, ValueLength, buf, sizeof(buf))) if (buf[0]) psong->track = atoi(buf); } else if (!strcasecmp(buf, "isVBR")) { fseek(fp, ValueLength, SEEK_CUR); psong->vbr_scale = 0; } else if (ValueLength) { fseek(fp, ValueLength, SEEK_CUR); } NumEntries--; } } else if (IsEqualGUID(&tmp.ID, &ASF_StreamHeader)) { _asf_read_stream_object(fp, psong, tmp.Size); } else if (IsEqualGUID(&tmp.ID, &ASF_HeaderExtension)) { _asf_read_header_extension(fp, psong, tmp.Size); } pos += tmp.Size; fseek(fp, pos, SEEK_SET); NumObjects--; } #if 0 if (sizeof(hdr) == fread(&hdr, 1, sizeof(hdr), fp) && IsEqualGUID(&hdr.ID, &ASF_DataObject)) { if (psong->song_length) { psong->bitrate = (hdr.Size * 8000) / psong->song_length; } } #endif fclose(fp); return 0; }