diff options
| author | Chris Lattner <sabre@nondot.org> | 2009-01-16 18:59:23 +0000 | 
|---|---|---|
| committer | Chris Lattner <sabre@nondot.org> | 2009-01-16 18:59:23 +0000 | 
| commit | 262d4e31b99e778714d3f83202bbf0e45bd7a0cb (patch) | |
| tree | 523aea96d79490150210d03c28d86f12152a7d46 | |
| parent | 8a24e588d73f9890c7acb100e19d8cdc44c8e929 (diff) | |
| download | bcm5719-llvm-262d4e31b99e778714d3f83202bbf0e45bd7a0cb.tar.gz bcm5719-llvm-262d4e31b99e778714d3f83202bbf0e45bd7a0cb.zip | |
Improve #pragma comment support by building the string argument and
notifying PPCallbacks about it.
llvm-svn: 62333
| -rw-r--r-- | clang/include/clang/Lex/PPCallbacks.h | 8 | ||||
| -rw-r--r-- | clang/lib/Lex/Pragma.cpp | 47 | 
2 files changed, 44 insertions, 11 deletions
| diff --git a/clang/include/clang/Lex/PPCallbacks.h b/clang/include/clang/Lex/PPCallbacks.h index 792c6c41f7c..bf03bb81f22 100644 --- a/clang/include/clang/Lex/PPCallbacks.h +++ b/clang/include/clang/Lex/PPCallbacks.h @@ -20,6 +20,7 @@  namespace clang {    class SourceLocation; +  class IdentifierInfo;  /// PPCallbacks - This interface provides a way to observe the actions of the  /// preprocessor as it does its thing.  Clients can define their hooks here to @@ -46,6 +47,13 @@ public:    virtual void Ident(SourceLocation Loc, const std::string &str) {    } +  /// PragmaComment - This callback is invoked when a #pragma comment directive +  /// is read. +  /// +  virtual void PragmaComment(SourceLocation Loc, const IdentifierInfo *Kind,  +                             const std::string &Str) { +  } +    };  }  // end namespace clang diff --git a/clang/lib/Lex/Pragma.cpp b/clang/lib/Lex/Pragma.cpp index b19dc2009d9..60877743531 100644 --- a/clang/lib/Lex/Pragma.cpp +++ b/clang/lib/Lex/Pragma.cpp @@ -14,6 +14,7 @@  #include "clang/Lex/Pragma.h"  #include "clang/Lex/HeaderSearch.h" +#include "clang/Lex/LiteralSupport.h"  #include "clang/Lex/Preprocessor.h"  #include "clang/Basic/Diagnostic.h"  #include "clang/Basic/FileManager.h" @@ -125,7 +126,11 @@ void Preprocessor::Handle_Pragma(Token &Tok) {      return;    } -  // The _Pragma is lexically sound.  Destringize according to C99 6.10.9.1. +  // The _Pragma is lexically sound.  Destringize according to C99 6.10.9.1: +  // "The string literal is destringized by deleting the L prefix, if present, +  // deleting the leading and trailing double-quotes, replacing each escape +  // sequence \" by a double-quote, and replacing each escape sequence \\ by a +  // single backslash."    if (StrVal[0] == 'L')  // Remove L prefix.      StrVal.erase(StrVal.begin());    assert(StrVal[0] == '"' && StrVal[StrVal.size()-1] == '"' && @@ -342,16 +347,11 @@ void Preprocessor::HandlePragmaComment(Token &Tok) {      return;    } -  // Check for optional string. -  // FIXME: If the kind is "compiler" warn if the string is present (it is -  // ignored). -  // FIXME: 'lib' requires a comment string. -  // FIXME: 'linker' requires a comment string, and has a specific list of -  // things that are allowable. +  // Read the optional string if present.    Lex(Tok); +  std::string ArgumentString;    if (Tok.is(tok::comma)) { -    // FIXME: for now, we parse but ignore the string. -    Lex(Tok); +    Lex(Tok); // eat the comma.      // We need at least one string.      if (Tok.getKind() != tok::string_literal) { @@ -362,20 +362,45 @@ void Preprocessor::HandlePragmaComment(Token &Tok) {      // String concatenation allows multiple strings, which can even come from      // macro expansion.      // "foo " "bar" "Baz" -    while (Tok.getKind() == tok::string_literal) +    llvm::SmallVector<Token, 4> StrToks; +    while (Tok.getKind() == tok::string_literal) { +      StrToks.push_back(Tok);        Lex(Tok); +    } + +    // Concatenate and parse the strings. +    StringLiteralParser Literal(&StrToks[0], StrToks.size(), *this); +    assert(!Literal.AnyWide && "Didn't allow wide strings in"); +    if (Literal.hadError) +      return; +    if (Literal.Pascal) { +      Diag(StrToks[0].getLocation(), diag::err_pragma_comment_malformed); +      return; +    } + +    ArgumentString = std::string(Literal.GetString(), +                                 Literal.GetString()+Literal.GetStringLength());    } +  // FIXME: If the kind is "compiler" warn if the string is present (it is +  // ignored). +  // FIXME: 'lib' requires a comment string. +  // FIXME: 'linker' requires a comment string, and has a specific list of +  // things that are allowable. +      if (Tok.isNot(tok::r_paren)) {      Diag(Tok.getLocation(), diag::err_pragma_comment_malformed);      return;    } -  Lex(Tok); +  Lex(Tok);  // eat the r_paren.    if (Tok.isNot(tok::eom)) {      Diag(Tok.getLocation(), diag::err_pragma_comment_malformed);      return;    } +   +  // If the pragma is lexically sound, notify any interested PPCallbacks. +  Callbacks->PragmaComment(CommentLoc, II, ArgumentString);  } | 

