aboutsummaryrefslogtreecommitdiff
path: root/src/cmd
diff options
context:
space:
mode:
Diffstat (limited to 'src/cmd')
-rw-r--r--src/cmd/auxstats/Darwin.c442
1 files changed, 301 insertions, 141 deletions
diff --git a/src/cmd/auxstats/Darwin.c b/src/cmd/auxstats/Darwin.c
index 46ba5cbf..94a8794d 100644
--- a/src/cmd/auxstats/Darwin.c
+++ b/src/cmd/auxstats/Darwin.c
@@ -1,10 +1,4 @@
-/*
- * No idea whether this will work. It does compile.
- */
-
#include <u.h>
-#include <kvm.h>
-#include <nlist.h>
#include <sys/types.h>
#include <sys/protosw.h>
#include <sys/socket.h>
@@ -15,119 +9,302 @@
#include <net/if_var.h>
#include <net/if_dl.h>
#include <net/if_types.h>
+#include <ifaddrs.h>
#include <sys/ioctl.h>
#include <limits.h>
#include <libc.h>
#include <bio.h>
+
+#include <mach/mach.h>
+#include <mach/mach_time.h>
+#include <CoreFoundation/CoreFoundation.h>
+#include <IOKit/ps/IOPowerSources.h>
+AUTOFRAMEWORK(CoreFoundation)
+AUTOFRAMEWORK(IOKit)
+
#include "dat.h"
+typedef struct Sample Sample;
+
+struct Sample
+{
+ uint seq;
+ host_cpu_load_info_data_t cpu, p_cpu;
+ vm_size_t pgsize;
+ double divisor;
+ uint64_t time, p_time;
+ vm_statistics_data_t vm_stat, p_vm_stat;
+ boolean_t purgeable_is_valid;
+ struct xsw_usage xsu;
+ boolean_t xsu_valid;
+ integer_t syscalls_mach, p_syscalls_mach;
+ integer_t syscalls_unix, p_syscalls_unix;
+ ulong csw, p_csw;
+ uint net_ifaces;
+ uvlong net_ipackets, p_net_ipackets;
+ uvlong net_opackets, p_net_opackets;
+ uvlong net_ibytes, p_net_ibytes;
+ uvlong net_obytes, p_net_obytes;
+ uvlong net_errors, p_net_errors;
+ ulong usecs;
+};
+
+static Sample sample;
+
+void xsample(int);
+void xapm(int);
void xloadavg(int);
void xcpu(int);
void xswap(int);
-void xsysctl(int);
+void xvm(int);
void xnet(int);
-void xkvm(int);
void (*statfn[])(int) =
{
- xkvm,
+ xsample,
+ xapm,
xloadavg,
xswap,
xcpu,
- xsysctl,
+ xvm,
xnet,
0
};
-static kvm_t *kvm;
-
-static struct nlist nl[] = {
- { "_ifnet" },
- { "_cp_time" },
- { "" },
-};
+static mach_port_t stat_port;
void
-kvminit(void)
+sampleinit(void)
{
- char buf[_POSIX2_LINE_MAX];
-
- if(kvm)
- return;
- kvm = kvm_openfiles(nil, nil, nil, OREAD, buf);
- if(kvm == nil)
- return;
- if(kvm_nlist(kvm, nl) < 0 || nl[0].n_type == 0){
- kvm = nil;
+ mach_timebase_info_data_t info;
+
+ if(stat_port)
return;
+
+ stat_port = mach_host_self();
+ memset(&sample, 0, sizeof sample);
+ if(host_page_size(stat_port, &sample.pgsize) != KERN_SUCCESS)
+ sample.pgsize = 4096;
+
+ // populate clock tick info for timestamps
+ mach_timebase_info(&info);
+ sample.divisor = 1000.0 * (double)info.denom/info.numer;
+ sample.time = mach_absolute_time();
+}
+
+void
+samplenet(void)
+{
+ struct ifaddrs *ifa_list, *ifa;
+
+ ifa_list = nil;
+ sample.net_ifaces = nil;
+ if(getifaddrs(&ifa_list) == 0){
+ sample.p_net_ipackets = sample.net_ipackets;
+ sample.p_net_opackets = sample.net_opackets;
+ sample.p_net_ibytes = sample.net_ibytes;
+ sample.p_net_obtypes = sample.net_obytes;
+ sample.p_net_errors = sample.net_errors;
+
+ sample.net_ipackets = 0;
+ sample.net_opackets = 0;
+ sample.net_ibytes = 0;
+ sample.net_obytes = 0;
+ sample.net_errors = 0;
+ sample.net_ifaces = 0;
+
+ for(ifa=ifa_list; ifa; ifa=ifa->next){
+ if(ifa->ifa_addr->sa_family != AF_LINK)
+ continue;
+ if((ifa->ifa_flags&(IFF_UP|IFF_RUNNING)) == 0)
+ continue;
+ if(ifa->ifa_data == nil)
+ continue;
+ if(strncmp(ifa->ifa_name, "lo", 2) == 0) /* loopback */
+ continue;
+
+ sample.net_ipackets += if_data->ifi_ipackets;
+ sample.net_opackets += if_data->ifi_opackets;
+ sample.net_ibytes += if_data->ifi_ibytes;
+ sample.net_obytes += if_data->ifi_obtypes;
+ sample.net_errors += if_data->ifi_ierrors + if_data->ifif_oerrors;
+ sample.net_ifaces++;
+ }
+ freeifaddrs(ifa_list);
}
}
+
+/*
+ * The following forces the program to be run with the suid root as
+ * all the other stat monitoring apps get set:
+ *
+ * -rwsr-xr-x 1 root wheel 83088 Mar 20 2005 /usr/bin/top
+ * -rwsrwxr-x 1 root admin 54048 Mar 20 2005
+ * /Applications/Utilities/Activity Monitor.app/Contents/Resources/pmTool
+ *
+ * If Darwin eventually encompases more into sysctl then this
+ * won't be required.
+ */
void
-xkvm(int first)
+sampevents(void)
{
- if(first)
- kvminit();
+ uint i, j, pcnt, tcnt;
+ mach_msg_type_number_t count;
+ kern_return_t error;
+ processor_set_t *psets, pset;
+ task_t *tasks;
+ task_events_info_data_t events;
+
+ if((error = host_processor_sets(stat_port, &psets, &pcnt)) != KERN_SUCCESS){
+ Bprint(&bout, "host_processor_sets: %s (make sure auxstats is setuid root)\n",
+ mach_error_string(error));
+ return;
+ }
+
+ sample.p_syscalls_mach = sample.syscalls_mach;
+ sample.p_syscalls_unix = sample.syscalls_unix;
+ sample.p_csw = sample.csw;
+
+ sample.syscalls_mach = 0;
+ sample.syscalls_unix = 0;
+ sample.csw = 0;
+
+ for(i=0; i<pcnt; i++){
+ if((error=host_processor_set_priv(stat_port, psets[i], &pset)) != KERN_SUCCESS){
+ Bprint(&bout, "host_processor_set_priv: %s\n", mach_error_string(error));
+ return;
+ }
+ if((error=host_processor_set_tasks(pset, &tasks, &tcnt)) != KERN_SUCCESS){
+ Bprint(&bout, "host_processor_set_tasks: %s\n", mach_error_string(error));
+ return;
+ }
+ for(j=0; j<tcnt; j++){
+ count = TASK_EVENTS_INFO_COUNT;
+ if(task_info(tasks[j], TASK_EVENTS_INFO, (task_info_t)&events, &count) == KERN_SUCCESS){
+ sample.syscalls_mach += events.syscalls_mach;
+ sample.syscalls_unix += events.syscalls_unix;
+ sample.csw += events.csw;
+ }
+
+ if(tasks[j] != mach_task_self())
+ mach_port_deallocate(mach_task_self(), tasks[j]);
+ }
+
+ if((error = vm_deallocate((vm_map_t)mach_task_self(),
+ (vm_address_t)tasks, tcnt*sizeof(task_t))) != KERN_SUCCESS){
+ Bprint(&bout, "vm_deallocate: %s\n", mach_error_string(error));
+ return;
+ }
+
+ if((error = mach_port_deallocate(mach_task_self(), pset)) != KERN_SUCCESS
+ || (error = mach_port_deallocate(mach_task_self(), psets[i])) != KERN_SUCCESS){
+ Bprint(&bout, "mach_port_deallocate: %s\n", mach_error_string(error));
+ return;
+ }
+ }
+
+ if((error = vm_deallocate((vm_map_t)mach_task_self(), (vm_address_t)psets,
+ pcnt*sizeof(processor_set_t))) != KERN_SUCCESS){
+ Bprint(&bout, "vm_deallocate: %s\n", mach_error_string(error));
+ return;
+ }
}
-int
-kread(ulong addr, char *buf, int size)
+void
+xsample(int first)
{
- if(kvm_read(kvm, addr, buf, size) != size){
- memset(buf, 0, size);
- return -1;
+ int mib[2];
+ mach_msg_type_number_t count;
+ kern_return_t error;
+ size_t len;
+
+ if(first){
+ sampleinit();
+ return;
}
- return size;
+
+ sample.seq++;
+ sample.p_time = sample.time;
+ sample.time = mach_absolute_time();
+
+ sample.p_vm_stat = sample.vm_stat;
+ count = sizeof(sample.vm_stat) / sizeof(natural_t);
+ host_statistics(stat_port, HOST_VM_INFO, (host_info_t)&sample.vm_stat, &count);
+
+ if(sample.seq == 1)
+ sample.p_vm_stat = sample.vm_stat;
+
+ mib[0] = CTL_VM;
+ mib[1] = VM_SWAPUSAGE;
+ len = sizeof sample.xsu;
+ sample.xsu_valid = TRUE;
+ if(sysctl(mib, 2, &asamp.xsu, &len, NULL, 0) < 0 && errno == ENOENT)
+ sample.xsu_value = FALSE;
+
+ samplenet();
+ sampleevents();
+
+ sample.p_cpu = sample.cpu;
+ count = HOST_CPU_LOAD_INFO_COUNT;
+ host_statistics(stat_port, HOST_CPU_LOAD_INFO, (host_info_t)&sample.cpu, &count);
+ sample.usecs = (double)(asamp.time - asamp.p_time)/asamp.divisor;
+ Bprint(&bout, "usecs %lud\n", sample.usecs);
}
void
-xnet(int first)
+xapm(int first)
{
-#if 0
- ulong out, in, outb, inb, err;
- static ulong ifnetaddr;
- ulong addr;
- struct ifnet ifnet;
- struct ifnethead ifnethead;
- char name[16];
+ int i, battery;
+ CFArrayRef array;
+ CFDictionaryRef dict;
+ CFTypeRef cf, src, value;
if(first)
return;
-
- if(ifnetaddr == 0){
- ifnetaddr = nl[0].n_value;
- if(ifnetaddr == 0)
- return;
+
+ src = IOPSCopyPowerSourcesInfo();
+ array = IOPSCopyPowerSourcesList(src);
+
+ for(i=0; i<CFArrayGetCount(array); i++){
+ cf = CFArrayGetValueAtIndex(array, i);
+ dict = IOPSGetPowerSourceDescription(src, cf);
+ if(dict != nil){
+ value = CFDictionaryGetValue(dict, CFSTR("Current Capacity"));
+ if(value != nil){
+ if(!CFNumberGetValue(value, kCFNumberIntType, &battery))
+ battery = 100;
+ Bprint(&bout, "battery =%d 100\n", battery);
+ break;
+ }
+ }
}
- if(kread(ifnetaddr, (char*)&ifnethead, sizeof ifnethead) < 0)
- return;
-
- out = in = outb = inb = err = 0;
- addr = (ulong)TAILQ_FIRST(&ifnethead);
- while(addr){
- if(kread(addr, (char*)&ifnet, sizeof ifnet) < 0
- || kread((ulong)ifnet.if_name, name, 16) < 0)
- return;
- name[15] = 0;
- addr = (ulong)TAILQ_NEXT(&ifnet, if_link);
- out += ifnet.if_opackets;
- in += ifnet.if_ipackets;
- outb += ifnet.if_obytes;
- inb += ifnet.if_ibytes;
- err += ifnet.if_oerrors+ifnet.if_ierrors;
- }
- Bprint(&bout, "etherin %lud\n", in);
- Bprint(&bout, "etherout %lud\n", out);
- Bprint(&bout, "etherinb %lud\n", inb);
- Bprint(&bout, "etheroutb %lud\n", outb);
- Bprint(&bout, "ethererr %lud\n", err);
- Bprint(&bout, "ether %lud\n", in+out);
- Bprint(&bout, "etherb %lud\n", inb+outb);
-#endif
- USED(first);
+ CFRelease(array);
+ CFRelease(src);
}
+void
+xnet(int first)
+{
+ uint n;
+ uvlong err, in, inb, out, outb;
+
+ n = sample.net_ifaces;
+ in = sample.net_ipackets;
+ out = sample.net_opackets;
+ inb = sample.net_ibytes;
+ outb = sample.net_obytes;
+ err = sample.net_errors;
+
+ Bprint(&bout, "etherb %llud %d\n", inb+outb, n*1000000);
+ Bprint(&bout, "ether %llud %d\n", in+out, n*1000);
+ Bprint(&bout, "ethererr %llud %d\n", err, n*1000);
+ Bprint(&bout, "etherin %llud %d\n", in, n*1000);
+ Bprint(&bout, "etherout %llud %d\n", out, n*1000);
+ Bprint(&bout, "etherinb %llud %d\n", inb, n*1000);
+ Bprint(&bout, "etheroutb %llud %d\n", outb, n*1000);
+}
int
rsys(char *name, char *buf, int len)
@@ -154,56 +331,60 @@ isys(char *name)
}
void
-xsysctl(int first)
+xvm(int first)
{
- static int pgsize;
- vlong t;
+ uvlong total;
- if(first){
- pgsize = isys("vm.stats.vm.v_page_size");
- if(pgsize == 0)
- pgsize = 4096;
- }
- if((t = isys("vm.stats.vm.v_page_count")) != 0)
- Bprint(&bout, "mem %lld %lld\n",
- isys("vm.stats.vm.v_active_count")*pgsize,
- t*pgsize);
- Bprint(&bout, "context %lld 1000\n", isys("vm.stats.sys.v_swtch"));
- Bprint(&bout, "syscall %lld 1000\n", isys("vm.stats.sys.v_syscall"));
- Bprint(&bout, "intr %lld 1000\n", isys("vm.stats.sys.v_intr")+isys("vm.stats.sys.v_trap"));
- Bprint(&bout, "fault %lld 1000\n", isys("vm.stats.vm.v_vm_faults"));
- Bprint(&bout, "fork %lld 1000\n", isys("vm.stats.vm.v_forks")
- +isys("vm.stats.vm.v_rforks")
+ if(first)
+ return;
+
+ total = sample.vm_stat.free_count
+ + sample.vm_stat.active_count
+ + sample.vm_stat.inactive_count
+ + sample.vm_stat.wire_count;
+ if(total)
+ Bprint(&bout, "mem =%lld %lld\n", sample.vm_stat_active_count, total);
+
+ Bprint(&bout, "context %lld 1000\n", (vlong)sample.csw);
+ Bprint(&bout, "syscall %lld 1000\n", (vlong)sample.syscalls_mach+sample.syscalls_unix);
+ Bprint(&bout, "intr %lld 1000\n",
+ isys("vm.stats.sys.v_intr")
+ +isys("vm.stats.sys.v_trap"));
+
+ Bprint(&bout, "fault %lld 1000\n", sample.vm_stats.faults);
+ Bprint(&bout, "fork %lld 1000\n",
+ isys("vm.stats.vm.v_rforks")
+isys("vm.stats.vm.v_vforks"));
+
+// Bprint(&bout, "hits %lud of %lud lookups (%d%% hit rate)\n",
+// (asamp.vm_stat.hits),
+// (asamp.vm_stat.lookups),
+// (natural_t)(((double)asamp.vm_stat.hits*100)/ (double)asamp.vm_stat.lookups));
}
void
xcpu(int first)
{
-#if 0
- static int stathz;
- ulong x[20];
- struct clockinfo *ci;
- int n;
-
- if(first){
- if(rsys("kern.clockrate", (char*)&x, sizeof x) < sizeof ci)
- stathz = 128;
- else{
- ci = (struct clockinfo*)x;
- stathz = ci->stathz;
- }
- return;
- }
+ ulong user, sys, idle, nice, t;
- if((n=rsys("kern.cp_time", (char*)x, sizeof x)) < 5*sizeof(ulong))
+ if(first)
return;
- Bprint(&bout, "user %lud %d\n", x[CP_USER]+x[CP_NICE], stathz);
- Bprint(&bout, "sys %lud %d\n", x[CP_SYS], stathz);
- Bprint(&bout, "cpu %lud %d\n", x[CP_USER]+x[CP_NICE]+x[CP_SYS], stathz);
- Bprint(&bout, "idle %lud %d\n", x[CP_IDLE], stathz);
-#endif
+ sys = sample.cpu.cpu_ticks[CPU_STATE_SYSTEM] -
+ sample.p_cpu.cpu_ticks[CPU_STATE_SYSTEM];
+ idle = sample.cpu.cpu_ticks[CPU_STATE_IDLE] -
+ sample.p_cpu.cpu_ticks[CPU_STATE_IDLE];
+ user = sample.cpu.cpu_ticks[CPU_STATE_USER] -
+ sample.p_cpu.cpu_ticks[CPU_STATE_USER];
+ nice = sample.cpu.cpu_ticks[CPU_STATE_NICE] -
+ sample.p_cpu.cpu_ticks[CPU_STATE_NICE];
+
+ t = sys+idle+user+nice;
+
+ Bprint(&bout, "user =%lud %lud\n", user, t);
+ Bprint(&bout, "sys =%lud %lud\n", sys, t);
+ Bprint(&bout, "idle =%lud %lud\n", idle, t);
+ Bprint(&bout, "nice =%lud %lud\n", nice, t);
}
void
@@ -222,32 +403,11 @@ xloadavg(int first)
void
xswap(int first)
{
-#if 0
- static struct kvm_swap s;
- static ulong pgin, pgout;
- int i, o;
- static int pgsize;
-
- if(first){
- pgsize = getpagesize();
- if(pgsize == 0)
- pgsize = 4096;
- return;
- }
-
- if(kvm == nil)
+ if(first)
return;
-
- i = isys("vm.stats.vm.v_swappgsin");
- o = isys("vm.stats.vm.v_swappgsout");
- if(i != pgin || o != pgout){
- pgin = i;
- pgout = o;
- kvm_getswapinfo(kvm, &s, 1, 0);
- }
-
-
- Bprint(&bout, "swap %lld %lld\n", s.ksw_used*(vlong)pgsize, s.ksw_total*(vlong)pgsize);
-#endif
+
+ if(sample.xsu_valid)
+ Bprint(&bout, "swap %lld %lld\n",
+ (vlong)samle.xsu.xsu_used,
+ (vlong)sample.xsu.xsu_total);
}
-