diff options
Diffstat (limited to 'agent/mibgroup/ucd-snmp/memory_freebsd2.c')
-rw-r--r-- | agent/mibgroup/ucd-snmp/memory_freebsd2.c | 389 |
1 files changed, 389 insertions, 0 deletions
diff --git a/agent/mibgroup/ucd-snmp/memory_freebsd2.c b/agent/mibgroup/ucd-snmp/memory_freebsd2.c new file mode 100644 index 0000000..22dde4c --- /dev/null +++ b/agent/mibgroup/ucd-snmp/memory_freebsd2.c @@ -0,0 +1,389 @@ +/* + * memory_freebsd2.c + */ + +#include <net-snmp/net-snmp-config.h> + + +/* + * Ripped from /usr/scr/usr.bin/vmstat/vmstat.c (covering all bases) + */ +#include <sys/param.h> +#include <sys/time.h> +#ifdef dragonfly +#include <sys/user.h> +#else +#include <sys/proc.h> +#include <sys/dkstat.h> +#endif +#ifdef freebsd5 +#include <sys/bio.h> +#endif +#include <sys/buf.h> +#include <sys/uio.h> +#include <sys/namei.h> +#include <sys/malloc.h> +#include <sys/signal.h> +#include <sys/fcntl.h> +#include <sys/ioctl.h> +#include <sys/sysctl.h> +#include <sys/vmmeter.h> + +#if HAVE_SYS_VMPARAM_H +#include <sys/vmparam.h> +#else +#include <vm/vm_param.h> +#endif + +#include <time.h> +#include <nlist.h> +#include <kvm.h> +#include <errno.h> +#include <unistd.h> +#include <stdio.h> +#include <ctype.h> +#include <stdlib.h> +#include <string.h> +#include <paths.h> +#include <limits.h> + +#include <net-snmp/net-snmp-includes.h> +#include <net-snmp/agent/net-snmp-agent-includes.h> +#include <net-snmp/agent/auto_nlist.h> + +#include "util_funcs.h" +#include "memory.h" +#include "memory_freebsd2.h" + +/* + * nlist symbols + */ +#define SUM_SYMBOL "cnt" +#ifndef openbsd2 +#define BUFSPACE_SYMBOL "bufspace" +#endif + +/* + * Default swap warning limit (kb) + */ +#define DEFAULTMINIMUMSWAP 16000 + +/* + * Swap warning limit + */ +long minimumswap; + +/* + * Swap info + */ +quad_t swapTotal; +quad_t swapUsed; +quad_t swapFree; + +static FindVarMethod var_extensible_mem; + +void +init_memory_freebsd2(void) +{ + + struct variable2 extensible_mem_variables[] = { + {MIBINDEX, ASN_INTEGER, NETSNMP_OLDAPI_RONLY, + var_extensible_mem, 1, {MIBINDEX}}, + {ERRORNAME, ASN_OCTET_STR, NETSNMP_OLDAPI_RONLY, + var_extensible_mem, 1, {ERRORNAME}}, + {MEMTOTALSWAP, ASN_INTEGER, NETSNMP_OLDAPI_RONLY, + var_extensible_mem, 1, {MEMTOTALSWAP}}, + {MEMAVAILSWAP, ASN_INTEGER, NETSNMP_OLDAPI_RONLY, + var_extensible_mem, 1, {MEMAVAILSWAP}}, + {MEMTOTALREAL, ASN_INTEGER, NETSNMP_OLDAPI_RONLY, + var_extensible_mem, 1, {MEMTOTALREAL}}, + {MEMAVAILREAL, ASN_INTEGER, NETSNMP_OLDAPI_RONLY, + var_extensible_mem, 1, {MEMAVAILREAL}}, + {MEMTOTALSWAPTXT, ASN_INTEGER, NETSNMP_OLDAPI_RONLY, + var_extensible_mem, 1, {MEMTOTALSWAPTXT}}, + {MEMUSEDSWAPTXT, ASN_INTEGER, NETSNMP_OLDAPI_RONLY, + var_extensible_mem, 1, {MEMUSEDSWAPTXT}}, + {MEMTOTALREALTXT, ASN_INTEGER, NETSNMP_OLDAPI_RONLY, + var_extensible_mem, 1, {MEMTOTALREALTXT}}, + {MEMUSEDREALTXT, ASN_INTEGER, NETSNMP_OLDAPI_RONLY, + var_extensible_mem, 1, {MEMUSEDREALTXT}}, + {MEMTOTALFREE, ASN_INTEGER, NETSNMP_OLDAPI_RONLY, + var_extensible_mem, 1, {MEMTOTALFREE}}, + {MEMSWAPMINIMUM, ASN_INTEGER, NETSNMP_OLDAPI_RONLY, + var_extensible_mem, 1, {MEMSWAPMINIMUM}}, + {MEMSHARED, ASN_INTEGER, NETSNMP_OLDAPI_RONLY, + var_extensible_mem, 1, {MEMSHARED}}, + {MEMBUFFER, ASN_INTEGER, NETSNMP_OLDAPI_RONLY, + var_extensible_mem, 1, {MEMBUFFER}}, + {MEMCACHED, ASN_INTEGER, NETSNMP_OLDAPI_RONLY, + var_extensible_mem, 1, {MEMCACHED}}, + {ERRORFLAG, ASN_INTEGER, NETSNMP_OLDAPI_RONLY, + var_extensible_mem, 1, {ERRORFLAG}}, + {ERRORMSG, ASN_OCTET_STR, NETSNMP_OLDAPI_RONLY, + var_extensible_mem, 1, {ERRORMSG}} + }; + + /* + * Define the OID pointer to the top of the mib tree that we're + * registering underneath + */ + oid mem_variables_oid[] = { NETSNMP_UCDAVIS_MIB, NETSNMP_MEMMIBNUM }; + + /* + * register ourselves with the agent to handle our mib tree + */ + REGISTER_MIB("ucd-snmp/memory", extensible_mem_variables, variable2, + mem_variables_oid); + + snmpd_register_config_handler("swap", memory_parse_config, + memory_free_config, "min-avail"); +} + + +void +memory_parse_config(const char *token, char *cptr) +{ + minimumswap = atoi(cptr); +} + +void +memory_free_config(void) +{ + minimumswap = DEFAULTMINIMUMSWAP; +} + +#ifndef freebsd4 +/* + * Executes swapinfo and parses last line + */ +/* + * This is just way too ugly ;) + */ + +void +swapmode(void) +{ + struct extensible ext; + int fd; + FILE *file; + + strcpy(ext.command, "/usr/sbin/swapinfo -k"); + + if ((fd = get_exec_output(&ext)) != -1) { + file = fdopen(fd, "r"); + + while (fgets(ext.output, sizeof(ext.output), file) != NULL); + + fclose(file); + wait_on_exec(&ext); + + sscanf(ext.output, "%*s%*d%qd%qd", &swapUsed, &swapFree); + + swapTotal = swapUsed + swapFree; + } +} +#else +/* + * swapmode is based on a program called swapinfo written + * by Kevin Lahey <kml@rokkaku.atl.ga.us>. + */ + +#include <sys/conf.h> + +void +swapmode(void) +{ + int pagesize; + int i, n; + static kvm_t *kd = NULL; + struct kvm_swap kswap[16]; + + if (kd == NULL) + kd = kvm_openfiles(NULL, NULL, NULL, O_RDONLY, NULL); + + n = kvm_getswapinfo(kd, kswap, sizeof(kswap) / sizeof(kswap[0]), 0); + + swapUsed = swapTotal = swapFree = 0; + /* + * Count up free swap space. + */ + for (i = 0; i < n; ++i) + swapFree += kswap[i].ksw_total - kswap[i].ksw_used; + + /* + * Count up total swap space + */ + for (i = 0; i < n; i++) + swapTotal += kswap[i].ksw_total; + + /* + * Calculate used swap space + */ + swapUsed = swapTotal - swapFree; + + /* + * Convert to kb + */ + pagesize = getpagesize() / 1024; + + swapTotal *= pagesize; + swapUsed *= pagesize; + swapFree *= pagesize; +} +#endif + + +/* + * var_extensible_mem(... + * Arguments: + * vp IN - pointer to variable entry that points here + * name IN/OUT - IN/name requested, OUT/name found + * length IN/OUT - length of IN/OUT oid's + * exact IN - TRUE if an exact match was requested + * var_len OUT - length of variable or 0 if function returned + * write_method + * + */ + +static unsigned char * +var_extensible_mem(struct variable *vp, + oid * name, + size_t * length, + int exact, + size_t * var_len, WriteMethod ** write_method) +{ + static long long_ret; + static char errmsg[1024]; + +#ifdef dragonfly + static struct vmstats mem; + size_t vmstats_size = sizeof(mem); +#endif + static struct vmmeter mem; +#endif + static struct vmtotal total; + + size_t total_size = sizeof(total); + int total_mib[] = { CTL_VM, VM_METER }; + + u_long phys_mem; + size_t phys_mem_size = sizeof(phys_mem); + int phys_mem_mib[] = { CTL_HW, HW_PHYSMEM }; + +#ifdef BUFSPACE_SYMBOL + long bufspace; +#endif + + if (header_generic(vp, name, length, exact, var_len, write_method)) + return (NULL); + + /* + * Memory info + */ +#ifdef dragonfly + sysctlbyname("vm.vmstats", &vmstats, &vmstats_size, NULL, 0); +#else + auto_nlist(SUM_SYMBOL, (char *) &mem, sizeof(mem)); +#endif + sysctl(total_mib, 2, &total, &total_size, NULL, 0); + + /* + * Swap info + */ + swapmode(); + /* + * getSwap(); + */ + + /* + * Physical memory + */ + sysctl(phys_mem_mib, 2, &phys_mem, &phys_mem_size, NULL, 0); + +#ifdef BUFSPACE_SYMBOL + /* + * Buffer space + */ + auto_nlist(BUFSPACE_SYMBOL, (char *) &bufspace, sizeof(bufspace)); +#endif + + long_ret = 0; /* set to 0 as default */ + + /* + * Page-to-kb macro + */ +#define ptok(p) ((p) * (mem.v_page_size >> 10)) + + switch (vp->magic) { + case MIBINDEX: + long_ret = 0; + return ((u_char *) (&long_ret)); + case ERRORNAME: /* dummy name */ + sprintf(errmsg, "swap"); + *var_len = strlen(errmsg); + return ((u_char *) (errmsg)); + case MEMTOTALSWAP: + long_ret = swapTotal; + return ((u_char *) (&long_ret)); + case MEMAVAILSWAP: /* FREE swap memory */ + long_ret = swapFree; + return ((u_char *) (&long_ret)); + case MEMTOTALREAL: + long_ret = phys_mem >> 10; + return ((u_char *) (&long_ret)); + case MEMAVAILREAL: /* FREE real memory */ + long_ret = ptok(mem.v_free_count); + return ((u_char *) (&long_ret)); + + /* + * these are not implemented + */ + case MEMTOTALSWAPTXT: + case MEMUSEDSWAPTXT: + case MEMTOTALREALTXT: + case MEMUSEDREALTXT: +#if NETSNMP_NO_DUMMY_VALUES + return NULL; +#endif + long_ret = -1; + return ((u_char *) (&long_ret)); + + case MEMTOTALFREE: + long_ret = ptok((int) total.t_free); + return ((u_char *) (&long_ret)); + case MEMSWAPMINIMUM: + long_ret = minimumswap; + return ((u_char *) (&long_ret)); + case MEMSHARED: + long_ret = ptok(total.t_vmshr + + total.t_avmshr + total.t_rmshr + total.t_armshr); + return ((u_char *) (&long_ret)); +#ifdef BUFSPACE_SYMBOL + case MEMBUFFER: + long_ret = bufspace >> 10; + return ((u_char *) (&long_ret)); +#endif +#ifndef openbsd2 + case MEMCACHED: +#ifdef darwin + long_ret = ptok(mem.v_lookups); +#elif defined(dragonfly) + long_ret = ptok(mem.v_cache_count); +#else + long_ret = ptok(mem.v_cache_count) + ptok(mem.v_inactive_count); +#endif + return ((u_char *) (&long_ret)); +#endif + case ERRORFLAG: + long_ret = (swapFree > minimumswap) ? 0 : 1; + return ((u_char *) (&long_ret)); + case ERRORMSG: + if (swapFree < minimumswap) + sprintf(errmsg, "Running out of swap space (%qd)", swapFree); + else + errmsg[0] = 0; + *var_len = strlen(errmsg); + return ((u_char *) (errmsg)); + } + return NULL; +} |