diff --git a/include/frg/formatting.hpp b/include/frg/formatting.hpp index bdbbc1e..086b718 100644 --- a/include/frg/formatting.hpp +++ b/include/frg/formatting.hpp @@ -58,6 +58,7 @@ struct format_options { format_conversion conversion; int minimum_width = 0; int arg_pos = -1; + bool dollar_arg_pos = false; optional precision; bool left_justify = false; bool always_sign = false; diff --git a/include/frg/printf.hpp b/include/frg/printf.hpp index b1e801c..2d6f27d 100644 --- a/include/frg/printf.hpp +++ b/include/frg/printf.hpp @@ -62,9 +62,16 @@ T pop_arg(va_struct *vsp, format_options *opts) { if (opts->arg_pos == -1) return pop_va_arg(); - FRG_ASSERT(opts->arg_pos <= vsp->num_args); - if (opts->arg_pos < vsp->num_args) + if(opts->dollar_arg_pos) { + // we copy out all previous and the requested argument into our vsp->arg_list + for(int i = vsp->num_args; i <= opts->arg_pos; i++) { + auto arg = pop_va_arg(); + *get_union_member(i) = arg; + } + + vsp->num_args = opts->arg_pos + 1; return *get_union_member(opts->arg_pos); + } auto arg = pop_va_arg(); *get_union_member(vsp->num_args++) = arg; @@ -74,6 +81,7 @@ T pop_arg(va_struct *vsp, format_options *opts) { template frg::expected printf_format(A agent, const char *s, va_struct *vsp) { FRG_ASSERT(s != nullptr); + bool dollar_arg_pos = false; while(*s) { if(*s != '%') { @@ -100,9 +108,12 @@ frg::expected printf_format(A agent, const char *s, va_struct *vsp format_options opts; + opts.dollar_arg_pos = dollar_arg_pos; while(true) { if (*s >= '0' && *s <= '9' && s[1] && s[1] == '$') { opts.arg_pos = *s - '0' - 1; // args are 1-indexed + opts.dollar_arg_pos = true; + dollar_arg_pos = true; s += 2; FRG_ASSERT(*s); } else if(*s == '-') { diff --git a/tests/tests.cpp b/tests/tests.cpp index 73ea24f..b4ba2d6 100644 --- a/tests/tests.cpp +++ b/tests/tests.cpp @@ -428,6 +428,8 @@ TEST(formatting, printf) { do_test("hello", "%.5s", "hello world"); do_test("hello", "%.*s", 5, "hello world"); do_test("hello", "%.10s", "hello\0!!!!"); + + do_test("55 33", "%2$d %1$d", 33, 55); } #include