htop/TraceScreen.c

181 lines
5.0 KiB
C
Raw Normal View History

2006-03-04 18:16:49 +00:00
/*
htop - TraceScreen.c
2006-03-23 18:55:29 +00:00
(C) 2005-2006 Hisham H. Muhammad
2006-03-04 18:16:49 +00:00
Released under the GNU GPL, see the COPYING file
in the source distribution for its full text.
*/
2011-12-26 21:35:57 +00:00
#include "TraceScreen.h"
#include "CRT.h"
#include "ProcessList.h"
#include "ListItem.h"
2006-03-04 18:16:49 +00:00
#include <stdio.h>
#include <unistd.h>
2011-12-26 21:35:57 +00:00
#include <string.h>
#include <stdlib.h>
2006-03-04 18:16:49 +00:00
#include <stdbool.h>
#include <unistd.h>
#include <fcntl.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <signal.h>
2006-03-04 18:16:49 +00:00
2011-12-26 21:35:57 +00:00
/*{
2006-03-04 18:16:49 +00:00
#include "Process.h"
2006-05-30 13:47:28 +00:00
#include "Panel.h"
2006-03-23 18:55:29 +00:00
#include "FunctionBar.h"
2006-03-04 18:16:49 +00:00
typedef struct TraceScreen_ {
Process* process;
2006-05-30 13:47:28 +00:00
Panel* display;
2006-03-04 18:16:49 +00:00
FunctionBar* bar;
bool tracing;
} TraceScreen;
}*/
static const char* tsFunctions[] = {"AutoScroll ", "Stop Tracing ", "Done ", NULL};
2006-03-04 18:16:49 +00:00
static const char* tsKeys[] = {"F4", "F5", "Esc"};
2006-03-04 18:16:49 +00:00
static int tsEvents[] = {KEY_F(4), KEY_F(5), 27};
2006-03-04 18:16:49 +00:00
TraceScreen* TraceScreen_new(Process* process) {
TraceScreen* this = (TraceScreen*) malloc(sizeof(TraceScreen));
this->process = process;
this->display = Panel_new(0, 1, COLS, LINES-2, LISTITEM_CLASS, true, ListItem_compare);
this->bar = FunctionBar_new(tsFunctions, tsKeys, tsEvents);
2006-03-04 18:16:49 +00:00
this->tracing = true;
return this;
}
void TraceScreen_delete(TraceScreen* this) {
2006-05-30 13:47:28 +00:00
Panel_delete((Object*)this->display);
2006-03-04 18:16:49 +00:00
FunctionBar_delete((Object*)this->bar);
free(this);
}
static void TraceScreen_draw(TraceScreen* this) {
2006-03-04 18:16:49 +00:00
attrset(CRT_colors[PANEL_HEADER_FOCUS]);
mvhline(0, 0, ' ', COLS);
mvprintw(0, 0, "Trace of process %d - %s", this->process->pid, this->process->comm);
attrset(CRT_colors[DEFAULT_COLOR]);
FunctionBar_draw(this->bar, NULL);
}
void TraceScreen_run(TraceScreen* this) {
// if (this->process->pid == getpid()) return;
char buffer[1001];
int fdpair[2];
int err = pipe(fdpair);
if (err == -1) return;
int child = fork();
if (child == -1) return;
if (child == 0) {
dup2(fdpair[1], STDERR_FILENO);
fcntl(fdpair[1], F_SETFL, O_NONBLOCK);
sprintf(buffer, "%d", this->process->pid);
execlp("strace", "strace", "-p", buffer, NULL);
const char* message = "Could not execute 'strace'. Please make sure it is available in your $PATH.";
write(fdpair[1], message, strlen(message));
2006-03-04 18:16:49 +00:00
exit(1);
}
fcntl(fdpair[0], F_SETFL, O_NONBLOCK);
FILE* strace = fdopen(fdpair[0], "r");
2006-05-30 14:00:18 +00:00
Panel* panel = this->display;
2006-03-04 18:16:49 +00:00
int fd_strace = fileno(strace);
TraceScreen_draw(this);
CRT_disableDelay();
bool contLine = false;
bool follow = false;
bool looping = true;
while (looping) {
fd_set fds;
FD_ZERO(&fds);
FD_SET(fd_strace, &fds);
struct timeval tv;
tv.tv_sec = 0; tv.tv_usec = 500;
int ready = select(fd_strace+1, &fds, NULL, NULL, &tv);
int nread = 0;
if (ready > 0)
nread = fread(buffer, 1, 1000, strace);
if (nread && this->tracing) {
char* line = buffer;
buffer[nread] = '\0';
for (int i = 0; i < nread; i++) {
if (buffer[i] == '\n') {
buffer[i] = '\0';
if (contLine) {
2006-05-30 14:00:18 +00:00
ListItem_append((ListItem*)Panel_get(panel,
2009-06-02 04:51:23 +00:00
Panel_size(panel)-1), line);
2006-03-04 18:16:49 +00:00
contLine = false;
} else {
2006-05-30 14:00:18 +00:00
Panel_add(panel, (Object*) ListItem_new(line, 0));
2006-03-04 18:16:49 +00:00
}
line = buffer+i+1;
}
}
if (line < buffer+nread) {
2006-05-30 14:00:18 +00:00
Panel_add(panel, (Object*) ListItem_new(line, 0));
2006-03-04 18:16:49 +00:00
buffer[nread] = '\0';
contLine = true;
}
if (follow)
2009-06-02 04:51:23 +00:00
Panel_setSelected(panel, Panel_size(panel)-1);
2006-05-30 14:00:18 +00:00
Panel_draw(panel, true);
2006-03-04 18:16:49 +00:00
}
int ch = getch();
if (ch == KEY_MOUSE) {
MEVENT mevent;
int ok = getmouse(&mevent);
if (ok == OK)
2006-05-30 14:00:18 +00:00
if (mevent.y >= panel->y && mevent.y < LINES - 1) {
Panel_setSelected(panel, mevent.y - panel->y + panel->scrollV);
2006-03-04 18:16:49 +00:00
follow = false;
ch = 0;
} if (mevent.y == LINES - 1)
ch = FunctionBar_synthesizeEvent(this->bar, mevent.x);
}
switch(ch) {
case ERR:
continue;
case KEY_F(5):
this->tracing = !this->tracing;
FunctionBar_setLabel(this->bar, KEY_F(5), this->tracing?"Stop Tracing ":"Resume Tracing ");
TraceScreen_draw(this);
break;
2010-07-31 00:11:34 +00:00
case KEY_HOME:
Panel_setSelected(panel, 0);
break;
case KEY_END:
Panel_setSelected(panel, Panel_size(panel)-1);
break;
2006-03-04 18:16:49 +00:00
case 'f':
case KEY_F(4):
follow = !follow;
if (follow)
2009-06-02 04:51:23 +00:00
Panel_setSelected(panel, Panel_size(panel)-1);
2006-03-04 18:16:49 +00:00
break;
case 'q':
case 27:
case KEY_F(10):
2006-03-04 18:16:49 +00:00
looping = false;
break;
case KEY_RESIZE:
2006-05-30 14:00:18 +00:00
Panel_resize(panel, COLS, LINES-2);
2006-03-04 18:16:49 +00:00
TraceScreen_draw(this);
break;
default:
follow = false;
2006-05-30 14:00:18 +00:00
Panel_onKey(panel, ch);
2006-03-04 18:16:49 +00:00
}
2006-05-30 14:00:18 +00:00
Panel_draw(panel, true);
2006-03-04 18:16:49 +00:00
}
kill(child, SIGTERM);
waitpid(child, NULL, 0);
2006-03-04 18:16:49 +00:00
fclose(strace);
2006-08-24 21:28:29 +00:00
CRT_enableDelay();
2006-03-04 18:16:49 +00:00
}