add support for parsing ape tags in musepack files

git-svn-id: https://svn.musicpd.org/mpd/trunk@3030 09075e82-0dd4-0310-85a5-a0d7c8717e4f
This commit is contained in:
Warren Dukes 2005-03-07 00:51:21 +00:00
parent 32a1f952e8
commit 01b708bc13
6 changed files with 159 additions and 9 deletions

2
TODO
View File

@ -12,8 +12,6 @@
*) Handle mp1 and mp2 files (including files with mp3 suffixes) *) Handle mp1 and mp2 files (including files with mp3 suffixes)
*) add support for playing aac streams (gee, thanks icecast) *) add support for playing aac streams (gee, thanks icecast)
*) parsing of lame tags (including getting replaygain and gapless info) *) parsing of lame tags (including getting replaygain and gapless info)
*) implement apev2 and id3v1 tag reader from xmms-musepack plugin
*) only use libid3tag for id3v2 tags, use internal impl for id3v1 tags
*) aduio output *) aduio output
*) allowing "pausing" of audio output devices *) allowing "pausing" of audio output devices

View File

@ -47,7 +47,7 @@ static mpc_int32_t mpc_read_cb(void * vdata, void * ptr, mpc_int32_t size) {
while(1) { while(1) {
ret = readFromInputStream(data->inStream, ptr, 1, size); ret = readFromInputStream(data->inStream, ptr, 1, size);
if(ret == 0 && !inputStreamAtEOF(data->inStream) && if(ret == 0 && !inputStreamAtEOF(data->inStream) &&
!data->dc->stop) (data->dc && !data->dc->stop))
{ {
my_usleep(10000); my_usleep(10000);
} }
@ -272,17 +272,50 @@ static int mpc_decode(OutputBuffer * cb, DecoderControl * dc,
return 0; return 0;
} }
static float mpcGetTime(char * file) {
InputStream inStream;
float time = -1;
mpc_reader reader;
mpc_streaminfo info;
MpcCallbackData data;
data.inStream = &inStream;
data.dc = NULL;
reader.read = mpc_read_cb;
reader.seek = mpc_seek_cb;
reader.tell = mpc_tell_cb;
reader.get_size = mpc_getsize_cb;
reader.canseek = mpc_canseek_cb;
reader.data = &data;
mpc_streaminfo_init(&info);
if(openInputStream(&inStream, file) < 0) return -1;
if(mpc_streaminfo_read(&info, &reader) != ERROR_CODE_OK) {
closeInputStream(&inStream);
return -1;
}
time = mpc_streaminfo_get_length(&info);
closeInputStream(&inStream);
return time;
}
static MpdTag * mpcTagDup(char * file) { static MpdTag * mpcTagDup(char * file) {
MpdTag * ret = NULL; MpdTag * ret = NULL;
FILE * fp; float time = mpcGetTime(file);
fp = fopen(file,"r"); if(time < 0) return NULL;
if(!fp) return NULL;
/* get tag info here */
ret = apeDup(file);
if(!ret) ret = id3Dup(file);
if(!ret) ret = newMpdTag(); if(!ret) ret = newMpdTag();
ret->time = 0; ret->time = time;
return ret; return ret;
} }

107
src/tag.c
View File

@ -217,6 +217,113 @@ MpdTag * id3Dup(char * file) {
return ret; return ret;
} }
MpdTag * apeDup(char * file) {
MpdTag * ret = NULL;
FILE * fp = NULL;
int tagCount;
unsigned char * buffer = NULL;
unsigned char * p;
int tagLen;
int size;
unsigned long flags;
int i;
unsigned char * key;
struct {
unsigned char id[8];
unsigned char version[4];
unsigned char length[4];
unsigned char tagCount[4];
unsigned char flags[4];
unsigned char reserved[8];
} footer;
char * apeItems[7] =
{
"title",
"artist",
"album",
"comment",
"genre",
"track",
"year"
};
int tagItems[7] =
{
TAG_ITEM_TITLE,
TAG_ITEM_ARTIST,
TAG_ITEM_ALBUM,
TAG_ITEM_COMMENT,
TAG_ITEM_GENRE,
TAG_ITEM_TRACK,
TAG_ITEM_DATE,
};
fp = fopen(file, "r");
if(!fp) return NULL;
/* determine if file has an apeV2 tag */
if(fseek(fp, 0, SEEK_END)) goto fail;
size = ftell(fp);
if(fseek(fp, size-sizeof(footer), SEEK_SET)) goto fail;
if(fread(&footer, 1, sizeof(footer), fp) != sizeof(footer)) goto fail;
if(memcmp(footer.id, "APETAGEX", sizeof(footer.id)) != 0) goto fail;
if(readLEuint32(footer.version) != 2000) goto fail;
/* find begining of ape tag */
tagLen = readLEuint32(footer.length);
if(tagLen < sizeof(footer)) goto fail;
if(fseek(fp, size-tagLen, SEEK_SET)) goto fail;
/* read tag into buffer */
tagLen -= sizeof(footer);
buffer = malloc(tagLen);
if(fread(buffer, 1, tagLen, fp) != tagLen) goto fail;
/* read tags */
tagCount = readLEuint32(footer.tagCount);
p = buffer;
while(tagCount-- && tagLen > 10) {
size = readLEuint32(p);
p += 4;
tagLen -= 4;
flags = readLEuint32(p);
p += 4;
tagLen -= 4;
/* get the key */
key = p;
while(tagLen-size > 0 && *p != '\0') {
p++;
tagLen--;
}
p++;
tagLen--;
/* get the value */
if(tagLen-size < 0) goto fail;
/* we only care about utf-8 text tags */
if(!(flags & (0x3 << 1))) {
for(i = 0; i < 7; i++) {
if(strcasecmp(key, apeItems[i]) == 0) {
if(!ret) ret = newMpdTag();
addItemToMpdTagWithLen(
ret, tagItems[i], p, size);
}
}
}
p += size;
tagLen -= size;
}
fail:
if(fp) fclose(fp);
if(buffer) free(buffer);
return ret;
}
MpdTag * newMpdTag() { MpdTag * newMpdTag() {
MpdTag * ret = malloc(sizeof(MpdTag)); MpdTag * ret = malloc(sizeof(MpdTag));
ret->items = NULL; ret->items = NULL;

View File

@ -64,6 +64,8 @@ typedef struct _MpdTag {
MpdTag * parseId3Tag(struct id3_tag *); MpdTag * parseId3Tag(struct id3_tag *);
#endif #endif
MpdTag * apeDup(char * file);
MpdTag * id3Dup(char * file); MpdTag * id3Dup(char * file);
MpdTag * newMpdTag(); MpdTag * newMpdTag();

View File

@ -92,3 +92,11 @@ char * appendToString(char * dest, const char * src) {
return dest; return dest;
} }
unsigned long readLEuint32(const unsigned char *p)
{
return ((unsigned long) p[0] << 0) |
((unsigned long) p[1] << 8) |
((unsigned long) p[2] << 16) | ((unsigned long) p[3] << 24);
}

View File

@ -35,4 +35,6 @@ int ipv6Supported();
char * appendToString(char * dest, const char * src); char * appendToString(char * dest, const char * src);
unsigned long readLEuint32(const unsigned char * p);
#endif #endif