/* * $Id: StatHist.cc,v 1.34 2007/04/28 22:26:37 hno Exp $ * * DEBUG: section 62 Generic Histogram * AUTHOR: Duane Wessels * * SQUID Web Proxy Cache http://www.squid-cache.org/ * ---------------------------------------------------------- * * Squid is the result of efforts by numerous individuals from * the Internet community; see the CONTRIBUTORS file for full * details. Many organizations have provided support for Squid's * development; see the SPONSORS file for full details. Squid is * Copyrighted (C) 2001 by the Regents of the University of * California; see the COPYRIGHT file for full details. Squid * incorporates software developed and/or copyrighted by other * sources; see the CREDITS file for full details. * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111, USA. * */ /* * Important restrictions on val_in and val_out functions: * * - val_in: ascending, defined on [0, oo), val_in(0) == 0; * - val_out: x == val_out(val_in(x)) where val_in(x) is defined * * In practice, the requirements are less strict, * but then it gets hard to define them without math notation. * val_in is applied after offseting the value but before scaling * See log and linear based histograms for examples */ #include "squid.h" #include "Store.h" /* Local functions */ static void statHistInit(StatHist * H, int capacity, hbase_f * val_in, hbase_f * val_out, double min, double max); static int statHistBin(const StatHist * H, double v); static double statHistVal(const StatHist * H, int bin); static StatHistBinDumper statHistBinDumper; #if !defined(_SQUID_HPUX_) || !defined(__GNUC__) /* * HP-UX and GCC (2.8?) give strange errors when these simple * functions are static. */ static hbase_f Log; static hbase_f Exp; static hbase_f Null; #endif /* low level init, higher level functions has less params */ static void statHistInit(StatHist * H, int capacity, hbase_f * val_in, hbase_f * val_out, double min, double max) { assert(H); assert(capacity > 0); assert(val_in && val_out); /* check before we divide to get scale */ assert(val_in(max - min) > 0); H->bins = (int *)xcalloc(capacity, sizeof(int)); H->min = min; H->max = max; H->capacity = capacity; H->scale = capacity / val_in(max - min); H->val_in = val_in; H->val_out = val_out; /* HPUX users: If you get one of the assertions below, please send * [at least] the values of all variables involved in the assertions * when reporting a bug! */ /* check that functions are valid */ /* a min value should go into bin[0] */ assert(statHistBin(H, min) == 0); /* a max value should go into the last bin */ assert(statHistBin(H, max) == H->capacity - 1); /* it is hard to test val_out, here is a crude test */ assert(((int) floor(0.99 + statHistVal(H, 0) - min)) == 0); } void statHistClean(StatHist * H) { xfree(H->bins); H->bins = NULL; } /* assumes that somebody already called init for Dest */ void statHistCopy(StatHist * Dest, const StatHist * Orig) { assert(Dest); assert(Orig); debugs(62, 3, "statHistCopy: Dest=" << Dest << ", Orig=" << Orig); assert(Dest->bins); /* better be safe than sorry */ debugs(62, 3, "statHistCopy: capacity " << Dest->capacity << " " << Orig->capacity); assert(Dest->capacity == Orig->capacity); debugs(62, 3, "statHistCopy: min " << Dest->min << " " << Orig->min ); assert(Dest->min == Orig->min); debugs(62, 3, "statHistCopy: max " << Dest->max << " " << Orig->max ); assert(Dest->max == Orig->max); debugs(62, 3, "statHistCopy: scale " << Dest->scale << " " << Orig->scale ); assert(fabs(Dest->scale - Orig->scale) < 0.0000001); assert(Dest->val_in == Orig->val_in); assert(Dest->val_out == Orig->val_out); /* actual copy */ debugs(62, 3, "statHistCopy: copying " << (long int) (Dest->capacity * sizeof(*Dest->bins)) << " bytes to " << Dest->bins << " from " << Orig->bins); xmemcpy(Dest->bins, Orig->bins, Dest->capacity * sizeof(*Dest->bins)); } /* * same as statHistCopy but will do nothing if capacities do not match; the * latter happens, for example, when #peers changes during reconfiguration; * if it happens too often we should think about more general solution.. */ void statHistSafeCopy(StatHist * Dest, const StatHist * Orig) { assert(Dest && Orig); assert(Dest->bins); if (Dest->capacity == Orig->capacity) statHistCopy(Dest, Orig); } void statHistCount(StatHist * H, double val) { const int bin = statHistBin(H, val); assert(H->bins); /* make sure it got initialized */ assert(0 <= bin && bin < H->capacity); H->bins[bin]++; } static int statHistBin(const StatHist * H, double v) { int bin; #if BROKEN_STAT_HIST_BIN return 0; /* NOTREACHED */ #endif v -= H->min; /* offset */ if (v <= 0.0) /* too small */ return 0; bin = (int) floor(H->scale * H->val_in(v) + 0.5); if (bin < 0) /* should not happen */ bin = 0; if (bin >= H->capacity) /* too big */ bin = H->capacity - 1; return bin; } static double statHistVal(const StatHist * H, int bin) { return H->val_out((double) bin / H->scale) + H->min; } double statHistDeltaMedian(const StatHist * A, const StatHist * B) { return statHistDeltaPctile(A, B, 0.5); } double statHistDeltaPctile(const StatHist * A, const StatHist * B, double pctile) { int i; int s1 = 0; int h = 0; int a = 0; int b = 0; int I = 0; int J = A->capacity; int K; double f; int *D = (int *)xcalloc(A->capacity, sizeof(int)); assert(A->capacity == B->capacity); for (i = 0; i < A->capacity; i++) { D[i] = B->bins[i] - A->bins[i]; assert(D[i] >= 0); } for (i = 0; i < A->capacity; i++) s1 += D[i]; h = int(s1 * pctile); for (i = 0; i < A->capacity; i++) { J = i; b += D[J]; if (a <= h && h <= b) break; I = i; a += D[I]; } xfree(D); if (s1 == 0) return 0.0; if (a > h) return 0.0; if (a >= b) return 0.0; if (I >= J) return 0.0; f = (h - a) / (b - a); K = (int) floor(f * (double) (J - I) + I); return statHistVal(A, K); } static void statHistBinDumper(StoreEntry * sentry, int idx, double val, double size, int count) { if (count) storeAppendPrintf(sentry, "\t%3d/%f\t%d\t%f\n", idx, val, count, count / size); } void statHistDump(const StatHist * H, StoreEntry * sentry, StatHistBinDumper * bd) { int i; double left_border = H->min; if (!bd) bd = statHistBinDumper; for (i = 0; i < H->capacity; i++) { const double right_border = statHistVal(H, i + 1); assert(right_border - left_border > 0.0); bd(sentry, i, left_border, right_border - left_border, H->bins[i]); left_border = right_border; } } /* log based histogram */ #if !defined(_SQUID_HPUX_) || !defined(__GNUC__) static #endif double Log(double x) { assert((x + 1.0) >= 0.0); return log(x + 1.0); } #if !defined(_SQUID_HPUX_) || !defined(__GNUC__) static #endif double Exp(double x) { return exp(x) - 1.0; } void statHistLogInit(StatHist * H, int capacity, double min, double max) { statHistInit(H, capacity, Log, Exp, min, max); } /* linear histogram for enums */ /* we want to be have [-1,last_enum+1] range to track out of range enums */ #if !defined(_SQUID_HPUX_) || !defined(__GNUC__) static #endif double Null(double x) { return x; } void statHistEnumInit(StatHist * H, int last_enum) { statHistInit(H, last_enum + 3, Null, Null, (double) -1, (double) (last_enum + 1 + 1)); } void statHistEnumDumper(StoreEntry * sentry, int idx, double val, double size, int count) { if (count) storeAppendPrintf(sentry, "%2d\t %5d\t %5d\n", idx, (int) val, count); } void statHistIntInit(StatHist * H, int n) { statHistInit(H, n, Null, Null, (double) 0, (double) n - 1); } void statHistIntDumper(StoreEntry * sentry, int idx, double val, double size, int count) { if (count) storeAppendPrintf(sentry, "%9d\t%9d\n", (int) val, count); }