Files
cmus/ape.c
2026-03-29 14:01:52 +03:00

259 lines
5.4 KiB
C

/*
* Copyright 2008-2013 Various Authors
* Copyright 2006 Chun-Yu Shei <cshei AT cs.indiana.edu>
*
* Cleaned up by Timo Hirvonen <tihirvon@gmail.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, see <http://www.gnu.org/licenses/>.
*/
#include "ape.h"
#include "file.h"
#include "xmalloc.h"
#include "utils.h"
#include <errno.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
#include <strings.h>
/* http://www.personal.uni-jena.de/~pfk/mpp/sv8/apetag.html */
#define PREAMBLE_SIZE (8)
static const char preamble[PREAMBLE_SIZE] = { 'A', 'P', 'E', 'T', 'A', 'G', 'E', 'X' };
/* NOTE: not sizeof(struct ape_header)! */
#define HEADER_SIZE (32)
/* returns position of APE header or -1 if not found */
static int find_ape_tag_slow(int fd)
{
char buf[4096];
int match = 0;
int pos = 0;
/* seek to start of file */
if (lseek(fd, pos, SEEK_SET) == -1)
return -1;
while (1) {
int i, got = read(fd, buf, sizeof(buf));
if (got == -1) {
if (errno == EAGAIN || errno == EINTR)
continue;
break;
}
if (got == 0)
break;
for (i = 0; i < got; i++) {
if (buf[i] != preamble[match]) {
match = 0;
continue;
}
match++;
if (match == PREAMBLE_SIZE)
return pos + i + 1 - PREAMBLE_SIZE;
}
pos += got;
}
return -1;
}
static int ape_parse_header(const char *buf, struct ape_header *h)
{
if (memcmp(buf, preamble, PREAMBLE_SIZE))
return 0;
h->version = read_le32(buf + 8);
h->size = read_le32(buf + 12);
h->count = read_le32(buf + 16);
h->flags = read_le32(buf + 20);
return 1;
}
static int read_header(int fd, struct ape_header *h)
{
char buf[HEADER_SIZE];
if (read_all(fd, buf, sizeof(buf)) != sizeof(buf))
return 0;
return ape_parse_header(buf, h);
}
/* sets fd right after the header and returns 1 if found,
* otherwise returns 0
*/
static int find_ape_tag(int fd, struct ape_header *h, int slow)
{
int pos;
if (lseek(fd, -HEADER_SIZE, SEEK_END) == -1)
return 0;
if (read_header(fd, h))
return 1;
/* try to skip ID3v1 tag at the end of the file */
if (lseek(fd, -(HEADER_SIZE + 128), SEEK_END) == -1)
return 0;
if (read_header(fd, h))
return 1;
if (!slow)
return 0;
pos = find_ape_tag_slow(fd);
if (pos == -1)
return 0;
if (lseek(fd, pos, SEEK_SET) == -1)
return 0;
return read_header(fd, h);
}
/*
* All keys are ASCII and length is 2..255
*
* UTF-8: Artist, Album, Title, Genre
* Integer: Track (N or N/M)
* Date: Year (release), "Record Date"
*
* UTF-8 strings are NOT zero terminated.
*
* Also support "discnumber" (vorbis) and "disc" (non-standard)
*/
static int ape_parse_one(const char *buf, int size, char **keyp, char **valp)
{
int pos = 0;
while (size - pos > 8) {
uint32_t val_len, flags;
char *key, *val;
int64_t max_key_len, key_len;
val_len = read_le32(buf + pos); pos += 4;
flags = read_le32(buf + pos); pos += 4;
max_key_len = size - pos - (int64_t)val_len - 1;
if (max_key_len < 0) {
/* corrupt */
break;
}
for (key_len = 0; key_len < max_key_len && buf[pos + key_len]; key_len++)
; /* nothing */
if (buf[pos + key_len]) {
/* corrupt */
break;
}
if (!AF_IS_UTF8(flags)) {
/* ignore binary data */
pos += key_len + 1 + val_len;
continue;
}
key = xstrdup(buf + pos);
pos += key_len + 1;
/* should not be NUL-terminated */
val = xstrndup(buf + pos, val_len);
pos += val_len;
/* could be moved to comment.c but I don't think anyone else would use it */
if (!strcasecmp(key, "record date") || !strcasecmp(key, "year")) {
free(key);
key = xstrdup("date");
}
if (!strcasecmp(key, "date")) {
/* Date format
*
* 1999-08-11 12:34:56
* 1999-08-11 12:34
* 1999-08-11
* 1999-08
* 1999
* 1999-W34 (week 34, totally crazy)
*
* convert to year, pl.c supports only years anyways
*
* FIXME: which one is the most common tag (year or record date)?
*/
if (strlen(val) > 4)
val[4] = 0;
}
*keyp = key;
*valp = val;
return pos;
}
return -1;
}
/* return the number of comments, or -1 */
int ape_read_tags(struct apetag *ape, int fd, int slow)
{
struct ape_header *h = &ape->header;
int rc = -1;
off_t old_pos;
/* save position */
old_pos = lseek(fd, 0, SEEK_CUR);
if (!find_ape_tag(fd, h, slow))
goto fail;
if (AF_IS_FOOTER(h->flags)) {
/* seek back right after the header */
if (lseek(fd, -((int)h->size), SEEK_CUR) == -1)
goto fail;
}
/* ignore insane tags */
if (h->size > 1024 * 1024)
goto fail;
ape->buf = xnew(char, h->size);
if (read_all(fd, ape->buf, h->size) != h->size)
goto fail;
rc = h->count;
fail:
lseek(fd, old_pos, SEEK_SET);
return rc;
}
/* returned key-name must be free'd */
char *ape_get_comment(struct apetag *ape, char **val)
{
struct ape_header *h = &ape->header;
char *key;
int rc;
if (ape->pos >= h->size)
return NULL;
rc = ape_parse_one(ape->buf + ape->pos, h->size - ape->pos, &key, val);
if (rc < 0)
return NULL;
ape->pos += rc;
return key;
}