#include "test-runner.h" #include <mgba/core/core.h> #include <mgba/feature/commandline.h> #include <stdio.h> _Static_assert(BYTES_PER_PIXEL == 4, "bytes per pixel MUST be four"); void log_output(struct mLogger* _log, int category, enum mLogLevel level, const char* format, va_list args); char* log_level_str(enum mLogLevel); struct MGBA { struct mLogger mlogger; struct mCore* core; struct video_buffer videoBuffer; char* filename; struct callback callback; }; struct MGBA* new_runner(char* filename) { struct MGBA* mgba = calloc(1, sizeof(struct MGBA)); mgba->mlogger.log = log_output; mgba->callback.callback = NULL; mLogSetDefaultLogger(&mgba->mlogger); char* filename_new = strdup(filename); mgba->filename = filename_new; struct mCore* core = mCoreFind(mgba->filename); if (!core) { printf("failed to find core\n"); free(mgba); return NULL; } core->init(core); unsigned width, height; core->desiredVideoDimensions(core, &width, &height); ssize_t videoBufferSize = width * height * BYTES_PER_PIXEL; uint32_t* videoBuffer = malloc(videoBufferSize * sizeof(*videoBuffer)); core->setVideoBuffer(core, videoBuffer, width); // load rom mCoreLoadFile(core, mgba->filename); mCoreConfigInit(&core->config, NULL); core->reset(core); mgba->core = core; mgba->videoBuffer = (struct video_buffer){ .buffer = videoBuffer, .width = width, .height = height, }; return mgba; } void set_logger(struct MGBA* mgba, struct callback callback) { mgba->callback = callback; } void free_runner(struct MGBA* mgba) { mgba->core->deinit(mgba->core); mgba->callback.destroy(mgba->callback.data); free(mgba->filename); free(mgba->videoBuffer.buffer); free(mgba); } void advance_frame(struct MGBA* mgba) { mgba->core->runFrame(mgba->core); } struct video_buffer get_video_buffer(struct MGBA* mgba) { return mgba->videoBuffer; } void log_output(struct mLogger* log, int category, enum mLogLevel level, const char* format, va_list args) { // cast log to mgba, this works as the logger is the top entry of the mgba // struct struct MGBA* mgba = (struct MGBA*)log; if (level & 31) { int32_t size = 0; size += snprintf(NULL, 0, "[%s] %s: ", log_level_str(level), mLogCategoryName(category)); va_list args_copy; va_copy(args_copy, args); size += vsnprintf(NULL, 0, format, args_copy); va_end(args_copy); size += 1; char* str = calloc(size, sizeof(*str)); int32_t offset = snprintf(str, size, "[%s] %s: ", log_level_str(level), mLogCategoryName(category)); size -= offset; vsnprintf(&str[offset], size, format, args); if (mgba->callback.callback != NULL) mgba->callback.callback(mgba->callback.data, str); else printf("%s\n", str); free(str); } } char* log_level_str(enum mLogLevel level) { switch (level) { case mLOG_FATAL: return "FATAL"; break; case mLOG_ERROR: return "ERROR"; break; case mLOG_WARN: return "WARNING"; break; case mLOG_INFO: return "INFO"; break; case mLOG_DEBUG: return "DEBUG"; break; case mLOG_STUB: return "STUB"; break; case mLOG_GAME_ERROR: return "GAME ERROR"; break; default: return "Unknown"; } }