From 2175d1edf65367fd3fe3cff266b8d6ea12930f2f Mon Sep 17 00:00:00 2001 From: Marcoen Hirschberg Date: Sat, 16 Jun 2007 18:19:51 +0000 Subject: split up the metadata parser git-svn-id: svn://svn.rockbox.org/rockbox/trunk@13637 a1c6a512-1295-4272-9138-f99709370657 --- apps/metadata/ape.c | 132 ++++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 132 insertions(+) create mode 100644 apps/metadata/ape.c (limited to 'apps/metadata/ape.c') diff --git a/apps/metadata/ape.c b/apps/metadata/ape.c new file mode 100644 index 0000000..ac071be --- /dev/null +++ b/apps/metadata/ape.c @@ -0,0 +1,132 @@ +/*************************************************************************** + * __________ __ ___. + * Open \______ \ ____ ____ | | _\_ |__ _______ ___ + * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ / + * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < < + * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \ + * \/ \/ \/ \/ \/ + * $Id$ + * + * Copyright (C) 2005 Dave Chapman + * + * 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 +#include +#include +#include +#include + +#include "system.h" +#include "id3.h" +#include "metadata_common.h" +#include "structec.h" + +#define APETAG_HEADER_LENGTH 32 +#define APETAG_HEADER_FORMAT "8llll8" +#define APETAG_ITEM_HEADER_FORMAT "ll" +#define APETAG_ITEM_TYPE_MASK 3 + +struct apetag_header +{ + char id[8]; + long version; + long length; + long item_count; + long flags; + char reserved[8]; +}; + +struct apetag_item_header +{ + long length; + long flags; +}; + +/* Read the items in an APEV2 tag. Only looks for a tag at the end of a + * file. Returns true if a tag was found and fully read, false otherwise. + */ +bool read_ape_tags(int fd, struct mp3entry* id3) +{ + struct apetag_header header; + + if ((lseek(fd, -APETAG_HEADER_LENGTH, SEEK_END) < 0) + || (ecread(fd, &header, 1, APETAG_HEADER_FORMAT, IS_BIG_ENDIAN) != APETAG_HEADER_LENGTH) + || (memcmp(header.id, "APETAGEX", sizeof(header.id)))) + { + return false; + } + + if ((header.version == 2000) && (header.item_count > 0) + && (header.length > APETAG_HEADER_LENGTH)) + { + char *buf = id3->id3v2buf; + unsigned int buf_remaining = sizeof(id3->id3v2buf) + + sizeof(id3->id3v1buf); + unsigned int tag_remaining = header.length - APETAG_HEADER_LENGTH; + int i; + + if (lseek(fd, -header.length, SEEK_END) < 0) + { + return false; + } + + for (i = 0; i < header.item_count; i++) + { + struct apetag_item_header item; + char name[TAG_NAME_LENGTH]; + char value[TAG_VALUE_LENGTH]; + long r; + + if (tag_remaining < sizeof(item)) + { + break; + } + + if (ecread(fd, &item, 1, APETAG_ITEM_HEADER_FORMAT, IS_BIG_ENDIAN) < (long) sizeof(item)) + { + return false; + } + + tag_remaining -= sizeof(item); + r = read_string(fd, name, sizeof(name), 0, tag_remaining); + + if (r == -1) + { + return false; + } + + tag_remaining -= r + item.length; + + if ((item.flags & APETAG_ITEM_TYPE_MASK) == 0) + { + long len; + + if (read_string(fd, value, sizeof(value), -1, item.length) + != item.length) + { + return false; + } + + len = parse_tag(name, value, id3, buf, buf_remaining, + TAGTYPE_APE); + buf += len; + buf_remaining -= len; + } + else + { + if (lseek(fd, item.length, SEEK_CUR) < 0) + { + return false; + } + } + } + } + + return true; +} -- cgit v1.1