2011-05-25 15:15:20 +02:00
|
|
|
/* SSNES - A Super Nintendo Entertainment System (SNES) Emulator frontend for libsnes.
|
|
|
|
* Copyright (C) 2010-2011 - 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 <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "snes_state.h"
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <assert.h>
|
|
|
|
#include "strl.h"
|
2011-06-06 17:44:05 +02:00
|
|
|
#include "general.h"
|
|
|
|
|
|
|
|
#ifdef HAVE_PYTHON
|
|
|
|
#include "py_state/py_state.h"
|
|
|
|
#endif
|
2011-05-25 15:15:20 +02:00
|
|
|
|
|
|
|
struct snes_tracker_internal
|
|
|
|
{
|
|
|
|
char id[64];
|
|
|
|
|
|
|
|
const uint8_t *ptr;
|
2011-06-06 17:44:05 +02:00
|
|
|
#ifdef HAVE_PYTHON
|
|
|
|
py_state_t *py;
|
|
|
|
#endif
|
|
|
|
|
2011-05-25 15:15:20 +02:00
|
|
|
uint32_t addr;
|
2011-05-29 23:58:04 +02:00
|
|
|
uint8_t mask;
|
2011-05-25 15:15:20 +02:00
|
|
|
|
|
|
|
enum snes_tracker_type type;
|
|
|
|
|
|
|
|
uint8_t prev[2];
|
|
|
|
int frame_count;
|
|
|
|
uint8_t old_value;
|
2011-06-06 21:32:10 +02:00
|
|
|
int transition_count;
|
2011-05-25 15:15:20 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
struct snes_tracker
|
|
|
|
{
|
|
|
|
struct snes_tracker_internal *info;
|
|
|
|
unsigned info_elem;
|
2011-06-06 17:44:05 +02:00
|
|
|
|
|
|
|
#ifdef HAVE_PYTHON
|
|
|
|
py_state_t *py;
|
|
|
|
#endif
|
2011-05-25 15:15:20 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
snes_tracker_t* snes_tracker_init(const struct snes_tracker_info *info)
|
|
|
|
{
|
|
|
|
snes_tracker_t *tracker = calloc(1, sizeof(*tracker));
|
|
|
|
if (!tracker)
|
|
|
|
return NULL;
|
|
|
|
|
2011-06-06 17:44:05 +02:00
|
|
|
#ifdef HAVE_PYTHON
|
|
|
|
if (info->script)
|
|
|
|
{
|
2011-06-07 15:33:29 +02:00
|
|
|
tracker->py = py_state_new(info->script, info->script_is_file, info->script_class ? info->script_class : "GameAware");
|
2011-06-06 17:44:05 +02:00
|
|
|
if (!tracker->py)
|
|
|
|
{
|
|
|
|
free(tracker);
|
|
|
|
SSNES_ERR("Failed to init Python script.\n");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2011-05-25 15:15:20 +02:00
|
|
|
tracker->info = 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;
|
2011-05-29 23:58:04 +02:00
|
|
|
tracker->info[i].mask = (info->info[i].mask == 0) ? 0xff : info->info[i].mask;
|
2011-05-25 15:15:20 +02:00
|
|
|
|
2011-06-06 17:44:05 +02:00
|
|
|
#ifdef HAVE_PYTHON
|
|
|
|
if (info->info[i].type == SSNES_STATE_PYTHON)
|
|
|
|
tracker->info[i].py = tracker->py;
|
|
|
|
#endif
|
|
|
|
|
2011-05-25 15:15:20 +02:00
|
|
|
assert(info->wram && info->vram && info->cgram &&
|
|
|
|
info->oam && info->apuram);
|
|
|
|
|
|
|
|
switch (info->info[i].ram_type)
|
|
|
|
{
|
|
|
|
case SSNES_STATE_WRAM:
|
|
|
|
tracker->info[i].ptr = info->wram;
|
|
|
|
break;
|
|
|
|
case SSNES_STATE_APURAM:
|
|
|
|
tracker->info[i].ptr = info->apuram;
|
|
|
|
break;
|
|
|
|
case SSNES_STATE_OAM:
|
|
|
|
tracker->info[i].ptr = info->oam;
|
|
|
|
break;
|
|
|
|
case SSNES_STATE_CGRAM:
|
|
|
|
tracker->info[i].ptr = info->cgram;
|
|
|
|
break;
|
|
|
|
case SSNES_STATE_VRAM:
|
|
|
|
tracker->info[i].ptr = info->vram;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
tracker->info[i].ptr = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return tracker;
|
|
|
|
}
|
|
|
|
|
|
|
|
void snes_tracker_free(snes_tracker_t *tracker)
|
|
|
|
{
|
|
|
|
free(tracker->info);
|
2011-06-06 17:44:05 +02:00
|
|
|
#ifdef HAVE_PYTHON
|
|
|
|
py_state_free(tracker->py);
|
|
|
|
#endif
|
2011-05-25 15:15:20 +02:00
|
|
|
free(tracker);
|
|
|
|
}
|
|
|
|
|
2011-05-30 00:04:26 +02:00
|
|
|
#define fetch() (info->ptr[info->addr] & info->mask)
|
2011-05-29 23:58:04 +02:00
|
|
|
|
2011-05-25 15:15:20 +02:00
|
|
|
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:
|
2011-05-30 00:04:26 +02:00
|
|
|
uniform->value = fetch();
|
2011-05-25 15:15:20 +02:00
|
|
|
break;
|
|
|
|
|
2011-05-26 16:23:11 +02:00
|
|
|
case SSNES_STATE_CAPTURE_PREV:
|
2011-05-30 00:04:26 +02:00
|
|
|
if (info->prev[0] != fetch())
|
2011-05-26 16:23:11 +02:00
|
|
|
{
|
|
|
|
info->prev[1] = info->prev[0];
|
2011-05-30 00:04:26 +02:00
|
|
|
info->prev[0] = fetch();
|
2011-05-26 16:23:11 +02:00
|
|
|
}
|
|
|
|
uniform->value = info->prev[1];
|
|
|
|
break;
|
|
|
|
|
2011-05-25 15:15:20 +02:00
|
|
|
case SSNES_STATE_TRANSITION:
|
2011-05-30 00:04:26 +02:00
|
|
|
if (info->old_value != fetch())
|
2011-05-25 15:15:20 +02:00
|
|
|
{
|
2011-05-30 00:04:26 +02:00
|
|
|
info->old_value = fetch();
|
2011-05-25 15:15:20 +02:00
|
|
|
info->frame_count = frame_count;
|
|
|
|
}
|
2011-05-26 16:23:11 +02:00
|
|
|
uniform->value = info->frame_count;
|
2011-05-25 15:15:20 +02:00
|
|
|
break;
|
|
|
|
|
2011-06-06 21:32:10 +02:00
|
|
|
case SSNES_STATE_TRANSITION_COUNT:
|
|
|
|
if (info->old_value != fetch())
|
|
|
|
{
|
|
|
|
info->old_value = fetch();
|
|
|
|
info->transition_count++;
|
|
|
|
}
|
|
|
|
uniform->value = info->transition_count;
|
|
|
|
break;
|
|
|
|
|
2011-05-26 16:23:11 +02:00
|
|
|
case SSNES_STATE_TRANSITION_PREV:
|
2011-05-30 00:04:26 +02:00
|
|
|
if (info->prev[0] != fetch())
|
2011-05-25 15:15:20 +02:00
|
|
|
{
|
2011-05-30 00:04:26 +02:00
|
|
|
info->old_value = fetch();
|
2011-05-25 15:15:20 +02:00
|
|
|
info->prev[1] = info->prev[0];
|
2011-05-26 16:23:11 +02:00
|
|
|
info->prev[0] = frame_count;
|
|
|
|
info->frame_count = frame_count;
|
2011-05-25 15:15:20 +02:00
|
|
|
}
|
|
|
|
uniform->value = info->prev[1];
|
2011-05-26 16:23:11 +02:00
|
|
|
break;
|
|
|
|
|
2011-06-06 17:44:05 +02:00
|
|
|
#ifdef HAVE_PYTHON
|
|
|
|
case SSNES_STATE_PYTHON:
|
|
|
|
uniform->value = py_state_get(info->py, info->id, frame_count);
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
|
2011-05-25 15:15:20 +02:00
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-05-29 23:58:04 +02:00
|
|
|
#undef fetch
|
|
|
|
|
2011-05-25 15:15:20 +02:00
|
|
|
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;
|
|
|
|
|
|
|
|
for (unsigned i = 0; i < elems; i++)
|
|
|
|
update_element(&uniforms[i], &tracker->info[i], frame_count);
|
|
|
|
|
|
|
|
return elems;
|
|
|
|
}
|
|
|
|
|