2012-03-18 00:01:14 +04:00
|
|
|
/* $NetBSD: humanize_number.c,v 1.16 2012/03/17 20:01:14 christos Exp $ */
|
2002-08-15 08:47:12 +04:00
|
|
|
|
2002-08-08 20:50:35 +04:00
|
|
|
/*
|
|
|
|
* Copyright (c) 1997, 1998, 1999, 2002 The NetBSD Foundation, Inc.
|
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
* This code is derived from software contributed to The NetBSD Foundation
|
|
|
|
* by Jason R. Thorpe of the Numerical Aerospace Simulation Facility,
|
|
|
|
* NASA Ames Research Center, by Luke Mewburn and by Tomas Svensson.
|
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions
|
|
|
|
* are met:
|
|
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer.
|
|
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
|
|
* documentation and/or other materials provided with the distribution.
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
|
|
|
|
* ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
|
|
|
|
* TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
|
|
|
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
|
|
|
|
* BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
|
|
|
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
|
|
|
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
|
|
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
|
|
|
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
|
|
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
|
|
* POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <sys/cdefs.h>
|
2005-09-13 05:44:08 +04:00
|
|
|
#if defined(LIBC_SCCS) && !defined(lint)
|
2012-03-18 00:01:14 +04:00
|
|
|
__RCSID("$NetBSD: humanize_number.c,v 1.16 2012/03/17 20:01:14 christos Exp $");
|
2005-09-13 05:44:08 +04:00
|
|
|
#endif /* LIBC_SCCS and not lint */
|
2002-08-08 20:50:35 +04:00
|
|
|
|
A little libc namespace housekeeping exercise:
* Make vfprintf_unlocked() an internal function, c.f. __svfscanf_unlocked().
* Add internal names for arc4random(), endnetpath(), fhstatvfs(),
fstatvfs(), mkstemp(), shquote(), statvfs(), taddr2uaddr(), uaddr2taddr(),
uuid_create_nil(), uuid_is_nil(), and wcwidth().
* Include namespace.h where supposed to.
2005-02-10 00:35:46 +03:00
|
|
|
#include "namespace.h"
|
2002-08-08 20:50:35 +04:00
|
|
|
#include <assert.h>
|
2006-06-09 01:08:56 +04:00
|
|
|
#include <inttypes.h>
|
2002-08-08 20:50:35 +04:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <locale.h>
|
|
|
|
|
|
|
|
int
|
|
|
|
humanize_number(char *buf, size_t len, int64_t bytes,
|
2002-08-15 08:47:12 +04:00
|
|
|
const char *suffix, int scale, int flags)
|
|
|
|
{
|
2004-07-27 05:56:24 +04:00
|
|
|
const char *prefixes, *sep;
|
2012-03-14 01:13:30 +04:00
|
|
|
int b, r, s1, s2, sign;
|
2012-03-18 00:01:14 +04:00
|
|
|
int64_t divisor, max, post = 1;
|
2012-03-14 01:13:30 +04:00
|
|
|
size_t i, baselen, maxscale;
|
2002-08-08 20:50:35 +04:00
|
|
|
|
|
|
|
_DIAGASSERT(buf != NULL);
|
|
|
|
_DIAGASSERT(suffix != NULL);
|
2002-11-11 20:56:11 +03:00
|
|
|
_DIAGASSERT(scale >= 0);
|
2002-08-08 20:50:35 +04:00
|
|
|
|
2002-09-27 22:37:43 +04:00
|
|
|
if (flags & HN_DIVISOR_1000) {
|
|
|
|
/* SI for decimal multiplies */
|
|
|
|
divisor = 1000;
|
2004-07-27 05:56:24 +04:00
|
|
|
if (flags & HN_B)
|
|
|
|
prefixes = "B\0k\0M\0G\0T\0P\0E";
|
|
|
|
else
|
|
|
|
prefixes = "\0\0k\0M\0G\0T\0P\0E";
|
2002-09-27 22:37:43 +04:00
|
|
|
} else {
|
|
|
|
/*
|
|
|
|
* binary multiplies
|
|
|
|
* XXX IEC 60027-2 recommends Ki, Mi, Gi...
|
|
|
|
*/
|
|
|
|
divisor = 1024;
|
2004-07-27 05:56:24 +04:00
|
|
|
if (flags & HN_B)
|
|
|
|
prefixes = "B\0K\0M\0G\0T\0P\0E";
|
|
|
|
else
|
|
|
|
prefixes = "\0\0K\0M\0G\0T\0P\0E";
|
2002-09-27 22:37:43 +04:00
|
|
|
}
|
|
|
|
|
2004-07-27 05:56:24 +04:00
|
|
|
#define SCALE2PREFIX(scale) (&prefixes[(scale) << 1])
|
|
|
|
maxscale = 7;
|
|
|
|
|
2012-03-14 01:13:30 +04:00
|
|
|
if ((size_t)scale >= maxscale &&
|
2004-07-27 05:56:24 +04:00
|
|
|
(scale & (HN_AUTOSCALE | HN_GETSCALE)) == 0)
|
2002-08-15 08:47:12 +04:00
|
|
|
return (-1);
|
2002-08-08 20:50:35 +04:00
|
|
|
|
|
|
|
if (buf == NULL || suffix == NULL)
|
2002-08-15 08:47:12 +04:00
|
|
|
return (-1);
|
2002-08-08 20:50:35 +04:00
|
|
|
|
|
|
|
if (len > 0)
|
|
|
|
buf[0] = '\0';
|
|
|
|
if (bytes < 0) {
|
|
|
|
sign = -1;
|
2004-07-27 05:56:24 +04:00
|
|
|
baselen = 3; /* sign, digit, prefix */
|
2012-03-18 00:01:14 +04:00
|
|
|
if (-bytes < INT64_MAX / 100)
|
|
|
|
bytes *= -100;
|
|
|
|
else {
|
|
|
|
bytes = -bytes;
|
|
|
|
post = 100;
|
|
|
|
baselen += 2;
|
|
|
|
}
|
2002-08-08 20:50:35 +04:00
|
|
|
} else {
|
|
|
|
sign = 1;
|
2004-07-27 05:56:24 +04:00
|
|
|
baselen = 2; /* digit, prefix */
|
2012-03-18 00:01:14 +04:00
|
|
|
if (bytes < INT64_MAX / 100)
|
|
|
|
bytes *= 100;
|
|
|
|
else {
|
|
|
|
post = 100;
|
|
|
|
baselen += 2;
|
|
|
|
}
|
2002-08-08 20:50:35 +04:00
|
|
|
}
|
2004-07-27 05:56:24 +04:00
|
|
|
if (flags & HN_NOSPACE)
|
|
|
|
sep = "";
|
|
|
|
else {
|
|
|
|
sep = " ";
|
|
|
|
baselen++;
|
|
|
|
}
|
|
|
|
baselen += strlen(suffix);
|
2002-08-08 20:50:35 +04:00
|
|
|
|
2004-07-27 05:56:24 +04:00
|
|
|
/* Check if enough room for `x y' + suffix + `\0' */
|
|
|
|
if (len < baselen + 1)
|
2002-08-08 20:50:35 +04:00
|
|
|
return (-1);
|
|
|
|
|
2004-07-27 05:56:24 +04:00
|
|
|
if (scale & (HN_AUTOSCALE | HN_GETSCALE)) {
|
|
|
|
/* See if there is additional columns can be used. */
|
|
|
|
for (max = 100, i = len - baselen; i-- > 0;)
|
|
|
|
max *= 10;
|
2002-08-08 20:50:35 +04:00
|
|
|
|
2007-03-13 05:52:10 +03:00
|
|
|
/*
|
|
|
|
* Divide the number until it fits the given column.
|
|
|
|
* If there will be an overflow by the rounding below,
|
|
|
|
* divide once more.
|
|
|
|
*/
|
|
|
|
for (i = 0; bytes >= max - 50 && i < maxscale; i++)
|
2002-08-08 20:50:35 +04:00
|
|
|
bytes /= divisor;
|
|
|
|
|
2012-03-14 01:13:30 +04:00
|
|
|
if (scale & HN_GETSCALE) {
|
|
|
|
_DIAGASSERT(__type_fit(int, i));
|
|
|
|
return (int)i;
|
|
|
|
}
|
2004-07-27 05:56:24 +04:00
|
|
|
} else
|
2012-03-14 01:13:30 +04:00
|
|
|
for (i = 0; i < (size_t)scale && i < maxscale; i++)
|
2004-07-27 05:56:24 +04:00
|
|
|
bytes /= divisor;
|
2012-03-18 00:01:14 +04:00
|
|
|
bytes *= post;
|
2002-08-08 20:50:35 +04:00
|
|
|
|
2004-07-27 05:56:24 +04:00
|
|
|
/* If a value <= 9.9 after rounding and ... */
|
|
|
|
if (bytes < 995 && i > 0 && flags & HN_DECIMAL) {
|
|
|
|
/* baselen + \0 + .N */
|
|
|
|
if (len < baselen + 1 + 2)
|
2002-08-08 20:50:35 +04:00
|
|
|
return (-1);
|
2004-07-27 05:56:24 +04:00
|
|
|
b = ((int)bytes + 5) / 10;
|
|
|
|
s1 = b / 10;
|
|
|
|
s2 = b % 10;
|
|
|
|
r = snprintf(buf, len, "%d%s%d%s%s%s",
|
|
|
|
sign * s1, localeconv()->decimal_point, s2,
|
|
|
|
sep, SCALE2PREFIX(i), suffix);
|
2002-08-08 20:50:35 +04:00
|
|
|
} else
|
2006-06-09 01:08:56 +04:00
|
|
|
r = snprintf(buf, len, "%" PRId64 "%s%s%s",
|
|
|
|
sign * ((bytes + 50) / 100),
|
2004-07-27 05:56:24 +04:00
|
|
|
sep, SCALE2PREFIX(i), suffix);
|
2002-08-08 20:50:35 +04:00
|
|
|
|
2002-08-15 08:47:12 +04:00
|
|
|
return (r);
|
2002-08-08 20:50:35 +04:00
|
|
|
}
|