mirror of
https://github.com/Gericom/teak-llvm.git
synced 2025-06-19 19:45:40 -04:00

This patch implements an extension to the preprocessor: __VA_OPT__(contents) --> which expands into its contents if variadic arguments are supplied to the parent macro, or behaves as an empty token if none. - Currently this feature is only enabled for C++2a (this could be enabled, with some careful tweaks, for other dialects with the appropriate extension or compatibility warnings) - The patch was reviewed here: https://reviews.llvm.org/D35782 and asides from the above (and moving some of the definition and expansion recognition logic into the corresponding state machines), I believe I incorporated all of Richard's suggestions. A few technicalities (most of which were clarified through private correspondence between rsmith, hubert and thomas) are worth mentioning. Given: #define F(a,...) a #__VA_OPT__(a ## a) a ## __VA_OPT__(__VA_ARGS__) - The call F(,) Does not supply any tokens for the variadic arguments and hence VA_OPT behaves as a placeholder. - When expanding VA_OPT (for e.g. F(,1) token pasting occurs eagerly within its contents if the contents need to be stringified. - A hash or a hashhash prior to VA_OPT does not inhibit expansion of arguments if they are the first token within VA_OPT. - When a variadic argument is supplied, argument substitution occurs within the contents as does stringification - and these resulting tokens are inserted back into the macro expansions token stream just prior to the entire stream being rescanned and concatenated. See wg21.link/P0306 for further details on the feature. Acknowledgment: This patch would have been poorer if not for Richard Smith's usual thoughtful analysis and feedback. llvm-svn: 315840
65 lines
1.5 KiB
C++
65 lines
1.5 KiB
C++
// RUN: %clang_cc1 %s -Eonly -verify -Wno-all -pedantic -std=c++2a
|
|
|
|
//expected-error@+1{{missing '('}}
|
|
#define V1(...) __VA_OPT__
|
|
#undef V1
|
|
// OK
|
|
#define V1(...) __VA_OPT__ ()
|
|
#undef V1
|
|
|
|
//expected-warning@+1{{can only appear in the expansion of a variadic macro}}
|
|
#define V2() __VA_OPT__(x)
|
|
#undef V2
|
|
|
|
//expected-error@+2{{missing ')' after}}
|
|
//expected-note@+1{{to match this '('}}
|
|
#define V3(...) __VA_OPT__(
|
|
#undef V3
|
|
|
|
#define V4(...) __VA_OPT__(__VA_ARGS__)
|
|
#undef V4
|
|
|
|
//expected-error@+1{{nested}}
|
|
#define V5(...) __VA_OPT__(__VA_OPT__())
|
|
#undef V5
|
|
|
|
//expected-error@+1{{not followed by}}
|
|
#define V1(...) __VA_OPT__ (#)
|
|
#undef V1
|
|
|
|
//expected-error@+1{{cannot appear at start}}
|
|
#define V1(...) __VA_OPT__ (##)
|
|
#undef V1
|
|
|
|
//expected-error@+1{{cannot appear at start}}
|
|
#define V1(...) __VA_OPT__ (## X) x
|
|
#undef V1
|
|
|
|
//expected-error@+1{{cannot appear at end}}
|
|
#define V1(...) y __VA_OPT__ (X ##)
|
|
#undef V1
|
|
|
|
|
|
#define FOO(x,...) # __VA_OPT__(x) #x #__VA_OPT__(__VA_ARGS__) //OK
|
|
|
|
//expected-error@+1{{not followed by a macro parameter}}
|
|
#define V1(...) __VA_OPT__(#)
|
|
#undef V1
|
|
|
|
//expected-error@+1{{cannot appear at start}}
|
|
#define V1(...) a __VA_OPT__(##) b
|
|
#undef V1
|
|
|
|
//expected-error@+1{{cannot appear at start}}
|
|
#define V1(...) a __VA_OPT__(a ## b) b __VA_OPT__(##)
|
|
#undef V1
|
|
|
|
#define V1(x,...) # __VA_OPT__(b x) // OK
|
|
#undef V1
|
|
|
|
//expected-error@+2{{missing ')' after}}
|
|
//expected-note@+1{{to match this '('}}
|
|
#define V1(...) __VA_OPT__ ((())
|
|
#undef V1
|
|
|