Toolbox snapshot
The Reactive C++ Toolbox
Loading...
Searching...
No Matches
OStreamBase.hpp
Go to the documentation of this file.
1// The Reactive C++ Toolbox.
2// Copyright (C) 2025 Reactive Markets Limited
3//
4// Licensed under the Apache License, Version 2.0 (the "License");
5// you may not use this file except in compliance with the License.
6// You may obtain a copy of the License at
7//
8// http://www.apache.org/licenses/LICENSE-2.0
9//
10// Unless required by applicable law or agreed to in writing, software
11// distributed under the License is distributed on an "AS IS" BASIS,
12// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13// See the License for the specific language governing permissions and
14// limitations under the License.
15
16#ifndef TOOLBOX_UTIL_OSTREAM_BASE_HPP
17#define TOOLBOX_UTIL_OSTREAM_BASE_HPP
18
19#include <algorithm>
20#include <charconv>
21#include <concepts>
22#include <type_traits>
23#include <string>
24#include <string_view>
25#include <cassert>
26#include <limits>
27
28#include "TypeTraits.hpp"
29
30namespace toolbox {
31inline namespace util {
32
33/*
34DerivedT is required to implement the following:
35
361) char* do_prepare_space(std::size_t num_bytes)
37 - returns writable buffer of atleast `num_bytes` bytes for immediate writing
38 - will be followed by a call to `do_relinquish_space` consuming up to `num_bytes`
39
402) void do_relinquish_space(std::size_t consumed_num_bytes)
41 - data of size `consumed_num_bytes` was written into given buffer (by `do_prepare_space`)
42 - The remaining unconsumed bytes are relinquished back to derived class
43
443) void do_set_badbit()
45 - informs derived type of failure when outputting, specifically invoked on:
46 - failure to allocate space
47 - failure to output
48*/
49
50template <class DerivedT>
51class OStreamBase;
52
53namespace detail {
54template <class T>
56
57template <class T>
58concept AllowedIntegral = std::is_integral_v<T> && !ExcludedIntegral<T>;
59
60template <class T>
61concept AllowedNumeric = AllowedIntegral<T> || std::floating_point<T>;
62
63template <class T>
65
66template <class T>
68 std::is_base_of_v<OStreamBase<std::remove_cvref_t<T>>, std::remove_cvref_t<T>>;
69
70} // namespace detail
71
72template <class DerivedT>
74 public:
75 DerivedT& put_data(const char* data, std::size_t data_size);
76
77 template <class T>
80
82
83 template <class T>
84 requires (detail::AllowedIntegral<T> && !std::same_as<T, bool>)
86
87 template <class T>
88 requires std::floating_point<T>
90
92 DerivedT& put(char ch);
93 DerivedT& write(const char* data, std::size_t sz);
94
95 protected:
96 // A number requires exactly N bytes to print, however put_num() will request a larger buffer
97 // size for performance reasons (ofcourse, when the buffer is relinquished it will correctly
98 // indicated that N bytes are consumed)
99 // However, if derived class cannot fullfill this request, then will fallback to slower path
100 // that requests exactly the N bytes required to print the number.
101 static constexpr std::size_t PutNumMaxBufRequest = 32u;
102
103 private:
104 DerivedT& get_derived() noexcept;
105
106 char* prepare_space(std::size_t num_bytes);
107 void relinquish_space(std::size_t consumed_num_bytes);
108 void set_badbit();
109};
110
112DerivedT& OStreamBase<DerivedT>::get_derived() noexcept
113{
114 // sizeof(T) results in a compiler error if T is an incomplete type
115 static_assert(sizeof(DerivedT) >= 0, "incomplete derived type");
116 static_assert(std::is_base_of_v<OStreamBase<DerivedT>, DerivedT>);
117 return static_cast<DerivedT&>(*this);
118}
119
120template <class DerivedT>
122{
123 return get_derived().do_prepare_space(num_bytes);
124}
125
126template <class DerivedT>
128{
129 return get_derived().do_relinquish_space(consumed_num_bytes);
130}
131
132template <class DerivedT>
133void OStreamBase<DerivedT>::set_badbit()
134{
135 get_derived().do_set_badbit();
136}
137
138template <class DerivedT>
140{
141 char* buf = prepare_space(data_size);
142 if (buf != nullptr) [[likely]] {
143 std::copy(data, data + data_size, buf);
144 relinquish_space(data_size);
145 } else {
146 set_badbit();
147 }
148 return get_derived();
149}
150
151template <class DerivedT>
152template <class T> requires detail::AllowedChar<T>
154{
155 char* buf = prepare_space(1);
156 if (buf != nullptr) [[likely]] {
157 *buf = ch;
158 relinquish_space(1);
159 } else {
160 set_badbit();
161 }
162 return get_derived();
163}
164
165template <class DerivedT>
167{
168 put_char(val ? '1' : '0');
169 return get_derived();
170}
171
172template <class DerivedT>
173template <class T> requires (detail::AllowedIntegral<T> && !std::same_as<T, bool>)
175{
176 constexpr std::size_t MaxBytesNeeded
177 = dec_digits(std::numeric_limits<T>::max())
178 + std::is_signed_v<T>;
179
180 static_assert(MaxBytesNeeded <= PutNumMaxBufRequest);
181
182 char* buf = prepare_space(MaxBytesNeeded);
183 std::size_t buf_sz = MaxBytesNeeded;
184
185 if (buf == nullptr) [[unlikely]] {
186 // slightly slower path
187 auto digits = dec_digits(val);
188 if constexpr (std::is_signed_v<T>) {
189 digits += (val < 0);
190 }
191 buf = prepare_space(digits);
192 buf_sz = digits;
193 }
194
195 if (buf != nullptr) [[likely]] {
196 // impossible for it to fail (N.B. to_chars is non-throwing)
197 const auto [end, ec] = std::to_chars(buf, buf + buf_sz, val);
198 assert(ec == std::errc());
199 relinquish_space(end - buf);
200 } else {
201 set_badbit();
202 }
203
204 return get_derived();
205}
206
207template <class DerivedT>
208template <class T> requires std::floating_point<T>
210{
211 char* buf = prepare_space(PutNumMaxBufRequest);
212 if (buf != nullptr) [[likely]] { // fast path
213 // impossible for to_chars to fail (N.B. to_chars is non-throwing)
214 const auto [end, ec] = std::to_chars(buf, buf + PutNumMaxBufRequest, val);
215 relinquish_space(end - buf);
216 assert(ec == std::errc());
217 }
218 else {
219 // slower path
220 // impossible for to_chars to fail (N.B. to_chars is non-throwing)
222 const auto [end, ec] = std::to_chars(local_space, local_space + PutNumMaxBufRequest, val);
223 assert(ec == std::errc());
224
225 std::size_t consumed = end - local_space;
226 buf = prepare_space(consumed);
227
228 if (buf != nullptr) [[likely]] {
229 std::copy(local_space, end, buf);
230 relinquish_space(consumed);
231 } else {
232 set_badbit();
233 }
234 }
235
236 return get_derived();
237}
238
239template <class DerivedT>
241{
242 return put_char(ch);
243}
244
245template <class DerivedT>
247{
248 return put_data(data, sz);
249}
250
251template <class StreamT>
253StreamT& operator<<(StreamT&& os, bool value)
254{
255 os.put_num(value);
256 return os;
257}
258
259template <class StreamT>
260 requires detail::InheritsBasicOStream<StreamT>
261StreamT& operator<<(StreamT&& os, short value)
262{
263 os.put_num(value);
264 return os;
265}
266
267template <class StreamT>
268 requires detail::InheritsBasicOStream<StreamT>
269StreamT& operator<<(StreamT&& os, unsigned short value)
270{
271 os.put_num(value);
272 return os;
273}
274
275template <class StreamT>
276 requires detail::InheritsBasicOStream<StreamT>
277StreamT& operator<<(StreamT&& os, int value)
278{
279 os.put_num(value);
280 return os;
281}
282
283template <class StreamT>
284 requires detail::InheritsBasicOStream<StreamT>
285StreamT& operator<<(StreamT&& os, unsigned int value)
286{
287 os.put_num(value);
288 return os;
289}
290
291template <class StreamT>
292 requires detail::InheritsBasicOStream<StreamT>
293StreamT& operator<<(StreamT&& os, long value)
294{
295 os.put_num(value);
296 return os;
297}
298
299template <class StreamT>
300 requires detail::InheritsBasicOStream<StreamT>
301StreamT& operator<<(StreamT&& os, unsigned long value)
302{
303 os.put_num(value);
304 return os;
305}
306
307template <class StreamT>
308 requires detail::InheritsBasicOStream<StreamT>
309StreamT& operator<<(StreamT&& os, long long value)
310{
311 os.put_num(value);
312 return os;
313}
314
315template <class StreamT>
316 requires detail::InheritsBasicOStream<StreamT>
317StreamT& operator<<(StreamT&& os, unsigned long long value)
318{
319 os.put_num(value);
320 return os;
321}
322
323template <class StreamT>
324 requires detail::InheritsBasicOStream<StreamT>
325StreamT& operator<<(StreamT&& os, float value)
326{
327 os.put_num(value);
328 return os;
329}
330
331template <class StreamT>
332 requires detail::InheritsBasicOStream<StreamT>
333StreamT& operator<<(StreamT&& os, double value)
334{
335 os.put_num(value);
336 return os;
337}
338
339template <class StreamT>
340 requires detail::InheritsBasicOStream<StreamT>
341StreamT& operator<<(StreamT&& os, long double value)
342{
343 os.put_num(value);
344 return os;
345}
346
347template <class StreamT>
348 requires detail::InheritsBasicOStream<StreamT>
350{
351 os.put_char(ch);
352 return os;
353}
354
355template <class StreamT>
356 requires detail::InheritsBasicOStream<StreamT>
357StreamT& operator<<(StreamT& os, signed char ch)
358{
359 os.put_char(ch);
360 return os;
361}
362
363template <class StreamT>
364 requires detail::InheritsBasicOStream<StreamT>
365StreamT& operator<<(StreamT& os, unsigned char ch)
366{
367 os.put_char(ch);
368 return os;
369}
370
371template <class StreamT>
372 requires detail::InheritsBasicOStream<StreamT>
373StreamT& operator<<(StreamT& os, const char* s)
374{
375 std::size_t len = std::char_traits<char>::length(s);
376 os.put_data(s, len);
377 return os;
378}
379
380template <class StreamT>
381 requires detail::InheritsBasicOStream<StreamT>
382StreamT& operator<<(StreamT& os, const signed char* s)
383{
384 const char* data = std::bit_cast<const char*>(s);
385 std::size_t len = std::char_traits<char>::length(data);
386 os.put_data(data, len);
387 return os;
388}
389
390template <class StreamT>
391 requires detail::InheritsBasicOStream<StreamT>
392StreamT& operator<<(StreamT& os, const unsigned char* s)
393{
394 const char* data = std::bit_cast<const char*>(s);
395 std::size_t len = std::char_traits<char>::length(data);
396 os.put_data(data, len);
397 return os;
398}
399
400template <class StreamT>
401 requires detail::InheritsBasicOStream<StreamT>
402StreamT& operator<<(StreamT& os, const std::string& s)
403{
404 os.put_data(s.data(), s.size());
405 return os;
406}
407
408template <class StreamT>
409 requires detail::InheritsBasicOStream<StreamT>
410StreamT& operator<<(StreamT& os, std::string_view s)
411{
412 os.put_data(s.data(), s.size());
413 return os;
414}
415
416template <class StreamT, class T>
417 requires (detail::InheritsBasicOStream<StreamT> &&
418 std::is_rvalue_reference_v<StreamT&&>)
419StreamT&& operator<<(StreamT&& os, const T& value)
420{
421 os << value;
422 return static_cast<StreamT&&>(os);
423}
424
425} // namespace util
426} // namespace toolbox
427
428#endif // TOOLBOX_UTIL_OSTREAM_BASE_HPP
DerivedT & put_num(bool val)
static constexpr std::size_t PutNumMaxBufRequest
DerivedT & write(const char *data, std::size_t sz)
DerivedT & put_num(T val)
DerivedT & put_num(T val)
DerivedT & put(char ch)
For compatability with std::ostream API.
DerivedT & put_data(const char *data, std::size_t data_size)
STL namespace.
ostream & operator<<(ostream &os, const pair< T, U > &p)
Definition Parser.ut.cpp:29
const DataT & data(const MsgEvent &ev) noexcept
Definition Event.hpp:54
constexpr int dec_digits(ValueT i) noexcept
Definition Utility.hpp:52
constexpr auto bind() noexcept
Definition Slot.hpp:97