2012-11-01 16:19:01 +01: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
|
2012-11-04 23:01:49 +01:00
|
|
|
// the Free Software Foundation, version 2.0 or later versions.
|
2012-11-01 16:19:01 +01: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>
|
2012-11-26 17:35:08 +01:00
|
|
|
#include <cmath>
|
2012-11-01 16:19:01 +01:00
|
|
|
|
2012-11-23 10:33:19 +01:00
|
|
|
// TODO: Move the relevant parts into common. Don't want the core
|
|
|
|
// to be dependent on "native", I think. Or maybe should get rid of common
|
|
|
|
// and move everything into native...
|
2013-08-16 01:00:26 +02:00
|
|
|
#include "base/logging.h"
|
2012-11-23 10:33:19 +01:00
|
|
|
#include "base/timeutil.h"
|
2012-11-01 16:19:01 +01:00
|
|
|
|
2013-07-29 08:09:33 -07:00
|
|
|
#include "Common/Thread.h"
|
|
|
|
#include "Core/CoreTiming.h"
|
|
|
|
#include "Core/CoreParameter.h"
|
2013-05-04 23:35:46 -07:00
|
|
|
#include "Core/Reporting.h"
|
2013-07-29 08:09:33 -07:00
|
|
|
#include "Core/Config.h"
|
|
|
|
#include "Core/System.h"
|
|
|
|
#include "Core/HLE/HLE.h"
|
|
|
|
#include "Core/HLE/sceDisplay.h"
|
|
|
|
#include "Core/HLE/sceKernel.h"
|
|
|
|
#include "Core/HLE/sceKernelThread.h"
|
|
|
|
#include "Core/HLE/sceKernelInterrupt.h"
|
|
|
|
|
|
|
|
#include "GPU/GPUState.h"
|
|
|
|
#include "GPU/GPUInterface.h"
|
2012-11-18 13:04:49 +01:00
|
|
|
|
2012-12-25 15:28:34 +01:00
|
|
|
struct FrameBufferState {
|
2012-11-01 16:19:01 +01:00
|
|
|
u32 topaddr;
|
2013-07-29 08:09:33 -07:00
|
|
|
GEBufferFormat pspFramebufFormat;
|
2012-11-01 16:19:01 +01:00
|
|
|
int pspFramebufLinesize;
|
|
|
|
};
|
|
|
|
|
2012-12-28 02:22:39 -08:00
|
|
|
struct WaitVBlankInfo
|
|
|
|
{
|
2013-02-10 19:02:00 -08:00
|
|
|
WaitVBlankInfo(u32 tid) : threadID(tid), vcountUnblock(1) {}
|
|
|
|
WaitVBlankInfo(u32 tid, int vcount) : threadID(tid), vcountUnblock(vcount) {}
|
2012-12-28 02:22:39 -08:00
|
|
|
u32 threadID;
|
|
|
|
int vcountUnblock; // what was this for again?
|
2013-02-04 01:39:52 -08:00
|
|
|
|
|
|
|
void DoState(PointerWrap &p)
|
|
|
|
{
|
|
|
|
p.Do(threadID);
|
|
|
|
p.Do(vcountUnblock);
|
|
|
|
}
|
2012-12-28 02:22:39 -08:00
|
|
|
};
|
|
|
|
|
2012-11-01 16:19:01 +01:00
|
|
|
// STATE BEGIN
|
|
|
|
static FrameBufferState framebuf;
|
|
|
|
static FrameBufferState latchedFramebuf;
|
|
|
|
static bool framebufIsLatched;
|
|
|
|
|
|
|
|
static int enterVblankEvent = -1;
|
|
|
|
static int leaveVblankEvent = -1;
|
2013-02-18 23:25:06 +01:00
|
|
|
static int afterFlipEvent = -1;
|
2012-11-01 16:19:01 +01:00
|
|
|
|
2013-03-06 20:29:40 +01:00
|
|
|
// hCount is computed now.
|
2012-12-23 11:16:32 +01:00
|
|
|
static int vCount;
|
|
|
|
static int isVblank;
|
2013-02-18 23:44:32 +01:00
|
|
|
static int numSkippedFrames;
|
2012-12-23 11:16:32 +01:00
|
|
|
static bool hasSetMode;
|
2013-04-30 21:29:18 +08:00
|
|
|
static int resumeMode;
|
|
|
|
static int holdMode;
|
2013-03-05 21:31:13 +08:00
|
|
|
static int mode;
|
|
|
|
static int width;
|
|
|
|
static int height;
|
2013-02-12 01:23:05 -08:00
|
|
|
// Don't include this in the state, time increases regardless of state.
|
2013-02-19 00:44:22 +01:00
|
|
|
static double curFrameTime;
|
|
|
|
static double nextFrameTime;
|
2013-08-07 23:32:28 +02:00
|
|
|
static int numVBlanksSinceFlip;
|
2012-11-19 14:16:37 +01:00
|
|
|
|
2013-03-06 20:29:40 +01:00
|
|
|
static u64 frameStartTicks;
|
2013-03-21 22:59:46 +01:00
|
|
|
const float hCountPerVblank = 285.72f; // insprired by jpcsp
|
|
|
|
|
2013-03-06 20:29:40 +01:00
|
|
|
|
2012-12-28 02:22:39 -08:00
|
|
|
std::vector<WaitVBlankInfo> vblankWaitingThreads;
|
|
|
|
|
2012-11-01 16:19:01 +01:00
|
|
|
// STATE END
|
2012-11-19 14:16:37 +01:00
|
|
|
|
2012-12-28 02:22:39 -08:00
|
|
|
// Called when vblank happens (like an internal interrupt.) Not part of state, should be static.
|
2012-12-02 15:44:23 -08:00
|
|
|
std::vector<VblankCallback> vblankListeners;
|
|
|
|
|
2012-11-01 16:19:01 +01:00
|
|
|
// The vblank period is 731.5 us (0.7315 ms)
|
|
|
|
const double vblankMs = 0.7315;
|
|
|
|
const double frameMs = 1000.0 / 60.0;
|
|
|
|
|
|
|
|
enum {
|
2012-11-19 14:16:37 +01:00
|
|
|
PSP_DISPLAY_SETBUF_IMMEDIATE = 0,
|
2012-11-01 16:19:01 +01:00
|
|
|
PSP_DISPLAY_SETBUF_NEXTFRAME = 1
|
|
|
|
};
|
|
|
|
|
2013-05-19 20:20:41 -07:00
|
|
|
static int lastFpsFrame = 0;
|
|
|
|
static double lastFpsTime = 0.0;
|
|
|
|
static double fps = 0.0;
|
|
|
|
static double fpsHistory[120];
|
|
|
|
static size_t fpsHistoryPos = 0;
|
|
|
|
static size_t fpsHistoryValid = 0;
|
2013-06-16 23:44:11 -07:00
|
|
|
static int lastNumFlips = 0;
|
|
|
|
static float flips = 0.0f;
|
2013-08-19 22:05:55 +02:00
|
|
|
static int actualFlips = 0; // taking frameskip into account
|
|
|
|
static int lastActualFlips = 0;
|
|
|
|
static float actualFps = 0;
|
2013-06-29 20:16:09 -07:00
|
|
|
static u64 lastFlipCycles = 0;
|
2013-08-26 20:12:26 -07:00
|
|
|
// For the "max 60 fps" setting.
|
|
|
|
static int lastFlipsTooFrequent = 0;
|
2013-05-19 20:20:41 -07:00
|
|
|
|
2012-11-01 16:19:01 +01:00
|
|
|
void hleEnterVblank(u64 userdata, int cyclesLate);
|
|
|
|
void hleLeaveVblank(u64 userdata, int cyclesLate);
|
2013-02-18 23:25:06 +01:00
|
|
|
void hleAfterFlip(u64 userdata, int cyclesLate);
|
2012-11-01 16:19:01 +01:00
|
|
|
|
2012-12-25 15:28:34 +01:00
|
|
|
void __DisplayInit() {
|
2013-08-07 22:32:04 +02:00
|
|
|
gpuStats.Reset();
|
2012-11-20 00:31:19 +01:00
|
|
|
hasSetMode = false;
|
2013-03-05 21:31:13 +08:00
|
|
|
mode = 0;
|
2013-04-30 21:29:18 +08:00
|
|
|
resumeMode = 0;
|
|
|
|
holdMode = 0;
|
2013-03-05 21:31:13 +08:00
|
|
|
width = 480;
|
|
|
|
height = 272;
|
2013-02-18 23:44:32 +01:00
|
|
|
numSkippedFrames = 0;
|
2013-08-07 23:32:28 +02:00
|
|
|
numVBlanksSinceFlip = 0;
|
2012-11-01 16:19:01 +01:00
|
|
|
framebufIsLatched = false;
|
|
|
|
framebuf.topaddr = 0x04000000;
|
2013-07-29 08:09:33 -07:00
|
|
|
framebuf.pspFramebufFormat = GE_FORMAT_8888;
|
2012-11-01 16:19:01 +01:00
|
|
|
framebuf.pspFramebufLinesize = 480; // ??
|
2013-06-29 20:16:09 -07:00
|
|
|
lastFlipCycles = 0;
|
2013-08-26 20:12:26 -07:00
|
|
|
lastFlipsTooFrequent = 0;
|
2012-11-01 16:19:01 +01:00
|
|
|
|
|
|
|
enterVblankEvent = CoreTiming::RegisterEvent("EnterVBlank", &hleEnterVblank);
|
|
|
|
leaveVblankEvent = CoreTiming::RegisterEvent("LeaveVBlank", &hleLeaveVblank);
|
2013-02-18 23:25:06 +01:00
|
|
|
afterFlipEvent = CoreTiming::RegisterEvent("AfterFlip", &hleAfterFlip);
|
2012-11-01 16:19:01 +01:00
|
|
|
|
|
|
|
CoreTiming::ScheduleEvent(msToCycles(frameMs - vblankMs), enterVblankEvent, 0);
|
|
|
|
isVblank = 0;
|
|
|
|
vCount = 0;
|
2013-02-19 00:44:22 +01:00
|
|
|
curFrameTime = 0.0;
|
|
|
|
nextFrameTime = 0.0;
|
2012-11-01 16:19:01 +01:00
|
|
|
|
2013-08-19 22:05:55 +02:00
|
|
|
flips = 0;
|
|
|
|
fps = 0.0;
|
|
|
|
actualFlips = 0;
|
|
|
|
lastActualFlips = 0;
|
|
|
|
lastNumFlips = 0;
|
|
|
|
fpsHistoryValid = 0;
|
2013-05-19 20:20:41 -07:00
|
|
|
fpsHistoryPos = 0;
|
|
|
|
fpsHistoryValid = 0;
|
|
|
|
|
2012-11-01 16:19:01 +01:00
|
|
|
InitGfxState();
|
|
|
|
}
|
|
|
|
|
2012-12-28 02:22:39 -08:00
|
|
|
void __DisplayDoState(PointerWrap &p) {
|
|
|
|
p.Do(framebuf);
|
|
|
|
p.Do(latchedFramebuf);
|
|
|
|
p.Do(framebufIsLatched);
|
2013-03-06 20:29:40 +01:00
|
|
|
p.Do(frameStartTicks);
|
2012-12-28 02:22:39 -08:00
|
|
|
p.Do(vCount);
|
|
|
|
p.Do(isVblank);
|
|
|
|
p.Do(hasSetMode);
|
2013-03-05 21:31:13 +08:00
|
|
|
p.Do(mode);
|
2013-04-30 21:29:18 +08:00
|
|
|
p.Do(resumeMode);
|
|
|
|
p.Do(holdMode);
|
2013-03-05 21:31:13 +08:00
|
|
|
p.Do(width);
|
|
|
|
p.Do(height);
|
2012-12-28 23:33:00 +01:00
|
|
|
WaitVBlankInfo wvi(0);
|
|
|
|
p.Do(vblankWaitingThreads, wvi);
|
2012-12-28 02:22:39 -08:00
|
|
|
|
|
|
|
p.Do(enterVblankEvent);
|
|
|
|
CoreTiming::RestoreRegisterEvent(enterVblankEvent, "EnterVBlank", &hleEnterVblank);
|
|
|
|
p.Do(leaveVblankEvent);
|
|
|
|
CoreTiming::RestoreRegisterEvent(leaveVblankEvent, "LeaveVBlank", &hleLeaveVblank);
|
2013-02-19 01:01:40 +01:00
|
|
|
p.Do(afterFlipEvent);
|
|
|
|
CoreTiming::RestoreRegisterEvent(afterFlipEvent, "AfterFlipEVent", &hleAfterFlip);
|
2012-12-28 02:22:39 -08:00
|
|
|
|
2012-12-29 12:21:56 -08:00
|
|
|
p.Do(gstate);
|
|
|
|
p.Do(gstate_c);
|
|
|
|
p.Do(gpuStats);
|
|
|
|
gpu->DoState(p);
|
|
|
|
|
|
|
|
ReapplyGfxState();
|
|
|
|
|
|
|
|
if (p.mode == p.MODE_READ) {
|
|
|
|
if (hasSetMode) {
|
|
|
|
gpu->InitClear();
|
|
|
|
}
|
|
|
|
gpu->SetDisplayFramebuffer(framebuf.topaddr, framebuf.pspFramebufLinesize, framebuf.pspFramebufFormat);
|
|
|
|
}
|
|
|
|
|
2012-12-28 02:22:39 -08:00
|
|
|
p.DoMarker("sceDisplay");
|
|
|
|
}
|
|
|
|
|
2012-12-25 15:28:34 +01:00
|
|
|
void __DisplayShutdown() {
|
2012-12-23 11:16:32 +01:00
|
|
|
vblankListeners.clear();
|
|
|
|
vblankWaitingThreads.clear();
|
2012-11-06 17:05:27 +01:00
|
|
|
ShutdownGfxState();
|
|
|
|
}
|
|
|
|
|
2012-12-25 15:28:34 +01:00
|
|
|
void __DisplayListenVblank(VblankCallback callback) {
|
2012-12-02 15:44:23 -08:00
|
|
|
vblankListeners.push_back(callback);
|
|
|
|
}
|
|
|
|
|
2012-12-25 15:28:34 +01:00
|
|
|
void __DisplayFireVblank() {
|
|
|
|
for (std::vector<VblankCallback>::iterator iter = vblankListeners.begin(), end = vblankListeners.end(); iter != end; ++iter) {
|
2012-12-02 15:44:23 -08:00
|
|
|
VblankCallback cb = *iter;
|
|
|
|
cb();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-20 00:49:25 +02:00
|
|
|
// TODO: Also average actualFps
|
2013-08-19 22:05:55 +02:00
|
|
|
void __DisplayGetFPS(float *out_vps, float *out_fps, float *out_actual_fps) {
|
2013-06-16 23:44:11 -07:00
|
|
|
*out_vps = fps;
|
|
|
|
*out_fps = flips;
|
2013-08-19 22:05:55 +02:00
|
|
|
*out_actual_fps = actualFps;
|
2013-04-07 22:43:23 +02:00
|
|
|
}
|
|
|
|
|
2013-05-19 20:20:41 -07:00
|
|
|
void __DisplayGetAveragedFPS(float *out_vps, float *out_fps) {
|
|
|
|
float avg = 0.0;
|
|
|
|
if (fpsHistoryValid > 0) {
|
|
|
|
if (fpsHistoryValid > ARRAY_SIZE(fpsHistory)) {
|
|
|
|
fpsHistoryValid = ARRAY_SIZE(fpsHistory);
|
|
|
|
}
|
|
|
|
for (size_t i = 0; i < fpsHistoryValid; ++i) {
|
|
|
|
avg += fpsHistory[i];
|
|
|
|
}
|
|
|
|
avg /= (double) fpsHistoryValid;
|
|
|
|
}
|
|
|
|
|
|
|
|
*out_vps = *out_fps = avg;
|
|
|
|
}
|
|
|
|
|
2013-08-19 22:05:55 +02:00
|
|
|
void CalculateFPS() {
|
2013-02-11 18:00:18 +08:00
|
|
|
time_update();
|
|
|
|
double now = time_now_d();
|
|
|
|
|
2013-08-20 00:49:25 +02:00
|
|
|
if (now >= lastFpsTime + 1.0) {
|
2013-08-07 22:32:04 +02:00
|
|
|
double frames = (gpuStats.numVBlanks - lastFpsFrame);
|
2013-08-19 22:05:55 +02:00
|
|
|
actualFps = (actualFlips - lastActualFlips);
|
|
|
|
|
2013-06-16 23:44:11 -07:00
|
|
|
fps = frames / (now - lastFpsTime);
|
2013-08-07 22:32:04 +02:00
|
|
|
flips = 60.0 * (double) (gpuStats.numFlips - lastNumFlips) / frames;
|
2013-02-11 18:00:18 +08:00
|
|
|
|
2013-08-07 22:32:04 +02:00
|
|
|
lastFpsFrame = gpuStats.numVBlanks;
|
|
|
|
lastNumFlips = gpuStats.numFlips;
|
2013-08-19 22:05:55 +02:00
|
|
|
lastActualFlips = actualFlips;
|
2013-02-11 18:00:18 +08:00
|
|
|
lastFpsTime = now;
|
2013-05-19 20:20:41 -07:00
|
|
|
|
|
|
|
fpsHistory[fpsHistoryPos++] = fps;
|
|
|
|
fpsHistoryPos = fpsHistoryPos % ARRAY_SIZE(fpsHistory);
|
|
|
|
++fpsHistoryValid;
|
2013-02-11 18:00:18 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-07 22:32:04 +02:00
|
|
|
void __DisplayGetDebugStats(char stats[2048]) {
|
2013-02-18 23:25:06 +01:00
|
|
|
gpu->UpdateStats();
|
|
|
|
|
2013-05-31 10:40:16 -07:00
|
|
|
float vertexAverageCycles = gpuStats.numVertsSubmitted > 0 ? (float)gpuStats.vertexGPUCycles / (float)gpuStats.numVertsSubmitted : 0.0f;
|
|
|
|
|
2013-02-18 23:25:06 +01:00
|
|
|
sprintf(stats,
|
|
|
|
"Frames: %i\n"
|
|
|
|
"DL processing time: %0.2f ms\n"
|
|
|
|
"Kernel processing time: %0.2f ms\n"
|
|
|
|
"Slowest syscall: %s : %0.2f ms\n"
|
|
|
|
"Most active syscall: %s : %0.2f ms\n"
|
|
|
|
"Draw calls: %i, flushes %i\n"
|
|
|
|
"Cached Draw calls: %i\n"
|
2013-08-27 20:58:27 +02:00
|
|
|
"Alpha Tested draws: %i\n"
|
|
|
|
"Non Alpha Tested draws: %i\n"
|
2013-02-18 23:25:06 +01:00
|
|
|
"Num Tracked Vertex Arrays: %i\n"
|
2013-05-31 10:40:16 -07:00
|
|
|
"Cycles executed: %d (%f per vertex)\n"
|
2013-08-23 11:26:13 +02:00
|
|
|
"Commands per call level: %i %i %i %i\n"
|
2013-02-18 23:25:06 +01:00
|
|
|
"Vertices Submitted: %i\n"
|
|
|
|
"Cached Vertices Drawn: %i\n"
|
|
|
|
"Uncached Vertices Drawn: %i\n"
|
|
|
|
"FBOs active: %i\n"
|
|
|
|
"Textures active: %i, decoded: %i\n"
|
|
|
|
"Texture invalidations: %i\n"
|
|
|
|
"Vertex shaders loaded: %i\n"
|
|
|
|
"Fragment shaders loaded: %i\n"
|
|
|
|
"Combined shaders loaded: %i\n",
|
2013-08-07 22:32:04 +02:00
|
|
|
gpuStats.numVBlanks,
|
2013-02-18 23:25:06 +01:00
|
|
|
gpuStats.msProcessingDisplayLists * 1000.0f,
|
|
|
|
kernelStats.msInSyscalls * 1000.0f,
|
|
|
|
kernelStats.slowestSyscallName ? kernelStats.slowestSyscallName : "(none)",
|
|
|
|
kernelStats.slowestSyscallTime * 1000.0f,
|
|
|
|
kernelStats.summedSlowestSyscallName ? kernelStats.summedSlowestSyscallName : "(none)",
|
|
|
|
kernelStats.summedSlowestSyscallTime * 1000.0f,
|
|
|
|
gpuStats.numDrawCalls,
|
|
|
|
gpuStats.numFlushes,
|
|
|
|
gpuStats.numCachedDrawCalls,
|
2013-08-27 20:58:27 +02:00
|
|
|
gpuStats.numAlphaTestedDraws,
|
|
|
|
gpuStats.numNonAlphaTestedDraws,
|
2013-02-18 23:25:06 +01:00
|
|
|
gpuStats.numTrackedVertexArrays,
|
2013-05-31 10:40:16 -07:00
|
|
|
gpuStats.vertexGPUCycles + gpuStats.otherGPUCycles,
|
|
|
|
vertexAverageCycles,
|
2013-08-23 11:26:13 +02:00
|
|
|
gpuStats.gpuCommandsAtCallLevel[0],gpuStats.gpuCommandsAtCallLevel[1],gpuStats.gpuCommandsAtCallLevel[2],gpuStats.gpuCommandsAtCallLevel[3],
|
2013-02-18 23:25:06 +01:00
|
|
|
gpuStats.numVertsSubmitted,
|
|
|
|
gpuStats.numCachedVertsDrawn,
|
|
|
|
gpuStats.numUncachedVertsDrawn,
|
|
|
|
gpuStats.numFBOs,
|
|
|
|
gpuStats.numTextures,
|
|
|
|
gpuStats.numTexturesDecoded,
|
|
|
|
gpuStats.numTextureInvalidations,
|
|
|
|
gpuStats.numVertexShaders,
|
|
|
|
gpuStats.numFragmentShaders,
|
|
|
|
gpuStats.numShaders
|
|
|
|
);
|
|
|
|
|
2013-08-07 22:32:04 +02:00
|
|
|
gpuStats.ResetFrame();
|
2013-02-18 23:25:06 +01:00
|
|
|
kernelStats.ResetFrame();
|
|
|
|
}
|
|
|
|
|
2013-05-20 08:28:07 -07:00
|
|
|
enum {
|
|
|
|
FPS_LIMIT_NORMAL = 0,
|
|
|
|
FPS_LIMIT_CUSTOM = 1,
|
|
|
|
};
|
|
|
|
|
2013-02-18 23:25:06 +01:00
|
|
|
// Let's collect all the throttling and frameskipping logic here.
|
2013-08-07 23:32:28 +02:00
|
|
|
void DoFrameTiming(bool &throttle, bool &skipFrame, float timestep) {
|
2013-05-20 08:28:07 -07:00
|
|
|
int fpsLimiter = PSP_CoreParameter().fpsLimit;
|
2013-08-18 18:53:45 +02:00
|
|
|
throttle = !PSP_CoreParameter().unthrottle;
|
2013-08-18 22:48:53 +02:00
|
|
|
if (fpsLimiter == FPS_LIMIT_CUSTOM && g_Config.iFpsLimit == 0)
|
|
|
|
throttle = false;
|
2013-02-18 23:44:32 +01:00
|
|
|
skipFrame = false;
|
2013-02-18 23:25:06 +01:00
|
|
|
|
2013-02-19 00:44:22 +01:00
|
|
|
// Check if the frameskipping code should be enabled. If neither throttling or frameskipping is on,
|
|
|
|
// we have nothing to do here.
|
2013-05-02 07:48:28 -07:00
|
|
|
bool doFrameSkip = g_Config.iFrameSkip != 0;
|
2013-04-14 12:11:49 +02:00
|
|
|
|
2013-04-14 12:45:58 +02:00
|
|
|
// On non windows, which is always vsync locked, we need to force frameskip when
|
|
|
|
// unthrottled.
|
|
|
|
#ifndef _WIN32
|
|
|
|
if (!throttle) {
|
|
|
|
doFrameSkip = true;
|
|
|
|
skipFrame = true;
|
2013-08-16 01:00:26 +02:00
|
|
|
if (numSkippedFrames >= 7) {
|
2013-04-14 12:45:58 +02:00
|
|
|
skipFrame = false;
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2013-02-19 00:44:22 +01:00
|
|
|
if (!throttle && !doFrameSkip)
|
|
|
|
return;
|
|
|
|
|
|
|
|
time_update();
|
|
|
|
|
|
|
|
curFrameTime = time_now_d();
|
|
|
|
if (nextFrameTime == 0.0)
|
2013-08-07 23:32:28 +02:00
|
|
|
nextFrameTime = time_now_d() + timestep;
|
2013-02-19 00:44:22 +01:00
|
|
|
|
2013-08-18 18:53:45 +02:00
|
|
|
// Argh, we are falling behind! Let's skip a frame and see if we catch up.
|
2013-08-16 01:00:26 +02:00
|
|
|
|
2013-08-18 18:53:45 +02:00
|
|
|
// Auto-frameskip automatically if speed limit is set differently than the default.
|
2013-08-18 22:48:53 +02:00
|
|
|
if (g_Config.iFrameSkip == 1 || (g_Config.iFrameSkip == 0 && fpsLimiter == FPS_LIMIT_CUSTOM && g_Config.iFpsLimit > 60)) {
|
2013-08-16 01:00:26 +02:00
|
|
|
// 1 == autoframeskip
|
|
|
|
if (curFrameTime > nextFrameTime && doFrameSkip) {
|
|
|
|
skipFrame = true;
|
|
|
|
}
|
|
|
|
} else if (g_Config.iFrameSkip > 1) {
|
|
|
|
// Other values = fixed frameskip
|
|
|
|
if (numSkippedFrames >= g_Config.iFrameSkip - 1)
|
|
|
|
skipFrame = false;
|
|
|
|
else
|
|
|
|
skipFrame = true;
|
2013-02-19 00:44:22 +01:00
|
|
|
}
|
2013-04-14 12:11:49 +02:00
|
|
|
|
2013-08-18 22:40:42 +02:00
|
|
|
if (curFrameTime < nextFrameTime && throttle) {
|
2013-02-19 00:44:22 +01:00
|
|
|
// If time gap is huge just jump (somebody unthrottled)
|
2013-08-18 22:40:42 +02:00
|
|
|
if ((nextFrameTime - curFrameTime > 2*timestep) && fpsLimiter == FPS_LIMIT_NORMAL) {
|
2013-08-07 23:32:28 +02:00
|
|
|
nextFrameTime = curFrameTime + timestep;
|
2013-02-19 00:44:22 +01:00
|
|
|
} else {
|
2013-05-20 08:28:07 -07:00
|
|
|
// Wait until we've caught up.
|
2013-02-19 00:44:22 +01:00
|
|
|
while (time_now_d() < nextFrameTime) {
|
|
|
|
Common::SleepCurrentThread(1);
|
|
|
|
time_update();
|
|
|
|
}
|
2013-02-18 23:25:06 +01:00
|
|
|
}
|
2013-02-19 00:44:22 +01:00
|
|
|
curFrameTime = time_now_d();
|
|
|
|
}
|
|
|
|
// Advance lastFrameTime by a constant amount each frame,
|
|
|
|
// but don't let it get too far behind as things can get very jumpy.
|
|
|
|
const double maxFallBehindFrames = 5.5;
|
|
|
|
|
2013-05-09 19:36:23 -05:00
|
|
|
// 3 states of fps limiter
|
2013-05-20 08:28:07 -07:00
|
|
|
if (fpsLimiter == FPS_LIMIT_NORMAL) {
|
2013-08-07 23:32:28 +02:00
|
|
|
nextFrameTime = std::max(nextFrameTime + timestep, time_now_d() - maxFallBehindFrames * timestep);
|
2013-05-20 08:28:07 -07:00
|
|
|
} else if (fpsLimiter == FPS_LIMIT_CUSTOM) {
|
2013-08-18 18:53:45 +02:00
|
|
|
double customLimiter = (g_Config.iFpsLimit / 60.0f) / timestep;
|
2013-05-09 19:36:23 -05:00
|
|
|
nextFrameTime = std::max(nextFrameTime + 1.0 / customLimiter, time_now_d() - maxFallBehindFrames / customLimiter);
|
2013-02-19 00:44:22 +01:00
|
|
|
}
|
2013-02-18 23:25:06 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-12-25 15:28:34 +01:00
|
|
|
void hleEnterVblank(u64 userdata, int cyclesLate) {
|
2012-11-01 16:19:01 +01:00
|
|
|
int vbCount = userdata;
|
|
|
|
|
|
|
|
DEBUG_LOG(HLE, "Enter VBlank %i", vbCount);
|
|
|
|
|
|
|
|
isVblank = 1;
|
2013-03-08 09:30:04 +08:00
|
|
|
vCount++; // // vCount increases at each VBLANK.
|
2012-11-01 16:19:01 +01:00
|
|
|
|
2012-12-02 15:44:23 -08:00
|
|
|
// Fire the vblank listeners before we wake threads.
|
|
|
|
__DisplayFireVblank();
|
|
|
|
|
2012-11-01 16:19:01 +01:00
|
|
|
// Wake up threads waiting for VBlank
|
2012-12-21 16:49:34 +01:00
|
|
|
for (size_t i = 0; i < vblankWaitingThreads.size(); i++) {
|
2013-02-10 19:02:00 -08:00
|
|
|
if (--vblankWaitingThreads[i].vcountUnblock == 0) {
|
|
|
|
__KernelResumeThreadFromWait(vblankWaitingThreads[i].threadID, 0);
|
|
|
|
vblankWaitingThreads.erase(vblankWaitingThreads.begin() + i--);
|
|
|
|
}
|
2012-12-17 20:14:06 +01:00
|
|
|
}
|
2012-11-01 16:19:01 +01:00
|
|
|
|
2013-02-04 00:41:16 +01:00
|
|
|
// Trigger VBlank interrupt handlers.
|
|
|
|
__TriggerInterrupt(PSP_INTR_IMMEDIATE | PSP_INTR_ONLY_IF_ENABLED | PSP_INTR_ALWAYS_RESCHED, PSP_VBLANK_INTR, PSP_INTR_SUB_ALL);
|
2012-11-01 16:19:01 +01:00
|
|
|
|
2013-02-18 23:25:06 +01:00
|
|
|
CoreTiming::ScheduleEvent(msToCycles(vblankMs) - cyclesLate, leaveVblankEvent, vbCount + 1);
|
2012-11-01 16:19:01 +01:00
|
|
|
|
2013-08-16 01:00:26 +02:00
|
|
|
gpuStats.numVBlanks++;
|
|
|
|
|
|
|
|
numVBlanksSinceFlip++;
|
|
|
|
|
2012-11-01 16:19:01 +01:00
|
|
|
// TODO: Should this be done here or in hleLeaveVblank?
|
2012-12-25 15:28:34 +01:00
|
|
|
if (framebufIsLatched) {
|
2012-11-01 16:19:01 +01:00
|
|
|
DEBUG_LOG(HLE, "Setting latched framebuffer %08x (prev: %08x)", latchedFramebuf.topaddr, framebuf.topaddr);
|
2013-08-16 09:00:40 +02:00
|
|
|
framebuf = latchedFramebuf;
|
|
|
|
framebufIsLatched = false;
|
|
|
|
gpu->SetDisplayFramebuffer(framebuf.topaddr, framebuf.pspFramebufLinesize, framebuf.pspFramebufFormat);
|
2012-11-01 16:19:01 +01:00
|
|
|
}
|
2013-08-10 18:36:11 +02:00
|
|
|
// We flip only if the framebuffer was dirty. This eliminates flicker when using
|
|
|
|
// non-buffered rendering. The interaction with frame skipping seems to need
|
|
|
|
// some work.
|
|
|
|
if (gpu->FramebufferDirty()) {
|
2013-08-20 00:49:25 +02:00
|
|
|
if (g_Config.iShowFPSCounter) {
|
|
|
|
CalculateFPS();
|
|
|
|
}
|
|
|
|
|
2013-08-16 01:00:26 +02:00
|
|
|
// Setting CORE_NEXTFRAME causes a swap.
|
|
|
|
// Check first though, might've just quit / been paused.
|
|
|
|
if (gpu->FramebufferReallyDirty()) {
|
|
|
|
if (coreState == CORE_RUNNING) {
|
|
|
|
coreState = CORE_NEXTFRAME;
|
|
|
|
gpu->CopyDisplayToOutput();
|
2013-08-19 22:05:55 +02:00
|
|
|
actualFlips++;
|
2013-08-16 01:00:26 +02:00
|
|
|
}
|
|
|
|
}
|
2013-08-11 22:11:51 +02:00
|
|
|
|
2013-08-16 01:00:26 +02:00
|
|
|
gpuStats.numFlips++;
|
2013-01-20 22:05:11 +01:00
|
|
|
|
2013-08-07 23:32:28 +02:00
|
|
|
bool throttle, skipFrame;
|
|
|
|
DoFrameTiming(throttle, skipFrame, (float)numVBlanksSinceFlip * (1.0f / 60.0f));
|
2012-12-18 10:25:57 +01:00
|
|
|
|
2013-08-16 01:00:26 +02:00
|
|
|
// Max 4 skipped frames in a row - 15 fps is really the bare minimum for playability.
|
|
|
|
// We check for 3 here so it's 3 skipped frames, 1 non skipped, 3 skipped, etc.
|
|
|
|
int maxFrameskip = throttle ? g_Config.iFrameSkip : 8;
|
|
|
|
if (numSkippedFrames >= maxFrameskip) {
|
|
|
|
skipFrame = false;
|
|
|
|
}
|
|
|
|
|
2013-08-07 23:32:28 +02:00
|
|
|
if (skipFrame) {
|
|
|
|
gstate_c.skipDrawReason |= SKIPDRAW_SKIPFRAME;
|
|
|
|
numSkippedFrames++;
|
|
|
|
} else {
|
2013-08-16 01:00:26 +02:00
|
|
|
gstate_c.skipDrawReason &= ~SKIPDRAW_SKIPFRAME;
|
2013-08-07 23:32:28 +02:00
|
|
|
numSkippedFrames = 0;
|
|
|
|
}
|
2013-02-18 23:44:32 +01:00
|
|
|
|
2013-08-07 23:32:28 +02:00
|
|
|
// Returning here with coreState == CORE_NEXTFRAME causes a buffer flip to happen (next frame).
|
|
|
|
// Right after, we regain control for a little bit in hleAfterFlip. I think that's a great
|
|
|
|
// place to do housekeeping.
|
|
|
|
CoreTiming::ScheduleEvent(0 - cyclesLate, afterFlipEvent, 0);
|
|
|
|
numVBlanksSinceFlip = 0;
|
|
|
|
}
|
2013-02-18 23:25:06 +01:00
|
|
|
}
|
2012-11-22 23:07:15 +01:00
|
|
|
|
2013-02-18 23:25:06 +01:00
|
|
|
void hleAfterFlip(u64 userdata, int cyclesLate)
|
|
|
|
{
|
|
|
|
gpu->BeginFrame(); // doesn't really matter if begin or end of frame.
|
2012-11-01 16:19:01 +01:00
|
|
|
}
|
|
|
|
|
2012-12-25 15:28:34 +01:00
|
|
|
void hleLeaveVblank(u64 userdata, int cyclesLate) {
|
2012-11-01 16:19:01 +01:00
|
|
|
isVblank = 0;
|
2012-11-20 00:31:19 +01:00
|
|
|
DEBUG_LOG(HLE,"Leave VBlank %i", (int)userdata - 1);
|
2013-03-06 20:29:40 +01:00
|
|
|
frameStartTicks = CoreTiming::GetTicks();
|
2012-11-01 16:19:01 +01:00
|
|
|
CoreTiming::ScheduleEvent(msToCycles(frameMs - vblankMs) - cyclesLate, enterVblankEvent, userdata);
|
|
|
|
}
|
|
|
|
|
2013-03-05 21:34:22 +08:00
|
|
|
u32 sceDisplayIsVblank() {
|
2012-11-01 16:19:01 +01:00
|
|
|
DEBUG_LOG(HLE,"%i=sceDisplayIsVblank()",isVblank);
|
2013-03-05 21:34:22 +08:00
|
|
|
return isVblank;
|
2012-11-01 16:19:01 +01:00
|
|
|
}
|
|
|
|
|
2013-03-05 21:31:13 +08:00
|
|
|
u32 sceDisplaySetMode(int displayMode, int displayWidth, int displayHeight) {
|
|
|
|
DEBUG_LOG(HLE,"sceDisplaySetMode(%i, %i, %i)", displayMode, displayWidth, displayHeight);
|
2012-11-01 16:19:01 +01:00
|
|
|
|
2012-12-25 15:28:34 +01:00
|
|
|
if (!hasSetMode) {
|
2012-11-20 10:59:23 +01:00
|
|
|
gpu->InitClear();
|
2012-11-20 00:31:19 +01:00
|
|
|
hasSetMode = true;
|
|
|
|
}
|
2013-03-05 21:31:13 +08:00
|
|
|
mode = displayMode;
|
|
|
|
width = displayWidth;
|
|
|
|
height = displayHeight;
|
2012-11-01 16:19:01 +01:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-08-20 00:49:25 +02:00
|
|
|
// Some games (GTA) never call this during gameplay, so bad place to put a framerate counter.
|
2013-03-02 23:36:11 +08:00
|
|
|
u32 sceDisplaySetFramebuf(u32 topaddr, int linesize, int pixelformat, int sync) {
|
2012-11-19 21:23:29 +01:00
|
|
|
FrameBufferState fbstate;
|
2013-03-02 23:36:11 +08:00
|
|
|
DEBUG_LOG(HLE,"sceDisplaySetFramebuf(topaddr=%08x,linesize=%d,pixelsize=%d,sync=%d)", topaddr, linesize, pixelformat, sync);
|
2012-12-25 15:28:34 +01:00
|
|
|
if (topaddr == 0) {
|
2012-11-19 21:23:29 +01:00
|
|
|
DEBUG_LOG(HLE,"- screen off");
|
2012-12-25 15:28:34 +01:00
|
|
|
} else {
|
2012-11-19 21:23:29 +01:00
|
|
|
fbstate.topaddr = topaddr;
|
2013-07-29 08:09:33 -07:00
|
|
|
fbstate.pspFramebufFormat = (GEBufferFormat)pixelformat;
|
2012-11-19 21:23:29 +01:00
|
|
|
fbstate.pspFramebufLinesize = linesize;
|
2012-11-01 16:19:01 +01:00
|
|
|
}
|
|
|
|
|
2013-08-25 09:46:21 -07:00
|
|
|
s64 delayCycles = 0;
|
2013-08-26 20:12:26 -07:00
|
|
|
if (topaddr != framebuf.topaddr && g_Config.iForceMaxEmulatedFPS > 0) {
|
|
|
|
// Sometimes we get a small number, there's probably no need to delay the thread for this.
|
|
|
|
// sceDisplaySetFramebuf() isn't supposed to delay threads at all. This is a hack.
|
|
|
|
const int FLIP_DELAY_CYCLES_MIN = 10;
|
|
|
|
// Some games (like Final Fantasy 4) only call this too much in spurts.
|
|
|
|
// The goal is to fix games where this would result in a consistent overhead.
|
|
|
|
const int FLIP_DELAY_MIN_FLIPS = 30;
|
|
|
|
|
|
|
|
u64 now = CoreTiming::GetTicks();
|
|
|
|
u64 expected = msToCycles(1000) / g_Config.iForceMaxEmulatedFPS;
|
|
|
|
u64 actual = now - lastFlipCycles;
|
|
|
|
if (actual < expected - FLIP_DELAY_CYCLES_MIN) {
|
|
|
|
if (lastFlipsTooFrequent >= FLIP_DELAY_MIN_FLIPS) {
|
2013-08-25 09:46:21 -07:00
|
|
|
delayCycles = expected - actual;
|
2013-08-26 20:12:26 -07:00
|
|
|
} else {
|
|
|
|
++lastFlipsTooFrequent;
|
2013-08-25 09:46:21 -07:00
|
|
|
}
|
2013-08-26 20:12:26 -07:00
|
|
|
} else {
|
|
|
|
--lastFlipsTooFrequent;
|
2013-06-29 20:16:09 -07:00
|
|
|
}
|
2013-08-26 20:12:26 -07:00
|
|
|
lastFlipCycles = CoreTiming::GetTicks();
|
2013-06-16 23:44:11 -07:00
|
|
|
}
|
|
|
|
|
2012-12-25 15:28:34 +01:00
|
|
|
if (sync == PSP_DISPLAY_SETBUF_IMMEDIATE) {
|
2012-11-19 21:23:29 +01:00
|
|
|
// Write immediately to the current framebuffer parameters
|
2013-04-08 20:59:45 +02:00
|
|
|
if (topaddr != 0) {
|
2013-08-14 17:54:58 +02:00
|
|
|
framebuf = fbstate;
|
|
|
|
gpu->SetDisplayFramebuffer(framebuf.topaddr, framebuf.pspFramebufLinesize, framebuf.pspFramebufFormat);
|
2013-04-08 20:59:45 +02:00
|
|
|
} else {
|
2012-12-21 16:25:05 -08:00
|
|
|
WARN_LOG(HLE, "%s: PSP_DISPLAY_SETBUF_IMMEDIATE without topaddr?", __FUNCTION__);
|
2013-04-08 20:59:45 +02:00
|
|
|
}
|
2012-12-25 15:28:34 +01:00
|
|
|
} else if (topaddr != 0) {
|
2012-11-19 21:23:29 +01:00
|
|
|
// Delay the write until vblank
|
|
|
|
latchedFramebuf = fbstate;
|
|
|
|
framebufIsLatched = true;
|
2012-11-01 16:19:01 +01:00
|
|
|
}
|
2013-08-25 09:46:21 -07:00
|
|
|
|
|
|
|
if (delayCycles > 0) {
|
|
|
|
// Okay, the game is going at too high a frame rate. God of War and Fat Princess both do this.
|
|
|
|
// Simply eating the cycles works and is fast, but breaks other games (like Jeanne d'Arc.)
|
2013-08-26 20:12:26 -07:00
|
|
|
// So, instead, we delay this HLE thread only (a small deviation from correct behavior.)
|
2013-08-25 09:46:21 -07:00
|
|
|
return hleDelayResult(0, "set framebuf", cyclesToUs(delayCycles));
|
|
|
|
} else {
|
|
|
|
return 0;
|
|
|
|
}
|
2012-11-01 16:19:01 +01:00
|
|
|
}
|
|
|
|
|
2013-05-18 10:20:13 -07:00
|
|
|
bool __DisplayGetFramebuf(u8 **topaddr, u32 *linesize, u32 *pixelFormat, int latchedMode) {
|
|
|
|
const FrameBufferState &fbState = latchedMode == 1 ? latchedFramebuf : framebuf;
|
2013-01-13 16:35:34 -08:00
|
|
|
if (topaddr != NULL)
|
|
|
|
*topaddr = Memory::GetPointer(fbState.topaddr);
|
|
|
|
if (linesize != NULL)
|
|
|
|
*linesize = fbState.pspFramebufLinesize;
|
|
|
|
if (pixelFormat != NULL)
|
|
|
|
*pixelFormat = fbState.pspFramebufFormat;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2013-05-18 10:20:13 -07:00
|
|
|
u32 sceDisplayGetFramebuf(u32 topaddrPtr, u32 linesizePtr, u32 pixelFormatPtr, int latchedMode) {
|
|
|
|
const FrameBufferState &fbState = latchedMode == 1 ? latchedFramebuf : framebuf;
|
2013-03-02 23:36:11 +08:00
|
|
|
DEBUG_LOG(HLE,"sceDisplayGetFramebuf(*%08x = %08x, *%08x = %08x, *%08x = %08x, %i)", topaddrPtr, fbState.topaddr, linesizePtr, fbState.pspFramebufLinesize, pixelFormatPtr, fbState.pspFramebufFormat, mode);
|
2012-12-25 15:28:34 +01:00
|
|
|
|
2012-11-19 23:53:38 +01:00
|
|
|
if (Memory::IsValidAddress(topaddrPtr))
|
|
|
|
Memory::Write_U32(fbState.topaddr, topaddrPtr);
|
|
|
|
if (Memory::IsValidAddress(linesizePtr))
|
|
|
|
Memory::Write_U32(fbState.pspFramebufLinesize, linesizePtr);
|
|
|
|
if (Memory::IsValidAddress(pixelFormatPtr))
|
|
|
|
Memory::Write_U32(fbState.pspFramebufFormat, pixelFormatPtr);
|
|
|
|
|
|
|
|
return 0;
|
2012-11-01 16:19:01 +01:00
|
|
|
}
|
|
|
|
|
2013-01-23 18:45:14 +01:00
|
|
|
u32 sceDisplayWaitVblankStart() {
|
2013-03-10 22:56:24 -07:00
|
|
|
VERBOSE_LOG(HLE,"sceDisplayWaitVblankStart()");
|
2012-12-17 20:14:06 +01:00
|
|
|
vblankWaitingThreads.push_back(WaitVBlankInfo(__KernelGetCurThread()));
|
2013-01-26 10:44:04 -08:00
|
|
|
__KernelWaitCurThread(WAITTYPE_VBLANK, 0, 0, 0, false, "vblank start waited");
|
2013-01-23 18:45:14 +01:00
|
|
|
return 0;
|
2012-11-01 16:19:01 +01:00
|
|
|
}
|
|
|
|
|
2013-01-23 18:45:14 +01:00
|
|
|
u32 sceDisplayWaitVblank() {
|
2013-01-12 17:53:23 +01:00
|
|
|
if (!isVblank) {
|
2013-03-10 22:56:24 -07:00
|
|
|
VERBOSE_LOG(HLE,"sceDisplayWaitVblank()");
|
2013-01-12 17:53:23 +01:00
|
|
|
vblankWaitingThreads.push_back(WaitVBlankInfo(__KernelGetCurThread()));
|
2013-01-26 10:44:04 -08:00
|
|
|
__KernelWaitCurThread(WAITTYPE_VBLANK, 0, 0, 0, false, "vblank waited");
|
2013-01-23 18:45:14 +01:00
|
|
|
return 0;
|
2013-01-12 17:53:23 +01:00
|
|
|
} else {
|
|
|
|
DEBUG_LOG(HLE,"sceDisplayWaitVblank() - not waiting since in vBlank");
|
2013-05-04 23:35:46 -07:00
|
|
|
hleEatCycles(1110);
|
2013-01-23 18:45:14 +01:00
|
|
|
return 1;
|
2013-01-12 17:53:23 +01:00
|
|
|
}
|
2012-11-01 16:19:01 +01:00
|
|
|
}
|
|
|
|
|
2013-02-10 19:02:00 -08:00
|
|
|
u32 sceDisplayWaitVblankStartMulti(int vblanks) {
|
2013-05-28 07:39:28 -07:00
|
|
|
if (vblanks <= 0) {
|
|
|
|
WARN_LOG(HLE, "sceDisplayWaitVblankStartMulti(%d): invalid number of vblanks", vblanks);
|
|
|
|
return SCE_KERNEL_ERROR_INVALID_VALUE;
|
|
|
|
}
|
|
|
|
VERBOSE_LOG(HLE, "sceDisplayWaitVblankStartMulti(%d)", vblanks);
|
2013-08-27 23:19:53 -07:00
|
|
|
if (!__KernelIsDispatchEnabled())
|
|
|
|
return SCE_KERNEL_ERROR_CAN_NOT_WAIT;
|
2013-02-10 19:02:00 -08:00
|
|
|
vblankWaitingThreads.push_back(WaitVBlankInfo(__KernelGetCurThread(), vblanks));
|
2013-01-26 10:44:04 -08:00
|
|
|
__KernelWaitCurThread(WAITTYPE_VBLANK, 0, 0, 0, false, "vblank start multi waited");
|
2013-01-23 18:45:14 +01:00
|
|
|
return 0;
|
2012-12-06 17:44:33 +00:00
|
|
|
}
|
|
|
|
|
2013-01-23 18:45:14 +01:00
|
|
|
u32 sceDisplayWaitVblankCB() {
|
2013-01-12 17:53:23 +01:00
|
|
|
if (!isVblank) {
|
2013-03-10 22:56:24 -07:00
|
|
|
VERBOSE_LOG(HLE,"sceDisplayWaitVblankCB()");
|
2013-01-12 17:53:23 +01:00
|
|
|
vblankWaitingThreads.push_back(WaitVBlankInfo(__KernelGetCurThread()));
|
2013-01-26 10:44:04 -08:00
|
|
|
__KernelWaitCurThread(WAITTYPE_VBLANK, 0, 0, 0, true, "vblank waited");
|
2013-01-23 18:45:14 +01:00
|
|
|
return 0;
|
2013-01-12 17:53:23 +01:00
|
|
|
} else {
|
|
|
|
DEBUG_LOG(HLE,"sceDisplayWaitVblank() - not waiting since in vBlank");
|
2013-05-04 23:35:46 -07:00
|
|
|
hleEatCycles(1110);
|
2013-01-23 18:45:14 +01:00
|
|
|
return 1;
|
2013-01-12 17:53:23 +01:00
|
|
|
}
|
2012-11-08 16:28:45 +01:00
|
|
|
}
|
|
|
|
|
2013-01-23 18:45:14 +01:00
|
|
|
u32 sceDisplayWaitVblankStartCB() {
|
2013-03-10 22:56:24 -07:00
|
|
|
VERBOSE_LOG(HLE,"sceDisplayWaitVblankStartCB()");
|
2012-12-17 20:14:06 +01:00
|
|
|
vblankWaitingThreads.push_back(WaitVBlankInfo(__KernelGetCurThread()));
|
2013-01-26 10:44:04 -08:00
|
|
|
__KernelWaitCurThread(WAITTYPE_VBLANK, 0, 0, 0, true, "vblank start waited");
|
2013-01-23 18:45:14 +01:00
|
|
|
return 0;
|
2012-11-01 16:19:01 +01:00
|
|
|
}
|
|
|
|
|
2013-02-10 19:02:00 -08:00
|
|
|
u32 sceDisplayWaitVblankStartMultiCB(int vblanks) {
|
2013-05-28 07:39:28 -07:00
|
|
|
if (vblanks <= 0) {
|
|
|
|
WARN_LOG(HLE, "sceDisplayWaitVblankStartMultiCB(%d): invalid number of vblanks", vblanks);
|
|
|
|
return SCE_KERNEL_ERROR_INVALID_VALUE;
|
|
|
|
}
|
|
|
|
VERBOSE_LOG(HLE,"sceDisplayWaitVblankStartMultiCB(%d)", vblanks);
|
2013-08-27 23:19:53 -07:00
|
|
|
if (!__KernelIsDispatchEnabled())
|
|
|
|
return SCE_KERNEL_ERROR_CAN_NOT_WAIT;
|
2013-02-10 19:02:00 -08:00
|
|
|
vblankWaitingThreads.push_back(WaitVBlankInfo(__KernelGetCurThread(), vblanks));
|
2013-01-26 10:44:04 -08:00
|
|
|
__KernelWaitCurThread(WAITTYPE_VBLANK, 0, 0, 0, true, "vblank start multi waited");
|
2013-01-23 18:45:14 +01:00
|
|
|
return 0;
|
2012-11-18 20:13:27 +00:00
|
|
|
}
|
|
|
|
|
2012-12-25 15:28:34 +01:00
|
|
|
u32 sceDisplayGetVcount() {
|
2013-03-10 22:56:24 -07:00
|
|
|
VERBOSE_LOG(HLE,"%i=sceDisplayGetVcount()", vCount);
|
2012-12-17 22:52:31 +01:00
|
|
|
|
2013-05-04 23:35:46 -07:00
|
|
|
hleEatCycles(150);
|
2012-12-17 22:52:31 +01:00
|
|
|
return vCount;
|
2012-11-01 16:19:01 +01:00
|
|
|
}
|
|
|
|
|
2013-03-05 21:31:13 +08:00
|
|
|
u32 sceDisplayGetCurrentHcount() {
|
2013-03-21 22:59:46 +01:00
|
|
|
u32 currentHCount = (CoreTiming::GetTicks() - frameStartTicks) / ((u64)CoreTiming::GetClockFrequencyMHz() * 1000000 / 60 / hCountPerVblank);
|
2013-03-08 10:35:17 +08:00
|
|
|
DEBUG_LOG(HLE,"%i=sceDisplayGetCurrentHcount()", currentHCount);
|
2013-05-04 23:35:46 -07:00
|
|
|
hleEatCycles(275);
|
2013-03-08 09:30:04 +08:00
|
|
|
return currentHCount;
|
2012-11-01 16:19:01 +01:00
|
|
|
}
|
|
|
|
|
2013-03-05 21:31:13 +08:00
|
|
|
u32 sceDisplayAdjustAccumulatedHcount() {
|
2013-05-04 23:35:46 -07:00
|
|
|
ERROR_LOG_REPORT(HLE,"UNIMPL sceDisplayAdjustAccumulatedHcount()");
|
2013-03-05 21:31:13 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
u32 sceDisplayGetAccumulatedHcount() {
|
2013-03-21 22:59:46 +01:00
|
|
|
u32 currentHCount = (CoreTiming::GetTicks() - frameStartTicks) / ((u64)CoreTiming::GetClockFrequencyMHz() * 1000000 / 60 / hCountPerVblank);
|
2013-03-07 16:47:09 +08:00
|
|
|
u32 accumHCount = currentHCount + (u32) (vCount * hCountPerVblank);
|
2013-03-08 10:35:17 +08:00
|
|
|
DEBUG_LOG(HLE,"%i=sceDisplayGetAccumulatedHcount()", accumHCount);
|
2013-05-04 23:35:46 -07:00
|
|
|
hleEatCycles(235);
|
2013-03-05 21:31:13 +08:00
|
|
|
return accumHCount;
|
2012-11-01 16:19:01 +01:00
|
|
|
}
|
|
|
|
|
2012-12-25 15:28:34 +01:00
|
|
|
float sceDisplayGetFramePerSec() {
|
2013-05-18 10:20:13 -07:00
|
|
|
const static float framePerSec = 59.9400599f;
|
|
|
|
DEBUG_LOG(HLE,"%f=sceDisplayGetFramePerSec()", framePerSec);
|
|
|
|
return framePerSec; // (9MHz * 1)/(525 * 286)
|
2012-11-01 16:19:01 +01:00
|
|
|
}
|
|
|
|
|
2013-03-03 12:39:37 +01:00
|
|
|
u32 sceDisplayIsForeground() {
|
2013-04-30 01:44:14 +08:00
|
|
|
DEBUG_LOG(HLE,"IMPL sceDisplayIsForeground()");
|
|
|
|
if (!hasSetMode || framebuf.topaddr == 0)
|
2013-04-30 21:29:18 +08:00
|
|
|
return 0;
|
2013-04-30 01:44:14 +08:00
|
|
|
else
|
2013-04-30 21:29:18 +08:00
|
|
|
return 1; // return value according to JPCSP comment
|
2013-03-05 21:31:13 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
u32 sceDisplayGetMode(u32 modeAddr, u32 widthAddr, u32 heightAddr) {
|
2013-03-15 00:43:35 -07:00
|
|
|
DEBUG_LOG(HLE,"sceDisplayGetMode(%08x, %08x, %08x)", modeAddr, widthAddr, heightAddr);
|
2013-03-05 21:31:13 +08:00
|
|
|
if (Memory::IsValidAddress(modeAddr))
|
|
|
|
Memory::Write_U32(mode, modeAddr);
|
|
|
|
if (Memory::IsValidAddress(widthAddr))
|
|
|
|
Memory::Write_U32(width, widthAddr);
|
|
|
|
if (Memory::IsValidAddress(heightAddr))
|
|
|
|
Memory::Write_U32(height, heightAddr);
|
|
|
|
return 0;
|
2013-03-03 12:39:37 +01:00
|
|
|
}
|
|
|
|
|
2013-04-30 21:29:18 +08:00
|
|
|
u32 sceDisplayIsVsync() {
|
|
|
|
ERROR_LOG(HLE,"UNIMPL sceDisplayIsVsync()");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
u32 sceDisplayGetResumeMode(u32 resumeModeAddr) {
|
|
|
|
ERROR_LOG(HLE,"sceDisplayGetResumeMode(%08x)", resumeModeAddr);
|
|
|
|
if (Memory::IsValidAddress(resumeModeAddr))
|
|
|
|
Memory::Write_U32(resumeMode, resumeModeAddr);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
u32 sceDisplaySetResumeMode(u32 rMode) {
|
|
|
|
ERROR_LOG(HLE,"sceDisplaySetResumeMode(%08x)", rMode);
|
|
|
|
resumeMode = rMode;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
u32 sceDisplayGetBrightness(u32 levelAddr) {
|
|
|
|
ERROR_LOG(HLE,"UNIMPL sceDisplayGetBrightness(%08x)", levelAddr);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
u32 sceDisplaySetHoldMode(u32 hMode) {
|
|
|
|
ERROR_LOG(HLE,"sceDisplaySetHoldMode(%08x)", hMode);
|
|
|
|
holdMode = hMode;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-12-25 15:28:34 +01:00
|
|
|
const HLEFunction sceDisplay[] = {
|
2013-03-05 21:31:13 +08:00
|
|
|
{0x0E20F177,WrapU_III<sceDisplaySetMode>, "sceDisplaySetMode"},
|
2013-05-19 19:21:59 -07:00
|
|
|
{0x289D82FE,WrapU_UIII<sceDisplaySetFramebuf>, "sceDisplaySetFrameBuf"},
|
2012-12-17 21:45:32 +01:00
|
|
|
{0xEEDA2E54,WrapU_UUUI<sceDisplayGetFramebuf>,"sceDisplayGetFrameBuf"},
|
2013-08-27 00:56:06 -07:00
|
|
|
{0x36CDFADE,WrapU_V<sceDisplayWaitVblank>, "sceDisplayWaitVblank", HLE_NOT_DISPATCH_SUSPENDED},
|
|
|
|
{0x984C27E7,WrapU_V<sceDisplayWaitVblankStart>, "sceDisplayWaitVblankStart", HLE_NOT_DISPATCH_SUSPENDED},
|
2013-08-27 23:19:53 -07:00
|
|
|
{0x40f1469c,WrapU_I<sceDisplayWaitVblankStartMulti>, "sceDisplayWaitVblankStartMulti"},
|
2013-08-27 00:56:06 -07:00
|
|
|
{0x8EB9EC49,WrapU_V<sceDisplayWaitVblankCB>, "sceDisplayWaitVblankCB", HLE_NOT_DISPATCH_SUSPENDED},
|
|
|
|
{0x46F186C3,WrapU_V<sceDisplayWaitVblankStartCB>, "sceDisplayWaitVblankStartCB", HLE_NOT_DISPATCH_SUSPENDED},
|
2013-08-27 23:19:53 -07:00
|
|
|
{0x77ed8b3a,WrapU_I<sceDisplayWaitVblankStartMultiCB>,"sceDisplayWaitVblankStartMultiCB"},
|
2012-12-17 21:45:32 +01:00
|
|
|
{0xdba6c4c4,WrapF_V<sceDisplayGetFramePerSec>,"sceDisplayGetFramePerSec"},
|
2013-03-05 21:31:13 +08:00
|
|
|
{0x773dd3a3,WrapU_V<sceDisplayGetCurrentHcount>,"sceDisplayGetCurrentHcount"},
|
|
|
|
{0x210eab3a,WrapU_V<sceDisplayGetAccumulatedHcount>,"sceDisplayGetAccumulatedHcount"},
|
|
|
|
{0xA83EF139,WrapU_V<sceDisplayAdjustAccumulatedHcount>,"sceDisplayAdjustAccumulatedHcount"},
|
2012-12-17 22:52:31 +01:00
|
|
|
{0x9C6EAAD7,WrapU_V<sceDisplayGetVcount>,"sceDisplayGetVcount"},
|
2013-03-05 21:31:13 +08:00
|
|
|
{0xDEA197D4,WrapU_UUU<sceDisplayGetMode>,"sceDisplayGetMode"},
|
2013-04-30 21:29:18 +08:00
|
|
|
{0x7ED59BC4,WrapU_U<sceDisplaySetHoldMode>,"sceDisplaySetHoldMode"},
|
|
|
|
{0xA544C486,WrapU_U<sceDisplaySetResumeMode>,"sceDisplaySetResumeMode"},
|
|
|
|
{0xBF79F646,WrapU_U<sceDisplayGetResumeMode>,"sceDisplayGetResumeMode"},
|
2013-03-03 12:39:37 +01:00
|
|
|
{0xB4F378FA,WrapU_V<sceDisplayIsForeground>,"sceDisplayIsForeground"},
|
2013-04-30 21:29:18 +08:00
|
|
|
{0x31C4BAA8,WrapU_U<sceDisplayGetBrightness>,"sceDisplayGetBrightness"},
|
2013-03-05 21:34:22 +08:00
|
|
|
{0x4D4E10EC,WrapU_V<sceDisplayIsVblank>,"sceDisplayIsVblank"},
|
2013-04-30 21:29:18 +08:00
|
|
|
{0x21038913,WrapU_V<sceDisplayIsVsync>,"sceDisplayIsVsync"},
|
2012-11-01 16:19:01 +01:00
|
|
|
};
|
|
|
|
|
2012-12-25 15:28:34 +01:00
|
|
|
void Register_sceDisplay() {
|
2012-11-01 16:19:01 +01:00
|
|
|
RegisterModule("sceDisplay", ARRAY_SIZE(sceDisplay), sceDisplay);
|
|
|
|
}
|