libcosmos
Linux C++ System Programming Library
Loading...
Searching...
No Matches
formatting.cxx
1// libc
2#include <stdarg.h>
3
4// C++
5#include <cstdint>
6#include <climits>
7#include <string>
8
9// cosmos
10#include <cosmos/formatting.hxx>
11
12namespace cosmos {
13
14namespace {
15
16/* silence clang warning about non-literal fmt */ COSMOS_FORMAT_PRINTF(1, 0)
17std::string sprintf_v(const char *fmt, va_list orig_args) {
18 std::string ret;
19 // let's use some arbitrary start size that should suffice for average uses
20 // and it will be increased if its not enough
21 ret.resize(128);
22 int written = 0;
23 va_list varargs;
24
25 while (true) {
26 // copy the list of arguments in any case, since we
27 // potentially need to evaluate it multiple times
28 va_copy(varargs, orig_args);
29
30 written = vsnprintf(&ret[0], ret.size() + 1, fmt, varargs);
31
32 if (written < 0) {
33 // some fatal error, simply return an empty string
34 // then
35 ret.clear();
36 break;
37 } else if (static_cast<size_t>(written) <= ret.size()) {
38 // the return value is excluding the null terminator
39 // so it's just right for resize()
40 ret.resize(written);
41 break;
42 }
43
44 // re-try with the correct string size
45 ret.resize(written);
46
47 va_end(varargs);
48 }
49
50 va_end(varargs);
51
52 return ret;
53}
54
55} // end anon ns
56
57std::string sprintf(const char *fmt, ...) {
58 va_list varargs;
59 va_start(varargs, fmt);
60 const auto ret = sprintf_v(fmt, varargs);
61 va_end(varargs);
62 return ret;
63}
64
65} // end ns
66
67template <typename NUM>
68std::ostream& operator<<(std::ostream& o, const cosmos::FormattedNumber<NUM> &fmtnum) {
69 const auto orig_flags = o.flags();
70 const auto orig_fill = o.fill();
71
72 static_assert(std::is_integral_v<NUM>, "template type needs to be an integral integer type");
73
74 // don't handle this with std::showbase, it's behaving badly e.g. the
75 // "0x" prefix counts towards the field with, also, the fill character
76 // will be prepended to the "0x" prefix resulting in things like
77 // "0000x64".
78 if (fmtnum.showBase())
79 o << fmtnum.basePrefix();
80 o << std::setw(fmtnum.width());
81 fmtnum.baseFN()(o);
82 o << std::setfill('0') << cosmos::to_printable_integer(fmtnum.num());
83
84 o.flags(orig_flags);
85 o.fill(orig_fill);
86 return o;
87}
88
89/* explicit instantiations of the templated operator<< */
90
91template COSMOS_API std::ostream& operator<<(std::ostream&, const cosmos::FormattedNumber<unsigned int>&);
92template COSMOS_API std::ostream& operator<<(std::ostream&, const cosmos::FormattedNumber<unsigned short>&);
93template COSMOS_API std::ostream& operator<<(std::ostream&, const cosmos::FormattedNumber<int>&);
94template COSMOS_API std::ostream& operator<<(std::ostream&, const cosmos::FormattedNumber<char>&);
95
96// on 32-bit archs these can be the same, causing a duplicate instantiation
97#if SIZE_MAX != UINT_MAX
98template COSMOS_API std::ostream& operator<<(std::ostream&, const cosmos::FormattedNumber<size_t>&);
99// similarly this ULL definition is then missing on 32-bit archs, provide it
100#else
101template COSMOS_API std::ostream& operator<<(std::ostream&, const cosmos::FormattedNumber<unsigned long long>&);
102template COSMOS_API std::ostream& operator<<(std::ostream&, const cosmos::FormattedNumber<unsigned long>&);
103#endif
auto to_printable_integer(T num) -> decltype(+num)
This helper makes sure that any integer is turned into a printable integer.
Base class for HexNum and OctNum format output helpers.
const FormattedNumber & showBase(bool yes_no)
If a prefix identifier for the number's base should be shown (e.g. 0x for hex, default: yes).