Minor correction to Signals implementation.

Patch submitted by Jeff Cohen. Thanks Jeff!

llvm-svn: 16401
This commit is contained in:
Reid Spencer 2004-09-19 05:37:39 +00:00
parent 9864df96ba
commit 1bdd0f0a08
1 changed files with 24 additions and 14 deletions

View File

@ -28,6 +28,8 @@ static BOOL WINAPI LLVMConsoleCtrlHandler(DWORD dwCtrlType);
static std::vector<std::string> *FilesToRemove = NULL; static std::vector<std::string> *FilesToRemove = NULL;
static std::vector<llvm::sys::Path> *DirectoriesToRemove = NULL; static std::vector<llvm::sys::Path> *DirectoriesToRemove = NULL;
static bool RegisteredUnhandledExceptionFilter = false; static bool RegisteredUnhandledExceptionFilter = false;
static bool CleanupExecuted = false;
static PTOP_LEVEL_EXCEPTION_FILTER OldFilter = NULL;
// Windows creates a new thread to execute the console handler when an event // Windows creates a new thread to execute the console handler when an event
// (such as CTRL/C) occurs. This causes concurrency issues with the above // (such as CTRL/C) occurs. This causes concurrency issues with the above
@ -43,8 +45,7 @@ namespace llvm {
static void RegisterHandler() { static void RegisterHandler() {
if (RegisteredUnhandledExceptionFilter) if (RegisteredUnhandledExceptionFilter) {
{
EnterCriticalSection(&CriticalSection); EnterCriticalSection(&CriticalSection);
return; return;
} }
@ -58,7 +59,7 @@ static void RegisterHandler() {
EnterCriticalSection(&CriticalSection); EnterCriticalSection(&CriticalSection);
RegisteredUnhandledExceptionFilter = true; RegisteredUnhandledExceptionFilter = true;
SetUnhandledExceptionFilter(LLVMUnhandledExceptionFilter); OldFilter = SetUnhandledExceptionFilter(LLVMUnhandledExceptionFilter);
SetConsoleCtrlHandler(LLVMConsoleCtrlHandler, TRUE); SetConsoleCtrlHandler(LLVMConsoleCtrlHandler, TRUE);
// IMPORTANT NOTE: Caller must call LeaveCriticalSection(&CriticalSection) or // IMPORTANT NOTE: Caller must call LeaveCriticalSection(&CriticalSection) or
@ -69,6 +70,9 @@ static void RegisterHandler() {
void sys::RemoveFileOnSignal(const std::string &Filename) { void sys::RemoveFileOnSignal(const std::string &Filename) {
RegisterHandler(); RegisterHandler();
if (CleanupExecuted)
throw std::string("Process terminating -- cannot register for removal");
if (FilesToRemove == NULL) if (FilesToRemove == NULL)
FilesToRemove = new std::vector<std::string>; FilesToRemove = new std::vector<std::string>;
@ -81,6 +85,9 @@ void sys::RemoveFileOnSignal(const std::string &Filename) {
void sys::RemoveDirectoryOnSignal(const sys::Path& path) { void sys::RemoveDirectoryOnSignal(const sys::Path& path) {
RegisterHandler(); RegisterHandler();
if (CleanupExecuted)
throw std::string("Process terminating -- cannot register for removal");
if (path.is_directory()) { if (path.is_directory()) {
if (DirectoriesToRemove == NULL) if (DirectoriesToRemove == NULL)
DirectoriesToRemove = new std::vector<sys::Path>; DirectoriesToRemove = new std::vector<sys::Path>;
@ -103,6 +110,12 @@ void sys::PrintStackTraceOnErrorSignal() {
static void Cleanup() { static void Cleanup() {
EnterCriticalSection(&CriticalSection); EnterCriticalSection(&CriticalSection);
// Prevent other thread from registering new files and directories for
// removal, should we be executing because of the console handler callback.
CleanupExecuted = true;
// FIXME: open files cannot be deleted.
if (FilesToRemove != NULL) if (FilesToRemove != NULL)
while (!FilesToRemove->empty()) { while (!FilesToRemove->empty()) {
try { try {
@ -144,7 +157,7 @@ static LONG WINAPI LLVMUnhandledExceptionFilter(LPEXCEPTION_POINTERS ep) {
// Initialize the symbol handler. // Initialize the symbol handler.
SymSetOptions(SYMOPT_DEFERRED_LOADS|SYMOPT_LOAD_LINES); SymSetOptions(SYMOPT_DEFERRED_LOADS|SYMOPT_LOAD_LINES);
SymInitialize(GetCurrentProcess(), NULL, TRUE); SymInitialize(hProcess, NULL, TRUE);
while (true) { while (true) {
if (!StackWalk(IMAGE_FILE_MACHINE_I386, hProcess, hThread, &StackFrame, if (!StackWalk(IMAGE_FILE_MACHINE_I386, hProcess, hThread, &StackFrame,
@ -158,7 +171,7 @@ static LONG WINAPI LLVMUnhandledExceptionFilter(LPEXCEPTION_POINTERS ep) {
// Print the PC in hexadecimal. // Print the PC in hexadecimal.
DWORD PC = StackFrame.AddrPC.Offset; DWORD PC = StackFrame.AddrPC.Offset;
fprintf(stderr, "%04X:%08X", ep->ContextRecord->SegCs, PC); fprintf(stderr, "%08X", PC);
// Print the parameters. Assume there are four. // Print the parameters. Assume there are four.
fprintf(stderr, " (0x%08X 0x%08X 0x%08X 0x%08X)", StackFrame.Params[0], fprintf(stderr, " (0x%08X 0x%08X 0x%08X 0x%08X)", StackFrame.Params[0],
@ -166,7 +179,7 @@ static LONG WINAPI LLVMUnhandledExceptionFilter(LPEXCEPTION_POINTERS ep) {
// Verify the PC belongs to a module in this process. // Verify the PC belongs to a module in this process.
if (!SymGetModuleBase(hProcess, PC)) { if (!SymGetModuleBase(hProcess, PC)) {
fputc('\n', stderr); fputs(" <unknown module>\n", stderr);
continue; continue;
} }
@ -201,21 +214,18 @@ static LONG WINAPI LLVMUnhandledExceptionFilter(LPEXCEPTION_POINTERS ep) {
fputc('\n', stderr); fputc('\n', stderr);
} }
} } catch (...) {
catch (...)
{
assert(!"Crashed in LLVMUnhandledExceptionFilter"); assert(!"Crashed in LLVMUnhandledExceptionFilter");
} }
// Allow dialog box to pop up allowing choice to start debugger. // Allow dialog box to pop up allowing choice to start debugger.
return EXCEPTION_CONTINUE_SEARCH; if (OldFilter)
return (*OldFilter)(ep);
else
return EXCEPTION_CONTINUE_SEARCH;
} }
static BOOL WINAPI LLVMConsoleCtrlHandler(DWORD dwCtrlType) { static BOOL WINAPI LLVMConsoleCtrlHandler(DWORD dwCtrlType) {
// FIXME: This handler executes on a different thread. The main thread
// is still running, potentially creating new files to be cleaned up
// in the tiny window between the call to Cleanup() and process termination.
// Also, any files currently open cannot be deleted.
Cleanup(); Cleanup();
// Allow normal processing to take place; i.e., the process dies. // Allow normal processing to take place; i.e., the process dies.