From 1c1a00cf51e5293e596e90d5c29a11c85197d996 Mon Sep 17 00:00:00 2001 From: Chris Lattner Date: Sun, 19 Apr 2009 20:29:42 +0000 Subject: [PATCH] move token paste poisoning diagnostics to after the instantiation loc for a token is set, this makes the diagnostic "expanded from stack" work for this diagnostic. Add a testcase for PR3918. llvm-svn: 69544 --- clang/lib/Lex/TokenLexer.cpp | 27 ++++++++++++++--------- clang/test/Preprocessor/macro_paste_bad.c | 13 +++++++++++ 2 files changed, 29 insertions(+), 11 deletions(-) diff --git a/clang/lib/Lex/TokenLexer.cpp b/clang/lib/Lex/TokenLexer.cpp index 8cec43322179..7ff473f090b4 100644 --- a/clang/lib/Lex/TokenLexer.cpp +++ b/clang/lib/Lex/TokenLexer.cpp @@ -308,13 +308,18 @@ void TokenLexer::Lex(Token &Tok) { // Get the next token to return. Tok = Tokens[CurToken++]; + bool TokenIsFromPaste = false; + // If this token is followed by a token paste (##) operator, paste the tokens! if (!isAtEnd() && Tokens[CurToken].is(tok::hashhash)) if (PasteTokens(Tok)) { // When handling the microsoft /##/ extension, the final token is // returned by PasteTokens, not the pasted token. return; + } else { + TokenIsFromPaste = true; } + // The token's current location indicate where the token was lexed from. We // need this information to compute the spelling of the token, but any @@ -342,6 +347,17 @@ void TokenLexer::Lex(Token &Tok) { // turning "for" into a keyword. Tok.setKind(II->getTokenID()); + // If this identifier was poisoned and from a paste, emit an error. This + // won't be handled by Preprocessor::HandleIdentifier because this is coming + // from a macro expansion. + if (II->isPoisoned() && TokenIsFromPaste) { + // We warn about __VA_ARGS__ with poisoning. + if (II->isStr("__VA_ARGS__")) + PP.Diag(Tok, diag::ext_pp_bad_vaargs_use); + else + PP.Diag(Tok, diag::err_pp_used_poisoned_id); + } + if (!DisableMacroExpansion && II->isHandleIdentifierCase()) PP.HandleIdentifier(Tok); } @@ -476,17 +492,6 @@ bool TokenLexer::PasteTokens(Token &Tok) { // by saying we're skipping contents, so we need to do this manually. IdentifierInfo *II = PP.LookUpIdentifierInfo(Tok, ResultTokStrPtr); Tok.setIdentifierInfo(II); - - // If this identifier was poisoned, emit an error. This won't be handled by - // Preprocessor::HandleIdentifier because this is coming from a macro - // expansion. - if (II->isPoisoned()) { - // We warn about __VA_ARGS__ with poisoning. - if (II->isStr("__VA_ARGS__")) - PP.Diag(Tok, diag::ext_pp_bad_vaargs_use); - else - PP.Diag(Tok, diag::err_pp_used_poisoned_id); - } } return false; } diff --git a/clang/test/Preprocessor/macro_paste_bad.c b/clang/test/Preprocessor/macro_paste_bad.c index f70c3b3ba2e6..b43d70b0c4e4 100644 --- a/clang/test/Preprocessor/macro_paste_bad.c +++ b/clang/test/Preprocessor/macro_paste_bad.c @@ -18,3 +18,16 @@ XYZ #define i ## // expected-error {{'##' cannot appear at start of macro expansion}} #define j() ## // expected-error {{'##' cannot appear at start of macro expansion}} +// Invalid token pasting. +// PR3918 + +// When pasting creates poisoned identifiers, we error. +#pragma GCC poison BLARG +BLARG // expected-error {{attempt to use a poisoned identifier}} +#define XX BL ## ARG +XX // expected-error {{attempt to use a poisoned identifier}} + +#define VA __VA_ ## ARGS__ +int VA; // expected-warning {{__VA_ARGS__ can only appear in the expansion of a C99 variadic macro}} + +