diff options
author | Hisham Muhammad <hisham@gobolinux.org> | 2015-03-16 23:01:48 -0300 |
---|---|---|
committer | Hisham Muhammad <hisham@gobolinux.org> | 2015-03-16 23:02:03 -0300 |
commit | 272e2d9b3459ceb3fe2f5ae34a07e44df6c45997 (patch) | |
tree | b7a7c2c68c3ebbe3b91f23a86da73ac352910cf5 /freebsd | |
parent | 9ff5d2b243472ae73d10dafdd7c0e24dc5052f6d (diff) |
Major advances in FreeBSD port.
Diffstat (limited to 'freebsd')
-rw-r--r-- | freebsd/FreeBSDProcess.c | 13 | ||||
-rw-r--r-- | freebsd/FreeBSDProcess.h | 8 | ||||
-rw-r--r-- | freebsd/FreeBSDProcessList.c | 102 | ||||
-rw-r--r-- | freebsd/FreeBSDProcessList.h | 4 |
4 files changed, 119 insertions, 8 deletions
diff --git a/freebsd/FreeBSDProcess.c b/freebsd/FreeBSDProcess.c index 614a6b8e..ab99ef7e 100644 --- a/freebsd/FreeBSDProcess.c +++ b/freebsd/FreeBSDProcess.c @@ -26,6 +26,14 @@ typedef struct FreeBSDProcess_ { Process super; } FreeBSDProcess; +#ifndef Process_isKernelThread +#define Process_isKernelThread(_process) (_process->pgrp == 0) +#endif + +#ifndef Process_isUserlandThread +#define Process_isUserlandThread(_process) (_process->pid != _process->tgid) +#endif + }*/ ProcessFieldData Process_fields[] = { @@ -44,6 +52,7 @@ ProcessFieldData Process_fields[] = { [NICE] = { .name = "NICE", .title = " NI ", .description = "Nice value (the higher the value, the more it lets other processes take priority)", .flags = 0, }, [STARTTIME] = { .name = "STARTTIME", .title = "START ", .description = "Time the process was started", .flags = 0, }, + [PROCESSOR] = { .name = "PROCESSOR", .title = "CPU ", .description = "Id of the CPU the process last executed on", .flags = 0, }, [M_SIZE] = { .name = "M_SIZE", .title = " VIRT ", .description = "Total program size in virtual memory", .flags = 0, }, [M_RESIDENT] = { .name = "M_RESIDENT", .title = " RES ", .description = "Resident set size, size of the text and data sections, plus stack usage", .flags = 0, }, [ST_UID] = { .name = "ST_UID", .title = " UID ", .description = "User ID of the process owner", .flags = 0, }, @@ -128,7 +137,5 @@ long Process_compare(const void* v1, const void* v2) { } bool Process_isThread(Process* this) { - (void) this; - // TODO - return false; + return (Process_isKernelThread(this)); } diff --git a/freebsd/FreeBSDProcess.h b/freebsd/FreeBSDProcess.h index 42a1fcb1..d0a1882c 100644 --- a/freebsd/FreeBSDProcess.h +++ b/freebsd/FreeBSDProcess.h @@ -19,6 +19,14 @@ typedef struct FreeBSDProcess_ { Process super; } FreeBSDProcess; +#ifndef Process_isKernelThread +#define Process_isKernelThread(_process) (_process->pgrp == 0) +#endif + +#ifndef Process_isUserlandThread +#define Process_isUserlandThread(_process) (_process->pid != _process->tgid) +#endif + extern ProcessFieldData Process_fields[]; diff --git a/freebsd/FreeBSDProcessList.c b/freebsd/FreeBSDProcessList.c index 24ab833e..6a8a5fcb 100644 --- a/freebsd/FreeBSDProcessList.c +++ b/freebsd/FreeBSDProcessList.c @@ -7,12 +7,15 @@ in the source distribution for its full text. #include "ProcessList.h" #include "FreeBSDProcessList.h" +#include "FreeBSDProcess.h" #include <unistd.h> #include <stdlib.h> #include <sys/types.h> #include <sys/sysctl.h> +#include <sys/user.h> #include <fcntl.h> +#include <string.h> /*{ @@ -103,10 +106,101 @@ static inline void FreeBSDProcessList_scanMemoryInfo(ProcessList* pl) { pl->buffersMem = 0; // not exposed to userspace } -void ProcessList_scan(ProcessList* this) { - (void) this; +char* FreeBSDProcessList_readProcessName(kvm_t* kd, struct kinfo_proc* kproc, int* basenameEnd) { + char** argv = kvm_getargv(kd, kproc, 0); + if (!argv) { + return strdup(kproc->ki_comm); + } + int len = 0; + for (int i = 0; argv[i]; i++) { + len += strlen(argv[i]) + 1; + } + char* comm = malloc(len * sizeof(char)); + char* at = comm; + *basenameEnd = 0; + for (int i = 0; argv[i]; i++) { + at = stpcpy(at, argv[i]); + if (!*basenameEnd) { + *basenameEnd = at - comm; + } + *at = ' '; + at++; + } + at--; + *at = '\0'; + return comm; +} +void ProcessList_goThroughEntries(ProcessList* this) { + FreeBSDProcessList* fpl = (FreeBSDProcessList*) this; + Settings* settings = this->settings; + bool hideKernelThreads = settings->hideKernelThreads; + bool hideUserlandThreads = settings->hideUserlandThreads; + FreeBSDProcessList_scanMemoryInfo(this); - - // stub! + + int count = 0; + struct kinfo_proc* kprocs = kvm_getprocs(fpl->kd, KERN_PROC_ALL, 0, &count); + + for (int i = 0; i < count; i++) { + struct kinfo_proc* kproc = &kprocs[i]; + + bool preExisting = false; + Process* proc = ProcessList_getProcess(this, kproc->ki_pid, &preExisting, (Process_new_fn) FreeBSDProcess_new); + FreeBSDProcess* fp = (FreeBSDProcess*) proc; + + proc->show = ! ((hideKernelThreads && Process_isKernelThread(proc)) || (hideUserlandThreads && Process_isUserlandThread(proc))); + + if (!preExisting) { + proc->ppid = kproc->ki_ppid; + proc->tpgid = kproc->ki_tpgid; + proc->tgid = kproc->ki_pid; + proc->session = kproc->ki_sid; + proc->tty_nr = kproc->ki_tdev; + proc->pgrp = kproc->ki_pgid; + proc->st_uid = kproc->ki_uid; + proc->starttime_ctime = kproc->ki_start.tv_sec; + proc->user = UsersTable_getRef(this->usersTable, proc->st_uid); + ProcessList_add((ProcessList*)this, proc); + proc->comm = FreeBSDProcessList_readProcessName(fpl->kd, kproc, &proc->basenameOffset); + } else { + if (settings->updateProcessNames) { + free(proc->comm); + proc->comm = FreeBSDProcessList_readProcessName(fpl->kd, kproc, &proc->basenameOffset); + } + } + + proc->m_size = kproc->ki_size / pageSizeKb / 1000; + proc->m_resident = kproc->ki_rssize; // * pageSizeKb; + proc->nlwp = kproc->ki_numthreads; + proc->time = (kproc->ki_runtime + 5000) / 10000; + proc->priority = kproc->ki_pri.pri_level - PZERO; + if (kproc->ki_pri.pri_class == PRI_TIMESHARE) { + proc->nice = kproc->ki_nice - NZERO; + } else if (PRI_IS_REALTIME(kproc->ki_pri.pri_class)) { + proc->nice = PRIO_MIN - 1 - (PRI_MAX_REALTIME - kproc->ki_pri.pri_level); + } else { + proc->nice = PRIO_MAX + 1 + kproc->ki_pri.pri_level - PRI_MIN_IDLE; + } + + switch (kproc->ki_stat) { + case SIDL: proc->state = 'I'; break; + case SRUN: proc->state = 'R'; break; + case SSLEEP: proc->state = 'S'; break; + case SSTOP: proc->state = 'T'; break; + case SZOMB: proc->state = 'Z'; break; + case SWAIT: proc->state = 'D'; break; + case SLOCK: proc->state = 'L'; break; + default: proc->state = '?'; + } + + if (Process_isKernelThread(proc)) { + this->kernelThreads++; + } + + this->totalTasks++; + if (proc->state == 'R') + this->runningTasks++; + proc->updated = true; + } } diff --git a/freebsd/FreeBSDProcessList.h b/freebsd/FreeBSDProcessList.h index 5127175b..9889f7c2 100644 --- a/freebsd/FreeBSDProcessList.h +++ b/freebsd/FreeBSDProcessList.h @@ -30,6 +30,8 @@ ProcessList* ProcessList_new(UsersTable* usersTable, Hashtable* pidWhiteList, ui void ProcessList_delete(ProcessList* this); -void ProcessList_scan(ProcessList* this); +char* FreeBSDProcessList_readProcessName(kvm_t* kd, struct kinfo_proc* kproc, int* basenameEnd); + +void ProcessList_goThroughEntries(ProcessList* this); #endif |