From 36707ec87aec7f92cc52d8250e18ce9ce98e912a Mon Sep 17 00:00:00 2001 From: Bananymous Date: Wed, 10 May 2023 22:36:03 +0300 Subject: [PATCH] LibC: implement printf conversions e, E, f, F --- libc/printf_impl.cpp | 111 +++++++++++++++++++++++++++++++------------ 1 file changed, 80 insertions(+), 31 deletions(-) diff --git a/libc/printf_impl.cpp b/libc/printf_impl.cpp index 98377ad6..c6194d51 100644 --- a/libc/printf_impl.cpp +++ b/libc/printf_impl.cpp @@ -1,6 +1,8 @@ #include +#include #include #include +#include #include #include @@ -35,7 +37,7 @@ static void integer_to_string(char* buffer, T value, int base, bool upper, const } else if (options.width != -1) width = options.width; - + auto digit_char = [](int digit, bool upper) { if (digit < 10) @@ -95,16 +97,6 @@ static void integer_to_string(char* buffer, T value, int base, bool upper, const buffer[offset++] = '\0'; } -template -static void floating_point_to_exponent_string(char* buffer, T value, bool upper, const format_options_t options) -{ - int percision = 6; - if (options.percision != -1) - percision = options.percision; - - strcpy(buffer, "??e??"); -} - template static void floating_point_to_string(char* buffer, T value, bool upper, const format_options_t options) { @@ -114,7 +106,8 @@ static void floating_point_to_string(char* buffer, T value, bool upper, const fo int offset = 0; - if (value < 0) + // Add sign if needed + if (value < (T)0.0) { buffer[offset++] = '-'; value = -value; @@ -124,42 +117,98 @@ static void floating_point_to_string(char* buffer, T value, bool upper, const fo else if (options.show_plus_sign_as_space) buffer[offset++] = ' '; - int exponent = 1; - while (value > exponent * 10) - exponent *= 10; + // Round last digit + value += (T)0.5 * BAN::Math::pow(10.0, -percision); - T fractional = value; - while (exponent >= 1) + // Add integer part of the decimal + if (value < (T)1.0) { - int scale = 0; - for (; scale < 10; scale++) - if (fractional < T(exponent * (scale + 1))) - break; - buffer[offset++] = '0' + scale; - fractional -= T(exponent * scale); - exponent /= 10; + buffer[offset++] = '0'; + value *= (T)10.0; + } + else + { + int exponent = (int)BAN::Math::log10(value); + T magnitude = BAN::Math::pow(10, exponent); + + value /= magnitude; + + for (int i = 0; i <= exponent; i++) + { + int digit = 0; + for (; digit < 10; digit++) + if (value < digit + 1) + break; + buffer[offset++] = '0' + digit; + value -= (T)digit; + value *= (T)10.0; + } } - if (!options.alternate_form && percision <= 0) + // We are done if the decimal part is not written + if (percision == 0 && !options.alternate_form) { buffer[offset++] = '\0'; return; } - buffer[offset++] = '.'; + + // Add the 'percision' digits after decimal point for (int i = 0; i < percision; i++) { - fractional *= T(10.0); - if (i == percision - 1) - fractional += T(0.5); - int digit = fractional; + int digit = 0; + for (; digit < 10; digit++) + if (value < digit + 1) + break; buffer[offset++] = '0' + digit; - fractional -= T(digit); + value -= (T)digit; + value *= (T)10.0; } buffer[offset++] = '\0'; } +template +static void floating_point_to_exponent_string(char* buffer, T value, bool upper, const format_options_t options) +{ + int percision = 6; + if (options.percision != -1) + percision = options.percision; + + int offset = 0; + + // Add sign if needed + if (value < (T)0.0) + { + buffer[offset++] = '-'; + value = -value; + } + else if (options.show_plus_sign) + buffer[offset++] = '+'; + else if (options.show_plus_sign_as_space) + buffer[offset++] = ' '; + + // Calculate which number to put as exponent + int exponent = 0; + if (value != (T)0.0) + { + exponent = (int)floorl(BAN::Math::log10(value)); + value /= BAN::Math::pow(10.0, exponent); + } + + // Add first numbers before 'e' + floating_point_to_string(buffer + offset, value, upper, options); + offset = strlen(buffer); + + // Add the exponent part + buffer[offset++] = (upper ? 'E' : 'e'); + format_options_t exponent_options; + exponent_options.show_plus_sign = true; + exponent_options.zero_padded = true; + exponent_options.width = 3; + integer_to_string(buffer + offset, exponent, 10, upper, exponent_options); +} + extern "C" int printf_impl(const char* format, va_list arguments, int (*putc_fun)(int, void*), void* data) { int written = 0;