33#ifndef _GLIBCXX_PRINT_H
34#define _GLIBCXX_PRINT_H 1
37#pragma GCC system_header
55namespace std _GLIBCXX_VISIBILITY(default)
57_GLIBCXX_BEGIN_NAMESPACE_VERSION
61#if _GLIBCXX_USE_STDIO_LOCKING && _GLIBCXX_USE_GLBC_STDIO_EXT
63 extern "C" int __fwritable(FILE*)
noexcept;
64 extern "C" int __flbf(FILE*)
noexcept;
65 extern "C" size_t __fbufsize(FILE*)
noexcept;
69 class _File_sink final : _Buf_sink<char>
74 _File(FILE* __f) : _M_file(__f)
78 if (!__fwritable(__f))
81 __throw_system_error(EACCES);
84 if (_M_write_buf().empty())
85 if (::__overflow(__f, EOF) == EOF)
87 const int __err = errno;
89 __throw_system_error(__err);
93 ~_File() { ::funlockfile(_M_file); }
95 _File(_File&&) =
delete;
99 _M_write_buf() noexcept
101 return {_M_file->_IO_write_ptr,
102 size_t(_M_file->_IO_buf_end - _M_file->_IO_write_ptr)};
109 if (::fflush_unlocked(_M_file))
110 __throw_system_error(errno);
117 _M_bump(
size_t __n)
noexcept
118 { _M_file->_IO_write_ptr += __n; }
121 _M_line_buffered() const noexcept
122 {
return __flbf(_M_file); }
125 _M_unbuffered() const noexcept
126 {
return __fbufsize(_M_file) == 1; }
135 _M_overflow()
override
137 auto __s = this->_M_used();
138 if (__s.data() == this->_M_buf)
141 auto __n = ::fwrite_unlocked(__s.data(), 1, __s.size(),
143 if (__n != __s.size())
144 __throw_system_error(errno);
145 this->_M_reset(this->_M_buf);
150 _M_file._M_bump(__s.size());
152 this->_M_reset(_M_file._M_write_buf());
157 _File_sink(FILE* __f,
bool __add_newline)
158 : _M_file(__f), _M_add_newline(__add_newline)
160 if (!_M_file._M_unbuffered())
162 this->_M_reset(_M_file._M_write_buf());
168 ~_File_sink() noexcept(false)
170 auto __s = this->_M_used();
171 if (__s.data() == this->_M_buf)
173 _File_sink::_M_overflow();
175 ::putc_unlocked(
'\n', _M_file._M_file);
179 _M_file._M_bump(__s.size());
181 ::putc_unlocked(
'\n', _M_file._M_file);
182 else if (_M_file._M_line_buffered() && __s.size()
183 && (__s.back() ==
'\n'
184 || __builtin_memchr(__s.data(),
'\n', __s.size())))
189 using _Sink<
char>::out;
191#elif _GLIBCXX_USE_STDIO_LOCKING
194 class _File_sink final : _Buf_sink<char>
198 explicit _File(FILE* __f) : _M_file(__f) { ::flockfile(_M_file); }
199 ~_File() { ::funlockfile(_M_file); }
207 _M_overflow()
override
209 auto __s = this->_M_used();
210#if _GLIBCXX_HAVE_FWRITE_UNLOCKED
211 auto __n = ::fwrite_unlocked(__s.data(), 1, __s.size(), _M_file._M_file);
212 if (__n != __s.size())
213 __throw_system_error(errno);
216 ::putc_unlocked(__c, _M_file._M_file);
217 if (::ferror(_M_file._M_file))
218 __throw_system_error(errno);
220 this->_M_reset(this->_M_buf);
224 _File_sink(FILE* __f,
bool __add_newline) noexcept
225 : _Buf_sink<char>(), _M_file(__f), _M_add_newline(__add_newline)
228 ~_File_sink() noexcept(false)
230 _File_sink::_M_overflow();
232 ::putc_unlocked(
'\n', _M_file._M_file);
235 using _Sink<
char>::out;
241 class _File_sink final
244 _Str_sink<char> _M_sink;
248 _File_sink(FILE* __f,
bool __add_newline) noexcept
249 : _M_file(__f), _M_add_newline(__add_newline)
252 ~_File_sink() noexcept(false)
257 auto __n = std::fwrite(__s.data(), 1, __s.size(), _M_file);
258 if (__n < __s.size())
259 __throw_system_error(EIO);
262 auto out() {
return _M_sink.out(); }
267#ifdef _GLIBCXX_NO_INLINE_PRINT
268# define _GLIBCXX_PRINT_INLINE_USED [[__gnu__::__used__]]
270# define _GLIBCXX_PRINT_INLINE_USED
273 _GLIBCXX_PRINT_INLINE_USED
275 vprint_nonunicode(FILE* __stream, string_view __fmt, format_args __args)
277 std::vformat_to(__format::_File_sink(__stream,
false).out(), __fmt, __args);
280 _GLIBCXX_PRINT_INLINE_USED
282 vprint_nonunicode_buffered(FILE* __stream, string_view __fmt,
285 __format::_Str_sink<char> __buf;
286 std::vformat_to(__buf.out(), __fmt, __args);
287 auto __out = __buf.view();
288 if (std::fwrite(__out.data(), 1, __out.size(), __stream) != __out.size())
289 __throw_system_error(EIO);
292 _GLIBCXX_PRINT_INLINE_USED
294 vprint_unicode(FILE* __stream, string_view __fmt, format_args __args)
296#if !defined(_WIN32) || defined(__CYGWIN__)
299 std::vprint_nonunicode(__stream, __fmt, __args);
301 __format::_Str_sink<char> __buf;
302 std::vformat_to(__buf.out(), __fmt, __args);
303 auto __out = __buf._M_span();
305 void* __open_terminal(FILE*);
306 error_code __write_to_terminal(
void*, span<char>);
308 if (
auto __term = __open_terminal(__stream))
311 if (!std::fflush(__stream))
313 __e = __write_to_terminal(__term, __out);
321 _GLIBCXX_THROW_OR_ABORT(
system_error(__e,
"std::vprint_unicode"));
325 if (std::fwrite(__out.data(), 1, __out.size(), __stream) != __out.size())
326 __throw_system_error(EIO);
330 _GLIBCXX_PRINT_INLINE_USED
332 vprint_unicode_buffered(FILE* __stream, string_view __fmt, format_args __args)
334#if !defined(_WIN32) || defined(__CYGWIN__)
337 std::vprint_nonunicode_buffered(__stream, __fmt, __args);
341 std::vprint_unicode(__stream, __fmt, __args);
344#undef _GLIBCXX_PRINT_INLINE_USED
346_GLIBCXX_END_NAMESPACE_VERSION
const error_category & generic_category() noexcept
Error category for errno error codes.
error_code make_error_code(future_errc __errc) noexcept
Overload of make_error_code for future_errc.
constexpr std::remove_reference< _Tp >::type && move(_Tp &&__t) noexcept
Convert a value to an rvalue.
ISO C++ entities toplevel namespace is std.
An exception type that includes an error_code value.