Preprocessor macro value to Objective-C string literal
Asked Answered
F

5

28

I have a preprocessor macro defined in build settings

FOO=BAR

That value I want to massage into an Objective-C string literal that can be passed to a method. The following #define does not work, but it should demonstrate what I am trying to achieve:

 #define FOOLITERAL @"FOO" //want FOOLITERAL to have the value of @"BAR"

 myMethodThatTakesAnNSString(FOOLITERAL);

I expect that I am just missing the obvious somehow, but I cannot seem to find the right preprocessor voodoo to get what I need.

Flickinger answered 30/9, 2011 at 4:8 Comment(0)
P
36

Use the stringizing operator # to make a C string out of the symbol. However, due to a quirk of the preprocessor, you need to use two extra layers of macros:

#define FOO BAR
#define STRINGIZE(x) #x
#define STRINGIZE2(x) STRINGIZE(x)
#define FOOLITERAL @ STRINGIZE2(FOO)
// FOOLITERAL now expands to @"BAR" 

The reason for the extra layers is that the stringizing operator can only be used on the arguments of the macro, not on other tokens. Secondly, if an argument of a macro has the stringizing operator applied to it in the body of the macro, then that argument is not expanded as another macro. So, to ensure that FOO gets expanded, we wrap in another macro, so that when STRINGIZE2 gets expanded, it also expands FOO because the stringizing operator does not appear in that macro's body.

Plexiglas answered 30/9, 2011 at 4:19 Comment(5)
That did it. I had tried the double indirection, but also had the "@" as a macro value and that seems to confuse it. One small difference is that the first line does not represent my case in that it does not retrieve the build setting #define FOOVAL FOO #define STRINGIZE(x) #x #define STRINGIZE2(x) STRINGIZE(x) #define FOOLITERAL @ STRINGIZE2(FOOVAL) is what actually works for me. But your answer got me there.Flickinger
Hi @vagrant. Does this solution still work for you with LLVM? I'm trying to get this to work, but the actual value of FOO as defined in my build settings preprocessor macros never shows up. Could you maybe update your question with the solution that works for you? Cheers.Sulfamerazine
Why not just #define Helper(STR) @ #STR ?Lakendra
This is correct, but @Noa's answer below is much easier to use IMHOWivern
Doesn't seem to work if FOO has commas in its valueKerwinn
O
29

Here's a modified version of Adam Rosenfield's answer with clearer semantics:

#define NSStringize_helper(x) #x
#define NSStringize(x) @NSStringize_helper(x)

I use it to replace code like this:

case OneEnumValue: name = @"OneEnumValue"; break;
case AnotherEnumValue: name = @"AnotherEnumValue"; break;

with this:

#define case_for_type(type) case type: name = NSStringize(type); break

case_for_type(OneEnumValue);
case_for_type(AnotherEnumValue);
Olecranon answered 4/2, 2013 at 1:9 Comment(0)
E
4

A convenient macro OS_STRINGIFY(s) is predefined in SDK's /usr/include/os/base.h, which is included in Foundation.framework.

So you can use the following code without additional definitions.

myMethodThatTakesAnNSString(@OS_STRINGIFY(FOO));
Estradiol answered 25/4, 2021 at 6:51 Comment(1)
Great find. Essentially the solution of the accepted answer, but can be used consistently, without cluttering up application code with something that already exists in Apple's frameworks: opensource.apple.com/source/xnu/xnu-4903.241.1/libkern/os/…Huebner
U
0

You need to define preprocessor macro like,

FOO=\@\"BAR\"

And use code side like,

[NSString stringWithFormat:@"macro: %@", FOO];
Uptown answered 17/8, 2017 at 11:31 Comment(0)
D
-2

What error are you seeing exactly? This type of thing does work as you expect:

#define kMyString @"MyString"

[NSString stringWithFormat:@"macro: %@", kMyString];
Dilly answered 30/9, 2011 at 4:12 Comment(1)
The actual value of the string needed is the value of the preprocessor macro. So I can't just put the value into the code like in your example, it has to come from the FOO variable.Flickinger

© 2022 - 2024 — McMap. All rights reserved.