ppsspp/GPU/Debugger/Debugger.cpp
Henrik Rydgård e01ca5b057
Logging API change (refactor) (#19324)
* Rename LogType to Log

* Explicitly use the Log:: enum when logging. Allows for autocomplete when editing.

* Mac/ARM64 buildfix

* Do the same with the hle result log macros

* Rename the log names to mixed case while at it.

* iOS buildfix

* Qt buildfix attempt, ARM32 buildfix
2024-07-14 14:42:59 +02:00

276 lines
7.1 KiB
C++

// Copyright (c) 2018- 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.
// 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 "Common/Log.h"
#include "Common/StringUtils.h"
#include "Common/TimeUtil.h"
#include "GPU/GPU.h"
#include "GPU/Debugger/Breakpoints.h"
#include "GPU/Debugger/Debugger.h"
#include "GPU/Debugger/Stepping.h"
namespace GPUDebug {
static bool active = false;
static bool inited = false;
static BreakNext breakNext = BreakNext::NONE;
static int breakAtCount = -1;
static bool hasBreakpoints = false;
static int primsLastFrame = 0;
static int primsThisFrame = 0;
static int thisFlipNum = 0;
static double lastStepTime = -1.0;
static std::vector<std::pair<int, int>> restrictPrimRanges;
static std::string restrictPrimRule;
static void Init() {
if (!inited) {
GPUBreakpoints::Init([](bool flag) {
hasBreakpoints = flag;
});
Core_ListenStopRequest(&GPUStepping::ForceUnpause);
inited = true;
}
}
void SetActive(bool flag) {
Init();
active = flag;
if (!active) {
breakNext = BreakNext::NONE;
breakAtCount = -1;
GPUStepping::ResumeFromStepping();
lastStepTime = -1.0;
}
}
bool IsActive() {
return active;
}
void SetBreakNext(BreakNext next) {
SetActive(true);
breakNext = next;
breakAtCount = -1;
if (next == BreakNext::TEX) {
GPUBreakpoints::AddTextureChangeTempBreakpoint();
} else if (next == BreakNext::PRIM || next == BreakNext::COUNT) {
GPUBreakpoints::AddCmdBreakpoint(GE_CMD_PRIM, true);
GPUBreakpoints::AddCmdBreakpoint(GE_CMD_BEZIER, true);
GPUBreakpoints::AddCmdBreakpoint(GE_CMD_SPLINE, true);
GPUBreakpoints::AddCmdBreakpoint(GE_CMD_VAP, true);
} else if (next == BreakNext::CURVE) {
GPUBreakpoints::AddCmdBreakpoint(GE_CMD_BEZIER, true);
GPUBreakpoints::AddCmdBreakpoint(GE_CMD_SPLINE, true);
}
GPUStepping::ResumeFromStepping();
lastStepTime = next == BreakNext::NONE ? -1.0 : time_now_d();
}
void SetBreakCount(int c, bool relative) {
if (relative) {
breakAtCount = primsThisFrame + c;
} else {
breakAtCount = c;
}
}
static bool IsBreakpoint(u32 pc, u32 op) {
if (breakNext == BreakNext::OP) {
return true;
} else if (breakNext == BreakNext::COUNT) {
return primsThisFrame == breakAtCount;
} else if (hasBreakpoints) {
return GPUBreakpoints::IsBreakpoint(pc, op);
}
return false;
}
bool NotifyCommand(u32 pc) {
if (!active)
return true;
u32 op = Memory::ReadUnchecked_U32(pc);
u32 cmd = op >> 24;
if (thisFlipNum != gpuStats.numFlips) {
primsLastFrame = primsThisFrame;
primsThisFrame = 0;
thisFlipNum = gpuStats.numFlips;
}
bool process = true;
if (cmd == GE_CMD_PRIM || cmd == GE_CMD_BEZIER || cmd == GE_CMD_SPLINE || cmd == GE_CMD_VAP) {
primsThisFrame++;
if (!restrictPrimRanges.empty()) {
process = false;
for (const auto &range : restrictPrimRanges) {
if (primsThisFrame >= range.first && primsThisFrame <= range.second) {
process = true;
break;
}
}
}
}
if (IsBreakpoint(pc, op)) {
GPUBreakpoints::ClearTempBreakpoints();
if (coreState == CORE_POWERDOWN || !gpuDebug) {
breakNext = BreakNext::NONE;
return process;
}
auto info = gpuDebug->DissassembleOp(pc);
if (lastStepTime >= 0.0) {
NOTICE_LOG(Log::G3D, "Waiting at %08x, %s (%fms)", pc, info.desc.c_str(), (time_now_d() - lastStepTime) * 1000.0);
lastStepTime = -1.0;
} else {
NOTICE_LOG(Log::G3D, "Waiting at %08x, %s", pc, info.desc.c_str());
}
GPUStepping::EnterStepping();
}
return process;
}
void NotifyDraw() {
if (!active)
return;
if (breakNext == BreakNext::DRAW && !GPUStepping::IsStepping()) {
if (lastStepTime >= 0.0) {
NOTICE_LOG(Log::G3D, "Waiting at a draw (%fms)", (time_now_d() - lastStepTime) * 1000.0);
lastStepTime = -1.0;
} else {
NOTICE_LOG(Log::G3D, "Waiting at a draw");
}
GPUStepping::EnterStepping();
}
}
void NotifyDisplay(u32 framebuf, u32 stride, int format) {
if (!active)
return;
if (breakNext == BreakNext::FRAME) {
// This should work fine, start stepping at the first op of the new frame.
breakNext = BreakNext::OP;
}
}
void NotifyBeginFrame() {
if (!active)
return;
if (breakNext == BreakNext::VSYNC) {
// Just start stepping as soon as we can once the vblank finishes.
breakNext = BreakNext::OP;
}
}
int PrimsThisFrame() {
return primsThisFrame;
}
int PrimsLastFrame() {
return primsLastFrame;
}
static bool ParseRange(const std::string &s, std::pair<int, int> &range) {
int c = sscanf(s.c_str(), "%d-%d", &range.first, &range.second);
if (c == 0)
return false;
if (c == 1)
range.second = range.first;
return true;
}
bool SetRestrictPrims(const char *rule) {
SetActive(true);
if (rule == nullptr || rule[0] == 0 || (rule[0] == '*' && rule[1] == 0)) {
restrictPrimRanges.clear();
restrictPrimRule.clear();
return true;
}
static constexpr int MAX_PRIMS = 0x7FFFFFFF;
std::vector<std::string> parts;
SplitString(rule, ',', parts);
// Parse expressions like: 0 or 0-1,4-5 or !2 or !2-3 or !2,!3
std::vector<std::pair<int, int>> updated;
for (auto &part : parts) {
std::pair<int, int> range;
if (part.size() > 1 && part[0] == '!') {
if (!ParseRange(part.substr(1), range))
return false;
// If there's nothing yet, add everything else.
if (updated.empty()) {
if (range.first > 0)
updated.emplace_back(0, range.first - 1);
if (range.second < MAX_PRIMS)
updated.emplace_back(range.second + 1, MAX_PRIMS);
continue;
}
// Otherwise, remove this range from any existing.
for (size_t i = 0; i < updated.size(); ++i) {
auto &sub = updated[i];
if (sub.second < range.first || sub.first > range.second)
continue;
if (sub.first >= range.first && sub.second <= range.second) {
// Entire subrange is inside the deleted entries, nuke.
sub.first = -1;
sub.second = -1;
continue;
}
if (sub.first < range.first && sub.second > range.second) {
// We're slicing a hole in this subrange.
int next = sub.second;
sub.second = range.first - 1;
updated.emplace_back(range.second + 1, next);
continue;
}
// If we got here, we're simply clipping the subrange.
if (sub.first < range.first && sub.second >= range.first && sub.second <= range.second)
sub.second = range.first - 1;
if (sub.first >= range.first && sub.first <= range.second && sub.second < range.second)
sub.first = range.second + 1;
}
} else {
if (!ParseRange(part, range))
return false;
updated.push_back(range);
}
}
restrictPrimRanges = updated;
restrictPrimRule = rule;
return true;
}
const char *GetRestrictPrims() {
return restrictPrimRule.c_str();
}
}