/* * $Id$ * Implementation for simple gdbm db * * 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 */ #include #include #include #include #define __USE_UNIX98 #include #include "err.h" #include "mp3-scanner.h" /* * Typedefs */ typedef struct tag_mp3record { MP3FILE mp3file; struct tag_mp3record *next; } MP3RECORD; typedef struct tag_playlistentry { unsigned int id; struct tag_playlistentry *next; } DB_PLAYLISTENTRY; typedef struct tag_playlist { unsigned int id; int songs; char *name; struct tag_playlistentry *nodes; struct tag_playlist *next; } DB_PLAYLIST; #define MAYBEFREE(a) { if((a)) free((a)); }; /* * Globals */ int db_version_no; int db_update_mode=0; int db_song_count; int db_playlist_count=0; pthread_rwlock_t db_rwlock; /* OSX doesn't have PTHREAD_RWLOCK_INITIALIZER */ pthread_once_t db_initlock=PTHREAD_ONCE_INIT; GDBM_FILE db_songs; GDBM_FILE db_playlists; /* * Forwards */ int db_start_initial_update(void); int db_end_initial_update(void); int db_is_empty(void); int db_init(char *parameters); int db_deinit(void); int db_version(void); int db_add(MP3FILE *mp3file); int db_add_playlist(unsigned int playlistid, char *name); int db_add_playlist_song(unsigned int playlistid, unsigned int itemid); MP3RECORD *db_enum_begin(void); MP3FILE *db_enum(MP3RECORD **current); int db_enum_end(void); DB_PLAYLIST *db_playlist_enum_begin(void); int db_playlist_enum(DB_PLAYLIST **current); int db_playlist_enum_end(void); DB_PLAYLISTENTRY *db_playlist_items_enum_begin(int playlistid); int db_playlist_items_enum(DB_PLAYLISTENTRY **current); int db_playlist_items_enum_end(void); int db_get_song_count(void); int db_get_playlist_count(void); int db_get_playlist_entry_count(int playlistid); char *db_get_playlist_name(int playlistid); MP3FILE *db_find(int id); void db_freerecord(MP3RECORD *mp3record); /* * db_init_once * * Must dynamically initialize the rwlock, as Mac OSX 10.3 (at least) * doesn't have a static initializer for rwlocks */ void db_init_once(void) { pthread_rwlock_init(&db_rwlock,NULL); } /* * db_init * * Initialize the database. Parameter is the directory * of the database files */ int db_init(char *parameters) { char db_path[MAX_PATH + 1]; snprintf(db_path,sizeof(db_path),"%s\\%s",parameters,"songs.gdb"); db_songs=gdbm_open(db_path,0,GDBM_WRCREAT | GDBM_SYNC | GDBM_NOLOCK, 0600,NULL); if(!db_songs) { DPRINTF(ERR_FATAL,"Could not open songs database (%s)\n", gdbm_strerror(gdbm_error)); return -1; } snprintf(db_path,sizeof(db_path),"%s\\%s",parameters,"playlists.gdb"); db_playlists=gdbm_open(db_path,0,GDBM_WRCREAT | GDBM_SYNC | GDBM_NOLOCK, 0600,NULL); if(!db_playlists) { DPRINTF(ERR_FATAL,"Could not open playlist database (%s)\n", gdbm_strerror(gdbm_error)); return -1; } db_version_no=1; db_song_count=0; /* count the actual songs... */ /* and the playlists */ return pthread_once(&db_initlock,db_init_once); } /* * db_deinit * * Close the db, in this case freeing memory */ int db_deinit(void) { MP3RECORD *current; DB_PLAYLIST *plist; DB_PLAYLISTENTRY *pentry; while(db_root.next) { current=db_root.next; db_root.next=current->next; db_freerecord(current); } while(db_playlists.next) { plist=db_playlists.next; db_playlists.next=plist->next; free(plist->name); /* free all the nodes */ while(plist->nodes) { pentry=plist->nodes; plist->nodes = pentry->next; free(pentry); } free(plist); } return 0; } /* * db_version * * return the db version */ int db_version(void) { return db_version_no; } /* * db_start_initial_update * * Set the db to bulk import mode */ int db_start_initial_update(void) { db_update_mode=1; return 0; } /* * db_end_initial_update * * Take the db out of bulk import mode */ int db_end_initial_update(void) { db_update_mode=0; return 0; } /* * db_is_empty * * See if the db is empty or not -- that is, should * the scanner start up in bulk update mode or in * background update mode */ int db_is_empty(void) { return !db_root.next; } /* * db_add_playlist * * Add a new playlist */ int db_add_playlist(unsigned int playlistid, char *name) { int err; DB_PLAYLIST *pnew; pnew=(DB_PLAYLIST*)malloc(sizeof(DB_PLAYLIST)); if(!pnew) return -1; pnew->name=strdup(name); pnew->id=playlistid; pnew->nodes=NULL; pnew->songs=0; if(!pnew->name) { free(pnew); return -1; } DPRINTF(ERR_DEBUG,"Adding new playlist %s\n",name); if(err=pthread_rwlock_wrlock(&db_rwlock)) { DPRINTF(ERR_WARN,"cannot lock wrlock in db_add\n"); free(pnew->name); free(pnew); errno=err; return -1; } /* we'll update playlist count when we add a song! */ // db_playlist_count++; pnew->next=db_playlists.next; db_playlists.next=pnew; if(!db_update_mode) { db_version_no++; } pthread_rwlock_unlock(&db_rwlock); DPRINTF(ERR_DEBUG,"Added playlist\n"); return 0; } /* * db_add_playlist_song * * Add a song to a particular playlist */ int db_add_playlist_song(unsigned int playlistid, unsigned int itemid) { DB_PLAYLIST *current; DB_PLAYLISTENTRY *pnew; int err; pnew=(DB_PLAYLISTENTRY*)malloc(sizeof(DB_PLAYLISTENTRY)); if(!pnew) return -1; pnew->id=itemid; pnew->next=NULL; DPRINTF(ERR_DEBUG,"Adding new playlist item\n"); if(err=pthread_rwlock_wrlock(&db_rwlock)) { DPRINTF(ERR_WARN,"cannot lock wrlock in db_add\n"); free(pnew); errno=err; return -1; } current=db_playlists.next; while(current && (current->id != playlistid)) current=current->next; if(!current) { DPRINTF(ERR_WARN,"Could not find playlist attempting to add to\n"); pthread_rwlock_unlock(&db_rwlock); free(pnew); return -1; } if(!current->songs) db_playlist_count++; current->songs++; pnew->next = current->nodes; current->nodes = pnew; if(!db_update_mode) { db_version_no++; } pthread_rwlock_unlock(&db_rwlock); DPRINTF(ERR_DEBUG,"Added playlist item\n"); return 0; } /* * db_add * * add an MP3 file to the database. */ int db_add(MP3FILE *mp3file) { int err; int g; MP3RECORD *pnew; DPRINTF(ERR_DEBUG,"Adding %s\n",mp3file->path); if((pnew=(MP3RECORD*)malloc(sizeof(MP3RECORD))) == NULL) { free(pnew); errno=ENOMEM; return -1; } memset(pnew,0,sizeof(MP3RECORD)); memcpy(&pnew->mp3file,mp3file,sizeof(MP3FILE)); g=(int) pnew->mp3file.path=strdup(mp3file->path); g = g && (pnew->mp3file.fname=strdup(mp3file->fname)); if(mp3file->title) g = g && (pnew->mp3file.title=strdup(mp3file->title)); if(mp3file->artist) g = g && (pnew->mp3file.artist=strdup(mp3file->artist)); if(mp3file->album) g = g && (pnew->mp3file.album=strdup(mp3file->album)); if(mp3file->genre) g = g && (pnew->mp3file.genre=strdup(mp3file->genre)); if(mp3file->comment) g = g && (pnew->mp3file.comment=strdup(mp3file->comment)); if(!g) { DPRINTF(ERR_WARN,"Malloc error in db_add\n"); db_freerecord(pnew); errno=ENOMEM; return -1; } if(err=pthread_rwlock_wrlock(&db_rwlock)) { DPRINTF(ERR_WARN,"cannot lock wrlock in db_add\n"); db_freerecord(pnew); errno=err; return -1; } pnew->next=db_root.next; db_root.next=pnew; if(!db_update_mode) { db_version_no++; } db_song_count++; pthread_rwlock_unlock(&db_rwlock); DPRINTF(ERR_DEBUG,"Added file\n"); return 0; } /* * db_freerecord * * free a complete mp3record */ void db_freerecord(MP3RECORD *mp3record) { MAYBEFREE(mp3record->mp3file.path); MAYBEFREE(mp3record->mp3file.fname); MAYBEFREE(mp3record->mp3file.title); MAYBEFREE(mp3record->mp3file.artist); MAYBEFREE(mp3record->mp3file.album); MAYBEFREE(mp3record->mp3file.genre); MAYBEFREE(mp3record->mp3file.comment); free(mp3record); } /* * db_enum_begin * * Begin to walk through an enum of * the database. * * this should be done quickly, as we'll be holding * a reader lock on the db */ MP3RECORD *db_enum_begin(void) { int err; if(err=pthread_rwlock_rdlock(&db_rwlock)) { log_err(0,"Cannot lock rwlock\n"); errno=err; return NULL; } return db_root.next; } /* * db_playlist_enum_begin * * Start enumerating playlists */ DB_PLAYLIST *db_playlist_enum_begin(void) { int err; DB_PLAYLIST *current; if(err=pthread_rwlock_rdlock(&db_rwlock)) { log_err(0,"Cannot lock rwlock\n"); errno=err; return NULL; } /* find first playlist with a song in it! */ current=db_playlists.next; while(current && (!current->songs)) current=current->next; return current; } /* * db_playlist_items_enum_begin * * Start enumerating playlist items */ DB_PLAYLISTENTRY *db_playlist_items_enum_begin(int playlistid) { DB_PLAYLIST *current; DB_PLAYLISTENTRY *retval; int err; if(err=pthread_rwlock_rdlock(&db_rwlock)) { log_err(0,"Cannot lock rwlock\n"); errno=err; return NULL; } current=db_playlists.next; while(current && (current->id != playlistid)) current=current->next; if(!current) return NULL; return current->nodes; } /* * db_enum * * Walk to the next entry */ MP3FILE *db_enum(MP3RECORD **current) { MP3FILE *retval; if(*current) { retval=&((*current)->mp3file); *current=(*current)->next; return retval; } return NULL; } /* * db_playlist_enum * * walk to the next entry */ int db_playlist_enum(DB_PLAYLIST **current) { int retval; DB_PLAYLIST *p; if(*current) { retval = (*current)->id; p=*current; p=p->next; while(p && (!p->songs)) p=p->next; *current=p; return retval; } return -1; } /* * db_playlist_items_enum * * walk to the next entry */ int db_playlist_items_enum(DB_PLAYLISTENTRY **current) { int retval; if(*current) { retval = (*current)->id; *current=(*current)->next; return retval; } return -1; } /* * db_enum_end * * quit walking the database (and give up reader lock) */ int db_enum_end(void) { return pthread_rwlock_unlock(&db_rwlock); } /* * db_playlist_enum_end * * quit walking the database */ int db_playlist_enum_end(void) { return pthread_rwlock_unlock(&db_rwlock); } /* * db_playlist_items_enum_end * * Quit walking the database */ int db_playlist_items_enum_end(void) { return pthread_rwlock_unlock(&db_rwlock); } /* * db_find * * Find a MP3FILE entry based on file id */ MP3FILE *db_find(int id) { MP3RECORD *current=db_root.next; while((current) && (current->mp3file.id != id)) { current=current->next; } if(!current) return NULL; return ¤t->mp3file; } /* * db_get_playlist_count * * return the number of playlists */ int db_get_playlist_count(void) { return db_playlist_count; } /* * db_get_song_count * * return the number of songs in the database. Used for the /database * request */ int db_get_song_count(void) { return db_song_count; } /* * db_get_playlist_entry_count * * return the number of songs in a particular playlist */ int db_get_playlist_entry_count(int playlistid) { int count; DB_PLAYLIST *current; int err; if(err=pthread_rwlock_rdlock(&db_rwlock)) { log_err(0,"Cannot lock rwlock\n"); errno=err; return -1; } current=db_playlists.next; while(current && (current->id != playlistid)) current=current->next; if(!current) { count = -1; } else { count = current->songs; } pthread_rwlock_unlock(&db_rwlock); return count; } /* * db_get_playlist_name * * return the name of a playlist * * FIXME: Small race here */ char *db_get_playlist_name(int playlistid) { char *name; DB_PLAYLIST *current; int err; if(err=pthread_rwlock_rdlock(&db_rwlock)) { log_err(0,"Cannot lock rwlock\n"); errno=err; return NULL; } current=db_playlists.next; while(current && (current->id != playlistid)) current=current->next; if(!current) { name = NULL; } else { name = current->name; } pthread_rwlock_unlock(&db_rwlock); return name; }