932 lines
		
	
	
		
			29 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			932 lines
		
	
	
		
			29 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| //===--- ParsePragma.cpp - Language specific pragma parsing ---------------===//
 | |
| //
 | |
| //                     The LLVM Compiler Infrastructure
 | |
| //
 | |
| // This file is distributed under the University of Illinois Open Source
 | |
| // License. See LICENSE.TXT for details.
 | |
| //
 | |
| //===----------------------------------------------------------------------===//
 | |
| //
 | |
| // This file implements the language specific #pragma handlers.
 | |
| //
 | |
| //===----------------------------------------------------------------------===//
 | |
| 
 | |
| #include "ParsePragma.h"
 | |
| #include "clang/Lex/Preprocessor.h"
 | |
| #include "clang/Parse/ParseDiagnostic.h"
 | |
| #include "clang/Parse/Parser.h"
 | |
| #include "clang/Sema/Scope.h"
 | |
| #include "llvm/ADT/StringSwitch.h"
 | |
| using namespace clang;
 | |
| 
 | |
| /// \brief Handle the annotation token produced for #pragma unused(...)
 | |
| ///
 | |
| /// Each annot_pragma_unused is followed by the argument token so e.g.
 | |
| /// "#pragma unused(x,y)" becomes:
 | |
| /// annot_pragma_unused 'x' annot_pragma_unused 'y'
 | |
| void Parser::HandlePragmaUnused() {
 | |
|   assert(Tok.is(tok::annot_pragma_unused));
 | |
|   SourceLocation UnusedLoc = ConsumeToken();
 | |
|   Actions.ActOnPragmaUnused(Tok, getCurScope(), UnusedLoc);
 | |
|   ConsumeToken(); // The argument token.
 | |
| }
 | |
| 
 | |
| void Parser::HandlePragmaVisibility() {
 | |
|   assert(Tok.is(tok::annot_pragma_vis));
 | |
|   const IdentifierInfo *VisType =
 | |
|     static_cast<IdentifierInfo *>(Tok.getAnnotationValue());
 | |
|   SourceLocation VisLoc = ConsumeToken();
 | |
|   Actions.ActOnPragmaVisibility(VisType, VisLoc);
 | |
| }
 | |
| 
 | |
| struct PragmaPackInfo {
 | |
|   Sema::PragmaPackKind Kind;
 | |
|   IdentifierInfo *Name;
 | |
|   Token Alignment;
 | |
|   SourceLocation LParenLoc;
 | |
|   SourceLocation RParenLoc;
 | |
| };
 | |
| 
 | |
| void Parser::HandlePragmaPack() {
 | |
|   assert(Tok.is(tok::annot_pragma_pack));
 | |
|   PragmaPackInfo *Info =
 | |
|     static_cast<PragmaPackInfo *>(Tok.getAnnotationValue());
 | |
|   SourceLocation PragmaLoc = ConsumeToken();
 | |
|   ExprResult Alignment;
 | |
|   if (Info->Alignment.is(tok::numeric_constant)) {
 | |
|     Alignment = Actions.ActOnNumericConstant(Info->Alignment);
 | |
|     if (Alignment.isInvalid())
 | |
|       return;
 | |
|   }
 | |
|   Actions.ActOnPragmaPack(Info->Kind, Info->Name, Alignment.get(), PragmaLoc,
 | |
|                           Info->LParenLoc, Info->RParenLoc);
 | |
| }
 | |
| 
 | |
| void Parser::HandlePragmaMSStruct() {
 | |
|   assert(Tok.is(tok::annot_pragma_msstruct));
 | |
|   Sema::PragmaMSStructKind Kind =
 | |
|     static_cast<Sema::PragmaMSStructKind>(
 | |
|     reinterpret_cast<uintptr_t>(Tok.getAnnotationValue()));
 | |
|   Actions.ActOnPragmaMSStruct(Kind);
 | |
|   ConsumeToken(); // The annotation token.
 | |
| }
 | |
| 
 | |
| void Parser::HandlePragmaAlign() {
 | |
|   assert(Tok.is(tok::annot_pragma_align));
 | |
|   Sema::PragmaOptionsAlignKind Kind =
 | |
|     static_cast<Sema::PragmaOptionsAlignKind>(
 | |
|     reinterpret_cast<uintptr_t>(Tok.getAnnotationValue()));
 | |
|   SourceLocation PragmaLoc = ConsumeToken();
 | |
|   Actions.ActOnPragmaOptionsAlign(Kind, PragmaLoc);
 | |
| }
 | |
| 
 | |
| void Parser::HandlePragmaWeak() {
 | |
|   assert(Tok.is(tok::annot_pragma_weak));
 | |
|   SourceLocation PragmaLoc = ConsumeToken();
 | |
|   Actions.ActOnPragmaWeakID(Tok.getIdentifierInfo(), PragmaLoc,
 | |
|                             Tok.getLocation());
 | |
|   ConsumeToken(); // The weak name.
 | |
| }
 | |
| 
 | |
| void Parser::HandlePragmaWeakAlias() {
 | |
|   assert(Tok.is(tok::annot_pragma_weakalias));
 | |
|   SourceLocation PragmaLoc = ConsumeToken();
 | |
|   IdentifierInfo *WeakName = Tok.getIdentifierInfo();
 | |
|   SourceLocation WeakNameLoc = Tok.getLocation();
 | |
|   ConsumeToken();
 | |
|   IdentifierInfo *AliasName = Tok.getIdentifierInfo();
 | |
|   SourceLocation AliasNameLoc = Tok.getLocation();
 | |
|   ConsumeToken();
 | |
|   Actions.ActOnPragmaWeakAlias(WeakName, AliasName, PragmaLoc,
 | |
|                                WeakNameLoc, AliasNameLoc);
 | |
| 
 | |
| }
 | |
