rockbox/apps/plugins/searchengine/dbinterface.c
Michiel Van Der Kolk c735ed7914 First runtime database support, self repairing, only playcount works for now,
which is still rather crude; playcount gets increased even if the song started playback but was skipped... track rating should be trivial to add, autorating also works since its based on playcount.


git-svn-id: svn://svn.rockbox.org/rockbox/trunk@6969 a1c6a512-1295-4272-9138-f99709370657
2005-07-01 17:29:44 +00:00

157 lines
5.4 KiB
C

/***************************************************************************
* __________ __ ___.
* Open \______ \ ____ ____ | | _\_ |__ _______ ___
* Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
* Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
* Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
* \/ \/ \/ \/ \/
* $Id$
*
* Copyright (C) 2005 by Michiel van der Kolk
*
* All files in this archive are subject to the GNU General Public License.
* See the file COPYING in the source tree root for full license agreement.
*
* This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
* KIND, either express or implied.
*
****************************************************************************/
#include "searchengine.h"
#include "dbinterface.h"
#undef SONGENTRY_SIZE
#undef FILEENTRY_SIZE
#undef ALBUMENTRY_SIZE
#undef ARTISTENTRY_SIZE
#undef FILERECORD2OFFSET
#define SONGENTRY_SIZE (rb->tagdbheader->songlen+12+rb->tagdbheader->genrelen+12)
#define FILEENTRY_SIZE (rb->tagdbheader->filelen+12)
#define ALBUMENTRY_SIZE (rb->tagdbheader->albumlen+4+rb->tagdbheader->songarraylen*4)
#define ARTISTENTRY_SIZE (rb->tagdbheader->artistlen+rb->tagdbheader->albumarraylen*4)
#define RUNDBENTRY_SIZE 20
#define FILERECORD2OFFSET(_x_) (rb->tagdbheader->filestart + _x_ * FILEENTRY_SIZE)
struct entry *currententry;
struct dbglobals dbglobal;
static struct entry *entryarray;
int database_init() {
char *p;
unsigned int i;
// allocate room for all entries
entryarray=(struct entry *)my_malloc(sizeof(struct entry)*rb->tagdbheader->filecount);
p=(char *)entryarray;
// zero all entries.
for(i=0;i<sizeof(struct entry)*rb->tagdbheader->filecount;i++)
*(p++)=0;
if(!*rb->tagdb_initialized) {
if(!rb->tagdb_init()) {
// failed loading db
return -1;
}
}
dbglobal.playcountmin=0;
dbglobal.playcountmax=0;
dbglobal.gotplaycountlimits=0;
return 0;
}
long readlong(int fd) {
long num;
rb->read(fd,&num,4);
#ifdef ROCKBOX_LITTLE_ENDIAN
num=BE32(num);
#endif
return num;
}
short readshort(int fd) {
short num;
rb->read(fd,&num,2);
#ifdef ROCKBOX_LITTLE_ENDIAN
num=BE16(num);
#endif
return num;
}
void loadentry(int filerecord) {
if(entryarray[filerecord].loadedfiledata==0) {
rb->lseek(*rb->tagdb_fd,FILERECORD2OFFSET(filerecord),SEEK_SET);
entryarray[filerecord].filename=(char *)my_malloc(rb->tagdbheader->filelen);
rb->read(*rb->tagdb_fd,entryarray[filerecord].filename,rb->tagdbheader->filelen);
entryarray[filerecord].hash=readlong(*rb->tagdb_fd);
entryarray[filerecord].songentry=readlong(*rb->tagdb_fd);
entryarray[filerecord].rundbentry=readlong(*rb->tagdb_fd);
entryarray[filerecord].loadedfiledata=1;
}
currententry=&entryarray[filerecord];
dbglobal.currententryindex=filerecord;
}
void loadsongdata() {
if(currententry->loadedsongdata)
return;
currententry->title=(char *)my_malloc(rb->tagdbheader->songlen);
currententry->genre=(char *)my_malloc(rb->tagdbheader->genrelen);
rb->lseek(*rb->tagdb_fd,currententry->songentry,SEEK_SET);
rb->read(*rb->tagdb_fd,currententry->title,rb->tagdbheader->songlen);
currententry->artistoffset=readlong(*rb->tagdb_fd);
currententry->albumoffset=readlong(*rb->tagdb_fd);
rb->lseek(*rb->tagdb_fd,4,SEEK_CUR);
rb->read(*rb->tagdb_fd,currententry->genre,rb->tagdbheader->genrelen);
currententry->bitrate=readshort(*rb->tagdb_fd);
currententry->year=readshort(*rb->tagdb_fd);
currententry->playtime=readlong(*rb->tagdb_fd);
currententry->track=readshort(*rb->tagdb_fd);
currententry->samplerate=readshort(*rb->tagdb_fd);
currententry->loadedsongdata=1;
}
void loadrundbdata() {
currententry->loadedrundbdata=1;
if(!*rb->rundb_initialized)
return;
if(currententry->rundbentry==-1)
return;
rb->lseek(*rb->rundb_fd,currententry->rundbentry,SEEK_SET);
currententry->rundbfe=readlong(*rb->rundb_fd);
currententry->rundbhash=readlong(*rb->rundb_fd);
currententry->rating=readshort(*rb->rundb_fd);
currententry->voladj=readshort(*rb->rundb_fd);
currententry->playcount=readlong(*rb->rundb_fd);
currententry->lastplayed=readlong(*rb->rundb_fd);
}
void loadartistname() {
/* memory optimization possible, only malloc for an album name once, then
* write that pointer to the entrys using it.
*/
if(currententry->loadedartistname)
return;
loadsongdata();
currententry->artistname=(char *)my_malloc(rb->tagdbheader->artistlen);
rb->lseek(*rb->tagdb_fd,currententry->artistoffset,SEEK_SET);
rb->read(*rb->tagdb_fd,currententry->artistname,rb->tagdbheader->artistlen);
currententry->loadedartistname=1;
}
void loadalbumname() {
/* see the note at loadartistname */
if(currententry->loadedalbumname)
return;
loadsongdata();
currententry->albumname=(char *)my_malloc(rb->tagdbheader->albumlen);
rb->lseek(*rb->tagdb_fd,currententry->albumoffset,SEEK_SET);
rb->read(*rb->tagdb_fd,currententry->albumname,rb->tagdbheader->albumlen);
currententry->loadedalbumname=1;
}
char *getfilename(int entry) {
if(entryarray[entry].loadedfiledata==0)
return "error O.o;;;";
else
return entryarray[entry].filename;
}