2021-02-27 00:44:50 +10:00

1994 lines
56 KiB
C

#include "rc_hash.h"
#include "../rcheevos/rc_compat.h"
#include "md5.h"
#include <ctype.h>
/* arbitrary limit to prevent allocating and hashing large files */
#define MAX_BUFFER_SIZE 64 * 1024 * 1024
const char* rc_path_get_filename(const char* path);
/* ===================================================== */
static rc_hash_message_callback error_message_callback = NULL;
rc_hash_message_callback verbose_message_callback = NULL;
void rc_hash_init_error_message_callback(rc_hash_message_callback callback)
{
error_message_callback = callback;
}
int rc_hash_error(const char* message)
{
if (error_message_callback)
error_message_callback(message);
return 0;
}
void rc_hash_init_verbose_message_callback(rc_hash_message_callback callback)
{
verbose_message_callback = callback;
}
static void rc_hash_verbose(const char* message)
{
if (verbose_message_callback)
verbose_message_callback(message);
}
/* ===================================================== */
static struct rc_hash_filereader filereader_funcs;
static struct rc_hash_filereader* filereader = NULL;
static void* filereader_open(const char* path)
{
return fopen(path, "rb");
}
static void filereader_seek(void* file_handle, size_t offset, int origin)
{
fseek((FILE*)file_handle, (long)offset, origin);
}
static size_t filereader_tell(void* file_handle)
{
return ftell((FILE*)file_handle);
}
static size_t filereader_read(void* file_handle, void* buffer, size_t requested_bytes)
{
return fread(buffer, 1, requested_bytes, (FILE*)file_handle);
}
static void filereader_close(void* file_handle)
{
fclose((FILE*)file_handle);
}
void rc_hash_init_custom_filereader(struct rc_hash_filereader* reader)
{
/* initialize with defaults first */
filereader_funcs.open = filereader_open;
filereader_funcs.seek = filereader_seek;
filereader_funcs.tell = filereader_tell;
filereader_funcs.read = filereader_read;
filereader_funcs.close = filereader_close;
/* hook up any provided custom handlers */
if (reader) {
if (reader->open)
filereader_funcs.open = reader->open;
if (reader->seek)
filereader_funcs.seek = reader->seek;
if (reader->tell)
filereader_funcs.tell = reader->tell;
if (reader->read)
filereader_funcs.read = reader->read;
if (reader->close)
filereader_funcs.close = reader->close;
}
filereader = &filereader_funcs;
}
void* rc_file_open(const char* path)
{
void* handle;
if (!filereader)
rc_hash_init_custom_filereader(NULL);
handle = filereader->open(path);
if (handle && verbose_message_callback)
{
char message[1024];
snprintf(message, sizeof(message), "Opened %s", rc_path_get_filename(path));
verbose_message_callback(message);
}
return handle;
}
void rc_file_seek(void* file_handle, size_t offset, int origin)
{
if (filereader)
filereader->seek(file_handle, offset, origin);
}
size_t rc_file_tell(void* file_handle)
{
return (filereader) ? filereader->tell(file_handle) : 0;
}
size_t rc_file_read(void* file_handle, void* buffer, int requested_bytes)
{
return (filereader) ? filereader->read(file_handle, buffer, requested_bytes) : 0;
}
void rc_file_close(void* file_handle)
{
if (filereader)
filereader->close(file_handle);
}
/* ===================================================== */
static struct rc_hash_cdreader cdreader_funcs;
struct rc_hash_cdreader* cdreader = NULL;
void rc_hash_init_custom_cdreader(struct rc_hash_cdreader* reader)
{
if (reader)
{
memcpy(&cdreader_funcs, reader, sizeof(cdreader_funcs));
cdreader = &cdreader_funcs;
}
else
{
cdreader = NULL;
}
}
static void* rc_cd_open_track(const char* path, uint32_t track)
{
if (cdreader && cdreader->open_track)
return cdreader->open_track(path, track);
rc_hash_error("no hook registered for cdreader_open_track");
return NULL;
}
static size_t rc_cd_read_sector(void* track_handle, uint32_t sector, void* buffer, size_t requested_bytes)
{
if (cdreader && cdreader->read_sector)
return cdreader->read_sector(track_handle, sector, buffer, requested_bytes);
rc_hash_error("no hook registered for cdreader_read_sector");
return 0;
}
static uint32_t rc_cd_absolute_sector_to_track_sector(void* track_handle, uint32_t sector)
{
if (cdreader && cdreader->absolute_sector_to_track_sector)
return cdreader->absolute_sector_to_track_sector(track_handle, sector);
rc_hash_error("no hook registered for cdreader_absolute_sector_to_track_sector");
return sector;
}
static void rc_cd_close_track(void* track_handle)
{
if (cdreader && cdreader->close_track)
{
cdreader->close_track(track_handle);
return;
}
rc_hash_error("no hook registered for cdreader_close_track");
}
static uint32_t rc_cd_find_file_sector(void* track_handle, const char* path, unsigned* size)
{
uint8_t buffer[2048], *tmp;
int sector;
size_t filename_length;
const char* slash;
if (!track_handle)
return 0;
filename_length = strlen(path);
slash = strrchr(path, '\\');
if (slash)
{
/* find the directory record for the first part of the path */
memcpy(buffer, path, slash - path);
buffer[slash - path] = '\0';
sector = rc_cd_find_file_sector(track_handle, (const char *)buffer, NULL);
if (!sector)
return 0;
++slash;
filename_length -= (slash - path);
path = slash;
}
else
{
/* find the cd information */
if (!rc_cd_read_sector(track_handle, 16, buffer, 256))
return 0;
/* the directory_record starts at 156, the sector containing the table of contents is 2 bytes into that.
* https://www.cdroller.com/htm/readdata.html
*/
sector = buffer[156 + 2] | (buffer[156 + 3] << 8) | (buffer[156 + 4] << 16);
}
/* fetch and process the directory record */
sector = rc_cd_absolute_sector_to_track_sector(track_handle, sector);
if (!rc_cd_read_sector(track_handle, sector, buffer, sizeof(buffer)))
return 0;
tmp = buffer;
while (tmp < buffer + sizeof(buffer))
{
if (!*tmp)
return 0;
/* filename is 33 bytes into the record and the format is "FILENAME;version" or "DIRECTORY" */
if ((tmp[33 + filename_length] == ';' || tmp[33 + filename_length] == '\0') &&
strncasecmp((const char*)(tmp + 33), path, filename_length) == 0)
{
sector = tmp[2] | (tmp[3] << 8) | (tmp[4] << 16);
if (verbose_message_callback)
{
snprintf((char*)buffer, sizeof(buffer), "Found %s at sector %d", path, sector);
verbose_message_callback((const char*)buffer);
}
if (size)
*size = tmp[10] | (tmp[11] << 8) | (tmp[12] << 16) | (tmp[13] << 24);
return sector;
}
/* the first byte of the record is the length of the record */
tmp += *tmp;
}
return 0;
}
/* ===================================================== */
const char* rc_path_get_filename(const char* path)
{
const char* ptr = path + strlen(path);
do
{
if (ptr[-1] == '/' || ptr[-1] == '\\')
break;
--ptr;
} while (ptr > path);
return ptr;
}
static const char* rc_path_get_extension(const char* path)
{
const char* ptr = path + strlen(path);
do
{
if (ptr[-1] == '.')
return ptr;
--ptr;
} while (ptr > path);
return path + strlen(path);
}
int rc_path_compare_extension(const char* path, const char* ext)
{
size_t path_len = strlen(path);
size_t ext_len = strlen(ext);
const char* ptr = path + path_len - ext_len;
if (ptr[-1] != '.')
return 0;
if (memcmp(ptr, ext, ext_len) == 0)
return 1;
do
{
if (tolower(*ptr) != *ext)
return 0;
++ext;
++ptr;
} while (*ptr);
return 1;
}
/* ===================================================== */
static int rc_hash_finalize(md5_state_t* md5, char hash[33])
{
md5_byte_t digest[16];
md5_finish(md5, digest);
/* NOTE: sizeof(hash) is 4 because it's still treated like a pointer, despite specifying a size */
snprintf(hash, 33, "%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x",
digest[0], digest[1], digest[2], digest[3], digest[4], digest[5], digest[6], digest[7],
digest[8], digest[9], digest[10], digest[11], digest[12], digest[13], digest[14], digest[15]
);
if (verbose_message_callback)
{
char message[128];
snprintf(message, sizeof(message), "Generated hash %s", hash);
verbose_message_callback(message);
}
return 1;
}
static int rc_hash_buffer(char hash[33], uint8_t* buffer, size_t buffer_size)
{
md5_state_t md5;
md5_init(&md5);
if (buffer_size > MAX_BUFFER_SIZE)
buffer_size = MAX_BUFFER_SIZE;
md5_append(&md5, buffer, (int)buffer_size);
if (verbose_message_callback)
{
char message[128];
snprintf(message, sizeof(message), "Hashing %u byte buffer", (unsigned)buffer_size);
verbose_message_callback(message);
}
return rc_hash_finalize(&md5, hash);
}
static int rc_hash_3do(char hash[33], const char* path)
{
uint8_t buffer[2048];
const uint8_t operafs_identifier[7] = { 0x01, 0x5A, 0x5A, 0x5A, 0x5A, 0x5A, 0x01 };
void* track_handle;
md5_state_t md5;
int sector;
int block_size, block_location;
int offset, stop;
size_t size = 0;
track_handle = rc_cd_open_track(path, 1);
if (!track_handle)
return rc_hash_error("Could not open track");
/* the Opera filesystem stores the volume information in the first 132 bytes of sector 0
* https://github.com/barbeque/3dodump/blob/master/OperaFS-Format.md
*/
rc_cd_read_sector(track_handle, 0, buffer, 132);
if (memcmp(buffer, operafs_identifier, sizeof(operafs_identifier)) == 0)
{
if (verbose_message_callback)
{
char message[128];
snprintf(message, sizeof(message), "Found 3DO CD, title=%.32s", &buffer[0x28]);
verbose_message_callback(message);
}
/* include the volume header in the hash */
md5_init(&md5);
md5_append(&md5, buffer, 132);
/* the block size is at offset 0x4C (assume 0x4C is always 0) */
block_size = buffer[0x4D] * 65536 + buffer[0x4E] * 256 + buffer[0x4F];
/* the root directory block location is at offset 0x64 (and duplicated several
* times, but we just look at the primary record) (assume 0x64 is always 0)*/
block_location = buffer[0x65] * 65536 + buffer[0x66] * 256 + buffer[0x67];
/* multiply the block index by the block size to get the real address */
block_location *= block_size;
/* convert that to a sector and read it */
sector = block_location / 2048;
do
{
rc_cd_read_sector(track_handle, sector, buffer, sizeof(buffer));
/* offset to start of entries is at offset 0x10 (assume 0x10 and 0x11 are always 0) */
offset = buffer[0x12] * 256 + buffer[0x13];
/* offset to end of entries is at offset 0x0C (assume 0x0C is always 0) */
stop = buffer[0x0D] * 65536 + buffer[0x0E] * 256 + buffer[0x0F];
while (offset < stop)
{
if (buffer[offset + 0x03] == 0x02) /* file */
{
if (strcasecmp((const char*)&buffer[offset + 0x20], "LaunchMe") == 0)
{
/* the block size is at offset 0x0C (assume 0x0C is always 0) */
block_size = buffer[offset + 0x0D] * 65536 + buffer[offset + 0x0E] * 256 + buffer[offset + 0x0F];
/* the block location is at offset 0x44 (assume 0x44 is always 0) */
block_location = buffer[offset + 0x45] * 65536 + buffer[offset + 0x46] * 256 + buffer[offset + 0x47];
block_location *= block_size;
/* the file size is at offset 0x10 (assume 0x10 is always 0) */
size = buffer[offset + 0x11] * 65536 + buffer[offset + 0x12] * 256 + buffer[offset + 0x13];
if (verbose_message_callback)
{
char message[128];
snprintf(message, sizeof(message), "Hashing header (%u bytes) and %.32s (%u bytes) ", 132, &buffer[offset + 0x20], (unsigned)size);
verbose_message_callback(message);
}
break;
}
}
/* the number of extra copies of the file is at offset 0x40 (assume 0x40-0x42 are always 0) */
offset += 0x48 + buffer[offset + 0x43] * 4;
}
if (size != 0)
break;
/* did not find the file, see if the directory listing is continued in another sector */
offset = buffer[0x02] * 256 + buffer[0x03];
/* no more sectors to search*/
if (offset == 0xFFFF)
break;
/* get next sector */
offset *= block_size;
sector = (block_location + offset) / 2048;
} while (1);
if (size == 0)
{
rc_cd_close_track(track_handle);
return rc_hash_error("Could not find LaunchMe");
}
sector = block_location / 2048;
while (size > 2048)
{
rc_cd_read_sector(track_handle, sector, buffer, sizeof(buffer));
md5_append(&md5, buffer, sizeof(buffer));
++sector;
size -= 2048;
}
rc_cd_read_sector(track_handle, sector, buffer, size);
md5_append(&md5, buffer, (int)size);
}
else
{
rc_cd_close_track(track_handle);
return rc_hash_error("Not a 3DO CD");
}
rc_cd_close_track(track_handle);
return rc_hash_finalize(&md5, hash);
}
static int rc_hash_arcade(char hash[33], const char* path)
{
/* arcade hash is just the hash of the filename (no extension) - the cores are pretty stringent about having the right ROM data */
const char* filename = rc_path_get_filename(path);
const char* ext = rc_path_get_extension(filename);
size_t filename_length = ext - filename - 1;
/* fbneo supports loading subsystems by using specific folder names.
* if one is found, include it in the hash.
* https://github.com/libretro/FBNeo/blob/master/src/burner/libretro/README.md#emulating-consoles
*/
if (filename > path + 1)
{
int include_folder = 0;
const char* folder = filename - 1;
size_t parent_folder_length = 0;
do
{
if (folder[-1] == '/' || folder[-1] == '\\')
break;
--folder;
} while (folder > path);
parent_folder_length = filename - folder - 1;
switch (parent_folder_length)
{
case 3:
if (memcmp(folder, "nes", 3) == 0 ||
memcmp(folder, "fds", 3) == 0 ||
memcmp(folder, "sms", 3) == 0 ||
memcmp(folder, "msx", 3) == 0 ||
memcmp(folder, "ngp", 3) == 0 ||
memcmp(folder, "pce", 3) == 0 ||
memcmp(folder, "sgx", 3) == 0)
include_folder = 1;
break;
case 4:
if (memcmp(folder, "tg16", 4) == 0)
include_folder = 1;
break;
case 6:
if (memcmp(folder, "coleco", 6) == 0 ||
memcmp(folder, "sg1000", 6) == 0)
include_folder = 1;
break;
case 8:
if (memcmp(folder, "gamegear", 8) == 0 ||
memcmp(folder, "megadriv", 8) == 0 ||
memcmp(folder, "spectrum", 8) == 0)
include_folder = 1;
break;
default:
break;
}
if (include_folder)
{
char buffer[128]; /* realistically, this should never need more than ~20 characters */
if (parent_folder_length + filename_length + 1 < sizeof(buffer))
{
memcpy(&buffer[0], folder, parent_folder_length);
buffer[parent_folder_length] = '_';
memcpy(&buffer[parent_folder_length + 1], filename, filename_length);
return rc_hash_buffer(hash, (uint8_t*)&buffer[0], parent_folder_length + filename_length + 1);
}
}
}
return rc_hash_buffer(hash, (uint8_t*)filename, filename_length);
}
static int rc_hash_lynx(char hash[33], uint8_t* buffer, size_t buffer_size)
{
/* if the file contains a header, ignore it */
if (buffer[0] == 'L' && buffer[1] == 'Y' && buffer[2] == 'N' && buffer[3] == 'X' && buffer[4] == 0)
{
rc_hash_verbose("Ignoring LYNX header");
buffer += 64;
buffer_size -= 64;
}
return rc_hash_buffer(hash, buffer, buffer_size);
}
static int rc_hash_nes(char hash[33], uint8_t* buffer, size_t buffer_size)
{
/* if the file contains a header, ignore it */
if (buffer[0] == 'N' && buffer[1] == 'E' && buffer[2] == 'S' && buffer[3] == 0x1A)
{
rc_hash_verbose("Ignoring NES header");
buffer += 16;
buffer_size -= 16;
}
else if (buffer[0] == 'F' && buffer[1] == 'D' && buffer[2] == 'S' && buffer[3] == 0x1A)
{
rc_hash_verbose("Ignoring FDS header");
buffer += 16;
buffer_size -= 16;
}
return rc_hash_buffer(hash, buffer, buffer_size);
}
static int rc_hash_nintendo_ds(char hash[33], const char* path)
{
uint8_t header[512];
uint8_t* hash_buffer;
unsigned int hash_size, arm9_size, arm9_addr, arm7_size, arm7_addr, icon_addr;
size_t num_read;
int offset = 0;
md5_state_t md5;
void* file_handle;
file_handle = rc_file_open(path);
if (!file_handle)
return rc_hash_error("Could not open file");
rc_file_seek(file_handle, 0, SEEK_SET);
if (rc_file_read(file_handle, header, sizeof(header)) != 512)
return rc_hash_error("Failed to read header");
if (header[0] == 0x2E && header[1] == 0x00 && header[2] == 0x00 && header[3] == 0xEA &&
header[0xB0] == 0x44 && header[0xB1] == 0x46 && header[0xB2] == 0x96 && header[0xB3] == 0)
{
/* SuperCard header detected, ignore it */
rc_hash_verbose("Ignoring SuperCard header");
offset = 512;
rc_file_seek(file_handle, offset, SEEK_SET);
rc_file_read(file_handle, header, sizeof(header));
}
arm9_addr = header[0x20] | (header[0x21] << 8) | (header[0x22] << 16) | (header[0x23] << 24);
arm9_size = header[0x2C] | (header[0x2D] << 8) | (header[0x2E] << 16) | (header[0x2F] << 24);
arm7_addr = header[0x30] | (header[0x31] << 8) | (header[0x32] << 16) | (header[0x33] << 24);
arm7_size = header[0x3C] | (header[0x3D] << 8) | (header[0x3E] << 16) | (header[0x3F] << 24);
icon_addr = header[0x68] | (header[0x69] << 8) | (header[0x6A] << 16) | (header[0x6B] << 24);
if (arm9_size + arm7_size > 16 * 1024 * 1024)
{
/* sanity check - code blocks are typically less than 1MB each - assume not a DS ROM */
snprintf((char*)header, sizeof(header), "arm9 code size (%u) + arm7 code size (%u) exceeds 16MB", arm9_size, arm7_size);
return rc_hash_error((const char*)header);
}
hash_size = 0xA00;
if (arm9_size > hash_size)
hash_size = arm9_size;
if (arm7_size > hash_size)
hash_size = arm7_size;
hash_buffer = (uint8_t*)malloc(hash_size);
if (!hash_buffer)
{
rc_file_close(file_handle);
snprintf((char*)header, sizeof(header), "Failed to allocate %u bytes", hash_size);
return rc_hash_error((const char*)header);
}
md5_init(&md5);
rc_hash_verbose("Hashing 352 byte header");
md5_append(&md5, header, 0x160);
if (verbose_message_callback)
{
snprintf((char*)header, sizeof(header), "Hashing %u byte arm9 code (at %08X)", arm9_size, arm9_addr);
verbose_message_callback((const char*)header);
}
rc_file_seek(file_handle, arm9_addr + offset, SEEK_SET);
rc_file_read(file_handle, hash_buffer, arm9_size);
md5_append(&md5, hash_buffer, arm9_size);
if (verbose_message_callback)
{
snprintf((char*)header, sizeof(header), "Hashing %u byte arm7 code (at %08X)", arm7_size, arm7_addr);
verbose_message_callback((const char*)header);
}
rc_file_seek(file_handle, arm7_addr + offset, SEEK_SET);
rc_file_read(file_handle, hash_buffer, arm7_size);
md5_append(&md5, hash_buffer, arm7_size);
if (verbose_message_callback)
{
snprintf((char*)header, sizeof(header), "Hashing 2560 byte icon and labels data (at %08X)", icon_addr);
verbose_message_callback((const char*)header);
}
rc_file_seek(file_handle, icon_addr + offset, SEEK_SET);
num_read = rc_file_read(file_handle, hash_buffer, 0xA00);
if (num_read < 0xA00)
{
/* some homebrew games don't provide a full icon block, and no data after the icon block.
* if we didn't get a full icon block, fill the remaining portion with 0s
*/
if (verbose_message_callback)
{
snprintf((char*)header, sizeof(header), "Warning: only got %u bytes for icon and labels data, 0-padding to 2560 bytes", (unsigned)num_read);
verbose_message_callback((const char*)header);
}
memset(&hash_buffer[num_read], 0, 0xA00 - num_read);
}
md5_append(&md5, hash_buffer, 0xA00);
free(hash_buffer);
rc_file_close(file_handle);
return rc_hash_finalize(&md5, hash);
}
static int rc_hash_pce(char hash[33], uint8_t* buffer, size_t buffer_size)
{
/* if the file contains a header, ignore it (expect ROM data to be multiple of 128KB) */
uint32_t calc_size = ((uint32_t)buffer_size / 0x20000) * 0x20000;
if (buffer_size - calc_size == 512)
{
rc_hash_verbose("Ignoring PCE header");
buffer += 512;
buffer_size -= 512;
}
return rc_hash_buffer(hash, buffer, buffer_size);
}
static int rc_hash_pce_track(char hash[33], void* track_handle)
{
uint8_t buffer[2048];
md5_state_t md5;
int sector, num_sectors;
unsigned size;
/* the PC-Engine uses the second sector to specify boot information and program name.
* the string "PC Engine CD-ROM SYSTEM" should exist at 32 bytes into the sector
* http://shu.sheldows.com/shu/download/pcedocs/pce_cdrom.html
*/
rc_cd_read_sector(track_handle, 1, buffer, 128);
/* normal PC Engine CD will have a header block in sector 1 */
if (memcmp("PC Engine CD-ROM SYSTEM", &buffer[32], 23) == 0)
{
/* the title of the disc is the last 22 bytes of the header */
md5_init(&md5);
md5_append(&md5, &buffer[106], 22);
if (verbose_message_callback)
{
char message[128];
buffer[128] = '\0';
snprintf(message, sizeof(message), "Found PC Engine CD, title=%.22s", &buffer[106]);
verbose_message_callback(message);
}
/* the first three bytes specify the sector of the program data, and the fourth byte
* is the number of sectors.
*/
sector = (buffer[0] << 16) + (buffer[1] << 8) + buffer[2];
num_sectors = buffer[3];
if (verbose_message_callback)
{
char message[128];
snprintf(message, sizeof(message), "Hashing %d sectors starting at sector %d", num_sectors, sector);
verbose_message_callback(message);
}
while (num_sectors > 0)
{
rc_cd_read_sector(track_handle, sector, buffer, sizeof(buffer));
md5_append(&md5, buffer, sizeof(buffer));
++sector;
--num_sectors;
}
}
/* GameExpress CDs use a standard Joliet filesystem - locate and hash the BOOT.BIN */
else if ((sector = rc_cd_find_file_sector(track_handle, "BOOT.BIN", &size)) != 0 && size < MAX_BUFFER_SIZE)
{
md5_init(&md5);
while (size > sizeof(buffer))
{
rc_cd_read_sector(track_handle, sector, buffer, sizeof(buffer));
md5_append(&md5, buffer, sizeof(buffer));
++sector;
size -= sizeof(buffer);
}
if (size > 0)
{
rc_cd_read_sector(track_handle, sector, buffer, size);
md5_append(&md5, buffer, size);
}
}
else
{
return rc_hash_error("Not a PC Engine CD");
}
return rc_hash_finalize(&md5, hash);
}
static int rc_hash_pce_cd(char hash[33], const char* path)
{
int result;
void* track_handle = rc_cd_open_track(path, RC_HASH_CDTRACK_FIRST_DATA);
if (!track_handle)
return rc_hash_error("Could not open track");
result = rc_hash_pce_track(hash, track_handle);
rc_cd_close_track(track_handle);
return result;
}
static int rc_hash_pcfx_cd(char hash[33], const char* path)
{
uint8_t buffer[2048];
void* track_handle;
md5_state_t md5;
int sector, num_sectors;
/* PC-FX executable can be in any track. Assume it's in the largest data track and check there first */
track_handle = rc_cd_open_track(path, RC_HASH_CDTRACK_LARGEST);
if (!track_handle)
return rc_hash_error("Could not open track");
/* PC-FX CD will have a header marker in sector 0 */
rc_cd_read_sector(track_handle, 0, buffer, 32);
if (memcmp("PC-FX:Hu_CD-ROM", &buffer[0], 15) != 0)
{
rc_cd_close_track(track_handle);
/* not found in the largest data track, check track 2 */
track_handle = rc_cd_open_track(path, 2);
if (!track_handle)
return rc_hash_error("Could not open track");
rc_cd_read_sector(track_handle, 0, buffer, 32);
}
if (memcmp("PC-FX:Hu_CD-ROM", &buffer[0], 15) == 0)
{
/* PC-FX boot header fills the first two sectors of the disc
* https://bitbucket.org/trap15/pcfxtools/src/master/pcfx-cdlink.c
* the important stuff is the first 128 bytes of the second sector (title being the first 32) */
rc_cd_read_sector(track_handle, 1, buffer, 128);
md5_init(&md5);
md5_append(&md5, buffer, 128);
if (verbose_message_callback)
{
char message[128];
buffer[128] = '\0';
snprintf(message, sizeof(message), "Found PC-FX CD, title=%.32s", &buffer[0]);
verbose_message_callback(message);
}
/* the program sector is in bytes 33-36 (assume byte 36 is 0) */
sector = (buffer[34] << 16) + (buffer[33] << 8) + buffer[32];
/* the number of sectors the program occupies is in bytes 37-40 (assume byte 40 is 0) */
num_sectors = (buffer[38] << 16) + (buffer[37] << 8) + buffer[36];
if (verbose_message_callback)
{
char message[128];
snprintf(message, sizeof(message), "Hashing %d sectors starting at sector %d", num_sectors, sector);
verbose_message_callback(message);
}
while (num_sectors > 0)
{
rc_cd_read_sector(track_handle, sector, buffer, sizeof(buffer));
md5_append(&md5, buffer, sizeof(buffer));
++sector;
--num_sectors;
}
}
else
{
int result = 0;
rc_cd_read_sector(track_handle, 1, buffer, 128);
/* some PC-FX CDs still identify as PCE CDs */
if (memcmp("PC Engine CD-ROM SYSTEM", &buffer[32], 23) == 0)
result = rc_hash_pce_track(hash, track_handle);
rc_cd_close_track(track_handle);
if (result)
return result;
return rc_hash_error("Not a PC-FX CD");
}
rc_cd_close_track(track_handle);
return rc_hash_finalize(&md5, hash);
}
static int rc_hash_dreamcast(char hash[33], const char* path)
{
uint8_t buffer[2048];
void* track_handle;
void* last_track_handle;
char exe_file[32] = "";
unsigned size;
size_t num_read = 0;
uint32_t sector;
int result = 0;
md5_state_t md5;
int i = 0;
/* track 03 is the data track that contains the TOC and IP.BIN */
track_handle = rc_cd_open_track(path, 3);
if (!track_handle)
return rc_hash_error("Could not open track");
/* first 256 bytes from first sector should have IP.BIN structure that stores game meta information
* https://mc.pp.se/dc/ip.bin.html */
rc_cd_read_sector(track_handle, 0, buffer, sizeof(buffer));
if (memcmp(&buffer[0], "SEGA SEGAKATANA ", 16) != 0)
{
rc_cd_close_track(track_handle);
return rc_hash_error("Not a Dreamcast CD");
}
md5_init(&md5);
md5_append(&md5, (md5_byte_t*)buffer, 256);
if (verbose_message_callback)
{
char message[256];
uint8_t* ptr = &buffer[0xFF];
while (ptr > &buffer[0x80] && ptr[-1] == ' ')
--ptr;
*ptr = '\0';
snprintf(message, sizeof(message), "Found Dreamcast CD: %.128s (%.16s)", (const char*)&buffer[0x80], (const char*)&buffer[0x40]);
verbose_message_callback(message);
}
/* remove whitespace from bootfile */
i = 0;
while (!isspace(buffer[96 + i]) && i < 16)
++i;
/* sometimes boot file isn't present on meta information.
* nothing can be done, as even the core doesn't run the game in this case. */
if (i == 0)
{
rc_cd_close_track(track_handle);
return rc_hash_error("Boot executable not specified on IP.BIN");
}
memcpy(exe_file, &buffer[96], i);
exe_file[i] = '\0';
sector = rc_cd_find_file_sector(track_handle, exe_file, &size);
rc_cd_close_track(track_handle);
if (sector == 0)
return rc_hash_error("Could not locate boot executable");
/* last track contains the boot executable */
last_track_handle = rc_cd_open_track(path, RC_HASH_CDTRACK_LAST);
sector = rc_cd_absolute_sector_to_track_sector(last_track_handle, sector);
if ((num_read = rc_cd_read_sector(last_track_handle, sector, buffer, sizeof(buffer))) < sizeof(buffer))
rc_hash_error("Could not read boot executable");
if (size > MAX_BUFFER_SIZE)
size = MAX_BUFFER_SIZE;
if (verbose_message_callback)
{
char message[128];
snprintf(message, sizeof(message), "Hashing %s contents (%u bytes)", exe_file, size);
verbose_message_callback(message);
}
do
{
md5_append(&md5, buffer, (int)num_read);
size -= (unsigned)num_read;
if (size == 0)
break;
++sector;
if (size >= sizeof(buffer))
num_read = rc_cd_read_sector(last_track_handle, sector, buffer, sizeof(buffer));
else
num_read = rc_cd_read_sector(last_track_handle, sector, buffer, size);
} while (num_read > 0);
rc_cd_close_track(last_track_handle);
result = rc_hash_finalize(&md5, hash);
return result;
}
static int rc_hash_psx(char hash[33], const char* path)
{
uint8_t buffer[2048];
char exe_name[64] = "";
char* ptr;
char* start;
void* track_handle;
uint32_t sector;
unsigned size;
size_t num_read;
int result = 0;
md5_state_t md5;
track_handle = rc_cd_open_track(path, 1);
if (!track_handle)
return rc_hash_error("Could not open track");
sector = rc_cd_find_file_sector(track_handle, "SYSTEM.CNF", NULL);
if (!sector)
{
sector = rc_cd_find_file_sector(track_handle, "PSX.EXE", &size);
if (sector)
strcpy(exe_name, "PSX.EXE");
}
else
{
size = (unsigned)rc_cd_read_sector(track_handle, sector, buffer, sizeof(buffer) - 1);
buffer[size] = '\0';
for (ptr = (char*)buffer; *ptr; ++ptr)
{
if (strncmp(ptr, "BOOT", 4) == 0)
{
ptr += 4;
while (isspace(*ptr))
++ptr;
if (*ptr == '=')
{
++ptr;
while (isspace(*ptr))
++ptr;
if (strncmp(ptr, "cdrom:", 6) == 0)
ptr += 6;
if (*ptr == '\\')
++ptr;
start = ptr;
while (!isspace(*ptr) && *ptr != ';')
++ptr;
size = (unsigned)(ptr - start);
if (size >= sizeof(exe_name))
size = sizeof(exe_name) - 1;
memcpy(exe_name, start, size);
exe_name[size] = '\0';
if (verbose_message_callback)
{
snprintf((char*)buffer, sizeof(buffer), "Looking for boot executable: %s", exe_name);
verbose_message_callback((const char*)buffer);
}
sector = rc_cd_find_file_sector(track_handle, exe_name, &size);
break;
}
}
/* advance to end of line */
while (*ptr && *ptr != '\n')
++ptr;
}
}
if (!sector)
{
rc_hash_error("Could not locate primary executable");
}
else if ((num_read = rc_cd_read_sector(track_handle, sector, buffer, sizeof(buffer))) < sizeof(buffer))
{
rc_hash_error("Could not read primary executable");
}
else
{
if (memcmp(buffer, "PS-X EXE", 7) != 0)
{
if (verbose_message_callback)
{
char message[128];
snprintf(message, sizeof(message), "%s did not contain PS-X EXE marker", exe_name);
verbose_message_callback(message);
}
}
else
{
/* the PS-X EXE header specifies the executable size as a 4-byte value 28 bytes into the header, which doesn't
* include the header itself. We want to include the header in the hash, so append another 2048 to that value.
*/
size = (((uint8_t)buffer[31] << 24) | ((uint8_t)buffer[30] << 16) | ((uint8_t)buffer[29] << 8) | (uint8_t)buffer[28]) + 2048;
}
if (size > MAX_BUFFER_SIZE)
size = MAX_BUFFER_SIZE;
if (verbose_message_callback)
{
char message[128];
snprintf(message, sizeof(message), "Hashing %s title (%u bytes) and contents (%u bytes) ", exe_name, (unsigned)strlen(exe_name), size);
verbose_message_callback(message);
}
/* there's also a few games that are use a singular engine and only differ via their data files. luckily, they have
* unique serial numbers, and use the serial number as the boot file in the standard way. include the boot file in the hash
*/
md5_init(&md5);
md5_append(&md5, (md5_byte_t*)exe_name, (int)strlen(exe_name));
do
{
md5_append(&md5, buffer, (int)num_read);
size -= (unsigned)num_read;
if (size == 0)
break;
++sector;
if (size >= sizeof(buffer))
num_read = rc_cd_read_sector(track_handle, sector, buffer, sizeof(buffer));
else
num_read = rc_cd_read_sector(track_handle, sector, buffer, size);
} while (num_read > 0);
result = rc_hash_finalize(&md5, hash);
}
rc_cd_close_track(track_handle);
return result;
}
static int rc_hash_sega_cd(char hash[33], const char* path)
{
uint8_t buffer[512];
void* track_handle;
track_handle = rc_cd_open_track(path, 1);
if (!track_handle)
return rc_hash_error("Could not open track");
/* the first 512 bytes of sector 0 are a volume header and ROM header that uniquely identify the game.
* After that is an arbitrary amount of code that ensures the game is being run in the correct region.
* Then more arbitrary code follows that actually starts the boot process. Somewhere in there, the
* primary executable is loaded. In many cases, a single game will have multiple executables, so even
* if we could determine the primary one, it's just the tip of the iceberg. As such, we've decided that
* hashing the volume and ROM headers is sufficient for identifying the game, and we'll have to trust
* that our players aren't modifying anything else on the disc.
*/
rc_cd_read_sector(track_handle, 0, buffer, sizeof(buffer));
rc_cd_close_track(track_handle);
if (memcmp(buffer, "SEGADISCSYSTEM ", 16) != 0 && /* Sega CD */
memcmp(buffer, "SEGA SEGASATURN ", 16) != 0) /* Sega Saturn */
{
return rc_hash_error("Not a Sega CD");
}
return rc_hash_buffer(hash, buffer, sizeof(buffer));
}
static int rc_hash_snes(char hash[33], uint8_t* buffer, size_t buffer_size)
{
/* if the file contains a header, ignore it */
uint32_t calc_size = ((uint32_t)buffer_size / 0x2000) * 0x2000;
if (buffer_size - calc_size == 512)
{
rc_hash_verbose("Ignoring SNES header");
buffer += 512;
buffer_size -= 512;
}
return rc_hash_buffer(hash, buffer, buffer_size);
}
int rc_hash_generate_from_buffer(char hash[33], int console_id, uint8_t* buffer, size_t buffer_size)
{
switch (console_id)
{
default:
{
char message[128];
snprintf(message, sizeof(message), "Unsupported console for buffer hash: %d", console_id);
return rc_hash_error(message);
}
case RC_CONSOLE_APPLE_II:
case RC_CONSOLE_ATARI_2600:
case RC_CONSOLE_ATARI_7800:
case RC_CONSOLE_ATARI_JAGUAR:
case RC_CONSOLE_COLECOVISION:
case RC_CONSOLE_GAMEBOY:
case RC_CONSOLE_GAMEBOY_ADVANCE:
case RC_CONSOLE_GAMEBOY_COLOR:
case RC_CONSOLE_GAME_GEAR:
case RC_CONSOLE_INTELLIVISION:
case RC_CONSOLE_MAGNAVOX_ODYSSEY2:
case RC_CONSOLE_MASTER_SYSTEM:
case RC_CONSOLE_MEGA_DRIVE:
case RC_CONSOLE_MSX:
case RC_CONSOLE_NEOGEO_POCKET:
case RC_CONSOLE_NINTENDO_64:
case RC_CONSOLE_ORIC:
case RC_CONSOLE_PC8800:
case RC_CONSOLE_POKEMON_MINI:
case RC_CONSOLE_SEGA_32X:
case RC_CONSOLE_SG1000:
case RC_CONSOLE_VECTREX:
case RC_CONSOLE_VIRTUAL_BOY:
case RC_CONSOLE_WONDERSWAN:
return rc_hash_buffer(hash, buffer, buffer_size);
case RC_CONSOLE_ATARI_LYNX:
return rc_hash_lynx(hash, buffer, buffer_size);
case RC_CONSOLE_NINTENDO:
return rc_hash_nes(hash, buffer, buffer_size);
case RC_CONSOLE_PC_ENGINE: /* NOTE: does not support PCEngine CD */
return rc_hash_pce(hash, buffer, buffer_size);
case RC_CONSOLE_SUPER_NINTENDO:
return rc_hash_snes(hash, buffer, buffer_size);
}
}
static int rc_hash_whole_file(char hash[33], int console_id, const char* path)
{
md5_state_t md5;
uint8_t* buffer;
size_t size;
const size_t buffer_size = 65536;
void* file_handle;
int result = 0;
file_handle = rc_file_open(path);
if (!file_handle)
return rc_hash_error("Could not open file");
rc_file_seek(file_handle, 0, SEEK_END);
size = rc_file_tell(file_handle);
if (verbose_message_callback)
{
char message[1024];
if (size > MAX_BUFFER_SIZE)
snprintf(message, sizeof(message), "Hashing first %u bytes (of %u bytes) of %s", MAX_BUFFER_SIZE, (unsigned)size, rc_path_get_filename(path));
else
snprintf(message, sizeof(message), "Hashing %s (%u bytes)", rc_path_get_filename(path), (unsigned)size);
verbose_message_callback(message);
}
if (size > MAX_BUFFER_SIZE)
size = MAX_BUFFER_SIZE;
md5_init(&md5);
buffer = (uint8_t*)malloc(buffer_size);
if (buffer)
{
rc_file_seek(file_handle, 0, SEEK_SET);
while (size >= buffer_size)
{
rc_file_read(file_handle, buffer, (int)buffer_size);
md5_append(&md5, buffer, (int)buffer_size);
size -= buffer_size;
}
if (size > 0)
{
rc_file_read(file_handle, buffer, (int)size);
md5_append(&md5, buffer, (int)size);
}
free(buffer);
result = rc_hash_finalize(&md5, hash);
}
rc_file_close(file_handle);
return result;
}
static int rc_hash_buffered_file(char hash[33], int console_id, const char* path)
{
uint8_t* buffer;
size_t size;
int result = 0;
void* file_handle;
file_handle = rc_file_open(path);
if (!file_handle)
return rc_hash_error("Could not open file");
rc_file_seek(file_handle, 0, SEEK_END);
size = rc_file_tell(file_handle);
if (verbose_message_callback)
{
char message[1024];
if (size > MAX_BUFFER_SIZE)
snprintf(message, sizeof(message), "Buffering first %u bytes (of %d bytes) of %s", MAX_BUFFER_SIZE, (unsigned)size, rc_path_get_filename(path));
else
snprintf(message, sizeof(message), "Buffering %s (%d bytes)", rc_path_get_filename(path), (unsigned)size);
verbose_message_callback(message);
}
if (size > MAX_BUFFER_SIZE)
size = MAX_BUFFER_SIZE;
buffer = (uint8_t*)malloc(size);
if (buffer)
{
rc_file_seek(file_handle, 0, SEEK_SET);
rc_file_read(file_handle, buffer, (int)size);
result = rc_hash_generate_from_buffer(hash, console_id, buffer, size);
free(buffer);
}
rc_file_close(file_handle);
return result;
}
static int rc_hash_path_is_absolute(const char* path)
{
if (!path[0])
return 0;
/* "/path/to/file" or "\path\to\file" */
if (path[0] == '/' || path[0] == '\\')
return 1;
/* "C:\path\to\file" */
if (path[1] == ':' && path[2] == '\\')
return 1;
/* "scheme:/path/to/file" */
while (*path)
{
if (path[0] == ':' && path[1] == '/')
return 1;
++path;
}
return 0;
}
static const char* rc_hash_get_first_item_from_playlist(const char* path)
{
char buffer[1024];
char* disc_path;
char* ptr, *start, *next;
size_t num_read, path_len, file_len;
void* file_handle;
file_handle = rc_file_open(path);
if (!file_handle)
{
rc_hash_error("Could not open playlist");
return NULL;
}
num_read = rc_file_read(file_handle, buffer, sizeof(buffer) - 1);
buffer[num_read] = '\0';
rc_file_close(file_handle);
ptr = start = buffer;
do
{
/* ignore empty and commented lines */
while (*ptr == '#' || *ptr == '\r' || *ptr == '\n')
{
while (*ptr && *ptr != '\n')
++ptr;
if (*ptr)
++ptr;
}
/* find and extract the current line */
start = ptr;
while (*ptr && *ptr != '\n')
++ptr;
next = ptr;
/* remove trailing whitespace - especially '\r' */
while (ptr > start && isspace(ptr[-1]))
--ptr;
/* if we found a non-empty line, break out of the loop to process it */
file_len = ptr - start;
if (file_len)
break;
/* did we reach the end of the file? */
if (!*next)
return NULL;
/* if the line only contained whitespace, keep searching */
ptr = next + 1;
} while (1);
if (verbose_message_callback)
{
char message[1024];
snprintf(message, sizeof(message), "Extracted %.*s from playlist", (int)file_len, start);
verbose_message_callback(message);
}
start[file_len++] = '\0';
if (rc_hash_path_is_absolute(start))
path_len = 0;
else
path_len = rc_path_get_filename(path) - path;
disc_path = (char*)malloc(path_len + file_len + 1);
if (!disc_path)
return NULL;
if (path_len)
memcpy(disc_path, path, path_len);
memcpy(&disc_path[path_len], start, file_len);
return disc_path;
}
static int rc_hash_generate_from_playlist(char hash[33], int console_id, const char* path)
{
int result;
const char* disc_path;
if (verbose_message_callback)
{
char message[1024];
snprintf(message, sizeof(message), "Processing playlist: %s", rc_path_get_filename(path));
verbose_message_callback(message);
}
disc_path = rc_hash_get_first_item_from_playlist(path);
if (!disc_path)
return rc_hash_error("Failed to get first item from playlist");
result = rc_hash_generate_from_file(hash, console_id, disc_path);
free((void*)disc_path);
return result;
}
int rc_hash_generate_from_file(char hash[33], int console_id, const char* path)
{
switch (console_id)
{
default:
{
char buffer[128];
snprintf(buffer, sizeof(buffer), "Unsupported console for file hash: %d", console_id);
return rc_hash_error(buffer);
}
case RC_CONSOLE_APPLE_II:
case RC_CONSOLE_ATARI_2600:
case RC_CONSOLE_ATARI_7800:
case RC_CONSOLE_ATARI_JAGUAR:
case RC_CONSOLE_COLECOVISION:
case RC_CONSOLE_GAMEBOY:
case RC_CONSOLE_GAMEBOY_ADVANCE:
case RC_CONSOLE_GAMEBOY_COLOR:
case RC_CONSOLE_GAME_GEAR:
case RC_CONSOLE_INTELLIVISION:
case RC_CONSOLE_MAGNAVOX_ODYSSEY2:
case RC_CONSOLE_MASTER_SYSTEM:
case RC_CONSOLE_MEGA_DRIVE:
case RC_CONSOLE_NEOGEO_POCKET:
case RC_CONSOLE_NINTENDO_64:
case RC_CONSOLE_ORIC:
case RC_CONSOLE_POKEMON_MINI:
case RC_CONSOLE_SEGA_32X:
case RC_CONSOLE_SG1000:
case RC_CONSOLE_VECTREX:
case RC_CONSOLE_VIRTUAL_BOY:
case RC_CONSOLE_WONDERSWAN:
/* generic whole-file hash - don't buffer */
return rc_hash_whole_file(hash, console_id, path);
case RC_CONSOLE_MSX:
case RC_CONSOLE_PC8800:
/* generic whole-file hash with m3u support - don't buffer */
if (rc_path_compare_extension(path, "m3u"))
return rc_hash_generate_from_playlist(hash, console_id, path);
return rc_hash_whole_file(hash, console_id, path);
case RC_CONSOLE_ATARI_LYNX:
case RC_CONSOLE_NINTENDO:
case RC_CONSOLE_SUPER_NINTENDO:
/* additional logic whole-file hash - buffer then call rc_hash_generate_from_buffer */
return rc_hash_buffered_file(hash, console_id, path);
case RC_CONSOLE_3DO:
if (rc_path_compare_extension(path, "m3u"))
return rc_hash_generate_from_playlist(hash, console_id, path);
return rc_hash_3do(hash, path);
case RC_CONSOLE_ARCADE:
return rc_hash_arcade(hash, path);
case RC_CONSOLE_NINTENDO_DS:
return rc_hash_nintendo_ds(hash, path);
case RC_CONSOLE_PC_ENGINE:
if (rc_path_compare_extension(path, "cue") || rc_path_compare_extension(path, "chd"))
return rc_hash_pce_cd(hash, path);
if (rc_path_compare_extension(path, "m3u"))
return rc_hash_generate_from_playlist(hash, console_id, path);
return rc_hash_buffered_file(hash, console_id, path);
case RC_CONSOLE_PCFX:
if (rc_path_compare_extension(path, "m3u"))
return rc_hash_generate_from_playlist(hash, console_id, path);
return rc_hash_pcfx_cd(hash, path);
case RC_CONSOLE_PLAYSTATION:
if (rc_path_compare_extension(path, "m3u"))
return rc_hash_generate_from_playlist(hash, console_id, path);
return rc_hash_psx(hash, path);
case RC_CONSOLE_DREAMCAST:
if (rc_path_compare_extension(path, "m3u"))
return rc_hash_generate_from_playlist(hash, console_id, path);
return rc_hash_dreamcast(hash, path);
case RC_CONSOLE_SEGA_CD:
case RC_CONSOLE_SATURN:
if (rc_path_compare_extension(path, "m3u"))
return rc_hash_generate_from_playlist(hash, console_id, path);
return rc_hash_sega_cd(hash, path);
}
}
static void rc_hash_iterator_append_console(struct rc_hash_iterator* iterator, int console_id)
{
int i = 0;
while (iterator->consoles[i] != 0)
{
if (iterator->consoles[i] == console_id)
return;
++i;
}
iterator->consoles[i] = console_id;
}
static void rc_hash_initialize_dsk_iterator(struct rc_hash_iterator* iterator, const char* path)
{
size_t size = iterator->buffer_size;
if (size == 0)
{
/* attempt to use disk size to determine system */
void* file = rc_file_open(path);
if (file)
{
rc_file_seek(file, 0, SEEK_END);
size = rc_file_tell(file);
rc_file_close(file);
}
}
if (size == 512 * 9 * 80) /* 360KB */
{
/* FAT-12 3.5" DD (512 byte sectors, 9 sectors per track, 80 tracks per side */
/* FAT-12 5.25" DD double-sided (512 byte sectors, 9 sectors per track, 80 tracks per side */
iterator->consoles[0] = RC_CONSOLE_MSX;
}
else if (size == 512 * 9 * 80 * 2) /* 720KB */
{
/* FAT-12 3.5" DD double-sided (512 byte sectors, 9 sectors per track, 80 tracks per side */
iterator->consoles[0] = RC_CONSOLE_MSX;
}
else if (size == 512 * 9 * 40) /* 180KB */
{
/* FAT-12 5.25" DD (512 byte sectors, 9 sectors per track, 40 tracks per side */
iterator->consoles[0] = RC_CONSOLE_MSX;
}
else if (size == 256 * 16 * 35) /* 140KB */
{
/* Apple II new format - 256 byte sectors, 16 sectors per track, 35 tracks per side */
iterator->consoles[0] = RC_CONSOLE_APPLE_II;
}
else if (size == 256 * 13 * 35) /* 113.75KB */
{
/* Apple II old format - 256 byte sectors, 13 sectors per track, 35 tracks per side */
iterator->consoles[0] = RC_CONSOLE_APPLE_II;
}
/* once a best guess has been identified, make sure the others are added as fallbacks */
/* check MSX first, as Apple II isn't supported by RetroArch, and RAppleWin won't use the iterator */
rc_hash_iterator_append_console(iterator, RC_CONSOLE_MSX);
rc_hash_iterator_append_console(iterator, RC_CONSOLE_APPLE_II);
}
void rc_hash_initialize_iterator(struct rc_hash_iterator* iterator, const char* path, uint8_t* buffer, size_t buffer_size)
{
int need_path = !buffer;
memset(iterator, 0, sizeof(*iterator));
iterator->buffer = buffer;
iterator->buffer_size = buffer_size;
iterator->consoles[0] = 0;
do
{
const char* ext = rc_path_get_extension(path);
switch (tolower(*ext))
{
case '7':
if (rc_path_compare_extension(ext, "7z"))
{
/* decompressing zip file not supported */
iterator->consoles[0] = RC_CONSOLE_ARCADE;
need_path = 1;
}
break;
case 'a':
if (rc_path_compare_extension(ext, "a78"))
{
iterator->consoles[0] = RC_CONSOLE_ATARI_7800;
}
break;
case 'b':
if (rc_path_compare_extension(ext, "bin"))
{
if (buffer_size == 0)
{
/* raw bin file may be a CD track. if it's more than 32MB, try a CD hash. */
void* file = rc_file_open(path);
if (file)
{
size_t size;
rc_file_seek(file, 0, SEEK_END);
size = rc_file_tell(file);
rc_file_close(file);
if (size > 32 * 1024 * 1024)
{
iterator->consoles[0] = RC_CONSOLE_3DO; /* 4DO supports directly opening the bin file */
iterator->consoles[1] = RC_CONSOLE_PLAYSTATION; /* PCSX ReARMed supports directly opening the bin file*/
iterator->consoles[2] = RC_CONSOLE_SEGA_CD; /* Genesis Plus GX supports directly opening the bin file*/
/* fallback to megadrive which just does a full hash */
iterator->consoles[3] = RC_CONSOLE_MEGA_DRIVE;
break;
}
}
}
/* bin is associated with MegaDrive, Sega32X and Atari 2600. Since they all use the same
* hashing algorithm, only specify one of them */
iterator->consoles[0] = RC_CONSOLE_MEGA_DRIVE;
}
else if (rc_path_compare_extension(ext, "bs"))
{
iterator->consoles[0] = RC_CONSOLE_SUPER_NINTENDO;
}
break;
case 'c':
if (rc_path_compare_extension(ext, "cue"))
{
iterator->consoles[0] = RC_CONSOLE_PLAYSTATION;
iterator->consoles[1] = RC_CONSOLE_PC_ENGINE;
iterator->consoles[2] = RC_CONSOLE_3DO;
iterator->consoles[3] = RC_CONSOLE_PCFX;
iterator->consoles[4] = RC_CONSOLE_SEGA_CD; /* ASSERT: handles both Sega CD and Saturn */
need_path = 1;
}
else if (rc_path_compare_extension(ext, "chd"))
{
iterator->consoles[0] = RC_CONSOLE_PLAYSTATION;
iterator->consoles[1] = RC_CONSOLE_DREAMCAST;
iterator->consoles[2] = RC_CONSOLE_PC_ENGINE;
iterator->consoles[3] = RC_CONSOLE_3DO;
iterator->consoles[4] = RC_CONSOLE_PCFX;
iterator->consoles[5] = RC_CONSOLE_SEGA_CD; /* ASSERT: handles both Sega CD and Saturn */
need_path = 1;
}
else if (rc_path_compare_extension(ext, "col"))
{
iterator->consoles[0] = RC_CONSOLE_COLECOVISION;
}
else if (rc_path_compare_extension(ext, "cas"))
{
iterator->consoles[0] = RC_CONSOLE_MSX;
}
break;
case 'd':
if (rc_path_compare_extension(ext, "dsk"))
{
rc_hash_initialize_dsk_iterator(iterator, path);
}
else if (rc_path_compare_extension(ext, "d88"))
{
iterator->consoles[0] = RC_CONSOLE_PC8800;
}
break;
case 'f':
if (rc_path_compare_extension(ext, "fig"))
{
iterator->consoles[0] = RC_CONSOLE_SUPER_NINTENDO;
}
else if (rc_path_compare_extension(ext, "fds"))
{
iterator->consoles[0] = RC_CONSOLE_NINTENDO;
}
break;
case 'g':
if (rc_path_compare_extension(ext, "gba"))
{
iterator->consoles[0] = RC_CONSOLE_GAMEBOY_ADVANCE;
}
else if (rc_path_compare_extension(ext, "gbc"))
{
iterator->consoles[0] = RC_CONSOLE_GAMEBOY_COLOR;
}
else if (rc_path_compare_extension(ext, "gb"))
{
iterator->consoles[0] = RC_CONSOLE_GAMEBOY;
}
else if (rc_path_compare_extension(ext, "gg"))
{
iterator->consoles[0] = RC_CONSOLE_GAME_GEAR;
}
else if (rc_path_compare_extension(ext, "gdi"))
{
iterator->consoles[0] = RC_CONSOLE_DREAMCAST;
}
break;
case 'i':
if (rc_path_compare_extension(ext, "iso"))
{
iterator->consoles[0] = RC_CONSOLE_3DO;
iterator->consoles[1] = RC_CONSOLE_SEGA_CD; /* ASSERT: handles both Sega CD and Saturn */
need_path = 1;
}
break;
case 'j':
if (rc_path_compare_extension(ext, "jag"))
{
iterator->consoles[0] = RC_CONSOLE_ATARI_JAGUAR;
}
break;
case 'l':
if (rc_path_compare_extension(ext, "lnx"))
{
iterator->consoles[0] = RC_CONSOLE_ATARI_LYNX;
}
break;
case 'm':
if (rc_path_compare_extension(ext, "m3u"))
{
const char* disc_path = rc_hash_get_first_item_from_playlist(path);
if (!disc_path) /* did not find a disc */
return;
iterator->buffer = NULL; /* ignore buffer; assume it's the m3u contents */
path = iterator->path = disc_path;
continue; /* retry with disc_path */
}
else if (rc_path_compare_extension(ext, "md"))
{
iterator->consoles[0] = RC_CONSOLE_MEGA_DRIVE;
}
else if (rc_path_compare_extension(ext, "min"))
{
iterator->consoles[0] = RC_CONSOLE_POKEMON_MINI;
}
else if (rc_path_compare_extension(ext, "mx1"))
{
iterator->consoles[0] = RC_CONSOLE_MSX;
}
else if (rc_path_compare_extension(ext, "mx2"))
{
iterator->consoles[0] = RC_CONSOLE_MSX;
}
break;
case 'n':
if (rc_path_compare_extension(ext, "nes"))
{
iterator->consoles[0] = RC_CONSOLE_NINTENDO;
}
else if (rc_path_compare_extension(ext, "nds"))
{
iterator->consoles[0] = RC_CONSOLE_NINTENDO_DS;
}
else if (rc_path_compare_extension(ext, "n64") ||
rc_path_compare_extension(ext, "ndd"))
{
iterator->consoles[0] = RC_CONSOLE_NINTENDO_64;
}
else if (rc_path_compare_extension(ext, "ngc"))
{
iterator->consoles[0] = RC_CONSOLE_NEOGEO_POCKET;
}
break;
case 'p':
if (rc_path_compare_extension(ext, "pce"))
{
iterator->consoles[0] = RC_CONSOLE_PC_ENGINE;
}
break;
case 'r':
if (rc_path_compare_extension(ext, "rom"))
{
iterator->consoles[0] = RC_CONSOLE_MSX;
}
if (rc_path_compare_extension(ext, "ri"))
{
iterator->consoles[0] = RC_CONSOLE_MSX;
}
break;
case 's':
if (rc_path_compare_extension(ext, "smc") ||
rc_path_compare_extension(ext, "sfc") ||
rc_path_compare_extension(ext, "swc"))
{
iterator->consoles[0] = RC_CONSOLE_SUPER_NINTENDO;
}
else if (rc_path_compare_extension(ext, "sg"))
{
iterator->consoles[0] = RC_CONSOLE_SG1000;
}
else if (rc_path_compare_extension(ext, "sgx"))
{
iterator->consoles[0] = RC_CONSOLE_PC_ENGINE;
}
break;
case 't':
if (rc_path_compare_extension(ext, "tap"))
{
iterator->consoles[0] = RC_CONSOLE_ORIC;
}
break;
case 'v':
if (rc_path_compare_extension(ext, "vb"))
{
iterator->consoles[0] = RC_CONSOLE_VIRTUAL_BOY;
}
break;
case 'w':
if (rc_path_compare_extension(ext, "wsc"))
{
iterator->consoles[0] = RC_CONSOLE_WONDERSWAN;
}
else if (rc_path_compare_extension(ext, "woz"))
{
iterator->consoles[0] = RC_CONSOLE_APPLE_II;
}
break;
case 'z':
if (rc_path_compare_extension(ext, "zip"))
{
/* decompressing zip file not supported */
iterator->consoles[0] = RC_CONSOLE_ARCADE;
need_path = 1;
}
break;
}
if (verbose_message_callback)
{
char message[256];
int count = 0;
while (iterator->consoles[count])
++count;
snprintf(message, sizeof(message), "Found %d potential consoles for %s file extension", count, ext);
verbose_message_callback(message);
}
/* loop is only for specific cases that redirect to another file - like m3u */
break;
} while (1);
if (need_path && !iterator->path)
iterator->path = strdup(path);
/* if we didn't match the extension, default to something that does a whole file hash */
if (!iterator->consoles[0])
iterator->consoles[0] = RC_CONSOLE_GAMEBOY;
}
void rc_hash_destroy_iterator(struct rc_hash_iterator* iterator)
{
if (iterator->path)
{
free((void*)iterator->path);
iterator->path = NULL;
}
}
int rc_hash_iterate(char hash[33], struct rc_hash_iterator* iterator)
{
int next_console;
int result = 0;
do
{
next_console = iterator->consoles[iterator->index];
if (next_console == 0)
{
hash[0] = '\0';
break;
}
++iterator->index;
if (verbose_message_callback)
{
char message[128];
snprintf(message, sizeof(message), "Trying console %d", next_console);
verbose_message_callback(message);
}
if (iterator->buffer)
result = rc_hash_generate_from_buffer(hash, next_console, iterator->buffer, iterator->buffer_size);
else
result = rc_hash_generate_from_file(hash, next_console, iterator->path);
} while (!result);
return result;
}