gambatte: fix video problem related to new frame timing
This commit is contained in:
parent
61c7537bfc
commit
df6fa1dec0
|
@ -235,7 +235,10 @@ namespace BizHawk.Emulation.Cores.Nintendo.Gameboy
|
|||
{
|
||||
uint samplesEmitted = TICKSINFRAME - frameOverflow; // according to gambatte docs, this is the nominal length of a frame in 2mhz clocks
|
||||
System.Diagnostics.Debug.Assert(samplesEmitted * 2 <= soundbuff.Length);
|
||||
LibGambatte.gambatte_runfor(GambatteState, VideoBuffer, 160, soundbuff, ref samplesEmitted);
|
||||
if (LibGambatte.gambatte_runfor(GambatteState, soundbuff, ref samplesEmitted) > 0)
|
||||
{
|
||||
LibGambatte.gambatte_blitto(GambatteState, VideoBuffer, 160);
|
||||
}
|
||||
|
||||
_cycleCount += (ulong)samplesEmitted;
|
||||
frameOverflow += samplesEmitted;
|
||||
|
@ -431,7 +434,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.Gameboy
|
|||
uint nlen = 0;
|
||||
IntPtr ndata = IntPtr.Zero;
|
||||
|
||||
if (!LibGambatte.gambatte_savestate(GambatteState, VideoBuffer, 160, ref ndata, ref nlen))
|
||||
if (!LibGambatte.gambatte_savestate(GambatteState, ref ndata, ref nlen))
|
||||
throw new Exception("Gambatte failed to save the savestate!");
|
||||
|
||||
if (nlen == 0)
|
||||
|
|
|
@ -151,13 +151,15 @@ namespace BizHawk.Emulation.Cores.Nintendo.Gameboy
|
|||
while (nL < target)
|
||||
{
|
||||
uint nsamp = (uint)(target - nL);
|
||||
LibGambatte.gambatte_runfor(L.GambatteState, leftvbuff, pitch, leftsbuff + nL * 2, ref nsamp);
|
||||
if (LibGambatte.gambatte_runfor(L.GambatteState, leftsbuff + nL * 2, ref nsamp) > 0)
|
||||
LibGambatte.gambatte_blitto(L.GambatteState, leftvbuff, pitch);
|
||||
nL += (int)nsamp;
|
||||
}
|
||||
while (nR < target)
|
||||
{
|
||||
uint nsamp = (uint)(target - nR);
|
||||
LibGambatte.gambatte_runfor(R.GambatteState, rightvbuff, pitch, rightsbuff + nR * 2, ref nsamp);
|
||||
if (LibGambatte.gambatte_runfor(R.GambatteState, rightsbuff + nR * 2, ref nsamp) > 0)
|
||||
LibGambatte.gambatte_blitto(R.GambatteState, rightvbuff, pitch);
|
||||
nR += (int)nsamp;
|
||||
}
|
||||
// poll link cable statuses
|
||||
|
|
|
@ -30,7 +30,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.Gameboy
|
|||
/// <summary>Use GBA intial CPU register values when in CGB mode.</summary>
|
||||
GBA_CGB = 2,
|
||||
/// <summary>Use heuristics to detect and support some multicart MBCs disguised as MBC1.</summary>
|
||||
MULTICART_COMPAT = 4
|
||||
MULTICART_COMPAT = 4
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
|
@ -60,15 +60,30 @@ namespace BizHawk.Emulation.Cores.Nintendo.Gameboy
|
|||
/// exact time (in number of samples) at which it was drawn.
|
||||
/// </summary>
|
||||
/// <param name="core">opaque state pointer</param>
|
||||
/// <param name="videobuf">160x144 RGB32 (native endian) video frame buffer or null</param>
|
||||
/// <param name="pitch">distance in number of pixels (not bytes) from the start of one line to the next in videoBuf.</param>
|
||||
/// <param name="soundbuf">buffer with space >= samples + 2064</param>
|
||||
/// <param name="samples">in: number of stereo samples to produce, out: actual number of samples produced</param>
|
||||
/// <returns>sample number at which the video frame was produced. -1 means no frame was produced.</returns>
|
||||
[DllImport("libgambatte.dll", CallingConvention = CallingConvention.Cdecl)]
|
||||
public static extern int gambatte_runfor(IntPtr core, int[] videobuf, int pitch, short[] soundbuf, ref uint samples);
|
||||
public static extern int gambatte_runfor(IntPtr core, short[] soundbuf, ref uint samples);
|
||||
[DllImport("libgambatte.dll", CallingConvention = CallingConvention.Cdecl)]
|
||||
unsafe public static extern int gambatte_runfor(IntPtr core, int* videobuf, int pitch, short* soundbuf, ref uint samples);
|
||||
unsafe public static extern int gambatte_runfor(IntPtr core, short* soundbuf, ref uint samples);
|
||||
|
||||
/// <summary>
|
||||
/// blit from internal framebuffer to provided framebuffer
|
||||
/// </summary>
|
||||
/// <param name="core">opaque state pointer</param>
|
||||
/// <param name="videobuf"></param>
|
||||
/// <param name="pitch">in pixels</param>
|
||||
[DllImport("libgambatte.dll", CallingConvention = CallingConvention.Cdecl)]
|
||||
unsafe public static extern void gambatte_blitto(IntPtr core, int* videobuf, int pitch);
|
||||
/// <summary>
|
||||
/// blit from internal framebuffer to provided framebuffer
|
||||
/// </summary>
|
||||
/// <param name="core">opaque state pointer</param>
|
||||
/// <param name="videobuf"></param>
|
||||
/// <param name="pitch">in pixels</param>
|
||||
[DllImport("libgambatte.dll", CallingConvention = CallingConvention.Cdecl)]
|
||||
public static extern void gambatte_blitto(IntPtr core, int[] videobuf, int pitch);
|
||||
|
||||
/// <summary>
|
||||
/// Reset to initial state.
|
||||
|
@ -112,7 +127,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.Gameboy
|
|||
/// </summary>
|
||||
[Flags]
|
||||
public enum Buttons
|
||||
{
|
||||
{
|
||||
A = 0x01,
|
||||
B = 0x02,
|
||||
SELECT = 0x04,
|
||||
|
@ -268,13 +283,11 @@ namespace BizHawk.Emulation.Cores.Nintendo.Gameboy
|
|||
/// Saves emulator state to a byte array
|
||||
/// </summary>
|
||||
/// <param name="core">opaque state pointer</param>
|
||||
/// <param name="videobuf">160x144 RGB32 (native endian) video frame buffer or 0. Used for saving a thumbnail. NOT USED ANYMORE</param>
|
||||
/// <param name="pitch">distance in number of pixels (not bytes) from the start of one line to the next in videoBuf.</param>
|
||||
/// <param name="data">private savestate data returned by the core</param>
|
||||
/// <param name="len">the length of the data in bytes</param>
|
||||
/// <returns>success</returns>
|
||||
[DllImport("libgambatte.dll", CallingConvention = CallingConvention.Cdecl)]
|
||||
public static extern bool gambatte_savestate(IntPtr core, int[] videobuf, int pitch, ref IntPtr data, ref uint len);
|
||||
public static extern bool gambatte_savestate(IntPtr core, ref IntPtr data, ref uint len);
|
||||
|
||||
/// <summary>
|
||||
/// destroy data returned by gambatte_savestate() to avoid memory leaks
|
||||
|
@ -293,43 +306,6 @@ namespace BizHawk.Emulation.Cores.Nintendo.Gameboy
|
|||
[DllImport("libgambatte.dll", CallingConvention = CallingConvention.Cdecl)]
|
||||
public static extern bool gambatte_loadstate(IntPtr core, byte[] data, uint len);
|
||||
|
||||
/// <summary>
|
||||
/// Saves emulator state to the file given by 'filepath'.
|
||||
/// </summary>
|
||||
/// <param name="core">opaque state pointer</param>
|
||||
/// <param name="videobuf">160x144 RGB32 (native endian) video frame buffer or 0. Used for saving a thumbnail.</param>
|
||||
/// <param name="pitch">distance in number of pixels (not bytes) from the start of one line to the next in videoBuf.</param>
|
||||
/// <param name="filepath"></param>
|
||||
/// <returns>success</returns>
|
||||
//[DllImport("libgambatte.dll", CallingConvention = CallingConvention.Cdecl)]
|
||||
//public static extern bool gambatte_savestate_file(IntPtr core, int[] videobuf, int pitch, string filepath);
|
||||
|
||||
/// <summary>
|
||||
/// Loads emulator state from the file given by 'filepath'.
|
||||
/// </summary>
|
||||
/// <param name="core">opaque state pointer</param>
|
||||
/// <param name="filepath"></param>
|
||||
/// <returns>success</returns>
|
||||
//[DllImport("libgambatte.dll", CallingConvention = CallingConvention.Cdecl)]
|
||||
//public static extern bool gambatte_loadstate_file(IntPtr core, string filepath);
|
||||
|
||||
/// <summary>
|
||||
/// Selects which state slot to save state to or load state from.
|
||||
/// There are 10 such slots, numbered from 0 to 9 (periodically extended for all n).
|
||||
/// </summary>
|
||||
/// <param name="core">opaque state pointer</param>
|
||||
/// <param name="n"></param>
|
||||
//[DllImport("libgambatte.dll", CallingConvention = CallingConvention.Cdecl)]
|
||||
//public static extern void gambatte_selectstate(IntPtr core, int n);
|
||||
|
||||
/// <summary>
|
||||
/// Current state slot selected with selectState(). Returns a value between 0 and 9 inclusive.
|
||||
/// </summary>
|
||||
/// <param name="core">opaque state pointer</param>
|
||||
/// <returns></returns>
|
||||
//[DllImport("libgambatte.dll", CallingConvention = CallingConvention.Cdecl)]
|
||||
//public static extern int gambatte_currentstate(IntPtr core);
|
||||
|
||||
/// <summary>
|
||||
/// ROM header title of currently loaded ROM image.
|
||||
/// </summary>
|
||||
|
|
|
@ -62,15 +62,14 @@ public:
|
|||
* The return value indicates whether a new video frame has been drawn, and the
|
||||
* exact time (in number of samples) at which it was drawn.
|
||||
*
|
||||
* @param videoBuf 160x144 RGB32 (native endian) video frame buffer or 0
|
||||
* @param pitch distance in number of pixels (not bytes) from the start of one line to the next in videoBuf.
|
||||
* @param soundBuf buffer with space >= samples + 2064
|
||||
* @param samples in: number of stereo samples to produce, out: actual number of samples produced
|
||||
* @return sample number at which the video frame was produced. -1 means no frame was produced.
|
||||
*/
|
||||
long runFor(gambatte::uint_least32_t *videoBuf, int pitch,
|
||||
gambatte::uint_least32_t *soundBuf, unsigned &samples);
|
||||
|
||||
long runFor(gambatte::uint_least32_t *soundBuf, unsigned &samples);
|
||||
|
||||
void blitTo(gambatte::uint_least32_t *videoBuf, int pitch);
|
||||
|
||||
/** Reset to initial state.
|
||||
* Equivalent to reloading a ROM image, or turning a Game Boy Color off and on again.
|
||||
*/
|
||||
|
@ -109,20 +108,6 @@ public:
|
|||
|
||||
// 0 = vram, 1 = rom, 2 = wram, 3 = cartram, 4 = oam, 5 = hram
|
||||
bool getMemoryArea(int which, unsigned char **data, int *length);
|
||||
|
||||
/** Saves emulator state to the state slot selected with selectState().
|
||||
* The data will be stored in the directory given by setSaveDir().
|
||||
*
|
||||
* @param videoBuf 160x144 RGB32 (native endian) video frame buffer or 0. Used for saving a thumbnail.
|
||||
* @param pitch distance in number of pixels (not bytes) from the start of one line to the next in videoBuf.
|
||||
* @return success
|
||||
*/
|
||||
//bool saveState(const gambatte::uint_least32_t *videoBuf, int pitch);
|
||||
|
||||
/** Loads emulator state from the state slot selected with selectState().
|
||||
* @return success
|
||||
*/
|
||||
//bool loadState();
|
||||
|
||||
/** Saves emulator state to the file given by 'filepath'.
|
||||
*
|
||||
|
@ -130,7 +115,7 @@ public:
|
|||
* @param pitch distance in number of pixels (not bytes) from the start of one line to the next in videoBuf.
|
||||
* @return success
|
||||
*/
|
||||
bool saveState(const gambatte::uint_least32_t *videoBuf, int pitch, std::ostream &file);
|
||||
bool saveState(std::ostream &file);
|
||||
|
||||
/** Loads emulator state from the file given by 'filepath'.
|
||||
* @return success
|
||||
|
|
|
@ -24,15 +24,21 @@ __declspec(dllexport) int gambatte_load(void *core, const char *romfiledata, uns
|
|||
return ret;
|
||||
}
|
||||
|
||||
__declspec(dllexport) long gambatte_runfor(void *core, unsigned long *videobuf, int pitch, short *soundbuf, unsigned *samples)
|
||||
__declspec(dllexport) long gambatte_runfor(void *core, short *soundbuf, unsigned *samples)
|
||||
{
|
||||
GB *g = (GB *) core;
|
||||
unsigned sampv = *samples;
|
||||
long ret = g->runFor((unsigned int*)videobuf, pitch, (unsigned int *) soundbuf, sampv);
|
||||
long ret = g->runFor((unsigned int *) soundbuf, sampv);
|
||||
*samples = sampv;
|
||||
return ret;
|
||||
}
|
||||
|
||||
__declspec(dllexport) void gambatte_blitto(void *core, unsigned long *videobuf, int pitch)
|
||||
{
|
||||
GB *g = (GB *) core;
|
||||
g->blitTo((unsigned int *)videobuf, pitch);
|
||||
}
|
||||
|
||||
__declspec(dllexport) void gambatte_reset(void *core, long long now)
|
||||
{
|
||||
GB *g = (GB *) core;
|
||||
|
@ -151,26 +157,12 @@ __declspec(dllexport) int gambatte_savesavedatalength(void *core)
|
|||
return g->saveSavedataLength();
|
||||
}
|
||||
|
||||
/*
|
||||
__declspec(dllexport) int gambatte_savestate(void *core, const unsigned long *videobuf, int pitch)
|
||||
{
|
||||
GB *g = (GB *) core;
|
||||
return g->saveState(videobuf, pitch);
|
||||
}
|
||||
|
||||
__declspec(dllexport) int gambatte_loadstate(void *core)
|
||||
{
|
||||
GB *g = (GB *) core;
|
||||
return g->loadState();
|
||||
}
|
||||
*/
|
||||
|
||||
__declspec(dllexport) int gambatte_savestate(void *core, const unsigned long *videobuf, int pitch, char **data, unsigned *len)
|
||||
__declspec(dllexport) int gambatte_savestate(void *core, char **data, unsigned *len)
|
||||
{
|
||||
GB *g = (GB *) core;
|
||||
|
||||
std::ostringstream os = std::ostringstream(std::ios_base::binary | std::ios_base::out);
|
||||
if (!g->saveState((const unsigned int*)videobuf, pitch, os))
|
||||
if (!g->saveState(os))
|
||||
return 0;
|
||||
|
||||
os.flush();
|
||||
|
@ -193,20 +185,6 @@ __declspec(dllexport) int gambatte_loadstate(void *core, const char *data, unsig
|
|||
return g->loadState(std::istringstream(std::string(data, len), std::ios_base::binary | std::ios_base::in));
|
||||
}
|
||||
|
||||
/*
|
||||
__declspec(dllexport) void gambatte_selectstate(void *core, int n)
|
||||
{
|
||||
GB *g = (GB *) core;
|
||||
g->selectState(n);
|
||||
}
|
||||
|
||||
__declspec(dllexport) int gambatte_currentstate(void *core)
|
||||
{
|
||||
GB *g = (GB *) core;
|
||||
return g->currentState();
|
||||
}
|
||||
*/
|
||||
|
||||
static char horriblebuff[64];
|
||||
__declspec(dllexport) const char *gambatte_romtitle(void *core)
|
||||
{
|
||||
|
|
|
@ -10,7 +10,8 @@ extern "C"
|
|||
|
||||
__declspec(dllexport) int gambatte_load(void *core, const char *romfiledata, unsigned romfilelength, long long now, unsigned flags);
|
||||
|
||||
__declspec(dllexport) long gambatte_runfor(void *core, unsigned long *videobuf, int pitch, short *soundbuf, unsigned *samples);
|
||||
__declspec(dllexport) long gambatte_runfor(void *core, short *soundbuf, unsigned *samples);
|
||||
__declspec(dllexport) void gambatte_blitto(void *core, unsigned long *videobuf, int pitch);
|
||||
|
||||
__declspec(dllexport) void gambatte_reset(void *core, long long now);
|
||||
|
||||
|
@ -46,7 +47,7 @@ extern "C"
|
|||
|
||||
//__declspec(dllexport) int gambatte_loadstate(void *core);
|
||||
|
||||
__declspec(dllexport) int gambatte_savestate(void *core, const unsigned long *videobuf, int pitch, char **data, unsigned *len);
|
||||
__declspec(dllexport) int gambatte_savestate(void *core, char **data, unsigned *len);
|
||||
__declspec(dllexport) void gambatte_savestate_destroy(char *data);
|
||||
|
||||
__declspec(dllexport) int gambatte_loadstate(void *core, const char *data, unsigned len);
|
||||
|
|
|
@ -39,8 +39,18 @@ struct GB::Priv {
|
|||
CPU cpu;
|
||||
int stateNo;
|
||||
bool gbaCgbMode;
|
||||
|
||||
gambatte::uint_least32_t *vbuff;
|
||||
|
||||
Priv() : stateNo(1), gbaCgbMode(false) {}
|
||||
Priv() : stateNo(1), gbaCgbMode(false)
|
||||
{
|
||||
vbuff = new gambatte::uint_least32_t[160*144];
|
||||
}
|
||||
|
||||
~Priv()
|
||||
{
|
||||
delete[] vbuff;
|
||||
}
|
||||
};
|
||||
|
||||
GB::GB() : p_(new Priv) {}
|
||||
|
@ -52,14 +62,13 @@ GB::~GB() {
|
|||
delete p_;
|
||||
}
|
||||
|
||||
long GB::runFor(gambatte::uint_least32_t *const videoBuf, const int pitch,
|
||||
gambatte::uint_least32_t *const soundBuf, unsigned &samples) {
|
||||
long GB::runFor(gambatte::uint_least32_t *const soundBuf, unsigned &samples) {
|
||||
if (!p_->cpu.loaded()) {
|
||||
samples = 0;
|
||||
return -1;
|
||||
}
|
||||
|
||||
p_->cpu.setVideoBuffer(videoBuf, pitch);
|
||||
p_->cpu.setVideoBuffer(p_->vbuff, 160);
|
||||
p_->cpu.setSoundBuffer(soundBuf);
|
||||
const long cyclesSinceBlit = p_->cpu.runFor(samples * 2);
|
||||
samples = p_->cpu.fillSoundBuffer();
|
||||
|
@ -67,6 +76,19 @@ long GB::runFor(gambatte::uint_least32_t *const videoBuf, const int pitch,
|
|||
return cyclesSinceBlit < 0 ? cyclesSinceBlit : static_cast<long>(samples) - (cyclesSinceBlit >> 1);
|
||||
}
|
||||
|
||||
void GB::blitTo(gambatte::uint_least32_t *videoBuf, int pitch)
|
||||
{
|
||||
gambatte::uint_least32_t *src = p_->vbuff;
|
||||
gambatte::uint_least32_t *dst = videoBuf;
|
||||
|
||||
for (int i = 0; i < 144; i++)
|
||||
{
|
||||
std::memcpy(dst, src, sizeof(gambatte::uint_least32_t) * 160);
|
||||
src += 160;
|
||||
dst += pitch;
|
||||
}
|
||||
}
|
||||
|
||||
void GB::reset(const std::uint32_t now) {
|
||||
if (p_->cpu.loaded()) {
|
||||
|
||||
|
@ -201,37 +223,23 @@ bool GB::loadState(std::istream &file) {
|
|||
|
||||
if (StateSaver::loadState(state, file)) {
|
||||
p_->cpu.loadState(state);
|
||||
file.read((char *)p_->vbuff, 160 * 144 * 4); // yes, sloppy
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
/*
|
||||
bool GB::saveState(const gambatte::uint_least32_t *const videoBuf, const int pitch) {
|
||||
if (saveState(videoBuf, pitch, statePath(p_->cpu.saveBasePath(), p_->stateNo))) {
|
||||
p_->cpu.setOsdElement(newStateSavedOsdElement(p_->stateNo));
|
||||
return true;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
bool GB::loadState() {
|
||||
if (loadState(statePath(p_->cpu.saveBasePath(), p_->stateNo))) {
|
||||
p_->cpu.setOsdElement(newStateLoadedOsdElement(p_->stateNo));
|
||||
return true;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
*/
|
||||
bool GB::saveState(const gambatte::uint_least32_t *const videoBuf, const int pitch, std::ostream &file) {
|
||||
bool GB::saveState(std::ostream &file) {
|
||||
if (p_->cpu.loaded()) {
|
||||
SaveState state;
|
||||
p_->cpu.setStatePtrs(state);
|
||||
p_->cpu.saveState(state);
|
||||
return StateSaver::saveState(state, videoBuf, pitch, file);
|
||||
bool ret = StateSaver::saveState(state, file);
|
||||
if (ret)
|
||||
file.write((const char *)p_->vbuff, 160 * 144 * 4); // yes, sloppy
|
||||
return ret;
|
||||
}
|
||||
|
||||
return false;
|
||||
|
|
|
@ -410,9 +410,7 @@ static SaverList list;
|
|||
|
||||
namespace gambatte {
|
||||
|
||||
bool StateSaver::saveState(const SaveState &state,
|
||||
const uint_least32_t *const videoBuf,
|
||||
const int pitch, std::ostream &file) {
|
||||
bool StateSaver::saveState(const SaveState &state, std::ostream &file) {
|
||||
//std::ostream file(filename.c_str(), std::ios_base::binary);
|
||||
|
||||
if (file.fail())
|
||||
|
|
|
@ -35,8 +35,7 @@ public:
|
|||
enum { SS_WIDTH = 160 >> SS_SHIFT };
|
||||
enum { SS_HEIGHT = 144 >> SS_SHIFT };
|
||||
|
||||
static bool saveState(const SaveState &state,
|
||||
const uint_least32_t *videoBuf, int pitch, std::ostream &file);
|
||||
static bool saveState(const SaveState &state, std::ostream &file);
|
||||
static bool loadState(SaveState &state, std::istream &file);
|
||||
};
|
||||
|
||||
|
|
Binary file not shown.
Loading…
Reference in New Issue