summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorThomas Martitz <kugel@rockbox.org>2011-01-21 20:09:22 +0000
committerThomas Martitz <kugel@rockbox.org>2011-01-21 20:09:22 +0000
commit01586b5ffe7ddce3a4c7e9401187a0a8f7bb594e (patch)
tree734d0d61c7efb9ebf3009ec7a7682661853b7b5a
parent52c678945063436da28a9801ce4a739ad0473579 (diff)
FS#11828: Fix in core mod parser to blindly accept any .mod you throw at it. Based on MikMod.
git-svn-id: svn://svn.rockbox.org/rockbox/trunk@29105 a1c6a512-1295-4272-9138-f99709370657
-rw-r--r--apps/metadata/mod.c62
1 files changed, 52 insertions, 10 deletions
diff --git a/apps/metadata/mod.c b/apps/metadata/mod.c
index dbedc6e62f..9595729c0b 100644
--- a/apps/metadata/mod.c
+++ b/apps/metadata/mod.c
@@ -19,40 +19,82 @@
*
****************************************************************************/
#include <stdio.h>
-#include <string.h>
#include <stdlib.h>
#include <ctype.h>
#include <inttypes.h>
#include "system.h"
#include "metadata.h"
+#include <string-extra.h>
#include "metadata_common.h"
#include "metadata_parsers.h"
#include "rbunicode.h"
+#define MODULEHEADERSIZE 0x438
+
bool get_mod_metadata(int fd, struct mp3entry* id3)
{
/* Use the trackname part of the id3 structure as a temporary buffer */
- unsigned char buf[1084];
- int read_bytes;
+ unsigned char buf[MODULEHEADERSIZE];
+ unsigned char id[4];
+ bool is_mod_file = false;
char *p;
-
if ((lseek(fd, 0, SEEK_SET) < 0)
- || ((read_bytes = read(fd, buf, sizeof(buf))) < 1084))
+ || (read(fd, buf, sizeof(buf)) < MODULEHEADERSIZE))
{
return false;
}
-
- /* We don't do file format checking here
- * There can be .mod files without any signatures out there */
+
+ if (read(fd, id, sizeof(id)) < (ssize_t)sizeof(id))
+ return false;
+
+ /* Mod type checking based on MikMod */
+ /* Protracker and variants */
+ if ((!memcmp(id, "M.K.", 4)) || (!memcmp(id, "M!K!", 4))) {
+ is_mod_file = true;
+ }
+
+ /* Star Tracker */
+ if (((!memcmp(id, "FLT", 3)) || (!memcmp(id, "EXO", 3))) &&
+ (isdigit(id[3]))) {
+ char numchn = id[3] - '0';
+ if (numchn == 4 || numchn == 8)
+ is_mod_file = true;
+ }
+
+ /* Oktalyzer (Amiga) */
+ if (!memcmp(id, "OKTA", 4)) {
+ is_mod_file = true;
+ }
+
+ /* Oktalyser (Atari) */
+ if (!memcmp(id, "CD81", 4)) {
+ is_mod_file = true;
+ }
+
+ /* Fasttracker */
+ if ((!memcmp(id + 1, "CHN", 3)) && (isdigit(id[0]))) {
+ is_mod_file = true;
+ }
+ /* Fasttracker or Taketracker */
+ if (((!memcmp(id + 2, "CH", 2)) || (!memcmp(id + 2, "CN", 2)))
+ && (isdigit(id[0])) && (isdigit(id[1]))) {
+ is_mod_file = true;
+ }
+
+ /* Don't try to play if we can't find a known mod type
+ * (there are mod files which have nothing to do with music) */
+ if (!is_mod_file)
+ return false;
p = id3->id3v2buf;
/* Copy Title */
- strcpy(p, &buf[0x00]);
+ if (strlcpy(p, buf, sizeof(id3->id3v2buf)) >= sizeof(id3->id3v2buf))
+ return false;
+
id3->title = p;
- p += strlen(p)+1;
id3->bitrate = filesize(fd)/1024; /* size in kb */
id3->frequency = 44100;