xref: /openbmc/linux/lib/string_helpers.c (revision ce932d0c5589e9766e089c22c66890dfc48fbd94)
1 /*
2  * Helpers for formatting and printing strings
3  *
4  * Copyright 31 August 2008 James Bottomley
5  */
6 #include <linux/kernel.h>
7 #include <linux/math64.h>
8 #include <linux/export.h>
9 #include <linux/string_helpers.h>
10 
11 /**
12  * string_get_size - get the size in the specified units
13  * @size:	The size to be converted
14  * @units:	units to use (powers of 1000 or 1024)
15  * @buf:	buffer to format to
16  * @len:	length of buffer
17  *
18  * This function returns a string formatted to 3 significant figures
19  * giving the size in the required units.  Returns 0 on success or
20  * error on failure.  @buf is always zero terminated.
21  *
22  */
23 int string_get_size(u64 size, const enum string_size_units units,
24 		    char *buf, int len)
25 {
26 	const char *units_10[] = { "B", "kB", "MB", "GB", "TB", "PB",
27 				   "EB", "ZB", "YB", NULL};
28 	const char *units_2[] = {"B", "KiB", "MiB", "GiB", "TiB", "PiB",
29 				 "EiB", "ZiB", "YiB", NULL };
30 	const char **units_str[] = {
31 		[STRING_UNITS_10] =  units_10,
32 		[STRING_UNITS_2] = units_2,
33 	};
34 	const unsigned int divisor[] = {
35 		[STRING_UNITS_10] = 1000,
36 		[STRING_UNITS_2] = 1024,
37 	};
38 	int i, j;
39 	u64 remainder = 0, sf_cap;
40 	char tmp[8];
41 
42 	tmp[0] = '\0';
43 	i = 0;
44 	if (size >= divisor[units]) {
45 		while (size >= divisor[units] && units_str[units][i]) {
46 			remainder = do_div(size, divisor[units]);
47 			i++;
48 		}
49 
50 		sf_cap = size;
51 		for (j = 0; sf_cap*10 < 1000; j++)
52 			sf_cap *= 10;
53 
54 		if (j) {
55 			remainder *= 1000;
56 			do_div(remainder, divisor[units]);
57 			snprintf(tmp, sizeof(tmp), ".%03lld",
58 				 (unsigned long long)remainder);
59 			tmp[j+1] = '\0';
60 		}
61 	}
62 
63 	snprintf(buf, len, "%lld%s %s", (unsigned long long)size,
64 		 tmp, units_str[units][i]);
65 
66 	return 0;
67 }
68 EXPORT_SYMBOL(string_get_size);
69