redream/src/host/sdl_host.c

1117 lines
28 KiB
C

#include <glad/glad.h>
#include <SDL.h>
#include <SDL_opengl.h>
#include "core/core.h"
#include "core/filesystem.h"
#include "core/profiler.h"
#include "core/ringbuf.h"
#include "core/time.h"
#include "core/version.h"
#include "emulator.h"
#include "guest/aica/aica.h"
#include "host/host.h"
#include "imgui.h"
#include "options.h"
#include "render/render_backend.h"
#include "tracer.h"
#include "ui.h"
/*
* sdl host implementation
*/
#define AUDIO_FREQ AICA_SAMPLE_FREQ
#define VIDEO_DEFAULT_WIDTH 853
#define VIDEO_DEFAULT_HEIGHT 480
#define INPUT_MAX_CONTROLLERS 4
#define AUDIO_FRAME_SIZE 4 /* stereo / pcm16 */
#define AUDIO_FRAMES_TO_MS(frames) \
(int)(((float)frames * 1000.0f) / (float)AUDIO_FREQ)
#define MS_TO_AUDIO_FRAMES(ms) (int)(((float)(ms) / 1000.0f) * AUDIO_FREQ)
#define NS_TO_AUDIO_FRAMES(ns) (int)(((float)(ns) / NS_PER_SEC) * AUDIO_FREQ)
struct host {
struct SDL_Window *win;
int closed;
struct ui *ui;
struct emu *emu;
struct tracer *tracer;
struct imgui *imgui;
struct {
SDL_AudioDeviceID dev;
SDL_AudioSpec spec;
int playing;
struct ringbuf *frames;
volatile int64_t last_cb;
} audio;
struct {
SDL_GLContext ctx;
struct render_backend *r;
int width;
int height;
} video;
struct {
struct button_map *keymap[K_NUM_KEYS];
SDL_GameController *controllers[INPUT_MAX_CONTROLLERS];
} input;
struct {
int show_menu;
int show_times;
unsigned frame;
float swap_times[360];
int64_t last_swap;
} dbg;
};
/*
* audio
*/
static int audio_read_frames(struct host *host, void *data, int num_frames) {
int buffered = ringbuf_available(host->audio.frames);
int size = MIN(buffered, num_frames * AUDIO_FRAME_SIZE);
CHECK_EQ(size % AUDIO_FRAME_SIZE, 0);
void *read_ptr = ringbuf_read_ptr(host->audio.frames);
memcpy(data, read_ptr, size);
ringbuf_advance_read_ptr(host->audio.frames, size);
return size / AUDIO_FRAME_SIZE;
}
static void audio_write_frames(struct host *host, const void *data,
int num_frames) {
int remaining = ringbuf_remaining(host->audio.frames);
int size = MIN(remaining, num_frames * AUDIO_FRAME_SIZE);
CHECK_EQ(size % AUDIO_FRAME_SIZE, 0);
void *write_ptr = ringbuf_write_ptr(host->audio.frames);
memcpy(write_ptr, data, size);
ringbuf_advance_write_ptr(host->audio.frames, size);
}
static int audio_buffered_frames(struct host *host) {
int buffered = ringbuf_available(host->audio.frames);
return buffered / AUDIO_FRAME_SIZE;
}
static int audio_buffer_low(struct host *host) {
if (!host->audio.dev) {
/* lie and say the audio buffer is low, forcing the emulator to run as fast
as possible */
return 1;
}
/* SDL's write callback is called very coarsely, seemingly, only each time
its buffered data has completely drained
since the main loop is designed to synchronize speed based on the amount
of buffered audio data, with larger buffer sizes (due to a larger latency
setting) this can result in the callback being called only one time for
multiple video frames
this creates a situation where multiple video frames are immediately ran
when the callback fires in order to push enough audio data to avoid an
underflow, and then multiple vblanks occur on the host where no new frame
is presented as the main loop again blocks waiting for another write
callback to decrease the amount of buffered audio data
in order to smooth out the video frame timings when the audio latency is
high, the host clock is used to interpolate the amount of buffered audio
data between callbacks */
int64_t now = time_nanoseconds();
int64_t since_last_cb = now - host->audio.last_cb;
int frames_buffered = audio_buffered_frames(host);
frames_buffered -= NS_TO_AUDIO_FRAMES(since_last_cb);
int low_water_mark = host->audio.spec.samples / 2;
return frames_buffered < low_water_mark;
}
static void audio_write_cb(void *userdata, Uint8 *stream, int len) {
struct host *host = userdata;
Sint32 *buf = (Sint32 *)stream;
int frame_count_max = len / AUDIO_FRAME_SIZE;
static uint32_t tmp[AUDIO_FREQ];
int frames_buffered = audio_buffered_frames(host);
int frames_remaining = MIN(frames_buffered, frame_count_max);
while (frames_remaining > 0) {
/* batch read frames from ring buffer */
int n = MIN(frames_remaining, ARRAY_SIZE(tmp));
n = audio_read_frames(host, tmp, n);
frames_remaining -= n;
/* copy frames to output stream */
memcpy(buf, tmp, n * AUDIO_FRAME_SIZE);
}
host->audio.last_cb = time_nanoseconds();
}
static void audio_destroy_device(struct host *host) {
if (!host->audio.dev) {
return;
}
SDL_CloseAudioDevice(host->audio.dev);
host->audio.dev = 0;
}
static int audio_create_device(struct host *host) {
/* SDL expects the number of buffered frames to be a power of two */
int target_frames = 1 << 12;
/* match AICA output format */
SDL_AudioSpec want;
SDL_zero(want);
want.freq = AUDIO_FREQ;
want.format = AUDIO_S16LSB;
want.channels = 2;
want.samples = target_frames;
want.userdata = host;
want.callback = audio_write_cb;
host->audio.dev = SDL_OpenAudioDevice(NULL, 0, &want, &host->audio.spec, 0);
if (!host->audio.dev) {
LOG_WARNING("audio_create_device failed to open device: %s",
SDL_GetError());
return 0;
}
LOG_INFO("audio_create_device latency=%d ms/%d frames",
AUDIO_FRAMES_TO_MS(host->audio.spec.samples),
host->audio.spec.samples);
return 1;
}
void audio_push(struct host *host, const int16_t *data, int num_frames) {
if (!host->audio.dev) {
return;
}
audio_write_frames(host, data, num_frames);
/* start playback once some audio is queued */
if (!host->audio.playing) {
SDL_PauseAudioDevice(host->audio.dev, 0);
host->audio.playing = 1;
}
}
static void audio_shutdown(struct host *host) {
if (host->audio.dev) {
SDL_CloseAudioDevice(host->audio.dev);
}
if (host->audio.frames) {
ringbuf_destroy(host->audio.frames);
}
}
static int audio_init(struct host *host) {
/* reset */
memset(&host->audio, 0, sizeof(host->audio));
/* if the audio sync is disabled, don't actually create a device, as audio
playback won't be possible without resampling the data */
if (!audio_sync_enabled()) {
return 1;
}
/* create ringbuffer to store data coming in from AICA. note, the buffer needs
to be at least two video frames in size, in order to handle the coarse
synchronization used by the main loop, where an entire guest video frame is
ran when the buffered audio data is deemed low */
host->audio.frames = ringbuf_create(AUDIO_FREQ * AUDIO_FRAME_SIZE);
int success = audio_create_device(host);
if (!success) {
LOG_WARNING("audio_init failed to open audio device: %s", SDL_GetError());
return 0;
}
return 1;
}
static int audio_restart(struct host *host) {
audio_shutdown(host);
return audio_init(host);
}
/*
* video
*/
static void video_destroy_context(struct host *host, SDL_GLContext ctx) {
/* make sure the context is no longer active before deleting */
int res = SDL_GL_MakeCurrent(host->win, NULL);
CHECK_EQ(res, 0);
SDL_GL_DeleteContext(ctx);
}
static SDL_GLContext video_create_context(struct host *host) {
#if PLATFORM_ANDROID
SDL_GL_SetAttribute(SDL_GL_CONTEXT_MAJOR_VERSION, 3);
SDL_GL_SetAttribute(SDL_GL_CONTEXT_MINOR_VERSION, 1);
SDL_GL_SetAttribute(SDL_GL_CONTEXT_PROFILE_MASK, SDL_GL_CONTEXT_PROFILE_ES);
#else
SDL_GL_SetAttribute(SDL_GL_CONTEXT_MAJOR_VERSION, 3);
SDL_GL_SetAttribute(SDL_GL_CONTEXT_MINOR_VERSION, 3);
SDL_GL_SetAttribute(SDL_GL_CONTEXT_PROFILE_MASK, SDL_GL_CONTEXT_PROFILE_CORE);
#endif
/* SDL defaults to allocating a 16-bit depth buffer, raise this to at least
24-bits to help with the depth precision lost when converting from PVR
coordinates to OpenGL */
SDL_GL_SetAttribute(SDL_GL_DEPTH_SIZE, 24);
SDL_GLContext ctx = SDL_GL_CreateContext(host->win);
CHECK_NOTNULL(ctx, "video_create_context failed: %s", SDL_GetError());
/* force vsync */
int vsync = video_sync_enabled();
int res = SDL_GL_SetSwapInterval(vsync);
CHECK_EQ(res, 0, "video_create_context failed to set swap interval");
/* link in gl functions at runtime */
res = gladLoadGLLoader((GLADloadproc)&SDL_GL_GetProcAddress);
CHECK_EQ(res, 1, "video_create_context failed to link");
return ctx;
}
static void video_set_fullscreen(struct host *host, int fullscreen) {
if (fullscreen) {
int res = SDL_SetWindowFullscreen(host->win, SDL_WINDOW_FULLSCREEN_DESKTOP);
CHECK(res >= 0);
res = SDL_ShowCursor(SDL_DISABLE);
CHECK(res >= 0);
} else {
int res = SDL_SetWindowFullscreen(host->win, 0);
CHECK(res >= 0);
res = SDL_ShowCursor(SDL_ENABLE);
CHECK(res >= 0);
}
}
static void video_shutdown(struct host *host) {
imgui_vid_destroyed(host->imgui);
if (host->tracer) {
tracer_vid_destroyed(host->tracer);
}
if (host->emu) {
emu_vid_destroyed(host->emu);
}
if (host->ui) {
ui_vid_destroyed(host->ui);
}
r_destroy(host->video.r);
video_destroy_context(host, host->video.ctx);
}
static int video_init(struct host *host) {
/* reset */
memset(&host->video, 0, sizeof(host->video));
/* immediately poll window size for platforms like Android where the window
starts fullscreen, ignoring the default width and height */
SDL_GetWindowSize(host->win, &host->video.width, &host->video.height);
host->video.ctx = video_create_context(host);
host->video.r = r_create(host->video.width, host->video.height);
if (host->ui) {
ui_vid_created(host->ui, host->video.r);
}
if (host->emu) {
emu_vid_created(host->emu, host->video.r);
}
if (host->tracer) {
tracer_vid_created(host->tracer, host->video.r);
}
imgui_vid_created(host->imgui, host->video.r);
return 1;
}
static int video_restart(struct host *host) {
video_shutdown(host);
return video_init(host);
}
/*
* input
*/
static void host_poll_events(struct host *host);
static int translate_sdl_key(SDL_Keysym keysym) {
int out = K_UNKNOWN;
if (keysym.sym >= SDLK_SPACE && keysym.sym <= SDLK_z) {
/* this range maps 1:1 with ASCII chars */
out = keysym.sym;
} else {
switch (keysym.sym) {
case SDLK_CAPSLOCK:
out = K_CAPSLOCK;
break;
case SDLK_RETURN:
out = K_RETURN;
break;
case SDLK_ESCAPE:
out = K_ESCAPE;
break;
case SDLK_BACKSPACE:
out = K_BACKSPACE;
break;
case SDLK_TAB:
out = K_TAB;
break;
case SDLK_PAGEUP:
out = K_PAGEUP;
break;
case SDLK_PAGEDOWN:
out = K_PAGEDOWN;
break;
case SDLK_DELETE:
out = K_DELETE;
break;
case SDLK_RIGHT:
out = K_RIGHT;
break;
case SDLK_LEFT:
out = K_LEFT;
break;
case SDLK_DOWN:
out = K_DOWN;
break;
case SDLK_UP:
out = K_UP;
break;
case SDLK_LCTRL:
out = K_LCTRL;
break;
case SDLK_LSHIFT:
out = K_LSHIFT;
break;
case SDLK_LALT:
out = K_LALT;
break;
case SDLK_LGUI:
out = K_LGUI;
break;
case SDLK_RCTRL:
out = K_RCTRL;
break;
case SDLK_RSHIFT:
out = K_RSHIFT;
break;
case SDLK_RALT:
out = K_RALT;
break;
case SDLK_RGUI:
out = K_RGUI;
break;
case SDLK_F1:
out = K_F1;
break;
case SDLK_F2:
out = K_F2;
break;
case SDLK_F3:
out = K_F3;
break;
case SDLK_F4:
out = K_F4;
break;
case SDLK_F5:
out = K_F5;
break;
case SDLK_F6:
out = K_F6;
break;
case SDLK_F7:
out = K_F7;
break;
case SDLK_F8:
out = K_F8;
break;
case SDLK_F9:
out = K_F9;
break;
case SDLK_F10:
out = K_F10;
break;
case SDLK_F11:
out = K_F11;
break;
case SDLK_F12:
out = K_F12;
break;
case SDLK_F13:
out = K_F13;
break;
case SDLK_F14:
out = K_F14;
break;
case SDLK_F15:
out = K_F15;
break;
case SDLK_F16:
out = K_F16;
break;
case SDLK_F17:
out = K_F17;
break;
case SDLK_F18:
out = K_F18;
break;
case SDLK_F19:
out = K_F19;
break;
case SDLK_F20:
out = K_F20;
break;
case SDLK_F21:
out = K_F21;
break;
case SDLK_F22:
out = K_F22;
break;
case SDLK_F23:
out = K_F23;
break;
case SDLK_F24:
out = K_F24;
break;
}
}
if (keysym.scancode == SDL_SCANCODE_GRAVE) {
out = K_CONSOLE;
}
return out;
}
static int16_t filter_sdl_motion(int16_t value, int deadzone) {
CHECK(deadzone >= 0 && deadzone <= INT16_MAX);
/* maximum input value after accounting for the deadzone */
float deadmax = (float)(INT16_MAX - deadzone);
if (value < -deadzone) {
value = (int16_t)((((float)value + deadzone) / deadmax) * INT16_MAX);
} else if (value > deadzone) {
value = (int16_t)((((float)value - deadzone) / deadmax) * INT16_MAX);
}
return value;
}
static int input_find_controller_port(struct host *host, int instance_id) {
for (int port = 0; port < INPUT_MAX_CONTROLLERS; port++) {
SDL_GameController *ctrl = host->input.controllers[port];
SDL_Joystick *joy = SDL_GameControllerGetJoystick(ctrl);
if (SDL_JoystickInstanceID(joy) == instance_id) {
return port;
}
}
return -1;
}
static void input_update_keymap(struct host *host) {
memset(host->input.keymap, 0, sizeof(host->input.keymap));
for (int i = 0; i < NUM_BUTTONS; i++) {
struct button_map *mapping = &BUTTONS[i];
if (mapping->key) {
host->input.keymap[*mapping->key] = mapping;
}
}
}
static void input_mousemove(struct host *host, int port, int x, int y) {
imgui_mousemove(host->imgui, x, y);
}
static void input_keydown(struct host *host, int port, int key, int16_t value) {
/* send event for both the key as well the button it's mapped to (if any) */
struct button_map *mapping = host->input.keymap[key];
int button = K_UNKNOWN;
if (mapping) {
button = mapping->btn;
/* if the mapping is to a negative axis, flip the value */
if (mapping->dir == DIR_NEG) {
value = -value;
}
}
if (key == K_F1 && value) {
host->dbg.show_menu = !host->dbg.show_menu;
return;
}
for (int i = 0; i < 2; i++) {
if (key == K_UNKNOWN) {
break;
}
if (host->ui && ui_keydown(host->ui, key, value)) {
continue;
}
if (host->emu && emu_keydown(host->emu, port, key, value)) {
continue;
}
if (host->tracer && tracer_keydown(host->tracer, key, value)) {
continue;
}
imgui_keydown(host->imgui, key, value);
/* send event for mapped button as well */
key = button;
}
}
static void input_controller_removed(struct host *host, int port) {
SDL_GameController *ctrl = host->input.controllers[port];
if (!ctrl) {
return;
}
const char *name = SDL_GameControllerName(ctrl);
LOG_INFO("input_controller_removed port=%d name=%s", port, name);
SDL_GameControllerClose(ctrl);
host->input.controllers[port] = NULL;
}
static void input_controller_added(struct host *host, int device_id) {
/* find the next open controller port */
int port;
for (port = 0; port < INPUT_MAX_CONTROLLERS; port++) {
if (!host->input.controllers[port]) {
break;
}
}
if (port >= INPUT_MAX_CONTROLLERS) {
LOG_WARNING("input_controller_added no open ports");
return;
}
SDL_GameController *ctrl = SDL_GameControllerOpen(device_id);
host->input.controllers[port] = ctrl;
const char *name = SDL_GameControllerName(ctrl);
LOG_INFO("input_controller_added port=%d name=%s", port, name);
}
static void input_shutdown(struct host *host) {
for (int i = 0; i < INPUT_MAX_CONTROLLERS; i++) {
input_controller_removed(host, i);
}
}
static int input_init(struct host *host) {
/* reset */
memset(&host->input, 0, sizeof(host->input));
input_update_keymap(host);
return 1;
}
int input_max_controllers(struct host *host) {
return INPUT_MAX_CONTROLLERS;
}
const char *input_controller_name(struct host *host, int port) {
CHECK(port >= 0 && port < INPUT_MAX_CONTROLLERS);
SDL_GameController *ctrl = host->input.controllers[port];
return SDL_GameControllerName(ctrl);
}
/*
* ui
*/
void ui_closed(struct host *host) {}
void ui_opened(struct host *host) {}
int ui_load_game(struct host *host, const char *path) {
return emu_load(host->emu, path);
}
/*
* internal
*/
static void host_swap_window(struct host *host) {
SDL_GL_SwapWindow(host->win);
/* keep track of the time between swaps */
int64_t now = time_nanoseconds();
if (host->dbg.last_swap) {
float swap_time_ms = (float)(now - host->dbg.last_swap) / 1000000.0f;
int num_times = ARRAY_SIZE(host->dbg.swap_times);
host->dbg.swap_times[host->dbg.frame % num_times] = swap_time_ms;
}
host->dbg.last_swap = now;
host->dbg.frame++;
}
static void host_debug_menu(struct host *host) {
#ifdef HAVE_IMGUI
if (!host->dbg.show_menu) {
return;
}
if (igBeginMainMenuBar()) {
if (igBeginMenu("HOST", 1)) {
if (igMenuItem("frame times", NULL, host->dbg.show_times, 1)) {
host->dbg.show_times = !host->dbg.show_times;
}
igEndMenu();
}
igEndMainMenuBar();
}
if (host->dbg.show_times) {
bool opened = true;
if (igBegin("frame times", &opened, ImGuiWindowFlags_AlwaysAutoResize)) {
struct ImVec2 graph_size = {300.0f, 50.0f};
int num_times = ARRAY_SIZE(host->dbg.swap_times);
float min_time = FLT_MAX;
float max_time = -FLT_MAX;
float avg_time = 0.0f;
for (int i = 0; i < num_times; i++) {
float time = host->dbg.swap_times[i];
min_time = MIN(min_time, time);
max_time = MAX(max_time, time);
avg_time += time;
}
avg_time /= num_times;
igValueFloat("min frame time", min_time, "%.2f");
igValueFloat("max frame time", max_time, "%.2f");
igValueFloat("avg frame time", avg_time, "%.2f");
igPlotLines("", host->dbg.swap_times, num_times,
host->dbg.frame % num_times, NULL, 0.0f, 60.0f, graph_size,
sizeof(float));
}
igEnd();
host->dbg.show_times = (int)opened;
}
emu_debug_menu(host->emu);
#endif
}
static void host_poll_events(struct host *host) {
SDL_Event ev;
while (SDL_PollEvent(&ev)) {
switch (ev.type) {
case SDL_KEYDOWN: {
int keycode = translate_sdl_key(ev.key.keysym);
if (keycode != K_UNKNOWN) {
input_keydown(host, 0, keycode, INT16_MAX);
}
} break;
case SDL_KEYUP: {
int keycode = translate_sdl_key(ev.key.keysym);
if (keycode != K_UNKNOWN) {
input_keydown(host, 0, keycode, 0);
}
} break;
case SDL_MOUSEBUTTONDOWN:
case SDL_MOUSEBUTTONUP: {
int keycode;
switch (ev.button.button) {
case SDL_BUTTON_LEFT:
keycode = K_MOUSE1;
break;
case SDL_BUTTON_RIGHT:
keycode = K_MOUSE2;
break;
case SDL_BUTTON_MIDDLE:
keycode = K_MOUSE3;
break;
case SDL_BUTTON_X1:
keycode = K_MOUSE4;
break;
case SDL_BUTTON_X2:
keycode = K_MOUSE5;
break;
default:
keycode = K_UNKNOWN;
break;
}
if (keycode != K_UNKNOWN) {
int16_t value = ev.type == SDL_MOUSEBUTTONDOWN ? INT16_MAX : 0;
input_keydown(host, 0, keycode, value);
}
} break;
case SDL_MOUSEWHEEL:
if (ev.wheel.y > 0) {
input_keydown(host, 0, K_MWHEELUP, INT16_MAX);
input_keydown(host, 0, K_MWHEELUP, 0);
} else {
input_keydown(host, 0, K_MWHEELDOWN, INT16_MAX);
input_keydown(host, 0, K_MWHEELDOWN, 0);
}
break;
case SDL_MOUSEMOTION:
input_mousemove(host, 0, ev.motion.x, ev.motion.y);
break;
case SDL_CONTROLLERDEVICEADDED: {
input_controller_added(host, ev.cdevice.which);
} break;
case SDL_CONTROLLERDEVICEREMOVED: {
int port = input_find_controller_port(host, ev.cdevice.which);
if (port != -1) {
input_controller_removed(host, port);
}
} break;
case SDL_CONTROLLERAXISMOTION: {
int port = input_find_controller_port(host, ev.caxis.which);
int key = K_UNKNOWN;
switch (ev.caxis.axis) {
case SDL_CONTROLLER_AXIS_LEFTX:
key = K_CONT_JOYX;
break;
case SDL_CONTROLLER_AXIS_LEFTY:
key = K_CONT_JOYY;
break;
case SDL_CONTROLLER_AXIS_TRIGGERLEFT:
key = K_CONT_LTRIG;
break;
case SDL_CONTROLLER_AXIS_TRIGGERRIGHT:
key = K_CONT_RTRIG;
break;
}
if (port != -1 && key != K_UNKNOWN) {
int16_t value = filter_sdl_motion(ev.caxis.value, *DEADZONES[port]);
input_keydown(host, port, key, value);
}
} break;
case SDL_CONTROLLERBUTTONDOWN:
case SDL_CONTROLLERBUTTONUP: {
int port = input_find_controller_port(host, ev.cbutton.which);
int key = K_UNKNOWN;
int16_t value = ev.type == SDL_CONTROLLERBUTTONDOWN ? INT16_MAX : 0;
switch (ev.cbutton.button) {
case SDL_CONTROLLER_BUTTON_A:
key = K_CONT_A;
break;
case SDL_CONTROLLER_BUTTON_B:
key = K_CONT_B;
break;
case SDL_CONTROLLER_BUTTON_X:
key = K_CONT_X;
break;
case SDL_CONTROLLER_BUTTON_Y:
key = K_CONT_Y;
break;
case SDL_CONTROLLER_BUTTON_START:
key = K_CONT_START;
break;
case SDL_CONTROLLER_BUTTON_DPAD_UP:
key = K_CONT_DPAD_UP;
break;
case SDL_CONTROLLER_BUTTON_DPAD_DOWN:
key = K_CONT_DPAD_DOWN;
break;
case SDL_CONTROLLER_BUTTON_DPAD_LEFT:
key = K_CONT_DPAD_LEFT;
break;
case SDL_CONTROLLER_BUTTON_DPAD_RIGHT:
key = K_CONT_DPAD_RIGHT;
break;
}
if (port != -1 && key != K_UNKNOWN) {
input_keydown(host, port, key, value);
}
} break;
case SDL_WINDOWEVENT:
if (ev.window.event == SDL_WINDOWEVENT_RESIZED) {
host->video.width = ev.window.data1;
host->video.height = ev.window.data2;
int res = video_restart(host);
CHECK(res, "video_restart failed");
}
break;
case SDL_QUIT:
host->closed = 1;
break;
}
}
/* check for option changes at this time as well */
if (OPTION_fullscreen_dirty) {
video_set_fullscreen(host, OPTION_fullscreen);
OPTION_fullscreen_dirty = 0;
}
if (OPTION_sync_dirty) {
int res = audio_restart(host);
CHECK(res, "audio_restart failed");
res = video_restart(host);
CHECK(res, "video_restart failed");
OPTION_sync_dirty = 0;
}
/* update reverse button map when options change */
int dirty_map = 0;
for (int i = 0; i < NUM_BUTTONS; i++) {
struct button_map *mapping = &BUTTONS[i];
if (!mapping->desc) {
continue;
}
if (*mapping->dirty) {
dirty_map = 1;
*mapping->dirty = 0;
}
}
if (dirty_map) {
input_update_keymap(host);
}
}
static void host_shutdown(struct host *host) {
input_shutdown(host);
video_shutdown(host);
audio_shutdown(host);
}
static int host_init(struct host *host) {
/* init sdl and create window */
int res = SDL_Init(SDL_INIT_AUDIO | SDL_INIT_VIDEO | SDL_INIT_GAMECONTROLLER);
CHECK_GE(res, 0, "host_create sdl initialization failed: %s", SDL_GetError());
uint32_t win_flags = SDL_WINDOW_OPENGL | SDL_WINDOW_RESIZABLE;
host->win = SDL_CreateWindow("redream", SDL_WINDOWPOS_UNDEFINED,
SDL_WINDOWPOS_UNDEFINED, VIDEO_DEFAULT_WIDTH,
VIDEO_DEFAULT_HEIGHT, win_flags);
CHECK_NOTNULL(host->win, "host_create window creation failed: %s",
SDL_GetError());
if (OPTION_fullscreen) {
video_set_fullscreen(host, 1);
}
if (!audio_init(host)) {
return 0;
}
if (!video_init(host)) {
return 0;
}
if (!input_init(host)) {
return 0;
}
return 1;
}
void host_destroy(struct host *host) {
if (host->win) {
SDL_DestroyWindow(host->win);
}
SDL_Quit();
free(host);
}
struct host *host_create() {
struct host *host = calloc(1, sizeof(struct host));
return host;
}
int main(int argc, char **argv) {
LOG_INFO("redream " GIT_VERSION);
#if PLATFORM_ANDROID
const char *appdir = SDL_AndroidGetExternalStoragePath();
fs_set_appdir(appdir);
#else
char userdir[PATH_MAX];
int r = fs_userdir(userdir, sizeof(userdir));
CHECK(r);
char appdir[PATH_MAX];
snprintf(appdir, sizeof(appdir), "%s" PATH_SEPARATOR ".redream", userdir);
fs_set_appdir(appdir);
#endif
/* load base options from config */
char config[PATH_MAX] = {0};
snprintf(config, sizeof(config), "%s" PATH_SEPARATOR "config", appdir);
options_read(config);
/* override options from the command line */
if (!options_parse(&argc, &argv)) {
return EXIT_FAILURE;
}
const char *load = argc > 1 ? argv[1] : NULL;
struct host *host = host_create();
host->imgui = imgui_create();
if (load && strstr(load, ".trace")) {
host->tracer = tracer_create(host);
} else {
host->ui = ui_create(host);
host->emu = emu_create(host);
}
/* init host after creating emulator / tracer client, so host can notify them
them that the audio / video / input subsystems have been initialized */
if (host_init(host)) {
if (host->tracer) {
if (tracer_load(host->tracer, load)) {
while (!host->closed) {
host_poll_events(host);
/* reset vertex buffers */
imgui_begin_frame(host->imgui);
/* render tracer output first */
tracer_render_frame(host->tracer);
/* overlay user interface */
imgui_end_frame(host->imgui);
host_swap_window(host);
}
}
} else if (host->emu) {
int success = 0;
if (load || OPTION_bios) {
success = emu_load(host->emu, load);
}
if (!success) {
/* if nothing is loaded, open the game select ui */
ui_set_page(host->ui, UI_PAGE_GAMES);
}
while (!host->closed) {
/* even though the emulator itself will poll for events when updating
controller input, the main loop needs to also poll to ensure the
close event is received */
host_poll_events(host);
/* only step the emulator if the available audio is running low. this
syncs the emulation speed with the host audio clock. note however,
if audio is disabled, the emulator will run unthrottled */
if (!audio_buffer_low(host)) {
continue;
}
/* reset vertex buffers */
imgui_begin_frame(host->imgui);
/* render emulator output and build up imgui buffers */
host_debug_menu(host);
emu_render_frame(host->emu);
ui_build_menus(host->ui);
/* overlay imgui */
imgui_end_frame(host->imgui);
/* flip profiler at end of frame */
int64_t now = time_nanoseconds();
prof_flip(time_nanoseconds());
host_swap_window(host);
}
}
}
host_shutdown(host);
if (host->ui) {
ui_destroy(host->ui);
host->ui = NULL;
}
if (host->emu) {
emu_destroy(host->emu);
}
if (host->tracer) {
tracer_destroy(host->tracer);
}
imgui_destroy(host->imgui);
host_destroy(host);
/* persist options for next run */
options_write(config);
return EXIT_SUCCESS;
}