/* SSNES - A Super Nintendo Entertainment System (SNES) Emulator frontend for libsnes. * Copyright (C) 2010-2012 - Hans-Kristian Arntzen * * Some code herein may be based on code found in BSNES. * * SSNES is free software: you can redistribute it and/or modify it under the terms * of the GNU General Public License as published by the Free Software Found- * ation, either version 3 of the License, or (at your option) any later version. * * SSNES is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR * PURPOSE. See the GNU General Public License for more details. * * You should have received a copy of the GNU General Public License along with SSNES. * If not, see . */ #include "snes_state.h" #include #include "../strl.h" #include "../general.h" #include "../libsnes.hpp" #ifdef HAVE_PYTHON #include "py_state/py_state.h" #endif struct snes_tracker_internal { char id[64]; bool is_input; const uint16_t *input_ptr; const uint8_t *ptr; #ifdef HAVE_PYTHON py_state_t *py; #endif uint32_t addr; unsigned mask; enum snes_tracker_type type; uint32_t prev[2]; int frame_count; int frame_count_prev; uint32_t old_value; int transition_count; }; struct snes_tracker { struct snes_tracker_internal *info; unsigned info_elem; uint16_t input_state[2]; #ifdef HAVE_PYTHON py_state_t *py; #endif }; snes_tracker_t* snes_tracker_init(const struct snes_tracker_info *info) { snes_tracker_t *tracker = (snes_tracker_t*)calloc(1, sizeof(*tracker)); if (!tracker) return NULL; #ifdef HAVE_PYTHON if (info->script) { tracker->py = py_state_new(info->script, info->script_is_file, info->script_class ? info->script_class : "GameAware"); if (!tracker->py) { free(tracker); SSNES_ERR("Failed to init Python script.\n"); return NULL; } } #endif tracker->info = (struct snes_tracker_internal*)calloc(info->info_elem, sizeof(struct snes_tracker_internal)); tracker->info_elem = info->info_elem; for (unsigned i = 0; i < info->info_elem; i++) { strlcpy(tracker->info[i].id, info->info[i].id, sizeof(tracker->info[i].id)); tracker->info[i].addr = info->info[i].addr; tracker->info[i].type = info->info[i].type; tracker->info[i].mask = (info->info[i].mask == 0) ? -1u : info->info[i].mask; #ifdef HAVE_PYTHON if (info->info[i].type == SSNES_STATE_PYTHON) tracker->info[i].py = tracker->py; #endif // If we don't have a valid pointer. static const uint8_t empty = 0; switch (info->info[i].ram_type) { case SSNES_STATE_WRAM: tracker->info[i].ptr = info->wram ? info->wram : ∅ break; case SSNES_STATE_APURAM: tracker->info[i].ptr = info->apuram ? info->apuram : ∅ break; case SSNES_STATE_OAM: tracker->info[i].ptr = info->oam ? info->oam : ∅ break; case SSNES_STATE_CGRAM: tracker->info[i].ptr = info->cgram ? info->cgram : ∅ break; case SSNES_STATE_VRAM: tracker->info[i].ptr = info->vram ? info->vram : ∅ break; case SSNES_STATE_INPUT_SLOT1: tracker->info[i].input_ptr = &tracker->input_state[0]; tracker->info[i].is_input = true; break; case SSNES_STATE_INPUT_SLOT2: tracker->info[i].input_ptr = &tracker->input_state[1]; tracker->info[i].is_input = true; break; default: tracker->info[i].ptr = ∅ } } return tracker; } void snes_tracker_free(snes_tracker_t *tracker) { free(tracker->info); #ifdef HAVE_PYTHON py_state_free(tracker->py); #endif free(tracker); } #define fetch() ((info->is_input ? *info->input_ptr : info->ptr[info->addr]) & info->mask) static void update_element( struct snes_tracker_uniform *uniform, struct snes_tracker_internal *info, unsigned frame_count) { uniform->id = info->id; switch (info->type) { case SSNES_STATE_CAPTURE: uniform->value = fetch(); break; case SSNES_STATE_CAPTURE_PREV: if (info->prev[0] != fetch()) { info->prev[1] = info->prev[0]; info->prev[0] = fetch(); } uniform->value = info->prev[1]; break; case SSNES_STATE_TRANSITION: if (info->old_value != fetch()) { info->old_value = fetch(); info->frame_count = frame_count; } uniform->value = info->frame_count; break; case SSNES_STATE_TRANSITION_COUNT: if (info->old_value != fetch()) { info->old_value = fetch(); info->transition_count++; } uniform->value = info->transition_count; break; case SSNES_STATE_TRANSITION_PREV: if (info->old_value != fetch()) { info->old_value = fetch(); info->frame_count_prev = info->frame_count; info->frame_count = frame_count; } uniform->value = info->frame_count_prev; break; #ifdef HAVE_PYTHON case SSNES_STATE_PYTHON: uniform->value = py_state_get(info->py, info->id, frame_count); break; #endif default: break; } } #undef fetch // Updates 16-bit input in same format as SNES itself. static void update_input(snes_tracker_t *tracker) { if (!driver.input_data) return; static const unsigned buttons[] = { SNES_DEVICE_ID_JOYPAD_R, SNES_DEVICE_ID_JOYPAD_L, SNES_DEVICE_ID_JOYPAD_X, SNES_DEVICE_ID_JOYPAD_A, SNES_DEVICE_ID_JOYPAD_RIGHT, SNES_DEVICE_ID_JOYPAD_LEFT, SNES_DEVICE_ID_JOYPAD_DOWN, SNES_DEVICE_ID_JOYPAD_UP, SNES_DEVICE_ID_JOYPAD_START, SNES_DEVICE_ID_JOYPAD_SELECT, SNES_DEVICE_ID_JOYPAD_Y, SNES_DEVICE_ID_JOYPAD_B }; static const struct snes_keybind *binds[MAX_PLAYERS] = { g_settings.input.binds[0], g_settings.input.binds[1], g_settings.input.binds[2], g_settings.input.binds[3], g_settings.input.binds[4] }; uint16_t state[2] = {0}; for (unsigned i = 4; i < 16; i++) { state[0] |= (driver.input->input_state(driver.input_data, binds, SNES_PORT_1, SNES_DEVICE_JOYPAD, 0, buttons[i - 4]) ? 1 : 0) << i; state[1] |= (driver.input->input_state(driver.input_data, binds, SNES_PORT_2, SNES_DEVICE_JOYPAD, 0, buttons[i - 4]) ? 1 : 0) << i; } for (unsigned i = 0; i < 2; i++) tracker->input_state[i] = state[i]; } unsigned snes_get_uniform(snes_tracker_t *tracker, struct snes_tracker_uniform *uniforms, unsigned elem, unsigned frame_count) { unsigned elems = tracker->info_elem < elem ? tracker->info_elem : elem; update_input(tracker); for (unsigned i = 0; i < elems; i++) update_element(&uniforms[i], &tracker->info[i], frame_count); return elems; }