Preprocessor functions evaluated at compile time in C
I want to write preprocessor functions/arrays that are evaluated at compile time. For example, if I define
#define MYARR[] {5,4,3,2,1,0}
then, the code
int x = R[0];
should be presented as
int x = 5;
to the compiler. (Of course only literals can be used in the index).
This is important if code size/mem开发者_运维百科ory is critical and we don't want to
store MYARR, but we need it for coding convenience.
Compile time functions would also be good. For example, something like
#define MYMAP(n)
#if n==1
5
#else
2
So, the statement
int x = MYMAP(4);
should be presented to the compiler as
int x = 2;
Obviously, we have to use a literal as the argument. Is this possible?
Sure its possible. While you could do that manually, Boost.Preprocessor already gives you the tools you need:
#include <boost/preprocessor.hpp>
#define SEQ (5)(4)(3)(2)(1)(0)
int x = BOOST_PP_SEQ_ELEM(0, SEQ);
... gets transformed to:
int x = 5;
It also includes arithmetic, comparisons and control structures like IF, FOR, FOR_EACH, enumerations, ... You just have to keep in mind that the data types you can work with are rather limited. 
Utilizing Boost.PP again, your second example can be done like so:
#define MYMAP(x) BOOST_PP_IF(BOOST_PP_EQUAL(x, 1), 5, 2)
You can of course implement manually what Boost.PP does, but considering the time and effort needed for this i personally wouldn't bother.
As a C user, you won't be interested in the rest of Boost, so you might want to use bcp to extract the preprocessor components.
The standard C preprocessor will not do what you want. To get this sort of behavior reliably you are going to need a more powerful, nonstandard preprocessing tool. However, I'm not familiar enough with what's available to direct you to which one you might want.
Although, in the second case, you may still be able to get the effect you want on most modern compilers. For example:
#define MYMAP(n) (((n) == 1) ? 5 : 2)
int x = MYMAP(4);
will still be presented to the compiler as:
int x = ((4 == 1) ? 5 : 2);
but the compiler itself, if it is a reasonable modern compiler and you allow it to optimize, will likely notice that this expression can be evaluated at compile-time and will emit code that is identical to
int x = 2;
However, there's nothing that guarantees that a compiler will perform this optimization.
In C99 (you really need that, C89 wouldn't do) you can do with something like
#define MyMap(N) (((int const[]){ 3, 4, 5, 6, 7})[N])
provided your type is int, but any other type would do. The weird thing (int const[]){ 3, 4, 5, 6, 7} is called a compound literal. The const in there for the base type tells the compiler that it will not be modified and that he may alias all occurrences with the same contents to the same fixed location.
Edit, after caf's remark:
Generally, for this approach most compilers will be able to completely optimize any reference to the array away, provided N is an expression that can be evaluated at compile time, such as a fixed value 7 or 'a' or so.
If this is not the case, the compiler has to create the array object somewhere. With the const he could be allowed to generate just one copy of it, regardless how often you call the macro in your code. When he manages to do so, the initialization of the array would be done at compile time, so there would be no compile time overhead.
I checked this for the three compilers that I have on my machine:
- gccand- clangallocate the array on the stack and this for every single call to- MyMap, bad, since the overhead is proportional to the size of the array.
- openccallocates the array statically but creates a new copy for every call to- MyMap, better but not yet ideal.
myarr for chars I can do
#define MYARR(x) ("1234567"[x])
still pondering ints. Mymap is
#define MYMAP(n) (n == 1 ? 5 : 2)
There is no standard preprocessor commands that handle arrays the way you are wanting. I suggest that you use a const.
static const int R[] = {5,4,3,2,1,0};
...
int a = R[0];
Then for the MYMAP question...
#define MYMAP(n) ((n)==1?5:2)
...
int a = MYMAP(1); // equals 5.
 
         加载中,请稍侯......
 加载中,请稍侯......
      
精彩评论