mirror of
https://github.com/jart/cosmopolitan.git
synced 2025-01-31 03:27:39 +00:00
180 lines
7.1 KiB
C
180 lines
7.1 KiB
C
|
//===---------------------------------------------------------------------===//
|
||
|
//
|
||
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
||
|
// See https://llvm.org/LICENSE.txt for license information.
|
||
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
||
|
//
|
||
|
//===---------------------------------------------------------------------===//
|
||
|
|
||
|
#ifndef _LIBCPP___OSTREAM_PRINT_H
|
||
|
#define _LIBCPP___OSTREAM_PRINT_H
|
||
|
|
||
|
#include <__config>
|
||
|
#include <__fwd/ostream.h>
|
||
|
#include <__iterator/ostreambuf_iterator.h>
|
||
|
#include <__ostream/basic_ostream.h>
|
||
|
#include <format>
|
||
|
#include <ios>
|
||
|
#include <locale>
|
||
|
#include <print>
|
||
|
|
||
|
#if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER)
|
||
|
# pragma GCC system_header
|
||
|
#endif
|
||
|
|
||
|
_LIBCPP_BEGIN_NAMESPACE_STD
|
||
|
|
||
|
#if _LIBCPP_STD_VER >= 23
|
||
|
|
||
|
template <class = void> // TODO PRINT template or availability markup fires too eagerly (http://llvm.org/PR61563).
|
||
|
_LIBCPP_HIDE_FROM_ABI inline void
|
||
|
__vprint_nonunicode(ostream& __os, string_view __fmt, format_args __args, bool __write_nl) {
|
||
|
// [ostream.formatted.print]/3
|
||
|
// Effects: Behaves as a formatted output function
|
||
|
// ([ostream.formatted.reqmts]) of os, except that:
|
||
|
// - failure to generate output is reported as specified below, and
|
||
|
// - any exception thrown by the call to vformat is propagated without regard
|
||
|
// to the value of os.exceptions() and without turning on ios_base::badbit
|
||
|
// in the error state of os.
|
||
|
// After constructing a sentry object, the function initializes an automatic
|
||
|
// variable via
|
||
|
// string out = vformat(os.getloc(), fmt, args);
|
||
|
|
||
|
ostream::sentry __s(__os);
|
||
|
if (__s) {
|
||
|
string __o = std::vformat(__os.getloc(), __fmt, __args);
|
||
|
if (__write_nl)
|
||
|
__o += '\n';
|
||
|
|
||
|
const char* __str = __o.data();
|
||
|
size_t __len = __o.size();
|
||
|
|
||
|
# ifndef _LIBCPP_HAS_NO_EXCEPTIONS
|
||
|
try {
|
||
|
# endif // _LIBCPP_HAS_NO_EXCEPTIONS
|
||
|
typedef ostreambuf_iterator<char> _Ip;
|
||
|
if (std::__pad_and_output(
|
||
|
_Ip(__os),
|
||
|
__str,
|
||
|
(__os.flags() & ios_base::adjustfield) == ios_base::left ? __str + __len : __str,
|
||
|
__str + __len,
|
||
|
__os,
|
||
|
__os.fill())
|
||
|
.failed())
|
||
|
__os.setstate(ios_base::badbit | ios_base::failbit);
|
||
|
|
||
|
# ifndef _LIBCPP_HAS_NO_EXCEPTIONS
|
||
|
} catch (...) {
|
||
|
__os.__set_badbit_and_consider_rethrow();
|
||
|
}
|
||
|
# endif // _LIBCPP_HAS_NO_EXCEPTIONS
|
||
|
}
|
||
|
}
|
||
|
|
||
|
template <class = void> // TODO PRINT template or availability markup fires too eagerly (http://llvm.org/PR61563).
|
||
|
_LIBCPP_HIDE_FROM_ABI inline void vprint_nonunicode(ostream& __os, string_view __fmt, format_args __args) {
|
||
|
std::__vprint_nonunicode(__os, __fmt, __args, false);
|
||
|
}
|
||
|
|
||
|
// Returns the FILE* associated with the __os.
|
||
|
// Returns a nullptr when no FILE* is associated with __os.
|
||
|
// This function is in the dylib since the type of the buffer associated
|
||
|
// with std::cout, std::cerr, and std::clog is only known in the dylib.
|
||
|
//
|
||
|
// This function implements part of the implementation-defined behavior
|
||
|
// of [ostream.formatted.print]/3
|
||
|
// If the function is vprint_unicode and os is a stream that refers to
|
||
|
// a terminal capable of displaying Unicode which is determined in an
|
||
|
// implementation-defined manner, writes out to the terminal using the
|
||
|
// native Unicode API;
|
||
|
// Whether the returned FILE* is "a terminal capable of displaying Unicode"
|
||
|
// is determined in the same way as the print(FILE*, ...) overloads.
|
||
|
_LIBCPP_EXPORTED_FROM_ABI FILE* __get_ostream_file(ostream& __os);
|
||
|
|
||
|
# ifndef _LIBCPP_HAS_NO_UNICODE
|
||
|
template <class = void> // TODO PRINT template or availability markup fires too eagerly (http://llvm.org/PR61563).
|
||
|
_LIBCPP_HIDE_FROM_ABI void __vprint_unicode(ostream& __os, string_view __fmt, format_args __args, bool __write_nl) {
|
||
|
# if _LIBCPP_AVAILABILITY_HAS_PRINT == 0
|
||
|
return std::__vprint_nonunicode(__os, __fmt, __args, __write_nl);
|
||
|
# else
|
||
|
FILE* __file = std::__get_ostream_file(__os);
|
||
|
if (!__file || !__print::__is_terminal(__file))
|
||
|
return std::__vprint_nonunicode(__os, __fmt, __args, __write_nl);
|
||
|
|
||
|
// [ostream.formatted.print]/3
|
||
|
// If the function is vprint_unicode and os is a stream that refers to a
|
||
|
// terminal capable of displaying Unicode which is determined in an
|
||
|
// implementation-defined manner, writes out to the terminal using the
|
||
|
// native Unicode API; if out contains invalid code units, the behavior is
|
||
|
// undefined and implementations are encouraged to diagnose it. If the
|
||
|
// native Unicode API is used, the function flushes os before writing out.
|
||
|
//
|
||
|
// This is the path for the native API, start with flushing.
|
||
|
__os.flush();
|
||
|
|
||
|
# ifndef _LIBCPP_HAS_NO_EXCEPTIONS
|
||
|
try {
|
||
|
# endif // _LIBCPP_HAS_NO_EXCEPTIONS
|
||
|
ostream::sentry __s(__os);
|
||
|
if (__s) {
|
||
|
# ifndef _LIBCPP_WIN32API
|
||
|
__print::__vprint_unicode_posix(__file, __fmt, __args, __write_nl, true);
|
||
|
# elif !defined(_LIBCPP_HAS_NO_WIDE_CHARACTERS)
|
||
|
__print::__vprint_unicode_windows(__file, __fmt, __args, __write_nl, true);
|
||
|
# else
|
||
|
# error "Windows builds with wchar_t disabled are not supported."
|
||
|
# endif
|
||
|
}
|
||
|
|
||
|
# ifndef _LIBCPP_HAS_NO_EXCEPTIONS
|
||
|
} catch (...) {
|
||
|
__os.__set_badbit_and_consider_rethrow();
|
||
|
}
|
||
|
# endif // _LIBCPP_HAS_NO_EXCEPTIONS
|
||
|
# endif // _LIBCPP_AVAILABILITY_HAS_PRINT
|
||
|
}
|
||
|
|
||
|
template <class = void> // TODO PRINT template or availability markup fires too eagerly (http://llvm.org/PR61563).
|
||
|
_LIBCPP_HIDE_FROM_ABI inline void vprint_unicode(ostream& __os, string_view __fmt, format_args __args) {
|
||
|
std::__vprint_unicode(__os, __fmt, __args, false);
|
||
|
}
|
||
|
# endif // _LIBCPP_HAS_NO_UNICODE
|
||
|
|
||
|
template <class... _Args>
|
||
|
_LIBCPP_HIDE_FROM_ABI void print(ostream& __os, format_string<_Args...> __fmt, _Args&&... __args) {
|
||
|
# ifndef _LIBCPP_HAS_NO_UNICODE
|
||
|
if constexpr (__print::__use_unicode_execution_charset)
|
||
|
std::__vprint_unicode(__os, __fmt.get(), std::make_format_args(__args...), false);
|
||
|
else
|
||
|
std::__vprint_nonunicode(__os, __fmt.get(), std::make_format_args(__args...), false);
|
||
|
# else // _LIBCPP_HAS_NO_UNICODE
|
||
|
std::__vprint_nonunicode(__os, __fmt.get(), std::make_format_args(__args...), false);
|
||
|
# endif // _LIBCPP_HAS_NO_UNICODE
|
||
|
}
|
||
|
|
||
|
template <class... _Args>
|
||
|
_LIBCPP_HIDE_FROM_ABI void println(ostream& __os, format_string<_Args...> __fmt, _Args&&... __args) {
|
||
|
# ifndef _LIBCPP_HAS_NO_UNICODE
|
||
|
// Note the wording in the Standard is inefficient. The output of
|
||
|
// std::format is a std::string which is then copied. This solution
|
||
|
// just appends a newline at the end of the output.
|
||
|
if constexpr (__print::__use_unicode_execution_charset)
|
||
|
std::__vprint_unicode(__os, __fmt.get(), std::make_format_args(__args...), true);
|
||
|
else
|
||
|
std::__vprint_nonunicode(__os, __fmt.get(), std::make_format_args(__args...), true);
|
||
|
# else // _LIBCPP_HAS_NO_UNICODE
|
||
|
std::__vprint_nonunicode(__os, __fmt.get(), std::make_format_args(__args...), true);
|
||
|
# endif // _LIBCPP_HAS_NO_UNICODE
|
||
|
}
|
||
|
|
||
|
template <class = void> // TODO PRINT template or availability markup fires too eagerly (http://llvm.org/PR61563).
|
||
|
_LIBCPP_HIDE_FROM_ABI inline void println(ostream& __os) {
|
||
|
std::print(__os, "\n");
|
||
|
}
|
||
|
|
||
|
#endif // _LIBCPP_STD_VER >= 23
|
||
|
|
||
|
_LIBCPP_END_NAMESPACE_STD
|
||
|
|
||
|
#endif // _LIBCPP___OSTREAM_PRINT_H
|