| 
 | |
| void Parser::HandlePragmaRedefineExtname() {
 | |
|   assert(Tok.is(tok::annot_pragma_redefine_extname));
 | |
|   SourceLocation RedefLoc = ConsumeToken();
 | |
|   IdentifierInfo *RedefName = Tok.getIdentifierInfo();
 | |
|   SourceLocation RedefNameLoc = Tok.getLocation();
 | |
|   ConsumeToken();
 | |
|   IdentifierInfo *AliasName = Tok.getIdentifierInfo();
 | |
|   SourceLocation AliasNameLoc = Tok.getLocation();
 | |
|   ConsumeToken();
 | |
|   Actions.ActOnPragmaRedefineExtname(RedefName, AliasName, RedefLoc,
 | |
|                                      RedefNameLoc, AliasNameLoc);
 | |
| }
 | |
| 
 | |
| void Parser::HandlePragmaFPContract() {
 | |
|   assert(Tok.is(tok::annot_pragma_fp_contract));
 | |
|   tok::OnOffSwitch OOS =
 | |
|     static_cast<tok::OnOffSwitch>(
 | |
|     reinterpret_cast<uintptr_t>(Tok.getAnnotationValue()));
 | |
|   Actions.ActOnPragmaFPContract(OOS);
 | |
|   ConsumeToken(); // The annotation token.
 | |
| }
 | |
| 
 | |
| StmtResult Parser::HandlePragmaCaptured()
 | |
| {
 | |
|   assert(Tok.is(tok::annot_pragma_captured));
 | |
|   ConsumeToken();
 | |
| 
 | |
|   if (Tok.isNot(tok::l_brace)) {
 | |
|     PP.Diag(Tok, diag::err_expected_lbrace);
 | |
|     return StmtError();
 | |
|   }
 | |
| 
 | |
|   SourceLocation Loc = Tok.getLocation();
 | |
| 
 | |
|   ParseScope CapturedRegionScope(this, Scope::FnScope | Scope::DeclScope);
 | |
|   Actions.ActOnCapturedRegionStart(Loc, getCurScope(), CR_Default,
 | |
|                                    /*NumParams=*/1);
 | |
| 
 | |
|   StmtResult R = ParseCompoundStatement();
 | |
|   CapturedRegionScope.Exit();
 | |
| 
 | |
|   if (R.isInvalid()) {
 | |
|     Actions.ActOnCapturedRegionError();
 | |
|     return StmtError();
 | |
|   }
 | |
| 
 | |
|   return Actions.ActOnCapturedRegionEnd(R.get());
 | |
| }
 | |
| 
 | |
| namespace {
 | |
|   typedef llvm::PointerIntPair<IdentifierInfo *, 1, bool> OpenCLExtData;
 | |
| }
 | |
| 
 | |
| void Parser::HandlePragmaOpenCLExtension() {
 | |
|   assert(Tok.is(tok::annot_pragma_opencl_extension));
 | |
|   OpenCLExtData data =
 | |
|       OpenCLExtData::getFromOpaqueValue(Tok.getAnnotationValue());
 | |
|   unsigned state = data.getInt();
 | |
|   IdentifierInfo *ename = data.getPointer();
 | |
|   SourceLocation NameLoc = Tok.getLocation();
 | |
|   ConsumeToken(); // The annotation token.
 | |
| 
 | |
|   OpenCLOptions &f = Actions.getOpenCLOptions();
 | |
|   // OpenCL 1.1 9.1: "The all variant sets the behavior for all extensions,
 | |
|   // overriding all previously issued extension directives, but only if the
 | |
|   // behavior is set to disable."
 | |
|   if (state == 0 && ename->isStr("all")) {
 | |
| #define OPENCLEXT(nm)   f.nm = 0;
 | |
| #include "clang/Basic/OpenCLExtensions.def"
 | |
|   }
 | |
| #define OPENCLEXT(nm) else if (ename->isStr(#nm)) { f.nm = state; }
 | |
| #include "clang/Basic/OpenCLExtensions.def"
 | |
|   else {
 | |
|     PP.Diag(NameLoc, diag::warn_pragma_unknown_extension) << ename;
 | |
|     return;
 | |
|   }
 | |
| }
 | |
| 
 | |
| 
 | |
| 
 | |
| // #pragma GCC visibility comes in two variants:
 | |
| //   'push' '(' [visibility] ')'
 | |
| //   'pop'
 | |
