You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
tdebase/drkonqi/backtrace.cpp

392 lines
13 KiB

/*****************************************************************
* drkonqi - The KDE Crash Handler
*
* Copyright (C) 2000-2003 Hans Petter Bieker <bieker@kde.org>
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
*
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
* OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
* IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
* NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
* THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*****************************************************************/
#include "config.h"
#include <tqfile.h>
#include <tqregexp.h>
#include <kprocess.h>
#include <kdebug.h>
#include <kstandarddirs.h>
#include <tdemessagebox.h>
#include <tdelocale.h>
#include <tdetempfile.h>
#ifdef WITH_TDEHWLIB
#include <tdehardwaredevices.h>
#endif
#ifdef HAVE_ELFICON
// Elven things
extern "C" {
#include <libr.h>
#include <libr-icons.h>
}
#include <tqfileinfo.h>
#include <tdeio/tdelficon.h>
#endif // HAVE_ELFICON
#include "krashconf.h"
#include "backtrace.h"
#include "backtrace.moc"
BackTrace::BackTrace(const KrashConfig *krashconf, TQObject *parent,
const char *name)
: TQObject(parent, name),
m_krashconf(krashconf), m_temp(NULL), m_temp_cmd(NULL)
{
m_proc = new TDEProcess;
}
BackTrace::~BackTrace()
{
pid_t pid = m_proc ? m_proc->pid() : 0;
// we don't want the gdb process to hang around
delete m_proc; // this will kill gdb (SIGKILL, signal 9)
// continue the process we ran backtrace on. Gdb sends SIGSTOP to the
// process. For some reason it doesn't work if we send the signal before
// gdb has exited, so we better wait for it.
// Do not touch it if we never ran backtrace.
if (pid)
{
waitpid(pid, NULL, 0);
kill(m_krashconf->pid(), SIGCONT);
}
delete m_temp;
delete m_temp_cmd;
}
void BackTrace::start()
{
TQString exec = m_krashconf->tryExec();
if ( !exec.isEmpty() && TDEStandardDirs::findExe(exec).isEmpty() )
{
TQObject * o = parent();
if (o && !o->inherits(TQWIDGET_OBJECT_NAME_STRING))
{
o = NULL;
}
KMessageBox::error(
(TQWidget *)o,
i18n("Could not generate a backtrace as the debugger '%1' was not found.").arg(exec));
return;
}
m_temp = new KTempFile;
m_temp->setAutoDelete(TRUE);
int handle = m_temp->handle();
TQString backtraceCommand = m_krashconf->backtraceCommand();
const char* bt = backtraceCommand.latin1();
::write(handle, bt, strlen(bt)); // the command for a backtrace
::write(handle, "\n", 1);
::fsync(handle);
// build the debugger command
TQString str = m_krashconf->debuggerBatch();
m_krashconf->expandString(str, true, m_temp->name());
// write the debugger command
m_temp_cmd = new KTempFile(TQString::null, TQString::null, 0700);
m_temp_cmd->setAutoDelete(TRUE);
handle = m_temp_cmd->handle();
const char* dbgcommand = str.latin1();
::write(handle, dbgcommand, strlen(dbgcommand)); // the command to execute the debugger
::write(handle, "\n", 1);
::fsync(handle);
m_temp_cmd->close();
// determine if yama has been used to prevent ptrace as normal user
bool need_root_access = false;
TQFile yamactl("/proc/sys/kernel/yama/ptrace_scope");
if (yamactl.exists()) {
if (yamactl.open(IO_ReadOnly)) {
TQTextStream stream(&yamactl);
TQString line;
line = stream.readLine();
if (line.toInt() != 0) {
need_root_access = true;
}
yamactl.close();
}
}
// start the debugger
m_proc = new TDEProcess;
m_proc->setUseShell(true);
if (need_root_access == false) {
*m_proc << m_temp_cmd->name();
}
else {
*m_proc << "tdesu -t --comment \"" << i18n("Administrative access is required to generate a backtrace") << "\" -c \"" << m_temp_cmd->name() << "\"";
}
connect(m_proc, TQT_SIGNAL(receivedStdout(TDEProcess*, char*, int)),
TQT_SLOT(slotReadInput(TDEProcess*, char*, int)));
connect(m_proc, TQT_SIGNAL(processExited(TDEProcess*)),
TQT_SLOT(slotProcessExited(TDEProcess*)));
m_proc->start ( TDEProcess::NotifyOnExit, TDEProcess::All );
}
void BackTrace::slotReadInput(TDEProcess *, char* buf, int buflen)
{
TQString newstr = TQString::fromLocal8Bit(buf, buflen);
newstr.replace("\n\n", "\n");
if (m_strBt.isEmpty()) {
if (newstr == "\n") {
newstr = "";
}
}
if (!newstr.startsWith(": ")) {
m_strBt.append(newstr);
emit append(newstr);
}
}
void BackTrace::slotProcessExited(TDEProcess *proc)
{
// start it again
kill(m_krashconf->pid(), SIGCONT);
if (proc->normalExit() && (proc->exitStatus() == 0) &&
usefulBacktrace())
{
processBacktrace();
emit done(m_strBt);
}
else
emit someError();
}
// analyze backtrace for usefulness
bool BackTrace::usefulBacktrace()
{
// remove crap
if( !m_krashconf->removeFromBacktraceRegExp().isEmpty()) {
m_strBt.replace(TQRegExp( m_krashconf->removeFromBacktraceRegExp()), TQString());
}
// fix threading info output
if( !m_krashconf->threadRegExp().isEmpty()) {
int pos = -1;
TQRegExp threadRegExpression( m_krashconf->threadRegExp());
do {
threadRegExpression.search(m_strBt);
pos = threadRegExpression.pos();
if (pos > -1) {
m_strBt.insert(threadRegExpression.pos()+1, "==== ");
}
} while (pos > -1);
}
if( m_krashconf->disableChecks())
return true;
// prepend and append newline, so that regexps like '\nwhatever\n' work on all lines
TQString strBt = '\n' + m_strBt + '\n';
// how many " ?? " in the bt ?
int unknown = 0;
if( !m_krashconf->invalidStackFrameRegExp().isEmpty()) {
int isfPos = strBt.find( TQRegExp( m_krashconf->invalidStackFrameRegExp()));
int islPos = strBt.find( m_krashconf->infoSharedLibraryHeader());
if ((isfPos >=0) && (isfPos < islPos)) {
unknown = true;
}
else {
unknown = false;
}
}
// how many stack frames in the bt ?
int frames = 0;
if( !m_krashconf->frameRegExp().isEmpty())
frames = strBt.contains( TQRegExp( m_krashconf->frameRegExp()));
else
frames = strBt.contains('\n');
bool tooShort = false;
if( !m_krashconf->neededInValidBacktraceRegExp().isEmpty())
tooShort = ( strBt.find( TQRegExp( m_krashconf->neededInValidBacktraceRegExp())) == -1 );
return !m_strBt.isNull() && !tooShort && (unknown < frames);
}
// remove stack frames added because of TDECrash
void BackTrace::processBacktrace()
{
if( !m_krashconf->kcrashRegExp().isEmpty()) {
TQRegExp kcrashregexp( m_krashconf->kcrashRegExp());
kcrashregexp.setMinimal(true);
int pos = 0;
int prevpos = 0;
while ((pos = kcrashregexp.search( m_strBt, pos )) >= 0) {
if (prevpos == pos) {
// Avoid infinite loop
// Shouldn't ever get here, but given that this is a crash handler, better safe than sorry!
break;
}
prevpos = pos;
if( pos >= 0 ) {
int len = kcrashregexp.matchedLength();
int nextinfochunkpos = m_strBt.find("====", pos);
if (nextinfochunkpos >= 0) {
// Trying to delete too much!
int limitedlen = nextinfochunkpos - pos;
TQString limitedstrBt = m_strBt.mid(pos, limitedlen);
int limitedpos = kcrashregexp.search( limitedstrBt );
if (limitedpos >= 0) {
len = kcrashregexp.matchedLength();
}
else {
len = 0;
}
}
if ((pos >= 0) && (len > 0)) {
if( m_strBt[ pos ] == '\n' ) {
++pos;
--len;
}
m_strBt.remove( pos, len );
m_strBt.insert( pos, TQString::fromLatin1( "[TDECrash handler]\n" ));
}
}
if (pos < 0) {
// Avoid infinite loop
// Shouldn't ever get here, but given that this is a crash handler, better safe than sorry!
break;
}
pos++;
if ((uint)pos >= m_strBt.length()) {
// Avoid infinite loop
// Shouldn't ever get here, but given that this is a crash handler, better safe than sorry!
break;
}
}
}
if( !m_krashconf->kcrashRegExpSingle().isEmpty()) {
TQRegExp kcrashregexp( m_krashconf->kcrashRegExpSingle());
int pos = kcrashregexp.search( m_strBt );
if( pos >= 0 ) {
int len = kcrashregexp.matchedLength();
if( m_strBt[ pos ] == '\n' ) {
++pos;
--len;
}
m_strBt.remove( pos, len );
}
}
#ifdef HAVE_ELFICON
m_strBt.append("\n==== (tdemetainfo) application version information ====\n");
// Extract embedded SCM metadata from the crashed application
TQString crashedExec = TDEStandardDirs::findExe(m_krashconf->execName());
if (crashedExec.startsWith("/")) {
libr_file *handle = NULL;
libr_access_t access = LIBR_READ;
if((handle = libr_open(const_cast<char*>(crashedExec.ascii()), access)) == NULL) {
kdWarning() << "failed to open file" << crashedExec << endl;
}
else {
TQString scmModule = elf_get_resource(handle, ".metadata_scmmodule");
TQString scmRevision = elf_get_resource(handle, ".metadata_scmrevision");
if (scmRevision != "") {
m_strBt.append(TQString("%1:\t%2:%3\n").arg(TQFileInfo(crashedExec).fileName()).arg(scmModule).arg(scmRevision));
}
}
libr_close(handle);
}
m_strBt.append("\n==== (tdemetainfo) library version information ====\n");
// Extract embedded SCM metadata from shared libraries
int islPos = m_strBt.find( m_krashconf->infoSharedLibraryHeader());
TQString infoSharedLibraryText = m_strBt.mid(islPos);
TQTextStream infoSharedLibraryTextStream(&infoSharedLibraryText, IO_ReadOnly);
infoSharedLibraryTextStream.readLine(); // Skip info header x1
infoSharedLibraryTextStream.readLine(); // Skip info header x2
TQString infoSharedLibraryLine = infoSharedLibraryTextStream.readLine();
while (infoSharedLibraryLine != TQString::null) {
TQStringList libraryInfoList = TQStringList::split(" ", infoSharedLibraryLine, false);
if (libraryInfoList.count() > 0) {
TQString libraryName = libraryInfoList[libraryInfoList.count()-1];
if (libraryName.startsWith("/")) {
libr_file *handle = NULL;
libr_access_t access = LIBR_READ;
if((handle = libr_open(const_cast<char*>(libraryName.ascii()), access)) == NULL) {
kdWarning() << "failed to open file " << libraryName << endl;
}
else {
TQString scmModule = elf_get_resource(handle, ".metadata_scmmodule");
TQString scmRevision = elf_get_resource(handle, ".metadata_scmrevision");
if (scmRevision != "") {
m_strBt.append(TQString("%1:\t%2:%3\n").arg(TQFileInfo(libraryName).fileName()).arg(scmModule).arg(scmRevision));
}
libr_close(handle);
}
}
}
infoSharedLibraryLine = infoSharedLibraryTextStream.readLine();
}
#endif // HAVE_ELFICON
#ifdef WITH_TDEHWLIB
// Append potentially important hardware information
m_strBt.append("\n==== (tdehwlib) hardware information ====\n");
TDEHardwareDevices *hwdevices = TDEGlobal::hardwareDevices();
TDEGenericHardwareList hwlist = hwdevices->listAllPhysicalDevices();
TDEGenericDevice *hwdevice;
for ( hwdevice = hwlist.first(); hwdevice; hwdevice = hwlist.next() ) {
if (hwdevice->type() == TDEGenericDeviceType::CPU) {
TDECPUDevice* cpudevice = static_cast<TDECPUDevice*>(hwdevice);
m_strBt.append(TQString("CPU core number:\t\t%1\n").arg(cpudevice->coreNumber()));
m_strBt.append(TQString("\tVendor:\t\t\t%1\n").arg(cpudevice->vendorName()));
m_strBt.append(TQString("\tModel:\t\t\t%1\n").arg(cpudevice->vendorModel()));
m_strBt.append(TQString("\tName:\t\t\t%1\n").arg(cpudevice->name()));
m_strBt.append(TQString("\tCurrent Frequency:\t%1 MHz\n").arg(cpudevice->frequency()));
m_strBt.append(TQString("\tMinimum Frequency:\t%1 MHz\n").arg(cpudevice->minFrequency()));
m_strBt.append(TQString("\tMaximum Frequency:\t%1 MHz\n").arg(cpudevice->maxFrequency()));
m_strBt.append("\n");
}
}
#endif
{
// Clean up hard to read debug blocks
TQRegExp kcrashregexp( "[^\n]\n==== ");
kcrashregexp.setMinimal(true);
int pos = 0;
while ((pos = kcrashregexp.search( m_strBt, pos )) >= 0) {
m_strBt.insert(pos+1, "\n");
}
}
}