dynamic construction of va_list

P

pete m

I would like to support stdio functions for an extremely primitive
type system, as shown in the attached sample program, using dynamic
creation of a va_list.

I've tested it on successfully a couple platforms, but what I don't
know is whether it's actually a legal program.

The function of interest is usnprintf, "union sprintf". (psnprintf is
a dummy wrapper for vsnprintf, useful for debugging.)


The signature is

int usnprintf (char *buffer, size_t size, const char *fmt, const
type_union *arg);

where the last argument is a flag-terminated array of type_union, as
defined below.



================Code begins =========================
#include <stdio.h>
#include <stdarg.h>
#include <assert.h>

typedef struct type_union type_union;

typedef enum type_val {
T_END = 0, /* terminator flag for variable-length array */
T_INT,
T_FLOAT,
T_CHAR,
T_STRING,
} type_val;

struct type_union
{
type_val t;
union {
float f;
int i;
char c;
const char *s;
};
};

/* peek into va_list offsets */
int psnprintf(char *buf, size_t s, const char *fmt, va_list va)
{
int i;
int r;
va_list vb = va;
for(i = 0; i < 2; i++) {
double t = va_arg(va, double);
}
r = vsnprintf(buf, s, fmt, vb);
return r;
}


/*
* Treat ac_list, ac_arg as lvalues.
*/

#define va_start_assign(list, var) ((list) = ((char*)&var))

#define va_assign(lvalp, type, rval) \
*((type*)lvalp) = rval; \
va_arg(lvalp, type);

#define VA_FLOAT double
#define VA_CHAR int

/*
* snprintf for typed unions. (Does not validate the parse.)
* Requires flag-terminated array of values.
* TODO: validate with parse_printf_format and eliminate terminator.
*/
int usnprintf (char *buffer, size_t size, const char *fmt, const
type_union *arg)
{
int i;
int ret;
va_list out;
double aligned[100]; /* aligned buffer of ridiculously large size */
va_start_assign(out, aligned[0]);
for(i = 0; arg.t != T_END; i++) {
switch(arg.t) {
case T_INT:
va_assign(out, int, arg.i);
break;
case T_FLOAT:
va_assign(out, VA_FLOAT, arg.f);
break;
case T_CHAR:
va_assign(out, VA_CHAR, arg.c);
break;
case T_STRING:
va_assign(out, const char *, arg.s);
break;
default:
assert(0);
va_end(out);
return -1;
}
}

va_end(out);
va_start_assign(out, aligned[0]);
ret = psnprintf(buffer, size, fmt, out);
va_end(out);
return ret;
}

#define TYPE_FUN(v2u, tv, T, v) \
type_union v2u(T v) { \
type_union r; \
r.v = v; \
r.t = tv; \
return r; \
}

TYPE_FUN(i2u, T_INT, int, i)
TYPE_FUN(c2u, T_CHAR, char, c)
TYPE_FUN(f2u, T_FLOAT, float, f)
TYPE_FUN(s2u, T_STRING, const char *, s)

static const type_union END = { T_END };


int main() {
char buffer[30];
type_union x[] = { f2u(2.5), f2u(3.5), i2u(5), c2u('a'), s2u("xyz
\n"), END };
usnprintf(buffer, 30, "%.1f; %.1f; %d; %c; %s", x);
printf("%s", buffer);
}
 
C

Chris Torek

(I am not sure why no one ever replied to this earlier, but I have
been saving it and have finally gotten around to replying myself.)

I would like to support stdio functions for an extremely primitive
type system, as shown in the attached sample program, using dynamic
creation of a va_list.

This cannot be done portably.

(It *can* be done *non*-portably, once for each target machine.
In some cases the technique may require compiler support and/or
sneaky assembly code. To get the "right" sizes, rather than using
a large array and simply hoping it is big enough -- as you did in
some code I snipped below -- may require a two-pass approach.)

[much snippage -- retaining only the va_assign related code]
#define va_start_assign(list, var) ((list) = ((char*)&var))

#define va_assign(lvalp, type, rval) \
*((type*)lvalp) = rval; \
va_arg(lvalp, type);

#define VA_FLOAT double
#define VA_CHAR int ...
va_list out;
double aligned[100]; /* aligned buffer of ridiculously large size */
va_start_assign(out, aligned[0]);
for(i = 0; arg.t != T_END; i++) {
switch(arg.t) {
case T_INT:
va_assign(out, int, arg.i);
break;
case T_FLOAT:
va_assign(out, VA_FLOAT, arg.f);
break;
case T_CHAR:
va_assign(out, VA_CHAR, arg.c);
break;
case T_STRING:
va_assign(out, const char *, arg.s);
break;
default:
assert(0);
va_end(out);
return -1;
}
}

va_end(out);
va_start_assign(out, aligned[0]);


("out" is then used as if it were an ordinary "va_list" parameter
created by va_start().)

This is likely to work on a typical stack-based system, such as
normally used on x86 implementations. It may even work on some
systems that pass arguments in registers. It will, however, fail
on some other systems that pass arguments in registers. For
instance, on V9 SPARC, "out" needs to point to a structure
object, where the structure contains two counters and three
pointers. The two counters indicate how many integral and
floating-point parameters remain unprocessed in the "save areas"
to which the first two pointers point. The final pointer points
to the "overflow area" for any additional parameters.

Hence, to "va_assign" (as it were) a VA_FLOAT, you would (at least
nominally) want to copy the "double" to the floating-point save
area and increment the floating-point count. In practice, you
could "cheat" and simply set the two counters to zero, then dump
all the remaining arguments into the "pseudo-stack" as you do above:

#define va_start_assign(ptr, save_area) \
(((struct __va_args *)&(save_area))->__va_ni = 0, \
((struct __va_args *)&(save_area))->__va_nf = 0, \
((struct __va_args *)&(save_area))->__va_rest = \
(void *)((struct va_args *)&(save_area) + 1), \
(ptr) = (struct __va_args *)&(save_area))

This method is of course quite specific to the SPARC-V9 ABI (and
makes some assumptions about the compiler's internal names for
various pieces involved in <stdarg.h>; and I may not recall them
quite correctly, and/or they may depend on specific compiler
revisions).
 

Ask a Question

Want to reply to this thread or ask your own question?

You'll need to choose a username for the site, which only take a couple of moments. After that, you can post your question and our members will help you out.

Ask a Question

Members online

Forum statistics

Threads
473,982
Messages
2,570,185
Members
46,736
Latest member
AdolphBig6

Latest Threads

Top