blob: 2f521544785c0dbab2a01a44160745c4ff8c1380 [file]
//===-- String Converter for printf -----------------------------*- 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 LLVM_LIBC_SRC_STDIO_PRINTF_CORE_STRING_CONVERTER_H
#define LLVM_LIBC_SRC_STDIO_PRINTF_CORE_STRING_CONVERTER_H
#ifndef LIBC_COPT_PRINTF_DISABLE_WIDE
#include "hdr/types/char32_t.h"
#include "hdr/types/char8_t.h"
#include "src/__support/wchar/mbstate.h"
#include "src/__support/wchar/string_converter.h"
#endif // LIBC_COPT_PRINTF_DISABLE_WIDE
#include "src/__support/macros/config.h"
#include "src/stdio/printf_core/converter_utils.h"
#include "src/stdio/printf_core/core_structs.h"
#include "src/stdio/printf_core/writer.h"
#include "src/string/string_utils.h" // string_length
#include <stddef.h>
namespace LIBC_NAMESPACE_DECL {
namespace printf_core {
template <WriteMode write_mode>
LIBC_INLINE int char_writer(Writer<write_mode> *writer,
const FormatSection &to_conv) {
const char *str_ptr = reinterpret_cast<const char *>(to_conv.conv_val_ptr);
size_t string_len = 0;
#ifndef LIBC_COPT_PRINTF_NO_NULLPTR_CHECKS
if (str_ptr == nullptr) {
str_ptr = "(null)";
}
#endif // LIBC_COPT_PRINTF_NO_NULLPTR_CHECKS
string_len = internal::string_length(str_ptr);
if (to_conv.precision >= 0 &&
static_cast<size_t>(to_conv.precision) < string_len)
string_len = to_conv.precision;
size_t padding_spaces = to_conv.min_width > static_cast<int>(string_len)
? to_conv.min_width - string_len
: 0;
// If the padding is on the left side, write the spaces first.
if (padding_spaces > 0 &&
(to_conv.flags & FormatFlags::LEFT_JUSTIFIED) == 0) {
RET_IF_RESULT_NEGATIVE(writer->write(' ', padding_spaces));
}
RET_IF_RESULT_NEGATIVE(writer->write({(str_ptr), string_len}));
// If the padding is on the right side, write the spaces last.
if (padding_spaces > 0 &&
(to_conv.flags & FormatFlags::LEFT_JUSTIFIED) != 0) {
RET_IF_RESULT_NEGATIVE(writer->write(' ', padding_spaces));
}
return WRITE_OK;
}
#ifndef LIBC_COPT_PRINTF_DISABLE_WIDE
template <WriteMode write_mode>
LIBC_INLINE int wchar_writer(Writer<write_mode> *writer,
const FormatSection &to_conv) {
size_t string_len = 0;
const char32_t *wstr_ptr =
reinterpret_cast<const char32_t *>(to_conv.conv_val_ptr);
size_t precision =
to_conv.precision < 0 ? SIZE_MAX : static_cast<size_t>(to_conv.precision);
#ifndef LIBC_COPT_PRINTF_NO_NULLPTR_CHECKS
if (wstr_ptr == nullptr) {
wstr_ptr = U"(null)";
}
#endif // LIBC_COPT_PRINTF_NO_NULLPTR_CHECKS
internal::mbstate mbstate;
internal::StringConverter<char32_t> length_counter(wstr_ptr, &mbstate,
precision);
for (auto converted = length_counter.pop<char8_t>();
converted.has_value() && converted.value() != '\0';
converted = length_counter.pop<char8_t>()) {
++string_len;
}
size_t padding_spaces = to_conv.min_width > static_cast<int>(string_len)
? to_conv.min_width - string_len
: 0;
// If the padding is on the left side, write the spaces first.
if (padding_spaces > 0 &&
(to_conv.flags & FormatFlags::LEFT_JUSTIFIED) == 0) {
RET_IF_RESULT_NEGATIVE(writer->write(' ', padding_spaces));
}
mbstate = internal::mbstate();
internal::StringConverter<char32_t> out_conv(wstr_ptr, &mbstate, precision);
for (auto converted = out_conv.pop<char8_t>();
converted.has_value() && converted.value() != '\0';
converted = out_conv.pop<char8_t>()) {
RET_IF_RESULT_NEGATIVE(writer->write(static_cast<char>(converted.value())));
}
// If the padding is on the right side, write the spaces last.
if (padding_spaces > 0 &&
(to_conv.flags & FormatFlags::LEFT_JUSTIFIED) != 0) {
RET_IF_RESULT_NEGATIVE(writer->write(' ', padding_spaces));
}
return WRITE_OK;
}
#endif // LIBC_COPT_PRINTF_DISABLE_WIDE
template <WriteMode write_mode>
LIBC_INLINE int convert_string(Writer<write_mode> *writer,
const FormatSection &to_conv) {
int ret = 0;
if (to_conv.length_modifier == LengthModifier::l) {
// find length and print wide char characters
#ifndef LIBC_COPT_PRINTF_DISABLE_WIDE
ret = wchar_writer(writer, to_conv);
#else
ret = char_writer(writer, to_conv);
#endif
} else {
ret = char_writer(writer, to_conv);
}
return ret;
}
} // namespace printf_core
} // namespace LIBC_NAMESPACE_DECL
#endif // LLVM_LIBC_SRC_STDIO_PRINTF_CORE_STRING_CONVERTER_H