Implement right-justified padding

This commit is contained in:
Steins7 2024-04-15 22:56:49 +02:00
parent 3bcb9c7df4
commit e477055e6e

View File

@ -21,9 +21,9 @@ static uint32_t buffer_write(uint8_t byte, void* arg);
static uint32_t render_format(FormatCallback callback, void* arg, static uint32_t render_format(FormatCallback callback, void* arg,
const char* restrict format, va_list va); const char* restrict format, va_list va);
static uint32_t render_signed(FormatCallback callback, void* arg, static uint32_t render_signed(FormatCallback callback, void* arg,
int32_t number, uint8_t base); int32_t number, uint8_t base, uint8_t width, bool zero_padding);
static uint32_t render_unsigned(FormatCallback callback, void* arg, static uint32_t render_unsigned(FormatCallback callback, void* arg,
uint32_t number, uint8_t base); uint32_t number, uint8_t base, uint8_t width, bool zero_padding);
//--local variables------------------------------------------------------------- //--local variables-------------------------------------------------------------
@ -74,7 +74,9 @@ static uint32_t buffer_write(uint8_t byte, void* arg)
static uint32_t render_format(FormatCallback callback, void* arg, static uint32_t render_format(FormatCallback callback, void* arg,
const char* restrict format, va_list va) const char* restrict format, va_list va)
{ {
uint8_t width = 0;
bool in_format = false; bool in_format = false;
bool zero_padding = false;
for (const char* c = format; *c != '\0'; ++c) { for (const char* c = format; *c != '\0'; ++c) {
@ -97,22 +99,40 @@ static uint32_t render_format(FormatCallback callback, void* arg,
} }
break; break;
case 'i': case 'i':
if (render_signed(callback, arg, va_arg(va, int32_t), 10)) { if (render_signed(callback, arg, va_arg(va, int32_t),
10, width, zero_padding)) {
return 1; return 1;
} }
break; break;
case 'u': case 'u':
if (render_unsigned(callback, arg, if (render_unsigned(callback, arg, va_arg(va, uint32_t),
va_arg(va, uint32_t), 10)) { 10, width, zero_padding)) {
return 1; return 1;
} }
break; break;
case 'x': case 'x':
if (render_unsigned(callback, arg, if (render_unsigned(callback, arg, va_arg(va, uint32_t),
va_arg(va, uint32_t), 16)) { 16, width, zero_padding)) {
return 1; return 1;
} }
break; break;
case '0':
if (width == 0) {
zero_padding = true;
continue;
}
case '1':
case '2':
case '3':
case '4':
case '5':
case '6':
case '7':
case '8':
case '9':
width *= 10;
width += (*c - '0');
continue;
default: default:
//TODO manage error //TODO manage error
break; break;
@ -121,6 +141,8 @@ static uint32_t render_format(FormatCallback callback, void* arg,
switch (*c) { switch (*c) {
case '%': case '%':
in_format = true; in_format = true;
zero_padding = false;
width = 0;
continue; continue;
default: default:
if (callback(*c, arg)) return 1; if (callback(*c, arg)) return 1;
@ -135,23 +157,70 @@ static uint32_t render_format(FormatCallback callback, void* arg,
} }
static uint32_t render_signed(FormatCallback callback, void* arg, static uint32_t render_signed(FormatCallback callback, void* arg,
int32_t number, uint8_t base) int32_t number, uint8_t base, uint8_t width, bool zero_padding)
{ {
if (width > 0 && number < 0) {
--width;
}
if (width > 0) {
uint32_t abs_number;
if (number < 0) {
abs_number = -number;
} else {
abs_number = number;
}
uint32_t base_power = base;
while (width > 0 && base_power <= abs_number) {
base_power *= base;
--width;
}
if (width > 0) {
--width;
}
while (width > 0) {
--width;
if (zero_padding) {
if (callback('0', arg)) return 1;
} else {
if (callback(' ', arg)) return 1;
}
}
}
if (number < 0) { if (number < 0) {
callback('-', arg); callback('-', arg);
number = -number; number = -number;
} }
return render_unsigned(callback, arg, (uint32_t)number, base); return render_unsigned(callback, arg, (uint32_t)number, base, 0,
zero_padding);
} }
static uint32_t render_unsigned(FormatCallback callback, void* arg, static uint32_t render_unsigned(FormatCallback callback, void* arg,
uint32_t number, uint8_t base) uint32_t number, uint8_t base, uint8_t width, bool zero_padding)
{ {
uint32_t base_power = base; uint32_t base_power = base;
while (base_power <= number) { while (base_power <= number) {
base_power *= base; base_power *= base;
if (width > 0) {
--width;
}
}
if (width > 0) {
--width;
}
while (width > 0) {
--width;
if (zero_padding) {
if (callback('0', arg)) return 1;
} else {
if (callback(' ', arg)) return 1;
}
} }
while (base_power > 1) { while (base_power > 1) {