C abuse of the week [entries|reading|network|archive]
simont

[ userinfo | dreamwidth userinfo ]
[ archive | journal archive ]

Tue 2012-02-21 11:53
C abuse of the week

I continue to be amazed at the number of bizarre things you can arrange to do using the C preprocessor, the switch statement, and a strong stomach. I've previously used the combination to implement coroutines, of course, and also a modified version of for which performs its test after the loop body rather than before it.

Chris mentioned to me this morning the fact that you could write this sort of thing:

    switch (get_value()) case 1: case 2: case 3: do_stuff();

which has the handy effect of calling get_value() only once and then testing its return value against several possibilities (though they're constrained to be compile-time constant integers). But of course it's hopelessly ugly, so he challenged me to wrap it up in a macro or two. (Not that that stops it being ugly, exactly. But differently ugly. :-)

The obvious solution seemed to me to be this:

#define IF_IN_LIST(expr,list) switch (expr) case list:
#define OR : case

which then lets you write compound statements that only look slightly un-C-ish:

    IF_IN_LIST(get_value(), 1 OR 2 OR 3)
do_stuff();

But then I realised that if you dangle a few more oddities off the switch statement, you can do better. If you do it like this:

#define IF_IN_LIST(expr,list) switch (expr) default: if (0) case list:
#define OR : case

then your uses of the macro can now include an optional else clause!

    IF_IN_LIST(get_value(), 1 OR 2 OR 3)
do_stuff();
else
do_other_stuff();

(And naturally both versions work with either bare statements or braced blocks.)

[xpost |http://simont.livejournal.com/235471.html]

LinkReply
[identity profile] ptc24.livejournal.comWed 2012-02-22 15:42
That's... quite special.

Your use of if(0) is rather better than the tawdry use I sometimes make of the Java equivalent, if(false) - the Java compiler complains of unreachable code, and sometimes when you're working on something, it's nice to suppress that with an if(false) return;. Of course, Java is boring and won't let you jump into blocks with a case: like that...
Link Reply to this | Thread
[personal profile] simontWed 2012-02-22 15:46
I've seen at least one C compiler get confused by jumps into the middle of an if(0) clause, because it had dead-code-eliminated it before noticing the case label. (PuTTY used to contain some code which did that. The compiler developer sent me an embarrassed email thanking me for exposing the compiler bug, which was now fixed.)

I didn't know it was actually illegal in Java, though! Or do you just mean that Java doesn't let you put case statements in sub-blocks in general rather than that it specifically disallows them in otherwise unreachable ones?
Link Reply to this | Parent | Thread
[identity profile] ptc24.livejournal.comWed 2012-02-22 15:50
The general case: it doesn't matter whether the code is reachable or not, you can't put case statements in sub-blocks. Not even sub-blocks that are just sub-blocks on their own (i.e. no if or for or while or whatever in front of them, they just start with {).
Link Reply to this | Parent
[identity profile] xraycb.livejournal.comSat 2012-02-25 12:30
I like this, almost enough that I might be tempted to use it in production code...
Link Reply to this
navigation
[ go | Previous Entry | Next Entry ]
[ add | to Memories ]