htop/darwin/DarwinProcess.c

344 lines
9.5 KiB
C
Raw Normal View History

/*
htop - DarwinProcess.c
(C) 2015 Hisham H. Muhammad
Released under the GNU GPLv2, see the COPYING file
in the source distribution for its full text.
*/
#include "DarwinProcess.h"
2015-07-13 06:17:14 +00:00
#include <libproc.h>
2015-07-14 16:46:16 +00:00
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <mach/mach.h>
#include "CRT.h"
#include "Process.h"
2020-10-13 12:35:30 +00:00
const ProcessClass DarwinProcess_class = {
2015-07-14 16:46:16 +00:00
.super = {
.extends = Class(Process),
.display = Process_display,
.delete = Process_delete,
.compare = Process_compare
},
.writeField = Process_writeField,
};
Process* DarwinProcess_new(const Settings* settings) {
2016-02-02 14:53:02 +00:00
DarwinProcess* this = xCalloc(1, sizeof(DarwinProcess));
2015-07-14 16:46:16 +00:00
Object_setClass(this, Class(DarwinProcess));
Process_init(&this->super, settings);
this->utime = 0;
this->stime = 0;
2016-03-31 03:18:42 +00:00
this->taskAccess = true;
2015-07-14 16:46:16 +00:00
return &this->super;
}
2015-07-13 06:17:14 +00:00
void Process_delete(Object* cast) {
2015-07-14 16:46:16 +00:00
DarwinProcess* this = (DarwinProcess*) cast;
Process_done(&this->super);
// free platform-specific fields here
free(this);
}
bool Process_isThread(const Process* this) {
2015-07-14 16:46:16 +00:00
(void) this;
2015-07-13 06:17:14 +00:00
return false;
}
static char* DarwinProcess_getCmdLine(const struct kinfo_proc* k, int* basenameOffset) {
2015-07-13 06:17:14 +00:00
/* This function is from the old Mac version of htop. Originally from ps? */
int mib[3], argmax, nargs, c = 0;
size_t size;
char *procargs, *sp, *np, *cp, *retval;
/* Get the maximum process arguments size. */
mib[0] = CTL_KERN;
mib[1] = KERN_ARGMAX;
size = sizeof( argmax );
if ( sysctl( mib, 2, &argmax, &size, NULL, 0 ) == -1 ) {
goto ERROR_A;
}
/* Allocate space for the arguments. */
2020-10-31 22:28:02 +00:00
procargs = (char*)xMalloc(argmax);
2015-07-13 06:17:14 +00:00
if ( procargs == NULL ) {
goto ERROR_A;
}
/*
* Make a sysctl() call to get the raw argument space of the process.
* The layout is documented in start.s, which is part of the Csu
* project. In summary, it looks like:
*
* /---------------\ 0x00000000
* : :
* : :
* |---------------|
* | argc |
* |---------------|
* | arg[0] |
* |---------------|
* : :
* : :
* |---------------|
* | arg[argc - 1] |
* |---------------|
* | 0 |
* |---------------|
* | env[0] |
* |---------------|
* : :
* : :
* |---------------|
* | env[n] |
* |---------------|
* | 0 |
* |---------------| <-- Beginning of data returned by sysctl() is here.
* | argc |
* |---------------|
* | exec_path |
* |:::::::::::::::|
* | |
* | String area. |
* | |
* |---------------| <-- Top of stack.
* : :
* : :
* \---------------/ 0xffffffff
*/
mib[0] = CTL_KERN;
mib[1] = KERN_PROCARGS2;
mib[2] = k->kp_proc.p_pid;
size = ( size_t ) argmax;
if ( sysctl( mib, 3, procargs, &size, NULL, 0 ) == -1 ) {
goto ERROR_B;
}
memcpy( &nargs, procargs, sizeof( nargs ) );
cp = procargs + sizeof( nargs );
/* Skip the saved exec_path. */
for ( ; cp < &procargs[size]; cp++ ) {
if ( *cp == '\0' ) {
/* End of exec_path reached. */
break;
}
}
if ( cp == &procargs[size] ) {
goto ERROR_B;
}
/* Skip trailing '\0' characters. */
for ( ; cp < &procargs[size]; cp++ ) {
if ( *cp != '\0' ) {
/* Beginning of first argument reached. */
break;
}
}
if ( cp == &procargs[size] ) {
goto ERROR_B;
}
/* Save where the argv[0] string starts. */
sp = cp;
*basenameOffset = 0;
2015-07-13 06:17:14 +00:00
for ( np = NULL; c < nargs && cp < &procargs[size]; cp++ ) {
if ( *cp == '\0' ) {
c++;
if ( np != NULL ) {
/* Convert previous '\0'. */
*np = ' ';
}
2020-10-31 21:14:27 +00:00
/* Note location of current '\0'. */
np = cp;
if (*basenameOffset == 0) {
*basenameOffset = cp - sp;
}
}
}
2015-07-13 06:17:14 +00:00
/*
* sp points to the beginning of the arguments/environment string, and
* np should point to the '\0' terminator for the string.
*/
if ( np == NULL || np == sp ) {
/* Empty or unterminated string. */
goto ERROR_B;
}
if (*basenameOffset == 0) {
*basenameOffset = np - sp;
}
2015-07-13 06:17:14 +00:00
/* Make a copy of the string. */
2016-02-02 14:53:02 +00:00
retval = xStrdup(sp);
2015-07-13 06:17:14 +00:00
/* Clean up. */
free( procargs );
return retval;
ERROR_B:
free( procargs );
ERROR_A:
2016-02-02 14:53:02 +00:00
retval = xStrdup(k->kp_proc.p_comm);
*basenameOffset = strlen(retval);
2019-10-31 16:39:12 +00:00
2015-07-13 06:17:14 +00:00
return retval;
}
void DarwinProcess_setFromKInfoProc(Process* proc, const struct kinfo_proc* ps, bool exists) {
const struct extern_proc* ep = &ps->kp_proc;
2015-08-19 16:56:46 +00:00
/* UNSET HERE :
*
* processor
* user (set at ProcessList level)
* nlwp
* percent_cpu
* percent_mem
* m_virt
2015-08-19 16:56:46 +00:00
* m_resident
* minflt
* majflt
*/
/* First, the "immutable" parts */
2020-10-31 19:52:20 +00:00
if (!exists) {
2015-08-19 16:56:46 +00:00
/* Set the PID/PGID/etc. */
proc->pid = ep->p_pid;
proc->ppid = ps->kp_eproc.e_ppid;
proc->pgrp = ps->kp_eproc.e_pgid;
proc->session = 0; /* TODO Get the session id */
proc->tpgid = ps->kp_eproc.e_tpgid;
proc->tgid = proc->pid;
2015-08-19 16:56:46 +00:00
proc->st_uid = ps->kp_eproc.e_ucred.cr_uid;
/* e_tdev = (major << 24) | (minor & 0xffffff) */
/* e_tdev == -1 for "no device" */
proc->tty_nr = ps->kp_eproc.e_tdev & 0xff; /* TODO tty_nr is unsigned */
proc->starttime_ctime = ep->p_starttime.tv_sec;
Process_fillStarttimeBuffer(proc);
proc->comm = DarwinProcess_getCmdLine(ps, &(proc->basenameOffset));
2015-08-19 16:56:46 +00:00
}
/* Mutable information */
proc->nice = ep->p_nice;
proc->priority = ep->p_priority;
proc->state = (ep->p_stat == SZOMB) ? 'Z' : '?';
2015-08-19 16:56:46 +00:00
/* Make sure the updated flag is set */
proc->updated = true;
2015-07-13 06:17:14 +00:00
}
2020-10-31 22:28:02 +00:00
void DarwinProcess_setFromLibprocPidinfo(DarwinProcess* proc, DarwinProcessList* dpl) {
2015-08-19 16:56:46 +00:00
struct proc_taskinfo pti;
2015-07-13 06:17:14 +00:00
2020-10-31 19:52:20 +00:00
if (sizeof(pti) == proc_pidinfo(proc->super.pid, PROC_PIDTASKINFO, 0, &pti, sizeof(pti))) {
if (0 != proc->utime || 0 != proc->stime) {
2015-08-19 16:56:46 +00:00
uint64_t diff = (pti.pti_total_system - proc->stime)
2020-10-31 21:14:27 +00:00
+ (pti.pti_total_user - proc->utime);
2015-07-14 16:46:16 +00:00
2015-08-19 16:56:46 +00:00
proc->super.percent_cpu = (double)diff * (double)dpl->super.cpuCount
2020-10-31 21:14:27 +00:00
/ ((double)dpl->global_diff * 100000.0);
2015-07-14 16:46:16 +00:00
2015-08-19 16:56:46 +00:00
// fprintf(stderr, "%f %llu %llu %llu %llu %llu\n", proc->super.percent_cpu,
// proc->stime, proc->utime, pti.pti_total_system, pti.pti_total_user, dpl->global_diff);
// exit(7);
}
2015-07-14 16:46:16 +00:00
2015-08-19 16:56:46 +00:00
proc->super.time = (pti.pti_total_system + pti.pti_total_user) / 10000000;
proc->super.nlwp = pti.pti_threadnum;
proc->super.m_virt = pti.pti_virtual_size / CRT_pageSize;
proc->super.m_resident = pti.pti_resident_size / CRT_pageSize;
2015-08-19 16:56:46 +00:00
proc->super.majflt = pti.pti_faults;
proc->super.percent_mem = (double)pti.pti_resident_size * 100.0
2020-10-31 21:14:27 +00:00
/ (double)dpl->host_info.max_mem;
2015-08-19 16:56:46 +00:00
proc->stime = pti.pti_total_system;
proc->utime = pti.pti_total_user;
dpl->super.kernelThreads += 0; /*pti.pti_threads_system;*/
dpl->super.userlandThreads += pti.pti_threadnum; /*pti.pti_threads_user;*/
dpl->super.totalTasks += pti.pti_threadnum;
dpl->super.runningTasks += pti.pti_numrunning;
}
2015-07-13 06:17:14 +00:00
}
/*
* Scan threads for process state information.
* Based on: http://stackoverflow.com/questions/6788274/ios-mac-cpu-usage-for-thread
* and https://github.com/max-horvath/htop-osx/blob/e86692e869e30b0bc7264b3675d2a4014866ef46/ProcessList.c
*/
2020-10-31 22:28:02 +00:00
void DarwinProcess_scanThreads(DarwinProcess* dp) {
Process* proc = (Process*) dp;
kern_return_t ret;
2019-10-31 16:39:12 +00:00
2016-03-31 03:18:42 +00:00
if (!dp->taskAccess) {
return;
}
2019-10-31 16:39:12 +00:00
if (proc->state == 'Z') {
return;
}
task_t port;
ret = task_for_pid(mach_task_self(), proc->pid, &port);
if (ret != KERN_SUCCESS) {
2016-03-31 03:18:42 +00:00
dp->taskAccess = false;
return;
}
2019-10-31 16:39:12 +00:00
task_info_data_t tinfo;
mach_msg_type_number_t task_info_count = TASK_INFO_MAX;
ret = task_info(port, TASK_BASIC_INFO, (task_info_t) tinfo, &task_info_count);
if (ret != KERN_SUCCESS) {
2016-03-31 03:18:42 +00:00
dp->taskAccess = false;
return;
}
2019-10-31 16:39:12 +00:00
thread_array_t thread_list;
mach_msg_type_number_t thread_count;
ret = task_threads(port, &thread_list, &thread_count);
if (ret != KERN_SUCCESS) {
2016-03-31 03:18:42 +00:00
dp->taskAccess = false;
mach_port_deallocate(mach_task_self(), port);
return;
}
2019-10-31 16:39:12 +00:00
integer_t run_state = 999;
for (unsigned int i = 0; i < thread_count; i++) {
thread_info_data_t thinfo;
mach_msg_type_number_t thread_info_count = THREAD_BASIC_INFO_COUNT;
ret = thread_info(thread_list[i], THREAD_BASIC_INFO, (thread_info_t)thinfo, &thread_info_count);
if (ret == KERN_SUCCESS) {
thread_basic_info_t basic_info_th = (thread_basic_info_t) thinfo;
if (basic_info_th->run_state < run_state) {
run_state = basic_info_th->run_state;
}
mach_port_deallocate(mach_task_self(), thread_list[i]);
}
}
vm_deallocate(mach_task_self(), (vm_address_t) thread_list, sizeof(thread_port_array_t) * thread_count);
mach_port_deallocate(mach_task_self(), port);
char state = '?';
switch (run_state) {
case TH_STATE_RUNNING: state = 'R'; break;
case TH_STATE_STOPPED: state = 'S'; break;
case TH_STATE_WAITING: state = 'W'; break;
case TH_STATE_UNINTERRUPTIBLE: state = 'U'; break;
case TH_STATE_HALTED: state = 'H'; break;
}
proc->state = state;
}