ppsspp/Core/HLE/sceDisplay.cpp

327 lines
8.4 KiB
C++
Raw Normal View History

2012-11-01 15:19:01 +00:00
// Copyright (c) 2012- PPSSPP Project.
// This program 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 Foundation, version 2.0 or later versions.
2012-11-01 15:19:01 +00:00
// This program 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 2.0 for more details.
// A copy of the GPL 2.0 should have been included with the program.
// If not, see http://www.gnu.org/licenses/
// Official git repository and contact information can be found at
// https://github.com/hrydgard/ppsspp and http://www.ppsspp.org/.
#include <vector>
//#include "base/timeutil.h"
#include "../Core/CoreTiming.h"
#include "../MIPS/MIPS.h"
#include "../HLE/HLE.h"
#include "sceAudio.h"
#include "../Host.h"
#include "../Config.h"
2012-11-19 20:23:29 +00:00
#include "../System.h"
#include "../Core/Core.h"
2012-11-01 15:19:01 +00:00
#include "sceDisplay.h"
#include "sceKernel.h"
#include "sceKernelThread.h"
#include "sceKernelInterrupt.h"
// TODO: This file should not depend directly on GLES code.
#include "../../GPU/GLES/Framebuffer.h"
#include "../../GPU/GLES/ShaderManager.h"
#include "../../GPU/GPUState.h"
#include "../../GPU/GPUInterface.h"
// Internal drawing library
#include "../Util/PPGeDraw.h"
2012-11-01 15:19:01 +00:00
extern ShaderManager shaderManager;
struct FrameBufferState
{
u32 topaddr;
PspDisplayPixelFormat pspFramebufFormat;
int pspFramebufLinesize;
};
// STATE BEGIN
static FrameBufferState framebuf;
static FrameBufferState latchedFramebuf;
static bool framebufIsLatched;
static int enterVblankEvent = -1;
static int leaveVblankEvent = -1;
static int hCount = 0;
static int hCountTotal = 0; //unused
2012-11-01 15:19:01 +00:00
static int vCount = 0;
static int isVblank = 0;
2012-11-01 15:19:01 +00:00
// STATE END
2012-11-01 15:19:01 +00:00
// The vblank period is 731.5 us (0.7315 ms)
const double vblankMs = 0.7315;
const double frameMs = 1000.0 / 60.0;
enum {
PSP_DISPLAY_SETBUF_IMMEDIATE = 0,
2012-11-01 15:19:01 +00:00
PSP_DISPLAY_SETBUF_NEXTFRAME = 1
};
struct WaitVBlankInfo
{
u32 threadID;
int vcountUnblock;
};
std::vector<WaitVBlankInfo> vblankWaitingThreads;
void hleEnterVblank(u64 userdata, int cyclesLate);
void hleLeaveVblank(u64 userdata, int cyclesLate);
void __DisplayInit()
{
framebufIsLatched = false;
framebuf.topaddr = 0x04000000;
framebuf.pspFramebufFormat = PSP_DISPLAY_PIXEL_FORMAT_8888;
framebuf.pspFramebufLinesize = 480; // ??
enterVblankEvent = CoreTiming::RegisterEvent("EnterVBlank", &hleEnterVblank);
leaveVblankEvent = CoreTiming::RegisterEvent("LeaveVBlank", &hleLeaveVblank);
CoreTiming::ScheduleEvent(msToCycles(frameMs - vblankMs), enterVblankEvent, 0);
isVblank = 0;
vCount = 0;
InitGfxState();
}
2012-11-06 16:05:27 +00:00
void __DisplayShutdown()
{
ShutdownGfxState();
}
2012-11-01 15:19:01 +00:00
void hleEnterVblank(u64 userdata, int cyclesLate)
{
int vbCount = userdata;
DEBUG_LOG(HLE, "Enter VBlank %i", vbCount);
isVblank = 1;
// Wake up threads waiting for VBlank
__KernelTriggerWait(WAITTYPE_VBLANK, 0, true);
// Trigger VBlank interrupt handlers.
__TriggerInterrupt(PSP_VBLANK_INTR);
CoreTiming::ScheduleEvent(msToCycles(vblankMs) - cyclesLate, leaveVblankEvent, vbCount+1);
// TODO: Should this be done here or in hleLeaveVblank?
if (framebufIsLatched)
2012-11-01 15:19:01 +00:00
{
DEBUG_LOG(HLE, "Setting latched framebuffer %08x (prev: %08x)", latchedFramebuf.topaddr, framebuf.topaddr);
framebuf = latchedFramebuf;
framebufIsLatched = false;
}
// Draw screen overlays before blitting. Saves and restores the Ge context.
2012-11-18 16:51:14 +00:00
/*
if (g_Config.bShowGPUStats)
{
char stats[512];
sprintf(stats, "Draw calls")
}*/
/*
PPGeBegin();
PPGeDrawImage(I_LOGO, 5, 5, 0, 0xFFFFFFFF);
PPGeDrawText("This is PPGeDraw speaking", 10, 100, 0, 0.5f, 0xFFFFFFFF);
PPGeEnd();
*/
2012-11-01 15:19:01 +00:00
// Yeah, this has to be the right moment to end the frame. Should possibly blit the right buffer
// depending on what's set in sceDisplaySetFramebuf, in order to support half-framerate games -
// an initial hack could be to NOT end the frame if the buffer didn't change? that should work okay.
{
host->EndFrame();
host->BeginFrame();
2012-11-19 20:23:29 +00:00
gpu->BeginFrame();
2012-11-01 15:19:01 +00:00
shaderManager.DirtyShader();
shaderManager.DirtyUniform(DIRTY_ALL);
}
// Tell the emu core that it's time to stop emulating
// Win32 doesn't need this.
#ifndef _WIN32
coreState = CORE_NEXTFRAME;
#endif
2012-11-01 15:19:01 +00:00
}
void hleLeaveVblank(u64 userdata, int cyclesLate)
{
isVblank = 0;
DEBUG_LOG(HLE,"Leave VBlank");
vCount++;
hCount = 0;
CoreTiming::ScheduleEvent(msToCycles(frameMs - vblankMs) - cyclesLate, enterVblankEvent, userdata);
}
void sceDisplayIsVblank()
2012-11-01 15:19:01 +00:00
{
DEBUG_LOG(HLE,"%i=sceDisplayIsVblank()",isVblank);
RETURN(isVblank);
2012-11-01 15:19:01 +00:00
}
u32 sceDisplaySetMode(u32 unknown, u32 xres, u32 yres)
{
DEBUG_LOG(HLE,"sceDisplaySetMode(%d,%d,%d)",unknown,xres,yres);
host->BeginFrame();
2012-11-19 20:23:29 +00:00
gpu->InitClear(PSP_CoreParameter().renderWidth, PSP_CoreParameter().renderHeight);
2012-11-01 15:19:01 +00:00
return 0;
}
void sceDisplaySetFramebuf()
2012-11-01 15:19:01 +00:00
{
//host->EndFrame();
u32 topaddr = PARAM(0);
int linesize = PARAM(1);
int pixelformat = PARAM(2);
int sync = PARAM(3);
2012-11-19 20:23:29 +00:00
FrameBufferState fbstate;
DEBUG_LOG(HLE,"sceDisplaySetFramebuf(topaddr=%08x,linesize=%d,pixelsize=%d,sync=%d)",topaddr,linesize,pixelformat,sync);
if (topaddr == 0)
2012-11-01 15:19:01 +00:00
{
2012-11-19 20:23:29 +00:00
DEBUG_LOG(HLE,"- screen off");
2012-11-01 15:19:01 +00:00
}
2012-11-19 20:23:29 +00:00
else
2012-11-01 15:19:01 +00:00
{
2012-11-19 20:23:29 +00:00
fbstate.topaddr = topaddr;
fbstate.pspFramebufFormat = (PspDisplayPixelFormat)pixelformat;
fbstate.pspFramebufLinesize = linesize;
2012-11-01 15:19:01 +00:00
}
2012-11-19 20:23:29 +00:00
if (sync == PSP_DISPLAY_SETBUF_IMMEDIATE)
2012-11-01 15:19:01 +00:00
{
2012-11-19 20:23:29 +00:00
// Write immediately to the current framebuffer parameters
framebuf = fbstate;
2012-11-01 15:19:01 +00:00
}
2012-11-19 20:23:29 +00:00
else if (topaddr != 0)
2012-11-01 15:19:01 +00:00
{
2012-11-19 20:23:29 +00:00
// Delay the write until vblank
latchedFramebuf = fbstate;
framebufIsLatched = true;
2012-11-01 15:19:01 +00:00
}
RETURN(0);
2012-11-01 15:19:01 +00:00
}
void sceDisplayGetFramebuf()
2012-11-01 15:19:01 +00:00
{
DEBUG_LOG(HLE,"sceDisplayGetFramebuf()");
RETURN(framebuf.topaddr);
2012-11-01 15:19:01 +00:00
}
void sceDisplayWaitVblankStart()
2012-11-01 15:19:01 +00:00
{
DEBUG_LOG(HLE,"sceDisplayWaitVblankStart()");
2012-11-01 15:19:01 +00:00
__KernelWaitCurThread(WAITTYPE_VBLANK, 0, 0, 0, false);
}
void sceDisplayWaitVblank()
2012-11-01 15:19:01 +00:00
{
DEBUG_LOG(HLE,"sceDisplayWaitVblank()");
2012-11-01 15:19:01 +00:00
__KernelWaitCurThread(WAITTYPE_VBLANK, 0, 0, 0, false);
}
2012-11-08 15:28:45 +00:00
void sceDisplayWaitVblankCB()
{
DEBUG_LOG(HLE,"sceDisplayWaitVblankCB()");
__KernelWaitCurThread(WAITTYPE_VBLANK, 0, 0, 0, true);
__KernelCheckCallbacks();
}
void sceDisplayWaitVblankStartCB()
2012-11-01 15:19:01 +00:00
{
DEBUG_LOG(HLE,"sceDisplayWaitVblankStartCB()");
2012-11-01 15:19:01 +00:00
__KernelWaitCurThread(WAITTYPE_VBLANK, 0, 0, 0, true);
__KernelCheckCallbacks();
2012-11-01 15:19:01 +00:00
}
void sceDisplayWaitVblankStartMultiCB()
{
DEBUG_LOG(HLE,"sceDisplayWaitVblankStartMultiCB()");
__KernelWaitCurThread(WAITTYPE_VBLANK, 0, 0, 0, true);
__KernelCheckCallbacks();
}
void sceDisplayGetVcount()
2012-11-01 15:19:01 +00:00
{
// Too spammy
// DEBUG_LOG(HLE,"%i=sceDisplayGetVcount()", vCount);
// Games like Puyo Puyo call this in a tight loop at end-of-frame. We could have it consume some time from CoreTiming?
CoreTiming::Idle(1000000);
RETURN(vCount);
2012-11-01 15:19:01 +00:00
}
void sceDisplayGetCurrentHcount()
2012-11-01 15:19:01 +00:00
{
RETURN(hCount++);
2012-11-01 15:19:01 +00:00
}
void sceDisplayGetAccumulatedHcount()
2012-11-01 15:19:01 +00:00
{
// Just do an estimate
u32 accumHCount = CoreTiming::GetTicks() / (222000000 / 60 / 272);
DEBUG_LOG(HLE,"%i=sceDisplayGetAccumulatedHcount()", accumHCount);
RETURN(accumHCount);
2012-11-01 15:19:01 +00:00
}
float sceDisplayGetFramePerSec()
{
float fps = 59.9400599f;
DEBUG_LOG(HLE,"%f=sceDisplayGetFramePerSec()", fps);
return fps; // (9MHz * 1)/(525 * 286)
}
const HLEFunction sceDisplay[] =
{
2012-11-06 17:14:09 +00:00
{0x0E20F177,&WrapU_UUU<sceDisplaySetMode>, "sceDisplaySetMode"},
{0x289D82FE,sceDisplaySetFramebuf, "sceDisplaySetFramebuf"},
{0x36CDFADE,sceDisplayWaitVblank, "sceDisplayWaitVblank"},
{0x984C27E7,sceDisplayWaitVblankStart, "sceDisplayWaitVblankStart"},
2012-11-08 15:28:45 +00:00
{0x8EB9EC49,sceDisplayWaitVblankCB, "sceDisplayWaitVblankCB"},
2012-11-06 17:14:09 +00:00
{0x46F186C3,sceDisplayWaitVblankStartCB, "sceDisplayWaitVblankStartCB"},
{0x77ed8b3a,sceDisplayWaitVblankStartMultiCB,"sceDisplayWaitVblankStartMultiCB"},
{0xdba6c4c4,&WrapF_V<sceDisplayGetFramePerSec>,"sceDisplayGetFramePerSec"},
{0x773dd3a3,sceDisplayGetCurrentHcount,"sceDisplayGetCurrentHcount"},
{0x210eab3a,sceDisplayGetAccumulatedHcount,"sceDisplayGetAccumulatedHcount"},
{0x9C6EAAD7,sceDisplayGetVcount,"sceDisplayGetVcount"},
2012-11-01 15:19:01 +00:00
{0x984C27E7,0,"sceDisplayWaitVblankStart"},
{0xDEA197D4,0,"sceDisplayGetMode"},
{0x7ED59BC4,0,"sceDisplaySetHoldMode"},
{0xA544C486,0,"sceDisplaySetResumeMode"},
{0x289D82FE,0,"sceDisplaySetFrameBuf"},
{0xEEDA2E54,sceDisplayGetFramebuf,"sceDisplayGetFrameBuf"},
2012-11-01 15:19:01 +00:00
{0xB4F378FA,0,"sceDisplayIsForeground"},
{0x31C4BAA8,0,"sceDisplayGetBrightness"},
{0x4D4E10EC,sceDisplayIsVblank,"sceDisplayIsVblank"},
2012-11-01 15:19:01 +00:00
};
void Register_sceDisplay()
{
RegisterModule("sceDisplay", ARRAY_SIZE(sceDisplay), sceDisplay);
}