summaryrefslogtreecommitdiff
path: root/apps/tagdb/parser.c
diff options
context:
space:
mode:
authorNiels Laukens <niobos@rockbox.org>2005-07-06 11:03:20 +0000
committerNiels Laukens <niobos@rockbox.org>2005-07-06 11:03:20 +0000
commitd1c294c17de95615b7af428da938b686830b42df (patch)
tree950080f5b6c9503c090df6e4f0929f13eae8891e /apps/tagdb/parser.c
parent5e9f52f6d1f3356bc6df75a675e1a2d5cdbf9d77 (diff)
downloadrockbox-d1c294c17de95615b7af428da938b686830b42df.zip
rockbox-d1c294c17de95615b7af428da938b686830b42df.tar.gz
rockbox-d1c294c17de95615b7af428da938b686830b42df.tar.bz2
rockbox-d1c294c17de95615b7af428da938b686830b42df.tar.xz
Initial import of tagdb
git-svn-id: svn://svn.rockbox.org/rockbox/trunk@7039 a1c6a512-1295-4272-9138-f99709370657
Diffstat (limited to 'apps/tagdb/parser.c')
-rw-r--r--apps/tagdb/parser.c218
1 files changed, 218 insertions, 0 deletions
diff --git a/apps/tagdb/parser.c b/apps/tagdb/parser.c
new file mode 100644
index 0000000..1d251dc
--- /dev/null
+++ b/apps/tagdb/parser.c
@@ -0,0 +1,218 @@
+#include <stdio.h>
+#include <stdint.h>
+#include <stdlib.h>
+
+#include "config.h"
+
+int errno;
+
+int read_failure(FILE *fd) {
+ fprintf(stderr, "Could not read from file: errno: %u ", errno);
+ if( feof(fd) ) fprintf(stderr, "EOF");
+ fprintf(stderr, "\n");
+ return 1;
+}
+
+int mem_failure() {
+ fprintf(stderr, "Could not (re)allocate memory\n");
+ return 1;
+}
+
+int main(int argc, char *argv[]) {
+ FILE *fd;
+ uint32_t artist_start, album_start, song_start, file_start;
+ uint32_t artist_count, album_count, song_count, file_count;
+ uint32_t artist_len, album_array_count;
+ uint32_t album_len, song_array_count;
+ uint32_t song_len, genre_len;
+ uint32_t file_len;
+#define header_start 0
+#define header_len 68
+
+ uint32_t i, j;
+ char *ct1 = NULL, *ct2 = NULL; // char temp 1 and 2
+ uint32_t it = 0; // integer temp
+
+ // input validation
+ if( argc != 2 ) {
+ fprintf(stderr, "usage: parser dbfile\n");
+ return 1;
+ }
+
+ // open file
+ fd = fopen(argv[1], "r");
+ if( fd == NULL ) {
+ fprintf(stderr, "Could not open file \"%s\"\n", argv[1]);
+ return 1;
+ }
+
+ // read the header
+ ct1 = realloc(ct1, 4); if( ct1 == NULL ) return mem_failure();
+ if( fread(ct1, 4, 1, fd) != 1 ) return read_failure(fd);
+ if( ct1[0] != 'R' || ct1[1] != 'D' || ct1[2] != 'B' ) {
+ printf("No header found\n");
+ return 1;
+ }
+ if( ct1[3] != 0x03 ) {
+ printf("Not version 3\n");
+ return 1;
+ }
+
+ if( fread(&artist_start, 4, 1, fd) != 1 ) return read_failure(fd); artist_start = BE32(artist_start);
+ if( fread(&album_start, 4, 1, fd) != 1 ) return read_failure(fd); album_start = BE32(album_start);
+ if( fread(&song_start, 4, 1, fd) != 1 ) return read_failure(fd); song_start = BE32(song_start);
+ if( fread(&file_start, 4, 1, fd) != 1 ) return read_failure(fd); file_start = BE32(file_start);
+
+ if( fread(&artist_count, 4, 1, fd) != 1 ) return read_failure(fd); artist_count = BE32(artist_count);
+ if( fread(&album_count, 4, 1, fd) != 1 ) return read_failure(fd); album_count = BE32(album_count);
+ if( fread(&song_count, 4, 1, fd) != 1 ) return read_failure(fd); song_count = BE32(song_count);
+ if( fread(&file_count, 4, 1, fd) != 1 ) return read_failure(fd); file_count = BE32(file_count);
+
+ if( fread(&artist_len, 4, 1, fd) != 1 ) return read_failure(fd); artist_len = BE32(artist_len);
+ if( fread(&album_len, 4, 1, fd) != 1 ) return read_failure(fd); album_len = BE32(album_len);
+ if( fread(&song_len, 4, 1, fd) != 1 ) return read_failure(fd); song_len = BE32(song_len);
+ if( fread(&genre_len, 4, 1, fd) != 1 ) return read_failure(fd); genre_len = BE32(genre_len);
+ if( fread(&file_len, 4, 1, fd) != 1 ) return read_failure(fd); file_len = BE32(file_len);
+
+ if( fread(&song_array_count, 4, 1, fd) != 1 ) return read_failure(fd); song_array_count = BE32(song_array_count);
+ if( fread(&album_array_count, 4, 1, fd) != 1 ) return read_failure(fd); album_array_count = BE32(album_array_count);
+
+ if( fread(ct1, 4, 1, fd) != 1 ) return read_failure(fd);
+
+ // print header info
+ printf("HEADER");
+ printf("\n Artist start: 0x%08x = %u", artist_start, artist_start);
+ if( artist_start != header_start + header_len )
+ printf(" should be 0x%08x = %u", header_start + header_len, header_start + header_len);
+ printf("\n Album start: 0x%08x = %u", album_start, album_start);
+ if( album_start != artist_start + artist_count*(artist_len + 4*album_array_count) )
+ printf(" should be 0x%08x = %u", artist_start + artist_count*(artist_len + 4*album_array_count),
+ artist_start + artist_count*(artist_len + 4*album_array_count));
+ printf("\n Song start: 0x%08x = %u", song_start, song_start);
+ if( song_start != album_start + album_count*(album_len + 4 + 4*song_array_count) )
+ printf(" should be 0x%08x = %u", album_start + album_count*(album_len + 4 + 4*song_array_count),
+ album_start + album_count*(album_len + 4 + 4*song_array_count));
+ printf("\n File start: 0x%08x = %u", file_start, file_start);
+ if( file_start != song_start + song_count*(song_len + genre_len + 24) )
+ printf(" should be 0x%08x = %u", song_start + song_count*(song_len + genre_len + 24),
+ song_start + song_count*(song_len + genre_len + 24));
+
+ printf("\n Artist count: 0x%08x = %u\n", artist_count, artist_count);
+ printf(" Album count: 0x%08x = %u\n", album_count, album_count);
+ printf(" Song count: 0x%08x = %u\n", song_count, song_count);
+ printf(" File count: 0x%08x = %u\n", file_count, file_count);
+
+ printf(" Artist len: 0x%08x = %u\n", artist_len, artist_len);
+ printf(" Album len: 0x%08x = %u\n", album_len, album_len);
+ printf(" Song len: 0x%08x = %u\n", song_len, song_len);
+ printf(" Genre len: 0x%08x = %u\n", genre_len, genre_len);
+ printf(" File len: 0x%08x = %u\n", file_len, file_len);
+
+ printf(" Song[] count: 0x%08x = %u\n", song_array_count, song_array_count);
+ printf(" Album[] count: 0x%08x = %u\n", album_array_count, album_array_count);
+
+ printf(" Reserved: 0x%08x\n", ct1[0] & 0xFFFFFFFE);
+ printf(" Rundb dirty: 0x%01x\n", ct1[3] & 0x01);
+
+ // iterate over artists:
+ ct1 = realloc(ct1, artist_len); if( ct1 == NULL && artist_count!=0 ) return mem_failure();
+ for(i=0; i < artist_count; i++) {
+ printf("ARTIST %u/%u (offset 0x%08lx)\n", i, artist_count, (unsigned long)ftell(fd));
+
+ if( fread(ct1, artist_len, 1, fd) != 1 ) return read_failure(fd);
+ printf(" Name: \"%s\"\n", ct1);
+
+ printf(" Albums:\n");
+ for(j=0; j < album_array_count; j++) {
+ if( fread(&it, 4, 1, fd) != 1 ) return read_failure(fd); it = BE32(it);
+ printf(" Offset 0x%08x = ", it);
+ if(it != 0) {
+ printf("item %u\n", (it - album_start) / (album_len + 4 + 4*song_array_count));
+ } else {
+ printf("padding\n");
+ }
+ }
+ }
+
+ // iterate over albums:
+ ct1 = realloc(ct1, album_len); if( ct1 == NULL && album_count!=0) return mem_failure();
+ for(i=0; i < album_count; i++) {
+ printf("ALBUM %u/%u (offset 0x%08lx)\n", i, album_count, (unsigned long)ftell(fd));
+
+ if( fread(ct1, album_len, 1, fd) != 1 ) return read_failure(fd);
+ printf(" Name: \"%s\"\n", ct1);
+
+ if( fread(&it, 4, 1, fd) != 1 ) return read_failure(fd); it = BE32(it);
+ printf(" Artist offset: 0x%08x = item %u\n", it, (it - artist_start) / (artist_len + 4*album_array_count));
+
+ printf(" Songs:\n");
+ for(j=0; j < song_array_count; j++) {
+ if( fread(&it, 4, 1, fd) != 1 ) return read_failure(fd); it = BE32(it);
+ printf(" Offset 0x%08x = ", it);
+ if(it != 0) {
+ printf("item %u\n", (it - song_start) / (song_len + genre_len + 24));
+ } else {
+ printf("padding\n");
+ }
+ }
+ }
+
+ // iterate over songs:
+ ct1 = realloc(ct1, song_len); if( ct1 == NULL && song_count!=0) return mem_failure();
+ ct2 = realloc(ct2, genre_len); if( ct2 == NULL && song_count!=0) return mem_failure();
+ for(i=0; i < song_count; i++) {
+ printf("SONG %u/%u (offset 0x%08lx)\n", i, song_count, (unsigned long)ftell(fd));
+
+ if( fread(ct1, song_len, 1, fd) != 1 ) return read_failure(fd);
+ printf(" Name: \"%s\"\n", ct1);
+
+ if( fread(&it, 4, 1, fd) != 1 ) return read_failure(fd); it = BE32(it);
+ printf(" Artist offset: 0x%08x = item %u\n", it, (it - artist_start) / (artist_len + 4*album_array_count));
+
+ if( fread(&it, 4, 1, fd) != 1 ) return read_failure(fd); it = BE32(it);
+ printf(" Album offset: 0x%08x = item %u\n", it, (it - album_start) / (album_len + 4 + 4*song_array_count));
+
+ if( fread(&it, 4, 1, fd) != 1 ) return read_failure(fd); it = BE32(it);
+ printf(" File offset: 0x%08x = item %u\n", it, (it - file_start) / (file_len + 12));
+
+ if( fread(ct2, genre_len, 1, fd) != 1 ) return read_failure(fd);
+ printf(" Genre: \"%s\"\n", ct2);
+
+ if( fread(&it, 4, 1, fd) != 1 ) return read_failure(fd); it = BE32(it);
+ printf(" Bitrate: 0x%04x = %u\n", (it & 0xFFFF0000) >> 16, (it & 0xFFFF0000) >> 16);
+ printf(" Year: 0x%04x = %u\n", it & 0x0000FFFF, it & 0x0000FFFF);
+
+ if( fread(&it, 4, 1, fd) != 1 ) return read_failure(fd); it = BE32(it);
+ printf(" Playtime: 0x%08x = %u\n", it, it);
+
+ if( fread(&it, 4, 1, fd) != 1 ) return read_failure(fd); it = BE32(it);
+ printf(" Track: 0x%04x = %u\n", (it & 0xFFFF0000) >> 16, (it & 0xFFFF0000) >> 16);
+ printf(" Samplerate: 0x%04x = %u\n", it & 0x0000FFFF, it & 0x0000FFFF);
+ }
+
+ // iterate over file:
+ ct1 = realloc(ct1, file_len); if( ct1 == NULL && file_count!=0) return mem_failure();
+ for(i=0; i < file_count; i++) {
+ printf("FILE %u/%u (offset 0x%08lx)\n", i, file_count, (unsigned long)ftell(fd));
+
+ if( fread(ct1, file_len, 1, fd) != 1 ) return read_failure(fd);
+ printf(" Name: \"%s\"\n", ct1);
+
+ if( fread(&it, 4, 1, fd) != 1 ) return read_failure(fd); it = BE32(it);
+ printf(" Hash: 0x%08x = %u\n", it, it);
+
+ if( fread(&it, 4, 1, fd) != 1 ) return read_failure(fd); it = BE32(it);
+ printf(" Song offset: 0x%08x = item %u\n", it, (it - song_start) / (song_len + genre_len + 24));
+
+ if( fread(&it, 4, 1, fd) != 1 ) return read_failure(fd); it = BE32(it);
+ printf(" Rundb offset: 0x%08x = %u\n", it, it);
+ }
+
+ // close the file
+ if( fclose(fd) != 0 ) {
+ fprintf(stderr, "Could not close file\n");
+ return 1;
+ }
+
+ return 0;
+}