How, exactly, does the double-stringize trick work?
Asked Answered
W

2

94

At least some C preprocessors let you stringize the value of a macro, rather than its name, by passing it through one function-like macro to another that stringizes it:

#define STR1(x) #x
#define STR2(x) STR1(x)
#define THE_ANSWER 42
#define THE_ANSWER_STR STR2(THE_ANSWER) /* "42" */

Example use cases here.

This does work, at least in GCC and Clang (both with -std=c99), but I'm not sure how it works in C-standard terms.

Is this behavior guaranteed by C99?
If so, how does C99 guarantee it?
If not, at what point does the behavior go from C-defined to GCC-defined?

Weizmann answered 1/5, 2010 at 23:17 Comment(2)
If you say "at least some", does that mean you have seen one where it does not work? I'm willing to write a bug report to the vendor.Adlai
@Jens: No; I haven't. Every compiler I've used (namely, GCC and Clang) implements this behavior.Weizmann
T
84

Yes, it's guaranteed.

It works because arguments to macros are themselves macro-expanded, except where the macro argument name appears in the macro body with the stringifier # or the token-paster ##.

6.10.3.1/1:

... After the arguments for the invocation of a function-like macro have been identified, argument substitution takes place. A parameter in the replacement list, unless preceded by a # or ## preprocessing token or followed by a ## preprocessing token (see below), is replaced by the corresponding argument after all macros contained therein have been expanded...

So, if you do STR1(THE_ANSWER) then you get "THE_ANSWER", because the argument of STR1 is not macro-expanded. However, the argument of STR2 is macro-expanded when it's substituted into the definition of STR2, which therefore gives STR1 an argument of 42, with the result of "42".

Ternate answered 1/5, 2010 at 23:24 Comment(0)
A
22

As Steve notes, this is guarenteed, and it has been guarenteed since the C89 standard -- that was the standard the codified the # and ## operators in macros and mandates recursively expanding macros in args before substituting them into the body if and only if the body does not apply a # or ## to the argument. C99 is unchanged from C89 in this respect.

Ashly answered 1/5, 2010 at 23:47 Comment(1)
+1 for taking the time to confirm that it is part of C89 as well. Some of us care about portability, including making code that people compiling in C89 mode can use - just a few years ago gcc releases still defaulted to C89 (plus gcc extensions to be fair), MSVC last I heard still only supports C89, and embedded or legacy systems also sometimes have only C89 compilers. C89 makes for a great portability "ground floor", a least common denominator standard which people can target to compile and run for anything in practical use today. So it is very nice to see it remembered.Kazukokb

© 2022 - 2024 — McMap. All rights reserved.