Skip to content

Commit

Permalink
Debugger: Lock startup/shutdown for threadsafety.
Browse files Browse the repository at this point in the history
Otherwise things can get freed while we're trying to inspect them.
  • Loading branch information
unknownbrackets committed Apr 24, 2018
1 parent 84ba2f4 commit 19bf79f
Show file tree
Hide file tree
Showing 8 changed files with 81 additions and 23 deletions.
9 changes: 4 additions & 5 deletions Core/Core.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -55,7 +55,7 @@ static std::condition_variable m_InactiveCond;
static std::mutex m_hInactiveMutex;
static bool singleStepPending = false;
static int steppingCounter = 0;
static std::set<Core_ShutdownFunc> shutdownFuncs;
static std::set<CoreLifecycleFunc> shutdownFuncs;
static bool windowHidden = false;
static double lastActivity = 0.0;
static double lastKeepAwake = 0.0;
Expand All @@ -76,13 +76,13 @@ void Core_NotifyActivity() {
lastActivity = time_now_d();
}

void Core_ListenShutdown(Core_ShutdownFunc func) {
void Core_ListenLifecycle(CoreLifecycleFunc func) {
shutdownFuncs.insert(func);
}

void Core_NotifyShutdown() {
void Core_NotifyLifecycle(CoreLifecycle stage) {
for (auto it = shutdownFuncs.begin(); it != shutdownFuncs.end(); ++it) {
(*it)();
(*it)(stage);
}
}

Expand All @@ -98,7 +98,6 @@ void Core_Halt(const char *msg) {

void Core_Stop() {
Core_UpdateState(CORE_POWERDOWN);
Core_NotifyShutdown();
m_StepCond.notify_all();
}

Expand Down
15 changes: 12 additions & 3 deletions Core/Core.h
Original file line number Diff line number Diff line change
Expand Up @@ -38,9 +38,18 @@ void Core_UpdateSingleStep();
// Changes every time we enter stepping.
int Core_GetSteppingCounter();

typedef void (* Core_ShutdownFunc)();
void Core_ListenShutdown(Core_ShutdownFunc func);
void Core_NotifyShutdown();
enum class CoreLifecycle {
STARTING,
// Note: includes failure cases. Guaranteed call after STARTING.
START_COMPLETE,
STOPPING,
// Guaranteed call after STOPPING.
STOPPED,
};

typedef void (* CoreLifecycleFunc)(CoreLifecycle stage);
void Core_ListenLifecycle(CoreLifecycleFunc func);
void Core_NotifyLifecycle(CoreLifecycle stage);
void Core_Halt(const char *msg);

bool Core_IsStepping();
Expand Down
36 changes: 36 additions & 0 deletions Core/Debugger/WebSocket.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -68,19 +68,51 @@ static volatile bool stopRequested = false;
static std::mutex stopLock;
static std::condition_variable stopCond;

// Prevent threading surprises and obscure crashes by locking startup/shutdown.
static bool lifecycleLockSetup = false;
static std::mutex lifecycleLock;

static void UpdateConnected(int delta) {
std::lock_guard<std::mutex> guard(stopLock);
debuggersConnected += delta;
stopCond.notify_all();
}

static void WebSocketNotifyLifecycle(CoreLifecycle stage) {
switch (stage) {
case CoreLifecycle::STARTING:
case CoreLifecycle::STOPPING:
if (debuggersConnected > 0) {
DEBUG_LOG(SYSTEM, "Waiting for debugger to complete on shutdown");
}
lifecycleLock.lock();
break;

case CoreLifecycle::START_COMPLETE:
case CoreLifecycle::STOPPED:
lifecycleLock.unlock();
if (debuggersConnected > 0) {
DEBUG_LOG(SYSTEM, "Debugger ready for shutdown");
}
break;
}
}

static void SetupDebuggerLock() {
if (!lifecycleLockSetup) {
Core_ListenLifecycle(&WebSocketNotifyLifecycle);
lifecycleLockSetup = true;
}
}

void HandleDebuggerRequest(const http::Request &request) {
net::WebSocketServer *ws = net::WebSocketServer::CreateAsUpgrade(request, "debugger.ppsspp.org");
if (!ws)
return;

setCurrentThreadName("Debugger");
UpdateConnected(1);
SetupDebuggerLock();

LogBroadcaster logger;
GameBroadcaster game;
Expand All @@ -89,6 +121,7 @@ void HandleDebuggerRequest(const http::Request &request) {
std::unordered_map<std::string, DebuggerEventHandler> eventHandlers;
std::vector<void *> subscriberData;
for (auto info : subscribers) {
std::lock_guard<std::mutex> guard(lifecycleLock);
subscriberData.push_back(info.init(eventHandlers));
}

Expand All @@ -109,6 +142,7 @@ void HandleDebuggerRequest(const http::Request &request) {
DebuggerRequest req(event, ws, root);
auto eventFunc = eventHandlers.find(event);
if (eventFunc != eventHandlers.end()) {
std::lock_guard<std::mutex> guard(lifecycleLock);
eventFunc->second(req);
req.Finish();
} else {
Expand All @@ -120,6 +154,7 @@ void HandleDebuggerRequest(const http::Request &request) {
});

while (ws->Process(1.0f / 60.0f)) {
std::lock_guard<std::mutex> guard(lifecycleLock);
// These send events that aren't just responses to requests.
logger.Broadcast(ws);
game.Broadcast(ws);
Expand All @@ -130,6 +165,7 @@ void HandleDebuggerRequest(const http::Request &request) {
}
}

std::lock_guard<std::mutex> guard(lifecycleLock);
for (size_t i = 0; i < subscribers.size(); ++i) {
if (subscribers[i].shutdown) {
subscribers[i].shutdown(subscriberData[i]);
Expand Down
10 changes: 6 additions & 4 deletions Core/HLE/sceIo.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -497,10 +497,12 @@ static void __IoManagerThread() {
}
}

static void __IoWakeManager() {
static void __IoWakeManager(CoreLifecycle stage) {
// Ping the thread so that it knows to check coreState.
ioManagerThreadEnabled = false;
ioManager.FinishEventLoop();
if (stage == CoreLifecycle::STOPPING) {
ioManagerThreadEnabled = false;
ioManager.FinishEventLoop();
}
}

static void __IoVblank() {
Expand Down Expand Up @@ -587,7 +589,7 @@ void __IoInit() {
ioManagerThreadEnabled = g_Config.bSeparateIOThread;
ioManager.SetThreadEnabled(ioManagerThreadEnabled);
if (ioManagerThreadEnabled) {
Core_ListenShutdown(&__IoWakeManager);
Core_ListenLifecycle(&__IoWakeManager);
ioManagerThread = new std::thread(&__IoManagerThread);
ioManagerThread->detach();
}
Expand Down
19 changes: 14 additions & 5 deletions Core/System.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -306,6 +306,7 @@ bool PSP_InitStart(const CoreParameter &coreParam, std::string *error_string) {
INFO_LOG(BOOT, "PPSSPP %s", PPSSPP_GIT_VERSION);
#endif

Core_NotifyLifecycle(CoreLifecycle::STARTING);
GraphicsContext *temp = coreParameter.graphicsContext;
coreParameter = coreParam;
if (coreParameter.graphicsContext == nullptr) {
Expand All @@ -320,6 +321,7 @@ bool PSP_InitStart(const CoreParameter &coreParam, std::string *error_string) {
*error_string = coreParameter.errorString;
bool success = coreParameter.fileToStart != "";
if (!success) {
Core_NotifyLifecycle(CoreLifecycle::START_COMPLETE);
pspIsIniting = false;
}
return success;
Expand All @@ -346,6 +348,9 @@ bool PSP_InitUpdate(std::string *error_string) {
}
pspIsInited = success && GPU_IsReady();
pspIsIniting = success && !pspIsInited;
if (!pspIsIniting) {
Core_NotifyLifecycle(CoreLifecycle::START_COMPLETE);
}
return !success || pspIsInited;
}

Expand Down Expand Up @@ -375,17 +380,20 @@ void PSP_Shutdown() {
return;
}

// Make sure things know right away that PSP memory, etc. is going away.
pspIsQuitting = true;
if (coreState == CORE_RUNNING)
Core_UpdateState(CORE_ERROR);

#ifndef MOBILE_DEVICE
if (g_Config.bFuncHashMap) {
MIPSAnalyst::StoreHashMap();
}
#endif

// Make sure things know right away that PSP memory, etc. is going away.
pspIsQuitting = true;
if (coreState == CORE_RUNNING)
Core_UpdateState(CORE_ERROR);
Core_NotifyShutdown();
if (pspIsIniting)
Core_NotifyLifecycle(CoreLifecycle::START_COMPLETE);
Core_NotifyLifecycle(CoreLifecycle::STOPPING);
CPU_Shutdown();
GPU_Shutdown();
g_paramSFO.Clear();
Expand All @@ -395,6 +403,7 @@ void PSP_Shutdown() {
pspIsIniting = false;
pspIsQuitting = false;
g_Config.unloadGameConfig();
Core_NotifyLifecycle(CoreLifecycle::STOPPED);
}

void PSP_BeginHostFrame() {
Expand Down
10 changes: 6 additions & 4 deletions GPU/Debugger/Stepping.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -197,10 +197,12 @@ void ResumeFromStepping() {
SetPauseAction(PAUSE_CONTINUE, false);
}

void ForceUnpause() {
SetPauseAction(PAUSE_CONTINUE, false);
actionComplete = true;
actionWait.notify_all();
void ForceUnpause(CoreLifecycle stage) {
if (stage == CoreLifecycle::STOPPING) {
SetPauseAction(PAUSE_CONTINUE, false);
actionComplete = true;
actionWait.notify_all();
}
}

} // namespace
3 changes: 2 additions & 1 deletion GPU/Debugger/Stepping.h
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,7 @@
#include <functional>

#include "Common/CommonTypes.h"
#include "Core/Core.h"
#include "GPU/Common/GPUDebugInterface.h"

namespace GPUStepping {
Expand All @@ -37,5 +38,5 @@ namespace GPUStepping {
bool GPU_SetCmdValue(u32 op);

void ResumeFromStepping();
void ForceUnpause();
void ForceUnpause(CoreLifecycle stage);
};
2 changes: 1 addition & 1 deletion Windows/GEDebugger/GEDebugger.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -67,7 +67,7 @@ void CGEDebugger::Init() {
CGEDebugger::CGEDebugger(HINSTANCE _hInstance, HWND _hParent)
: Dialog((LPCSTR)IDD_GEDEBUGGER, _hInstance, _hParent) {
GPUBreakpoints::Init();
Core_ListenShutdown(ForceUnpause);
Core_ListenLifecycle(ForceUnpause);

// minimum size = a little more than the default
RECT windowRect;
Expand Down

0 comments on commit 19bf79f

Please sign in to comment.