/*
* Copyright (C), 2000-2007 by the monit project group.
* All Rights Reserved.
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
#include <config.h>
#ifdef HAVE_STRING_H
#include <string.h>
#endif
#ifdef HAVE_ERRNO_H
#include <errno.h>
#endif
#ifdef HAVE_FCNTL_H
#include <fcntl.h>
#endif
#ifdef HAVE_KVM_H
#include <kvm.h>
#endif
#ifdef HAVE_SYS_PARAM_H
#include <sys/param.h>
#endif
#ifdef HAVE_SYS_PROC_H
#include <sys/proc.h>
#endif
#ifdef HAVE_UVM_UVM_H
#include <uvm/uvm.h>
#endif
#ifdef HAVE_SYS_SYSCTL_H
#include <sys/sysctl.h>
#endif
#ifdef HAVE_SYS_DKSTAT_H
#include <sys/dkstat.h>
#endif
#include "monitor.h"
#include "process.h"
#include "process_sysdep.h"
/**
* System dependent resource gathering code for OpenBSD.
*
* @author Margarida Sequeira <margarida@openbsd.org>
* @author Jan-Henrik Haukeland, <hauk@tildeslash.com>
* @author Christian Hopp <chopp@iei.tu-clausthal.de>
* @author Rory Toma <rory@digeo.com>
* @author Martin Pala<martinp@tildeslash.com>
*
* @version \$Id: sysdep_OPENBSD.c,v 1.29 2007/07/25 12:54:32 hauk Exp $
*
* @file
*/
/* ----------------------------------------------------------------- Private */
static int hz;
static int pagesize_kbyte;
static long total_old = 0;
static long cpu_user_old = 0;
static long cpu_syst_old = 0;
/* ------------------------------------------------------------------ Public */
int init_process_info_sysdep(void) {
int mib[2];
size_t len;
struct clockinfo clock;
mib[0] = CTL_KERN;
mib[1] = KERN_CLOCKRATE;
len = sizeof(clock);
if(sysctl(mib, 2, &clock, &len, NULL, 0) == -1)
{
DEBUG("system statistic error -- cannot get clock rate: %s\n", STRERROR);
return FALSE;
}
hz = clock.hz;
mib[0] = CTL_HW;
mib[1] = HW_NCPU;
len = sizeof(num_cpus);
if(sysctl(mib, 2, &num_cpus, &len, NULL, 0) == -1)
{
DEBUG("system statistic error -- cannot get cpu count: %s\n", STRERROR);
return FALSE;
}
mib[1] = HW_PHYSMEM;
len = sizeof(systeminfo.mem_kbyte_max);
if(sysctl(mib, 2, &systeminfo.mem_kbyte_max, &len, NULL, 0) == -1)
{
DEBUG("system statistic error -- cannot get real memory amount: %s\n", STRERROR);
return FALSE;
}
systeminfo.mem_kbyte_max /= 1024;
mib[1] = HW_PAGESIZE;
len = sizeof(pagesize_kbyte);
if(sysctl(mib, 2, &pagesize_kbyte, &len, NULL, 0) == -1)
{
DEBUG("system statistic error -- cannot get memory page size: %s\n", STRERROR);
return FALSE;
}
pagesize_kbyte /= 1024;
return TRUE;
}
/**
* Read all processes to initialize the information tree.
* @param reference reference of ProcessTree
* @return treesize>0 if succeeded otherwise =0.
*/
int initprocesstree_sysdep(ProcessTree_T ** reference) {
int i;
int treesize;
static kvm_t *kvm_handle;
ProcessTree_T *pt;
struct kinfo_proc *pinfo;
if(getuid()!=0) {
LogError("system statistic error -- permission denied\n");
return FALSE;
}
if(!(kvm_handle = kvm_open(NULL, NULL, NULL, O_RDONLY, NULL)))
{
LogError("system statistic error -- cannot initialize kvm interface\n");
return FALSE;
}
pinfo = kvm_getprocs(kvm_handle, KERN_PROC_ALL, 0, &treesize);
if(!pinfo || (treesize < 1))
{
LogError("system statistic error -- cannot get process tree\n");
kvm_close(kvm_handle);
return FALSE;
}
pt = xcalloc(sizeof(ProcessTree_T), treesize);
for(i = 0; i < treesize; i++)
{
pt[i].pid = pinfo[i].kp_proc.p_pid;
pt[i].ppid = pinfo[i].kp_eproc.e_ppid;
pt[i].cputime = (long)((pinfo[i].kp_proc.p_rtime.tv_sec * 10) +
(pinfo[i].kp_proc.p_rtime.tv_usec / 100000)
);
pt[i].mem_kbyte = (unsigned long)(pinfo[i].kp_eproc.e_vm.vm_rssize * pagesize_kbyte);
if(pinfo[i].kp_proc.p_stat == SZOMB)
{
pt[i].status_flag |= PROCESS_ZOMBIE;
}
pt[i].time = get_float_time();
}
*reference = pt;
kvm_close(kvm_handle);
return treesize;
}
/**
* This routine returns 'nelem' double precision floats containing
* the load averages in 'loadv'; at most 3 values will be returned.
* @param loadv destination of the load averages
* @param nelem number of averages
* @return: 0 if successful, -1 if failed (and all load averages are 0).
*/
int getloadavg_sysdep (double *loadv, int nelem) {
return getloadavg(loadv, nelem);
}
/**
* This routine returns kbyte of real memory in use.
* @return: TRUE if successful, FALSE if failed (or not available)
*/
int used_system_memory_sysdep(SystemInfo_T *si) {
int mib[] = {CTL_VM, VM_METER};
size_t len = sizeof(struct vmtotal);
struct vmtotal vm;
if(sysctl(mib, 2, &vm, &len, NULL, 0) == -1)
{
LogError("system statistic error -- cannot get real memory usage: %s\n",
STRERROR);
return FALSE;
}
si->total_mem_kbyte = (unsigned long)(vm.t_arm * pagesize_kbyte);
return TRUE;
}
/**
* This routine returns system/user CPU time in use.
* @return: TRUE if successful, FALSE if failed
*/
int used_system_cpu_sysdep(SystemInfo_T *si) {
int i;
int mib[] = {CTL_KERN, KERN_CPTIME};
long cp_time[CPUSTATES];
long total_new = 0;
long total;
size_t len;
len = sizeof(cp_time);
if(sysctl(mib, 2, &cp_time, &len, NULL, 0) == -1)
{
LogError("system statistic error -- cannot get cpu time: %s\n", STRERROR);
return FALSE;
}
for(i = 0; i < CPUSTATES; i++)
{
total_new += cp_time[i];
}
total = total_new - total_old;
total_old = total_new;
si->total_cpu_user_percent =
(total > 0)?(int)(1000 * (double)(cp_time[CP_USER] - cpu_user_old) / total):0;
si->total_cpu_syst_percent =
(total > 0)?(int)(1000 * (double)(cp_time[CP_SYS] - cpu_syst_old) / total):0;
si->total_cpu_wait_percent =
0; /* there is no wait statistic available */
cpu_user_old = cp_time[CP_USER];
cpu_syst_old = cp_time[CP_SYS];
return TRUE;
}
syntax highlighted by Code2HTML, v. 0.9.1