2017-04-17 20:16:17 +03:00
|
|
|
// The tester requires low-level access to the GB struct to detect failures
|
|
|
|
#define GB_INTERNAL
|
|
|
|
|
2016-09-03 03:39:32 +03:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdbool.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <time.h>
|
|
|
|
#include <assert.h>
|
|
|
|
#include <signal.h>
|
|
|
|
#ifdef _WIN32
|
|
|
|
#include <direct.h>
|
|
|
|
#include <windows.h>
|
|
|
|
#define snprintf _snprintf
|
|
|
|
#endif
|
|
|
|
|
2017-10-13 00:02:02 +03:00
|
|
|
#include <Core/gb.h>
|
2016-09-03 03:39:32 +03:00
|
|
|
|
|
|
|
static bool running = false;
|
|
|
|
static char *filename;
|
|
|
|
static char *bmp_filename;
|
|
|
|
static char *log_filename;
|
|
|
|
static FILE *log_file;
|
|
|
|
static void replace_extension(const char *src, size_t length, char *dest, const char *ext);
|
2017-02-25 16:06:38 +02:00
|
|
|
static bool push_start_a, start_is_not_first, a_is_bad, b_is_confirm, push_faster, push_slower,
|
|
|
|
do_not_stop, push_a_twice, start_is_bad, allow_weird_sp_values, large_stack, push_right;
|
2016-09-03 03:39:32 +03:00
|
|
|
static unsigned int test_length = 60 * 40;
|
|
|
|
GB_gameboy_t gb;
|
|
|
|
|
|
|
|
static unsigned int frames = 0;
|
|
|
|
const char bmp_header[] = {
|
|
|
|
0x42, 0x4D, 0x48, 0x68, 0x01, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x46, 0x00, 0x00, 0x00, 0x38, 0x00,
|
|
|
|
0x00, 0x00, 0xA0, 0x00, 0x00, 0x00, 0x70, 0xFF,
|
|
|
|
0xFF, 0xFF, 0x01, 0x00, 0x20, 0x00, 0x03, 0x00,
|
|
|
|
0x00, 0x00, 0x02, 0x68, 0x01, 0x00, 0x12, 0x0B,
|
|
|
|
0x00, 0x00, 0x12, 0x0B, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0xFF, 0x00, 0x00, 0xFF, 0x00, 0x00, 0xFF,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
};
|
|
|
|
|
|
|
|
uint32_t bitmap[160*144];
|
|
|
|
|
2016-10-22 00:49:32 +03:00
|
|
|
static char *async_input_callback(GB_gameboy_t *gb)
|
|
|
|
{
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2016-09-03 03:39:32 +03:00
|
|
|
static void vblank(GB_gameboy_t *gb)
|
|
|
|
{
|
|
|
|
/* Do not press any buttons during the last two seconds, this might cause a
|
|
|
|
screenshot to be taken while the LCD is off if the press makes the game
|
|
|
|
load graphics. */
|
2016-10-05 23:56:44 +03:00
|
|
|
if (push_start_a && (frames < test_length - 120 || do_not_stop)) {
|
2016-09-09 19:29:14 +03:00
|
|
|
unsigned combo_length = 40;
|
2016-10-14 01:30:54 +03:00
|
|
|
if (start_is_not_first || push_a_twice) combo_length = 60; /* The start item in the menu is not the first, so also push down */
|
2017-02-24 01:19:44 +02:00
|
|
|
else if (a_is_bad || start_is_bad) combo_length = 20; /* Pressing A has a negative effect (when trying to start the game). */
|
2016-09-09 19:29:14 +03:00
|
|
|
|
2016-10-14 01:30:54 +03:00
|
|
|
switch ((push_faster ? frames * 2 :
|
|
|
|
push_slower ? frames / 2 :
|
|
|
|
push_a_twice? frames / 4:
|
2017-02-24 01:19:44 +02:00
|
|
|
frames) % combo_length + (start_is_bad? 20 : 0) ) {
|
2016-09-03 03:39:32 +03:00
|
|
|
case 0:
|
2017-02-25 16:06:38 +02:00
|
|
|
gb->keys[push_right? 0 : 7] = true; // Start (Or right) down
|
2016-09-03 03:39:32 +03:00
|
|
|
break;
|
|
|
|
case 10:
|
2017-02-25 16:06:38 +02:00
|
|
|
gb->keys[push_right? 0 : 7] = false; // Start (Or right) up
|
2016-09-03 03:39:32 +03:00
|
|
|
break;
|
|
|
|
case 20:
|
2016-09-09 19:29:14 +03:00
|
|
|
gb->keys[b_is_confirm? 5: 4] = true; // A down (or B)
|
2016-09-03 03:39:32 +03:00
|
|
|
break;
|
|
|
|
case 30:
|
2016-09-09 19:29:14 +03:00
|
|
|
gb->keys[b_is_confirm? 5: 4] = false; // A up (or B)
|
2016-09-03 03:39:32 +03:00
|
|
|
break;
|
2016-09-06 18:00:05 +03:00
|
|
|
case 40:
|
2016-10-14 01:30:54 +03:00
|
|
|
if (push_a_twice) {
|
|
|
|
gb->keys[b_is_confirm? 5: 4] = true; // A down (or B)
|
|
|
|
}
|
|
|
|
else if (gb->boot_rom_finished) {
|
2016-09-09 19:29:14 +03:00
|
|
|
gb->keys[3] = true; // D-Pad Down down
|
|
|
|
}
|
2016-09-06 18:00:05 +03:00
|
|
|
break;
|
|
|
|
case 50:
|
2016-10-14 01:30:54 +03:00
|
|
|
gb->keys[b_is_confirm? 5: 4] = false; // A down (or B)
|
2016-09-06 18:00:05 +03:00
|
|
|
gb->keys[3] = false; // D-Pad Down up
|
|
|
|
break;
|
2016-09-03 03:39:32 +03:00
|
|
|
}
|
|
|
|
}
|
2016-09-03 22:59:23 +03:00
|
|
|
|
|
|
|
/* Detect common crashes and stop the test early */
|
|
|
|
if (frames < test_length - 1) {
|
2017-02-25 16:06:38 +02:00
|
|
|
if (gb->backtrace_size >= 0x200 + (large_stack? 0x80: 0) || (!allow_weird_sp_values && (gb->registers[GB_REGISTER_SP] >= 0xfe00 && gb->registers[GB_REGISTER_SP] < 0xff80))) {
|
2017-02-24 18:25:27 +02:00
|
|
|
GB_log(gb, "A stack overflow has probably occurred. (SP = $%04x; backtrace size = %d) \n",
|
|
|
|
gb->registers[GB_REGISTER_SP], gb->backtrace_size);
|
2016-09-03 22:59:23 +03:00
|
|
|
frames = test_length - 1;
|
|
|
|
}
|
2016-09-06 18:00:05 +03:00
|
|
|
if (gb->halted && !gb->interrupt_enable) {
|
2016-09-03 22:59:23 +03:00
|
|
|
GB_log(gb, "The game is deadlocked.\n");
|
|
|
|
frames = test_length - 1;
|
|
|
|
}
|
|
|
|
}
|
2016-09-03 03:39:32 +03:00
|
|
|
|
2016-09-20 22:59:00 +03:00
|
|
|
if (frames >= test_length ) {
|
2016-09-30 01:10:50 +03:00
|
|
|
bool is_screen_blank = true;
|
|
|
|
for (unsigned i = 160*144; i--;) {
|
|
|
|
if (bitmap[i] != bitmap[0]) {
|
|
|
|
is_screen_blank = false;
|
|
|
|
break;
|
|
|
|
}
|
2016-09-20 22:59:00 +03:00
|
|
|
}
|
|
|
|
|
2016-09-30 14:12:41 +03:00
|
|
|
/* Let the test run for extra four seconds if the screen is off/disabled */
|
|
|
|
if (!is_screen_blank || frames >= test_length + 60 * 4) {
|
2016-09-20 22:59:00 +03:00
|
|
|
FILE *f = fopen(bmp_filename, "wb");
|
|
|
|
fwrite(&bmp_header, 1, sizeof(bmp_header), f);
|
|
|
|
fwrite(&bitmap, 1, sizeof(bitmap), f);
|
|
|
|
fclose(f);
|
|
|
|
if (!gb->boot_rom_finished) {
|
|
|
|
GB_log(gb, "Boot ROM did not finish.\n");
|
|
|
|
}
|
|
|
|
if (is_screen_blank) {
|
|
|
|
GB_log(gb, "Game probably stuck with blank screen. \n");
|
|
|
|
}
|
|
|
|
running = false;
|
2016-09-03 22:59:23 +03:00
|
|
|
}
|
2016-09-03 03:39:32 +03:00
|
|
|
}
|
|
|
|
else if (frames == test_length - 1) {
|
|
|
|
gb->disable_rendering = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
frames++;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void log_callback(GB_gameboy_t *gb, const char *string, GB_log_attributes attributes)
|
|
|
|
{
|
|
|
|
if (!log_file) log_file = fopen(log_filename, "w");
|
|
|
|
fprintf(log_file, "%s", string);
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef __APPLE__
|
|
|
|
#include <mach-o/dyld.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static const char *executable_folder(void)
|
|
|
|
{
|
|
|
|
static char path[1024] = {0,};
|
|
|
|
if (path[0]) {
|
|
|
|
return path;
|
|
|
|
}
|
|
|
|
/* Ugly unportable code! :( */
|
|
|
|
#ifdef __APPLE__
|
|
|
|
unsigned int length = sizeof(path) - 1;
|
|
|
|
_NSGetExecutablePath(&path[0], &length);
|
|
|
|
#else
|
|
|
|
#ifdef __linux__
|
|
|
|
ssize_t length = readlink("/proc/self/exe", &path[0], sizeof(path) - 1);
|
|
|
|
assert (length != -1);
|
|
|
|
#else
|
|
|
|
#ifdef _WIN32
|
|
|
|
HMODULE hModule = GetModuleHandle(NULL);
|
|
|
|
GetModuleFileName(hModule, path, sizeof(path) - 1);
|
|
|
|
#else
|
|
|
|
/* No OS-specific way, assume running from CWD */
|
|
|
|
getcwd(&path[0], sizeof(path) - 1);
|
|
|
|
return path;
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
size_t pos = strlen(path);
|
|
|
|
while (pos) {
|
|
|
|
pos--;
|
|
|
|
#ifdef _WIN32
|
|
|
|
if (path[pos] == '\\') {
|
|
|
|
#else
|
|
|
|
if (path[pos] == '/') {
|
|
|
|
#endif
|
|
|
|
path[pos] = 0;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return path;
|
|
|
|
}
|
|
|
|
|
|
|
|
static char *executable_relative_path(const char *filename)
|
|
|
|
{
|
|
|
|
static char path[1024];
|
|
|
|
snprintf(path, sizeof(path), "%s/%s", executable_folder(), filename);
|
|
|
|
return path;
|
|
|
|
}
|
|
|
|
|
|
|
|
static uint32_t rgb_encode(GB_gameboy_t *gb, uint8_t r, uint8_t g, uint8_t b)
|
|
|
|
{
|
|
|
|
return (r << 24) | (g << 16) | (b << 8);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void replace_extension(const char *src, size_t length, char *dest, const char *ext)
|
|
|
|
{
|
|
|
|
memcpy(dest, src, length);
|
|
|
|
dest[length] = 0;
|
|
|
|
|
|
|
|
/* Remove extension */
|
|
|
|
for (size_t i = length; i--;) {
|
|
|
|
if (dest[i] == '/') break;
|
|
|
|
if (dest[i] == '.') {
|
|
|
|
dest[i] = 0;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Add new extension */
|
|
|
|
strcat(dest, ext);
|
|
|
|
}
|
|
|
|
|
2017-06-21 23:25:39 +03:00
|
|
|
|
2016-09-03 03:39:32 +03:00
|
|
|
int main(int argc, char **argv)
|
|
|
|
{
|
|
|
|
#define str(x) #x
|
|
|
|
#define xstr(x) str(x)
|
|
|
|
fprintf(stderr, "SameBoy Tester v" xstr(VERSION) "\n");
|
|
|
|
|
|
|
|
if (argc == 1) {
|
2017-06-21 23:25:39 +03:00
|
|
|
fprintf(stderr, "Usage: %s [--dmg] [--start] [--length seconds] [--boot path to boot ROM]"
|
2016-09-03 03:39:32 +03:00
|
|
|
#ifndef _WIN32
|
|
|
|
" [--jobs number of tests to run simultaneously]"
|
|
|
|
#endif
|
|
|
|
" rom ...\n", argv[0]);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifndef _WIN32
|
|
|
|
unsigned int max_forks = 1;
|
|
|
|
unsigned int current_forks = 0;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
bool dmg = false;
|
2017-06-21 23:25:39 +03:00
|
|
|
const char *boot_rom_path = NULL;
|
|
|
|
|
2016-09-03 03:39:32 +03:00
|
|
|
for (unsigned i = 1; i < argc; i++) {
|
|
|
|
if (strcmp(argv[i], "--dmg") == 0) {
|
|
|
|
fprintf(stderr, "Using DMG mode\n");
|
|
|
|
dmg = true;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (strcmp(argv[i], "--start") == 0) {
|
|
|
|
fprintf(stderr, "Pushing Start and A\n");
|
|
|
|
push_start_a = true;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (strcmp(argv[i], "--length") == 0 && i != argc - 1) {
|
|
|
|
test_length = atoi(argv[++i]) * 60;
|
|
|
|
fprintf(stderr, "Test length is %d seconds\n", test_length / 60);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2017-06-21 23:25:39 +03:00
|
|
|
if (strcmp(argv[i], "--boot") == 0 && i != argc - 1) {
|
|
|
|
fprintf(stderr, "Using boot ROM %s\n", argv[i + 1]);
|
|
|
|
boot_rom_path = argv[++i];
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2016-09-03 03:39:32 +03:00
|
|
|
#ifndef _WIN32
|
|
|
|
if (strcmp(argv[i], "--jobs") == 0 && i != argc - 1) {
|
|
|
|
max_forks = atoi(argv[++i]);
|
|
|
|
/* Make sure wrong input doesn't blow anything up. */
|
|
|
|
if (max_forks < 1) max_forks = 1;
|
|
|
|
if (max_forks > 16) max_forks = 16;
|
|
|
|
fprintf(stderr, "Running up to %d tests simultaneously\n", max_forks);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (max_forks > 1) {
|
|
|
|
while (current_forks >= max_forks) {
|
|
|
|
int wait_out;
|
|
|
|
while(wait(&wait_out) == -1);
|
|
|
|
current_forks--;
|
|
|
|
}
|
|
|
|
|
|
|
|
current_forks++;
|
|
|
|
if (fork() != 0) continue;
|
|
|
|
}
|
|
|
|
#endif
|
2016-09-03 22:59:23 +03:00
|
|
|
filename = argv[i];
|
|
|
|
size_t path_length = strlen(filename);
|
2016-09-03 03:39:32 +03:00
|
|
|
|
2016-09-03 22:59:23 +03:00
|
|
|
char bitmap_path[path_length + 5]; /* At the worst case, size is strlen(path) + 4 bytes for .bmp + NULL */
|
|
|
|
replace_extension(filename, path_length, bitmap_path, ".bmp");
|
|
|
|
bmp_filename = &bitmap_path[0];
|
|
|
|
|
|
|
|
char log_path[path_length + 5];
|
|
|
|
replace_extension(filename, path_length, log_path, ".log");
|
|
|
|
log_filename = &log_path[0];
|
|
|
|
|
|
|
|
fprintf(stderr, "Testing ROM %s\n", filename);
|
|
|
|
|
2016-09-03 03:39:32 +03:00
|
|
|
if (dmg) {
|
|
|
|
GB_init(&gb);
|
2017-06-21 23:25:39 +03:00
|
|
|
if (GB_load_boot_rom(&gb, boot_rom_path? boot_rom_path : executable_relative_path("dmg_boot.bin"))) {
|
2016-09-03 03:39:32 +03:00
|
|
|
perror("Failed to load boot ROM");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
GB_init_cgb(&gb);
|
2017-06-21 23:25:39 +03:00
|
|
|
if (GB_load_boot_rom(&gb, boot_rom_path? boot_rom_path : executable_relative_path("cgb_boot.bin"))) {
|
2016-09-03 03:39:32 +03:00
|
|
|
perror("Failed to load boot ROM");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
}
|
2016-09-03 22:59:23 +03:00
|
|
|
|
2016-09-03 03:39:32 +03:00
|
|
|
GB_set_vblank_callback(&gb, (GB_vblank_callback_t) vblank);
|
|
|
|
GB_set_pixels_output(&gb, &bitmap[0]);
|
|
|
|
GB_set_rgb_encode_callback(&gb, rgb_encode);
|
|
|
|
GB_set_log_callback(&gb, log_callback);
|
2016-10-22 00:49:32 +03:00
|
|
|
GB_set_async_input_callback(&gb, async_input_callback);
|
2016-09-03 03:39:32 +03:00
|
|
|
|
2016-09-03 22:59:23 +03:00
|
|
|
if (GB_load_rom(&gb, filename)) {
|
|
|
|
perror("Failed to load ROM");
|
|
|
|
exit(1);
|
|
|
|
}
|
2016-09-09 19:29:14 +03:00
|
|
|
|
|
|
|
/* Game specific hacks for start attempt automations */
|
|
|
|
/* It's OK. No overflow is possible here. */
|
|
|
|
start_is_not_first = strcmp((const char *)(gb.rom + 0x134), "NEKOJARA") == 0 ||
|
|
|
|
strcmp((const char *)(gb.rom + 0x134), "GINGA") == 0;
|
2016-10-05 23:56:44 +03:00
|
|
|
a_is_bad = strcmp((const char *)(gb.rom + 0x134), "DESERT STRIKE") == 0 ||
|
|
|
|
/* Restarting in Puzzle Boy/Kwirk (Start followed by A) leaks stack. */
|
|
|
|
strcmp((const char *)(gb.rom + 0x134), "KWIRK") == 0 ||
|
|
|
|
strcmp((const char *)(gb.rom + 0x134), "PUZZLE BOY") == 0;
|
2017-02-24 01:19:44 +02:00
|
|
|
start_is_bad = strcmp((const char *)(gb.rom + 0x134), "BLUESALPHA") == 0;
|
2016-09-09 19:29:14 +03:00
|
|
|
b_is_confirm = strcmp((const char *)(gb.rom + 0x134), "ELITE SOCCER") == 0;
|
|
|
|
push_faster = strcmp((const char *)(gb.rom + 0x134), "MOGURA DE PON!") == 0;
|
2016-10-01 14:31:34 +03:00
|
|
|
push_slower = strcmp((const char *)(gb.rom + 0x134), "BAKENOU") == 0;
|
2016-10-05 23:56:44 +03:00
|
|
|
do_not_stop = strcmp((const char *)(gb.rom + 0x134), "SPACE INVADERS") == 0;
|
2017-05-13 21:03:28 +03:00
|
|
|
push_right = memcmp((const char *)(gb.rom + 0x134), "BOB ET BOB", strlen("BOB ET BOB")) == 0 ||
|
|
|
|
strcmp((const char *)(gb.rom + 0x134), "LITTLE MASTER") == 0 ||
|
|
|
|
/* M&M's Minis Madness Demo (which has no menu but the same title as the full game) */
|
|
|
|
(memcmp((const char *)(gb.rom + 0x134), "MINIMADNESSBMIE", strlen("MINIMADNESSBMIE")) == 0 &&
|
|
|
|
gb.rom[0x14e] == 0x6c);
|
2017-02-25 16:06:38 +02:00
|
|
|
|
2016-10-14 01:30:54 +03:00
|
|
|
|
2017-02-24 23:00:10 +02:00
|
|
|
/* This game temporarily sets SP to OAM RAM */
|
2017-06-21 23:25:39 +03:00
|
|
|
allow_weird_sp_values = strcmp((const char *)(gb.rom + 0x134), "WDL:TT") == 0 ||
|
|
|
|
/* Some mooneye-gb tests abuse the stack */
|
|
|
|
strcmp((const char *)(gb.rom + 0x134), "mooneye-gb test") == 0;
|
2017-02-24 23:00:10 +02:00
|
|
|
|
2017-02-25 16:06:38 +02:00
|
|
|
/* This game uses some recursive algorithms and therefore requires quite a large call stack */
|
2017-05-13 21:03:28 +03:00
|
|
|
large_stack = memcmp((const char *)(gb.rom + 0x134), "MICRO EPAK1BM", strlen("MICRO EPAK1BM")) == 0 ||
|
|
|
|
strcmp((const char *)(gb.rom + 0x134), "TECMO BOWL") == 0;
|
2017-02-25 16:06:38 +02:00
|
|
|
|
2017-06-21 23:25:39 +03:00
|
|
|
/* Pressing start while in the map in Tsuri Sensei will leak an internal screen-stack which
|
2016-10-14 01:30:54 +03:00
|
|
|
will eventually overflow, override an array of jump-table indexes, jump to a random
|
|
|
|
address, execute an invalid opcode, and crash. Pressing A twice while slowing down
|
|
|
|
will prevent this scenario. */
|
|
|
|
push_a_twice = strcmp((const char *)(gb.rom + 0x134), "TURI SENSEI V1") == 0;
|
2016-09-09 19:29:14 +03:00
|
|
|
|
2016-09-03 03:39:32 +03:00
|
|
|
/* Run emulation */
|
|
|
|
running = true;
|
|
|
|
gb.turbo = gb.turbo_dont_skip = gb.disable_rendering = true;
|
|
|
|
frames = 0;
|
|
|
|
while (running) {
|
|
|
|
GB_run(&gb);
|
2016-09-30 18:24:01 +03:00
|
|
|
/* This early crash test must not run in vblank because PC might not point to the next instruction. */
|
|
|
|
if (gb.pc == 0x38 && frames < test_length - 1 && GB_read_memory(&gb, 0x38) == 0xFF) {
|
|
|
|
GB_log(&gb, "The game is probably stuck in an FF loop.\n");
|
|
|
|
frames = test_length - 1;
|
|
|
|
}
|
2016-09-03 03:39:32 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
if (log_file) {
|
|
|
|
fclose(log_file);
|
|
|
|
log_file = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
GB_free(&gb);
|
|
|
|
#ifndef _WIN32
|
|
|
|
if (max_forks > 1) {
|
|
|
|
exit(0);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
#ifndef _WIN32
|
|
|
|
int wait_out;
|
|
|
|
while(wait(&wait_out) != -1);
|
|
|
|
#endif
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|