Skip to content

Commit c7394d2

Browse files
committed
[libc] Template the writing mode for the writer class
Summary: Currently we dispatch the writing mode off of a runtime enum passed in by the constructor. This causes very unfortunate codegen for the GPU targets where we get worst-case codegen because of the unused function pointer for `sprintf`. Instead, this patch moves all of this to a template so it can be masked out. This results in no dynamic stack and uses 60 VGPRs instead of 117. It also compiles about 5x as fast.
1 parent ee57a68 commit c7394d2

31 files changed

+335
-352
lines changed

libc/src/stdio/printf_core/CMakeLists.txt

+3-9
Original file line numberDiff line numberDiff line change
@@ -59,10 +59,8 @@ add_header_library(
5959
libc.src.__support.common
6060
)
6161

62-
add_object_library(
62+
add_header_library(
6363
writer
64-
SRCS
65-
writer.cpp
6664
HDRS
6765
writer.h
6866
DEPENDS
@@ -73,10 +71,8 @@ add_object_library(
7371
libc.src.string.memory_utils.inline_memset
7472
)
7573

76-
add_object_library(
74+
add_header_library(
7775
converter
78-
SRCS
79-
converter.cpp
8076
HDRS
8177
converter.h
8278
converter_atlas.h
@@ -110,10 +106,8 @@ add_object_library(
110106
libc.src.__support.StringUtil.error_to_string
111107
)
112108

113-
add_object_library(
109+
add_header_library(
114110
printf_main
115-
SRCS
116-
printf_main.cpp
117111
HDRS
118112
printf_main.h
119113
DEPENDS

libc/src/stdio/printf_core/char_converter.h

+3-1
Original file line numberDiff line numberDiff line change
@@ -17,7 +17,9 @@
1717
namespace LIBC_NAMESPACE_DECL {
1818
namespace printf_core {
1919

20-
LIBC_INLINE int convert_char(Writer *writer, const FormatSection &to_conv) {
20+
template <WriteMode write_mode>
21+
LIBC_INLINE int convert_char(Writer<write_mode> *writer,
22+
const FormatSection &to_conv) {
2123
char c = static_cast<char>(to_conv.conv_val_raw);
2224

2325
constexpr int STRING_LEN = 1;

libc/src/stdio/printf_core/converter.cpp

-105
This file was deleted.

libc/src/stdio/printf_core/converter.h

+84-1
Original file line numberDiff line numberDiff line change
@@ -11,8 +11,18 @@
1111

1212
#include "src/__support/macros/config.h"
1313
#include "src/stdio/printf_core/core_structs.h"
14+
#include "src/stdio/printf_core/printf_config.h"
15+
#include "src/stdio/printf_core/strerror_converter.h"
1416
#include "src/stdio/printf_core/writer.h"
1517

18+
// This option allows for replacing all of the conversion functions with custom
19+
// replacements. This allows conversions to be replaced at compile time.
20+
#ifndef LIBC_COPT_PRINTF_CONV_ATLAS
21+
#include "src/stdio/printf_core/converter_atlas.h"
22+
#else
23+
#include LIBC_COPT_PRINTF_CONV_ATLAS
24+
#endif
25+
1626
#include <stddef.h>
1727

1828
namespace LIBC_NAMESPACE_DECL {
@@ -21,7 +31,80 @@ namespace printf_core {
2131
// convert will call a conversion function to convert the FormatSection into
2232
// its string representation, and then that will write the result to the
2333
// writer.
24-
int convert(Writer *writer, const FormatSection &to_conv);
34+
template <WriteMode write_mode>
35+
int convert(Writer<write_mode> *writer, const FormatSection &to_conv) {
36+
if (!to_conv.has_conv)
37+
return writer->write(to_conv.raw_string);
38+
39+
#if !defined(LIBC_COPT_PRINTF_DISABLE_FLOAT) && \
40+
defined(LIBC_COPT_PRINTF_HEX_LONG_DOUBLE)
41+
if (to_conv.length_modifier == LengthModifier::L) {
42+
switch (to_conv.conv_name) {
43+
case 'f':
44+
case 'F':
45+
case 'e':
46+
case 'E':
47+
case 'g':
48+
case 'G':
49+
return convert_float_hex_exp(writer, to_conv);
50+
default:
51+
break;
52+
}
53+
}
54+
#endif // LIBC_COPT_PRINTF_DISABLE_FLOAT
55+
56+
switch (to_conv.conv_name) {
57+
case '%':
58+
return writer->write("%");
59+
case 'c':
60+
return convert_char(writer, to_conv);
61+
case 's':
62+
return convert_string(writer, to_conv);
63+
case 'd':
64+
case 'i':
65+
case 'u':
66+
case 'o':
67+
case 'x':
68+
case 'X':
69+
case 'b':
70+
case 'B':
71+
return convert_int(writer, to_conv);
72+
#ifndef LIBC_COPT_PRINTF_DISABLE_FLOAT
73+
case 'f':
74+
case 'F':
75+
return convert_float_decimal(writer, to_conv);
76+
case 'e':
77+
case 'E':
78+
return convert_float_dec_exp(writer, to_conv);
79+
case 'a':
80+
case 'A':
81+
return convert_float_hex_exp(writer, to_conv);
82+
case 'g':
83+
case 'G':
84+
return convert_float_dec_auto(writer, to_conv);
85+
#endif // LIBC_COPT_PRINTF_DISABLE_FLOAT
86+
#ifdef LIBC_INTERNAL_PRINTF_HAS_FIXED_POINT
87+
case 'r':
88+
case 'R':
89+
case 'k':
90+
case 'K':
91+
return convert_fixed(writer, to_conv);
92+
#endif // LIBC_INTERNAL_PRINTF_HAS_FIXED_POINT
93+
#ifndef LIBC_COPT_PRINTF_DISABLE_STRERROR
94+
case 'm':
95+
return convert_strerror(writer, to_conv);
96+
#endif // LIBC_COPT_PRINTF_DISABLE_STRERROR
97+
#ifndef LIBC_COPT_PRINTF_DISABLE_WRITE_INT
98+
case 'n':
99+
return convert_write_int(writer, to_conv);
100+
#endif // LIBC_COPT_PRINTF_DISABLE_WRITE_INT
101+
case 'p':
102+
return convert_pointer(writer, to_conv);
103+
default:
104+
return writer->write(to_conv.raw_string);
105+
}
106+
return -1;
107+
}
25108

26109
} // namespace printf_core
27110
} // namespace LIBC_NAMESPACE_DECL

libc/src/stdio/printf_core/fixed_converter.h

+3-1
Original file line numberDiff line numberDiff line change
@@ -62,7 +62,9 @@ LIBC_INLINE constexpr uint32_t const_ten_exp(uint32_t exponent) {
6262
} \
6363
} while (false)
6464

65-
LIBC_INLINE int convert_fixed(Writer *writer, const FormatSection &to_conv) {
65+
template <WriteMode write_mode>
66+
LIBC_INLINE int convert_fixed(Writer<write_mode> *writer,
67+
const FormatSection &to_conv) {
6668
// Long accum should be the largest type, so we can store all the smaller
6769
// numbers in things sized for it.
6870
using LARep = fixed_point::FXRep<unsigned long accum>;

0 commit comments

Comments
 (0)