Raptor
A fast and space-efficient pre-filter
All Classes Namespaces Files Functions Variables Macros Pages Concepts
formatted_bytes.hpp
Go to the documentation of this file.
1// SPDX-FileCopyrightText: 2006-2024 Knut Reinert & Freie Universität Berlin
2// SPDX-FileCopyrightText: 2016-2024 Knut Reinert & MPI für molekulare Genetik
3// SPDX-License-Identifier: BSD-3-Clause
4
10#pragma once
11
12#include <cassert>
13#include <cstddef>
14#include <cstdint>
15#include <string>
16
17namespace raptor
18{
19
20[[nodiscard]] inline std::string formatted_bytes(size_t const bytes)
21{
22 assert(bytes > 0);
23
24 size_t iterations{};
25 size_t integer{bytes};
26
27 while (integer >> 10u && iterations < 6u)
28 {
29 integer >>= 10u;
30 ++iterations;
31 }
32
33 // While this is a bit more involved, we can avoid using floating point numbers.
34 auto first_decimal_position = [&]()
35 {
36 assert(iterations > 0u);
37 size_t decimal{bytes};
38 decimal -= integer << (iterations * 10u); // Substract bytes represented by integer, e.g. -5GiB
39 decimal >>= (iterations - 1u) * 10u; // Shift to next smallest unit, e.g. 800MiB
40 decimal = decimal * 1000u / 1024u; // Account for using decimal system, i.e. 800MiB != 0.8GiB
41 size_t const diff{decimal - (decimal / 100u) * 100u}; // We want to round up to 1 decimal position
42 uint32_t const round_up{diff >= 50u};
43 decimal += round_up * 100u - diff;
44 decimal /= 100u;
45 return decimal;
46 };
47
48 auto formatted_string = [&]()
49 {
50 static constexpr int8_t int_to_char_offset{'0'}; // int 0 as char: char{0 + 48} = '0'
51 size_t const decimal = iterations ? first_decimal_position() : 0u;
52 assert(decimal <= 10u);
53
54 if (!iterations) // No decimals for Bytes
55 return std::to_string(integer);
56 else if (decimal < 10u) // No need to round integer part
57 return std::to_string(integer) + '.' + static_cast<char>(decimal + int_to_char_offset);
58 else // Round integer part, e.g., 5.99 MiB should report 6.0 MiB
59 {
60 ++integer;
61 // Check whether rounding results in a change of unit, e.g. 1023.99MiB to 1.0GiB
62 if (integer >> 10u)
63 {
64 ++iterations;
65 integer >>= 10u;
66 }
67 return std::to_string(integer) + ".0";
68 }
69 };
70
71 std::string const formatted{formatted_string()};
72 switch (iterations)
73 {
74 case 0:
75 return "[Bytes]: " + formatted;
76 case 1:
77 return "[KiB]: " + formatted;
78 case 2:
79 return "[MiB]: " + formatted;
80 case 3:
81 return "[GiB]: " + formatted;
82 case 4:
83 return "[TiB]: " + formatted;
84 case 5:
85 return "[PiB]: " + formatted;
86 default:
87 return "[EiB]: " + formatted;
88 }
89}
90
91} // namespace raptor
T to_string(T... args)
Hide me