LUAInterface should have all its functionality now (excluding input/controller management). Now we need a GUI to see how it works. Added new DSP function: ClearAudioBuffer, which clears the audio buffer for pausing. Currently it doesn't work with DSound.

git-svn-id: https://dolphin-emu.googlecode.com/svn/trunk@4507 8ced0084-cf51-0410-be5f-012b33b47a6e
This commit is contained in:
XTra.KrazzY 2009-11-07 20:01:39 +00:00
parent ebaad032eb
commit a0129e51a3
18 changed files with 255 additions and 185 deletions

View File

@ -82,6 +82,13 @@ void AOSound::Update()
soundSyncEvent.Set(); soundSyncEvent.Set();
} }
void AOSound::Clear()
{
memset(realtimeBuffer, 0, sizeof(realtimeBuffer));
Update();
}
void AOSound::Stop() void AOSound::Stop()
{ {
soundCriticalSection.Enter(); soundCriticalSection.Enter();

View File

@ -52,6 +52,8 @@ public:
virtual void Stop(); virtual void Stop();
virtual void Clear();
static bool isValid() { static bool isValid() {
return true; return true;
} }

View File

@ -170,6 +170,13 @@ void DSound::Update()
soundSyncEvent.Set(); soundSyncEvent.Set();
} }
void DSound::Clear()
{
memset(realtimeBuffer, 0, sizeof(realtimeBuffer));
Update();
}
void DSound::Stop() void DSound::Stop()
{ {
soundCriticalSection.Enter(); soundCriticalSection.Enter();

View File

@ -78,6 +78,7 @@ public:
virtual void SoundLoop(); virtual void SoundLoop();
virtual void SetVolume(int volume); virtual void SetVolume(int volume);
virtual void Stop(); virtual void Stop();
virtual void Clear();
static bool isValid() { return true; } static bool isValid() { return true; }
virtual bool usesMixer() const { return true; } virtual bool usesMixer() const { return true; }
virtual void Update(); virtual void Update();

View File

@ -92,6 +92,13 @@ void OpenALStream::Update()
} }
} }
void OpenALStream::Clear()
{
memset(realtimeBuffer, 0, sizeof(realtimeBuffer));
Update();
}
THREAD_RETURN OpenALStream::ThreadFunc(void* args) THREAD_RETURN OpenALStream::ThreadFunc(void* args)
{ {
(reinterpret_cast<OpenALStream *>(args))->SoundLoop(); (reinterpret_cast<OpenALStream *>(args))->SoundLoop();

View File

@ -50,6 +50,7 @@ public:
virtual bool Start(); virtual bool Start();
virtual void SoundLoop(); virtual void SoundLoop();
virtual void Stop(); virtual void Stop();
virtual void Clear();
static bool isValid() { return true; } static bool isValid() { return true; }
virtual bool usesMixer() const { return true; } virtual bool usesMixer() const { return true; }
virtual void Update(); virtual void Update();

View File

@ -44,6 +44,7 @@ public:
virtual void SoundLoop() {} virtual void SoundLoop() {}
virtual void Stop() {} virtual void Stop() {}
virtual void Update() {} virtual void Update() {}
virtual void Clear() {}
virtual void StartLogAudio(const char *filename) { virtual void StartLogAudio(const char *filename) {
if (! m_logAudio) { if (! m_logAudio) {
m_logAudio = true; m_logAudio = true;

View File

@ -40,6 +40,8 @@ PluginDSP::PluginDSP(const char *_Filename)
(LoadSymbol("DSP_SendAIBuffer")); (LoadSymbol("DSP_SendAIBuffer"));
DSP_StopSoundStream = reinterpret_cast<TDSP_StopSoundStream> DSP_StopSoundStream = reinterpret_cast<TDSP_StopSoundStream>
(LoadSymbol("DSP_StopSoundStream")); (LoadSymbol("DSP_StopSoundStream"));
DSP_ClearAudioBuffer = reinterpret_cast<TDSP_ClearAudioBuffer>
(LoadSymbol("DSP_ClearAudioBuffer"));
if ((DSP_ReadMailboxHigh != 0) && if ((DSP_ReadMailboxHigh != 0) &&
(DSP_ReadMailboxLow != 0) && (DSP_ReadMailboxLow != 0) &&
@ -49,7 +51,8 @@ PluginDSP::PluginDSP(const char *_Filename)
(DSP_WriteControlRegister != 0) && (DSP_WriteControlRegister != 0) &&
(DSP_SendAIBuffer != 0) && (DSP_SendAIBuffer != 0) &&
(DSP_Update != 0) && (DSP_Update != 0) &&
(DSP_StopSoundStream != 0)) (DSP_StopSoundStream != 0) &&
(DSP_ClearAudioBuffer != 0))
validDSP = true; validDSP = true;
} }

View File

@ -30,6 +30,7 @@ typedef unsigned short (__cdecl* TDSP_WriteControlRegister)(unsigned short);
typedef void (__cdecl *TDSP_SendAIBuffer)(unsigned int address, int sample_rate); typedef void (__cdecl *TDSP_SendAIBuffer)(unsigned int address, int sample_rate);
typedef void (__cdecl *TDSP_Update)(int cycles); typedef void (__cdecl *TDSP_Update)(int cycles);
typedef void (__cdecl *TDSP_StopSoundStream)(); typedef void (__cdecl *TDSP_StopSoundStream)();
typedef void (__cdecl *TDSP_ClearAudioBuffer)();
class PluginDSP : public CPlugin class PluginDSP : public CPlugin
{ {
@ -47,6 +48,7 @@ public:
TDSP_SendAIBuffer DSP_SendAIBuffer; TDSP_SendAIBuffer DSP_SendAIBuffer;
TDSP_Update DSP_Update; TDSP_Update DSP_Update;
TDSP_StopSoundStream DSP_StopSoundStream; TDSP_StopSoundStream DSP_StopSoundStream;
TDSP_ClearAudioBuffer DSP_ClearAudioBuffer;
private: private:
bool validDSP; bool validDSP;

View File

@ -575,6 +575,7 @@ void SetState(EState _State)
break; break;
case CORE_PAUSE: case CORE_PAUSE:
CCPU::EnableStepping(true); // Break CCPU::EnableStepping(true); // Break
CPluginManager::GetInstance().GetDSP()->DSP_ClearAudioBuffer();
break; break;
case CORE_RUN: case CORE_RUN:
CCPU::EnableStepping(false); CCPU::EnableStepping(false);

View File

@ -18,13 +18,18 @@
#include <vector> #include <vector>
#include <map> #include <map>
#include <algorithm> #include <algorithm>
#include <assert.h> #include <cassert>
#include "zlib.h"
#include "LuaInterface.h" #include "LuaInterface.h"
#include "Common.h" #include "Common.h"
#include "Core.h"
#include "OnFrame.h" #include "OnFrame.h"
#include "zlib.h" #include "Core.h"
#include "State.h"
#include "ConfigManager.h"
#include "PluginManager.h"
#include "HW/Memmap.h"
#include "Host.h"
extern "C" { extern "C" {
@ -34,7 +39,7 @@ extern "C" {
#include "lstate.h" #include "lstate.h"
}; };
// TODO Count: 22 // TODO Count: 8
// TODO: GUI // TODO: GUI
@ -46,44 +51,38 @@ bool Debug = false;
namespace Lua { namespace Lua {
// TODO: Find proper implementations of these
// the emulator must provide these so that we can implement // the emulator must provide these so that we can implement
// the various functions the user can call from their lua script // the various functions the user can call from their lua script
// (this interface with the emulator needs cleanup, I know) // (this interface with the emulator needs cleanup, I know)
extern int (*Update_Frame)(); //extern unsigned int ReadValueAtHardwareAddress(unsigned int address, unsigned int size);
extern int (*Update_Frame_Fast)(); //extern bool WriteValueAtHardwareAddress(unsigned int address, unsigned int value, unsigned int size, bool hookless=false);
extern unsigned int ReadValueAtHardwareAddress(unsigned int address, unsigned int size); //extern bool WriteValueAtHardwareRAMAddress(unsigned int address, unsigned int value, unsigned int size, bool hookless=false);
extern bool WriteValueAtHardwareAddress(unsigned int address, unsigned int value, unsigned int size, bool hookless=false); //extern bool WriteValueAtHardwareROMAddress(unsigned int address, unsigned int value, unsigned int size);
extern bool WriteValueAtHardwareRAMAddress(unsigned int address, unsigned int value, unsigned int size, bool hookless=false); //extern bool IsHardwareAddressValid(unsigned int address);
extern bool WriteValueAtHardwareROMAddress(unsigned int address, unsigned int value, unsigned int size); //extern bool IsHardwareRAMAddressValid(unsigned int address);
extern bool IsHardwareAddressValid(unsigned int address); //extern bool IsHardwareROMAddressValid(unsigned int address);
extern bool IsHardwareRAMAddressValid(unsigned int address); //extern int Clear_Sound_Buffer(void);
extern bool IsHardwareROMAddressValid(unsigned int address); //extern long long GetCurrentInputCondensed();
extern "C" int disableSound2, disableRamSearchUpdate; //extern long long PeekInputCondensed();
extern "C" int Clear_Sound_Buffer(void); //extern void SetNextInputCondensed(long long input, long long mask);
extern long long GetCurrentInputCondensed(); //extern int Set_Current_State(int Num, bool showOccupiedMessage, bool showEmptyMessage);
extern long long PeekInputCondensed(); //extern int Update_Emulation_One(HWND hWnd);
extern void SetNextInputCondensed(long long input, long long mask); //extern void Update_Emulation_One_Before(HWND hWnd);
extern int Set_Current_State(int Num, bool showOccupiedMessage, bool showEmptyMessage); //extern void Update_Emulation_After_Fast(HWND hWnd);
extern int Update_Emulation_One(HWND hWnd); //extern void Update_Emulation_One_Before_Minimal();
extern void Update_Emulation_One_Before(HWND hWnd); //extern int Update_Frame_Adjusted();
extern void Update_Emulation_After_Fast(HWND hWnd); //extern int Update_Frame_Hook();
extern void Update_Emulation_One_Before_Minimal(); //extern int Update_Frame_Fast_Hook();
extern int Update_Frame_Adjusted(); //extern void Update_Emulation_After_Controlled(HWND hWnd, bool flip);
extern int Update_Frame_Hook(); //extern void Prevent_Next_Frame_Skipping();
extern int Update_Frame_Fast_Hook(); //extern void UpdateLagCount();
extern void Update_Emulation_After_Controlled(HWND hWnd, bool flip); //extern bool Step_emulua_MainLoop(bool allowSleep, bool allowEmulate);
extern void Prevent_Next_Frame_Skipping();
extern void UpdateLagCount(); extern int disableSound2, disableRamSearchUpdate;
extern bool BackgroundInput; extern bool BackgroundInput;
extern bool g_disableStatestateWarnings; extern bool g_disableStatestateWarnings;
extern bool g_onlyCallSavestateCallbacks; extern bool g_onlyCallSavestateCallbacks;
extern bool Step_emulua_MainLoop(bool allowSleep, bool allowEmulate);
extern bool frameadvSkipLagForceDisable; extern bool frameadvSkipLagForceDisable;
extern "C" void Put_Info_NonImmediate(char *Message, int Duration);
extern int Show_Genesis_Screen();
extern void emuluaReplayMovie();
extern bool SkipNextRerecordIncrement; extern bool SkipNextRerecordIncrement;
enum SpeedMode enum SpeedMode
@ -1022,7 +1021,7 @@ DEFINE_LUA_FUNCTION(print, "...")
DEFINE_LUA_FUNCTION(emulua_message, "str") DEFINE_LUA_FUNCTION(emulua_message, "str")
{ {
const char* str = toCString(L); const char* str = toCString(L);
Put_Info_NonImmediate((char*)str, 500); Core::DisplayMessage(str, 2000);
return 0; return 0;
} }
@ -1171,8 +1170,7 @@ void LuaRescueHook(lua_State* L, lua_Debug *dbg)
bool stopworrying = true; bool stopworrying = true;
if(!info.panic) if(!info.panic)
{ {
Clear_Sound_Buffer(); CPluginManager::GetInstance().GetDSP()->DSP_ClearAudioBuffer();
stoprunning = PanicYesNo("A Lua script has been running for quite a while. Maybe it is in an infinite loop.\n\nWould you like to stop the script?\n\n(Yes to stop it now,\n No to keep running and not ask again)"); stoprunning = PanicYesNo("A Lua script has been running for quite a while. Maybe it is in an infinite loop.\n\nWould you like to stop the script?\n\n(Yes to stop it now,\n No to keep running and not ask again)");
} }
@ -1367,8 +1365,6 @@ DEFINE_LUA_FUNCTION(emulua_wait, "")
} }
*/ */
// TODO
/*
DEFINE_LUA_FUNCTION(emulua_frameadvance, "") DEFINE_LUA_FUNCTION(emulua_frameadvance, "")
{ {
if(FailVerifyAtFrameBoundary(L, "emulua.frameadvance", 0,1)) if(FailVerifyAtFrameBoundary(L, "emulua.frameadvance", 0,1))
@ -1377,66 +1373,29 @@ DEFINE_LUA_FUNCTION(emulua_frameadvance, "")
int uid = luaStateToUIDMap[L]; int uid = luaStateToUIDMap[L];
LuaContextInfo& info = GetCurrentInfo(); LuaContextInfo& info = GetCurrentInfo();
if(!info.ranFrameAdvance) info.ranFrameAdvance = !info.ranFrameAdvance;
{
// otherwise we'll never see the first frame of GUI drawing Frame::SetFrameStepping(info.ranFrameAdvance);
if(info.speedMode != SPEEDMODE_MAXIMUM)
Show_Genesis_Screen();
info.ranFrameAdvance = true;
}
switch(info.speedMode)
{
default:
case SPEEDMODE_NORMAL:
while(!Step_emulua_MainLoop(true, true) && !info.panic);
break;
case SPEEDMODE_NOTHROTTLE:
while(!Step_emulua_MainLoop(Core::GetState() == Core::CORE_PAUSE, false) && !info.panic);
if(!(FastForwardKeyDown && (GetActiveWindow()==(HWND)Core::g_CoreStartupParameter.hMainWindow || BackgroundInput)))
emulua_emulateframefastnoskipping(L);
else
emulua_emulateframefast(L);
break;
case SPEEDMODE_TURBO:
while(!Step_emulua_MainLoop(Core::GetState() == Core::CORE_PAUSE, false) && !info.panic);
emulua_emulateframefast(L);
break;
case SPEEDMODE_MAXIMUM:
while(!Step_emulua_MainLoop(Core::GetState() == Core::CORE_PAUSE, false) && !info.panic);
emulua_emulateframeinvisible(L);
break;
}
return 0; return 0;
} }
*/
DEFINE_LUA_FUNCTION(emulua_pause, "") DEFINE_LUA_FUNCTION(emulua_pause, "")
{ {
LuaContextInfo& info = GetCurrentInfo();
Core::SetState(Core::CORE_PAUSE); Core::SetState(Core::CORE_PAUSE);
while(!Step_emulua_MainLoop(true, false) && !info.panic);
// allow the user to not have to manually unpause
// after restarting a script that used emulua.pause()
if(info.panic)
Core::SetState(Core::CORE_RUN);
return 0; return 0;
} }
DEFINE_LUA_FUNCTION(emulua_unpause, "") DEFINE_LUA_FUNCTION(emulua_unpause, "")
{ {
LuaContextInfo& info = GetCurrentInfo();
Core::SetState(Core::CORE_RUN); Core::SetState(Core::CORE_RUN);
return 0; return 0;
} }
DEFINE_LUA_FUNCTION(emulua_redraw, "") DEFINE_LUA_FUNCTION(emulua_redraw, "")
{ {
Show_Genesis_Screen(); Host_UpdateMainFrame();
worry(L,250); worry(L,250);
return 0; return 0;
} }
@ -1445,7 +1404,7 @@ DEFINE_LUA_FUNCTION(emulua_redraw, "")
DEFINE_LUA_FUNCTION(memory_readbyte, "address") DEFINE_LUA_FUNCTION(memory_readbyte, "address")
{ {
int address = (int)luaL_checkinteger(L,1); int address = (int)luaL_checkinteger(L,1);
unsigned char value = (unsigned char)(ReadValueAtHardwareAddress(address, 1) & 0xFF); unsigned char value = Memory::Read_U8(address);
lua_settop(L,0); lua_settop(L,0);
lua_pushinteger(L, value); lua_pushinteger(L, value);
return 1; // we return the number of return values return 1; // we return the number of return values
@ -1453,7 +1412,7 @@ DEFINE_LUA_FUNCTION(memory_readbyte, "address")
DEFINE_LUA_FUNCTION(memory_readbytesigned, "address") DEFINE_LUA_FUNCTION(memory_readbytesigned, "address")
{ {
int address = (int)luaL_checkinteger(L,1); int address = (int)luaL_checkinteger(L,1);
signed char value = (signed char)(ReadValueAtHardwareAddress(address, 1) & 0xFF); signed char value = (signed char)(Memory::Read_U8(address));
lua_settop(L,0); lua_settop(L,0);
lua_pushinteger(L, value); lua_pushinteger(L, value);
return 1; return 1;
@ -1461,7 +1420,7 @@ DEFINE_LUA_FUNCTION(memory_readbytesigned, "address")
DEFINE_LUA_FUNCTION(memory_readword, "address") DEFINE_LUA_FUNCTION(memory_readword, "address")
{ {
int address = (int)luaL_checkinteger(L,1); int address = (int)luaL_checkinteger(L,1);
unsigned short value = (unsigned short)(ReadValueAtHardwareAddress(address, 2) & 0xFFFF); unsigned short value = Memory::Read_U16(address);
lua_settop(L,0); lua_settop(L,0);
lua_pushinteger(L, value); lua_pushinteger(L, value);
return 1; return 1;
@ -1469,7 +1428,7 @@ DEFINE_LUA_FUNCTION(memory_readword, "address")
DEFINE_LUA_FUNCTION(memory_readwordsigned, "address") DEFINE_LUA_FUNCTION(memory_readwordsigned, "address")
{ {
int address = (int)luaL_checkinteger(L,1); int address = (int)luaL_checkinteger(L,1);
signed short value = (signed short)(ReadValueAtHardwareAddress(address, 2) & 0xFFFF); signed short value = (signed short)(Memory::Read_U16(address));
lua_settop(L,0); lua_settop(L,0);
lua_pushinteger(L, value); lua_pushinteger(L, value);
return 1; return 1;
@ -1477,7 +1436,7 @@ DEFINE_LUA_FUNCTION(memory_readwordsigned, "address")
DEFINE_LUA_FUNCTION(memory_readdword, "address") DEFINE_LUA_FUNCTION(memory_readdword, "address")
{ {
int address = (int)luaL_checkinteger(L,1); int address = (int)luaL_checkinteger(L,1);
unsigned long value = (unsigned long)(ReadValueAtHardwareAddress(address, 4)); unsigned long value = Memory::Read_U32(address);
lua_settop(L,0); lua_settop(L,0);
lua_pushinteger(L, value); lua_pushinteger(L, value);
return 1; return 1;
@ -1485,7 +1444,23 @@ DEFINE_LUA_FUNCTION(memory_readdword, "address")
DEFINE_LUA_FUNCTION(memory_readdwordsigned, "address") DEFINE_LUA_FUNCTION(memory_readdwordsigned, "address")
{ {
int address = (int)luaL_checkinteger(L,1); int address = (int)luaL_checkinteger(L,1);
signed long value = (signed long)(ReadValueAtHardwareAddress(address, 4)); signed long value = (signed long)(Memory::Read_U32(address));
lua_settop(L,0);
lua_pushinteger(L, value);
return 1;
}
DEFINE_LUA_FUNCTION(memory_readqword, "address")
{
int address = (int)luaL_checkinteger(L,1);
unsigned long long value = Memory::Read_U64(address);
lua_settop(L,0);
lua_pushinteger(L, value);
return 1;
}
DEFINE_LUA_FUNCTION(memory_readqwordsigned, "address")
{
int address = (int)luaL_checkinteger(L,1);
signed long long value = (signed long long)(Memory::Read_U64(address));
lua_settop(L,0); lua_settop(L,0);
lua_pushinteger(L, value); lua_pushinteger(L, value);
return 1; return 1;
@ -1495,21 +1470,28 @@ DEFINE_LUA_FUNCTION(memory_writebyte, "address,value")
{ {
int address = (int)luaL_checkinteger(L,1); int address = (int)luaL_checkinteger(L,1);
unsigned char value = (unsigned char)(luaL_checkinteger(L,2) & 0xFF); unsigned char value = (unsigned char)(luaL_checkinteger(L,2) & 0xFF);
WriteValueAtHardwareRAMAddress(address, value, 1); Memory::Write_U8(value, address);
return 0; return 0;
} }
DEFINE_LUA_FUNCTION(memory_writeword, "address,value") DEFINE_LUA_FUNCTION(memory_writeword, "address,value")
{ {
int address = (int)luaL_checkinteger(L,1); int address = (int)luaL_checkinteger(L,1);
unsigned short value = (unsigned short)(luaL_checkinteger(L,2) & 0xFFFF); unsigned short value = (unsigned short)(luaL_checkinteger(L,2) & 0xFFFF);
WriteValueAtHardwareRAMAddress(address, value, 2); Memory::Write_U16(value, address);
return 0; return 0;
} }
DEFINE_LUA_FUNCTION(memory_writedword, "address,value") DEFINE_LUA_FUNCTION(memory_writedword, "address,value")
{ {
int address = (int)luaL_checkinteger(L,1); int address = (int)luaL_checkinteger(L,1);
unsigned long value = (unsigned long)(luaL_checkinteger(L,2)); unsigned long value = (unsigned long)(luaL_checkinteger(L,2));
WriteValueAtHardwareRAMAddress(address, value, 4); Memory::Write_U32(value, address);
return 0;
}
DEFINE_LUA_FUNCTION(memory_writeqword, "address,value")
{
int address = (int)luaL_checkinteger(L,1);
unsigned long value = (unsigned long)(luaL_checkinteger(L,2));
Memory::Write_U64(value, address);
return 0; return 0;
} }
@ -1530,9 +1512,9 @@ DEFINE_LUA_FUNCTION(memory_readbyterange, "address,length")
// put all the values into the (1-based) array // put all the values into the (1-based) array
for(int a = address, n = 1; n <= length; a++, n++) for(int a = address, n = 1; n <= length; a++, n++)
{ {
if(IsHardwareAddressValid(a)) if(Memory::IsRAMAddress(a))
{ {
unsigned char value = (unsigned char)(ReadValueAtHardwareAddress(a, 1) & 0xFF); unsigned char value = Memory::Read_U8(a);
lua_pushinteger(L, value); lua_pushinteger(L, value);
lua_rawseti(L, -2, n); lua_rawseti(L, -2, n);
} }
@ -1546,7 +1528,7 @@ DEFINE_LUA_FUNCTION(memory_isvalid, "address")
{ {
int address = (int)luaL_checkinteger(L,1); int address = (int)luaL_checkinteger(L,1);
lua_settop(L,0); lua_settop(L,0);
lua_pushboolean(L, IsHardwareAddressValid(address)); lua_pushboolean(L, Memory::IsRAMAddress(address));
return 1; return 1;
} }
@ -1682,8 +1664,7 @@ DEFINE_LUA_FUNCTION(memory_setregister, "cpu_dot_registername_string,value")
return 0; return 0;
} }
// TODO: (also not implemented in DeSmuME) // Creates a savestate object
/*
DEFINE_LUA_FUNCTION(state_create, "[location]") DEFINE_LUA_FUNCTION(state_create, "[location]")
{ {
if(lua_isnumber(L,1)) if(lua_isnumber(L,1))
@ -1694,11 +1675,7 @@ DEFINE_LUA_FUNCTION(state_create, "[location]")
return 1; return 1;
} }
int len = GENESIS_STATE_LENGTH; size_t len = State_GetSize();
if (SegaCD_Started) len += SEGACD_LENGTH_EX;
if (_32X_Started) len += G32X_LENGTH_EX;
if (!((Genesis_Started)||(SegaCD_Started)||(_32X_Started)))
len += max(SEGACD_LENGTH_EX, G32X_LENGTH_EX);
// allocate the in-memory/anonymous savestate // allocate the in-memory/anonymous savestate
unsigned char* stateBuffer = (unsigned char*)lua_newuserdata(L, len + 16); // 16 is for performance alignment reasons unsigned char* stateBuffer = (unsigned char*)lua_newuserdata(L, len + 16); // 16 is for performance alignment reasons
@ -1706,7 +1683,7 @@ DEFINE_LUA_FUNCTION(state_create, "[location]")
return 1; return 1;
} }
*/
// savestate.save(location [, option]) // savestate.save(location [, option])
// saves the current emulation state to the given location // saves the current emulation state to the given location
// you can pass in either a savestate file number (an integer), // you can pass in either a savestate file number (an integer),
@ -1715,8 +1692,6 @@ DEFINE_LUA_FUNCTION(state_create, "[location]")
// if option is "scriptdataonly" then the state will not actually be saved, but any save callbacks will still get called and their results will be saved (see savestate.registerload()/savestate.registersave()) // if option is "scriptdataonly" then the state will not actually be saved, but any save callbacks will still get called and their results will be saved (see savestate.registerload()/savestate.registersave())
DEFINE_LUA_FUNCTION(state_save, "location[,option]") DEFINE_LUA_FUNCTION(state_save, "location[,option]")
{ {
// TODO
/*
const char* option = (lua_type(L,2) == LUA_TSTRING) ? lua_tostring(L,2) : NULL; const char* option = (lua_type(L,2) == LUA_TSTRING) ? lua_tostring(L,2) : NULL;
if(option) if(option)
{ {
@ -1736,24 +1711,21 @@ DEFINE_LUA_FUNCTION(state_save, "location[,option]")
case LUA_TNUMBER: // numbered save file case LUA_TNUMBER: // numbered save file
default: default:
{ {
int stateNumber = (int)luaL_checkinteger(L,1); State_Save((int)luaL_checkinteger(L,1));
Set_Current_State(stateNumber, false,false); return 0;
char Name [1024] = {0}; }
Get_State_File_Name(Name);
Save_State(Name);
} return 0;
case LUA_TUSERDATA: // in-memory save slot case LUA_TUSERDATA: // in-memory save slot
{ {
unsigned char* stateBuffer = (unsigned char*)lua_touserdata(L,1); unsigned char* stateBuffer = (unsigned char*)lua_touserdata(L,1);
if(stateBuffer) if(stateBuffer)
{ {
stateBuffer += ((16 - (int)stateBuffer) & 15); // for performance alignment reasons stateBuffer += ((16 - (int)stateBuffer) & 15); // for performance alignment reasons
Save_State_To_Buffer(stateBuffer); State_SaveToBuffer(&stateBuffer);
} }
} return 0;
}
*/
return 0; return 0;
}
}
} }
// savestate.load(location [, option]) // savestate.load(location [, option])
@ -1788,14 +1760,11 @@ DEFINE_LUA_FUNCTION(state_load, "location[,option]")
LuaContextInfo& info = GetCurrentInfo(); LuaContextInfo& info = GetCurrentInfo();
if(info.rerecordCountingDisabled) if(info.rerecordCountingDisabled)
SkipNextRerecordIncrement = true; SkipNextRerecordIncrement = true;
int stateNumber = (int)luaL_checkinteger(L,1); State_Load((int)luaL_checkinteger(L,1));
Set_Current_State(stateNumber, false,!g_disableStatestateWarnings);
char Name [1024] = {0}; return 0;
}
//TODO
//Get_State_File_Name(Name);
//Load_State(Name);
} return 0;
case LUA_TUSERDATA: // in-memory save slot case LUA_TUSERDATA: // in-memory save slot
{ {
unsigned char* stateBuffer = (unsigned char*)lua_touserdata(L,1); unsigned char* stateBuffer = (unsigned char*)lua_touserdata(L,1);
@ -1803,11 +1772,12 @@ DEFINE_LUA_FUNCTION(state_load, "location[,option]")
{ {
stateBuffer += ((16 - (int)stateBuffer) & 15); // for performance alignment reasons stateBuffer += ((16 - (int)stateBuffer) & 15); // for performance alignment reasons
if(stateBuffer[0]) if(stateBuffer[0])
;// Load_State_From_Buffer(stateBuffer); TODO State_LoadFromBuffer(&stateBuffer);
else // the first byte of a valid savestate is never 0 else // the first byte of a valid savestate is never 0
luaL_error(L, "attempted to load an anonymous savestate before saving it"); luaL_error(L, "attempted to load an anonymous savestate before saving it");
} }
} return 0; return 0;
}
} }
} }
@ -1832,10 +1802,11 @@ DEFINE_LUA_FUNCTION(state_loadscriptdata, "location")
case LUA_TNUMBER: // numbered save file case LUA_TNUMBER: // numbered save file
default: default:
{ {
// TODO
int stateNumber = (int)luaL_checkinteger(L,1); int stateNumber = (int)luaL_checkinteger(L,1);
Set_Current_State(stateNumber, false,false); //Set_Current_State(stateNumber, false,false);
char Name [1024] = {0}; char Name [1024] = {0};
//Get_State_File_Name(Name); TODO //Get_State_File_Name(Name);
{ {
LuaSaveData saveData; LuaSaveData saveData;
@ -1876,6 +1847,7 @@ DEFINE_LUA_FUNCTION(state_savescriptdata, "location")
// TODO: Convert to Dolphin? // TODO: Convert to Dolphin?
/*
static const struct ButtonDesc static const struct ButtonDesc
{ {
unsigned short controllerNum; unsigned short controllerNum;
@ -2095,7 +2067,7 @@ DEFINE_LUA_FUNCTION(joy_peekup, "[controller=1]")
{ {
return joy_peek_internal(L, true, false); return joy_peek_internal(L, true, false);
} }
*/
static const struct ColorMapping static const struct ColorMapping
{ {
@ -2744,29 +2716,63 @@ DEFINE_LUA_FUNCTION(emu_openscript, "filename")
return 0; return 0;
} }
// TODO
/*
DEFINE_LUA_FUNCTION(emulua_loadrom, "filename") DEFINE_LUA_FUNCTION(emulua_loadrom, "filename")
{ {
struct Temp { Temp() {EnableStopAllLuaScripts(false);} ~Temp() {EnableStopAllLuaScripts(true);}} dontStopScriptsHere; struct Temp { Temp() {EnableStopAllLuaScripts(false);} ~Temp() {EnableStopAllLuaScripts(true);}} dontStopScriptsHere;
const char* filename = lua_isstring(L,1) ? lua_tostring(L,1) : NULL; const char* filename = lua_isstring(L,1) ? lua_tostring(L,1) : NULL;
char curScriptDir[1024]; GetCurrentScriptDir(curScriptDir, 1024); char curScriptDir[1024]; GetCurrentScriptDir(curScriptDir, 1024);
filename = MakeRomPathAbsolute(filename, curScriptDir);
int result = emuluaLoadRom(filename); if(Core::GetState() != Core::CORE_UNINITIALIZED)
if(result <= 0) Core::Stop();
luaL_error(L, "Failed to load ROM \"%s\": %s", filename, result ? "invalid or unsupported" : "cancelled or not found");
SCoreStartupParameter& StartUp = SConfig::GetInstance().m_LocalCoreStartupParameter;
StartUp.m_BootType = SCoreStartupParameter::BOOT_ISO;
StartUp.m_strFilename = filename;
SConfig::GetInstance().m_LastFilename = filename;
StartUp.bRunCompareClient = false;
StartUp.bRunCompareServer = false;
// If for example the ISO file is bad we return here
if (!StartUp.AutoSetup(SCoreStartupParameter::BOOT_DEFAULT)) return 1;
// Load game specific settings
IniFile game_ini;
std::string unique_id = StartUp.GetUniqueID();
StartUp.m_strGameIni = FULL_GAMECONFIG_DIR + unique_id + ".ini";
if (unique_id.size() == 6 && game_ini.Load(StartUp.m_strGameIni.c_str()))
{
// General settings
game_ini.Get("Core", "CPUOnThread", &StartUp.bCPUThread, StartUp.bCPUThread);
game_ini.Get("Core", "SkipIdle", &StartUp.bSkipIdle, StartUp.bSkipIdle);
game_ini.Get("Core", "OptimizeQuantizers", &StartUp.bOptimizeQuantizers, StartUp.bOptimizeQuantizers);
game_ini.Get("Core", "EnableFPRF", &StartUp.bEnableFPRF, StartUp.bEnableFPRF);
game_ini.Get("Core", "TLBHack", &StartUp.iTLBHack, StartUp.iTLBHack);
// Wii settings
if (StartUp.bWii)
{
// Flush possible changes to SYSCONF to file
SConfig::GetInstance().m_SYSCONF->Save();
}
}
if (!Core::Init())
return 1;
Core::SetState(Core::CORE_RUN);
CallRegisteredLuaFunctions(LUACALL_ONSTART); CallRegisteredLuaFunctions(LUACALL_ONSTART);
return 0; return 0;
} }
*/
DEFINE_LUA_FUNCTION(emulua_getframecount, "") DEFINE_LUA_FUNCTION(emulua_getframecount, "")
{ {
lua_pushinteger(L, Frame::g_frameCounter); lua_pushinteger(L, (lua_Integer)Frame::g_frameCounter);
return 1; return 1;
} }
DEFINE_LUA_FUNCTION(emulua_getlagcount, "") DEFINE_LUA_FUNCTION(emulua_getlagcount, "")
{ {
lua_pushinteger(L, Frame::g_lagCounter); lua_pushinteger(L, (lua_Integer)Frame::g_lagCounter);
return 1; return 1;
} }
DEFINE_LUA_FUNCTION(emulua_lagged, "") DEFINE_LUA_FUNCTION(emulua_lagged, "")
@ -2788,7 +2794,7 @@ DEFINE_LUA_FUNCTION(emulua_atframeboundary, "")
DEFINE_LUA_FUNCTION(movie_getlength, "") DEFINE_LUA_FUNCTION(movie_getlength, "")
{ {
lua_pushinteger(L, Frame::g_recordfd ? Frame::g_frameCounter : 0); lua_pushinteger(L, Frame::g_recordfd ? (lua_Integer)Frame::g_frameCounter : 0);
return 1; return 1;
} }
DEFINE_LUA_FUNCTION(movie_isactive, "") DEFINE_LUA_FUNCTION(movie_isactive, "")
@ -2798,13 +2804,12 @@ DEFINE_LUA_FUNCTION(movie_isactive, "")
} }
DEFINE_LUA_FUNCTION(movie_rerecordcount, "") DEFINE_LUA_FUNCTION(movie_rerecordcount, "")
{ {
// TODO lua_pushinteger(L, Frame::g_recordfd ? (lua_Integer)Frame::g_numRerecords : 0);
lua_pushinteger(L, /*Frame::g_recordfd ? MainMovie.NbRerecords :*/ 0);
return 1; return 1;
} }
DEFINE_LUA_FUNCTION(movie_setrerecordcount, "") DEFINE_LUA_FUNCTION(movie_setrerecordcount, "")
{ {
//TODO: MainMovie.NbRerecords = (int)luaL_checkinteger(L, 1); Frame::g_numRerecords = (int)luaL_checkinteger(L, 1);
return 0; return 0;
} }
DEFINE_LUA_FUNCTION(emulua_rerecordcounting, "[enabled]") DEFINE_LUA_FUNCTION(emulua_rerecordcounting, "[enabled]")
@ -2825,20 +2830,14 @@ DEFINE_LUA_FUNCTION(emulua_rerecordcounting, "[enabled]")
} }
DEFINE_LUA_FUNCTION(movie_getreadonly, "") DEFINE_LUA_FUNCTION(movie_getreadonly, "")
{ {
//TODO // We don't support read-only rerecords
lua_pushboolean(L, /*MainMovie.File ? MainMovie.ReadOnly :*/ 0); lua_pushboolean(L, 0);
return 1; return 1;
} }
DEFINE_LUA_FUNCTION(movie_setreadonly, "readonly") DEFINE_LUA_FUNCTION(movie_setreadonly, "readonly")
{ {
//TODO // We don't support read-only rerecords
/* luaL_error(L, "movie.setreadonly failed: No such feature");
int readonly = lua_toboolean(L,1) ? 1 : 0;
if(MainMovie.ReadOnly != 2)
MainMovie.ReadOnly = readonly;
else if(!readonly)
luaL_error(L, "movie.setreadonly failed: write permission denied");
*/
return 0; return 0;
} }
DEFINE_LUA_FUNCTION(movie_isrecording, "") DEFINE_LUA_FUNCTION(movie_isrecording, "")
@ -2872,7 +2871,7 @@ DEFINE_LUA_FUNCTION(movie_getmode, "")
} }
DEFINE_LUA_FUNCTION(movie_getname, "") DEFINE_LUA_FUNCTION(movie_getname, "")
{ {
lua_pushstring(L, "TODO" /*MainMovie.FileName*/); lua_pushstring(L, Frame::g_recordFile.c_str());
return 1; return 1;
} }
// movie.play() -- plays a movie of the user's choice // movie.play() -- plays a movie of the user's choice
@ -2880,32 +2879,30 @@ DEFINE_LUA_FUNCTION(movie_getname, "")
// throws an error (with a description) if for whatever reason the movie couldn't be played // throws an error (with a description) if for whatever reason the movie couldn't be played
DEFINE_LUA_FUNCTION(movie_play, "[filename]") DEFINE_LUA_FUNCTION(movie_play, "[filename]")
{ {
// TODO
/*
const char* filename = lua_isstring(L,1) ? lua_tostring(L,1) : NULL; const char* filename = lua_isstring(L,1) ? lua_tostring(L,1) : NULL;
const char* errorMsg = emuluaPlayMovie(filename, true); if(!Frame::PlayInput(filename))
if(errorMsg) luaL_error(L, "ERROR playing rereecording");
luaL_error(L, errorMsg);*/
return 0; return 0;
} }
DEFINE_LUA_FUNCTION(movie_replay, "") DEFINE_LUA_FUNCTION(movie_replay, "")
{ {
if(Frame::g_recordfd) if(Frame::g_recordfd) {
emuluaReplayMovie(); std::string filename = Frame::g_recordFile;
else Frame::EndPlayInput();
Frame::PlayInput(filename.c_str());
} else
luaL_error(L, "it is invalid to call movie.replay when no movie open."); luaL_error(L, "it is invalid to call movie.replay when no movie open.");
return 0; return 0;
} }
DEFINE_LUA_FUNCTION(movie_close, "") DEFINE_LUA_FUNCTION(movie_close, "")
{ {
// TODO Frame::EndPlayInput();
//CloseMovieFile(&MainMovie);
return 0; return 0;
} }
DEFINE_LUA_FUNCTION(sound_clear, "") DEFINE_LUA_FUNCTION(sound_clear, "")
{ {
Clear_Sound_Buffer(); CPluginManager::GetInstance().GetDSP()->DSP_ClearAudioBuffer();
return 0; return 0;
} }
@ -3006,9 +3003,9 @@ int dontworry(LuaContextInfo& info)
static const struct luaL_reg emulualib [] = static const struct luaL_reg emulualib [] =
{ {
// {"frameadvance", emulua_frameadvance}, {"frameadvance", emulua_frameadvance},
// {"speedmode", emulua_speedmode}, // {"speedmode", emulua_speedmode},
// {"wait", emulua_wait}, {"wait", emulua_wait},
{"pause", emulua_pause}, {"pause", emulua_pause},
{"unpause", emulua_unpause}, {"unpause", emulua_unpause},
// {"emulateframe", emulua_emulateframe}, // {"emulateframe", emulua_emulateframe},
@ -3028,9 +3025,9 @@ static const struct luaL_reg emulualib [] =
{"message", emulua_message}, {"message", emulua_message},
{"print", print}, // sure, why not {"print", print}, // sure, why not
// {"openscript", emulua_openscript}, // {"openscript", emulua_openscript},
// {"loadrom", emulua_loadrom}, {"loadrom", emulua_loadrom},
// alternative names // alternative names
// {"openrom", emulua_loadrom}, {"openrom", emulua_loadrom},
{NULL, NULL} {NULL, NULL}
}; };
static const struct luaL_reg guilib [] = static const struct luaL_reg guilib [] =
@ -3064,11 +3061,11 @@ static const struct luaL_reg guilib [] =
}; };
static const struct luaL_reg statelib [] = static const struct luaL_reg statelib [] =
{ {
// {"create", state_create}, {"create", state_create},
// {"save", state_save}, {"save", state_save},
// {"load", state_load}, {"load", state_load},
// {"loadscriptdata", state_loadscriptdata}, {"loadscriptdata", state_loadscriptdata},
// {"savescriptdata", state_savescriptdata}, {"savescriptdata", state_savescriptdata},
{"registersave", state_registersave}, {"registersave", state_registersave},
{"registerload", state_registerload}, {"registerload", state_registerload},
{NULL, NULL} {NULL, NULL}
@ -3098,8 +3095,11 @@ static const struct luaL_reg memorylib [] =
{"readlong", memory_readdword}, {"readlong", memory_readdword},
{"readlongunsigned", memory_readdword}, {"readlongunsigned", memory_readdword},
{"readlongsigned", memory_readdwordsigned}, {"readlongsigned", memory_readdwordsigned},
{"readlonglongunsigned", memory_readqword},
{"readlonglongsigned", memory_readqwordsigned},
{"writeshort", memory_writeword}, {"writeshort", memory_writeword},
{"writelong", memory_writedword}, {"writelong", memory_writedword},
{"writelonglong", memory_writeqword},
// memory hooks // memory hooks
{"registerwrite", memory_registerwrite}, {"registerwrite", memory_registerwrite},
@ -3114,18 +3114,18 @@ static const struct luaL_reg memorylib [] =
}; };
static const struct luaL_reg joylib [] = static const struct luaL_reg joylib [] =
{ {
{"get", joy_get}, //{"get", joy_get},
{"getdown", joy_getdown}, //{"getdown", joy_getdown},
{"getup", joy_getup}, //{"getup", joy_getup},
{"peek", joy_peek}, //{"peek", joy_peek},
{"peekdown", joy_peekdown}, //{"peekdown", joy_peekdown},
{"peekup", joy_peekup}, //{"peekup", joy_peekup},
{"set", joy_set}, //{"set", joy_set},
// alternative names //// alternative names
{"read", joy_get}, //{"read", joy_get},
{"write", joy_set}, //{"write", joy_set},
{"readdown", joy_getdown}, //{"readdown", joy_getdown},
{"readup", joy_getup}, //{"readup", joy_getup},
{NULL, NULL} {NULL, NULL}
}; };
static const struct luaL_reg inputlib [] = static const struct luaL_reg inputlib [] =
@ -3520,7 +3520,6 @@ void RunLuaScriptFile(int uid, const char* filenameCStr)
} }
else else
{ {
Show_Genesis_Screen();
StopScriptIfFinished(uid, true); StopScriptIfFinished(uid, true);
} }
} while(info.restart); } while(info.restart);

View File

@ -41,6 +41,9 @@ FILE *g_recordfd = NULL;
u64 g_frameCounter = 0, g_lagCounter = 0; u64 g_frameCounter = 0, g_lagCounter = 0;
bool g_bPolled = false; bool g_bPolled = false;
int g_numRerecords = 0;
std::string g_recordFile;
void FrameUpdate() { void FrameUpdate() {
g_frameCounter++; g_frameCounter++;
@ -205,6 +208,8 @@ bool BeginRecordingInput(const char *filename, int controllers)
g_playMode = MODE_RECORDING; g_playMode = MODE_RECORDING;
g_recordFile = filename;
return true; return true;
} }
@ -313,6 +318,8 @@ bool PlayInput(const char *filename)
g_numPads = header.numControllers; g_numPads = header.numControllers;
g_padStates = new ControllerState[g_numPads]; g_padStates = new ControllerState[g_numPads];
g_numRerecords = header.numRerecords;
g_recordFile = filename;
g_playMode = MODE_PLAYING; g_playMode = MODE_PLAYING;

View File

@ -21,6 +21,8 @@
#include "Common.h" #include "Common.h"
#include "pluginspecs_pad.h" #include "pluginspecs_pad.h"
#include <string>
// Per-(video )Frame actions // Per-(video )Frame actions
namespace Frame { namespace Frame {
@ -52,9 +54,12 @@ extern PlayMode g_playMode;
extern int g_framesToSkip, g_frameSkipCounter, g_numPads; extern int g_framesToSkip, g_frameSkipCounter, g_numPads;
extern ControllerState *g_padStates; extern ControllerState *g_padStates;
extern FILE *g_recordfd; extern FILE *g_recordfd;
extern std::string g_recordFile;
extern u64 g_frameCounter, g_lagCounter; extern u64 g_frameCounter, g_lagCounter;
extern int g_numRerecords;
typedef struct { typedef struct {
u8 filetype[4]; // Unique Identifier (always "DTM"0x1A) u8 filetype[4]; // Unique Identifier (always "DTM"0x1A)

View File

@ -36,6 +36,7 @@
#include "minilzo.h" #include "minilzo.h"
// TODO: Move to namespace
// TODO: Investigate a memory leak on save/load state // TODO: Investigate a memory leak on save/load state
@ -453,3 +454,12 @@ void State_UndoSaveState()
{ {
State_LoadAs(FULL_STATESAVES_DIR "lastState.sav"); State_LoadAs(FULL_STATESAVES_DIR "lastState.sav");
} }
size_t State_GetSize()
{
// Measure the size of the buffer.
u8 *ptr = 0;
PointerWrap p(&ptr, PointerWrap::MODE_MEASURE);
DoState(p);
return (size_t)ptr;
}

View File

@ -46,6 +46,8 @@ void State_LoadLastSaved();
void State_UndoSaveState(); void State_UndoSaveState();
void State_UndoLoadState(); void State_UndoLoadState();
size_t State_GetSize();
typedef struct typedef struct
{ {

View File

@ -104,5 +104,10 @@ EXPORT void CALL DSP_SendAIBuffer(unsigned int address, int sample_rate);
// Purpose: Stops audio playback. Must be called before Shutdown(). // Purpose: Stops audio playback. Must be called before Shutdown().
EXPORT void CALL DSP_StopSoundStream(); EXPORT void CALL DSP_StopSoundStream();
// __________________________________________________________________________________________________
// Function: DSP_ClearAudioBuffer
// Purpose: Stops audio. Called while pausing to stop the annoying noises.
EXPORT void CALL DSP_ClearAudioBuffer();
#include "ExportEpilog.h" #include "ExportEpilog.h"
#endif #endif

View File

@ -383,3 +383,7 @@ void DSP_SendAIBuffer(unsigned int address, int sample_rate)
soundStream->Update(); soundStream->Update();
} }
void DSP_ClearAudioBuffer()
{
soundStream->Clear();
}

View File

@ -360,3 +360,9 @@ void DSP_SendAIBuffer(unsigned int address, int sample_rate)
if (/*(counter & 31) == 0 &&*/ soundStream) if (/*(counter & 31) == 0 &&*/ soundStream)
soundStream->Update(); soundStream->Update();
} }
void DSP_ClearAudioBuffer()
{
soundStream->Clear();
}