Fix an AppVerifier STOP in OOP server code. In the destructor of
the OOP server, we need to wait for any pending I/O to be done. git-svn-id: http://google-breakpad.googlecode.com/svn/trunk@308 4c0a9323-5329-0410-9bdc-e9ce6186880e
This commit is contained in:
parent
308947d1ce
commit
104e4e0114
2 changed files with 22 additions and 1 deletions
|
@ -129,6 +129,19 @@ CrashGenerationServer::~CrashGenerationServer() {
|
||||||
// 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
|
||||||
|
// an I/O request is pending on the pipe right now but not yet done. In fact,
|
||||||
|
// it's very likely this is the case unless we are in an ERROR state. If we
|
||||||
|
// don't wait for the pending I/O to be done, then when the I/O completes,
|
||||||
|
// it may write to invalid memory. AppVerifier will flag this problem too.
|
||||||
|
// So we disconnect from the pipe and then wait for the server to get into
|
||||||
|
// error state so that the pending I/O will fail and get cleared.
|
||||||
|
DisconnectNamedPipe(pipe_);
|
||||||
|
int num_tries = 100;
|
||||||
|
while (num_tries-- && server_state_ != IPC_SERVER_STATE_ERROR) {
|
||||||
|
Sleep(10);
|
||||||
|
}
|
||||||
|
|
||||||
// Unregister wait on the pipe.
|
// Unregister wait on the pipe.
|
||||||
if (pipe_wait_handle_) {
|
if (pipe_wait_handle_) {
|
||||||
// Wait for already executing callbacks to finish.
|
// Wait for already executing callbacks to finish.
|
||||||
|
@ -629,6 +642,14 @@ 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() {
|
||||||
|
// If we are shutting doen then get into ERROR state, reset the event so more
|
||||||
|
// workers don't run and return immediately.
|
||||||
|
if (shutting_down_) {
|
||||||
|
server_state_ = IPC_SERVER_STATE_ERROR;
|
||||||
|
ResetEvent(overlapped_.hEvent);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
switch (server_state_) {
|
switch (server_state_) {
|
||||||
case IPC_SERVER_STATE_ERROR:
|
case IPC_SERVER_STATE_ERROR:
|
||||||
HandleErrorState();
|
HandleErrorState();
|
||||||
|
|
|
@ -241,7 +241,7 @@ 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.
|
||||||
IPCServerState server_state_;
|
volatile IPCServerState server_state_;
|
||||||
|
|
||||||
// Whether the server is shutting down.
|
// Whether the server is shutting down.
|
||||||
volatile bool shutting_down_;
|
volatile bool shutting_down_;
|
||||||
|
|
Loading…
Reference in a new issue