ported memory hook codes from gens, then re-implemented memory.registerwrite, and add memory.registerexec; FIXME: CallRegisteredLuaMemHook in nsf.c, I couldn't get how it should be; there might be other issues possibly

This commit is contained in:
gocha 2009-10-11 08:34:10 +00:00
parent d7b02d9549
commit 3ab41b359c
5 changed files with 389 additions and 133 deletions

View File

@ -325,7 +325,7 @@ static DECLFW(BRAML)
{ {
RAM[A]=V; RAM[A]=V;
#ifdef _S9XLUA_H #ifdef _S9XLUA_H
FCEU_LuaWriteInform(); CallRegisteredLuaMemHook(A, 1, V, LUAMEMHOOK_WRITE);
#endif #endif
} }
@ -333,7 +333,7 @@ static DECLFW(BRAMH)
{ {
RAM[A&0x7FF]=V; RAM[A&0x7FF]=V;
#ifdef _S9XLUA_H #ifdef _S9XLUA_H
FCEU_LuaWriteInform(); CallRegisteredLuaMemHook(A&0x7FF, 1, V, LUAMEMHOOK_WRITE);
#endif #endif
} }

View File

@ -10,6 +10,19 @@ enum LuaCallID
}; };
extern void CallRegisteredLuaFunctions(LuaCallID calltype); extern void CallRegisteredLuaFunctions(LuaCallID calltype);
enum LuaMemHookType
{
LUAMEMHOOK_WRITE,
LUAMEMHOOK_READ,
LUAMEMHOOK_EXEC,
LUAMEMHOOK_WRITE_SUB,
LUAMEMHOOK_READ_SUB,
LUAMEMHOOK_EXEC_SUB,
LUAMEMHOOK_COUNT
};
void CallRegisteredLuaMemHook(unsigned int address, int size, unsigned int value, LuaMemHookType hookType);
// Just forward function declarations // Just forward function declarations
//void FCEU_LuaWrite(uint32 addr); //void FCEU_LuaWrite(uint32 addr);
@ -33,7 +46,7 @@ char *FCEU_GetFreezeFilename(int slot);
// Call this before writing into a buffer passed to FCEU_CheatAddRAM(). // Call this before writing into a buffer passed to FCEU_CheatAddRAM().
// (That way, Lua-based memwatch will work as expected for somebody // (That way, Lua-based memwatch will work as expected for somebody
// used to FCEU's memwatch.) // used to FCEU's memwatch.)
void FCEU_LuaWriteInform(); //void FCEU_LuaWriteInform(); // DEADBEEF
#endif #endif

View File