| void PragmaGCCVisibilityHandler::HandlePragma(Preprocessor &PP, 
 | |
|                                               PragmaIntroducerKind Introducer,
 | |
|                                               Token &VisTok) {
 | |
|   SourceLocation VisLoc = VisTok.getLocation();
 | |
| 
 | |
|   Token Tok;
 | |
|   PP.LexUnexpandedToken(Tok);
 | |
| 
 | |
|   const IdentifierInfo *PushPop = Tok.getIdentifierInfo();
 | |
| 
 | |
|   const IdentifierInfo *VisType;
 | |
|   if (PushPop && PushPop->isStr("pop")) {
 | |
|     VisType = 0;
 | |
|   } else if (PushPop && PushPop->isStr("push")) {
 | |
|     PP.LexUnexpandedToken(Tok);
 | |
|     if (Tok.isNot(tok::l_paren)) {
 | |
|       PP.Diag(Tok.getLocation(), diag::warn_pragma_expected_lparen)
 | |
|         << "visibility";
 | |
|       return;
 | |
|     }
 | |
|     PP.LexUnexpandedToken(Tok);
 | |
|     VisType = Tok.getIdentifierInfo();
 | |
|     if (!VisType) {
 | |
|       PP.Diag(Tok.getLocation(), diag::warn_pragma_expected_identifier)
 | |
|         << "visibility";
 | |
|       return;
 | |
|     }
 | |
|     PP.LexUnexpandedToken(Tok);
 | |
|     if (Tok.isNot(tok::r_paren)) {
 | |
|       PP.Diag(Tok.getLocation(), diag::warn_pragma_expected_rparen)
 | |
|         << "visibility";
 | |
|       return;
 | |
|     }
 | |
|   } else {
 | |
|     PP.Diag(Tok.getLocation(), diag::warn_pragma_expected_identifier)
 | |
|       << "visibility";
 | |
|     return;
 | |
|   }
 | |
|   PP.LexUnexpandedToken(Tok);
 | |
|   if (Tok.isNot(tok::eod)) {
 | |
|     PP.Diag(Tok.getLocation(), diag::warn_pragma_extra_tokens_at_eol)
 | |
|       << "visibility";
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   Token *Toks = new Token[1];
 | |
|   Toks[0].startToken();
 | |
|   Toks[0].setKind(tok::annot_pragma_vis);
 | |
|   Toks[0].setLocation(VisLoc);
 | |
|   Toks[0].setAnnotationValue(
 | |
|                           const_cast<void*>(static_cast<const void*>(VisType)));
 | |
|   PP.EnterTokenStream(Toks, 1, /*DisableMacroExpansion=*/true,
 | |
|                       /*OwnsTokens=*/true);
 | |
| }
 | |
| 
 | |
| // #pragma pack(...) comes in the following delicious flavors:
 | |
| //   pack '(' [integer] ')'
 | |
| //   pack '(' 'show' ')'
 | |
| //   pack '(' ('push' | 'pop') [',' identifier] [, integer] ')'
 | |
| void PragmaPackHandler::HandlePragma(Preprocessor &PP, 
 | |
|                                      PragmaIntroducerKind Introducer,
 | |
|                                      Token &PackTok) {
 | |
|   SourceLocation PackLoc = PackTok.getLocation();
 | |
| 
 | |
|   Token Tok;
 | |
|   PP.Lex(Tok);
 | |
|   if (Tok.isNot(tok::l_paren)) {
 | |
|     PP.Diag(Tok.getLocation(), diag::warn_pragma_expected_lparen) << "pack";
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   Sema::PragmaPackKind Kind = Sema::PPK_Default;
 | |
|   IdentifierInfo *Name = 0;
 | |
|   Token Alignment;
 | |
|   Alignment.startToken();
 | |
|   SourceLocation LParenLoc = Tok.getLocation();
 | |
|   PP.Lex(Tok);
 | |
|   if (Tok.is(tok::numeric_constant)) {
 | |
|     Alignment = Tok;
 | |
| 
 | |
|     PP.Lex(Tok);
 | |
| 
 | |
|     // In MSVC/gcc, #pragma pack(4) sets the alignment without affecting
 | |
|     // the push/pop stack.
 | |
|     // In Apple gcc, #pragma pack(4) is equivalent to #pragma pack(push, 4)
 | |
|     if (PP.getLangOpts().ApplePragmaPack)
 | |
|       Kind = Sema::PPK_Push;
 | |
|   } else if (Tok.is(tok::identifier)) {
 | |
|     const IdentifierInfo *II = Tok.getIdentifierInfo();
 | |
|     if (II->isStr("show")) {
 | |
|       Kind = Sema::PPK_Show;
 | |
|       PP.Lex(Tok);
 | |
|     } else {
 | |
|       if (II->isStr("push")) {
 | |
|         Kind = Sema::PPK_Push;
 | |
|       } else if (II->isStr("pop")) {
 | |
|         Kind = Sema::PPK_Pop;
 | |
|       } else {
 | |
|         PP.Diag(Tok.getLocation(), diag::warn_pragma_pack_invalid_action);
 | |
|         return;
 | |
|       }
 | |
|       PP.Lex(Tok);
 | |
| 
 | |
|       if (Tok.is(tok::comma)) {
 | |
|         PP.Lex(Tok);
 | |
| 
 | |
|         if (Tok.is(tok::numeric_constant)) {
 | |
|           Alignment = Tok;
 | |
| 
 | |
|           PP.Lex(Tok);
 | |
|         } else if (Tok.is(tok::identifier)) {
 | |
|           Name = Tok.getIdentifierInfo();
 | |
|           PP.Lex(Tok);
 | |
| 
 | |
|           if (Tok.is(tok::comma)) {
 | |
|             PP.Lex(Tok);
 | |
| 
 | |
|             if (Tok.isNot(tok::numeric_constant)) {
 | |
|               PP.Diag(Tok.getLocation(), diag::warn_pragma_pack_malformed);
 | |
|               return;
 | |
|             }
 | |
| 
 | |
|             Alignment = Tok;
 | |
| 
 | |
|             PP.Lex(Tok);
 | |
|           }
 | |
|         } else {
 | |
|           PP.Diag(Tok.getLocation(), diag::warn_pragma_pack_malformed);
 | |
|           return;
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|   } else if (PP.getLangOpts().ApplePragmaPack) {
 | |
|     // In MSVC/gcc, #pragma pack() resets the alignment without affecting
 | |
|     // the push/pop stack.
 | |
|     // In Apple gcc #pragma pack() is equivalent to #pragma pack(pop).
 | |
|     Kind = Sema::PPK_Pop;
 | |
|   }
 | |
| 
 | |
|   if (Tok.isNot(tok::r_paren)) {
 | |
|     PP.Diag(Tok.getLocation(), diag::warn_pragma_expected_rparen) << "pack";
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   SourceLocation RParenLoc = Tok.getLocation();
 | |
|   PP.Lex(Tok);
 | |
|   if (Tok.isNot(tok::eod)) {
 | |
|     PP.Diag(Tok.getLocation(), diag::warn_pragma_extra_tokens_at_eol) << "pack";
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   PragmaPackInfo *Info = 
 | |
|     (PragmaPackInfo*) PP.getPreprocessorAllocator().Allocate(
 | |
|       sizeof(PragmaPackInfo), llvm::alignOf<PragmaPackInfo>());
 | |
|   new (Info) PragmaPackInfo();
 | |
|   Info->Kind = Kind;
 | |
|   Info->Name = Name;
 | |
|   Info->Alignment = Alignment;
 | |
|   Info->LParenLoc = LParenLoc;
 | |
|   Info->RParenLoc = RParenLoc;
 | |
| 
 | |
|   Token *Toks = 
 | |
|     (Token*) PP.getPreprocessorAllocator().Allocate(
 | |
|       sizeof(Token) * 1, llvm::alignOf<Token>());
 | |
|   new (Toks) Token();
 | |
|   Toks[0].startToken();
 | |
|   Toks[0].setKind(tok::annot_pragma_pack);
 | |
|   Toks[0].setLocation(PackLoc);
 | |
|   Toks[0].setAnnotationValue(static_cast<void*>(Info));
 | |
|   PP.EnterTokenStream(Toks, 1, /*DisableMacroExpansion=*/true,
 | |
|                       /*OwnsTokens=*/false);
 | |
| }
 | |
| 
 | |
| // #pragma ms_struct on
 | |
| // #pragma ms_struct off
 | |
| void PragmaMSStructHandler::HandlePragma(Preprocessor &PP, 
 | |
|                                          PragmaIntroducerKind Introducer,
 | |
|                                          Token &MSStructTok) {
 | |
|   Sema::PragmaMSStructKind Kind = Sema::PMSST_OFF;
 | |
|   
 | |
|   Token Tok;
 | |
|   PP.Lex(Tok);
 | |
|   if (Tok.isNot(tok::identifier)) {
 | |
|     PP.Diag(Tok.getLocation(), diag::warn_pragma_ms_struct);
 | |
|     return;
 | |
|   }
 | |
|   const IdentifierInfo *II = Tok.getIdentifierInfo();
 | |
|   if (II->isStr("on")) {
 | |
|     Kind = Sema::PMSST_ON;
 | |
|     PP.Lex(Tok);
 | |
|   }
 | |
|   else if (II->isStr("off") || II->isStr("reset"))
 | |
|     PP.Lex(Tok);
 | |
|   else {
 | |
|     PP.Diag(Tok.getLocation(), diag::warn_pragma_ms_struct);
 | |
|     return;
 | |
|   }
 | |
|   
 | |
|   if (Tok.isNot(tok::eod)) {
 | |
|     PP.Diag(Tok.getLocation(), diag::warn_pragma_extra_tokens_at_eol)
 | |
|       << "ms_struct";
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   Token *Toks =
 | |
|     (Token*) PP.getPreprocessorAllocator().Allocate(
 | |
|       sizeof(Token) * 1, llvm::alignOf<Token>());
 | |
|   new (Toks) Token();
 | |
|   Toks[0].startToken();
 | |
|   Toks[0].setKind(tok::annot_pragma_msstruct);
 | |
|   Toks[0].setLocation(MSStructTok.getLocation());
 | |
|   Toks[0].setAnnotationValue(reinterpret_cast<void*>(
 | |
|                              static_cast<uintptr_t>(Kind)));
 | |
|   PP.EnterTokenStream(Toks, 1, /*DisableMacroExpansion=*/true,
 | |
|                       /*OwnsTokens=*/false);
 | |
| }
 | |
| 
 | |
| // #pragma 'align' '=' {'native','natural','mac68k','power','reset'}
 | |
| // #pragma 'options 'align' '=' {'native','natural','mac68k','power','reset'}
 | |
| static void ParseAlignPragma(Preprocessor &PP, Token &FirstTok,
 | |
|                              bool IsOptions) {
 | |
|   Token Tok;
 | |
| 
 | |
|   if (IsOptions) {
 | |
|     PP.Lex(Tok);
 | |
|     if (Tok.isNot(tok::identifier) ||
 | |
|         !Tok.getIdentifierInfo()->isStr("align")) {
 | |
|       PP.Diag(Tok.getLocation(), diag::warn_pragma_options_expected_align);
 | |
|       return;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   PP.Lex(Tok);
 | |
|   if (Tok.isNot(tok::equal)) {
 | |
|     PP.Diag(Tok.getLocation(), diag::warn_pragma_align_expected_equal)
 | |
|       << IsOptions;
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   PP.Lex(Tok);
 | |
|   if (Tok.isNot(tok::identifier)) {
 | |
|     PP.Diag(Tok.getLocation(), diag::warn_pragma_expected_identifier)
 | |
|       << (IsOptions ? "options" : "align");
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   Sema::PragmaOptionsAlignKind Kind = Sema::POAK_Natural;
 | |
|   const IdentifierInfo *II = Tok.getIdentifierInfo();
 | |
|   if (II->isStr("native"))
 | |
|     Kind = Sema::POAK_Native;
 | |
|   else if (II->isStr("natural"))
 | |
|     Kind = Sema::POAK_Natural;
 | |
|   else if (II->isStr("packed"))
 | |
|     Kind = Sema::POAK_Packed;
 | |
|   else if (II->isStr("power"))
 | |
|     Kind = Sema::POAK_Power;
 | |
|   else if (II->isStr("mac68k"))
 | |
|     Kind = Sema::POAK_Mac68k;
 | |
|   else if (II->isStr("reset"))
 | |
|     Kind = Sema::POAK_Reset;
 | |
|   else {
 | |
|     PP.Diag(Tok.getLocation(), diag::warn_pragma_align_invalid_option)
 | |
|       << IsOptions;
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   PP.Lex(Tok);
 | |
|   if (Tok.isNot(tok::eod)) {
 | |
|     PP.Diag(Tok.getLocation(), diag::warn_pragma_extra_tokens_at_eol)
 | |
|       << (IsOptions ? "options" : "align");
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   Token *Toks =
 | |
|     (Token*) PP.getPreprocessorAllocator().Allocate(
 | |
|       sizeof(Token) * 1, llvm::alignOf<Token>());
 | |
|   new (Toks) Token();
 | |
|   Toks[0].startToken();
 | |
|   Toks[0].setKind(tok::annot_pragma_align);
 | |
|   Toks[0].setLocation(FirstTok.getLocation());
 | |
|   Toks[0].setAnnotationValue(reinterpret_cast<void*>(
 | |
|                              static_cast<uintptr_t>(Kind)));
 | |
|   PP.EnterTokenStream(Toks, 1, /*DisableMacroExpansion=*/true,
 | |
|                       /*OwnsTokens=*/false);
 | |
| }
 | |
| 
 | |
| void PragmaAlignHandler::HandlePragma(Preprocessor &PP, 
 | |
|                                       PragmaIntroducerKind Introducer,
 | |
|                                       Token &AlignTok) {
 | |
|   ParseAlignPragma(PP, AlignTok, /*IsOptions=*/false);
 | |
| }
 | |
| 
 | |
| void PragmaOptionsHandler::HandlePragma(Preprocessor &PP, 
 | |
|                                         PragmaIntroducerKind Introducer,
 | |
|                                         Token &OptionsTok) {
 | |
|   ParseAlignPragma(PP, OptionsTok, /*IsOptions=*/true);
 | |
| }
 | |
| 
 | |
| // #pragma unused(identifier)
 | |
| void PragmaUnusedHandler::HandlePragma(Preprocessor &PP, 
 | |
|                                        PragmaIntroducerKind Introducer,
 | |
|                                        Token &UnusedTok) {
 | |
|   // FIXME: Should we be expanding macros here? My guess is no.
 | |
|   SourceLocation UnusedLoc = UnusedTok.getLocation();
 | |
| 
 | |
|   // Lex the left '('.
 | |
|   Token Tok;
 | |
|   PP.Lex(Tok);
 | |
|   if (Tok.isNot(tok::l_paren)) {
 | |
|     PP.Diag(Tok.getLocation(), diag::warn_pragma_expected_lparen) << "unused";
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   // Lex the declaration reference(s).
 | |
|   SmallVector<Token, 5> Identifiers;
 | |
|   SourceLocation RParenLoc;
 | |
|   bool LexID = true;
 | |
| 
 | |
|   while (true) {
 | |
|     PP.Lex(Tok);
 | |
| 
 | |
|     if (LexID) {
 | |
|       if (Tok.is(tok::identifier)) {
 | |
|         Identifiers.push_back(Tok);
 | |
|         LexID = false;
 | |
|         continue;
 | |
|       }
 | |
| 
 | |
|       // Illegal token!
 | |
|       PP.Diag(Tok.getLocation(), diag::warn_pragma_unused_expected_var);
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     // We are execting a ')' or a ','.
 | |
|     if (Tok.is(tok::comma)) {
 | |
|       LexID = true;
 | |
|       continue;
 | |
|     }
 | |
| 
 | |
|     if (Tok.is(tok::r_paren)) {
 | |
|       RParenLoc = Tok.getLocation();
 | |
|       break;
 | |
|     }
 | |
| 
 | |
|     // Illegal token!
 | |
|     PP.Diag(Tok.getLocation(), diag::warn_pragma_unused_expected_punc);
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   PP.Lex(Tok);
 | |
|   if (Tok.isNot(tok::eod)) {
 | |
|     PP.Diag(Tok.getLocation(), diag::warn_pragma_extra_tokens_at_eol) <<
 | |
|         "unused";
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   // Verify that we have a location for the right parenthesis.
 | |
|   assert(RParenLoc.isValid() && "Valid '#pragma unused' must have ')'");
 | |
|   assert(!Identifiers.empty() && "Valid '#pragma unused' must have arguments");
 | |
| 
 | |
|   // For each identifier token, insert into the token stream a
 | |
|   // annot_pragma_unused token followed by the identifier token.
 | |
|   // This allows us to cache a "#pragma unused" that occurs inside an inline
 | |
|   // C++ member function.
 | |
| 
 | |
|   Token *Toks = 
 | |
|     (Token*) PP.getPreprocessorAllocator().Allocate(
 | |
|       sizeof(Token) * 2 * Identifiers.size(), llvm::alignOf<Token>());
 | |
|   for (unsigned i=0; i != Identifiers.size(); i++) {
 | |
|     Token &pragmaUnusedTok = Toks[2*i], &idTok = Toks[2*i+1];
 | |
|     pragmaUnusedTok.startToken();
 | |
|     pragmaUnusedTok.setKind(tok::annot_pragma_unused);
 | |
|     pragmaUnusedTok.setLocation(UnusedLoc);
 | |
|     idTok = Identifiers[i];
 | |
|   }
 | |
|   PP.EnterTokenStream(Toks, 2*Identifiers.size(),
 | |
|                       /*DisableMacroExpansion=*/true, /*OwnsTokens=*/false);
 | |
| }
 | |
| 
 | |
| // #pragma weak identifier
 | |
| // #pragma weak identifier '=' identifier
 | |
| void PragmaWeakHandler::HandlePragma(Preprocessor &PP, 
 | |
|                                      PragmaIntroducerKind Introducer,
 | |
|                                      Token &WeakTok) {
 | |
|   SourceLocation WeakLoc = WeakTok.getLocation();
 | |
| 
 | |
|   Token Tok;
 | |
|   PP.Lex(Tok);
 | |
|   if (Tok.isNot(tok::identifier)) {
 | |
|     PP.Diag(Tok.getLocation(), diag::warn_pragma_expected_identifier) << "weak";
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   Token WeakName = Tok;
 | |
|   bool HasAlias = false;
 | |
|   Token AliasName;
 | |
| 
 | |
|   PP.Lex(Tok);
 | |
|   if (Tok.is(tok::equal)) {
 | |
|     HasAlias = true;
 | |
|     PP.Lex(Tok);
 | |
|     if (Tok.isNot(tok::identifier)) {
 | |
|       PP.Diag(Tok.getLocation(), diag::warn_pragma_expected_identifier)
 | |
|           << "weak";
 | |
|       return;
 | |
|     }
 | |
|     AliasName = Tok;
 | |
|     PP.Lex(Tok);
 | |
|   }
 | |
| 
 | |
|   if (Tok.isNot(tok::eod)) {
 | |
|     PP.Diag(Tok.getLocation(), diag::warn_pragma_extra_tokens_at_eol) << "weak";
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (HasAlias) {
 | |
|     Token *Toks = 
 | |
|       (Token*) PP.getPreprocessorAllocator().Allocate(
 | |
|         sizeof(Token) * 3, llvm::alignOf<Token>());
 | |
|     Token &pragmaUnusedTok = Toks[0];
 | |
|     pragmaUnusedTok.startToken();
 | |
|     pragmaUnusedTok.setKind(tok::annot_pragma_weakalias);
 | |
|     pragmaUnusedTok.setLocation(WeakLoc);
 | |
|     Toks[1] = WeakName;
 | |
|     Toks[2] = AliasName;
 | |
|     PP.EnterTokenStream(Toks, 3,
 | |
|                         /*DisableMacroExpansion=*/true, /*OwnsTokens=*/false);
 | |
|   } else {
 | |
|     Token *Toks = 
 | |
|       (Token*) PP.getPreprocessorAllocator().Allocate(
 | |
|         sizeof(Token) * 2, llvm::alignOf<Token>());
 | |
|     Token &pragmaUnusedTok = Toks[0];
 | |
|     pragmaUnusedTok.startToken();
 | |
|     pragmaUnusedTok.setKind(tok::annot_pragma_weak);
 | |
|     pragmaUnusedTok.setLocation(WeakLoc);
 | |
|     Toks[1] = WeakName;
 | |
|     PP.EnterTokenStream(Toks, 2,
 | |
|                         /*DisableMacroExpansion=*/true, /*OwnsTokens=*/false);
 | |
|   }
 | |
| }
 | |
| 
 | |
| // #pragma redefine_extname identifier identifier
 | |
| void PragmaRedefineExtnameHandler::HandlePragma(Preprocessor &PP, 
 | |
|                                                PragmaIntroducerKind Introducer,
 | |
|                                                 Token &RedefToken) {
 | |
|   SourceLocation RedefLoc = RedefToken.getLocation();
 | |
| 
 | |
|   Token Tok;
 | |
|   PP.Lex(Tok);
 | |
|   if (Tok.isNot(tok::identifier)) {
 | |
|     PP.Diag(Tok.getLocation(), diag::warn_pragma_expected_identifier) <<
 | |
|       "redefine_extname";
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   Token RedefName = Tok;
 | |
|   PP.Lex(Tok);
 | |
| 
 | |
|   if (Tok.isNot(tok::identifier)) {
 | |
|     PP.Diag(Tok.getLocation(), diag::warn_pragma_expected_identifier)
 | |
|         << "redefine_extname";
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   Token AliasName = Tok;
 | |
|   PP.Lex(Tok);
 | |
| 
 | |
|   if (Tok.isNot(tok::eod)) {
 | |
|     PP.Diag(Tok.getLocation(), diag::warn_pragma_extra_tokens_at_eol) <<
 | |
|       "redefine_extname";
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   Token *Toks = 
 | |
|     (Token*) PP.getPreprocessorAllocator().Allocate(
 | |
|       sizeof(Token) * 3, llvm::alignOf<Token>());
 | |
|   Token &pragmaRedefTok = Toks[0];
 | |
|   pragmaRedefTok.startToken();
 | |
|   pragmaRedefTok.setKind(tok::annot_pragma_redefine_extname);
 | |
|   pragmaRedefTok.setLocation(RedefLoc);
 | |
|   Toks[1] = RedefName;
 | |
|   Toks[2] = AliasName;
 | |
|   PP.EnterTokenStream(Toks, 3,
 | |
|                       /*DisableMacroExpansion=*/true, /*OwnsTokens=*/false);
 | |
| }
 | |
| 
 | |
| 
 | |
| void
 | |
| PragmaFPContractHandler::HandlePragma(Preprocessor &PP, 
 | |
|                                       PragmaIntroducerKind Introducer,
 | |
|                                       Token &Tok) {
 | |
|   tok::OnOffSwitch OOS;
 | |
|   if (PP.LexOnOffSwitch(OOS))
 | |
|     return;
 | |
| 
 | |
|   Token *Toks =
 | |
|     (Token*) PP.getPreprocessorAllocator().Allocate(
 | |
|       sizeof(Token) * 1, llvm::alignOf<Token>());
 | |
|   new (Toks) Token();
 | |
|   Toks[0].startToken();
 | |
|   Toks[0].setKind(tok::annot_pragma_fp_contract);
 | |
|   Toks[0].setLocation(Tok.getLocation());
 | |
|   Toks[0].setAnnotationValue(reinterpret_cast<void*>(
 | |
|                              static_cast<uintptr_t>(OOS)));
 | |
|   PP.EnterTokenStream(Toks, 1, /*DisableMacroExpansion=*/true,
 | |
|                       /*OwnsTokens=*/false);
 | |
| }
 | |
| 
 | |
| void 
 | |
| PragmaOpenCLExtensionHandler::HandlePragma(Preprocessor &PP, 
 | |
|                                            PragmaIntroducerKind Introducer,
 | |
|                                            Token &Tok) {
 | |
|   PP.LexUnexpandedToken(Tok);
 | |
|   if (Tok.isNot(tok::identifier)) {
 | |
|     PP.Diag(Tok.getLocation(), diag::warn_pragma_expected_identifier) <<
 | |
|       "OPENCL";
 | |
|     return;
 | |
|   }
 | |
|   IdentifierInfo *ename = Tok.getIdentifierInfo();
 | |
|   SourceLocation NameLoc = Tok.getLocation();
 | |
| 
 | |
|   PP.Lex(Tok);
 | |
|   if (Tok.isNot(tok::colon)) {
 | |
|     PP.Diag(Tok.getLocation(), diag::warn_pragma_expected_colon) << ename;
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   PP.Lex(Tok);
 | |
|   if (Tok.isNot(tok::identifier)) {
 | |
|     PP.Diag(Tok.getLocation(), diag::warn_pragma_expected_enable_disable);
 | |
|     return;
 | |
|   }
 | |
|   IdentifierInfo *op = Tok.getIdentifierInfo();
 | |
| 
 | |
|   unsigned state;
 | |
|   if (op->isStr("enable")) {
 | |
|     state = 1;
 | |
|   } else if (op->isStr("disable")) {
 | |
|     state = 0;
 | |
|   } else {
 | |
|     PP.Diag(Tok.getLocation(), diag::warn_pragma_expected_enable_disable);
 | |
|     return;
 | |
|   }
 | |
|   SourceLocation StateLoc = Tok.getLocation();
 | |
| 
 | |
|   PP.Lex(Tok);
 | |
|   if (Tok.isNot(tok::eod)) {
 | |
|     PP.Diag(Tok.getLocation(), diag::warn_pragma_extra_tokens_at_eol) <<
 | |
|       "OPENCL EXTENSION";
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   OpenCLExtData data(ename, state);
 | |
|   Token *Toks =
 | |
|     (Token*) PP.getPreprocessorAllocator().Allocate(
 | |
|       sizeof(Token) * 1, llvm::alignOf<Token>());
 | |
|   new (Toks) Token();
 | |
|   Toks[0].startToken();
 | |
|   Toks[0].setKind(tok::annot_pragma_opencl_extension);
 | |
|   Toks[0].setLocation(NameLoc);
 | |
|   Toks[0].setAnnotationValue(data.getOpaqueValue());
 | |
|   PP.EnterTokenStream(Toks, 1, /*DisableMacroExpansion=*/true,
 | |
|                       /*OwnsTokens=*/false);
 | |
| 
 | |
|   if (PP.getPPCallbacks())
 | |
|     PP.getPPCallbacks()->PragmaOpenCLExtension(NameLoc, ename, 
 | |
|                                                StateLoc, state);
 | |
| }
 | |
| 
 | |
| /// \brief Handle '#pragma omp ...' when OpenMP is disabled.
 | |
| ///
 | |
| void
 | |
| PragmaNoOpenMPHandler::HandlePragma(Preprocessor &PP,
 | |
|                                     PragmaIntroducerKind Introducer,
 | |
|                                     Token &FirstTok) {
 | |
|   if (PP.getDiagnostics().getDiagnosticLevel(diag::warn_pragma_omp_ignored,
 | |
|                                              FirstTok.getLocation()) !=
 | |
|       DiagnosticsEngine::Ignored) {
 | |
|     PP.Diag(FirstTok, diag::warn_pragma_omp_ignored);
 | |
|     PP.getDiagnostics().setDiagnosticMapping(diag::warn_pragma_omp_ignored,
 | |
|                                              diag::MAP_IGNORE,
 | |
|                                              SourceLocation());
 | |
|   }
 | |
|   PP.DiscardUntilEndOfDirective();
 | |
| }
 | |
| 
 | |
| /// \brief Handle '#pragma omp ...' when OpenMP is enabled.
 | |
| ///
 | |
| void
 | |
| PragmaOpenMPHandler::HandlePragma(Preprocessor &PP,
 | |
|                                   PragmaIntroducerKind Introducer,
 | |
|                                   Token &FirstTok) {
 | |
|   SmallVector<Token, 16> Pragma;
 | |
|   Token Tok;
 | |
|   Tok.startToken();
 | |
|   Tok.setKind(tok::annot_pragma_openmp);
 | |
|   Tok.setLocation(FirstTok.getLocation());
 | |
| 
 | |
|   while (Tok.isNot(tok::eod)) {
 | |
|     Pragma.push_back(Tok);
 | |
|     PP.Lex(Tok);
 | |
|   }
 | |
|   SourceLocation EodLoc = Tok.getLocation();
 | |
|   Tok.startToken();
 | |
|   Tok.setKind(tok::annot_pragma_openmp_end);
 | |
|   Tok.setLocation(EodLoc);
 | |
|   Pragma.push_back(Tok);
 | |
| 
 | |
|   Token *Toks = new Token[Pragma.size()];
 | |
|   std::copy(Pragma.begin(), Pragma.end(), Toks);
 | |
|   PP.EnterTokenStream(Toks, Pragma.size(),
 | |
|                       /*DisableMacroExpansion=*/true, /*OwnsTokens=*/true);
 | |
| }
 | |
| 
 | |
| /// \brief Handle the Microsoft \#pragma detect_mismatch extension.
 | |
| ///
 | |
| /// The syntax is:
 | |
| /// \code
 | |
| ///   #pragma detect_mismatch("name", "value")
 | |
| /// \endcode
 | |
| /// Where 'name' and 'value' are quoted strings.  The values are embedded in
 | |
| /// the object file and passed along to the linker.  If the linker detects a
 | |
| /// mismatch in the object file's values for the given name, a LNK2038 error
 | |
| /// is emitted.  See MSDN for more details.
 | |
| void PragmaDetectMismatchHandler::HandlePragma(Preprocessor &PP,
 | |
|                                                PragmaIntroducerKind Introducer,
 | |
|                                                Token &Tok) {
 | |
|   SourceLocation CommentLoc = Tok.getLocation();
 | |
|   PP.Lex(Tok);
 | |
|   if (Tok.isNot(tok::l_paren)) {
 | |
|     PP.Diag(CommentLoc, diag::err_expected_lparen);
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   // Read the name to embed, which must be a string literal.
 | |
|   std::string NameString;
 | |
|   if (!PP.LexStringLiteral(Tok, NameString,
 | |
|                            "pragma detect_mismatch",
 | |
|                            /*MacroExpansion=*/true))
 | |
|     return;
 | |
| 
 | |
|   // Read the comma followed by a second string literal.
 | |
|   std::string ValueString;
 | |
|   if (Tok.isNot(tok::comma)) {
 | |
|     PP.Diag(Tok.getLocation(), diag::err_pragma_detect_mismatch_malformed);
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (!PP.LexStringLiteral(Tok, ValueString, "pragma detect_mismatch",
 | |
|                            /*MacroExpansion=*/true))
 | |
|     return;
 | |
| 
 | |
|   if (Tok.isNot(tok::r_paren)) {
 | |
|     PP.Diag(Tok.getLocation(), diag::err_expected_rparen);
 | |
|     return;
 | |
|   }
 | |
|   PP.Lex(Tok);  // Eat the r_paren.
 | |
| 
 | |
|   if (Tok.isNot(tok::eod)) {
 | |
|     PP.Diag(Tok.getLocation(), diag::err_pragma_detect_mismatch_malformed);
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   // If the pragma is lexically sound, notify any interested PPCallbacks.
 | |
|   if (PP.getPPCallbacks())
 | |
|     PP.getPPCallbacks()->PragmaDetectMismatch(CommentLoc, NameString,
 | |
|                                               ValueString);
 | |
| 
 | |
|   Actions.ActOnPragmaDetectMismatch(NameString, ValueString);
 | |
| }
 | |
| 
 | |
| /// \brief Handle the microsoft \#pragma comment extension.
 | |
| ///
 | |
| /// The syntax is:
 | |
| /// \code
 | |
| ///   #pragma comment(linker, "foo")
 | |
| /// \endcode
 | |
| /// 'linker' is one of five identifiers: compiler, exestr, lib, linker, user.
 | |
| /// "foo" is a string, which is fully macro expanded, and permits string
 | |
| /// concatenation, embedded escape characters etc.  See MSDN for more details.
 | |
| void PragmaCommentHandler::HandlePragma(Preprocessor &PP,
 | |
|                                         PragmaIntroducerKind Introducer,
 | |
|                                         Token &Tok) {
 | |
|   SourceLocation CommentLoc = Tok.getLocation();
 | |
|   PP.Lex(Tok);
 | |
|   if (Tok.isNot(tok::l_paren)) {
 | |
|     PP.Diag(CommentLoc, diag::err_pragma_comment_malformed);
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   // Read the identifier.
 | |
|   PP.Lex(Tok);
 | |
|   if (Tok.isNot(tok::identifier)) {
 | |
|     PP.Diag(CommentLoc, diag::err_pragma_comment_malformed);
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   // Verify that this is one of the 5 whitelisted options.
 | |
|   IdentifierInfo *II = Tok.getIdentifierInfo();
 | |
|   Sema::PragmaMSCommentKind Kind =
 | |
|     llvm::StringSwitch<Sema::PragmaMSCommentKind>(II->getName())
 | |
|     .Case("linker",   Sema::PCK_Linker)
 | |
|     .Case("lib",      Sema::PCK_Lib)
 | |
|     .Case("compiler", Sema::PCK_Compiler)
 | |
|     .Case("exestr",   Sema::PCK_ExeStr)
 | |
|     .Case("user",     Sema::PCK_User)
 | |
|     .Default(Sema::PCK_Unknown);
 | |
|   if (Kind == Sema::PCK_Unknown) {
 | |
|     PP.Diag(Tok.getLocation(), diag::err_pragma_comment_unknown_kind);
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   // Read the optional string if present.
 | |
|   PP.Lex(Tok);
 | |
|   std::string ArgumentString;
 | |
|   if (Tok.is(tok::comma) && !PP.LexStringLiteral(Tok, ArgumentString,
 | |
|                                                  "pragma comment",
 | |
|                                                  /*MacroExpansion=*/true))
 | |
|     return;
 | |
| 
 | |
|   // FIXME: warn that 'exestr' is deprecated.
 | |
|   // FIXME: If the kind is "compiler" warn if the string is present (it is
 | |
|   // ignored).
 | |
|   // The MSDN docs say that "lib" and "linker" require a string and have a short
 | |
|   // whitelist of linker options they support, but in practice MSVC doesn't
 | |
|   // issue a diagnostic.  Therefore neither does clang.
 | |
| 
 | |
|   if (Tok.isNot(tok::r_paren)) {
 | |
|     PP.Diag(Tok.getLocation(), diag::err_pragma_comment_malformed);
 | |
|     return;
 | |
|   }
 | |
|   PP.Lex(Tok);  // eat the r_paren.
 | |
| 
 | |
|   if (Tok.isNot(tok::eod)) {
 | |
|     PP.Diag(Tok.getLocation(), diag::err_pragma_comment_malformed);
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   // If the pragma is lexically sound, notify any interested PPCallbacks.
 | |
|   if (PP.getPPCallbacks())
 | |
|     PP.getPPCallbacks()->PragmaComment(CommentLoc, II, ArgumentString);
 | |
| 
 | |
|   Actions.ActOnPragmaMSComment(Kind, ArgumentString);
 | |
| }
 | 
