//========================================================================= // 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); /* DLNA Profile Name */ switch( le16_to_cpu(s.wfx.wFormatTag) ) { case WMAV1: case WMAV2: if( (psong->bitrate/1000+1) >= 385 || psong->samplerate > 48000 ) asprintf(&(psong->dlna_pn), "WMAPRO"); else if( (psong->bitrate / 1000)+1 < 192 ) asprintf(&(psong->dlna_pn), "WMABASE"); else asprintf(&(psong->dlna_pn), "WMAFULL"); break; case WMAPRO: asprintf(&(psong->dlna_pn), "WMAPRO"); break; default: break; } 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); /* DLNA Profile Name */ switch( le16_to_cpu(wfx.wFormatTag) ) { case WMAV1: case WMAV2: if( (psong->bitrate/1000+1) >= 385 || psong->samplerate > 48000 ) asprintf(&(psong->dlna_pn), "WMAPRO"); else if( (psong->bitrate / 1000)+1 < 192 ) asprintf(&(psong->dlna_pn), "WMABASE"); else asprintf(&(psong->dlna_pn), "WMAFULL"); break; case WMAPRO: asprintf(&(psong->dlna_pn), "WMAPRO"); break; default: break; } } 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_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 < sizeof(buf) - 1; i++) { buf[i] = fget_le16(fp); size -= 2; if(!buf[i]) break; } buf[i] = '\0'; if(i == sizeof(buf) - 1) { while(fget_le16(fp)) size -= 2; } if(!strcasecmp(buf, "image/jpeg") || !strcasecmp(buf, "image/jpg") || !strcasecmp(buf, "image/peg")) { while(0 != fget_le16(fp)) size -= 2; if(size > 0) { if(!(bm = malloc(size))) { DPRINTF(E_ERROR, L_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_SCANNER, "Overrun %d bytes required\n", size); free(bm); bm = NULL; } } } else { DPRINTF(E_ERROR, L_SCANNER, "No binary data\n"); size = 0; bm = NULL; } } else { DPRINTF(E_ERROR, L_SCANNER, "Invalid mime type %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_SCANNER, "Could not open %s for reading\n", file); return -1; } if(sizeof(hdr) != fread(&hdr, 1, sizeof(hdr), fp)) { DPRINTF(E_ERROR, L_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_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_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; }