@ -4,6 +4,11 @@
#include <string.h> #include <string.h>
#include <ctype.h> #include <ctype.h>
#include <zlib.h> #include <zlib.h>
#include <assert.h>
#include <vector>
#include <map>
#include <string>
#include <algorithm>
#ifdef __linux #ifdef __linux
#include <unistd.h> #include <unistd.h>
@ -96,8 +101,6 @@ static int skipRerecords = FALSE;
// Used by the registry to find our functions // Used by the registry to find our functions
static const char *frameAdvanceThread = "FCEU.FrameAdvance"; static const char *frameAdvanceThread = "FCEU.FrameAdvance";
static const char *memoryWatchTable = "FCEU.Memory";
static const char *memoryValueTable = "FCEU.MemValues";
static const char *guiCallbackTable = "FCEU.GUI"; static const char *guiCallbackTable = "FCEU.GUI";
// True if there's a thread waiting to run after a run of frame-advance. // True if there's a thread waiting to run after a run of frame-advance.
@ -136,6 +139,8 @@ enum
// over time. The script gets knifed once this reaches zero. // over time. The script gets knifed once this reaches zero.
static int numTries; static int numTries;
// number of registered memory functions (1 per hooked byte)
static unsigned int numMemHooks;
// Look in fceu.h for macros named like JOY_UP to determine the order. // Look in fceu.h for macros named like JOY_UP to determine the order.
static const char *button_mappings[] = { static const char *button_mappings[] = {
@ -148,9 +153,19 @@ static const char* luaCallIDStrings [] =
"CALL_AFTEREMULATION", "CALL_AFTEREMULATION",
"CALL_BEFOREEXIT", "CALL_BEFOREEXIT",
}; };
#ifdef WIN32 // uh... yeah
static const int _makeSureWeHaveTheRightNumberOfStrings [sizeof(luaCallIDStrings)/sizeof(*luaCallIDStrings) == LUACALL_COUNT ? 1 : 0]; static const int _makeSureWeHaveTheRightNumberOfStrings [sizeof(luaCallIDStrings)/sizeof(*luaCallIDStrings) == LUACALL_COUNT ? 1 : 0];
#endif
static const char* luaMemHookTypeStrings [] =
{
"MEMHOOK_WRITE",
"MEMHOOK_READ",
"MEMHOOK_EXEC",
"MEMHOOK_WRITE_SUB",
"MEMHOOK_READ_SUB",
"MEMHOOK_EXEC_SUB",
};
static const int _makeSureWeHaveTheRightNumberOfStrings2 [sizeof(luaMemHookTypeStrings)/sizeof(*luaMemHookTypeStrings) == LUAMEMHOOK_COUNT ? 1 : 0];
/** /**
* Resets emulator speed / pause states after script exit. * Resets emulator speed / pause states after script exit.
@ -213,53 +228,6 @@ int FCEU_LuaFrameSkip() {
return 0; return 0;
} }
/**
* When code determines that a write has occurred
* (not necessarily worth informing Lua), call this.
*
*/
void FCEU_LuaWriteInform() {
if (!L || !luaRunning) return;
// Nuke the stack, just in case.
lua_settop(L,0);
lua_getfield(L, LUA_REGISTRYINDEX, memoryWatchTable);
lua_pushnil(L);
while (lua_next(L, 1) != 0)
{
unsigned int addr = luaL_checkinteger(L, 2);
lua_Integer value;
lua_getfield(L, LUA_REGISTRYINDEX, memoryValueTable);
lua_pushvalue(L, 2);
lua_gettable(L, 4);
value = luaL_checkinteger(L, 5);
if (FCEU_CheatGetByte(addr) != value)
{
// Value changed; update & invoke the Lua callback
lua_pushinteger(L, addr);
lua_pushinteger(L, FCEU_CheatGetByte(addr));
lua_settable(L, 4);
lua_pop(L, 2);
numTries = 1000;
int res = lua_pcall(L, 0, 0, 0);
if (res) {
const char *err = lua_tostring(L, -1);
#ifdef WIN32
//StopSound(); //mbg merge 7/23/08
MessageBox(hAppWnd, err, "Lua Engine", MB_OK);
#else
fprintf(stderr, "Lua error: %s\n", err);
#endif
}
}
lua_settop(L, 2);
}
lua_settop(L, 0);
}
/** /**
* Toggle certain rendering planes * Toggle certain rendering planes
* emu.setrenderingplanes(sprites, background) * emu.setrenderingplanes(sprites, background)
@ -463,6 +431,236 @@ static int memory_readbyterange(lua_State *L) {
} }
void HandleCallbackError(lua_State* L)
{
//if(L->errfunc || L->errorJmp)
// luaL_error(L, "%s", lua_tostring(L,-1));
//else
{
lua_pushnil(L);
lua_setfield(L, LUA_REGISTRYINDEX, guiCallbackTable);
// Error?
#ifdef WIN32
MessageBox( hAppWnd, lua_tostring(L,-1), "Lua run error", MB_OK | MB_ICONSTOP);
#else
fprintf(stderr, "Lua thread bombed out: %s\n", lua_tostring(L,-1));
#endif
FCEU_LuaStop();
}
}
// the purpose of this structure is to provide a way of
// QUICKLY determining whether a memory address range has a hook associated with it,
// with a bias toward fast rejection because the majority of addresses will not be hooked.
// (it must not use any part of Lua or perform any per-script operations,
// otherwise it would definitely be too slow.)
// calculating the regions when a hook is added/removed may be slow,
// but this is an intentional tradeoff to obtain a high speed of checking during later execution
struct TieredRegion
{
template<unsigned int maxGap>
struct Region
{
struct Island
{
unsigned int start;
unsigned int end;
__forceinline bool Contains(unsigned int address, int size) const { return address < end && address+size > start; }
};
std::vector<Island> islands;
void Calculate(const std::vector<unsigned int>& bytes)
{
islands.clear();
unsigned int lastEnd = ~0;
std::vector<unsigned int>::const_iterator iter = bytes.begin();
std::vector<unsigned int>::const_iterator end = bytes.end();
for(; iter != end; ++iter)
{
unsigned int addr = *iter;
if(addr < lastEnd || addr > lastEnd + (long long)maxGap)
{
islands.push_back(Island());
islands.back().start = addr;
}
islands.back().end = addr+1;
lastEnd = addr+1;
}
}
bool Contains(unsigned int address, int size) const
{
std::vector<Island>::const_iterator iter = islands.begin();
std::vector<Island>::const_iterator end = islands.end();
for(; iter != end; ++iter)
if(iter->Contains(address, size))
return true;
return false;
}
};
Region<0xFFFFFFFF> broad;
Region<0x1000> mid;
Region<0> narrow;
void Calculate(std::vector<unsigned int>& bytes)
{
std::sort(bytes.begin(), bytes.end());
broad.Calculate(bytes);
mid.Calculate(bytes);
narrow.Calculate(bytes);
}
TieredRegion()
{
Calculate(std::vector<unsigned int>());
}
__forceinline int NotEmpty()
{
return broad.islands.size();
}
// note: it is illegal to call this if NotEmpty() returns 0
__forceinline bool Contains(unsigned int address, int size)
{
return broad.islands[0].Contains(address,size) &&
mid.Contains(address,size) &&
narrow.Contains(address,size);
}
};
TieredRegion hookedRegions [LUAMEMHOOK_COUNT];
static void CalculateMemHookRegions(LuaMemHookType hookType)
{
std::vector<unsigned int> hookedBytes;
// std::map<int, LuaContextInfo*>::iterator iter = luaContextInfo.begin();
// std::map<int, LuaContextInfo*>::iterator end = luaContextInfo.end();
// while(iter != end)
// {
// LuaContextInfo& info = *iter->second;
if(/*info.*/ numMemHooks)
{
// lua_State* L = info.L;
if(L)
{
lua_settop(L, 0);
lua_getfield(L, LUA_REGISTRYINDEX, luaMemHookTypeStrings[hookType]);
lua_pushnil(L);
while(lua_next(L, -2))
{
if(lua_isfunction(L, -1))
{
unsigned int addr = lua_tointeger(L, -2);
hookedBytes.push_back(addr);
}
lua_pop(L, 1);
}
lua_settop(L, 0);
}
}
// ++iter;
// }
hookedRegions[hookType].Calculate(hookedBytes);
}
static void CallRegisteredLuaMemHook_LuaMatch(unsigned int address, int size, unsigned int value, LuaMemHookType hookType)
{
// std::map<int, LuaContextInfo*>::iterator iter = luaContextInfo.begin();
// std::map<int, LuaContextInfo*>::iterator end = luaContextInfo.end();
// while(iter != end)
// {
// LuaContextInfo& info = *iter->second;
if(/*info.*/ numMemHooks)
{
// lua_State* L = info.L;
if(L/* && !info.panic*/)
{
#ifdef USE_INFO_STACK
infoStack.insert(infoStack.begin(), &info);
struct Scope { ~Scope(){ infoStack.erase(infoStack.begin()); } } scope;
#endif
lua_settop(L, 0);
lua_getfield(L, LUA_REGISTRYINDEX, luaMemHookTypeStrings[hookType]);
for(int i = address; i != address+size; i++)
{
lua_rawgeti(L, -1, i);
if (lua_isfunction(L, -1))
{
bool wasRunning = (luaRunning!=0) /*info.running*/;
luaRunning /*info.running*/ = true;
//RefreshScriptSpeedStatus();
lua_pushinteger(L, address);
lua_pushinteger(L, size);
int errorcode = lua_pcall(L, 2, 0, 0);
luaRunning /*info.running*/ = wasRunning;
//RefreshScriptSpeedStatus();
if (errorcode)
{
HandleCallbackError(L);
//int uid = iter->first;
//HandleCallbackError(L,info,uid,true);
}
break;
}
else
{
lua_pop(L,1);
}
}
lua_settop(L, 0);
}
}
// ++iter;
// }
}
void CallRegisteredLuaMemHook(unsigned int address, int size, unsigned int value, LuaMemHookType hookType)
{
// performance critical! (called VERY frequently)
// I suggest timing a large number of calls to this function in Release if you change anything in here,
// before and after, because even the most innocent change can make it become 30% to 400% slower.
// a good amount to test is: 100000000 calls with no hook set, and another 100000000 with a hook set.
// (on my system that consistently took 200 ms total in the former case and 350 ms total in the latter case)
if(hookedRegions[hookType].NotEmpty())
{
//if((hookType <= LUAMEMHOOK_EXEC) && (address >= 0xE00000))
// address |= 0xFF0000; // account for mirroring of RAM
if(hookedRegions[hookType].Contains(address, size))
CallRegisteredLuaMemHook_LuaMatch(address, size, value, hookType); // something has hooked this specific address
}
}
void CallRegisteredLuaFunctions(LuaCallID calltype)
{
assert((unsigned int)calltype < (unsigned int)LUACALL_COUNT);
const char* idstring = luaCallIDStrings[calltype];
if (!L)
return;
lua_settop(L, 0);
lua_getfield(L, LUA_REGISTRYINDEX, idstring);
int errorcode = 0;
if (lua_isfunction(L, -1))
{
errorcode = lua_pcall(L, 0, 0, 0);
if (errorcode)
HandleCallbackError(L);
}
else
{
lua_pop(L, 1);
}
}
// Not for the signed versions though // Not for the signed versions though
static int memory_readbytesigned(lua_State *L) { static int memory_readbytesigned(lua_State *L) {
signed char c = (signed char) FCEU_CheatGetByte(luaL_checkinteger(L,1)); signed char c = (signed char) FCEU_CheatGetByte(luaL_checkinteger(L,1));
@ -470,37 +668,112 @@ static int memory_readbytesigned(lua_State *L) {
return 1; return 1;
} }
// memory.registerwrite(int address, function func) static int memory_registerHook(lua_State* L, LuaMemHookType hookType, int defaultSize)
// {
// Calls the given function when the indicated memory address is // get first argument: address
// written to. No args are given to the function. The write has already
// occurred, so the new address is readable.
static int memory_registerwrite(lua_State *L) {
// Check args
unsigned int addr = luaL_checkinteger(L,1); unsigned int addr = luaL_checkinteger(L,1);
if (lua_type(L,2) != LUA_TNIL && lua_type(L,2) != LUA_TFUNCTION) if((addr & ~0xFFFFFF) == ~0xFFFFFF)
luaL_error(L, "function or nil expected in arg 2 to memory.register"); addr &= 0xFFFFFF;
// get optional second argument: size
int size = defaultSize;
int funcIdx = 2;
if(lua_isnumber(L,2))
{
size = luaL_checkinteger(L,2);
if(size < 0)
{
size = -size;
addr -= size;
}
funcIdx++;
}
// Check the address range // check last argument: callback function
if (addr > 0xffff) bool clearing = lua_isnil(L,funcIdx);
luaL_error(L, "arg 1 should be between 0x0000 and 0x0fff"); if(!clearing)
luaL_checktype(L, funcIdx, LUA_TFUNCTION);
lua_settop(L,funcIdx);
// Commit it to the registery // get the address-to-callback table for this hook type of the current script
lua_getfield(L, LUA_REGISTRYINDEX, memoryWatchTable); lua_getfield(L, LUA_REGISTRYINDEX, luaMemHookTypeStrings[hookType]);
lua_pushvalue(L,1);
lua_pushvalue(L,2);
lua_settable(L, -3);
lua_getfield(L, LUA_REGISTRYINDEX, memoryValueTable);
lua_pushvalue(L,1);
if (lua_isnil(L,2)) lua_pushnil(L);
else lua_pushinteger(L, FCEU_CheatGetByte(addr));
lua_settable(L, -3);
// count how many callback functions we'll be displacing
int numFuncsAfter = clearing ? 0 : size;
int numFuncsBefore = 0;
for(unsigned int i = addr; i != addr+size; i++)
{
lua_rawgeti(L, -1, i);
if(lua_isfunction(L, -1))
numFuncsBefore++;
lua_pop(L,1);
}
// put the callback function in the address slots
for(unsigned int i = addr; i != addr+size; i++)
{
lua_pushvalue(L, -2);
lua_rawseti(L, -2, i);
}
// adjust the count of active hooks
//LuaContextInfo& info = GetCurrentInfo();
/*info.*/ numMemHooks += numFuncsAfter - numFuncsBefore;
// re-cache regions of hooked memory across all scripts
CalculateMemHookRegions(hookType);
//StopScriptIfFinished(luaStateToUIDMap[L]);
return 0; return 0;
} }
LuaMemHookType MatchHookTypeToCPU(lua_State* L, LuaMemHookType hookType)
{
int cpuID = 0;
int cpunameIndex = 0;
if(lua_type(L,2) == LUA_TSTRING)
cpunameIndex = 2;
else if(lua_type(L,3) == LUA_TSTRING)
cpunameIndex = 3;
if(cpunameIndex)
{
const char* cpuName = lua_tostring(L, cpunameIndex);
if(!stricmp(cpuName, "sub"))
cpuID = 1;
lua_remove(L, cpunameIndex);
}
switch(cpuID)
{
case 0:
return hookType;
case 1:
switch(hookType)
{
case LUAMEMHOOK_WRITE: return LUAMEMHOOK_WRITE_SUB;
case LUAMEMHOOK_READ: return LUAMEMHOOK_READ_SUB;
case LUAMEMHOOK_EXEC: return LUAMEMHOOK_EXEC_SUB;
}
}
return hookType;
}
static int memory_registerwrite(lua_State *L)
{
return memory_registerHook(L, MatchHookTypeToCPU(L,LUAMEMHOOK_WRITE), 1);
}
static int memory_registerread(lua_State *L)
{
return memory_registerHook(L, MatchHookTypeToCPU(L,LUAMEMHOOK_READ), 1);
}
static int memory_registerexec(lua_State *L)
{
return memory_registerHook(L, MatchHookTypeToCPU(L,LUAMEMHOOK_EXEC), 2);
}
//adelikat: table pulled from GENS. credz nitsuja! //adelikat: table pulled from GENS. credz nitsuja!
#ifdef _WIN32 #ifdef _WIN32
@ -2115,8 +2388,12 @@ static const struct luaL_reg memorylib [] = {
// memory hooks // memory hooks
{"registerwrite", memory_registerwrite}, {"registerwrite", memory_registerwrite},
//{"registerread", memory_registerread}, TODO
{"registerexec", memory_registerexec},
// alternate names // alternate names
{"register", memory_registerwrite}, {"register", memory_registerwrite},
{"registerrun", memory_registerexec},
{"registerexecute", memory_registerexec},
{NULL,NULL} {NULL,NULL}
}; };
@ -2207,26 +2484,6 @@ static const struct luaL_reg guilib[] = {
{NULL,NULL} {NULL,NULL}
}; };
void HandleCallbackError(lua_State* L)
{
//if(L->errfunc || L->errorJmp)
// luaL_error(L, "%s", lua_tostring(L,-1));
//else
{
lua_pushnil(L);
lua_setfield(L, LUA_REGISTRYINDEX, guiCallbackTable);
// Error?
#ifdef WIN32
MessageBox( hAppWnd, lua_tostring(L,-1), "Lua run error", MB_OK | MB_ICONSTOP);
#else
fprintf(stderr, "Lua thread bombed out: %s\n", lua_tostring(L,-1));
#endif
FCEU_LuaStop();
}
}
void CallExitFunction() { void CallExitFunction() {
if (!L) if (!L)
return; return;
@ -2246,30 +2503,6 @@ void CallExitFunction() {
HandleCallbackError(L); HandleCallbackError(L);
} }
void CallRegisteredLuaFunctions(LuaCallID calltype)
{
assert((unsigned int)calltype < (unsigned int)LUACALL_COUNT);
const char* idstring = luaCallIDStrings[calltype];
if (!L)
return;
lua_settop(L, 0);
lua_getfield(L, LUA_REGISTRYINDEX, idstring);
int errorcode = 0;
if (lua_isfunction(L, -1))
{
errorcode = lua_pcall(L, 0, 0, 0);
if (errorcode)
HandleCallbackError(L);
}
else
{
lua_pop(L, 1);
}
}
void FCEU_LuaFrameBoundary() { void FCEU_LuaFrameBoundary() {
// printf("Lua Frame\n"); // printf("Lua Frame\n");
@ -2375,10 +2608,12 @@ int FCEU_LoadLuaCode(const char *filename) {
luabitop_validate(L); luabitop_validate(L);
// push arrays for storing hook functions in
for(int i = 0; i < LUAMEMHOOK_COUNT; i++)
{
lua_newtable(L); lua_newtable(L);
lua_setfield(L, LUA_REGISTRYINDEX, memoryWatchTable); lua_setfield(L, LUA_REGISTRYINDEX, luaMemHookTypeStrings[i]);
lua_newtable(L); }
lua_setfield(L, LUA_REGISTRYINDEX, memoryValueTable);
} }
// We make our thread NOW because we want it at the bottom of the stack. // We make our thread NOW because we want it at the bottom of the stack.
@ -2415,6 +2650,7 @@ int FCEU_LoadLuaCode(const char *filename) {
// Initialize settings // Initialize settings
luaRunning = TRUE; luaRunning = TRUE;
skipRerecords = FALSE; skipRerecords = FALSE;
numMemHooks = 0;
wasPaused = FCEUI_EmulationPaused(); wasPaused = FCEUI_EmulationPaused();
if (wasPaused) FCEUI_ToggleEmulationPause(); if (wasPaused) FCEUI_ToggleEmulationPause();
@ -2455,6 +2691,10 @@ void FCEU_LuaStop() {
//execute the user's shutdown callbacks //execute the user's shutdown callbacks
CallExitFunction(); CallExitFunction();
/*info.*/numMemHooks = 0;
for(int i = 0; i < LUAMEMHOOK_COUNT; i++)
CalculateMemHookRegions((LuaMemHookType)i);
//sometimes iup uninitializes com //sometimes iup uninitializes com
//MBG TODO - test whether this is really necessary. i dont think it is //MBG TODO - test whether this is really necessary. i dont think it is
#ifdef WIN32 #ifdef WIN32

View File

@ -442,7 +442,7 @@ static DECLFR(NSF_read)
BANKSET(0x8000+x*4096,NSFHeader.BankSwitch[x]); BANKSET(0x8000+x*4096,NSFHeader.BankSwitch[x]);
} }
#ifdef _S9XLUA_H #ifdef _S9XLUA_H
FCEU_LuaWriteInform(); //CallRegisteredLuaMemHook(A, 1, V, LUAMEMHOOK_WRITE); FIXME
#endif #endif
return (CurrentSong-1); return (CurrentSong-1);
} }

