summaryrefslogtreecommitdiff
path: root/agent/mibgroup/ucd-snmp/memory_freebsd2.c
diff options
context:
space:
mode:
Diffstat (limited to 'agent/mibgroup/ucd-snmp/memory_freebsd2.c')
-rw-r--r--agent/mibgroup/ucd-snmp/memory_freebsd2.c389
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;
+}