call printf using va_list
Asked Answered
T

5

74
void TestPrint(char* format, ...)
{
    va_list argList;

    va_start(argList, format);
    printf(format, argList);
    va_end(argList);
}


int main()
{
    TestPrint("Test print %s %d\n", "string", 55);
    return 0;
}

I need to get:

Test print string 55

Actually, I get garbage output. What is wrong in this code?

Thrombo answered 12/5, 2011 at 11:34 Comment(0)
K
91

Use vprintf() instead.

Kris answered 12/5, 2011 at 11:36 Comment(1)
@PingwinTux: Yup. This kind of response was acceptable back in 2011 ;)Kris
F
74

Instead of printf, I recommend you try vprintf instead, which was created for this specific purpose:

#include <stdio.h>
#include <stdlib.h>
#include <stdarg.h>

void errmsg( const char* format, ... )
{
    va_list arglist;

    printf( "Error: " );
    va_start( arglist, format );
    vprintf( format, arglist );
    va_end( arglist );
}

int main( void )
{
    errmsg( "%s %d %s", "Failed", 100, "times" );
    return EXIT_SUCCESS;
}

Source

Fellowman answered 12/5, 2011 at 11:39 Comment(0)
M
27

As others have pointed out already: In this case you should use vprintf instead.

But if you really want to wrap printf, or want to wrap a function that does not have a v... version, you can do that in GCC using the non-standard __builtin_apply feature:

int myfunction(char *fmt, ...)
{
    void *arg = __builtin_apply_args();
    void *ret = __builtin_apply((void*)printf, arg, 100);
    __builtin_return(ret);
}

The last argument to __builtin_apply is the max. total size of the arguments in bytes. Make sure that you use a value here that is large enough.

Myrwyn answered 21/5, 2014 at 17:13 Comment(2)
thanks for that answer, which gives a straight solution to OP's problem. For interception purpose, this code saved my life! (or at least avoided me to defer to assembly :)Affiliation
Can there EVER be a maximum for this value? The __builtin_apply() should parse the call stack frames in order to find out the value on its own. But it depends on the architecture if and how good this would work. I once grafted a similar solution for myself. It was horrible.Altdorfer
A
4

This is not how you use printf(). If you want to use va_lists, use vprintf() instead. Look here for reference.

Althing answered 12/5, 2011 at 11:37 Comment(0)
P
0

For anybody who needs more universal solution - try this template function:

template<typename... Args>
void myprintf(const char *format, Args... args)
{
    myObject.printf(format, args...);
}
Poussette answered 27/4, 2023 at 13:21 Comment(0)

© 2022 - 2025 — McMap. All rights reserved.