/* * $Id$ * Implementation file for mp3 scanner and monitor * * Ironically, this now scans file types other than mp3 files, * but the name is the same for historical purposes, not to mention * the fact that sf.net makes it virtually impossible to manage a cvs * root reasonably. Perhaps one day soon they will move to subversion. * * /me crosses his fingers * * Copyright (C) 2003 Ron Pedde (ron@pedde.com) * * 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 */ #ifdef HAVE_CONFIG_H # include "config.h" #endif #define _POSIX_PTHREAD_SEMANTICS #include #include #include #include #include #include #include #include #include #include /* htons and friends */ #include #include /* why here? For osx 10.2, of course! */ #include "daapd.h" #include "db-memory.h" #include "err.h" #include "mp3-scanner.h" #include "playlist.h" #include "ssc.h" #ifndef HAVE_STRCASESTR # include "strcasestr.h" #endif /* * Typedefs */ /** * Struct to keep info about the information gleaned from * the mp3 frame header. */ typedef struct tag_scan_frameinfo { int layer; /**< 1, 2, or 3, representing Layer I, II, and III */ int bitrate; /**< Bitrate in kbps (128, 64, etc) */ int samplerate; /**< Samplerate (e.g. 44100) */ int stereo; /**< Any kind of stereo.. joint, dual mono, etc */ int frame_length; /**< Frame length in bytes - calculated */ int crc_protected; /**< Is the frame crc protected? */ int samples_per_frame; /**< Samples per frame - calculated field */ int padding; /**< Whether or not there is a padding sample */ int xing_offset; /**< Where the xing header should be relative to end of hdr */ int number_of_frames; /**< Number of frames in the song */ int frame_offset; /**< Where this frame was found */ double version; /**< MPEG version (e.g. 2.0, 2.5, 1.0) */ int is_valid; } SCAN_FRAMEINFO; typedef struct tag_scan_id3header { unsigned char id[3]; unsigned char version[2]; unsigned char flags; unsigned char size[4]; } __attribute((packed)) SCAN_ID3HEADER; #define MAYBEFREE(a) { if((a)) free((a)); }; /* * Globals */ int scan_br_table[5][16] = { { 0,32,64,96,128,160,192,224,256,288,320,352,384,416,448,0 }, /* MPEG1, Layer 1 */ { 0,32,48,56,64,80,96,112,128,160,192,224,256,320,384,0 }, /* MPEG1, Layer 2 */ { 0,32,40,48,56,64,80,96,112,128,160,192,224,256,320,0 }, /* MPEG1, Layer 3 */ { 0,32,48,56,64,80,96,112,128,144,160,176,192,224,256,0 }, /* MPEG2/2.5, Layer 1 */ { 0,8,16,24,32,40,48,56,64,80,96,112,128,144,160,0 } /* MPEG2/2.5, Layer 2/3 */ }; int scan_sample_table[3][4] = { { 44100, 48000, 32000, 0 }, /* MPEG 1 */ { 22050, 24000, 16000, 0 }, /* MPEG 2 */ { 11025, 12000, 8000, 0 } /* MPEG 2.5 */ }; int scan_mode_foreground=1; char *scan_winamp_genre[] = { "Blues", // 0 "Classic Rock", "Country", "Dance", "Disco", "Funk", // 5 "Grunge", "Hip-Hop", "Jazz", "Metal", "New Age", // 10 "Oldies", "Other", "Pop", "R&B", "Rap", // 15 "Reggae", "Rock", "Techno", "Industrial", "Alternative", // 20 "Ska", "Death Metal", "Pranks", "Soundtrack", "Euro-Techno", // 25 "Ambient", "Trip-Hop", "Vocal", "Jazz+Funk", "Fusion", // 30 "Trance", "Classical", "Instrumental", "Acid", "House", // 35 "Game", "Sound Clip", "Gospel", "Noise", "AlternRock", // 40 "Bass", "Soul", "Punk", "Space", "Meditative", // 45 "Instrumental Pop", "Instrumental Rock", "Ethnic", "Gothic", "Darkwave", // 50 "Techno-Industrial", "Electronic", "Pop-Folk", "Eurodance", "Dream", // 55 "Southern Rock", "Comedy", "Cult", "Gangsta", "Top 40", // 60 "Christian Rap", "Pop/Funk", "Jungle", "Native American", "Cabaret", // 65 "New Wave", "Psychadelic", "Rave", "Showtunes", "Trailer", // 70 "Lo-Fi", "Tribal", "Acid Punk", "Acid Jazz", "Polka", // 75 "Retro", "Musical", "Rock & Roll", "Hard Rock", "Folk", // 80 "Folk/Rock", "National folk", "Swing", "Fast-fusion", "Bebob", // 85 "Latin", "Revival", "Celtic", "Bluegrass", "Avantgarde", // 90 "Gothic Rock", "Progressive Rock", "Psychedelic Rock", "Symphonic Rock", "Slow Rock", // 95 "Big Band", "Chorus", "Easy Listening", "Acoustic", "Humour", // 100 "Speech", "Chanson", "Opera", "Chamber Music", "Sonata", // 105 "Symphony", "Booty Bass", "Primus", "Porn Groove", "Satire", // 110 "Slow Jam", "Club", "Tango", "Samba", "Folklore", // 115 "Ballad", "Powder Ballad", "Rhythmic Soul", "Freestyle", "Duet", // 120 "Punk Rock", "Drum Solo", "A Capella", "Euro-House", "Dance Hall", // 125 "Goa", "Drum & Bass", "Club House", "Hardcore", "Terror", // 130 "Indie", "BritPop", "NegerPunk", "Polsk Punk", "Beat", // 135 "Christian Gangsta", "Heavy Metal", "Black Metal", "Crossover", "Contemporary C", // 140 "Christian Rock", "Merengue", "Salsa", "Thrash Metal", "Anime", // 145 "JPop", "SynthPop", "Unknown" }; #define WINAMP_GENRE_UNKNOWN 148 /* * Forwards */ static int scan_path(char *path); static int scan_gettags(char *file, MP3FILE *pmp3); static int scan_get_mp3tags(char *file, MP3FILE *pmp3); static int scan_get_aactags(char *file, MP3FILE *pmp3); static int scan_get_nultags(char *file, MP3FILE *pmp3) { return 0; }; static int scan_get_fileinfo(char *file, MP3FILE *pmp3); static int scan_get_mp3fileinfo(char *file, MP3FILE *pmp3); static int scan_get_aacfileinfo(char *file, MP3FILE *pmp3); static int scan_get_nulfileinfo(char *file, MP3FILE *pmp3) { return 0; }; static int scan_get_urlfileinfo(char *file, MP3FILE *pmp3); static int scan_freetags(MP3FILE *pmp3); static void scan_static_playlist(char *path, struct dirent *pde, struct stat *psb); static void scan_music_file(char *path, struct dirent *pde, struct stat *psb); static int scan_decode_mp3_frame(unsigned char *frame, SCAN_FRAMEINFO *pfi); static time_t mac_to_unix_time(int t); #ifdef OGGVORBIS extern int scan_get_oggfileinfo(char *filename, MP3FILE *pmp3); #endif /* * Typedefs */ typedef struct { char* suffix; int (*tags)(char* file, MP3FILE* pmp3); int (*files)(char* file, MP3FILE* pmp3); } taghandler; static taghandler taghandlers[] = { { "aac", scan_get_aactags, scan_get_aacfileinfo }, { "mp4", scan_get_aactags, scan_get_aacfileinfo }, { "m4a", scan_get_aactags, scan_get_aacfileinfo }, { "m4p", scan_get_aactags, scan_get_aacfileinfo }, { "mp3", scan_get_mp3tags, scan_get_mp3fileinfo }, { "url", scan_get_nultags, scan_get_urlfileinfo }, #ifdef OGGVORBIS { "ogg", scan_get_nultags, scan_get_oggfileinfo }, #endif { NULL, 0 } }; /** * Convert mac time to unix time (different epochs) * * param t time since mac epoch */ time_t mac_to_unix_time(int t) { struct timeval tv; struct timezone tz; gettimeofday(&tv, &tz); return (t - (365L * 66L * 24L * 60L * 60L + 17L * 60L * 60L * 24L) + (tz.tz_minuteswest * 60)); } /* * scan_init * * This assumes the database is already initialized. * * Ideally, this would check to see if the database is empty. * If it is, it sets the database into bulk-import mode, and scans * the MP3 directory. * * If not empty, it would start a background monitor thread * and update files on a file-by-file basis */ int scan_init(char *path) { int err=0; scan_mode_foreground=0; if(db_is_empty()) { scan_mode_foreground=1; } if(db_start_initial_update()) return -1; DPRINTF(E_DBG,L_SCAN,"%s scanning for MP3s in %s\n", scan_mode_foreground ? "Foreground" : "Background", path); err=scan_path(path); if(db_end_initial_update()) return -1; scan_mode_foreground=0; return err; } /* * scan_path * * Do a brute force scan of a path, finding all the MP3 files there */ int scan_path(char *path) { DIR *current_dir; char de[sizeof(struct dirent) + MAXNAMLEN + 1]; /* overcommit for solaris */ struct dirent *pde; int err; char mp3_path[PATH_MAX]; struct stat sb; int modified_time; char *ext; if((current_dir=opendir(path)) == NULL) { DPRINTF(E_WARN,L_SCAN,"opendir: %s\n",strerror(errno)); return -1; } while(1) { if(config.stop) { DPRINTF(E_WARN,L_SCAN,"Stop detected. Aborting scan of %s.\n",path); closedir(current_dir); return 0; } pde=(struct dirent *)&de; err=readdir_r(current_dir,(struct dirent *)de,&pde); if(err == -1) { DPRINTF(E_DBG,L_SCAN,"Error on readdir_r: %s\n",strerror(errno)); err=errno; closedir(current_dir); errno=err; return -1; } if(!pde) break; if(pde->d_name[0] == '.') /* skip hidden and directories */ continue; snprintf(mp3_path,PATH_MAX,"%s/%s",path,pde->d_name); DPRINTF(E_DBG,L_SCAN,"Found %s\n",mp3_path); if(stat(mp3_path,&sb)) { DPRINTF(E_WARN,L_SCAN,"Error statting: %s\n",strerror(errno)); } else { if(sb.st_mode & S_IFDIR) { /* dir -- recurse */ DPRINTF(E_DBG,L_SCAN,"Found dir %s... recursing\n",pde->d_name); scan_path(mp3_path); } else { /* process the file */ if(strlen(pde->d_name) > 4) { if((strcasecmp(".m3u",(char*)&pde->d_name[strlen(pde->d_name) - 4]) == 0) && config.process_m3u){ /* we found an m3u file */ scan_static_playlist(path, pde, &sb); } else if (((ext = strrchr(pde->d_name, '.')) != NULL) && (strcasestr(config.extensions, ext))) { /* only scan if it's been changed, or empty db */ modified_time=sb.st_mtime; DPRINTF(E_DBG,L_SCAN,"FS Mod time: %d\n",modified_time); DPRINTF(E_DBG,L_SCAN,"DB Mod time: %d\n",db_last_modified(sb.st_ino)); if((scan_mode_foreground) || !db_exists(sb.st_ino) || db_last_modified(sb.st_ino) < modified_time) { scan_music_file(path,pde,&sb); } else { DPRINTF(E_DBG,L_SCAN,"Skipping file... not modified\n"); } } } } } } closedir(current_dir); return 0; } /* * scan_static_playlist * * Scan a file as a static playlist */ void scan_static_playlist(char *path, struct dirent *pde, struct stat *psb) { char playlist_path[PATH_MAX]; char m3u_path[PATH_MAX]; char linebuffer[PATH_MAX]; int fd; int playlistid; struct stat sb; DPRINTF(E_WARN,L_SCAN|L_PL,"Processing static playlist: %s\n",pde->d_name); /* see if we should update it */ if(db_playlist_last_modified(psb->st_ino) == psb->st_mtime) return; db_delete_playlist(psb->st_ino); strcpy(m3u_path,pde->d_name); snprintf(playlist_path,sizeof(playlist_path),"%s/%s",path,pde->d_name); m3u_path[strlen(pde->d_name) - 4] = '\0'; playlistid=psb->st_ino; fd=open(playlist_path,O_RDONLY); if(fd != -1) { db_add_playlist(playlistid,m3u_path,psb->st_mtime,0); DPRINTF(E_INF,L_SCAN|L_PL,"Added playlist as id %d\n",playlistid); memset(linebuffer,0x00,sizeof(linebuffer)); while(readline(fd,linebuffer,sizeof(linebuffer)) > 0) { while((linebuffer[strlen(linebuffer)-1] == '\n') || (linebuffer[strlen(linebuffer)-1] == '\r')) /* windows? */ linebuffer[strlen(linebuffer)-1] = '\0'; if((linebuffer[0] == ';') || (linebuffer[0] == '#')) continue; /* FIXME - should chomp trailing comments */ /* otherwise, assume it is a path */ if(linebuffer[0] == '/') { strcpy(m3u_path,linebuffer); } else { snprintf(m3u_path,sizeof(m3u_path),"%s/%s",path,linebuffer); } DPRINTF(E_DBG,L_SCAN|L_PL,"Checking %s\n",m3u_path); /* might be valid, might not... */ if(!stat(m3u_path,&sb)) { /* FIXME: check to see if valid inode! */ db_add_playlist_song(playlistid,sb.st_ino); } else { DPRINTF(E_WARN,L_SCAN|L_PL,"Playlist entry %s bad: %s\n", m3u_path,strerror(errno)); } } close(fd); } DPRINTF(E_WARN,L_SCAN|L_PL,"Done processing playlist\n"); } /* * scan_music_file * * scan a particular file as a music file */ void scan_music_file(char *path, struct dirent *pde, struct stat *psb) { MP3FILE mp3file; char mp3_path[PATH_MAX]; snprintf(mp3_path,sizeof(mp3_path),"%s/%s",path,pde->d_name); /* we found an mp3 file */ DPRINTF(E_INF,L_SCAN,"Found music file: %s\n",pde->d_name); memset((void*)&mp3file,0,sizeof(mp3file)); mp3file.path=strdup(mp3_path); mp3file.fname=strdup(pde->d_name); if(strlen(pde->d_name) > 4) mp3file.type=strdup(strrchr(pde->d_name, '.') + 1); /* FIXME: assumes that st_ino is a u_int_32 DWB: also assumes that the library is contained entirely within one file system */ mp3file.id=psb->st_ino; /* Do the tag lookup here */ if(!scan_gettags(mp3file.path,&mp3file) && !scan_get_fileinfo(mp3file.path,&mp3file)) { make_composite_tags(&mp3file); /* fill in the time_added. I'm not sure of the logic in this. My thinking is to use time created, but what is that? Best guess would be earliest of st_mtime and st_ctime... */ mp3file.time_added=psb->st_mtime; if(psb->st_ctime < mp3file.time_added) mp3file.time_added=psb->st_ctime; mp3file.time_modified=psb->st_mtime; server_side_convert_set(&mp3file); DPRINTF(E_DBG,L_SCAN," Date Added: %d\n",mp3file.time_added); db_add(&mp3file); pl_eval(&mp3file); /* FIXME: move to db_add? */ } else { DPRINTF(E_WARN,L_SCAN,"Skipping %s - scan_gettags failed\n",pde->d_name); } scan_freetags(&mp3file); } /* * scan_aac_findatom * * Find an AAC atom */ long scan_aac_findatom(FILE *fin, long max_offset, char *which_atom, int *atom_size) { long current_offset=0; int size; char atom[4]; while(current_offset < max_offset) { if(fread((void*)&size,1,sizeof(int),fin) != sizeof(int)) return -1; size=ntohl(size); if(size <= 7) /* something not right */ return -1; if(fread(atom,1,4,fin) != 4) return -1; if(strncasecmp(atom,which_atom,4) == 0) { *atom_size=size; return current_offset; } fseek(fin,size-8,SEEK_CUR); current_offset+=size; } return -1; } /* * scan_get_aactags * * Get tags from an AAC (m4a) file */ int scan_get_aactags(char *file, MP3FILE *pmp3) { FILE *fin; long atom_offset; int atom_length; long current_offset=0; int current_size; char current_atom[4]; char *current_data; unsigned short us_data; int genre; int len; if(!(fin=fopen(file,"rb"))) { DPRINTF(E_INF,L_SCAN,"Cannot open file %s for reading\n",file); return -1; } fseek(fin,0,SEEK_SET); atom_offset = aac_drilltoatom(fin, "moov:udta:meta:ilst", &atom_length); if(atom_offset != -1) { /* found the tag section - need to walk through now */ while(current_offset < atom_length) { if(fread((void*)¤t_size,1,sizeof(int),fin) != sizeof(int)) break; current_size=ntohl(current_size); if(current_size <= 7) /* something not right */ break; if(fread(current_atom,1,4,fin) != 4) break; len=current_size-7; /* for ill-formed too-short tags */ if(len < 22) len=22; current_data=(char*)malloc(len); /* extra byte */ memset(current_data,0x00,len); if(fread(current_data,1,current_size-8,fin) != current_size-8) break; if(!memcmp(current_atom,"\xA9" "nam",4)) { /* Song name */ pmp3->title=strdup((char*)¤t_data[16]); } else if(!memcmp(current_atom,"\xA9" "ART",4)) { pmp3->artist=strdup((char*)¤t_data[16]); } else if(!memcmp(current_atom,"\xA9" "alb",4)) { pmp3->album=strdup((char*)¤t_data[16]); } else if(!memcmp(current_atom,"\xA9" "cmt",4)) { pmp3->comment=strdup((char*)¤t_data[16]); } else if(!memcmp(current_atom,"\xA9" "wrt",4)) { pmp3->composer=strdup((char*)¤t_data[16]); } else if(!memcmp(current_atom,"\xA9" "grp",4)) { pmp3->grouping=strdup((char*)¤t_data[16]); } else if(!memcmp(current_atom,"\xA9" "gen",4)) { /* can this be a winamp genre??? */ pmp3->genre=strdup((char*)¤t_data[16]); } else if(!memcmp(current_atom,"tmpo",4)) { us_data=*((unsigned short *)¤t_data[16]); us_data=ntohs(us_data); pmp3->bpm=us_data; } else if(!memcmp(current_atom,"trkn",4)) { us_data=*((unsigned short *)¤t_data[18]); us_data=ntohs(us_data); pmp3->track=us_data; us_data=*((unsigned short *)¤t_data[20]); us_data=ntohs(us_data); pmp3->total_tracks=us_data; } else if(!memcmp(current_atom,"disk",4)) { us_data=*((unsigned short *)¤t_data[18]); us_data=ntohs(us_data); pmp3->disc=us_data; us_data=*((unsigned short *)¤t_data[20]); us_data=ntohs(us_data); pmp3->total_discs=us_data; } else if(!memcmp(current_atom,"\xA9" "day",4)) { pmp3->year=atoi((char*)¤t_data[16]); } else if(!memcmp(current_atom,"gnre",4)) { genre=(int)(*((char*)¤t_data[17])); genre--; if((genre < 0) || (genre > WINAMP_GENRE_UNKNOWN)) genre=WINAMP_GENRE_UNKNOWN; pmp3->genre=strdup(scan_winamp_genre[genre]); } else if (!memcmp(current_atom, "cpil", 4)) { pmp3->compilation = current_data[16]; } free(current_data); current_offset+=current_size; } } fclose(fin); return 0; /* we'll return as much as we got. */ } /* * scan_gettags * * Scan an mp3 file for id3 tags using libid3tag */ int scan_gettags(char *file, MP3FILE *pmp3) { taghandler *hdl; /* dispatch to appropriate tag handler */ for(hdl = taghandlers ; hdl->suffix ; ++hdl) if(!strcasecmp(hdl->suffix, pmp3->type)) break; if(hdl->tags) return hdl->tags(file, pmp3); /* maybe this is an extension that we've manually * specified in the config file, but don't know how * to extract tags from. Ogg, maybe. */ return 0; } int scan_get_mp3tags(char *file, MP3FILE *pmp3) { struct id3_file *pid3file; struct id3_tag *pid3tag; struct id3_frame *pid3frame; int err; int index; int used; unsigned char *utf8_text; int genre=WINAMP_GENRE_UNKNOWN; int have_utf8; int have_text; id3_ucs4_t const *native_text; char *tmp; int got_numeric_genre; if(strcasecmp(pmp3->type,"mp3")) /* can't get tags for non-mp3 */ return 0; pid3file=id3_file_open(file,ID3_FILE_MODE_READONLY); if(!pid3file) { DPRINTF(E_WARN,L_SCAN,"Cannot open %s\n",file); return -1; } pid3tag=id3_file_tag(pid3file); if(!pid3tag) { err=errno; id3_file_close(pid3file); errno=err; DPRINTF(E_WARN,L_SCAN,"Cannot get ID3 tag for %s\n",file); return -1; } index=0; while((pid3frame=id3_tag_findframe(pid3tag,"",index))) { used=0; utf8_text=NULL; native_text=NULL; have_utf8=0; have_text=0; if(!strcmp(pid3frame->id,"YTCP")) { /* for id3v2.2 */ pmp3->compilation = 1; DPRINTF(E_DBG,L_SCAN,"Compilation: %d\n", pmp3->compilation); } if(((pid3frame->id[0] == 'T')||(strcmp(pid3frame->id,"COMM")==0)) && (id3_field_getnstrings(&pid3frame->fields[1]))) have_text=1; if(have_text) { native_text=id3_field_getstrings(&pid3frame->fields[1],0); if(native_text) { have_utf8=1; if(config.latin1_tags) { utf8_text=id3_ucs4_latin1duplicate(native_text); } else { utf8_text=id3_ucs4_utf8duplicate(native_text); } MEMNOTIFY(utf8_text); if(!strcmp(pid3frame->id,"TIT2")) { /* Title */ used=1; pmp3->title = utf8_text; DPRINTF(E_DBG,L_SCAN," Title: %s\n",utf8_text); } else if(!strcmp(pid3frame->id,"TPE1")) { used=1; pmp3->artist = utf8_text; DPRINTF(E_DBG,L_SCAN," Artist: %s\n",utf8_text); } else if(!strcmp(pid3frame->id,"TALB")) { used=1; pmp3->album = utf8_text; DPRINTF(E_DBG,L_SCAN," Album: %s\n",utf8_text); } else if(!strcmp(pid3frame->id,"TCOM")) { used=1; pmp3->composer = utf8_text; DPRINTF(E_DBG,L_SCAN," Composer: %s\n",utf8_text); } else if(!strcmp(pid3frame->id,"TIT1")) { used=1; pmp3->grouping = utf8_text; DPRINTF(E_DBG,L_SCAN," Grouping: %s\n",utf8_text); } else if(!strcmp(pid3frame->id,"TPE2")) { used=1; pmp3->orchestra = utf8_text; DPRINTF(E_DBG,L_SCAN," Orchestra: %s\n",utf8_text); } else if(!strcmp(pid3frame->id,"TPE3")) { used=1; pmp3->conductor = utf8_text; DPRINTF(E_DBG,L_SCAN," Conductor: %s\n",utf8_text); } else if(!strcmp(pid3frame->id,"TCON")) { used=1; pmp3->genre = utf8_text; got_numeric_genre=0; DPRINTF(E_DBG,L_SCAN," Genre: %s\n",utf8_text); if(pmp3->genre) { if(!strlen(pmp3->genre)) { genre=WINAMP_GENRE_UNKNOWN; got_numeric_genre=1; } else if (isdigit(pmp3->genre[0])) { genre=atoi(pmp3->genre); got_numeric_genre=1; } else if ((pmp3->genre[0] == '(') && (isdigit(pmp3->genre[1]))) { genre=atoi((char*)&pmp3->genre[1]); got_numeric_genre=1; } if(got_numeric_genre) { if((genre < 0) || (genre > WINAMP_GENRE_UNKNOWN)) genre=WINAMP_GENRE_UNKNOWN; free(pmp3->genre); pmp3->genre=strdup(scan_winamp_genre[genre]); } } } else if(!strcmp(pid3frame->id,"COMM")) { used=1; pmp3->comment = utf8_text; DPRINTF(E_DBG,L_SCAN," Comment: %s\n",pmp3->comment); } else if(!strcmp(pid3frame->id,"TPOS")) { tmp=(char*)utf8_text; strsep(&tmp,"/"); if(tmp) { pmp3->total_discs=atoi(tmp); } pmp3->disc=atoi((char*)utf8_text); DPRINTF(E_DBG,L_SCAN," Disc %d of %d\n",pmp3->disc,pmp3->total_discs); } else if(!strcmp(pid3frame->id,"TRCK")) { tmp=(char*)utf8_text; strsep(&tmp,"/"); if(tmp) { pmp3->total_tracks=atoi(tmp); } pmp3->track=atoi((char*)utf8_text); DPRINTF(E_DBG,L_SCAN," Track %d of %d\n",pmp3->track,pmp3->total_tracks); } else if(!strcmp(pid3frame->id,"TDRC")) { pmp3->year = atoi(utf8_text); DPRINTF(E_DBG,L_SCAN," Year: %d\n",pmp3->year); } else if(!strcmp(pid3frame->id,"TLEN")) { pmp3->song_length = atoi(utf8_text); /* now in ms */ DPRINTF(E_DBG,L_SCAN," Length: %d\n", pmp3->song_length); } else if(!strcmp(pid3frame->id,"TBPM")) { pmp3->bpm = atoi(utf8_text); DPRINTF(E_DBG,L_SCAN,"BPM: %d\n", pmp3->bpm); } else if(!strcmp(pid3frame->id,"TCMP")) { /* for id3v2.3 */ pmp3->compilation = (char)atoi(utf8_text); DPRINTF(E_DBG,L_SCAN,"Compilation: %d\n", pmp3->compilation); } } } /* can check for non-text tags here */ if((!used) && (have_utf8) && (utf8_text)) free(utf8_text); /* v2 COMM tags are a bit different than v1 */ if((!strcmp(pid3frame->id,"COMM")) && (pid3frame->nfields == 4)) { /* Make sure it isn't a application-specific comment... * This currently includes the following: * * iTunes_CDDB_IDs * iTunNORM * * If other apps stuff crap into comment fields, then we'll ignore them * here. */ native_text=id3_field_getstring(&pid3frame->fields[2]); if(native_text) { utf8_text=id3_ucs4_utf8duplicate(native_text); if((utf8_text) && (strncasecmp(utf8_text,"iTun",4) != 0)) { /* it's a real comment */ if(utf8_text) free(utf8_text); native_text=id3_field_getfullstring(&pid3frame->fields[3]); if(native_text) { if(pmp3->comment) free(pmp3->comment); utf8_text=id3_ucs4_utf8duplicate(native_text); if(utf8_text) { pmp3->comment=utf8_text; MEMNOTIFY(pmp3->comment); } } } else { if(utf8_text) free(utf8_text); } } } index++; } id3_file_close(pid3file); DPRINTF(E_DBG,L_SCAN,"Got id3 tag successfully\n"); return 0; } /* * scan_freetags * * Free up the tags that were dynamically allocated */ int scan_freetags(MP3FILE *pmp3) { MAYBEFREE(pmp3->path); MAYBEFREE(pmp3->fname); MAYBEFREE(pmp3->title); MAYBEFREE(pmp3->artist); MAYBEFREE(pmp3->album); MAYBEFREE(pmp3->genre); MAYBEFREE(pmp3->comment); MAYBEFREE(pmp3->type); MAYBEFREE(pmp3->composer); MAYBEFREE(pmp3->orchestra); MAYBEFREE(pmp3->conductor); MAYBEFREE(pmp3->grouping); MAYBEFREE(pmp3->description); return 0; } /* * scan_get_fileinfo * * Dispatch to actual file info handlers */ int scan_get_fileinfo(char *file, MP3FILE *pmp3) { FILE *infile; off_t file_size; taghandler *hdl; /* dispatch to appropriate tag handler */ for(hdl = taghandlers ; hdl->suffix ; ++hdl) if(!strcasecmp(hdl->suffix, pmp3->type)) break; if(hdl->files) return hdl->files(file, pmp3); /* a file we don't know anything about... ogg or aiff maybe */ if(!(infile=fopen(file,"rb"))) { DPRINTF(E_WARN,L_SCAN,"Could not open %s for reading\n",file); return -1; } /* we can at least get this */ fseek(infile,0,SEEK_END); file_size=ftell(infile); fseek(infile,0,SEEK_SET); pmp3->file_size=file_size; fclose(infile); return 0; } /* * aac_drilltoatom * * Returns the offset of the atom specified by the given path or -1 if * not found. atom_path is a colon separated list of atoms specifying * a path from parent node to the target node. All paths must be specified * from the root. */ off_t aac_drilltoatom(FILE *aac_fp, char *atom_path, unsigned int *atom_length) { long atom_offset; off_t file_size; char *cur_p, *end_p; char atom_name[5]; fseek(aac_fp, 0, SEEK_END); file_size = ftell(aac_fp); rewind(aac_fp); end_p = atom_path; while (*end_p != '\0') { end_p++; } atom_name[4] = '\0'; cur_p = atom_path; while (cur_p != NULL) { if ((end_p - cur_p) < 4) { return -1; } strncpy(atom_name, cur_p, 4); atom_offset = scan_aac_findatom(aac_fp, file_size, atom_name, atom_length); if (atom_offset == -1) { return -1; } DPRINTF(E_DBG,L_SCAN,"Found %s atom at off %ld.\n", atom_name, ftell(aac_fp) - 8); cur_p = strchr(cur_p, ':'); if (cur_p != NULL) { cur_p++; /* PENDING: Hack to deal with atoms that have extra data in addition to having child atoms. This should be dealt in a better fashion than this (table with skip offsets or an actual real mp4 parser.) */ if (!strcmp(atom_name, "meta")) { fseek(aac_fp, 4, SEEK_CUR); } else if (!strcmp(atom_name, "stsd")) { fseek(aac_fp, 8, SEEK_CUR); } else if (!strcmp(atom_name, "mp4a")) { fseek(aac_fp, 28, SEEK_CUR); } } } return ftell(aac_fp) - 8; } /* * scan_get_urlfileinfo * * Get info from a "url" file -- a media stream file */ int scan_get_urlfileinfo(char *file, MP3FILE *pmp3) { FILE *infile; char *head, *tail; char linebuffer[256]; DPRINTF(E_DBG,L_SCAN,"Getting URL file info\n"); if(!(infile=fopen(file,"rb"))) { DPRINTF(E_WARN,L_SCAN,"Could not open %s for reading\n",file); return -1; } fgets(linebuffer,sizeof(linebuffer),infile); while((linebuffer[strlen(linebuffer)-1] == '\n') || (linebuffer[strlen(linebuffer)-1] == '\r')) { linebuffer[strlen(linebuffer)-1] = '\0'; } head=linebuffer; tail=strchr(head,','); if(!tail) { DPRINTF(E_LOG,L_SCAN,"Badly formatted .url file - must be bitrate,descr,url\n"); fclose(infile); return -1; } pmp3->bitrate=atoi(head); head=++tail; tail=strchr(head,','); if(!tail) { DPRINTF(E_LOG,L_SCAN,"Badly formatted .url file - must be bitrate,descr,url\n"); fclose(infile); return -1; } *tail++='\0'; pmp3->title=strdup(head); pmp3->url=strdup(tail); fclose(infile); DPRINTF(E_DBG,L_SCAN," Title: %s\n",pmp3->title); DPRINTF(E_DBG,L_SCAN," Bitrate: %d\n",pmp3->bitrate); DPRINTF(E_DBG,L_SCAN," URL: %s\n",pmp3->url); return 0; } /* * scan_get_aacfileinfo * * Get info from the actual aac headers */ int scan_get_aacfileinfo(char *file, MP3FILE *pmp3) { FILE *infile; long atom_offset; int atom_length; int sample_size; int samples; unsigned int bit_rate; off_t file_size; int ms; unsigned char buffer[2]; int time = 0; DPRINTF(E_DBG,L_SCAN,"Getting AAC file info\n"); if(!(infile=fopen(file,"rb"))) { DPRINTF(E_WARN,L_SCAN,"Could not open %s for reading\n",file); return -1; } fseek(infile,0,SEEK_END); file_size=ftell(infile); fseek(infile,0,SEEK_SET); pmp3->file_size=file_size; /* now, hunt for the mvhd atom */ atom_offset = aac_drilltoatom(infile, "moov:mvhd", &atom_length); if(atom_offset != -1) { fseek(infile, 4, SEEK_CUR); fread((void *)&time, sizeof(int), 1, infile); time = ntohl(time); pmp3->time_added = mac_to_unix_time(time); fread((void *)&time, sizeof(int), 1, infile); time = ntohl(time); pmp3->time_modified = mac_to_unix_time(time); fread((void*)&sample_size,1,sizeof(int),infile); fread((void*)&samples,1,sizeof(int),infile); sample_size=ntohl(sample_size); samples=ntohl(samples); /* avoid overflowing on large sample_sizes (90000) */ ms=1000; while((ms > 9) && (!(sample_size % 10))) { sample_size /= 10; ms /= 10; } /* DWB: use ms time instead of sec */ pmp3->song_length=(int)((samples * ms) / sample_size); DPRINTF(E_DBG,L_SCAN,"Song length: %d seconds\n", pmp3->song_length / 1000); } pmp3->bitrate = 0; /* Get the sample rate from the 'mp4a' atom (timescale). This is also found in the 'mdhd' atom which is a bit closer but we need to navigate to the 'mp4a' atom anyways to get to the 'esds' atom. */ atom_offset = aac_drilltoatom(infile, "moov:trak:mdia:minf:stbl:stsd:mp4a", &atom_length); if (atom_offset != -1) { fseek(infile, atom_offset + 32, SEEK_SET); /* Timescale here seems to be 2 bytes here (the 2 bytes before it are "reserved") though the timescale in the 'mdhd' atom is 4. Not sure how this is dealt with when sample rate goes higher than 64K. */ fread(buffer, sizeof(unsigned char), 2, infile); pmp3->samplerate = (buffer[0] << 8) | (buffer[1]); /* Seek to end of atom. */ fseek(infile, 2, SEEK_CUR); /* Get the bit rate from the 'esds' atom. We are already positioned in the parent atom so just scan ahead. */ atom_offset = scan_aac_findatom(infile, atom_length - (ftell(infile) - atom_offset), "esds", &atom_length); if (atom_offset != -1) { fseek(infile, atom_offset + 22, SEEK_CUR); fread((void *)&bit_rate, sizeof(unsigned int), 1, infile); pmp3->bitrate = ntohl(bit_rate) / 1000; } else { DPRINTF(E_DBG,L_SCAN, "Could not find 'esds' atom to determine bit rate.\n"); } } else { DPRINTF(E_DBG,L_SCAN, "Could not find 'mp4a' atom to determine sample rate.\n"); } /* Fallback if we can't find the info in the atoms. */ if (pmp3->bitrate == 0) { /* calculate bitrate from song length... Kinda cheesy */ DPRINTF(E_DBG,L_SCAN, "Could not find 'esds' atom. Calculating bit rate.\n"); atom_offset=aac_drilltoatom(infile,"mdat",&atom_length); if ((atom_offset != -1) && (pmp3->song_length)) { pmp3->bitrate = atom_length / ((pmp3->song_length / 1000) * 128); } } fclose(infile); return 0; } /** * Decode an mp3 frame header. Determine layer, bitrate, * samplerate, etc, and fill in the passed structure. * * @param frame 4 byte mp3 frame header * @param pfi pointer to an allocated SCAN_FRAMEINFO struct * @return 0 on success (valid frame), -1 otherwise */ int scan_decode_mp3_frame(unsigned char *frame, SCAN_FRAMEINFO *pfi) { int ver; int layer_index; int sample_index; int bitrate_index; int samplerate_index; if((frame[0] != 0xFF) || (frame[1] < 224)) { pfi->is_valid=0; return -1; } ver=(frame[1] & 0x18) >> 3; pfi->layer = 4 - ((frame[1] & 0x6) >> 1); layer_index=-1; sample_index=-1; switch(ver) { case 0: pfi->version = 2.5; sample_index=2; if(pfi->layer == 1) layer_index = 3; if((pfi->layer == 2) || (pfi->layer == 3)) layer_index = 4; break; case 2: pfi->version = 2.0; sample_index=1; if(pfi->layer == 1) layer_index=3; if((pfi->layer == 2) || (pfi->layer == 3)) layer_index=4; break; case 3: pfi->version = 1.0; sample_index=0; if(pfi->layer == 1) layer_index = 0; if(pfi->layer == 2) layer_index = 1; if(pfi->layer == 3) layer_index = 2; break; } if((layer_index < 0) || (layer_index > 4)) { pfi->is_valid=0; return -1; } if((sample_index < 0) || (sample_index > 2)) { pfi->is_valid=0; return -1; } if(pfi->layer==1) pfi->samples_per_frame=384; if(pfi->layer==2) pfi->samples_per_frame=1152; if(pfi->layer==3) { if(pfi->version == 1.0) { pfi->samples_per_frame=1152; } else { pfi->samples_per_frame=576; } } bitrate_index=(frame[2] & 0xF0) >> 4; samplerate_index=(frame[2] & 0x0C) >> 2; if((bitrate_index == 0xF) || (bitrate_index==0x0)) { pfi->is_valid=0; return -1; } if(samplerate_index == 3) { pfi->is_valid=0; return -1; } pfi->bitrate = scan_br_table[layer_index][bitrate_index]; pfi->samplerate = scan_sample_table[sample_index][samplerate_index]; if((frame[3] & 0xC0 >> 6) == 3) pfi->stereo = 0; else pfi->stereo = 1; if(frame[2] & 0x02) { /* Padding bit set */ pfi->padding=1; } else { pfi->padding=0; } if(pfi->version == 1.0) { if(pfi->stereo) { pfi->xing_offset=32; } else { pfi->xing_offset=17; } } else { if(pfi->stereo) { pfi->xing_offset=17; } else { pfi->xing_offset=9; } } pfi->crc_protected=(frame[1] & 0xFE); if(pfi->layer == 1) { pfi->frame_length = (12 * pfi->bitrate * 1000 / pfi->samplerate + pfi->padding) * 4; } else { pfi->frame_length = 144 * pfi->bitrate * 1000 / pfi->samplerate + pfi->padding; } if((pfi->frame_length > 2880) || (pfi->frame_length <= 0)) { pfi->is_valid=0; return -1; } pfi->is_valid=1; return 0; } /** * Scan 10 frames from the middle of the file and determine an * average bitrate from that. It might not be as accurate as a full * frame count, but it's probably Close Enough (tm) * * @param infile file to scan for average bitrate * @param pfi pointer to frame info struct to put the bitrate into */ void scan_get_average_bitrate(FILE *infile, SCAN_FRAMEINFO *pfi) { off_t file_size; unsigned char frame_buffer[2900]; unsigned char header[4]; int index=0; int found=0; off_t pos; SCAN_FRAMEINFO fi; int frame_count=0; int bitrate_total=0; DPRINTF(E_DBG,L_SCAN,"Starting averaging bitrate\n"); fseek(infile,0,SEEK_END); file_size=ftell(infile); pos=file_size/2; /* now, find the first frame */ fseek(infile,pos,SEEK_SET); if(fread(frame_buffer,1,sizeof(frame_buffer),infile) != sizeof(frame_buffer)) return; while(!found) { while((frame_buffer[index] != 0xFF) && (index < (sizeof(frame_buffer)-4))) index++; if(index >= (sizeof(frame_buffer)-4)) { /* largest mp3 frame is 2880 bytes */ DPRINTF(E_DBG,L_SCAN,"Could not find frame... quitting\n"); return; } if(!scan_decode_mp3_frame(&frame_buffer[index],&fi)) { /* see if next frame is valid */ fseek(infile,pos + index + fi.frame_length,SEEK_SET); if(fread(header,1,sizeof(header),infile) != sizeof(header)) { DPRINTF(E_DBG,L_SCAN,"Could not read frame header\n"); return; } if(!scan_decode_mp3_frame(header,&fi)) found=1; } if(!found) index++; } pos += index; /* found first frame. Let's move */ while(frame_count < 10) { fseek(infile,pos,SEEK_SET); if(fread(header,1,sizeof(header),infile) != sizeof(header)) { DPRINTF(E_DBG,L_SCAN,"Could not read frame header\n"); return; } if(scan_decode_mp3_frame(header,&fi)) { DPRINTF(E_DBG,L_SCAN,"Invalid frame header while averaging\n"); return; } bitrate_total += fi.bitrate; frame_count++; pos += fi.frame_length; } DPRINTF(E_DBG,L_SCAN,"Old bitrate: %d\n",pfi->bitrate); pfi->bitrate = bitrate_total/frame_count; DPRINTF(E_DBG,L_SCAN,"New bitrate: %d\n",pfi->bitrate); return; } /** * do a full frame-by-frame scan of the file, counting frames * as we go to try and get a more accurate song length estimate. * If the song turns out to be CBR, then we'll not set the frame * length. Instead we'll use the file size estimate, since it is * more consistent with iTunes. * * @param infile file to scan for frame count * @param pfi pointer to frame info struct to put framecount into */ void scan_get_frame_count(FILE *infile, SCAN_FRAMEINFO *pfi) { int pos; int frames=0; unsigned char frame_buffer[4]; SCAN_FRAMEINFO fi; off_t file_size; int err=0; int cbr=1; int last_bitrate=0; DPRINTF(E_DBG,L_SCAN,"Starting frame count\n"); fseek(infile,0,SEEK_END); file_size=ftell(infile); pos=pfi->frame_offset; while(1) { err=1; DPRINTF(E_SPAM,L_SCAN,"Seeking to %d\n",pos); fseek(infile,pos,SEEK_SET); if(fread(frame_buffer,1,sizeof(frame_buffer),infile) == sizeof(frame_buffer)) { /* check for valid frame */ if(!scan_decode_mp3_frame(frame_buffer,&fi)) { frames++; pos += fi.frame_length; err=0; if((last_bitrate) && (fi.bitrate != last_bitrate)) cbr=0; last_bitrate=fi.bitrate; /* no point in brute scan of a cbr file... */ if(cbr && (frames > 100)) { DPRINTF(E_DBG,L_SCAN,"File appears to be CBR... quitting frame count\n"); return; } } } if(err) { if(pos > (file_size - 4096)) { /* probably good enough */ pfi->number_of_frames=frames; DPRINTF(E_DBG,L_SCAN,"Estimated frame count: %d\n",frames); return; } else { DPRINTF(E_DBG,L_SCAN,"Frame count aborted on error. Pos=%d, Count=%d\n", pos, frames); return; } } } } /** * Get information from the file headers itself -- like * song length, bit rate, etc. * * @param file File to get info for * @param pmp3 where to put the found information */ int scan_get_mp3fileinfo(char *file, MP3FILE *pmp3) { FILE *infile; SCAN_ID3HEADER *pid3; SCAN_FRAMEINFO fi; unsigned int size=0; off_t fp_size=0; off_t file_size; unsigned char buffer[1024]; int index; int xing_flags; int found; int first_check; char frame_buffer[4]; if(!(infile=fopen(file,"rb"))) { DPRINTF(E_WARN,L_SCAN,"Could not open %s for reading\n",file); return -1; } memset((void*)&fi,0x00,sizeof(fi)); fseek(infile,0,SEEK_END); file_size=ftell(infile); fseek(infile,0,SEEK_SET); pmp3->file_size=file_size; if(fread(buffer,1,sizeof(buffer),infile) != sizeof(buffer)) { if(ferror(infile)) { DPRINTF(E_LOG,L_SCAN,"Error reading: %s\n",strerror(errno)); } else { DPRINTF(E_LOG,L_SCAN,"Short file: %s\n",file); } fclose(infile); return -1; } pid3=(SCAN_ID3HEADER*)buffer; found=0; fp_size=0; if(strncmp(pid3->id,"ID3",3)==0) { /* found an ID3 header... */ DPRINTF(E_DBG,L_SCAN,"Found ID3 header\n"); size = (pid3->size[0] << 21 | pid3->size[1] << 14 | pid3->size[2] << 7 | pid3->size[3]); fp_size=size + sizeof(SCAN_ID3HEADER); first_check=1; DPRINTF(E_DBG,L_SCAN,"Header length: %d\n",size); } index = 0; /* Here we start the brute-force header seeking. Sure wish there * weren't so many crappy mp3 files out there */ while(!found) { fseek(infile,fp_size,SEEK_SET); DPRINTF(E_DBG,L_SCAN,"Reading in new block at %d\n",(int)fp_size); if(fread(buffer,1,sizeof(buffer),infile) < sizeof(buffer)) { DPRINTF(E_LOG,L_SCAN,"Short read: %s\n",file); fclose(infile); return 0; } index=0; while(!found) { while((buffer[index] != 0xFF) && (index < (sizeof(buffer)-50))) index++; if((first_check) && (index)) { fp_size=0; DPRINTF(E_DBG,L_SCAN,"Bad header... dropping back for full frame search\n"); first_check=0; break; } if(index > sizeof(buffer) - 50) { fp_size += index; DPRINTF(E_DBG,L_SCAN,"Block exhausted\n"); break; } if(!scan_decode_mp3_frame(&buffer[index],&fi)) { DPRINTF(E_DBG,L_SCAN,"valid header at %d\n",index); if(strncasecmp((char*)&buffer[index+fi.xing_offset+4],"XING",4) == 0) { /* no need to check further... if there is a xing header there, * this is definately a valid frame */ found=1; fp_size += index; } else { /* No Xing... check for next frame */ DPRINTF(E_DBG,L_SCAN,"Found valid frame at %04x\n",(int)fp_size+index); DPRINTF(E_DBG,L_SCAN,"Checking at %04x\n",(int)fp_size+index+fi.frame_length); fseek(infile,fp_size + index + fi.frame_length,SEEK_SET); if(fread(frame_buffer,1,sizeof(frame_buffer),infile) == sizeof(frame_buffer)) { if(!scan_decode_mp3_frame(frame_buffer,&fi)) { found=1; fp_size += index; } } else { DPRINTF(E_LOG,L_SCAN,"Could not read frame header: %s\n",file); fclose(infile); return 0; } if(!found) { DPRINTF(E_DBG,L_SCAN,"Didn't pan out.\n"); } } } if(!found) { index++; if (first_check) { /* if the header info was wrong about where the data started, * then start a brute-force scan from the beginning of the file. * don't want to just scan forward, because we might have already * missed the xing header */ DPRINTF(E_DBG,L_SCAN,"Bad header... dropping back for full frame search\n"); first_check=0; fp_size=0; break; } } } } file_size -= fp_size; fi.frame_offset=fp_size; if(scan_decode_mp3_frame(&buffer[index],&fi)) { fclose(infile); DPRINTF(E_LOG,L_SCAN,"Could not find sync frame: %s\n",file); DPRINTF(E_LOG,L_SCAN,"If this is a valid mp3 file that plays in " "other applications, please email me at rpedde@users.sourceforge.net " "and tell me you got this error. Thanks"); return 0; } DPRINTF(E_DBG,L_SCAN," MPEG Version: %0.1g\n",fi.version); DPRINTF(E_DBG,L_SCAN," Layer: %d\n",fi.layer); DPRINTF(E_DBG,L_SCAN," Sample Rate: %d\n",fi.samplerate); DPRINTF(E_DBG,L_SCAN," Bit Rate: %d\n",fi.bitrate); /* now check for an XING header */ if(strncasecmp((char*)&buffer[index+fi.xing_offset+4],"XING",4) == 0) { DPRINTF(E_DBG,L_SCAN,"Found Xing header\n"); xing_flags=*((int*)&buffer[index+fi.xing_offset+4+4]); xing_flags=ntohs(xing_flags); DPRINTF(E_DBG,L_SCAN,"Xing Flags: %02X\n",xing_flags); if(xing_flags & 0x1) { /* Frames field is valid... */ fi.number_of_frames=*((int*)&buffer[index+fi.xing_offset+4+8]); fi.number_of_frames=ntohs(fi.number_of_frames); } } if((config.scan_type != 0) && (fi.number_of_frames == 0) && (!pmp3->song_length)) { /* We have no good estimate of song time, and we want more * aggressive scanning */ DPRINTF(E_DBG,L_SCAN,"Starting aggressive file length scan\n"); if(config.scan_type == 1) { /* get average bitrate */ scan_get_average_bitrate(infile, &fi); } else { /* get full frame count */ scan_get_frame_count(infile, &fi); } } pmp3->bitrate=fi.bitrate; pmp3->samplerate=fi.samplerate; /* guesstimate the file length */ if(!pmp3->song_length) { /* could have gotten it from the tag */ /* DWB: use ms time instead of seconds, use doubles to avoid overflow */ if(!fi.number_of_frames) { /* not vbr */ pmp3->song_length = (int) ((double) file_size * 8. / (double) fi.bitrate); } else { pmp3->song_length = (int) ((double)(fi.number_of_frames*fi.samples_per_frame*1000.)/ (double) fi.samplerate); } } DPRINTF(E_DBG,L_SCAN," Song Length: %d\n",pmp3->song_length); fclose(infile); return 0; } /** * Manually build tags. Set artist to computer/orchestra * if there is already no artist. Perhaps this could be * done better, but I'm not sure what else to do here. * * @param song MP3FILE of the file to build composite tags for */ void make_composite_tags(MP3FILE *song) { int len; char fdescr[50]; len=0; if(!song->artist && (song->orchestra || song->conductor)) { if(song->orchestra) len += strlen(song->orchestra); if(song->conductor) len += strlen(song->conductor); len += 3; song->artist=(char*)calloc(len, 1); if(song->artist) { if(song->orchestra) strcat(song->artist,song->orchestra); if(song->orchestra && song->conductor) strcat(song->artist," - "); if(song->conductor) strcat(song->artist,song->conductor); } } sprintf(fdescr,"%s audio file",song->type); song->description = strdup(fdescr); if(song->url) { song->description = strdup("Playlist URL"); song->data_kind=1; /* bit of a hack for the roku soundbridge - type *has* to be pls */ if(song->type) free(song->type); song->type = strdup("pls"); } else { song->data_kind=0; } if(!song->title) song->title = strdup(song->fname); /* Ogg used to be set as an item_kind of 4. Dunno why */ song->item_kind = 2; }