223 lines
4.5 KiB
C++
223 lines
4.5 KiB
C++
#include "nevs.h"
|
|
|
|
#include "debug.h"
|
|
#include "interpreter_lib.h"
|
|
#include "memory_manager.h"
|
|
#include "platform_compat.h"
|
|
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
|
|
#define NEVS_COUNT 40
|
|
|
|
typedef struct Nevs {
|
|
bool used;
|
|
char name[32];
|
|
Program* program;
|
|
int proc;
|
|
int type;
|
|
int hits;
|
|
bool busy;
|
|
void (*field_38)();
|
|
} Nevs;
|
|
|
|
static Nevs* _nevs_alloc();
|
|
static void _nevs_reset(Nevs* nevs);
|
|
static void _nevs_removeprogramreferences(Program* program);
|
|
static Nevs* _nevs_find(const char* name);
|
|
|
|
// 0x6391C8
|
|
static Nevs* gNevs;
|
|
|
|
// 0x6391CC
|
|
static int gNevsHits;
|
|
|
|
// nevs_alloc
|
|
// 0x488340
|
|
static Nevs* _nevs_alloc()
|
|
{
|
|
if (gNevs == NULL) {
|
|
debugPrint("nevs_alloc(): nevs_initonce() not called!");
|
|
exit(99);
|
|
}
|
|
|
|
for (int index = 0; index < NEVS_COUNT; index++) {
|
|
Nevs* nevs = &(gNevs[index]);
|
|
if (!nevs->used) {
|
|
// NOTE: Uninline.
|
|
_nevs_reset(nevs);
|
|
return nevs;
|
|
}
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
// NOTE: Inlined.
|
|
//
|
|
// 0x488394
|
|
static void _nevs_reset(Nevs* nevs)
|
|
{
|
|
nevs->used = false;
|
|
memset(nevs, 0, sizeof(*nevs));
|
|
}
|
|
|
|
// 0x4883AC
|
|
void _nevs_close()
|
|
{
|
|
if (gNevs != NULL) {
|
|
internal_free_safe(gNevs, __FILE__, __LINE__); // "..\\int\\NEVS.C", 97
|
|
gNevs = NULL;
|
|
}
|
|
}
|
|
|
|
// 0x4883D4
|
|
static void _nevs_removeprogramreferences(Program* program)
|
|
{
|
|
if (gNevs != NULL) {
|
|
for (int i = 0; i < NEVS_COUNT; i++) {
|
|
Nevs* nevs = &(gNevs[i]);
|
|
if (nevs->used && nevs->program == program) {
|
|
// NOTE: Uninline.
|
|
_nevs_reset(nevs);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// nevs_initonce
|
|
// 0x488418
|
|
void _nevs_initonce()
|
|
{
|
|
intLibRegisterProgramDeleteCallback(_nevs_removeprogramreferences);
|
|
|
|
if (gNevs == NULL) {
|
|
gNevs = (Nevs*)internal_calloc_safe(sizeof(Nevs), NEVS_COUNT, __FILE__, __LINE__); // "..\\int\\NEVS.C", 131
|
|
if (gNevs == NULL) {
|
|
debugPrint("nevs_initonce(): out of memory");
|
|
exit(99);
|
|
}
|
|
}
|
|
}
|
|
|
|
// nevs_find
|
|
// 0x48846C
|
|
static Nevs* _nevs_find(const char* name)
|
|
{
|
|
if (gNevs == NULL) {
|
|
debugPrint("nevs_find(): nevs_initonce() not called!");
|
|
exit(99);
|
|
}
|
|
|
|
for (int index = 0; index < NEVS_COUNT; index++) {
|
|
Nevs* nevs = &(gNevs[index]);
|
|
if (nevs->used && compat_stricmp(nevs->name, name) == 0) {
|
|
return nevs;
|
|
}
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
// 0x4884C8
|
|
int _nevs_addevent(const char* name, Program* program, int proc, int type)
|
|
{
|
|
Nevs* nevs = _nevs_find(name);
|
|
if (nevs == NULL) {
|
|
nevs = _nevs_alloc();
|
|
}
|
|
|
|
if (nevs == NULL) {
|
|
return 1;
|
|
}
|
|
|
|
nevs->used = true;
|
|
strcpy(nevs->name, name);
|
|
nevs->program = program;
|
|
nevs->proc = proc;
|
|
nevs->type = type;
|
|
nevs->field_38 = NULL;
|
|
|
|
return 0;
|
|
}
|
|
|
|
// nevs_clearevent
|
|
// 0x48859C
|
|
int _nevs_clearevent(const char* a1)
|
|
{
|
|
debugPrint("nevs_clearevent( '%s');\n", a1);
|
|
|
|
Nevs* nevs = _nevs_find(a1);
|
|
if (nevs != NULL) {
|
|
// NOTE: Uninline.
|
|
_nevs_reset(nevs);
|
|
return 0;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
// nevs_signal
|
|
// 0x48862C
|
|
int _nevs_signal(const char* name)
|
|
{
|
|
debugPrint("nevs_signal( '%s');\n", name);
|
|
|
|
Nevs* nevs = _nevs_find(name);
|
|
if (nevs == NULL) {
|
|
return 1;
|
|
}
|
|
|
|
debugPrint("nep: %p, used = %u, prog = %p, proc = %d", nevs, nevs->used, nevs->program, nevs->proc);
|
|
|
|
if (nevs->used
|
|
&& ((nevs->program != NULL && nevs->proc != 0) || nevs->field_38 != NULL)
|
|
&& !nevs->busy) {
|
|
nevs->hits++;
|
|
gNevsHits++;
|
|
return 0;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
// nevs_update
|
|
// 0x4886AC
|
|
void _nevs_update()
|
|
{
|
|
if (gNevsHits == 0) {
|
|
return;
|
|
}
|
|
|
|
debugPrint("nevs_update(): we have anyhits = %u\n", gNevsHits);
|
|
|
|
gNevsHits = 0;
|
|
|
|
for (int index = 0; index < NEVS_COUNT; index++) {
|
|
Nevs* nevs = &(gNevs[index]);
|
|
if (nevs->used
|
|
&& ((nevs->program != NULL && nevs->proc != 0) || nevs->field_38 != NULL)
|
|
&& !nevs->busy) {
|
|
if (nevs->hits > 0) {
|
|
nevs->busy = true;
|
|
|
|
nevs->hits -= 1;
|
|
gNevsHits += nevs->hits;
|
|
|
|
if (nevs->field_38 == NULL) {
|
|
_executeProc(nevs->program, nevs->proc);
|
|
} else {
|
|
nevs->field_38();
|
|
}
|
|
|
|
nevs->busy = false;
|
|
|
|
if (nevs->type == NEVS_TYPE_EVENT) {
|
|
// NOTE: Uninline.
|
|
_nevs_reset(nevs);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|