2014-11-24 20:55:49 +00:00
|
|
|
/*
|
|
|
|
htop - LinuxProcess.c
|
|
|
|
(C) 2014 Hisham H. Muhammad
|
|
|
|
Released under the GNU GPL, see the COPYING file
|
|
|
|
in the source distribution for its full text.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "Process.h"
|
2014-11-24 21:22:50 +00:00
|
|
|
#include "ProcessList.h"
|
|
|
|
#include "LinuxProcess.h"
|
|
|
|
#include "CRT.h"
|
|
|
|
|
2015-02-20 16:52:10 +00:00
|
|
|
#include <stdlib.h>
|
2014-11-24 21:22:50 +00:00
|
|
|
#include <unistd.h>
|
|
|
|
#include <sys/syscall.h>
|
2014-11-24 20:55:49 +00:00
|
|
|
|
|
|
|
/*{
|
|
|
|
|
2014-11-24 21:22:50 +00:00
|
|
|
#include "IOPriority.h"
|
|
|
|
|
2014-11-24 20:55:49 +00:00
|
|
|
typedef struct LinuxProcess_ {
|
|
|
|
Process super;
|
|
|
|
IOPriority ioPriority;
|
|
|
|
} LinuxProcess;
|
|
|
|
|
2015-02-20 16:52:10 +00:00
|
|
|
#define Process_delete LinuxProcess_delete
|
|
|
|
|
2014-11-24 20:55:49 +00:00
|
|
|
}*/
|
|
|
|
|
2015-02-23 06:53:36 +00:00
|
|
|
LinuxProcess* LinuxProcess_new(ProcessList* pl) {
|
2015-02-20 16:52:10 +00:00
|
|
|
LinuxProcess* this = calloc(sizeof(LinuxProcess), 1);
|
2015-02-23 06:53:36 +00:00
|
|
|
Object_setClass(this, Class(Process));
|
|
|
|
Process_init(&this->super, pl);
|
2015-02-20 16:52:10 +00:00
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
|
|
|
void LinuxProcess_delete(Object* cast) {
|
2015-03-08 22:45:56 +00:00
|
|
|
LinuxProcess* this = (LinuxProcess*) cast;
|
2015-02-20 16:52:10 +00:00
|
|
|
Process_done((Process*)cast);
|
|
|
|
free(this);
|
|
|
|
}
|
|
|
|
|
2014-11-24 20:55:49 +00:00
|
|
|
/*
|
|
|
|
[1] Note that before kernel 2.6.26 a process that has not asked for
|
|
|
|
an io priority formally uses "none" as scheduling class, but the
|
|
|
|
io scheduler will treat such processes as if it were in the best
|
|
|
|
effort class. The priority within the best effort class will be
|
|
|
|
dynamically derived from the cpu nice level of the process:
|
|
|
|
io_priority = (cpu_nice + 20) / 5. -- From ionice(1) man page
|
|
|
|
*/
|
2014-11-24 21:22:50 +00:00
|
|
|
#define LinuxProcess_effectiveIOPriority(p_) (IOPriority_class(p_->ioPriority) == IOPRIO_CLASS_NONE ? IOPriority_tuple(IOPRIO_CLASS_BE, (p_->super.nice + 20) / 5) : p_->ioPriority)
|
2014-11-24 20:55:49 +00:00
|
|
|
|
2014-11-24 21:22:50 +00:00
|
|
|
IOPriority LinuxProcess_updateIOPriority(LinuxProcess* this) {
|
|
|
|
IOPriority ioprio = syscall(SYS_ioprio_get, IOPRIO_WHO_PROCESS, this->super.pid);
|
2014-11-24 20:55:49 +00:00
|
|
|
this->ioPriority = ioprio;
|
|
|
|
return ioprio;
|
|
|
|
}
|
|
|
|
|
2014-11-24 21:22:50 +00:00
|
|
|
bool LinuxProcess_setIOPriority(LinuxProcess* this, IOPriority ioprio) {
|
|
|
|
syscall(SYS_ioprio_set, IOPRIO_WHO_PROCESS, this->super.pid, ioprio);
|
|
|
|
return (LinuxProcess_updateIOPriority(this) == ioprio);
|
2014-11-24 20:55:49 +00:00
|
|
|
}
|
|
|
|
|
2014-11-24 21:22:50 +00:00
|
|
|
void LinuxProcess_writeField(LinuxProcess* this, RichString* str, ProcessField field) {
|
|
|
|
char buffer[256]; buffer[255] = '\0';
|
|
|
|
int attr = CRT_colors[DEFAULT_COLOR];
|
|
|
|
int n = sizeof(buffer) - 1;
|
2014-11-24 20:55:49 +00:00
|
|
|
switch (field) {
|
|
|
|
case IO_PRIORITY: {
|
|
|
|
int klass = IOPriority_class(this->ioPriority);
|
|
|
|
if (klass == IOPRIO_CLASS_NONE) {
|
|
|
|
// see note [1] above
|
2014-11-24 21:22:50 +00:00
|
|
|
snprintf(buffer, n, "B%1d ", (int) (this->super.nice + 20) / 5);
|
2014-11-24 20:55:49 +00:00
|
|
|
} else if (klass == IOPRIO_CLASS_BE) {
|
|
|
|
snprintf(buffer, n, "B%1d ", IOPriority_data(this->ioPriority));
|
|
|
|
} else if (klass == IOPRIO_CLASS_RT) {
|
|
|
|
attr = CRT_colors[PROCESS_HIGH_PRIORITY];
|
|
|
|
snprintf(buffer, n, "R%1d ", IOPriority_data(this->ioPriority));
|
|
|
|
} else if (this->ioPriority == IOPriority_Idle) {
|
|
|
|
attr = CRT_colors[PROCESS_LOW_PRIORITY];
|
|
|
|
snprintf(buffer, n, "id ");
|
|
|
|
} else {
|
|
|
|
snprintf(buffer, n, "?? ");
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
snprintf(buffer, n, "- ");
|
|
|
|
}
|
|
|
|
RichString_append(str, attr, buffer);
|
|
|
|
}
|
|
|
|
|
|
|
|
long LinuxProcess_compare(const void* v1, const void* v2) {
|
|
|
|
LinuxProcess *p1, *p2;
|
|
|
|
ProcessList *pl = ((Process*)v1)->pl;
|
|
|
|
if (pl->direction == 1) {
|
|
|
|
p1 = (LinuxProcess*)v1;
|
|
|
|
p2 = (LinuxProcess*)v2;
|
|
|
|
} else {
|
|
|
|
p2 = (LinuxProcess*)v1;
|
|
|
|
p1 = (LinuxProcess*)v2;
|
|
|
|
}
|
|
|
|
switch (pl->sortKey) {
|
|
|
|
case IO_PRIORITY:
|
2014-11-24 21:22:50 +00:00
|
|
|
return LinuxProcess_effectiveIOPriority(p1) - LinuxProcess_effectiveIOPriority(p2);
|
2014-11-24 20:55:49 +00:00
|
|
|
default:
|
2014-11-24 21:22:50 +00:00
|
|
|
return (p1->super.pid - p2->super.pid);
|
2014-11-24 20:55:49 +00:00
|
|
|
}
|
|
|
|
}
|