View File

@ -52,7 +52,7 @@ static INLINE void WrMem(unsigned int A, uint8 V)
{ {
BWrite[A](A,V); BWrite[A](A,V);
#ifdef _S9XLUA_H #ifdef _S9XLUA_H
FCEU_LuaWriteInform(); CallRegisteredLuaMemHook(A, 1, V, LUAMEMHOOK_WRITE);
#endif #endif
} }
@ -67,7 +67,7 @@ static INLINE void WrRAM(unsigned int A, uint8 V)
{ {
RAM[A]=V; RAM[A]=V;
#ifdef _S9XLUA_H #ifdef _S9XLUA_H
FCEU_LuaWriteInform(); CallRegisteredLuaMemHook(A, 1, V, LUAMEMHOOK_WRITE);
#endif #endif
} }
@ -82,7 +82,7 @@ void X6502_DMW(uint32 A, uint8 V)
ADDCYC(1); ADDCYC(1);
BWrite[A](A,V); BWrite[A](A,V);
#ifdef _S9XLUA_H #ifdef _S9XLUA_H
FCEU_LuaWriteInform(); CallRegisteredLuaMemHook(A, 1, V, LUAMEMHOOK_WRITE);
#endif #endif
} }
@ -490,6 +490,9 @@ extern int test; test++;
_tcount=0; _tcount=0;
if(MapIRQHook) MapIRQHook(temp); if(MapIRQHook) MapIRQHook(temp);
FCEU_SoundCPUHook(temp); FCEU_SoundCPUHook(temp);
#ifdef _S9XLUA_H
CallRegisteredLuaMemHook(_PC, 1, 0, LUAMEMHOOK_EXEC);
#endif
_PC++; _PC++;
switch(b1) switch(b1)
{ {