| /* |
| htop - dragonflybsd/Platform.c |
| (C) 2014 Hisham H. Muhammad |
| (C) 2017 Diederik de Groot |
| Released under the GNU GPL, see the COPYING file |
| in the source distribution for its full text. |
| */ |
| |
| #include "Platform.h" |
| #include "Meter.h" |
| #include "CPUMeter.h" |
| #include "MemoryMeter.h" |
| #include "SwapMeter.h" |
| #include "TasksMeter.h" |
| #include "LoadAverageMeter.h" |
| #include "UptimeMeter.h" |
| #include "ClockMeter.h" |
| #include "HostnameMeter.h" |
| #include "DragonFlyBSDProcess.h" |
| #include "DragonFlyBSDProcessList.h" |
| |
| #include <sys/types.h> |
| #include <sys/sysctl.h> |
| #include <sys/time.h> |
| #include <sys/resource.h> |
| #include <sys/proc.h> |
| #include <vm/vm_param.h> |
| #include <signal.h> |
| #include <fcntl.h> |
| #include <kvm.h> |
| #include <limits.h> |
| #include <time.h> |
| #include <math.h> |
| |
| /*{ |
| #include "bsd/Platform.h" |
| #include "Action.h" |
| #include "BatteryMeter.h" |
| #include "SignalsPanel.h" |
| |
| extern ProcessFieldData Process_fields[]; |
| |
| }*/ |
| |
| #ifndef CLAMP |
| #define CLAMP(x,low,high) (((x)>(high))?(high):(((x)<(low))?(low):(x))) |
| #endif |
| |
| ProcessField Platform_defaultFields[] = { PID, EFFECTIVE_USER, PRIORITY, NICE, M_SIZE, M_RESIDENT, STATE, PERCENT_CPU, PERCENT_MEM, TIME, COMM, 0 }; |
| |
| int Platform_numberOfFields = LAST_PROCESSFIELD; |
| |
| const SignalItem Platform_signals[] = { |
| { .name = "Cancel", .number = 0 }, |
| #define SIG(NAME) { .name = #NAME, .number = SIG##NAME } |
| SIG(HUP), |
| SIG(INT), |
| SIG(QUIT), |
| SIG(ILL), |
| SIG(TRAP), |
| SIG(ABRT), |
| SIG(EMT), |
| SIG(FPE), |
| SIG(KILL), |
| SIG(BUS), |
| SIG(SEGV), |
| SIG(SYS), |
| SIG(PIPE), |
| SIG(ALRM), |
| SIG(TERM), |
| SIG(URG), |
| SIG(STOP), |
| SIG(TSTP), |
| SIG(CONT), |
| SIG(CHLD), |
| SIG(TTIN), |
| SIG(TTOU), |
| SIG(IO), |
| SIG(XCPU), |
| SIG(XFSZ), |
| SIG(VTALRM), |
| SIG(PROF), |
| SIG(WINCH), |
| SIG(INFO), |
| SIG(USR1), |
| SIG(USR2), |
| SIG(THR), |
| #undef SIG |
| }; |
| |
| const unsigned int Platform_numberOfSignals = sizeof(Platform_signals)/sizeof(SignalItem); |
| |
| void Platform_setBindings(Htop_Action* keys) { |
| (void) keys; |
| } |
| |
| MeterClass* Platform_meterTypes[] = { |
| &CPUMeter_class, |
| &ClockMeter_class, |
| &LoadAverageMeter_class, |
| &LoadMeter_class, |
| &MemoryMeter_class, |
| &SwapMeter_class, |
| &TasksMeter_class, |
| &UptimeMeter_class, |
| &BatteryMeter_class, |
| &HostnameMeter_class, |
| &AllCPUsMeter_class, |
| &AllCPUs2Meter_class, |
| &LeftCPUsMeter_class, |
| &RightCPUsMeter_class, |
| &LeftCPUs2Meter_class, |
| &RightCPUs2Meter_class, |
| &BlankMeter_class, |
| NULL |
| }; |
| |
| void Platform_getLoadAverage(double* one, double* five, double* fifteen) { |
| struct loadavg loadAverage; |
| int mib[2] = { CTL_VM, VM_LOADAVG }; |
| size_t size = sizeof(loadAverage); |
| |
| int err = sysctl(mib, 2, &loadAverage, &size, NULL, 0); |
| if (err) { |
| *one = 0; |
| *five = 0; |
| *fifteen = 0; |
| return; |
| } |
| *one = (double) loadAverage.ldavg[0] / loadAverage.fscale; |
| *five = (double) loadAverage.ldavg[1] / loadAverage.fscale; |
| *fifteen = (double) loadAverage.ldavg[2] / loadAverage.fscale; |
| } |
| |
| int Platform_getMaxPid() { |
| return PID_MAX; |
| } |
| |
| double Platform_setCPUValues(Meter* this, int cpu) { |
| DragonFlyBSDProcessList* fpl = (DragonFlyBSDProcessList*) this->pl; |
| int cpus = this->pl->cpuCount; |
| CPUData* cpuData; |
| |
| if (cpus == 1) { |
| // single CPU box has everything in fpl->cpus[0] |
| cpuData = &(fpl->cpus[0]); |
| } else { |
| cpuData = &(fpl->cpus[cpu]); |
| } |
| |
| double percent; |
| double* v = this->values; |
| |
| v[CPU_METER_NICE] = cpuData->nicePercent; |
| v[CPU_METER_NORMAL] = cpuData->userPercent; |
| if (this->pl->settings->detailedCPUTime) { |
| v[CPU_METER_KERNEL] = cpuData->systemPercent; |
| v[CPU_METER_IRQ] = cpuData->irqPercent; |
| Meter_setItems(this, 4); |
| percent = v[0]+v[1]+v[2]+v[3]; |
| } else { |
| v[2] = cpuData->systemAllPercent; |
| Meter_setItems(this, 3); |
| percent = v[0]+v[1]+v[2]; |
| } |
| |
| percent = CLAMP(percent, 0.0, 100.0); |
| if (isnan(percent)) percent = 0.0; |
| return percent; |
| } |
| |
| void Platform_setMemoryValues(Meter* this) { |
| // TODO |
| ProcessList* pl = (ProcessList*) this->pl; |
| |
| this->total = pl->totalMem; |
| this->values[0] = pl->usedMem; |
| this->values[1] = pl->buffersMem; |
| this->values[2] = pl->cachedMem; |
| } |
| |
| void Platform_setSwapValues(Meter* this) { |
| ProcessList* pl = (ProcessList*) this->pl; |
| this->total = pl->totalSwap; |
| this->values[0] = pl->usedSwap; |
| } |
| |
| char **Platform_getProcessEnv(Process *proc) { |
| if(Process_isKernelProcess(proc)) return NULL; |
| |
| char **env = xMalloc(2 * sizeof(char *)); |
| env[1] = NULL; |
| char errmsg[_POSIX2_LINE_MAX]; |
| kvm_t *kvm = kvm_openfiles(NULL, "/dev/null", NULL, 0, errmsg); |
| if(!kvm) { |
| env[0] = xStrdup(errmsg); |
| return env; |
| } |
| int count; |
| struct kinfo_proc *kip = kvm_getprocs(kvm, KERN_PROC_PID, proc->pid, &count); |
| if(!kip || count < 1) { |
| const char *e = kip ? NULL : kvm_geterr(kvm); |
| if(e && *e) { |
| env[0] = xStrdup(e); |
| } else { |
| free(env); |
| env = NULL; |
| } |
| kvm_close(kvm); |
| return env; |
| } |
| char **v = kvm_getenvv(kvm, kip, 0); |
| if(!v) { |
| const char *e = kvm_geterr(kvm); |
| if(*e) { |
| env[0] = xStrdup(e); |
| } else { |
| free(env); |
| env = NULL; |
| } |
| kvm_close(kvm); |
| return env; |
| } |
| free(env); |
| count = 0; |
| while(v[count]) count++; |
| env = xMalloc((count + 1) * sizeof(char *)); |
| env[count] = NULL; |
| while(count-- > 0) env[count] = xStrdup(v[count]); |
| kvm_close(kvm); |
| return env; |
| } |