-
-
Notifications
You must be signed in to change notification settings - Fork 442
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
5 changed files
with
289 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,211 @@ | ||
/* | ||
htop - Action.c | ||
(C) 2015 Hisham H. Muhammad | ||
Released under the GNU GPLv2+, see the COPYING file | ||
in the source distribution for its full text. | ||
*/ | ||
|
||
#include "BacktraceScreen.h" | ||
|
||
#if (defined(HTOP_LINUX) && defined(HAVE_LIBUNWIND_PTRACE)) | ||
|
||
#include <sys/wait.h> | ||
|
||
#include "CRT.h" | ||
#include "Object.h" | ||
#include "Panel.h" | ||
#include "Process.h" | ||
#include "RichString.h" | ||
#include "XUtils.h" | ||
#include "errno.h" | ||
|
||
#ifdef HAVE_LIBIBERTY | ||
#include <libiberty/demangle.h> | ||
#endif | ||
|
||
#include <libunwind-ptrace.h> | ||
#include <sys/ptrace.h> | ||
|
||
#define MAX_FRAME 256 | ||
|
||
static const char* const BacktraceScreenFunctions[] = {"Done ", NULL}; | ||
|
||
static const char* const BacktraceScreenKeys[] = {"Esc"}; | ||
|
||
static const int BacktraceScreenEvents[] = {27}; | ||
|
||
static void Frame_display(const Object* super, RichString* out) { | ||
const Frame* const frame = (const Frame*)super; | ||
if (frame->isError) { | ||
RichString_appendAscii(out, CRT_colors[DEFAULT_COLOR], frame->error); | ||
return; | ||
} | ||
|
||
char bufferNumberOfFrame[16] = {'\0'}; | ||
int len = snprintf(bufferNumberOfFrame, sizeof(bufferNumberOfFrame), "#%-3d ", frame->index); | ||
RichString_appendnAscii(out, CRT_colors[DYNAMIC_GREEN], bufferNumberOfFrame, len); | ||
|
||
char bufferAddress[32] = {'\0'}; | ||
len = snprintf(bufferAddress, sizeof(bufferAddress), "0x%016zx ", frame->address); | ||
RichString_appendnAscii(out, CRT_colors[DYNAMIC_BLUE], bufferAddress, len); | ||
|
||
RichString_appendAscii(out, CRT_colors[DEFAULT_COLOR], frame->functionName); | ||
if (frame->isSignalFrame) { | ||
RichString_appendAscii(out, CRT_colors[DYNAMIC_RED], " signal frame"); | ||
} | ||
|
||
char bufferFrameOffset[16] = {'\0'}; | ||
len = snprintf(bufferFrameOffset, sizeof(bufferFrameOffset), "+%zu", frame->offset); | ||
RichString_appendAscii(out, CRT_colors[DYNAMIC_YELLOW], bufferFrameOffset); | ||
} | ||
|
||
static void BacktracePanel_getFrames(BacktracePanel* this) { | ||
Panel* super = (Panel*) this; | ||
|
||
unw_addr_space_t addrSpace = unw_create_addr_space(&_UPT_accessors, 0); | ||
if (!addrSpace) { | ||
xAsprintf(&this->error, "Unable to init libunwind."); | ||
return; | ||
} | ||
|
||
const pid_t pid = Process_getPid(this->process); | ||
|
||
if (pid == 0) { | ||
xAsprintf(&this->error, "Unable to get the pid"); | ||
goto addr_space_error; | ||
} | ||
|
||
if (ptrace(PTRACE_ATTACH, pid, 0, 0) == -1) { | ||
xAsprintf(&this->error, "ptrace: %s", strerror(errno)); | ||
goto addr_space_error; | ||
} | ||
wait(NULL); | ||
|
||
struct UPT_info* context = _UPT_create(pid); | ||
if (!context) { | ||
xAsprintf(&this->error, "Unable to init backtrace panel."); | ||
goto ptrace_error; | ||
} | ||
|
||
unw_cursor_t cursor; | ||
int ret = unw_init_remote(&cursor, addrSpace, context); | ||
if (ret < 0) { | ||
xAsprintf(&this->error, "libunwind cursor: ret=%d", ret); | ||
goto context_error; | ||
} | ||
|
||
int index = 0; | ||
do { | ||
char procName[256] = "?"; | ||
unw_word_t offset; | ||
unw_word_t pc; | ||
|
||
if (unw_get_proc_name(&cursor, procName, sizeof(procName), &offset) == 0) { | ||
ret = unw_get_reg(&cursor, UNW_REG_IP, &pc); | ||
if (ret < 0) { | ||
xAsprintf(&this->error, "unable to get register rip : %d", ret); | ||
break; | ||
} | ||
|
||
Frame* frame = Frame_new(); | ||
frame->index = index; | ||
frame->address = pc; | ||
frame->offset = offset; | ||
frame->isSignalFrame = unw_is_signal_frame(&cursor); | ||
#if HAVE_LIBIBERTY | ||
char* demangledName = cplus_demangle(procName, | ||
DMGL_PARAMS | DMGL_ANSI | DMGL_VERBOSE | DMGL_RET_POSTFIX); | ||
if (demangledName == NULL) { | ||
xAsprintf(&frame->functionName, "%s", procName); | ||
} else { | ||
xAsprintf(&frame->functionName, "%s", demangledName); | ||
free(demangledName); | ||
} | ||
#else | ||
xAsprintf(&frame->functionName, "%s", procName); | ||
#endif | ||
Panel_add(super, (Object*)frame); | ||
} | ||
index++; | ||
} while (unw_step(&cursor) > 0 && index < MAX_FRAME); | ||
|
||
context_error: | ||
_UPT_destroy(context); | ||
|
||
ptrace_error: | ||
ptrace(PTRACE_DETACH, pid, 0, 0); | ||
|
||
addr_space_error: | ||
unw_destroy_addr_space(addrSpace); | ||
} | ||
|
||
BacktracePanel* BacktracePanel_new(const Process* process) { | ||
BacktracePanel* this = CallocThis(BacktracePanel); | ||
this->process = process; | ||
|
||
Panel* super = (Panel*) this; | ||
Panel_init(super, 1, 1, 1, 1, Class(Frame), true, FunctionBar_new(BacktraceScreenFunctions, BacktraceScreenKeys, BacktraceScreenEvents)); | ||
BacktracePanel_getFrames(this); | ||
if (this->error) { | ||
Panel_prune(super); | ||
|
||
Frame* errorFrame = Frame_new(); | ||
errorFrame->error = xStrdup(this->error); | ||
errorFrame->isError = true; | ||
Panel_add(super, (Object*)errorFrame); | ||
} | ||
|
||
char* header = NULL; | ||
xAsprintf(&header, "Backtrace of '%s' (%d)", process->procComm, Process_getPid(process)); | ||
Panel_setHeader(super, header); | ||
free(header); | ||
return this; | ||
} | ||
|
||
Frame* Frame_new(void) { | ||
Frame* this = CallocThis(Frame); | ||
return this; | ||
} | ||
|
||
static int Frame_compare(const void* object1, const void* object2) { | ||
const Frame* frame1 = (const Frame*)object1; | ||
const Frame* frame2 = (const Frame*)object2; | ||
return String_eq(frame1->functionName, frame2->functionName); | ||
} | ||
|
||
static void Frame_delete(Object* object) { | ||
Frame* this = (Frame*)object; | ||
if (this->functionName) { | ||
free(this->functionName); | ||
} | ||
|
||
if (this->isError && this->error) { | ||
free(this->error); | ||
} | ||
|
||
free(this); | ||
} | ||
|
||
void BacktracePanel_delete(Object* object) { | ||
BacktracePanel* this = (BacktracePanel*)object; | ||
if (this->error) { | ||
free(this->error); | ||
} | ||
Panel_delete(object); | ||
} | ||
|
||
const PanelClass BacktracePanel_class = { | ||
.super = { | ||
.extends = Class(Panel), | ||
.delete = BacktracePanel_delete, | ||
}, | ||
}; | ||
|
||
const ObjectClass Frame_class = { | ||
.extends = Class(Object), | ||
.compare = Frame_compare, | ||
.delete = Frame_delete, | ||
.display = Frame_display, | ||
}; | ||
|
||
#endif /* HTOP_LINUX && HAVE_LIBUNWIND_PTRACE */ |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,46 @@ | ||
#ifndef HEADER_BacktraceScreen | ||
#define HEADER_BacktraceScreen | ||
/* | ||
htop - Filename.h | ||
(C) 2021 htop dev team | ||
Released under the GNU GPLv2+, see the COPYING file | ||
in the source distribution for its full text. | ||
*/ | ||
|
||
#include "config.h" // IWYU pragma: keep | ||
|
||
#if (defined(HTOP_LINUX) && defined(HAVE_LIBUNWIND_PTRACE)) | ||
|
||
#include <stddef.h> | ||
|
||
#include "Panel.h" | ||
#include "Process.h" | ||
|
||
typedef struct BacktracePanel_ { | ||
Panel super; | ||
const Process* process; | ||
char* error; | ||
} BacktracePanel; | ||
|
||
typedef struct Frame_ { | ||
Object super; | ||
int index; | ||
size_t address; | ||
size_t offset; | ||
char* functionName; | ||
bool isSignalFrame; | ||
|
||
bool isError; | ||
char* error; | ||
} Frame; | ||
|
||
BacktracePanel* BacktracePanel_new(const Process* process); | ||
void BacktracePanel_delete(Object* object); | ||
Frame* Frame_new(void); | ||
|
||
extern const PanelClass BacktracePanel_class; | ||
extern const ObjectClass Frame_class; | ||
|
||
#endif /* HTOP_LINUX && HAVE_LIBUNWIND_PTRACE */ | ||
|
||
#endif |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters