mirror of
https://github.com/llvm/llvm-project.git
synced 2025-05-12 20:36:07 +00:00

feels a kinship to machine stacks that grow down. Now we get stuff like this: Stack dump: 0. Program arguments: clang clang_crash_Iw2Osj.mi 1. /Developer/SDKs/MacOSX10.5.sdk/usr/lib/gcc/i686-apple-darwin9/4.0.1/include/xmmintrin.h:624:1: parsing function body '_mm_cvtpi16_ps' 2. /Developer/SDKs/MacOSX10.5.sdk/usr/lib/gcc/i686-apple-darwin9/4.0.1/include/xmmintrin.h:624:1: in compound statement ('{}') Abort llvm-svn: 66145
80 lines
2.4 KiB
C++
80 lines
2.4 KiB
C++
//===- PrettyStackTrace.cpp - Pretty Crash Handling -----------------------===//
|
|
//
|
|
// The LLVM Compiler Infrastructure
|
|
//
|
|
// This file is distributed under the University of Illinois Open Source
|
|
// License. See LICENSE.TXT for details.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// This file defines some helpful functions for dealing with the possibility of
|
|
// Unix signals occuring while your program is running.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "llvm/Support/PrettyStackTrace.h"
|
|
#include "llvm/Support/raw_ostream.h"
|
|
#include "llvm/System/Signals.h"
|
|
using namespace llvm;
|
|
|
|
// FIXME: This should be thread local when llvm supports threads.
|
|
static const PrettyStackTraceEntry *PrettyStackTraceHead = 0;
|
|
|
|
static unsigned PrintStack(const PrettyStackTraceEntry *Entry, raw_ostream &OS){
|
|
unsigned NextID = 0;
|
|
if (Entry->getNextEntry())
|
|
NextID = PrintStack(Entry->getNextEntry(), OS);
|
|
OS << NextID << ".\t";
|
|
Entry->print(OS);
|
|
|
|
return NextID+1;
|
|
}
|
|
|
|
/// CrashHandler - This callback is run if a fatal signal is delivered to the
|
|
/// process, it prints the pretty stack trace.
|
|
static void CrashHandler(void *Cookie) {
|
|
// Don't print an empty trace.
|
|
if (PrettyStackTraceHead == 0) return;
|
|
|
|
// If there are pretty stack frames registered, walk and emit them.
|
|
raw_ostream &OS = errs();
|
|
OS << "Stack dump:\n";
|
|
|
|
PrintStack(PrettyStackTraceHead, OS);
|
|
OS.flush();
|
|
}
|
|
|
|
static bool RegisterCrashPrinter() {
|
|
sys::AddSignalHandler(CrashHandler, 0);
|
|
return false;
|
|
}
|
|
|
|
PrettyStackTraceEntry::PrettyStackTraceEntry() {
|
|
// The first time this is called, we register the crash printer.
|
|
static bool HandlerRegistered = RegisterCrashPrinter();
|
|
HandlerRegistered = HandlerRegistered;
|
|
|
|
// Link ourselves.
|
|
NextEntry = PrettyStackTraceHead;
|
|
PrettyStackTraceHead = this;
|
|
}
|
|
|
|
PrettyStackTraceEntry::~PrettyStackTraceEntry() {
|
|
assert(PrettyStackTraceHead == this &&
|
|
"Pretty stack trace entry destruction is out of order");
|
|
PrettyStackTraceHead = getNextEntry();
|
|
}
|
|
|
|
void PrettyStackTraceString::print(raw_ostream &OS) const {
|
|
OS << Str << "\n";
|
|
}
|
|
|
|
void PrettyStackTraceProgram::print(raw_ostream &OS) const {
|
|
OS << "Program arguments: ";
|
|
// Print the argument list.
|
|
for (unsigned i = 0, e = ArgC; i != e; ++i)
|
|
OS << ArgV[i] << ' ';
|
|
OS << '\n';
|
|
}
|
|
|