diff options
Diffstat (limited to 'lib')
-rw-r--r-- | lib/string_helpers.c | 68 |
1 files changed, 48 insertions, 20 deletions
diff --git a/lib/string_helpers.c b/lib/string_helpers.c index 8f8c4417f228..4a913ec3acf9 100644 --- a/lib/string_helpers.c +++ b/lib/string_helpers.c | |||
@@ -4,6 +4,7 @@ | |||
4 | * Copyright 31 August 2008 James Bottomley | 4 | * Copyright 31 August 2008 James Bottomley |
5 | * Copyright (C) 2013, Intel Corporation | 5 | * Copyright (C) 2013, Intel Corporation |
6 | */ | 6 | */ |
7 | #include <linux/bug.h> | ||
7 | #include <linux/kernel.h> | 8 | #include <linux/kernel.h> |
8 | #include <linux/math64.h> | 9 | #include <linux/math64.h> |
9 | #include <linux/export.h> | 10 | #include <linux/export.h> |
@@ -14,7 +15,8 @@ | |||
14 | 15 | ||
15 | /** | 16 | /** |
16 | * string_get_size - get the size in the specified units | 17 | * string_get_size - get the size in the specified units |
17 | * @size: The size to be converted | 18 | * @size: The size to be converted in blocks |
19 | * @blk_size: Size of the block (use 1 for size in bytes) | ||
18 | * @units: units to use (powers of 1000 or 1024) | 20 | * @units: units to use (powers of 1000 or 1024) |
19 | * @buf: buffer to format to | 21 | * @buf: buffer to format to |
20 | * @len: length of buffer | 22 | * @len: length of buffer |
@@ -24,14 +26,14 @@ | |||
24 | * at least 9 bytes and will always be zero terminated. | 26 | * at least 9 bytes and will always be zero terminated. |
25 | * | 27 | * |
26 | */ | 28 | */ |
27 | void string_get_size(u64 size, const enum string_size_units units, | 29 | void string_get_size(u64 size, u64 blk_size, const enum string_size_units units, |
28 | char *buf, int len) | 30 | char *buf, int len) |
29 | { | 31 | { |
30 | static const char *const units_10[] = { | 32 | static const char *const units_10[] = { |
31 | "B", "kB", "MB", "GB", "TB", "PB", "EB" | 33 | "B", "kB", "MB", "GB", "TB", "PB", "EB", "ZB", "YB" |
32 | }; | 34 | }; |
33 | static const char *const units_2[] = { | 35 | static const char *const units_2[] = { |
34 | "B", "KiB", "MiB", "GiB", "TiB", "PiB", "EiB" | 36 | "B", "KiB", "MiB", "GiB", "TiB", "PiB", "EiB", "ZiB", "YiB" |
35 | }; | 37 | }; |
36 | static const char *const *const units_str[] = { | 38 | static const char *const *const units_str[] = { |
37 | [STRING_UNITS_10] = units_10, | 39 | [STRING_UNITS_10] = units_10, |
@@ -42,31 +44,57 @@ void string_get_size(u64 size, const enum string_size_units units, | |||
42 | [STRING_UNITS_2] = 1024, | 44 | [STRING_UNITS_2] = 1024, |
43 | }; | 45 | }; |
44 | int i, j; | 46 | int i, j; |
45 | u32 remainder = 0, sf_cap; | 47 | u32 remainder = 0, sf_cap, exp; |
46 | char tmp[8]; | 48 | char tmp[8]; |
49 | const char *unit; | ||
47 | 50 | ||
48 | tmp[0] = '\0'; | 51 | tmp[0] = '\0'; |
49 | i = 0; | 52 | i = 0; |
50 | if (size >= divisor[units]) { | 53 | if (!size) |
51 | while (size >= divisor[units]) { | 54 | goto out; |
52 | remainder = do_div(size, divisor[units]); | ||
53 | i++; | ||
54 | } | ||
55 | 55 | ||
56 | sf_cap = size; | 56 | while (blk_size >= divisor[units]) { |
57 | for (j = 0; sf_cap*10 < 1000; j++) | 57 | remainder = do_div(blk_size, divisor[units]); |
58 | sf_cap *= 10; | 58 | i++; |
59 | } | ||
59 | 60 | ||
60 | if (j) { | 61 | exp = divisor[units] / (u32)blk_size; |
61 | remainder *= 1000; | 62 | if (size >= exp) { |
62 | remainder /= divisor[units]; | 63 | remainder = do_div(size, divisor[units]); |
63 | snprintf(tmp, sizeof(tmp), ".%03u", remainder); | 64 | remainder *= blk_size; |
64 | tmp[j+1] = '\0'; | 65 | i++; |
65 | } | 66 | } else { |
67 | remainder *= size; | ||
68 | } | ||
69 | |||
70 | size *= blk_size; | ||
71 | size += remainder / divisor[units]; | ||
72 | remainder %= divisor[units]; | ||
73 | |||
74 | while (size >= divisor[units]) { | ||
75 | remainder = do_div(size, divisor[units]); | ||
76 | i++; | ||
66 | } | 77 | } |
67 | 78 | ||
79 | sf_cap = size; | ||
80 | for (j = 0; sf_cap*10 < 1000; j++) | ||
81 | sf_cap *= 10; | ||
82 | |||
83 | if (j) { | ||
84 | remainder *= 1000; | ||
85 | remainder /= divisor[units]; | ||
86 | snprintf(tmp, sizeof(tmp), ".%03u", remainder); | ||
87 | tmp[j+1] = '\0'; | ||
88 | } | ||
89 | |||
90 | out: | ||
91 | if (i >= ARRAY_SIZE(units_2)) | ||
92 | unit = "UNK"; | ||
93 | else | ||
94 | unit = units_str[units][i]; | ||
95 | |||
68 | snprintf(buf, len, "%u%s %s", (u32)size, | 96 | snprintf(buf, len, "%u%s %s", (u32)size, |
69 | tmp, units_str[units][i]); | 97 | tmp, unit); |
70 | } | 98 | } |
71 | EXPORT_SYMBOL(string_get_size); | 99 | EXPORT_SYMBOL(string_get_size); |
72 | 100 | ||