Passing parameters dynamically to variadic functions

前端 未结 4 1853
余生分开走
余生分开走 2020-11-30 13:02

I was wondering if there was any way to pass parameters dynamically to variadic functions. i.e. If I have a function

int some_function (int a, int b, ...){/*         


        
相关标签:
4条回答
  • 2020-11-30 13:37

    A standard approach is to have each variadic function accompanied by a va_list-taking counterpart (as in printf and vprintf). The variadic version just converts ... to a va_list (using macros from stdarg.h) and calls its va_list-taking sister, which does actual work.

    0 讨论(0)
  • 2020-11-30 13:41

    Depending on what it is you're passing around, it could be a discriminated union you're after here (as hinted at in the comments). That would avoid the need for variadic functions or arrays of void*, and answers the question "how does some_function know what you actually passed it". You might have code something like this:

    enum thing_code { INTEGER, DOUBLE, LONG };
    
    struct thing
    {
     enum thing_code code;
     union
     {
        int a;
        double b;
        long c;
     };
    };
    
    void some_function(size_t n_things, struct thing *things)
    {
        /* ... for each thing ... */
        switch(things[i].code)
        {
          case INTEGER:
          /* ... */
        }
    }
    

    You can take this a step further and avoid the switch by replacing the code with one or more pointers to functions that do something useful with each thing. For example, if what you wanted to do was to simply print out each thing, you could have this:

    struct thing
    {
     void (*print)(struct thing*);
     union
     {
       ...
     };
    }
    
    void some_function(size_t n_things, struct thing *things)
    {
      /* .. for each thing .. */
      things[i]->print(things[i]);
      /* ... */
    }
    
    0 讨论(0)
  • 2020-11-30 13:42

    It might be interesting to try just passing an array, and then use the vararg macros anyway. Depending on stack alignment, it might Just Work (tm).

    This is probably not an optimal solution, I mainly posted it because I found the idea interesting. After trying it out, this approach worked on my linux x86, but not on x86-64 - it can probably be improved. This method will depend on stack alignment, struct alignment and probably more.

    void varprint(int count, ...)
    {
        va_list ap;
        int32_t i;
    
        va_start(ap, count);
        while(count-- ) {
            i = va_arg(ap, int32_t);
            printf("Argument: %d\n", i);
        }
        va_end(ap); 
    }
    
    struct intstack
    {
        int32_t pos[99];
    };
    
    int main(int argc, char** argv)
    {
        struct intstack *args = malloc(sizeof(struct intstack));
        args->pos[0] = 1;
        args->pos[1] = 2;
        args->pos[2] = 3;
        args->pos[3] = 4;
        args->pos[4] = 5;
    
        varprint(5, *args);
        return 0;
    }
    
    0 讨论(0)
  • 2020-11-30 13:48

    Variadic functions use a calling convention where the caller is responsible for popping the function parameters from the stack, so yes, it is possible to do this dynamically. It's not standardized in C, and normally would require some assembly to manually push the desired parameters, and invoke the variadic function correctly.

    The cdecl calling convention requires that the arguments be pushed in the correct order, and after the call, the bytes pushed as arguments before the call are popped. In this way, the called function can receive an arbitrary number of parameters, as the caller will handle reverting the stack pointer to it's pre-call state. The space occupied by the arguments before the ... is the safe lower bound for number of bytes pushed. Additional variadic arguments are interpreted at runtime.

    FFCALL is a library which provides wrappers for passing parameters dynamically to variadic functions. The group of functions you're interested in is avcall. Here's an example calling the functions you gave above:

    #include <avcall.h>
    
    av_alist argList;
    int retVal;
    av_start_int(argList, some_function, retval);
    av_int(argList, a);
    av_int(argList, b);
    av_type(argList, val1);
    ...
    av_type(argList, valn);
    av_call(argList);
    

    You might also find this link discussing generating wrappers around variadic functions in C, to be of interest in justifying why this isn't part of standard C.

    0 讨论(0)
提交回复
热议问题