2004-12-24 01:43:36 +00:00
|
|
|
/* md5table - Convert a MD5 table to either PHP or C++ code
|
2010-01-23 15:27:47 +00:00
|
|
|
*
|
|
|
|
* ScummVM is the legal property of its developers, whose names
|
|
|
|
* are too numerous to list here. Please refer to the COPYRIGHT
|
|
|
|
* file distributed with this source distribution.
|
2004-12-24 01:43:36 +00:00
|
|
|
*
|
|
|
|
* 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, write to the Free Software
|
2005-10-18 01:30:26 +00:00
|
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
2004-12-24 01:43:36 +00:00
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <assert.h>
|
|
|
|
#include <stdarg.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <time.h>
|
|
|
|
|
|
|
|
void error(const char *s, ...) {
|
|
|
|
char buf[1024];
|
|
|
|
va_list va;
|
|
|
|
|
|
|
|
va_start(va, s);
|
2005-04-09 01:52:44 +00:00
|
|
|
vsnprintf(buf, 1024, s, va);
|
2004-12-24 01:43:36 +00:00
|
|
|
va_end(va);
|
|
|
|
|
|
|
|
fprintf(stderr, "ERROR: %s!\n", buf);
|
|
|
|
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
void warning(const char *s, ...) {
|
|
|
|
char buf[1024];
|
|
|
|
va_list va;
|
|
|
|
|
|
|
|
va_start(va, s);
|
2005-04-09 01:52:44 +00:00
|
|
|
vsnprintf(buf, 1024, s, va);
|
2004-12-24 01:43:36 +00:00
|
|
|
va_end(va);
|
|
|
|
|
|
|
|
fprintf(stderr, "WARNING: %s!\n", buf);
|
|
|
|
}
|
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
const char *key;
|
|
|
|
const char *value;
|
|
|
|
} StringMap;
|
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
const char *md5;
|
2006-02-12 19:13:03 +00:00
|
|
|
const char *language;
|
|
|
|
const char *platform;
|
2006-02-16 18:00:22 +00:00
|
|
|
const char *variant;
|
|
|
|
const char *extra;
|
2006-08-25 17:27:59 +00:00
|
|
|
const char *size;
|
2006-02-12 19:13:03 +00:00
|
|
|
const char *desc;
|
|
|
|
const char *comments;
|
2004-12-24 01:43:36 +00:00
|
|
|
const char *infoSource;
|
|
|
|
} Entry;
|
|
|
|
|
|
|
|
/* Map MD5 table platform names to ScummVM constant names.
|
|
|
|
* Note: Currently not many constants are defined within ScummVM. However, more
|
|
|
|
* will probably be added eventually (see also commented out constants in
|
|
|
|
* common/util.h).
|
|
|
|
*/
|
|
|
|
static const StringMap platformMap[] = {
|
2010-06-22 16:08:57 +00:00
|
|
|
{ "2gs", "kPlatformApple2GS" },
|
2006-02-16 04:52:10 +00:00
|
|
|
{ "3DO", "kPlatform3DO" },
|
2004-12-24 01:43:36 +00:00
|
|
|
{ "Amiga", "kPlatformAmiga" },
|
|
|
|
{ "Atari", "kPlatformAtariST" },
|
2005-09-07 16:01:11 +00:00
|
|
|
{ "C64", "kPlatformC64" },
|
2013-05-02 18:26:58 -04:00
|
|
|
{ "DOS", "kPlatformDOS" },
|
2007-03-03 12:59:48 +00:00
|
|
|
{ "FM-TOWNS", "kPlatformFMTowns" },
|
2011-11-27 20:11:46 -05:00
|
|
|
{ "iOS", "kPlatformIOS" },
|
2004-12-24 01:43:36 +00:00
|
|
|
{ "Mac", "kPlatformMacintosh" },
|
2005-04-11 01:45:26 +00:00
|
|
|
{ "NES", "kPlatformNES" },
|
2007-03-03 12:59:48 +00:00
|
|
|
{ "PC-Engine", "kPlatformPCEngine" },
|
2006-01-14 09:28:38 +00:00
|
|
|
{ "SEGA", "kPlatformSegaCD" },
|
2007-03-03 12:59:48 +00:00
|
|
|
{ "Windows", "kPlatformWindows" },
|
2008-09-03 01:47:01 +00:00
|
|
|
{ "Wii", "kPlatformWii" },
|
2004-12-24 01:43:36 +00:00
|
|
|
|
|
|
|
{ "All?", "kPlatformUnknown" },
|
|
|
|
{ "All", "kPlatformUnknown" },
|
|
|
|
|
|
|
|
{ 0, "kPlatformUnknown" }
|
|
|
|
};
|
|
|
|
|
|
|
|
static const StringMap langMap[] = {
|
2006-04-08 23:12:56 +00:00
|
|
|
{ "en", "EN_ANY" },
|
|
|
|
{ "us", "EN_USA" },
|
|
|
|
{ "gb", "EN_GRB" },
|
2004-12-24 01:43:36 +00:00
|
|
|
{ "de", "DE_DEU" },
|
|
|
|
{ "fr", "FR_FRA" },
|
|
|
|
{ "it", "IT_ITA" },
|
2006-08-15 19:54:41 +00:00
|
|
|
{ "br", "PT_BRA" },
|
2004-12-24 01:43:36 +00:00
|
|
|
{ "es", "ES_ESP" },
|
|
|
|
{ "jp", "JA_JPN" },
|
|
|
|
{ "zh", "ZH_TWN" },
|
|
|
|
{ "ko", "KO_KOR" },
|
|
|
|
{ "se", "SE_SWE" },
|
|
|
|
{ "en", "EN_GRB" },
|
2010-04-12 21:21:06 +00:00
|
|
|
{ "he", "HE_ISR" },
|
2004-12-24 01:43:36 +00:00
|
|
|
{ "ru", "RU_RUS" },
|
|
|
|
{ "cz", "CZ_CZE" },
|
|
|
|
{ "nl", "NL_NLD" },
|
2005-05-22 02:07:32 +00:00
|
|
|
{ "nb", "NB_NOR" },
|
2008-01-28 00:14:17 +00:00
|
|
|
{ "pl", "PL_POL" },
|
2004-12-24 01:43:36 +00:00
|
|
|
|
|
|
|
{ "All", "UNK_LANG" },
|
|
|
|
{ "All?", "UNK_LANG" },
|
|
|
|
|
|
|
|
{ 0, "UNK_LANG" }
|
|
|
|
};
|
|
|
|
|
|
|
|
static const char *php_header =
|
|
|
|
"<!--\n"
|
|
|
|
" This file was generated by the md5table tool on %s"
|
|
|
|
" DO NOT EDIT MANUALLY!\n"
|
|
|
|
" -->\n"
|
2006-02-16 18:00:22 +00:00
|
|
|
"<?php\n"
|
2004-12-24 01:43:36 +00:00
|
|
|
"\n";
|
|
|
|
|
|
|
|
static const char *c_header =
|
|
|
|
"/*\n"
|
|
|
|
" This file was generated by the md5table tool on %s"
|
|
|
|
" DO NOT EDIT MANUALLY!\n"
|
|
|
|
" */\n"
|
|
|
|
"\n"
|
|
|
|
"struct MD5Table {\n"
|
|
|
|
" const char *md5;\n"
|
2006-01-21 13:01:20 +00:00
|
|
|
" const char *gameid;\n"
|
2006-02-12 19:03:39 +00:00
|
|
|
" const char *variant;\n"
|
|
|
|
" const char *extra;\n"
|
2006-08-25 17:27:59 +00:00
|
|
|
" int32 filesize;\n"
|
2004-12-24 01:43:36 +00:00
|
|
|
" Common::Language language;\n"
|
|
|
|
" Common::Platform platform;\n"
|
|
|
|
"};\n"
|
|
|
|
"\n"
|
|
|
|
"static const MD5Table md5table[] = {\n";
|
|
|
|
|
|
|
|
static const char *c_footer =
|
2006-08-25 17:27:59 +00:00
|
|
|
" { 0, 0, 0, 0, 0, Common::UNK_LANG, Common::kPlatformUnknown }\n"
|
2004-12-24 01:43:36 +00:00
|
|
|
"};\n";
|
|
|
|
|
|
|
|
static void parseEntry(Entry *entry, char *line) {
|
|
|
|
assert(entry);
|
|
|
|
assert(line);
|
2008-01-27 19:47:41 +00:00
|
|
|
|
2004-12-24 01:43:36 +00:00
|
|
|
/* Split at the tabs */
|
2006-02-12 19:03:39 +00:00
|
|
|
entry->md5 = strtok(line, "\t\n\r");
|
2006-08-25 17:27:59 +00:00
|
|
|
entry->size = strtok(NULL, "\t\n\r");
|
2004-12-24 01:43:36 +00:00
|
|
|
entry->language = strtok(NULL, "\t\n\r");
|
2006-02-12 19:03:39 +00:00
|
|
|
entry->platform = strtok(NULL, "\t\n\r");
|
2006-02-16 18:00:22 +00:00
|
|
|
entry->variant = strtok(NULL, "\t\n\r");
|
|
|
|
entry->extra = strtok(NULL, "\t\n\r");
|
2006-02-12 19:03:39 +00:00
|
|
|
entry->desc = strtok(NULL, "\t\n\r");
|
2004-12-24 01:43:36 +00:00
|
|
|
entry->infoSource = strtok(NULL, "\t\n\r");
|
|
|
|
}
|
|
|
|
|
|
|
|
static int isEmptyLine(const char *line) {
|
|
|
|
const char *whitespace = " \t\n\r";
|
|
|
|
while (*line) {
|
|
|
|
if (!strchr(whitespace, *line))
|
|
|
|
return 0;
|
|
|
|
line++;
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const char *mapStr(const char *str, const StringMap *map) {
|
|
|
|
assert(str);
|
|
|
|
assert(map);
|
|
|
|
while (map->key) {
|
|
|
|
if (0 == strcmp(map->key, str))
|
|
|
|
return map->value;
|
|
|
|
map++;
|
|
|
|
}
|
|
|
|
warning("mapStr: unknown string '%s', defaulting to '%s'", str, map->value);
|
|
|
|
return map->value;
|
|
|
|
}
|
|
|
|
|
|
|
|
void showhelp(const char *exename)
|
|
|
|
{
|
|
|
|
printf("\nUsage: %s <params>\n", exename);
|
|
|
|
printf("\nParams:\n");
|
|
|
|
printf(" --c++ output C++ code for inclusion in ScummVM (default)\n");
|
|
|
|
printf(" --php output PHP code for the web site\n");
|
2006-02-12 19:03:39 +00:00
|
|
|
printf(" --txt output TXT file (should be identical to input file)\n");
|
2004-12-24 01:43:36 +00:00
|
|
|
exit(2);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* needed to call from qsort */
|
|
|
|
int strcmp_wrapper(const void *s1, const void *s2)
|
|
|
|
{
|
|
|
|
return strcmp((const char *)s1, (const char *)s2);
|
|
|
|
}
|
|
|
|
|
|
|
|
int main(int argc, char *argv[])
|
|
|
|
{
|
|
|
|
FILE *inFile = stdin;
|
|
|
|
FILE *outFile = stdout;
|
|
|
|
char buffer[1024];
|
2006-02-12 19:03:39 +00:00
|
|
|
char section[256];
|
|
|
|
char gameid[32];
|
2004-12-24 01:43:36 +00:00
|
|
|
char *line;
|
|
|
|
int err;
|
|
|
|
int i;
|
|
|
|
time_t theTime;
|
2009-10-02 14:42:24 +00:00
|
|
|
char *generationDate;
|
2004-12-24 01:43:36 +00:00
|
|
|
|
|
|
|
const int entrySize = 256;
|
|
|
|
int numEntries = 0, maxEntries = 1;
|
2010-05-24 13:20:35 +00:00
|
|
|
char *entriesBuffer = (char *)malloc(maxEntries * entrySize);
|
2004-12-24 01:43:36 +00:00
|
|
|
|
2006-02-12 19:03:39 +00:00
|
|
|
typedef enum {
|
|
|
|
kCPPOutput,
|
|
|
|
kPHPOutput,
|
|
|
|
kTXTOutput
|
|
|
|
} OutputMode;
|
|
|
|
|
|
|
|
OutputMode outputMode = kCPPOutput;
|
2004-12-24 01:43:36 +00:00
|
|
|
|
|
|
|
if (argc != 2)
|
|
|
|
showhelp(argv[0]);
|
|
|
|
if (strcmp(argv[1], "--c++") == 0) {
|
2006-02-12 19:03:39 +00:00
|
|
|
outputMode = kCPPOutput;
|
2004-12-24 01:43:36 +00:00
|
|
|
} else if (strcmp(argv[1], "--php") == 0) {
|
2006-02-12 19:03:39 +00:00
|
|
|
outputMode = kPHPOutput;
|
|
|
|
} else if (strcmp(argv[1], "--txt") == 0) {
|
|
|
|
outputMode = kTXTOutput;
|
2004-12-24 01:43:36 +00:00
|
|
|
} else {
|
|
|
|
showhelp(argv[0]);
|
|
|
|
}
|
2008-01-27 19:47:41 +00:00
|
|
|
|
2004-12-24 01:43:36 +00:00
|
|
|
time(&theTime);
|
|
|
|
generationDate = strdup(asctime(gmtime(&theTime)));
|
|
|
|
|
2006-02-12 19:03:39 +00:00
|
|
|
if (outputMode == kPHPOutput)
|
2004-12-24 01:43:36 +00:00
|
|
|
fprintf(outFile, php_header, generationDate);
|
|
|
|
|
2006-02-12 19:03:39 +00:00
|
|
|
section[0] = 0;
|
|
|
|
gameid[0] = 0;
|
2004-12-24 01:43:36 +00:00
|
|
|
while ((line = fgets(buffer, sizeof(buffer), inFile))) {
|
|
|
|
/* Parse line */
|
2006-02-12 19:13:03 +00:00
|
|
|
if (line[0] == '#' || isEmptyLine(line)) {
|
|
|
|
if (outputMode == kTXTOutput)
|
|
|
|
fprintf(outFile, "%s", line);
|
2004-12-24 01:43:36 +00:00
|
|
|
continue; /* Skip comments & empty lines */
|
2006-02-12 19:13:03 +00:00
|
|
|
}
|
2004-12-24 01:43:36 +00:00
|
|
|
if (line[0] == '\t') {
|
|
|
|
Entry entry;
|
|
|
|
assert(section[0]);
|
|
|
|
parseEntry(&entry, line+1);
|
2006-02-12 19:03:39 +00:00
|
|
|
if (outputMode == kPHPOutput) {
|
2006-02-16 18:00:22 +00:00
|
|
|
fprintf(outFile, "\taddEntry(");
|
2008-01-27 19:47:41 +00:00
|
|
|
|
2006-02-16 18:00:22 +00:00
|
|
|
// Print the description string
|
|
|
|
fprintf(outFile, "\"");
|
|
|
|
if (entry.extra && strcmp(entry.extra, "-")) {
|
|
|
|
fprintf(outFile, "%s", entry.extra);
|
|
|
|
if (entry.desc && strcmp(entry.desc, "-"))
|
|
|
|
fprintf(outFile, " (%s)", entry.desc);
|
|
|
|
}
|
|
|
|
fprintf(outFile, "\", ");
|
|
|
|
|
2004-12-24 01:43:36 +00:00
|
|
|
fprintf(outFile, "\"%s\", ", entry.platform);
|
|
|
|
fprintf(outFile, "\"%s\", ", entry.language);
|
2006-02-16 18:00:22 +00:00
|
|
|
fprintf(outFile, "\"%s\"", entry.md5);
|
2004-12-24 01:43:36 +00:00
|
|
|
if (entry.infoSource)
|
|
|
|
fprintf(outFile, ", \"%s\"", entry.infoSource);
|
2006-02-16 18:00:22 +00:00
|
|
|
fprintf(outFile, ");\n");
|
2006-02-12 19:03:39 +00:00
|
|
|
} else if (outputMode == kTXTOutput) {
|
2006-08-25 17:27:59 +00:00
|
|
|
fprintf(outFile, "\t%s\t%s\t%s\t%s\t%s\t%s\t%s\t%s\n",
|
2006-02-12 19:03:39 +00:00
|
|
|
entry.md5,
|
2006-08-25 17:27:59 +00:00
|
|
|
entry.size,
|
2006-02-12 19:03:39 +00:00
|
|
|
entry.language,
|
|
|
|
entry.platform,
|
2006-02-16 18:00:22 +00:00
|
|
|
entry.variant,
|
|
|
|
entry.extra,
|
2006-02-12 19:03:39 +00:00
|
|
|
entry.desc,
|
|
|
|
entry.infoSource ? entry.infoSource : ""
|
|
|
|
);
|
2004-12-24 01:43:36 +00:00
|
|
|
} else if (entry.md5) {
|
|
|
|
if (numEntries >= maxEntries) {
|
|
|
|
maxEntries *= 2;
|
2010-05-24 13:20:35 +00:00
|
|
|
entriesBuffer = (char *)realloc(entriesBuffer, maxEntries * entrySize);
|
2004-12-24 01:43:36 +00:00
|
|
|
}
|
2006-02-16 18:00:22 +00:00
|
|
|
if (0 == strcmp(entry.variant, "-"))
|
|
|
|
entry.variant = "";
|
|
|
|
if (0 == strcmp(entry.extra, "-"))
|
|
|
|
entry.extra = "";
|
2006-02-12 19:03:39 +00:00
|
|
|
snprintf(entriesBuffer + numEntries * entrySize, entrySize,
|
2006-08-25 17:27:59 +00:00
|
|
|
"\t{ \"%s\", \"%s\", \"%s\", \"%s\", %s, Common::%s, Common::%s },\n",
|
|
|
|
entry.md5,
|
|
|
|
gameid,
|
|
|
|
entry.variant,
|
|
|
|
entry.extra,
|
|
|
|
entry.size,
|
|
|
|
mapStr(entry.language, langMap),
|
|
|
|
mapStr(entry.platform, platformMap));
|
2004-12-24 01:43:36 +00:00
|
|
|
numEntries++;
|
|
|
|
}
|
|
|
|
} else {
|
2006-02-16 18:00:22 +00:00
|
|
|
if (outputMode == kPHPOutput && gameid[0] != 0) {
|
|
|
|
// If there is an active section, close it now
|
|
|
|
fprintf(outFile, "endSection();\n");
|
|
|
|
}
|
2006-02-12 19:03:39 +00:00
|
|
|
// Read the gameid, followed by a tab
|
|
|
|
for (i = 0; *line && *line != '\t'; ++i)
|
|
|
|
gameid[i] = *line++;
|
|
|
|
assert(i > 0);
|
|
|
|
gameid[i] = 0;
|
|
|
|
assert(*line != 0);
|
|
|
|
line++;
|
|
|
|
|
|
|
|
// Read the section header (usually the full game name)
|
|
|
|
for (i = 0; *line && *line != '\n'; ++i)
|
|
|
|
section[i] = *line++;
|
|
|
|
assert(i > 0);
|
2004-12-24 01:43:36 +00:00
|
|
|
section[i] = 0;
|
2006-02-12 19:03:39 +00:00
|
|
|
|
|
|
|
// If in PHP or TXT mode, we write the output immediately
|
|
|
|
if (outputMode == kPHPOutput) {
|
2006-02-16 18:00:22 +00:00
|
|
|
fprintf(outFile, "beginSection(\"%s\", \"%s\");\n", section, gameid);
|
2006-02-12 19:03:39 +00:00
|
|
|
} else if (outputMode == kTXTOutput) {
|
|
|
|
fprintf(outFile, "%s\t%s\n", gameid, section);
|
2004-12-24 01:43:36 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
err = ferror(inFile);
|
|
|
|
if (err)
|
|
|
|
error("Failed reading from input file, error %d", err);
|
|
|
|
|
2006-02-16 18:00:22 +00:00
|
|
|
if (outputMode == kPHPOutput) {
|
|
|
|
if (gameid[0] != 0) // If there is an active section, close it now
|
|
|
|
fprintf(outFile, "endSection();\n");
|
2008-01-27 19:47:41 +00:00
|
|
|
|
2006-02-16 18:00:22 +00:00
|
|
|
fprintf(outFile, "?>\n");
|
|
|
|
}
|
|
|
|
|
2006-02-12 19:03:39 +00:00
|
|
|
if (outputMode == kCPPOutput) {
|
2004-12-24 01:43:36 +00:00
|
|
|
/* Printf header */
|
|
|
|
fprintf(outFile, c_header, generationDate);
|
|
|
|
/* Now sort the MD5 table (this allows for binary searches) */
|
|
|
|
qsort(entriesBuffer, numEntries, entrySize, strcmp_wrapper);
|
2005-03-30 20:20:06 +00:00
|
|
|
/* Output the table and emit warnings if duplicate md5s are found */
|
|
|
|
buffer[0] = '\0';
|
|
|
|
for (i = 0; i < numEntries; ++i) {
|
|
|
|
const char *currentEntry = entriesBuffer + i * entrySize;
|
2009-04-04 13:15:41 +00:00
|
|
|
fprintf(outFile, "%s", currentEntry);
|
2005-03-30 20:20:06 +00:00
|
|
|
if (strncmp(currentEntry + 4, buffer, 32) == 0) {
|
|
|
|
warning("Duplicate MD5 found '%.32s'", buffer);
|
|
|
|
} else {
|
|
|
|
strncpy(buffer, currentEntry + 4, 32);
|
|
|
|
}
|
|
|
|
}
|
2004-12-24 01:43:36 +00:00
|
|
|
/* Finally, print the footer */
|
2009-04-04 13:15:41 +00:00
|
|
|
fprintf(outFile, "%s", c_footer);
|
2004-12-24 01:43:36 +00:00
|
|
|
}
|
2008-01-27 19:47:41 +00:00
|
|
|
|
2004-12-24 01:43:36 +00:00
|
|
|
free(entriesBuffer);
|
2009-10-02 14:42:24 +00:00
|
|
|
free(generationDate);
|
2004-12-24 01:43:36 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|