#include "stdinc.h" #include "dat.h" #include "fns.h" int collectstats = 1; /* keep in sync with dat.h:/NStat */ Statdesc statdesc[NStat] = { { "rpc total", }, { "rpc reads", }, { "rpc reads ok", }, { "rpc reads failed", }, { "rpc read bytes", }, { "rpc read time", }, { "rpc read cached", }, { "rpc read cached time", }, { "rpc read uncached", }, { "rpc read uncached time "}, { "rpc writes", }, { "rpc writes new", }, { "rpc writes old", }, { "rpc writes failed", }, { "rpc write bytes", }, { "rpc write time", }, { "rpc write new time", }, { "rpc write old time", }, { "lump cache hits", }, { "lump cache misses", }, { "lump cache reads", }, { "lump cache writes", }, { "lump cache size", }, { "lump cache stall", }, { "lump cache read time", }, { "disk cache hits", }, { "disk cache misses", }, { "disk cache lookups", }, { "disk cache reads", }, { "disk cache writes", }, { "disk cache dirty", }, { "disk cache size", }, { "disk cache flushes", }, { "disk cache stalls", }, { "disk cache lookup time", }, { "disk block stalls", }, { "lump stalls", }, { "index cache hits", }, { "index cache misses", }, { "index cache reads", }, { "index cache writes", }, { "index cache fills", }, { "index cache prefetches", }, { "index cache dirty", }, { "index cache size", }, { "index cache flushes", }, { "index cache stalls", }, { "index cache read time", }, { "index cache lookups" }, { "index cache summary hits" }, { "index cache summary prefetches" }, { "bloom filter hits", }, { "bloom filter misses", }, { "bloom filter false misses", }, { "bloom filter lookups", }, { "bloom filter ones", }, { "bloom filter bits", }, { "arena block reads", }, { "arena block read bytes", }, { "arena block writes", }, { "arena block write bytes", }, { "isect block reads", }, { "isect block read bytes", }, { "isect block writes", }, { "isect block write bytes", }, { "sum reads", }, { "sum read bytes", }, { "cig loads" }, { "cig load time" }, }; QLock statslock; Stats stats; Stats *stathist; int nstathist; ulong statind; ulong stattime; void statsproc(void *v) { USED(v); for(;;){ stats.now = time(0); stathist[stattime%nstathist] = stats; stattime++; sleep(1000); } } void statsinit(void) { nstathist = 90000; stathist = MKNZ(Stats, nstathist); vtproc(statsproc, nil); } void setstat(int index, long val) { qlock(&statslock); stats.n[index] = val; qunlock(&statslock); } void addstat(int index, int inc) { if(!collectstats) return; qlock(&statslock); stats.n[index] += inc; qunlock(&statslock); } void addstat2(int index, int inc, int index1, int inc1) { if(!collectstats) return; qlock(&statslock); stats.n[index] += inc; stats.n[index1] += inc1; qunlock(&statslock); } void printstats(void) { } void binstats(long (*fn)(Stats *s0, Stats *s1, void *arg), void *arg, long t0, long t1, Statbin *bin, int nbin) { long xt0, t, te, v; int i, j, lo, hi, m; vlong tot; Statbin *b; t = stats.now; /* negative times mean relative to now. */ if(t0 <= 0) t0 += t; if(t1 <= 0) t1 += t; /* ten minute range if none given */ if(t1 <= t0) t0 = t1 - 60*10; if(0) fprint(2, "stats %ld-%ld\n", t0, t1); /* binary search to find t0-1 or close */ lo = stattime; hi = stattime+nstathist; while(lo+1 < hi){ m = (lo+hi)/2; if(stathist[m%nstathist].now >= t0) hi = m; else lo = m; } xt0 = stathist[lo%nstathist].now; if(xt0 >= t1){ /* no samples */ memset(bin, 0, nbin*sizeof bin[0]); return; } hi = stattime+nstathist; j = lo+1; for(i=0; i<nbin; i++){ te = t0 + (t1-t0)*i/nbin; b = &bin[i]; memset(b, 0, sizeof *b); tot = 0; for(; j<hi && stathist[j%nstathist].now<te; j++){ v = fn(&stathist[(j-1)%nstathist], &stathist[j%nstathist], arg); if(b->nsamp==0 || v < b->min) b->min = v; if(b->nsamp==0 || v > b->max) b->max = v; tot += v; b->nsamp++; } if(b->nsamp) b->avg = tot / b->nsamp; if(b->nsamp==0 && i>0) *b = bin[i-1]; } }