mirror of
https://github.com/yuzu-emu/breakpad.git
synced 2025-01-11 07:35:33 +00:00
CrashGenerationServer's state machine can be invoked from both the application
thread and thread pool threads. This CL serializes access to the FSM state. Handling of crash dump and client shutdown requests is still done asynchronously. Patch by Alex Pakhunov <alexeypa@chromium.org> BUG=132164 TEST=remoting_unittests.BreakpadWinDeathTest.* Review URL: https://breakpad.appspot.com/396002/ git-svn-id: http://google-breakpad.googlecode.com/svn/trunk@970 4c0a9323-5329-0410-9bdc-e9ce6186880e
This commit is contained in:
parent
fd67ff286e
commit
c50346b341
|
@ -40,14 +40,31 @@ class AutoCriticalSection {
|
||||||
public:
|
public:
|
||||||
// Creates a new instance with the given critical section object
|
// Creates a new instance with the given critical section object
|
||||||
// and enters the critical section immediately.
|
// and enters the critical section immediately.
|
||||||
explicit AutoCriticalSection(CRITICAL_SECTION* cs) : cs_(cs) {
|
explicit AutoCriticalSection(CRITICAL_SECTION* cs) : cs_(cs), taken_(false) {
|
||||||
assert(cs_);
|
assert(cs_);
|
||||||
EnterCriticalSection(cs_);
|
Acquire();
|
||||||
}
|
}
|
||||||
|
|
||||||
// Destructor: leaves the critical section.
|
// Destructor: leaves the critical section.
|
||||||
~AutoCriticalSection() {
|
~AutoCriticalSection() {
|
||||||
|
if (taken_) {
|
||||||
|
Release();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Enters the critical section. Recursive Acquire() calls are not allowed.
|
||||||
|
void Acquire() {
|
||||||
|
assert(!taken_);
|
||||||
|
EnterCriticalSection(cs_);
|
||||||
|
taken_ = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Leaves the critical section. The caller should not call Release() unless
|
||||||
|
// the critical seciton has been entered already.
|
||||||
|
void Release() {
|
||||||
|
assert(taken_);
|
||||||
LeaveCriticalSection(cs_);
|
LeaveCriticalSection(cs_);
|
||||||
|
taken_ = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
@ -56,6 +73,7 @@ class AutoCriticalSection {
|
||||||
AutoCriticalSection& operator=(const AutoCriticalSection&);
|
AutoCriticalSection& operator=(const AutoCriticalSection&);
|
||||||
|
|
||||||
CRITICAL_SECTION* cs_;
|
CRITICAL_SECTION* cs_;
|
||||||
|
bool taken_;
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace google_breakpad
|
} // namespace google_breakpad
|
||||||
|
|
|
@ -125,7 +125,7 @@ CrashGenerationServer::CrashGenerationServer(
|
||||||
overlapped_(),
|
overlapped_(),
|
||||||
client_info_(NULL),
|
client_info_(NULL),
|
||||||
cleanup_item_count_(0) {
|
cleanup_item_count_(0) {
|
||||||
InitializeCriticalSection(&clients_sync_);
|
InitializeCriticalSection(&sync_);
|
||||||
|
|
||||||
if (dump_path) {
|
if (dump_path) {
|
||||||
dump_generator_.reset(new MinidumpGenerator(*dump_path));
|
dump_generator_.reset(new MinidumpGenerator(*dump_path));
|
||||||
|
@ -133,20 +133,27 @@ CrashGenerationServer::CrashGenerationServer(
|
||||||
}
|
}
|
||||||
|
|
||||||
CrashGenerationServer::~CrashGenerationServer() {
|
CrashGenerationServer::~CrashGenerationServer() {
|
||||||
|
// New scope to release the lock automatically.
|
||||||
|
{
|
||||||
|
AutoCriticalSection lock(&sync_);
|
||||||
|
|
||||||
// Indicate to existing threads that server is shutting down.
|
// Indicate to existing threads that server is shutting down.
|
||||||
shutting_down_ = true;
|
shutting_down_ = true;
|
||||||
|
|
||||||
// Even if there are no current worker threads running, it is possible that
|
// Even if there are no current worker threads running, it is possible that
|
||||||
// an I/O request is pending on the pipe right now but not yet done. In fact,
|
// an I/O request is pending on the pipe right now but not yet done.
|
||||||
// it's very likely this is the case unless we are in an ERROR state. If we
|
// In fact, it's very likely this is the case unless we are in an ERROR
|
||||||
// don't wait for the pending I/O to be done, then when the I/O completes,
|
// state. If we don't wait for the pending I/O to be done, then when the I/O
|
||||||
// it may write to invalid memory. AppVerifier will flag this problem too.
|
// completes, it may write to invalid memory. AppVerifier will flag this
|
||||||
// So we disconnect from the pipe and then wait for the server to get into
|
// problem too. So we disconnect from the pipe and then wait for the server
|
||||||
// error state so that the pending I/O will fail and get cleared.
|
// to get into error state so that the pending I/O will fail and get
|
||||||
|
// cleared.
|
||||||
DisconnectNamedPipe(pipe_);
|
DisconnectNamedPipe(pipe_);
|
||||||
int num_tries = 100;
|
int num_tries = 100;
|
||||||
while (num_tries-- && server_state_ != IPC_SERVER_STATE_ERROR) {
|
while (num_tries-- && server_state_ != IPC_SERVER_STATE_ERROR) {
|
||||||
|
lock.Release();
|
||||||
Sleep(10);
|
Sleep(10);
|
||||||
|
lock.Acquire();
|
||||||
}
|
}
|
||||||
|
|
||||||
// Unregister wait on the pipe.
|
// Unregister wait on the pipe.
|
||||||
|
@ -161,10 +168,6 @@ CrashGenerationServer::~CrashGenerationServer() {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Request all ClientInfo objects to unregister all waits.
|
// Request all ClientInfo objects to unregister all waits.
|
||||||
// New scope to hold the lock for the shortest time.
|
|
||||||
{
|
|
||||||
AutoCriticalSection lock(&clients_sync_);
|
|
||||||
|
|
||||||
std::list<ClientInfo*>::iterator iter;
|
std::list<ClientInfo*>::iterator iter;
|
||||||
for (iter = clients_.begin(); iter != clients_.end(); ++iter) {
|
for (iter = clients_.begin(); iter != clients_.end(); ++iter) {
|
||||||
ClientInfo* client_info = *iter;
|
ClientInfo* client_info = *iter;
|
||||||
|
@ -185,17 +188,16 @@ CrashGenerationServer::~CrashGenerationServer() {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Clean up all the ClientInfo objects.
|
|
||||||
// New scope to hold the lock for the shortest time.
|
// New scope to hold the lock for the shortest time.
|
||||||
{
|
{
|
||||||
AutoCriticalSection lock(&clients_sync_);
|
AutoCriticalSection lock(&sync_);
|
||||||
|
|
||||||
|
// Clean up all the ClientInfo objects.
|
||||||
std::list<ClientInfo*>::iterator iter;
|
std::list<ClientInfo*>::iterator iter;
|
||||||
for (iter = clients_.begin(); iter != clients_.end(); ++iter) {
|
for (iter = clients_.begin(); iter != clients_.end(); ++iter) {
|
||||||
ClientInfo* client_info = *iter;
|
ClientInfo* client_info = *iter;
|
||||||
delete client_info;
|
delete client_info;
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
if (server_alive_handle_) {
|
if (server_alive_handle_) {
|
||||||
// Release the mutex before closing the handle so that clients requesting
|
// Release the mutex before closing the handle so that clients requesting
|
||||||
|
@ -207,11 +209,14 @@ CrashGenerationServer::~CrashGenerationServer() {
|
||||||
if (overlapped_.hEvent) {
|
if (overlapped_.hEvent) {
|
||||||
CloseHandle(overlapped_.hEvent);
|
CloseHandle(overlapped_.hEvent);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
DeleteCriticalSection(&clients_sync_);
|
DeleteCriticalSection(&sync_);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool CrashGenerationServer::Start() {
|
bool CrashGenerationServer::Start() {
|
||||||
|
AutoCriticalSection lock(&sync_);
|
||||||
|
|
||||||
if (server_state_ != IPC_SERVER_STATE_UNINITIALIZED) {
|
if (server_state_ != IPC_SERVER_STATE_UNINITIALIZED) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
@ -682,6 +687,8 @@ bool CrashGenerationServer::RespondToClient(ClientInfo* client_info) {
|
||||||
// implements the state machine described in ReadMe.txt along with the
|
// implements the state machine described in ReadMe.txt along with the
|
||||||
// helper methods HandleXXXState.
|
// helper methods HandleXXXState.
|
||||||
void CrashGenerationServer::HandleConnectionRequest() {
|
void CrashGenerationServer::HandleConnectionRequest() {
|
||||||
|
AutoCriticalSection lock(&sync_);
|
||||||
|
|
||||||
// If we are shutting doen then get into ERROR state, reset the event so more
|
// If we are shutting doen then get into ERROR state, reset the event so more
|
||||||
// workers don't run and return immediately.
|
// workers don't run and return immediately.
|
||||||
if (shutting_down_) {
|
if (shutting_down_) {
|
||||||
|
@ -768,7 +775,7 @@ bool CrashGenerationServer::AddClient(ClientInfo* client_info) {
|
||||||
|
|
||||||
// New scope to hold the lock for the shortest time.
|
// New scope to hold the lock for the shortest time.
|
||||||
{
|
{
|
||||||
AutoCriticalSection lock(&clients_sync_);
|
AutoCriticalSection lock(&sync_);
|
||||||
clients_.push_back(client_info);
|
clients_.push_back(client_info);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -836,7 +843,7 @@ void CrashGenerationServer::DoCleanup(ClientInfo* client_info) {
|
||||||
|
|
||||||
// Start a new scope to release lock automatically.
|
// Start a new scope to release lock automatically.
|
||||||
{
|
{
|
||||||
AutoCriticalSection lock(&clients_sync_);
|
AutoCriticalSection lock(&sync_);
|
||||||
clients_.remove(client_info);
|
clients_.remove(client_info);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -216,8 +216,9 @@ class CrashGenerationServer {
|
||||||
// asynchronous IO operation.
|
// asynchronous IO operation.
|
||||||
void EnterStateWhenSignaled(IPCServerState state);
|
void EnterStateWhenSignaled(IPCServerState state);
|
||||||
|
|
||||||
// Sync object for thread-safe access to the shared list of clients.
|
// Sync object for thread-safe access to the shared list of clients and
|
||||||
CRITICAL_SECTION clients_sync_;
|
// the server's state.
|
||||||
|
CRITICAL_SECTION sync_;
|
||||||
|
|
||||||
// List of clients.
|
// List of clients.
|
||||||
std::list<ClientInfo*> clients_;
|
std::list<ClientInfo*> clients_;
|
||||||
|
@ -271,10 +272,10 @@ class CrashGenerationServer {
|
||||||
// Note that since we restrict the pipe to one instance, we
|
// Note that since we restrict the pipe to one instance, we
|
||||||
// only need to keep one state of the server. Otherwise, server
|
// only need to keep one state of the server. Otherwise, server
|
||||||
// would have one state per client it is talking to.
|
// would have one state per client it is talking to.
|
||||||
volatile IPCServerState server_state_;
|
IPCServerState server_state_;
|
||||||
|
|
||||||
// Whether the server is shutting down.
|
// Whether the server is shutting down.
|
||||||
volatile bool shutting_down_;
|
bool shutting_down_;
|
||||||
|
|
||||||
// Overlapped instance for async I/O on the pipe.
|
// Overlapped instance for async I/O on the pipe.
|
||||||
OVERLAPPED overlapped_;
|
OVERLAPPED overlapped_;
|
||||||
|
|
Loading…
Reference in a new issue