/* * memory_freebsd2.c */ #include /* * Ripped from /usr/scr/usr.bin/vmstat/vmstat.c (covering all bases) */ #include #include #ifdef dragonfly #include #else #include #include #endif #ifdef freebsd5 #include #endif #include #include #include #include #include #include #include #include #include #if HAVE_SYS_VMPARAM_H #include #else #include #endif #include #include #include #include #include #include #include #include #include #include #include #include #include #include #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 . */ #include 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; }