libcosmos
Linux C++ System Programming Library
Loading...
Searching...
No Matches
formatting.hxx
Go to the documentation of this file.
1#pragma once
2
3// C++
4#include <iomanip>
5#include <functional>
6#include <ostream>
7#include <string>
8#include <sstream>
9#include <type_traits>
10
11// cosmos
12#include <cosmos/compiler.hxx>
13#include <cosmos/dso_export.h>
14#include <cosmos/fs/types.hxx>
15#include <cosmos/proc/types.hxx>
16#include <cosmos/string.hxx>
17#include <cosmos/thread/thread.hxx>
18#include <cosmos/types.hxx>
19#include <cosmos/utils.hxx>
20
28namespace cosmos {
29 template <typename NUM>
30 struct FormattedNumber;
31}
32
33inline std::ostream& operator<<(std::ostream &o, const cosmos::ProcessID &pid) {
34 // we could also think about using a consistent annotation of process
35 // ids in the output like @1234 or <pid: 1234> something like that.
36 o << cosmos::to_integral(pid);
37 return o;
38}
39
40inline std::ostream& operator<<(std::ostream &o, const cosmos::ThreadID &tid) {
41 // similarly we could use special annotation here
42 o << cosmos::to_integral(tid);
43 return o;
44}
45
46inline std::ostream& operator<<(std::ostream &o, const cosmos::UserID &uid) {
47 // similarly we could use special annotation here
48 o << cosmos::to_integral(uid);
49 return o;
50}
51
52inline std::ostream& operator<<(std::ostream &o, const cosmos::GroupID &gid) {
53 // similarly we could use special annotation here
54 o << cosmos::to_integral(gid);
55 return o;
56}
57
58inline std::ostream& operator<<(std::ostream &o, const cosmos::SignalNr &sig) {
59 // similarly we could use special annotation here
60 o << cosmos::to_integral(sig);
61 return o;
62}
63
64inline std::ostream& operator<<(std::ostream &o, const cosmos::FileNum &fd) {
65 // similarly we could use special annotation here
66 o << cosmos::to_integral(fd);
67 return o;
68}
69
70// this is implemented outlined with explicit template instantiations for the
71// currently necessary primitive types
72template <typename NUM>
73std::ostream& operator<<(std::ostream& o, const cosmos::FormattedNumber<NUM> &fmtnum);
74
75
76namespace cosmos {
77
79template <typename NUM>
81protected: // types
82
84 using SetBaseFN = std::function<void(std::ostream&)>;
85
86protected: // functions
87
88 FormattedNumber(const NUM num, size_t width, SetBaseFN fn, std::string base_prefix) :
89 m_num{num}, m_width{width}, m_setbase_fn{fn}, m_base_prefix{base_prefix} {}
90
91public: // functions
92
94 const FormattedNumber& showBase(bool yes_no) { m_show_base = yes_no; return *this; }
95
96 size_t width() const { return m_width; }
97 bool showBase() const { return m_show_base; }
98 auto num() const { return m_num; }
99 const std::string& basePrefix() const { return m_base_prefix; }
100 SetBaseFN baseFN() const { return m_setbase_fn; }
101
102 explicit operator std::string() const {
103 std::stringstream ss;
104 ss << *this;
105 return ss.str();
106 }
107
108protected: // data
109
110 NUM m_num;
111 size_t m_width = 0;
112 SetBaseFN m_setbase_fn;
113 std::string m_base_prefix;
114 bool m_show_base = true;
115};
116
118
129template <typename NUM>
130struct HexNum :
131 public FormattedNumber<NUM> {
132 HexNum(const NUM num, size_t width) :
133 FormattedNumber<NUM>{num, width, [](std::ostream &o){ o << std::hex; }, "0x"}
134 {}
135};
136
138
141template <typename NUM>
142struct OctNum :
143 public FormattedNumber<NUM> {
144 OctNum(const NUM num, size_t width) :
145 FormattedNumber<NUM>{num, width, [](std::ostream &o){ o << std::oct; }, "0o"}
146 {}
147};
148
150
156template <typename T>
157auto to_printable_integer(T num) -> decltype(+num) {
158 // using the unary + operator promotes the number to a printable type
159 return +num;
160}
161
163
169COSMOS_API std::string sprintf(const char *fmt, ...) COSMOS_FORMAT_PRINTF(1, 2);
170
171} // end ns cosmos
auto to_printable_integer(T num) -> decltype(+num)
This helper makes sure that any integer is turned into a printable integer.
FileNum
Primitive file descriptor.
Definition types.hxx:32
ProcessID
Definition types.hxx:25
SignalNr
A primitive signal number specification.
Definition types.hxx:50
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).
std::function< void(std::ostream &)> SetBaseFN
This function is supposed to apply the desired number base to the stream.
Helper to output a primitive integer as hexadecimal in the style of 0x1234.
Helper to output a primitive integer as octal in the style of 0o123.