X-Git-Url: https://git.ozlabs.org/?a=blobdiff_plain;ds=sidebyside;f=ccan%2Ftally%2Ftally.c;h=9f2a4591f8c3ce5ddd05b8f29702ff2761784dad;hb=fdea8abee49e9158275741c210bbb27caf6754fa;hp=1433a3aa2650e45cf2bfc44c3bce646400f8dfba;hpb=7a36f69eea4ee572bdf191d6048350751cd2ab5b;p=ccan diff --git a/ccan/tally/tally.c b/ccan/tally/tally.c index 1433a3aa..9f2a4591 100644 --- a/ccan/tally/tally.c +++ b/ccan/tally/tally.c @@ -8,19 +8,18 @@ #include #include -#define MAX_STEP_BITS (sizeof(size_t)*CHAR_BIT) +#define SIZET_BITS (sizeof(size_t)*CHAR_BIT) /* We use power of 2 steps. I tried being tricky, but it got buggy. */ struct tally { ssize_t min, max; size_t total[2]; /* This allows limited frequency analysis. */ - size_t buckets; - size_t step_bits; + unsigned buckets, step_bits; size_t counts[1 /* [buckets] */ ]; }; -struct tally *tally_new(size_t buckets) +struct tally *tally_new(unsigned buckets) { struct tally *tally; @@ -29,20 +28,8 @@ struct tally *tally_new(size_t buckets) return NULL; tally = malloc(sizeof(*tally) + sizeof(tally->counts[0])*buckets); if (tally) { - /* SSIZE_MAX isn't portable, so make it one of these types. */ - BUILD_ASSERT(sizeof(tally->min) == sizeof(int) - || sizeof(tally->min) == sizeof(long) - || sizeof(tally->min) == sizeof(long long)); - if (sizeof(tally->min) == sizeof(int)) { - tally->min = INT_MAX; - tally->max = INT_MIN; - } else if (sizeof(tally->min) == sizeof(long)) { - tally->min = LONG_MAX; - tally->max = LONG_MIN; - } else if (sizeof(tally->min) == sizeof(long long)) { - tally->min = (ssize_t)LLONG_MAX; - tally->max = (ssize_t)LLONG_MIN; - } + tally->max = ((size_t)1 << (SIZET_BITS - 1)); + tally->min = ~tally->max; tally->total[0] = tally->total[1] = 0; /* There is always 1 bucket. */ tally->buckets = buckets+1; @@ -55,9 +42,9 @@ struct tally *tally_new(size_t buckets) static unsigned bucket_of(ssize_t min, unsigned step_bits, ssize_t val) { /* Don't over-shift. */ - if (step_bits == MAX_STEP_BITS) + if (step_bits == SIZET_BITS) return 0; - assert(step_bits < MAX_STEP_BITS); + assert(step_bits < SIZET_BITS); return (size_t)(val - min) >> step_bits; } @@ -65,9 +52,9 @@ static unsigned bucket_of(ssize_t min, unsigned step_bits, ssize_t val) static ssize_t bucket_min(ssize_t min, unsigned step_bits, unsigned b) { /* Don't over-shift. */ - if (step_bits == MAX_STEP_BITS) + if (step_bits == SIZET_BITS) return min; - assert(step_bits < MAX_STEP_BITS); + assert(step_bits < SIZET_BITS); return min + ((ssize_t)b << step_bits); } @@ -324,6 +311,33 @@ ssize_t tally_mean(const struct tally *tally) return divls64(tally->total[1], tally->total[0], count); } +ssize_t tally_total(const struct tally *tally, ssize_t *overflow) +{ + if (overflow) { + *overflow = tally->total[1]; + return tally->total[0]; + } + + /* If result is negative, make sure we can represent it. */ + if (tally->total[1] & (1 << (SIZET_BITS-1))) { + /* Must have only underflowed once, and must be able to + * represent result at ssize_t. */ + if ((~tally->total[1])+1 != 0 + || (ssize_t)tally->total[0] >= 0) { + /* Underflow, return minimum. */ + return (ssize_t)((size_t)1 << (SIZET_BITS - 1)); + } + } else { + /* Result is positive, must not have overflowed, and must be + * able to represent as ssize_t. */ + if (tally->total[1] || (ssize_t)tally->total[0] < 0) { + /* Overflow. Return maximum. */ + return (ssize_t)~((size_t)1 << (SIZET_BITS - 1)); + } + } + return tally->total[0]; +} + static ssize_t bucket_range(const struct tally *tally, unsigned b, size_t *err) { ssize_t min, max; @@ -417,7 +431,7 @@ char *tally_histogram(const struct tally *tally, memcpy(tmp->counts, tally->counts, sizeof(tally->counts[0]) * tmp->buckets); while ((max_bucket = get_max_bucket(tmp)) >= height) - renormalize(tmp, tmp->min, tmp->max *= 2); + renormalize(tmp, tmp->min, tmp->max * 2); /* Restore max */ tmp->max = tally->max; tally = tmp; @@ -436,23 +450,29 @@ char *tally_histogram(const struct tally *tally, free(tmp); return NULL; } + for (i = 0; i < height; i++) { - unsigned covered = 0; - count = (double)tally->counts[i] / largest_bucket * width; + unsigned covered = 1; + count = (double)tally->counts[i] / largest_bucket * (width-1)+1; if (i == 0) covered = snprintf(p, width, "%zi", tally->min); else if (i == height - 1) covered = snprintf(p, width, "%zi", tally->max); - if (covered) { - if (covered > width) - covered = width; - p += covered; - if (count > covered) - count -= covered; - else - count = 0; - } + else if (i == bucket_of(tally->min, tally->step_bits, 0)) + *p = '+'; + else + *p = '|'; + + if (covered > width) + covered = width; + p += covered; + + if (count > covered) + count -= covered; + else + count = 0; + memset(p, '*', count); p += count; *p = '\n';