1 //===--- ParsePragma.cpp - Language specific pragma parsing ---------------===// 2 // 3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 4 // See https://llvm.org/LICENSE.txt for license information. 5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 6 // 7 //===----------------------------------------------------------------------===// 8 // 9 // This file implements the language specific #pragma handlers. 10 // 11 //===----------------------------------------------------------------------===// 12 13 #include "clang/AST/ASTContext.h" 14 #include "clang/Basic/PragmaKinds.h" 15 #include "clang/Basic/TargetInfo.h" 16 #include "clang/Lex/Preprocessor.h" 17 #include "clang/Parse/LoopHint.h" 18 #include "clang/Parse/ParseDiagnostic.h" 19 #include "clang/Parse/Parser.h" 20 #include "clang/Parse/RAIIObjectsForParser.h" 21 #include "clang/Sema/Scope.h" 22 #include "llvm/ADT/StringSwitch.h" 23 using namespace clang; 24 25 namespace { 26 27 struct PragmaAlignHandler : public PragmaHandler { 28 explicit PragmaAlignHandler() : PragmaHandler("align") {} 29 void HandlePragma(Preprocessor &PP, PragmaIntroducer Introducer, 30 Token &FirstToken) override; 31 }; 32 33 struct PragmaGCCVisibilityHandler : public PragmaHandler { 34 explicit PragmaGCCVisibilityHandler() : PragmaHandler("visibility") {} 35 void HandlePragma(Preprocessor &PP, PragmaIntroducer Introducer, 36 Token &FirstToken) override; 37 }; 38 39 struct PragmaOptionsHandler : public PragmaHandler { 40 explicit PragmaOptionsHandler() : PragmaHandler("options") {} 41 void HandlePragma(Preprocessor &PP, PragmaIntroducer Introducer, 42 Token &FirstToken) override; 43 }; 44 45 struct PragmaPackHandler : public PragmaHandler { 46 explicit PragmaPackHandler() : PragmaHandler("pack") {} 47 void HandlePragma(Preprocessor &PP, PragmaIntroducer Introducer, 48 Token &FirstToken) override; 49 }; 50 51 struct PragmaClangSectionHandler : public PragmaHandler { 52 explicit PragmaClangSectionHandler(Sema &S) 53 : PragmaHandler("section"), Actions(S) {} 54 void HandlePragma(Preprocessor &PP, PragmaIntroducer Introducer, 55 Token &FirstToken) override; 56 57 private: 58 Sema &Actions; 59 }; 60 61 struct PragmaMSStructHandler : public PragmaHandler { 62 explicit PragmaMSStructHandler() : PragmaHandler("ms_struct") {} 63 void HandlePragma(Preprocessor &PP, PragmaIntroducer Introducer, 64 Token &FirstToken) override; 65 }; 66 67 struct PragmaUnusedHandler : public PragmaHandler { 68 PragmaUnusedHandler() : PragmaHandler("unused") {} 69 void HandlePragma(Preprocessor &PP, PragmaIntroducer Introducer, 70 Token &FirstToken) override; 71 }; 72 73 struct PragmaWeakHandler : public PragmaHandler { 74 explicit PragmaWeakHandler() : PragmaHandler("weak") {} 75 void HandlePragma(Preprocessor &PP, PragmaIntroducer Introducer, 76 Token &FirstToken) override; 77 }; 78 79 struct PragmaRedefineExtnameHandler : public PragmaHandler { 80 explicit PragmaRedefineExtnameHandler() : PragmaHandler("redefine_extname") {} 81 void HandlePragma(Preprocessor &PP, PragmaIntroducer Introducer, 82 Token &FirstToken) override; 83 }; 84 85 struct PragmaOpenCLExtensionHandler : public PragmaHandler { 86 PragmaOpenCLExtensionHandler() : PragmaHandler("EXTENSION") {} 87 void HandlePragma(Preprocessor &PP, PragmaIntroducer Introducer, 88 Token &FirstToken) override; 89 }; 90 91 92 struct PragmaFPContractHandler : public PragmaHandler { 93 PragmaFPContractHandler() : PragmaHandler("FP_CONTRACT") {} 94 void HandlePragma(Preprocessor &PP, PragmaIntroducer Introducer, 95 Token &FirstToken) override; 96 }; 97 98 // Pragma STDC implementations. 99 100 /// PragmaSTDC_FENV_ACCESSHandler - "\#pragma STDC FENV_ACCESS ...". 101 struct PragmaSTDC_FENV_ACCESSHandler : public PragmaHandler { 102 PragmaSTDC_FENV_ACCESSHandler() : PragmaHandler("FENV_ACCESS") {} 103 104 void HandlePragma(Preprocessor &PP, PragmaIntroducer Introducer, 105 Token &Tok) override { 106 tok::OnOffSwitch OOS; 107 if (PP.LexOnOffSwitch(OOS)) 108 return; 109 if (OOS == tok::OOS_ON) { 110 PP.Diag(Tok, diag::warn_stdc_fenv_access_not_supported); 111 } 112 113 MutableArrayRef<Token> Toks(PP.getPreprocessorAllocator().Allocate<Token>(1), 114 1); 115 Toks[0].startToken(); 116 Toks[0].setKind(tok::annot_pragma_fenv_access); 117 Toks[0].setLocation(Tok.getLocation()); 118 Toks[0].setAnnotationEndLoc(Tok.getLocation()); 119 Toks[0].setAnnotationValue(reinterpret_cast<void*>( 120 static_cast<uintptr_t>(OOS))); 121 PP.EnterTokenStream(Toks, /*DisableMacroExpansion=*/true, 122 /*IsReinject=*/false); 123 } 124 }; 125 126 /// PragmaSTDC_CX_LIMITED_RANGEHandler - "\#pragma STDC CX_LIMITED_RANGE ...". 127 struct PragmaSTDC_CX_LIMITED_RANGEHandler : public PragmaHandler { 128 PragmaSTDC_CX_LIMITED_RANGEHandler() : PragmaHandler("CX_LIMITED_RANGE") {} 129 130 void HandlePragma(Preprocessor &PP, PragmaIntroducer Introducer, 131 Token &Tok) override { 132 tok::OnOffSwitch OOS; 133 PP.LexOnOffSwitch(OOS); 134 } 135 }; 136 137 /// PragmaSTDC_UnknownHandler - "\#pragma STDC ...". 138 struct PragmaSTDC_UnknownHandler : public PragmaHandler { 139 PragmaSTDC_UnknownHandler() = default; 140 141 void HandlePragma(Preprocessor &PP, PragmaIntroducer Introducer, 142 Token &UnknownTok) override { 143 // C99 6.10.6p2, unknown forms are not allowed. 144 PP.Diag(UnknownTok, diag::ext_stdc_pragma_ignored); 145 } 146 }; 147 148 struct PragmaFPHandler : public PragmaHandler { 149 PragmaFPHandler() : PragmaHandler("fp") {} 150 void HandlePragma(Preprocessor &PP, PragmaIntroducer Introducer, 151 Token &FirstToken) override; 152 }; 153 154 struct PragmaNoOpenMPHandler : public PragmaHandler { 155 PragmaNoOpenMPHandler() : PragmaHandler("omp") { } 156 void HandlePragma(Preprocessor &PP, PragmaIntroducer Introducer, 157 Token &FirstToken) override; 158 }; 159 160 struct PragmaOpenMPHandler : public PragmaHandler { 161 PragmaOpenMPHandler() : PragmaHandler("omp") { } 162 void HandlePragma(Preprocessor &PP, PragmaIntroducer Introducer, 163 Token &FirstToken) override; 164 }; 165 166 /// PragmaCommentHandler - "\#pragma comment ...". 167 struct PragmaCommentHandler : public PragmaHandler { 168 PragmaCommentHandler(Sema &Actions) 169 : PragmaHandler("comment"), Actions(Actions) {} 170 void HandlePragma(Preprocessor &PP, PragmaIntroducer Introducer, 171 Token &FirstToken) override; 172 173 private: 174 Sema &Actions; 175 }; 176 177 struct PragmaDetectMismatchHandler : public PragmaHandler { 178 PragmaDetectMismatchHandler(Sema &Actions) 179 : PragmaHandler("detect_mismatch"), Actions(Actions) {} 180 void HandlePragma(Preprocessor &PP, PragmaIntroducer Introducer, 181 Token &FirstToken) override; 182 183 private: 184 Sema &Actions; 185 }; 186 187 struct PragmaMSPointersToMembers : public PragmaHandler { 188 explicit PragmaMSPointersToMembers() : PragmaHandler("pointers_to_members") {} 189 void HandlePragma(Preprocessor &PP, PragmaIntroducer Introducer, 190 Token &FirstToken) override; 191 }; 192 193 struct PragmaMSVtorDisp : public PragmaHandler { 194 explicit PragmaMSVtorDisp() : PragmaHandler("vtordisp") {} 195 void HandlePragma(Preprocessor &PP, PragmaIntroducer Introducer, 196 Token &FirstToken) override; 197 }; 198 199 struct PragmaMSPragma : public PragmaHandler { 200 explicit PragmaMSPragma(const char *name) : PragmaHandler(name) {} 201 void HandlePragma(Preprocessor &PP, PragmaIntroducer Introducer, 202 Token &FirstToken) override; 203 }; 204 205 /// PragmaOptimizeHandler - "\#pragma clang optimize on/off". 206 struct PragmaOptimizeHandler : public PragmaHandler { 207 PragmaOptimizeHandler(Sema &S) 208 : PragmaHandler("optimize"), Actions(S) {} 209 void HandlePragma(Preprocessor &PP, PragmaIntroducer Introducer, 210 Token &FirstToken) override; 211 212 private: 213 Sema &Actions; 214 }; 215 216 struct PragmaLoopHintHandler : public PragmaHandler { 217 PragmaLoopHintHandler() : PragmaHandler("loop") {} 218 void HandlePragma(Preprocessor &PP, PragmaIntroducer Introducer, 219 Token &FirstToken) override; 220 }; 221 222 struct PragmaUnrollHintHandler : public PragmaHandler { 223 PragmaUnrollHintHandler(const char *name) : PragmaHandler(name) {} 224 void HandlePragma(Preprocessor &PP, PragmaIntroducer Introducer, 225 Token &FirstToken) override; 226 }; 227 228 struct PragmaMSRuntimeChecksHandler : public EmptyPragmaHandler { 229 PragmaMSRuntimeChecksHandler() : EmptyPragmaHandler("runtime_checks") {} 230 }; 231 232 struct PragmaMSIntrinsicHandler : public PragmaHandler { 233 PragmaMSIntrinsicHandler() : PragmaHandler("intrinsic") {} 234 void HandlePragma(Preprocessor &PP, PragmaIntroducer Introducer, 235 Token &FirstToken) override; 236 }; 237 238 struct PragmaMSOptimizeHandler : public PragmaHandler { 239 PragmaMSOptimizeHandler() : PragmaHandler("optimize") {} 240 void HandlePragma(Preprocessor &PP, PragmaIntroducer Introducer, 241 Token &FirstToken) override; 242 }; 243 244 struct PragmaForceCUDAHostDeviceHandler : public PragmaHandler { 245 PragmaForceCUDAHostDeviceHandler(Sema &Actions) 246 : PragmaHandler("force_cuda_host_device"), Actions(Actions) {} 247 void HandlePragma(Preprocessor &PP, PragmaIntroducer Introducer, 248 Token &FirstToken) override; 249 250 private: 251 Sema &Actions; 252 }; 253 254 /// PragmaAttributeHandler - "\#pragma clang attribute ...". 255 struct PragmaAttributeHandler : public PragmaHandler { 256 PragmaAttributeHandler(AttributeFactory &AttrFactory) 257 : PragmaHandler("attribute"), AttributesForPragmaAttribute(AttrFactory) {} 258 void HandlePragma(Preprocessor &PP, PragmaIntroducer Introducer, 259 Token &FirstToken) override; 260 261 /// A pool of attributes that were parsed in \#pragma clang attribute. 262 ParsedAttributes AttributesForPragmaAttribute; 263 }; 264 265 } // end namespace 266 267 void Parser::initializePragmaHandlers() { 268 AlignHandler = std::make_unique<PragmaAlignHandler>(); 269 PP.AddPragmaHandler(AlignHandler.get()); 270 271 GCCVisibilityHandler = std::make_unique<PragmaGCCVisibilityHandler>(); 272 PP.AddPragmaHandler("GCC", GCCVisibilityHandler.get()); 273 274 OptionsHandler = std::make_unique<PragmaOptionsHandler>(); 275 PP.AddPragmaHandler(OptionsHandler.get()); 276 277 PackHandler = std::make_unique<PragmaPackHandler>(); 278 PP.AddPragmaHandler(PackHandler.get()); 279 280 MSStructHandler = std::make_unique<PragmaMSStructHandler>(); 281 PP.AddPragmaHandler(MSStructHandler.get()); 282 283 UnusedHandler = std::make_unique<PragmaUnusedHandler>(); 284 PP.AddPragmaHandler(UnusedHandler.get()); 285 286 WeakHandler = std::make_unique<PragmaWeakHandler>(); 287 PP.AddPragmaHandler(WeakHandler.get()); 288 289 RedefineExtnameHandler = std::make_unique<PragmaRedefineExtnameHandler>(); 290 PP.AddPragmaHandler(RedefineExtnameHandler.get()); 291 292 FPContractHandler = std::make_unique<PragmaFPContractHandler>(); 293 PP.AddPragmaHandler("STDC", FPContractHandler.get()); 294 295 STDCFENVHandler = std::make_unique<PragmaSTDC_FENV_ACCESSHandler>(); 296 PP.AddPragmaHandler("STDC", STDCFENVHandler.get()); 297 298 STDCCXLIMITHandler = std::make_unique<PragmaSTDC_CX_LIMITED_RANGEHandler>(); 299 PP.AddPragmaHandler("STDC", STDCCXLIMITHandler.get()); 300 301 STDCUnknownHandler = std::make_unique<PragmaSTDC_UnknownHandler>(); 302 PP.AddPragmaHandler("STDC", STDCUnknownHandler.get()); 303 304 PCSectionHandler = std::make_unique<PragmaClangSectionHandler>(Actions); 305 PP.AddPragmaHandler("clang", PCSectionHandler.get()); 306 307 if (getLangOpts().OpenCL) { 308 OpenCLExtensionHandler = std::make_unique<PragmaOpenCLExtensionHandler>(); 309 PP.AddPragmaHandler("OPENCL", OpenCLExtensionHandler.get()); 310 311 PP.AddPragmaHandler("OPENCL", FPContractHandler.get()); 312 } 313 if (getLangOpts().OpenMP) 314 OpenMPHandler = std::make_unique<PragmaOpenMPHandler>(); 315 else 316 OpenMPHandler = std::make_unique<PragmaNoOpenMPHandler>(); 317 PP.AddPragmaHandler(OpenMPHandler.get()); 318 319 if (getLangOpts().MicrosoftExt || 320 getTargetInfo().getTriple().isOSBinFormatELF()) { 321 MSCommentHandler = std::make_unique<PragmaCommentHandler>(Actions); 322 PP.AddPragmaHandler(MSCommentHandler.get()); 323 } 324 325 if (getLangOpts().MicrosoftExt) { 326 MSDetectMismatchHandler = 327 std::make_unique<PragmaDetectMismatchHandler>(Actions); 328 PP.AddPragmaHandler(MSDetectMismatchHandler.get()); 329 MSPointersToMembers = std::make_unique<PragmaMSPointersToMembers>(); 330 PP.AddPragmaHandler(MSPointersToMembers.get()); 331 MSVtorDisp = std::make_unique<PragmaMSVtorDisp>(); 332 PP.AddPragmaHandler(MSVtorDisp.get()); 333 MSInitSeg = std::make_unique<PragmaMSPragma>("init_seg"); 334 PP.AddPragmaHandler(MSInitSeg.get()); 335 MSDataSeg = std::make_unique<PragmaMSPragma>("data_seg"); 336 PP.AddPragmaHandler(MSDataSeg.get()); 337 MSBSSSeg = std::make_unique<PragmaMSPragma>("bss_seg"); 338 PP.AddPragmaHandler(MSBSSSeg.get()); 339 MSConstSeg = std::make_unique<PragmaMSPragma>("const_seg"); 340 PP.AddPragmaHandler(MSConstSeg.get()); 341 MSCodeSeg = std::make_unique<PragmaMSPragma>("code_seg"); 342 PP.AddPragmaHandler(MSCodeSeg.get()); 343 MSSection = std::make_unique<PragmaMSPragma>("section"); 344 PP.AddPragmaHandler(MSSection.get()); 345 MSRuntimeChecks = std::make_unique<PragmaMSRuntimeChecksHandler>(); 346 PP.AddPragmaHandler(MSRuntimeChecks.get()); 347 MSIntrinsic = std::make_unique<PragmaMSIntrinsicHandler>(); 348 PP.AddPragmaHandler(MSIntrinsic.get()); 349 MSOptimize = std::make_unique<PragmaMSOptimizeHandler>(); 350 PP.AddPragmaHandler(MSOptimize.get()); 351 } 352 353 if (getLangOpts().CUDA) { 354 CUDAForceHostDeviceHandler = 355 std::make_unique<PragmaForceCUDAHostDeviceHandler>(Actions); 356 PP.AddPragmaHandler("clang", CUDAForceHostDeviceHandler.get()); 357 } 358 359 OptimizeHandler = std::make_unique<PragmaOptimizeHandler>(Actions); 360 PP.AddPragmaHandler("clang", OptimizeHandler.get()); 361 362 LoopHintHandler = std::make_unique<PragmaLoopHintHandler>(); 363 PP.AddPragmaHandler("clang", LoopHintHandler.get()); 364 365 UnrollHintHandler = std::make_unique<PragmaUnrollHintHandler>("unroll"); 366 PP.AddPragmaHandler(UnrollHintHandler.get()); 367 368 NoUnrollHintHandler = std::make_unique<PragmaUnrollHintHandler>("nounroll"); 369 PP.AddPragmaHandler(NoUnrollHintHandler.get()); 370 371 UnrollAndJamHintHandler = 372 std::make_unique<PragmaUnrollHintHandler>("unroll_and_jam"); 373 PP.AddPragmaHandler(UnrollAndJamHintHandler.get()); 374 375 NoUnrollAndJamHintHandler = 376 std::make_unique<PragmaUnrollHintHandler>("nounroll_and_jam"); 377 PP.AddPragmaHandler(NoUnrollAndJamHintHandler.get()); 378 379 FPHandler = std::make_unique<PragmaFPHandler>(); 380 PP.AddPragmaHandler("clang", FPHandler.get()); 381 382 AttributePragmaHandler = 383 std::make_unique<PragmaAttributeHandler>(AttrFactory); 384 PP.AddPragmaHandler("clang", AttributePragmaHandler.get()); 385 } 386 387 void Parser::resetPragmaHandlers() { 388 // Remove the pragma handlers we installed. 389 PP.RemovePragmaHandler(AlignHandler.get()); 390 AlignHandler.reset(); 391 PP.RemovePragmaHandler("GCC", GCCVisibilityHandler.get()); 392 GCCVisibilityHandler.reset(); 393 PP.RemovePragmaHandler(OptionsHandler.get()); 394 OptionsHandler.reset(); 395 PP.RemovePragmaHandler(PackHandler.get()); 396 PackHandler.reset(); 397 PP.RemovePragmaHandler(MSStructHandler.get()); 398 MSStructHandler.reset(); 399 PP.RemovePragmaHandler(UnusedHandler.get()); 400 UnusedHandler.reset(); 401 PP.RemovePragmaHandler(WeakHandler.get()); 402 WeakHandler.reset(); 403 PP.RemovePragmaHandler(RedefineExtnameHandler.get()); 404 RedefineExtnameHandler.reset(); 405 406 if (getLangOpts().OpenCL) { 407 PP.RemovePragmaHandler("OPENCL", OpenCLExtensionHandler.get()); 408 OpenCLExtensionHandler.reset(); 409 PP.RemovePragmaHandler("OPENCL", FPContractHandler.get()); 410 } 411 PP.RemovePragmaHandler(OpenMPHandler.get()); 412 OpenMPHandler.reset(); 413 414 if (getLangOpts().MicrosoftExt || 415 getTargetInfo().getTriple().isOSBinFormatELF()) { 416 PP.RemovePragmaHandler(MSCommentHandler.get()); 417 MSCommentHandler.reset(); 418 } 419 420 PP.RemovePragmaHandler("clang", PCSectionHandler.get()); 421 PCSectionHandler.reset(); 422 423 if (getLangOpts().MicrosoftExt) { 424 PP.RemovePragmaHandler(MSDetectMismatchHandler.get()); 425 MSDetectMismatchHandler.reset(); 426 PP.RemovePragmaHandler(MSPointersToMembers.get()); 427 MSPointersToMembers.reset(); 428 PP.RemovePragmaHandler(MSVtorDisp.get()); 429 MSVtorDisp.reset(); 430 PP.RemovePragmaHandler(MSInitSeg.get()); 431 MSInitSeg.reset(); 432 PP.RemovePragmaHandler(MSDataSeg.get()); 433 MSDataSeg.reset(); 434 PP.RemovePragmaHandler(MSBSSSeg.get()); 435 MSBSSSeg.reset(); 436 PP.RemovePragmaHandler(MSConstSeg.get()); 437 MSConstSeg.reset(); 438 PP.RemovePragmaHandler(MSCodeSeg.get()); 439 MSCodeSeg.reset(); 440 PP.RemovePragmaHandler(MSSection.get()); 441 MSSection.reset(); 442 PP.RemovePragmaHandler(MSRuntimeChecks.get()); 443 MSRuntimeChecks.reset(); 444 PP.RemovePragmaHandler(MSIntrinsic.get()); 445 MSIntrinsic.reset(); 446 PP.RemovePragmaHandler(MSOptimize.get()); 447 MSOptimize.reset(); 448 } 449 450 if (getLangOpts().CUDA) { 451 PP.RemovePragmaHandler("clang", CUDAForceHostDeviceHandler.get()); 452 CUDAForceHostDeviceHandler.reset(); 453 } 454 455 PP.RemovePragmaHandler("STDC", FPContractHandler.get()); 456 FPContractHandler.reset(); 457 458 PP.RemovePragmaHandler("STDC", STDCFENVHandler.get()); 459 STDCFENVHandler.reset(); 460 461 PP.RemovePragmaHandler("STDC", STDCCXLIMITHandler.get()); 462 STDCCXLIMITHandler.reset(); 463 464 PP.RemovePragmaHandler("STDC", STDCUnknownHandler.get()); 465 STDCUnknownHandler.reset(); 466 467 PP.RemovePragmaHandler("clang", OptimizeHandler.get()); 468 OptimizeHandler.reset(); 469 470 PP.RemovePragmaHandler("clang", LoopHintHandler.get()); 471 LoopHintHandler.reset(); 472 473 PP.RemovePragmaHandler(UnrollHintHandler.get()); 474 UnrollHintHandler.reset(); 475 476 PP.RemovePragmaHandler(NoUnrollHintHandler.get()); 477 NoUnrollHintHandler.reset(); 478 479 PP.RemovePragmaHandler(UnrollAndJamHintHandler.get()); 480 UnrollAndJamHintHandler.reset(); 481 482 PP.RemovePragmaHandler(NoUnrollAndJamHintHandler.get()); 483 NoUnrollAndJamHintHandler.reset(); 484 485 PP.RemovePragmaHandler("clang", FPHandler.get()); 486 FPHandler.reset(); 487 488 PP.RemovePragmaHandler("clang", AttributePragmaHandler.get()); 489 AttributePragmaHandler.reset(); 490 } 491 492 /// Handle the annotation token produced for #pragma unused(...) 493 /// 494 /// Each annot_pragma_unused is followed by the argument token so e.g. 495 /// "#pragma unused(x,y)" becomes: 496 /// annot_pragma_unused 'x' annot_pragma_unused 'y' 497 void Parser::HandlePragmaUnused() { 498 assert(Tok.is(tok::annot_pragma_unused)); 499 SourceLocation UnusedLoc = ConsumeAnnotationToken(); 500 Actions.ActOnPragmaUnused(Tok, getCurScope(), UnusedLoc); 501 ConsumeToken(); // The argument token. 502 } 503 504 void Parser::HandlePragmaVisibility() { 505 assert(Tok.is(tok::annot_pragma_vis)); 506 const IdentifierInfo *VisType = 507 static_cast<IdentifierInfo *>(Tok.getAnnotationValue()); 508 SourceLocation VisLoc = ConsumeAnnotationToken(); 509 Actions.ActOnPragmaVisibility(VisType, VisLoc); 510 } 511 512 namespace { 513 struct PragmaPackInfo { 514 Sema::PragmaMsStackAction Action; 515 StringRef SlotLabel; 516 Token Alignment; 517 }; 518 } // end anonymous namespace 519 520 void Parser::HandlePragmaPack() { 521 assert(Tok.is(tok::annot_pragma_pack)); 522 PragmaPackInfo *Info = 523 static_cast<PragmaPackInfo *>(Tok.getAnnotationValue()); 524 SourceLocation PragmaLoc = Tok.getLocation(); 525 ExprResult Alignment; 526 if (Info->Alignment.is(tok::numeric_constant)) { 527 Alignment = Actions.ActOnNumericConstant(Info->Alignment); 528 if (Alignment.isInvalid()) { 529 ConsumeAnnotationToken(); 530 return; 531 } 532 } 533 Actions.ActOnPragmaPack(PragmaLoc, Info->Action, Info->SlotLabel, 534 Alignment.get()); 535 // Consume the token after processing the pragma to enable pragma-specific 536 // #include warnings. 537 ConsumeAnnotationToken(); 538 } 539 540 void Parser::HandlePragmaMSStruct() { 541 assert(Tok.is(tok::annot_pragma_msstruct)); 542 PragmaMSStructKind Kind = static_cast<PragmaMSStructKind>( 543 reinterpret_cast<uintptr_t>(Tok.getAnnotationValue())); 544 Actions.ActOnPragmaMSStruct(Kind); 545 ConsumeAnnotationToken(); 546 } 547 548 void Parser::HandlePragmaAlign() { 549 assert(Tok.is(tok::annot_pragma_align)); 550 Sema::PragmaOptionsAlignKind Kind = 551 static_cast<Sema::PragmaOptionsAlignKind>( 552 reinterpret_cast<uintptr_t>(Tok.getAnnotationValue())); 553 Actions.ActOnPragmaOptionsAlign(Kind, Tok.getLocation()); 554 // Consume the token after processing the pragma to enable pragma-specific 555 // #include warnings. 556 ConsumeAnnotationToken(); 557 } 558 559 void Parser::HandlePragmaDump() { 560 assert(Tok.is(tok::annot_pragma_dump)); 561 IdentifierInfo *II = 562 reinterpret_cast<IdentifierInfo *>(Tok.getAnnotationValue()); 563 Actions.ActOnPragmaDump(getCurScope(), Tok.getLocation(), II); 564 ConsumeAnnotationToken(); 565 } 566 567 void Parser::HandlePragmaWeak() { 568 assert(Tok.is(tok::annot_pragma_weak)); 569 SourceLocation PragmaLoc = ConsumeAnnotationToken(); 570 Actions.ActOnPragmaWeakID(Tok.getIdentifierInfo(), PragmaLoc, 571 Tok.getLocation()); 572 ConsumeToken(); // The weak name. 573 } 574 575 void Parser::HandlePragmaWeakAlias() { 576 assert(Tok.is(tok::annot_pragma_weakalias)); 577 SourceLocation PragmaLoc = ConsumeAnnotationToken(); 578 IdentifierInfo *WeakName = Tok.getIdentifierInfo(); 579 SourceLocation WeakNameLoc = Tok.getLocation(); 580 ConsumeToken(); 581 IdentifierInfo *AliasName = Tok.getIdentifierInfo(); 582 SourceLocation AliasNameLoc = Tok.getLocation(); 583 ConsumeToken(); 584 Actions.ActOnPragmaWeakAlias(WeakName, AliasName, PragmaLoc, 585 WeakNameLoc, AliasNameLoc); 586 587 } 588 589 void Parser::HandlePragmaRedefineExtname() { 590 assert(Tok.is(tok::annot_pragma_redefine_extname)); 591 SourceLocation RedefLoc = ConsumeAnnotationToken(); 592 IdentifierInfo *RedefName = Tok.getIdentifierInfo(); 593 SourceLocation RedefNameLoc = Tok.getLocation(); 594 ConsumeToken(); 595 IdentifierInfo *AliasName = Tok.getIdentifierInfo(); 596 SourceLocation AliasNameLoc = Tok.getLocation(); 597 ConsumeToken(); 598 Actions.ActOnPragmaRedefineExtname(RedefName, AliasName, RedefLoc, 599 RedefNameLoc, AliasNameLoc); 600 } 601 602 void Parser::HandlePragmaFPContract() { 603 assert(Tok.is(tok::annot_pragma_fp_contract)); 604 tok::OnOffSwitch OOS = 605 static_cast<tok::OnOffSwitch>( 606 reinterpret_cast<uintptr_t>(Tok.getAnnotationValue())); 607 608 LangOptions::FPContractModeKind FPC; 609 switch (OOS) { 610 case tok::OOS_ON: 611 FPC = LangOptions::FPC_On; 612 break; 613 case tok::OOS_OFF: 614 FPC = LangOptions::FPC_Off; 615 break; 616 case tok::OOS_DEFAULT: 617 FPC = getLangOpts().getDefaultFPContractMode(); 618 break; 619 } 620 621 Actions.ActOnPragmaFPContract(FPC); 622 ConsumeAnnotationToken(); 623 } 624 625 void Parser::HandlePragmaFEnvAccess() { 626 assert(Tok.is(tok::annot_pragma_fenv_access)); 627 tok::OnOffSwitch OOS = 628 static_cast<tok::OnOffSwitch>( 629 reinterpret_cast<uintptr_t>(Tok.getAnnotationValue())); 630 631 LangOptions::FEnvAccessModeKind FPC; 632 switch (OOS) { 633 case tok::OOS_ON: 634 FPC = LangOptions::FEA_On; 635 break; 636 case tok::OOS_OFF: 637 FPC = LangOptions::FEA_Off; 638 break; 639 case tok::OOS_DEFAULT: // FIXME: Add this cli option when it makes sense. 640 FPC = LangOptions::FEA_Off; 641 break; 642 } 643 644 Actions.ActOnPragmaFEnvAccess(FPC); 645 ConsumeAnnotationToken(); 646 } 647 648 649 StmtResult Parser::HandlePragmaCaptured() 650 { 651 assert(Tok.is(tok::annot_pragma_captured)); 652 ConsumeAnnotationToken(); 653 654 if (Tok.isNot(tok::l_brace)) { 655 PP.Diag(Tok, diag::err_expected) << tok::l_brace; 656 return StmtError(); 657 } 658 659 SourceLocation Loc = Tok.getLocation(); 660 661 ParseScope CapturedRegionScope(this, Scope::FnScope | Scope::DeclScope | 662 Scope::CompoundStmtScope); 663 Actions.ActOnCapturedRegionStart(Loc, getCurScope(), CR_Default, 664 /*NumParams=*/1); 665 666 StmtResult R = ParseCompoundStatement(); 667 CapturedRegionScope.Exit(); 668 669 if (R.isInvalid()) { 670 Actions.ActOnCapturedRegionError(); 671 return StmtError(); 672 } 673 674 return Actions.ActOnCapturedRegionEnd(R.get()); 675 } 676 677 namespace { 678 enum OpenCLExtState : char { 679 Disable, Enable, Begin, End 680 }; 681 typedef std::pair<const IdentifierInfo *, OpenCLExtState> OpenCLExtData; 682 } 683 684 void Parser::HandlePragmaOpenCLExtension() { 685 assert(Tok.is(tok::annot_pragma_opencl_extension)); 686 OpenCLExtData *Data = static_cast<OpenCLExtData*>(Tok.getAnnotationValue()); 687 auto State = Data->second; 688 auto Ident = Data->first; 689 SourceLocation NameLoc = Tok.getLocation(); 690 ConsumeAnnotationToken(); 691 692 auto &Opt = Actions.getOpenCLOptions(); 693 auto Name = Ident->getName(); 694 // OpenCL 1.1 9.1: "The all variant sets the behavior for all extensions, 695 // overriding all previously issued extension directives, but only if the 696 // behavior is set to disable." 697 if (Name == "all") { 698 if (State == Disable) { 699 Opt.disableAll(); 700 Opt.enableSupportedCore(getLangOpts()); 701 } else { 702 PP.Diag(NameLoc, diag::warn_pragma_expected_predicate) << 1; 703 } 704 } else if (State == Begin) { 705 if (!Opt.isKnown(Name) || !Opt.isSupported(Name, getLangOpts())) { 706 Opt.support(Name); 707 } 708 Actions.setCurrentOpenCLExtension(Name); 709 } else if (State == End) { 710 if (Name != Actions.getCurrentOpenCLExtension()) 711 PP.Diag(NameLoc, diag::warn_pragma_begin_end_mismatch); 712 Actions.setCurrentOpenCLExtension(""); 713 } else if (!Opt.isKnown(Name)) 714 PP.Diag(NameLoc, diag::warn_pragma_unknown_extension) << Ident; 715 else if (Opt.isSupportedExtension(Name, getLangOpts())) 716 Opt.enable(Name, State == Enable); 717 else if (Opt.isSupportedCore(Name, getLangOpts())) 718 PP.Diag(NameLoc, diag::warn_pragma_extension_is_core) << Ident; 719 else 720 PP.Diag(NameLoc, diag::warn_pragma_unsupported_extension) << Ident; 721 } 722 723 void Parser::HandlePragmaMSPointersToMembers() { 724 assert(Tok.is(tok::annot_pragma_ms_pointers_to_members)); 725 LangOptions::PragmaMSPointersToMembersKind RepresentationMethod = 726 static_cast<LangOptions::PragmaMSPointersToMembersKind>( 727 reinterpret_cast<uintptr_t>(Tok.getAnnotationValue())); 728 SourceLocation PragmaLoc = ConsumeAnnotationToken(); 729 Actions.ActOnPragmaMSPointersToMembers(RepresentationMethod, PragmaLoc); 730 } 731 732 void Parser::HandlePragmaMSVtorDisp() { 733 assert(Tok.is(tok::annot_pragma_ms_vtordisp)); 734 uintptr_t Value = reinterpret_cast<uintptr_t>(Tok.getAnnotationValue()); 735 Sema::PragmaMsStackAction Action = 736 static_cast<Sema::PragmaMsStackAction>((Value >> 16) & 0xFFFF); 737 MSVtorDispMode Mode = MSVtorDispMode(Value & 0xFFFF); 738 SourceLocation PragmaLoc = ConsumeAnnotationToken(); 739 Actions.ActOnPragmaMSVtorDisp(Action, PragmaLoc, Mode); 740 } 741 742 void Parser::HandlePragmaMSPragma() { 743 assert(Tok.is(tok::annot_pragma_ms_pragma)); 744 // Grab the tokens out of the annotation and enter them into the stream. 745 auto TheTokens = 746 (std::pair<std::unique_ptr<Token[]>, size_t> *)Tok.getAnnotationValue(); 747 PP.EnterTokenStream(std::move(TheTokens->first), TheTokens->second, true, 748 /*IsReinject=*/true); 749 SourceLocation PragmaLocation = ConsumeAnnotationToken(); 750 assert(Tok.isAnyIdentifier()); 751 StringRef PragmaName = Tok.getIdentifierInfo()->getName(); 752 PP.Lex(Tok); // pragma kind 753 754 // Figure out which #pragma we're dealing with. The switch has no default 755 // because lex shouldn't emit the annotation token for unrecognized pragmas. 756 typedef bool (Parser::*PragmaHandler)(StringRef, SourceLocation); 757 PragmaHandler Handler = llvm::StringSwitch<PragmaHandler>(PragmaName) 758 .Case("data_seg", &Parser::HandlePragmaMSSegment) 759 .Case("bss_seg", &Parser::HandlePragmaMSSegment) 760 .Case("const_seg", &Parser::HandlePragmaMSSegment) 761 .Case("code_seg", &Parser::HandlePragmaMSSegment) 762 .Case("section", &Parser::HandlePragmaMSSection) 763 .Case("init_seg", &Parser::HandlePragmaMSInitSeg); 764 765 if (!(this->*Handler)(PragmaName, PragmaLocation)) { 766 // Pragma handling failed, and has been diagnosed. Slurp up the tokens 767 // until eof (really end of line) to prevent follow-on errors. 768 while (Tok.isNot(tok::eof)) 769 PP.Lex(Tok); 770 PP.Lex(Tok); 771 } 772 } 773 774 bool Parser::HandlePragmaMSSection(StringRef PragmaName, 775 SourceLocation PragmaLocation) { 776 if (Tok.isNot(tok::l_paren)) { 777 PP.Diag(PragmaLocation, diag::warn_pragma_expected_lparen) << PragmaName; 778 return false; 779 } 780 PP.Lex(Tok); // ( 781 // Parsing code for pragma section 782 if (Tok.isNot(tok::string_literal)) { 783 PP.Diag(PragmaLocation, diag::warn_pragma_expected_section_name) 784 << PragmaName; 785 return false; 786 } 787 ExprResult StringResult = ParseStringLiteralExpression(); 788 if (StringResult.isInvalid()) 789 return false; // Already diagnosed. 790 StringLiteral *SegmentName = cast<StringLiteral>(StringResult.get()); 791 if (SegmentName->getCharByteWidth() != 1) { 792 PP.Diag(PragmaLocation, diag::warn_pragma_expected_non_wide_string) 793 << PragmaName; 794 return false; 795 } 796 int SectionFlags = ASTContext::PSF_Read; 797 bool SectionFlagsAreDefault = true; 798 while (Tok.is(tok::comma)) { 799 PP.Lex(Tok); // , 800 // Ignore "long" and "short". 801 // They are undocumented, but widely used, section attributes which appear 802 // to do nothing. 803 if (Tok.is(tok::kw_long) || Tok.is(tok::kw_short)) { 804 PP.Lex(Tok); // long/short 805 continue; 806 } 807 808 if (!Tok.isAnyIdentifier()) { 809 PP.Diag(PragmaLocation, diag::warn_pragma_expected_action_or_r_paren) 810 << PragmaName; 811 return false; 812 } 813 ASTContext::PragmaSectionFlag Flag = 814 llvm::StringSwitch<ASTContext::PragmaSectionFlag>( 815 Tok.getIdentifierInfo()->getName()) 816 .Case("read", ASTContext::PSF_Read) 817 .Case("write", ASTContext::PSF_Write) 818 .Case("execute", ASTContext::PSF_Execute) 819 .Case("shared", ASTContext::PSF_Invalid) 820 .Case("nopage", ASTContext::PSF_Invalid) 821 .Case("nocache", ASTContext::PSF_Invalid) 822 .Case("discard", ASTContext::PSF_Invalid) 823 .Case("remove", ASTContext::PSF_Invalid) 824 .Default(ASTContext::PSF_None); 825 if (Flag == ASTContext::PSF_None || Flag == ASTContext::PSF_Invalid) { 826 PP.Diag(PragmaLocation, Flag == ASTContext::PSF_None 827 ? diag::warn_pragma_invalid_specific_action 828 : diag::warn_pragma_unsupported_action) 829 << PragmaName << Tok.getIdentifierInfo()->getName(); 830 return false; 831 } 832 SectionFlags |= Flag; 833 SectionFlagsAreDefault = false; 834 PP.Lex(Tok); // Identifier 835 } 836 // If no section attributes are specified, the section will be marked as 837 // read/write. 838 if (SectionFlagsAreDefault) 839 SectionFlags |= ASTContext::PSF_Write; 840 if (Tok.isNot(tok::r_paren)) { 841 PP.Diag(PragmaLocation, diag::warn_pragma_expected_rparen) << PragmaName; 842 return false; 843 } 844 PP.Lex(Tok); // ) 845 if (Tok.isNot(tok::eof)) { 846 PP.Diag(PragmaLocation, diag::warn_pragma_extra_tokens_at_eol) 847 << PragmaName; 848 return false; 849 } 850 PP.Lex(Tok); // eof 851 Actions.ActOnPragmaMSSection(PragmaLocation, SectionFlags, SegmentName); 852 return true; 853 } 854 855 bool Parser::HandlePragmaMSSegment(StringRef PragmaName, 856 SourceLocation PragmaLocation) { 857 if (Tok.isNot(tok::l_paren)) { 858 PP.Diag(PragmaLocation, diag::warn_pragma_expected_lparen) << PragmaName; 859 return false; 860 } 861 PP.Lex(Tok); // ( 862 Sema::PragmaMsStackAction Action = Sema::PSK_Reset; 863 StringRef SlotLabel; 864 if (Tok.isAnyIdentifier()) { 865 StringRef PushPop = Tok.getIdentifierInfo()->getName(); 866 if (PushPop == "push") 867 Action = Sema::PSK_Push; 868 else if (PushPop == "pop") 869 Action = Sema::PSK_Pop; 870 else { 871 PP.Diag(PragmaLocation, 872 diag::warn_pragma_expected_section_push_pop_or_name) 873 << PragmaName; 874 return false; 875 } 876 if (Action != Sema::PSK_Reset) { 877 PP.Lex(Tok); // push | pop 878 if (Tok.is(tok::comma)) { 879 PP.Lex(Tok); // , 880 // If we've got a comma, we either need a label or a string. 881 if (Tok.isAnyIdentifier()) { 882 SlotLabel = Tok.getIdentifierInfo()->getName(); 883 PP.Lex(Tok); // identifier 884 if (Tok.is(tok::comma)) 885 PP.Lex(Tok); 886 else if (Tok.isNot(tok::r_paren)) { 887 PP.Diag(PragmaLocation, diag::warn_pragma_expected_punc) 888 << PragmaName; 889 return false; 890 } 891 } 892 } else if (Tok.isNot(tok::r_paren)) { 893 PP.Diag(PragmaLocation, diag::warn_pragma_expected_punc) << PragmaName; 894 return false; 895 } 896 } 897 } 898 // Grab the string literal for our section name. 899 StringLiteral *SegmentName = nullptr; 900 if (Tok.isNot(tok::r_paren)) { 901 if (Tok.isNot(tok::string_literal)) { 902 unsigned DiagID = Action != Sema::PSK_Reset ? !SlotLabel.empty() ? 903 diag::warn_pragma_expected_section_name : 904 diag::warn_pragma_expected_section_label_or_name : 905 diag::warn_pragma_expected_section_push_pop_or_name; 906 PP.Diag(PragmaLocation, DiagID) << PragmaName; 907 return false; 908 } 909 ExprResult StringResult = ParseStringLiteralExpression(); 910 if (StringResult.isInvalid()) 911 return false; // Already diagnosed. 912 SegmentName = cast<StringLiteral>(StringResult.get()); 913 if (SegmentName->getCharByteWidth() != 1) { 914 PP.Diag(PragmaLocation, diag::warn_pragma_expected_non_wide_string) 915 << PragmaName; 916 return false; 917 } 918 // Setting section "" has no effect 919 if (SegmentName->getLength()) 920 Action = (Sema::PragmaMsStackAction)(Action | Sema::PSK_Set); 921 } 922 if (Tok.isNot(tok::r_paren)) { 923 PP.Diag(PragmaLocation, diag::warn_pragma_expected_rparen) << PragmaName; 924 return false; 925 } 926 PP.Lex(Tok); // ) 927 if (Tok.isNot(tok::eof)) { 928 PP.Diag(PragmaLocation, diag::warn_pragma_extra_tokens_at_eol) 929 << PragmaName; 930 return false; 931 } 932 PP.Lex(Tok); // eof 933 Actions.ActOnPragmaMSSeg(PragmaLocation, Action, SlotLabel, 934 SegmentName, PragmaName); 935 return true; 936 } 937 938 // #pragma init_seg({ compiler | lib | user | "section-name" [, func-name]} ) 939 bool Parser::HandlePragmaMSInitSeg(StringRef PragmaName, 940 SourceLocation PragmaLocation) { 941 if (getTargetInfo().getTriple().getEnvironment() != llvm::Triple::MSVC) { 942 PP.Diag(PragmaLocation, diag::warn_pragma_init_seg_unsupported_target); 943 return false; 944 } 945 946 if (ExpectAndConsume(tok::l_paren, diag::warn_pragma_expected_lparen, 947 PragmaName)) 948 return false; 949 950 // Parse either the known section names or the string section name. 951 StringLiteral *SegmentName = nullptr; 952 if (Tok.isAnyIdentifier()) { 953 auto *II = Tok.getIdentifierInfo(); 954 StringRef Section = llvm::StringSwitch<StringRef>(II->getName()) 955 .Case("compiler", "\".CRT$XCC\"") 956 .Case("lib", "\".CRT$XCL\"") 957 .Case("user", "\".CRT$XCU\"") 958 .Default(""); 959 960 if (!Section.empty()) { 961 // Pretend the user wrote the appropriate string literal here. 962 Token Toks[1]; 963 Toks[0].startToken(); 964 Toks[0].setKind(tok::string_literal); 965 Toks[0].setLocation(Tok.getLocation()); 966 Toks[0].setLiteralData(Section.data()); 967 Toks[0].setLength(Section.size()); 968 SegmentName = 969 cast<StringLiteral>(Actions.ActOnStringLiteral(Toks, nullptr).get()); 970 PP.Lex(Tok); 971 } 972 } else if (Tok.is(tok::string_literal)) { 973 ExprResult StringResult = ParseStringLiteralExpression(); 974 if (StringResult.isInvalid()) 975 return false; 976 SegmentName = cast<StringLiteral>(StringResult.get()); 977 if (SegmentName->getCharByteWidth() != 1) { 978 PP.Diag(PragmaLocation, diag::warn_pragma_expected_non_wide_string) 979 << PragmaName; 980 return false; 981 } 982 // FIXME: Add support for the '[, func-name]' part of the pragma. 983 } 984 985 if (!SegmentName) { 986 PP.Diag(PragmaLocation, diag::warn_pragma_expected_init_seg) << PragmaName; 987 return false; 988 } 989 990 if (ExpectAndConsume(tok::r_paren, diag::warn_pragma_expected_rparen, 991 PragmaName) || 992 ExpectAndConsume(tok::eof, diag::warn_pragma_extra_tokens_at_eol, 993 PragmaName)) 994 return false; 995 996 Actions.ActOnPragmaMSInitSeg(PragmaLocation, SegmentName); 997 return true; 998 } 999 1000 namespace { 1001 struct PragmaLoopHintInfo { 1002 Token PragmaName; 1003 Token Option; 1004 ArrayRef<Token> Toks; 1005 }; 1006 } // end anonymous namespace 1007 1008 static std::string PragmaLoopHintString(Token PragmaName, Token Option) { 1009 StringRef Str = PragmaName.getIdentifierInfo()->getName(); 1010 std::string ClangLoopStr = (llvm::Twine("clang loop ") + Str).str(); 1011 return llvm::StringSwitch<StringRef>(Str) 1012 .Case("loop", ClangLoopStr) 1013 .Case("unroll_and_jam", Str) 1014 .Case("unroll", Str) 1015 .Default(""); 1016 } 1017 1018 bool Parser::HandlePragmaLoopHint(LoopHint &Hint) { 1019 assert(Tok.is(tok::annot_pragma_loop_hint)); 1020 PragmaLoopHintInfo *Info = 1021 static_cast<PragmaLoopHintInfo *>(Tok.getAnnotationValue()); 1022 1023 IdentifierInfo *PragmaNameInfo = Info->PragmaName.getIdentifierInfo(); 1024 Hint.PragmaNameLoc = IdentifierLoc::create( 1025 Actions.Context, Info->PragmaName.getLocation(), PragmaNameInfo); 1026 1027 // It is possible that the loop hint has no option identifier, such as 1028 // #pragma unroll(4). 1029 IdentifierInfo *OptionInfo = Info->Option.is(tok::identifier) 1030 ? Info->Option.getIdentifierInfo() 1031 : nullptr; 1032 Hint.OptionLoc = IdentifierLoc::create( 1033 Actions.Context, Info->Option.getLocation(), OptionInfo); 1034 1035 llvm::ArrayRef<Token> Toks = Info->Toks; 1036 1037 // Return a valid hint if pragma unroll or nounroll were specified 1038 // without an argument. 1039 auto IsLoopHint = llvm::StringSwitch<bool>(PragmaNameInfo->getName()) 1040 .Cases("unroll", "nounroll", "unroll_and_jam", 1041 "nounroll_and_jam", true) 1042 .Default(false); 1043 1044 if (Toks.empty() && IsLoopHint) { 1045 ConsumeAnnotationToken(); 1046 Hint.Range = Info->PragmaName.getLocation(); 1047 return true; 1048 } 1049 1050 // The constant expression is always followed by an eof token, which increases 1051 // the TokSize by 1. 1052 assert(!Toks.empty() && 1053 "PragmaLoopHintInfo::Toks must contain at least one token."); 1054 1055 // If no option is specified the argument is assumed to be a constant expr. 1056 bool OptionUnroll = false; 1057 bool OptionUnrollAndJam = false; 1058 bool OptionDistribute = false; 1059 bool OptionPipelineDisabled = false; 1060 bool StateOption = false; 1061 if (OptionInfo) { // Pragma Unroll does not specify an option. 1062 OptionUnroll = OptionInfo->isStr("unroll"); 1063 OptionUnrollAndJam = OptionInfo->isStr("unroll_and_jam"); 1064 OptionDistribute = OptionInfo->isStr("distribute"); 1065 OptionPipelineDisabled = OptionInfo->isStr("pipeline"); 1066 StateOption = llvm::StringSwitch<bool>(OptionInfo->getName()) 1067 .Case("vectorize", true) 1068 .Case("interleave", true) 1069 .Case("vectorize_predicate", true) 1070 .Default(false) || 1071 OptionUnroll || OptionUnrollAndJam || OptionDistribute || 1072 OptionPipelineDisabled; 1073 } 1074 1075 bool AssumeSafetyArg = !OptionUnroll && !OptionUnrollAndJam && 1076 !OptionDistribute && !OptionPipelineDisabled; 1077 // Verify loop hint has an argument. 1078 if (Toks[0].is(tok::eof)) { 1079 ConsumeAnnotationToken(); 1080 Diag(Toks[0].getLocation(), diag::err_pragma_loop_missing_argument) 1081 << /*StateArgument=*/StateOption 1082 << /*FullKeyword=*/(OptionUnroll || OptionUnrollAndJam) 1083 << /*AssumeSafetyKeyword=*/AssumeSafetyArg; 1084 return false; 1085 } 1086 1087 // Validate the argument. 1088 if (StateOption) { 1089 ConsumeAnnotationToken(); 1090 SourceLocation StateLoc = Toks[0].getLocation(); 1091 IdentifierInfo *StateInfo = Toks[0].getIdentifierInfo(); 1092 1093 bool Valid = StateInfo && 1094 llvm::StringSwitch<bool>(StateInfo->getName()) 1095 .Case("disable", true) 1096 .Case("enable", !OptionPipelineDisabled) 1097 .Case("full", OptionUnroll || OptionUnrollAndJam) 1098 .Case("assume_safety", AssumeSafetyArg) 1099 .Default(false); 1100 if (!Valid) { 1101 if (OptionPipelineDisabled) { 1102 Diag(Toks[0].getLocation(), diag::err_pragma_pipeline_invalid_keyword); 1103 } else { 1104 Diag(Toks[0].getLocation(), diag::err_pragma_invalid_keyword) 1105 << /*FullKeyword=*/(OptionUnroll || OptionUnrollAndJam) 1106 << /*AssumeSafetyKeyword=*/AssumeSafetyArg; 1107 } 1108 return false; 1109 } 1110 if (Toks.size() > 2) 1111 Diag(Tok.getLocation(), diag::warn_pragma_extra_tokens_at_eol) 1112 << PragmaLoopHintString(Info->PragmaName, Info->Option); 1113 Hint.StateLoc = IdentifierLoc::create(Actions.Context, StateLoc, StateInfo); 1114 } else { 1115 // Enter constant expression including eof terminator into token stream. 1116 PP.EnterTokenStream(Toks, /*DisableMacroExpansion=*/false, 1117 /*IsReinject=*/false); 1118 ConsumeAnnotationToken(); 1119 1120 ExprResult R = ParseConstantExpression(); 1121 1122 // Tokens following an error in an ill-formed constant expression will 1123 // remain in the token stream and must be removed. 1124 if (Tok.isNot(tok::eof)) { 1125 Diag(Tok.getLocation(), diag::warn_pragma_extra_tokens_at_eol) 1126 << PragmaLoopHintString(Info->PragmaName, Info->Option); 1127 while (Tok.isNot(tok::eof)) 1128 ConsumeAnyToken(); 1129 } 1130 1131 ConsumeToken(); // Consume the constant expression eof terminator. 1132 1133 if (R.isInvalid() || 1134 Actions.CheckLoopHintExpr(R.get(), Toks[0].getLocation())) 1135 return false; 1136 1137 // Argument is a constant expression with an integer type. 1138 Hint.ValueExpr = R.get(); 1139 } 1140 1141 Hint.Range = SourceRange(Info->PragmaName.getLocation(), 1142 Info->Toks.back().getLocation()); 1143 return true; 1144 } 1145 1146 namespace { 1147 struct PragmaAttributeInfo { 1148 enum ActionType { Push, Pop, Attribute }; 1149 ParsedAttributes &Attributes; 1150 ActionType Action; 1151 const IdentifierInfo *Namespace = nullptr; 1152 ArrayRef<Token> Tokens; 1153 1154 PragmaAttributeInfo(ParsedAttributes &Attributes) : Attributes(Attributes) {} 1155 }; 1156 1157 #include "clang/Parse/AttrSubMatchRulesParserStringSwitches.inc" 1158 1159 } // end anonymous namespace 1160 1161 static StringRef getIdentifier(const Token &Tok) { 1162 if (Tok.is(tok::identifier)) 1163 return Tok.getIdentifierInfo()->getName(); 1164 const char *S = tok::getKeywordSpelling(Tok.getKind()); 1165 if (!S) 1166 return ""; 1167 return S; 1168 } 1169 1170 static bool isAbstractAttrMatcherRule(attr::SubjectMatchRule Rule) { 1171 using namespace attr; 1172 switch (Rule) { 1173 #define ATTR_MATCH_RULE(Value, Spelling, IsAbstract) \ 1174 case Value: \ 1175 return IsAbstract; 1176 #include "clang/Basic/AttrSubMatchRulesList.inc" 1177 } 1178 llvm_unreachable("Invalid attribute subject match rule"); 1179 return false; 1180 } 1181 1182 static void diagnoseExpectedAttributeSubjectSubRule( 1183 Parser &PRef, attr::SubjectMatchRule PrimaryRule, StringRef PrimaryRuleName, 1184 SourceLocation SubRuleLoc) { 1185 auto Diagnostic = 1186 PRef.Diag(SubRuleLoc, 1187 diag::err_pragma_attribute_expected_subject_sub_identifier) 1188 << PrimaryRuleName; 1189 if (const char *SubRules = validAttributeSubjectMatchSubRules(PrimaryRule)) 1190 Diagnostic << /*SubRulesSupported=*/1 << SubRules; 1191 else 1192 Diagnostic << /*SubRulesSupported=*/0; 1193 } 1194 1195 static void diagnoseUnknownAttributeSubjectSubRule( 1196 Parser &PRef, attr::SubjectMatchRule PrimaryRule, StringRef PrimaryRuleName, 1197 StringRef SubRuleName, SourceLocation SubRuleLoc) { 1198 1199 auto Diagnostic = 1200 PRef.Diag(SubRuleLoc, diag::err_pragma_attribute_unknown_subject_sub_rule) 1201 << SubRuleName << PrimaryRuleName; 1202 if (const char *SubRules = validAttributeSubjectMatchSubRules(PrimaryRule)) 1203 Diagnostic << /*SubRulesSupported=*/1 << SubRules; 1204 else 1205 Diagnostic << /*SubRulesSupported=*/0; 1206 } 1207 1208 bool Parser::ParsePragmaAttributeSubjectMatchRuleSet( 1209 attr::ParsedSubjectMatchRuleSet &SubjectMatchRules, SourceLocation &AnyLoc, 1210 SourceLocation &LastMatchRuleEndLoc) { 1211 bool IsAny = false; 1212 BalancedDelimiterTracker AnyParens(*this, tok::l_paren); 1213 if (getIdentifier(Tok) == "any") { 1214 AnyLoc = ConsumeToken(); 1215 IsAny = true; 1216 if (AnyParens.expectAndConsume()) 1217 return true; 1218 } 1219 1220 do { 1221 // Parse the subject matcher rule. 1222 StringRef Name = getIdentifier(Tok); 1223 if (Name.empty()) { 1224 Diag(Tok, diag::err_pragma_attribute_expected_subject_identifier); 1225 return true; 1226 } 1227 std::pair<Optional<attr::SubjectMatchRule>, 1228 Optional<attr::SubjectMatchRule> (*)(StringRef, bool)> 1229 Rule = isAttributeSubjectMatchRule(Name); 1230 if (!Rule.first) { 1231 Diag(Tok, diag::err_pragma_attribute_unknown_subject_rule) << Name; 1232 return true; 1233 } 1234 attr::SubjectMatchRule PrimaryRule = *Rule.first; 1235 SourceLocation RuleLoc = ConsumeToken(); 1236 1237 BalancedDelimiterTracker Parens(*this, tok::l_paren); 1238 if (isAbstractAttrMatcherRule(PrimaryRule)) { 1239 if (Parens.expectAndConsume()) 1240 return true; 1241 } else if (Parens.consumeOpen()) { 1242 if (!SubjectMatchRules 1243 .insert( 1244 std::make_pair(PrimaryRule, SourceRange(RuleLoc, RuleLoc))) 1245 .second) 1246 Diag(RuleLoc, diag::err_pragma_attribute_duplicate_subject) 1247 << Name 1248 << FixItHint::CreateRemoval(SourceRange( 1249 RuleLoc, Tok.is(tok::comma) ? Tok.getLocation() : RuleLoc)); 1250 LastMatchRuleEndLoc = RuleLoc; 1251 continue; 1252 } 1253 1254 // Parse the sub-rules. 1255 StringRef SubRuleName = getIdentifier(Tok); 1256 if (SubRuleName.empty()) { 1257 diagnoseExpectedAttributeSubjectSubRule(*this, PrimaryRule, Name, 1258 Tok.getLocation()); 1259 return true; 1260 } 1261 attr::SubjectMatchRule SubRule; 1262 if (SubRuleName == "unless") { 1263 SourceLocation SubRuleLoc = ConsumeToken(); 1264 BalancedDelimiterTracker Parens(*this, tok::l_paren); 1265 if (Parens.expectAndConsume()) 1266 return true; 1267 SubRuleName = getIdentifier(Tok); 1268 if (SubRuleName.empty()) { 1269 diagnoseExpectedAttributeSubjectSubRule(*this, PrimaryRule, Name, 1270 SubRuleLoc); 1271 return true; 1272 } 1273 auto SubRuleOrNone = Rule.second(SubRuleName, /*IsUnless=*/true); 1274 if (!SubRuleOrNone) { 1275 std::string SubRuleUnlessName = "unless(" + SubRuleName.str() + ")"; 1276 diagnoseUnknownAttributeSubjectSubRule(*this, PrimaryRule, Name, 1277 SubRuleUnlessName, SubRuleLoc); 1278 return true; 1279 } 1280 SubRule = *SubRuleOrNone; 1281 ConsumeToken(); 1282 if (Parens.consumeClose()) 1283 return true; 1284 } else { 1285 auto SubRuleOrNone = Rule.second(SubRuleName, /*IsUnless=*/false); 1286 if (!SubRuleOrNone) { 1287 diagnoseUnknownAttributeSubjectSubRule(*this, PrimaryRule, Name, 1288 SubRuleName, Tok.getLocation()); 1289 return true; 1290 } 1291 SubRule = *SubRuleOrNone; 1292 ConsumeToken(); 1293 } 1294 SourceLocation RuleEndLoc = Tok.getLocation(); 1295 LastMatchRuleEndLoc = RuleEndLoc; 1296 if (Parens.consumeClose()) 1297 return true; 1298 if (!SubjectMatchRules 1299 .insert(std::make_pair(SubRule, SourceRange(RuleLoc, RuleEndLoc))) 1300 .second) { 1301 Diag(RuleLoc, diag::err_pragma_attribute_duplicate_subject) 1302 << attr::getSubjectMatchRuleSpelling(SubRule) 1303 << FixItHint::CreateRemoval(SourceRange( 1304 RuleLoc, Tok.is(tok::comma) ? Tok.getLocation() : RuleEndLoc)); 1305 continue; 1306 } 1307 } while (IsAny && TryConsumeToken(tok::comma)); 1308 1309 if (IsAny) 1310 if (AnyParens.consumeClose()) 1311 return true; 1312 1313 return false; 1314 } 1315 1316 namespace { 1317 1318 /// Describes the stage at which attribute subject rule parsing was interrupted. 1319 enum class MissingAttributeSubjectRulesRecoveryPoint { 1320 Comma, 1321 ApplyTo, 1322 Equals, 1323 Any, 1324 None, 1325 }; 1326 1327 MissingAttributeSubjectRulesRecoveryPoint 1328 getAttributeSubjectRulesRecoveryPointForToken(const Token &Tok) { 1329 if (const auto *II = Tok.getIdentifierInfo()) { 1330 if (II->isStr("apply_to")) 1331 return MissingAttributeSubjectRulesRecoveryPoint::ApplyTo; 1332 if (II->isStr("any")) 1333 return MissingAttributeSubjectRulesRecoveryPoint::Any; 1334 } 1335 if (Tok.is(tok::equal)) 1336 return MissingAttributeSubjectRulesRecoveryPoint::Equals; 1337 return MissingAttributeSubjectRulesRecoveryPoint::None; 1338 } 1339 1340 /// Creates a diagnostic for the attribute subject rule parsing diagnostic that 1341 /// suggests the possible attribute subject rules in a fix-it together with 1342 /// any other missing tokens. 1343 DiagnosticBuilder createExpectedAttributeSubjectRulesTokenDiagnostic( 1344 unsigned DiagID, ParsedAttr &Attribute, 1345 MissingAttributeSubjectRulesRecoveryPoint Point, Parser &PRef) { 1346 SourceLocation Loc = PRef.getEndOfPreviousToken(); 1347 if (Loc.isInvalid()) 1348 Loc = PRef.getCurToken().getLocation(); 1349 auto Diagnostic = PRef.Diag(Loc, DiagID); 1350 std::string FixIt; 1351 MissingAttributeSubjectRulesRecoveryPoint EndPoint = 1352 getAttributeSubjectRulesRecoveryPointForToken(PRef.getCurToken()); 1353 if (Point == MissingAttributeSubjectRulesRecoveryPoint::Comma) 1354 FixIt = ", "; 1355 if (Point <= MissingAttributeSubjectRulesRecoveryPoint::ApplyTo && 1356 EndPoint > MissingAttributeSubjectRulesRecoveryPoint::ApplyTo) 1357 FixIt += "apply_to"; 1358 if (Point <= MissingAttributeSubjectRulesRecoveryPoint::Equals && 1359 EndPoint > MissingAttributeSubjectRulesRecoveryPoint::Equals) 1360 FixIt += " = "; 1361 SourceRange FixItRange(Loc); 1362 if (EndPoint == MissingAttributeSubjectRulesRecoveryPoint::None) { 1363 // Gather the subject match rules that are supported by the attribute. 1364 SmallVector<std::pair<attr::SubjectMatchRule, bool>, 4> SubjectMatchRuleSet; 1365 Attribute.getMatchRules(PRef.getLangOpts(), SubjectMatchRuleSet); 1366 if (SubjectMatchRuleSet.empty()) { 1367 // FIXME: We can emit a "fix-it" with a subject list placeholder when 1368 // placeholders will be supported by the fix-its. 1369 return Diagnostic; 1370 } 1371 FixIt += "any("; 1372 bool NeedsComma = false; 1373 for (const auto &I : SubjectMatchRuleSet) { 1374 // Ensure that the missing rule is reported in the fix-it only when it's 1375 // supported in the current language mode. 1376 if (!I.second) 1377 continue; 1378 if (NeedsComma) 1379 FixIt += ", "; 1380 else 1381 NeedsComma = true; 1382 FixIt += attr::getSubjectMatchRuleSpelling(I.first); 1383 } 1384 FixIt += ")"; 1385 // Check if we need to remove the range 1386 PRef.SkipUntil(tok::eof, Parser::StopBeforeMatch); 1387 FixItRange.setEnd(PRef.getCurToken().getLocation()); 1388 } 1389 if (FixItRange.getBegin() == FixItRange.getEnd()) 1390 Diagnostic << FixItHint::CreateInsertion(FixItRange.getBegin(), FixIt); 1391 else 1392 Diagnostic << FixItHint::CreateReplacement( 1393 CharSourceRange::getCharRange(FixItRange), FixIt); 1394 return Diagnostic; 1395 } 1396 1397 } // end anonymous namespace 1398 1399 void Parser::HandlePragmaAttribute() { 1400 assert(Tok.is(tok::annot_pragma_attribute) && 1401 "Expected #pragma attribute annotation token"); 1402 SourceLocation PragmaLoc = Tok.getLocation(); 1403 auto *Info = static_cast<PragmaAttributeInfo *>(Tok.getAnnotationValue()); 1404 if (Info->Action == PragmaAttributeInfo::Pop) { 1405 ConsumeAnnotationToken(); 1406 Actions.ActOnPragmaAttributePop(PragmaLoc, Info->Namespace); 1407 return; 1408 } 1409 // Parse the actual attribute with its arguments. 1410 assert((Info->Action == PragmaAttributeInfo::Push || 1411 Info->Action == PragmaAttributeInfo::Attribute) && 1412 "Unexpected #pragma attribute command"); 1413 1414 if (Info->Action == PragmaAttributeInfo::Push && Info->Tokens.empty()) { 1415 ConsumeAnnotationToken(); 1416 Actions.ActOnPragmaAttributeEmptyPush(PragmaLoc, Info->Namespace); 1417 return; 1418 } 1419 1420 PP.EnterTokenStream(Info->Tokens, /*DisableMacroExpansion=*/false, 1421 /*IsReinject=*/false); 1422 ConsumeAnnotationToken(); 1423 1424 ParsedAttributes &Attrs = Info->Attributes; 1425 Attrs.clearListOnly(); 1426 1427 auto SkipToEnd = [this]() { 1428 SkipUntil(tok::eof, StopBeforeMatch); 1429 ConsumeToken(); 1430 }; 1431 1432 if (Tok.is(tok::l_square) && NextToken().is(tok::l_square)) { 1433 // Parse the CXX11 style attribute. 1434 ParseCXX11AttributeSpecifier(Attrs); 1435 } else if (Tok.is(tok::kw___attribute)) { 1436 ConsumeToken(); 1437 if (ExpectAndConsume(tok::l_paren, diag::err_expected_lparen_after, 1438 "attribute")) 1439 return SkipToEnd(); 1440 if (ExpectAndConsume(tok::l_paren, diag::err_expected_lparen_after, "(")) 1441 return SkipToEnd(); 1442 1443 if (Tok.isNot(tok::identifier)) { 1444 Diag(Tok, diag::err_pragma_attribute_expected_attribute_name); 1445 SkipToEnd(); 1446 return; 1447 } 1448 IdentifierInfo *AttrName = Tok.getIdentifierInfo(); 1449 SourceLocation AttrNameLoc = ConsumeToken(); 1450 1451 if (Tok.isNot(tok::l_paren)) 1452 Attrs.addNew(AttrName, AttrNameLoc, nullptr, AttrNameLoc, nullptr, 0, 1453 ParsedAttr::AS_GNU); 1454 else 1455 ParseGNUAttributeArgs(AttrName, AttrNameLoc, Attrs, /*EndLoc=*/nullptr, 1456 /*ScopeName=*/nullptr, 1457 /*ScopeLoc=*/SourceLocation(), ParsedAttr::AS_GNU, 1458 /*Declarator=*/nullptr); 1459 1460 if (ExpectAndConsume(tok::r_paren)) 1461 return SkipToEnd(); 1462 if (ExpectAndConsume(tok::r_paren)) 1463 return SkipToEnd(); 1464 } else if (Tok.is(tok::kw___declspec)) { 1465 ParseMicrosoftDeclSpecs(Attrs); 1466 } else { 1467 Diag(Tok, diag::err_pragma_attribute_expected_attribute_syntax); 1468 if (Tok.getIdentifierInfo()) { 1469 // If we suspect that this is an attribute suggest the use of 1470 // '__attribute__'. 1471 if (ParsedAttr::getParsedKind( 1472 Tok.getIdentifierInfo(), /*ScopeName=*/nullptr, 1473 ParsedAttr::AS_GNU) != ParsedAttr::UnknownAttribute) { 1474 SourceLocation InsertStartLoc = Tok.getLocation(); 1475 ConsumeToken(); 1476 if (Tok.is(tok::l_paren)) { 1477 ConsumeAnyToken(); 1478 SkipUntil(tok::r_paren, StopBeforeMatch); 1479 if (Tok.isNot(tok::r_paren)) 1480 return SkipToEnd(); 1481 } 1482 Diag(Tok, diag::note_pragma_attribute_use_attribute_kw) 1483 << FixItHint::CreateInsertion(InsertStartLoc, "__attribute__((") 1484 << FixItHint::CreateInsertion(Tok.getEndLoc(), "))"); 1485 } 1486 } 1487 SkipToEnd(); 1488 return; 1489 } 1490 1491 if (Attrs.empty() || Attrs.begin()->isInvalid()) { 1492 SkipToEnd(); 1493 return; 1494 } 1495 1496 // Ensure that we don't have more than one attribute. 1497 if (Attrs.size() > 1) { 1498 SourceLocation Loc = Attrs[1].getLoc(); 1499 Diag(Loc, diag::err_pragma_attribute_multiple_attributes); 1500 SkipToEnd(); 1501 return; 1502 } 1503 1504 ParsedAttr &Attribute = *Attrs.begin(); 1505 if (!Attribute.isSupportedByPragmaAttribute()) { 1506 Diag(PragmaLoc, diag::err_pragma_attribute_unsupported_attribute) 1507 << Attribute; 1508 SkipToEnd(); 1509 return; 1510 } 1511 1512 // Parse the subject-list. 1513 if (!TryConsumeToken(tok::comma)) { 1514 createExpectedAttributeSubjectRulesTokenDiagnostic( 1515 diag::err_expected, Attribute, 1516 MissingAttributeSubjectRulesRecoveryPoint::Comma, *this) 1517 << tok::comma; 1518 SkipToEnd(); 1519 return; 1520 } 1521 1522 if (Tok.isNot(tok::identifier)) { 1523 createExpectedAttributeSubjectRulesTokenDiagnostic( 1524 diag::err_pragma_attribute_invalid_subject_set_specifier, Attribute, 1525 MissingAttributeSubjectRulesRecoveryPoint::ApplyTo, *this); 1526 SkipToEnd(); 1527 return; 1528 } 1529 const IdentifierInfo *II = Tok.getIdentifierInfo(); 1530 if (!II->isStr("apply_to")) { 1531 createExpectedAttributeSubjectRulesTokenDiagnostic( 1532 diag::err_pragma_attribute_invalid_subject_set_specifier, Attribute, 1533 MissingAttributeSubjectRulesRecoveryPoint::ApplyTo, *this); 1534 SkipToEnd(); 1535 return; 1536 } 1537 ConsumeToken(); 1538 1539 if (!TryConsumeToken(tok::equal)) { 1540 createExpectedAttributeSubjectRulesTokenDiagnostic( 1541 diag::err_expected, Attribute, 1542 MissingAttributeSubjectRulesRecoveryPoint::Equals, *this) 1543 << tok::equal; 1544 SkipToEnd(); 1545 return; 1546 } 1547 1548 attr::ParsedSubjectMatchRuleSet SubjectMatchRules; 1549 SourceLocation AnyLoc, LastMatchRuleEndLoc; 1550 if (ParsePragmaAttributeSubjectMatchRuleSet(SubjectMatchRules, AnyLoc, 1551 LastMatchRuleEndLoc)) { 1552 SkipToEnd(); 1553 return; 1554 } 1555 1556 // Tokens following an ill-formed attribute will remain in the token stream 1557 // and must be removed. 1558 if (Tok.isNot(tok::eof)) { 1559 Diag(Tok, diag::err_pragma_attribute_extra_tokens_after_attribute); 1560 SkipToEnd(); 1561 return; 1562 } 1563 1564 // Consume the eof terminator token. 1565 ConsumeToken(); 1566 1567 // Handle a mixed push/attribute by desurging to a push, then an attribute. 1568 if (Info->Action == PragmaAttributeInfo::Push) 1569 Actions.ActOnPragmaAttributeEmptyPush(PragmaLoc, Info->Namespace); 1570 1571 Actions.ActOnPragmaAttributeAttribute(Attribute, PragmaLoc, 1572 std::move(SubjectMatchRules)); 1573 } 1574 1575 // #pragma GCC visibility comes in two variants: 1576 // 'push' '(' [visibility] ')' 1577 // 'pop' 1578 void PragmaGCCVisibilityHandler::HandlePragma(Preprocessor &PP, 1579 PragmaIntroducer Introducer, 1580 Token &VisTok) { 1581 SourceLocation VisLoc = VisTok.getLocation(); 1582 1583 Token Tok; 1584 PP.LexUnexpandedToken(Tok); 1585 1586 const IdentifierInfo *PushPop = Tok.getIdentifierInfo(); 1587 1588 const IdentifierInfo *VisType; 1589 if (PushPop && PushPop->isStr("pop")) { 1590 VisType = nullptr; 1591 } else if (PushPop && PushPop->isStr("push")) { 1592 PP.LexUnexpandedToken(Tok); 1593 if (Tok.isNot(tok::l_paren)) { 1594 PP.Diag(Tok.getLocation(), diag::warn_pragma_expected_lparen) 1595 << "visibility"; 1596 return; 1597 } 1598 PP.LexUnexpandedToken(Tok); 1599 VisType = Tok.getIdentifierInfo(); 1600 if (!VisType) { 1601 PP.Diag(Tok.getLocation(), diag::warn_pragma_expected_identifier) 1602 << "visibility"; 1603 return; 1604 } 1605 PP.LexUnexpandedToken(Tok); 1606 if (Tok.isNot(tok::r_paren)) { 1607 PP.Diag(Tok.getLocation(), diag::warn_pragma_expected_rparen) 1608 << "visibility"; 1609 return; 1610 } 1611 } else { 1612 PP.Diag(Tok.getLocation(), diag::warn_pragma_expected_identifier) 1613 << "visibility"; 1614 return; 1615 } 1616 SourceLocation EndLoc = Tok.getLocation(); 1617 PP.LexUnexpandedToken(Tok); 1618 if (Tok.isNot(tok::eod)) { 1619 PP.Diag(Tok.getLocation(), diag::warn_pragma_extra_tokens_at_eol) 1620 << "visibility"; 1621 return; 1622 } 1623 1624 auto Toks = std::make_unique<Token[]>(1); 1625 Toks[0].startToken(); 1626 Toks[0].setKind(tok::annot_pragma_vis); 1627 Toks[0].setLocation(VisLoc); 1628 Toks[0].setAnnotationEndLoc(EndLoc); 1629 Toks[0].setAnnotationValue( 1630 const_cast<void *>(static_cast<const void *>(VisType))); 1631 PP.EnterTokenStream(std::move(Toks), 1, /*DisableMacroExpansion=*/true, 1632 /*IsReinject=*/false); 1633 } 1634 1635 // #pragma pack(...) comes in the following delicious flavors: 1636 // pack '(' [integer] ')' 1637 // pack '(' 'show' ')' 1638 // pack '(' ('push' | 'pop') [',' identifier] [, integer] ')' 1639 void PragmaPackHandler::HandlePragma(Preprocessor &PP, 1640 PragmaIntroducer Introducer, 1641 Token &PackTok) { 1642 SourceLocation PackLoc = PackTok.getLocation(); 1643 1644 Token Tok; 1645 PP.Lex(Tok); 1646 if (Tok.isNot(tok::l_paren)) { 1647 PP.Diag(Tok.getLocation(), diag::warn_pragma_expected_lparen) << "pack"; 1648 return; 1649 } 1650 1651 Sema::PragmaMsStackAction Action = Sema::PSK_Reset; 1652 StringRef SlotLabel; 1653 Token Alignment; 1654 Alignment.startToken(); 1655 PP.Lex(Tok); 1656 if (Tok.is(tok::numeric_constant)) { 1657 Alignment = Tok; 1658 1659 PP.Lex(Tok); 1660 1661 // In MSVC/gcc, #pragma pack(4) sets the alignment without affecting 1662 // the push/pop stack. 1663 // In Apple gcc, #pragma pack(4) is equivalent to #pragma pack(push, 4) 1664 Action = 1665 PP.getLangOpts().ApplePragmaPack ? Sema::PSK_Push_Set : Sema::PSK_Set; 1666 } else if (Tok.is(tok::identifier)) { 1667 const IdentifierInfo *II = Tok.getIdentifierInfo(); 1668 if (II->isStr("show")) { 1669 Action = Sema::PSK_Show; 1670 PP.Lex(Tok); 1671 } else { 1672 if (II->isStr("push")) { 1673 Action = Sema::PSK_Push; 1674 } else if (II->isStr("pop")) { 1675 Action = Sema::PSK_Pop; 1676 } else { 1677 PP.Diag(Tok.getLocation(), diag::warn_pragma_invalid_action) << "pack"; 1678 return; 1679 } 1680 PP.Lex(Tok); 1681 1682 if (Tok.is(tok::comma)) { 1683 PP.Lex(Tok); 1684 1685 if (Tok.is(tok::numeric_constant)) { 1686 Action = (Sema::PragmaMsStackAction)(Action | Sema::PSK_Set); 1687 Alignment = Tok; 1688 1689 PP.Lex(Tok); 1690 } else if (Tok.is(tok::identifier)) { 1691 SlotLabel = Tok.getIdentifierInfo()->getName(); 1692 PP.Lex(Tok); 1693 1694 if (Tok.is(tok::comma)) { 1695 PP.Lex(Tok); 1696 1697 if (Tok.isNot(tok::numeric_constant)) { 1698 PP.Diag(Tok.getLocation(), diag::warn_pragma_pack_malformed); 1699 return; 1700 } 1701 1702 Action = (Sema::PragmaMsStackAction)(Action | Sema::PSK_Set); 1703 Alignment = Tok; 1704 1705 PP.Lex(Tok); 1706 } 1707 } else { 1708 PP.Diag(Tok.getLocation(), diag::warn_pragma_pack_malformed); 1709 return; 1710 } 1711 } 1712 } 1713 } else if (PP.getLangOpts().ApplePragmaPack) { 1714 // In MSVC/gcc, #pragma pack() resets the alignment without affecting 1715 // the push/pop stack. 1716 // In Apple gcc #pragma pack() is equivalent to #pragma pack(pop). 1717 Action = Sema::PSK_Pop; 1718 } 1719 1720 if (Tok.isNot(tok::r_paren)) { 1721 PP.Diag(Tok.getLocation(), diag::warn_pragma_expected_rparen) << "pack"; 1722 return; 1723 } 1724 1725 SourceLocation RParenLoc = Tok.getLocation(); 1726 PP.Lex(Tok); 1727 if (Tok.isNot(tok::eod)) { 1728 PP.Diag(Tok.getLocation(), diag::warn_pragma_extra_tokens_at_eol) << "pack"; 1729 return; 1730 } 1731 1732 PragmaPackInfo *Info = 1733 PP.getPreprocessorAllocator().Allocate<PragmaPackInfo>(1); 1734 Info->Action = Action; 1735 Info->SlotLabel = SlotLabel; 1736 Info->Alignment = Alignment; 1737 1738 MutableArrayRef<Token> Toks(PP.getPreprocessorAllocator().Allocate<Token>(1), 1739 1); 1740 Toks[0].startToken(); 1741 Toks[0].setKind(tok::annot_pragma_pack); 1742 Toks[0].setLocation(PackLoc); 1743 Toks[0].setAnnotationEndLoc(RParenLoc); 1744 Toks[0].setAnnotationValue(static_cast<void*>(Info)); 1745 PP.EnterTokenStream(Toks, /*DisableMacroExpansion=*/true, 1746 /*IsReinject=*/false); 1747 } 1748 1749 // #pragma ms_struct on 1750 // #pragma ms_struct off 1751 void PragmaMSStructHandler::HandlePragma(Preprocessor &PP, 1752 PragmaIntroducer Introducer, 1753 Token &MSStructTok) { 1754 PragmaMSStructKind Kind = PMSST_OFF; 1755 1756 Token Tok; 1757 PP.Lex(Tok); 1758 if (Tok.isNot(tok::identifier)) { 1759 PP.Diag(Tok.getLocation(), diag::warn_pragma_ms_struct); 1760 return; 1761 } 1762 SourceLocation EndLoc = Tok.getLocation(); 1763 const IdentifierInfo *II = Tok.getIdentifierInfo(); 1764 if (II->isStr("on")) { 1765 Kind = PMSST_ON; 1766 PP.Lex(Tok); 1767 } 1768 else if (II->isStr("off") || II->isStr("reset")) 1769 PP.Lex(Tok); 1770 else { 1771 PP.Diag(Tok.getLocation(), diag::warn_pragma_ms_struct); 1772 return; 1773 } 1774 1775 if (Tok.isNot(tok::eod)) { 1776 PP.Diag(Tok.getLocation(), diag::warn_pragma_extra_tokens_at_eol) 1777 << "ms_struct"; 1778 return; 1779 } 1780 1781 MutableArrayRef<Token> Toks(PP.getPreprocessorAllocator().Allocate<Token>(1), 1782 1); 1783 Toks[0].startToken(); 1784 Toks[0].setKind(tok::annot_pragma_msstruct); 1785 Toks[0].setLocation(MSStructTok.getLocation()); 1786 Toks[0].setAnnotationEndLoc(EndLoc); 1787 Toks[0].setAnnotationValue(reinterpret_cast<void*>( 1788 static_cast<uintptr_t>(Kind))); 1789 PP.EnterTokenStream(Toks, /*DisableMacroExpansion=*/true, 1790 /*IsReinject=*/false); 1791 } 1792 1793 // #pragma clang section bss="abc" data="" rodata="def" text="" relro="" 1794 void PragmaClangSectionHandler::HandlePragma(Preprocessor &PP, 1795 PragmaIntroducer Introducer, 1796 Token &FirstToken) { 1797 1798 Token Tok; 1799 auto SecKind = Sema::PragmaClangSectionKind::PCSK_Invalid; 1800 1801 PP.Lex(Tok); // eat 'section' 1802 while (Tok.isNot(tok::eod)) { 1803 if (Tok.isNot(tok::identifier)) { 1804 PP.Diag(Tok.getLocation(), diag::err_pragma_expected_clang_section_name) << "clang section"; 1805 return; 1806 } 1807 1808 const IdentifierInfo *SecType = Tok.getIdentifierInfo(); 1809 if (SecType->isStr("bss")) 1810 SecKind = Sema::PragmaClangSectionKind::PCSK_BSS; 1811 else if (SecType->isStr("data")) 1812 SecKind = Sema::PragmaClangSectionKind::PCSK_Data; 1813 else if (SecType->isStr("rodata")) 1814 SecKind = Sema::PragmaClangSectionKind::PCSK_Rodata; 1815 else if (SecType->isStr("relro")) 1816 SecKind = Sema::PragmaClangSectionKind::PCSK_Relro; 1817 else if (SecType->isStr("text")) 1818 SecKind = Sema::PragmaClangSectionKind::PCSK_Text; 1819 else { 1820 PP.Diag(Tok.getLocation(), diag::err_pragma_expected_clang_section_name) << "clang section"; 1821 return; 1822 } 1823 1824 PP.Lex(Tok); // eat ['bss'|'data'|'rodata'|'text'] 1825 if (Tok.isNot(tok::equal)) { 1826 PP.Diag(Tok.getLocation(), diag::err_pragma_clang_section_expected_equal) << SecKind; 1827 return; 1828 } 1829 1830 std::string SecName; 1831 if (!PP.LexStringLiteral(Tok, SecName, "pragma clang section", false)) 1832 return; 1833 1834 Actions.ActOnPragmaClangSection(Tok.getLocation(), 1835 (SecName.size()? Sema::PragmaClangSectionAction::PCSA_Set : 1836 Sema::PragmaClangSectionAction::PCSA_Clear), 1837 SecKind, SecName); 1838 } 1839 } 1840 1841 // #pragma 'align' '=' {'native','natural','mac68k','power','reset'} 1842 // #pragma 'options 'align' '=' {'native','natural','mac68k','power','reset'} 1843 static void ParseAlignPragma(Preprocessor &PP, Token &FirstTok, 1844 bool IsOptions) { 1845 Token Tok; 1846 1847 if (IsOptions) { 1848 PP.Lex(Tok); 1849 if (Tok.isNot(tok::identifier) || 1850 !Tok.getIdentifierInfo()->isStr("align")) { 1851 PP.Diag(Tok.getLocation(), diag::warn_pragma_options_expected_align); 1852 return; 1853 } 1854 } 1855 1856 PP.Lex(Tok); 1857 if (Tok.isNot(tok::equal)) { 1858 PP.Diag(Tok.getLocation(), diag::warn_pragma_align_expected_equal) 1859 << IsOptions; 1860 return; 1861 } 1862 1863 PP.Lex(Tok); 1864 if (Tok.isNot(tok::identifier)) { 1865 PP.Diag(Tok.getLocation(), diag::warn_pragma_expected_identifier) 1866 << (IsOptions ? "options" : "align"); 1867 return; 1868 } 1869 1870 Sema::PragmaOptionsAlignKind Kind = Sema::POAK_Natural; 1871 const IdentifierInfo *II = Tok.getIdentifierInfo(); 1872 if (II->isStr("native")) 1873 Kind = Sema::POAK_Native; 1874 else if (II->isStr("natural")) 1875 Kind = Sema::POAK_Natural; 1876 else if (II->isStr("packed")) 1877 Kind = Sema::POAK_Packed; 1878 else if (II->isStr("power")) 1879 Kind = Sema::POAK_Power; 1880 else if (II->isStr("mac68k")) 1881 Kind = Sema::POAK_Mac68k; 1882 else if (II->isStr("reset")) 1883 Kind = Sema::POAK_Reset; 1884 else { 1885 PP.Diag(Tok.getLocation(), diag::warn_pragma_align_invalid_option) 1886 << IsOptions; 1887 return; 1888 } 1889 1890 SourceLocation EndLoc = Tok.getLocation(); 1891 PP.Lex(Tok); 1892 if (Tok.isNot(tok::eod)) { 1893 PP.Diag(Tok.getLocation(), diag::warn_pragma_extra_tokens_at_eol) 1894 << (IsOptions ? "options" : "align"); 1895 return; 1896 } 1897 1898 MutableArrayRef<Token> Toks(PP.getPreprocessorAllocator().Allocate<Token>(1), 1899 1); 1900 Toks[0].startToken(); 1901 Toks[0].setKind(tok::annot_pragma_align); 1902 Toks[0].setLocation(FirstTok.getLocation()); 1903 Toks[0].setAnnotationEndLoc(EndLoc); 1904 Toks[0].setAnnotationValue(reinterpret_cast<void*>( 1905 static_cast<uintptr_t>(Kind))); 1906 PP.EnterTokenStream(Toks, /*DisableMacroExpansion=*/true, 1907 /*IsReinject=*/false); 1908 } 1909 1910 void PragmaAlignHandler::HandlePragma(Preprocessor &PP, 1911 PragmaIntroducer Introducer, 1912 Token &AlignTok) { 1913 ParseAlignPragma(PP, AlignTok, /*IsOptions=*/false); 1914 } 1915 1916 void PragmaOptionsHandler::HandlePragma(Preprocessor &PP, 1917 PragmaIntroducer Introducer, 1918 Token &OptionsTok) { 1919 ParseAlignPragma(PP, OptionsTok, /*IsOptions=*/true); 1920 } 1921 1922 // #pragma unused(identifier) 1923 void PragmaUnusedHandler::HandlePragma(Preprocessor &PP, 1924 PragmaIntroducer Introducer, 1925 Token &UnusedTok) { 1926 // FIXME: Should we be expanding macros here? My guess is no. 1927 SourceLocation UnusedLoc = UnusedTok.getLocation(); 1928 1929 // Lex the left '('. 1930 Token Tok; 1931 PP.Lex(Tok); 1932 if (Tok.isNot(tok::l_paren)) { 1933 PP.Diag(Tok.getLocation(), diag::warn_pragma_expected_lparen) << "unused"; 1934 return; 1935 } 1936 1937 // Lex the declaration reference(s). 1938 SmallVector<Token, 5> Identifiers; 1939 SourceLocation RParenLoc; 1940 bool LexID = true; 1941 1942 while (true) { 1943 PP.Lex(Tok); 1944 1945 if (LexID) { 1946 if (Tok.is(tok::identifier)) { 1947 Identifiers.push_back(Tok); 1948 LexID = false; 1949 continue; 1950 } 1951 1952 // Illegal token! 1953 PP.Diag(Tok.getLocation(), diag::warn_pragma_unused_expected_var); 1954 return; 1955 } 1956 1957 // We are execting a ')' or a ','. 1958 if (Tok.is(tok::comma)) { 1959 LexID = true; 1960 continue; 1961 } 1962 1963 if (Tok.is(tok::r_paren)) { 1964 RParenLoc = Tok.getLocation(); 1965 break; 1966 } 1967 1968 // Illegal token! 1969 PP.Diag(Tok.getLocation(), diag::warn_pragma_expected_punc) << "unused"; 1970 return; 1971 } 1972 1973 PP.Lex(Tok); 1974 if (Tok.isNot(tok::eod)) { 1975 PP.Diag(Tok.getLocation(), diag::warn_pragma_extra_tokens_at_eol) << 1976 "unused"; 1977 return; 1978 } 1979 1980 // Verify that we have a location for the right parenthesis. 1981 assert(RParenLoc.isValid() && "Valid '#pragma unused' must have ')'"); 1982 assert(!Identifiers.empty() && "Valid '#pragma unused' must have arguments"); 1983 1984 // For each identifier token, insert into the token stream a 1985 // annot_pragma_unused token followed by the identifier token. 1986 // This allows us to cache a "#pragma unused" that occurs inside an inline 1987 // C++ member function. 1988 1989 MutableArrayRef<Token> Toks( 1990 PP.getPreprocessorAllocator().Allocate<Token>(2 * Identifiers.size()), 1991 2 * Identifiers.size()); 1992 for (unsigned i=0; i != Identifiers.size(); i++) { 1993 Token &pragmaUnusedTok = Toks[2*i], &idTok = Toks[2*i+1]; 1994 pragmaUnusedTok.startToken(); 1995 pragmaUnusedTok.setKind(tok::annot_pragma_unused); 1996 pragmaUnusedTok.setLocation(UnusedLoc); 1997 idTok = Identifiers[i]; 1998 } 1999 PP.EnterTokenStream(Toks, /*DisableMacroExpansion=*/true, 2000 /*IsReinject=*/false); 2001 } 2002 2003 // #pragma weak identifier 2004 // #pragma weak identifier '=' identifier 2005 void PragmaWeakHandler::HandlePragma(Preprocessor &PP, 2006 PragmaIntroducer Introducer, 2007 Token &WeakTok) { 2008 SourceLocation WeakLoc = WeakTok.getLocation(); 2009 2010 Token Tok; 2011 PP.Lex(Tok); 2012 if (Tok.isNot(tok::identifier)) { 2013 PP.Diag(Tok.getLocation(), diag::warn_pragma_expected_identifier) << "weak"; 2014 return; 2015 } 2016 2017 Token WeakName = Tok; 2018 bool HasAlias = false; 2019 Token AliasName; 2020 2021 PP.Lex(Tok); 2022 if (Tok.is(tok::equal)) { 2023 HasAlias = true; 2024 PP.Lex(Tok); 2025 if (Tok.isNot(tok::identifier)) { 2026 PP.Diag(Tok.getLocation(), diag::warn_pragma_expected_identifier) 2027 << "weak"; 2028 return; 2029 } 2030 AliasName = Tok; 2031 PP.Lex(Tok); 2032 } 2033 2034 if (Tok.isNot(tok::eod)) { 2035 PP.Diag(Tok.getLocation(), diag::warn_pragma_extra_tokens_at_eol) << "weak"; 2036 return; 2037 } 2038 2039 if (HasAlias) { 2040 MutableArrayRef<Token> Toks( 2041 PP.getPreprocessorAllocator().Allocate<Token>(3), 3); 2042 Token &pragmaUnusedTok = Toks[0]; 2043 pragmaUnusedTok.startToken(); 2044 pragmaUnusedTok.setKind(tok::annot_pragma_weakalias); 2045 pragmaUnusedTok.setLocation(WeakLoc); 2046 pragmaUnusedTok.setAnnotationEndLoc(AliasName.getLocation()); 2047 Toks[1] = WeakName; 2048 Toks[2] = AliasName; 2049 PP.EnterTokenStream(Toks, /*DisableMacroExpansion=*/true, 2050 /*IsReinject=*/false); 2051 } else { 2052 MutableArrayRef<Token> Toks( 2053 PP.getPreprocessorAllocator().Allocate<Token>(2), 2); 2054 Token &pragmaUnusedTok = Toks[0]; 2055 pragmaUnusedTok.startToken(); 2056 pragmaUnusedTok.setKind(tok::annot_pragma_weak); 2057 pragmaUnusedTok.setLocation(WeakLoc); 2058 pragmaUnusedTok.setAnnotationEndLoc(WeakLoc); 2059 Toks[1] = WeakName; 2060 PP.EnterTokenStream(Toks, /*DisableMacroExpansion=*/true, 2061 /*IsReinject=*/false); 2062 } 2063 } 2064 2065 // #pragma redefine_extname identifier identifier 2066 void PragmaRedefineExtnameHandler::HandlePragma(Preprocessor &PP, 2067 PragmaIntroducer Introducer, 2068 Token &RedefToken) { 2069 SourceLocation RedefLoc = RedefToken.getLocation(); 2070 2071 Token Tok; 2072 PP.Lex(Tok); 2073 if (Tok.isNot(tok::identifier)) { 2074 PP.Diag(Tok.getLocation(), diag::warn_pragma_expected_identifier) << 2075 "redefine_extname"; 2076 return; 2077 } 2078 2079 Token RedefName = Tok; 2080 PP.Lex(Tok); 2081 2082 if (Tok.isNot(tok::identifier)) { 2083 PP.Diag(Tok.getLocation(), diag::warn_pragma_expected_identifier) 2084 << "redefine_extname"; 2085 return; 2086 } 2087 2088 Token AliasName = Tok; 2089 PP.Lex(Tok); 2090 2091 if (Tok.isNot(tok::eod)) { 2092 PP.Diag(Tok.getLocation(), diag::warn_pragma_extra_tokens_at_eol) << 2093 "redefine_extname"; 2094 return; 2095 } 2096 2097 MutableArrayRef<Token> Toks(PP.getPreprocessorAllocator().Allocate<Token>(3), 2098 3); 2099 Token &pragmaRedefTok = Toks[0]; 2100 pragmaRedefTok.startToken(); 2101 pragmaRedefTok.setKind(tok::annot_pragma_redefine_extname); 2102 pragmaRedefTok.setLocation(RedefLoc); 2103 pragmaRedefTok.setAnnotationEndLoc(AliasName.getLocation()); 2104 Toks[1] = RedefName; 2105 Toks[2] = AliasName; 2106 PP.EnterTokenStream(Toks, /*DisableMacroExpansion=*/true, 2107 /*IsReinject=*/false); 2108 } 2109 2110 void PragmaFPContractHandler::HandlePragma(Preprocessor &PP, 2111 PragmaIntroducer Introducer, 2112 Token &Tok) { 2113 tok::OnOffSwitch OOS; 2114 if (PP.LexOnOffSwitch(OOS)) 2115 return; 2116 2117 MutableArrayRef<Token> Toks(PP.getPreprocessorAllocator().Allocate<Token>(1), 2118 1); 2119 Toks[0].startToken(); 2120 Toks[0].setKind(tok::annot_pragma_fp_contract); 2121 Toks[0].setLocation(Tok.getLocation()); 2122 Toks[0].setAnnotationEndLoc(Tok.getLocation()); 2123 Toks[0].setAnnotationValue(reinterpret_cast<void*>( 2124 static_cast<uintptr_t>(OOS))); 2125 PP.EnterTokenStream(Toks, /*DisableMacroExpansion=*/true, 2126 /*IsReinject=*/false); 2127 } 2128 2129 void PragmaOpenCLExtensionHandler::HandlePragma(Preprocessor &PP, 2130 PragmaIntroducer Introducer, 2131 Token &Tok) { 2132 PP.LexUnexpandedToken(Tok); 2133 if (Tok.isNot(tok::identifier)) { 2134 PP.Diag(Tok.getLocation(), diag::warn_pragma_expected_identifier) << 2135 "OPENCL"; 2136 return; 2137 } 2138 IdentifierInfo *Ext = Tok.getIdentifierInfo(); 2139 SourceLocation NameLoc = Tok.getLocation(); 2140 2141 PP.Lex(Tok); 2142 if (Tok.isNot(tok::colon)) { 2143 PP.Diag(Tok.getLocation(), diag::warn_pragma_expected_colon) << Ext; 2144 return; 2145 } 2146 2147 PP.Lex(Tok); 2148 if (Tok.isNot(tok::identifier)) { 2149 PP.Diag(Tok.getLocation(), diag::warn_pragma_expected_predicate) << 0; 2150 return; 2151 } 2152 IdentifierInfo *Pred = Tok.getIdentifierInfo(); 2153 2154 OpenCLExtState State; 2155 if (Pred->isStr("enable")) { 2156 State = Enable; 2157 } else if (Pred->isStr("disable")) { 2158 State = Disable; 2159 } else if (Pred->isStr("begin")) 2160 State = Begin; 2161 else if (Pred->isStr("end")) 2162 State = End; 2163 else { 2164 PP.Diag(Tok.getLocation(), diag::warn_pragma_expected_predicate) 2165 << Ext->isStr("all"); 2166 return; 2167 } 2168 SourceLocation StateLoc = Tok.getLocation(); 2169 2170 PP.Lex(Tok); 2171 if (Tok.isNot(tok::eod)) { 2172 PP.Diag(Tok.getLocation(), diag::warn_pragma_extra_tokens_at_eol) << 2173 "OPENCL EXTENSION"; 2174 return; 2175 } 2176 2177 auto Info = PP.getPreprocessorAllocator().Allocate<OpenCLExtData>(1); 2178 Info->first = Ext; 2179 Info->second = State; 2180 MutableArrayRef<Token> Toks(PP.getPreprocessorAllocator().Allocate<Token>(1), 2181 1); 2182 Toks[0].startToken(); 2183 Toks[0].setKind(tok::annot_pragma_opencl_extension); 2184 Toks[0].setLocation(NameLoc); 2185 Toks[0].setAnnotationValue(static_cast<void*>(Info)); 2186 Toks[0].setAnnotationEndLoc(StateLoc); 2187 PP.EnterTokenStream(Toks, /*DisableMacroExpansion=*/true, 2188 /*IsReinject=*/false); 2189 2190 if (PP.getPPCallbacks()) 2191 PP.getPPCallbacks()->PragmaOpenCLExtension(NameLoc, Ext, 2192 StateLoc, State); 2193 } 2194 2195 /// Handle '#pragma omp ...' when OpenMP is disabled. 2196 /// 2197 void PragmaNoOpenMPHandler::HandlePragma(Preprocessor &PP, 2198 PragmaIntroducer Introducer, 2199 Token &FirstTok) { 2200 if (!PP.getDiagnostics().isIgnored(diag::warn_pragma_omp_ignored, 2201 FirstTok.getLocation())) { 2202 PP.Diag(FirstTok, diag::warn_pragma_omp_ignored); 2203 PP.getDiagnostics().setSeverity(diag::warn_pragma_omp_ignored, 2204 diag::Severity::Ignored, SourceLocation()); 2205 } 2206 PP.DiscardUntilEndOfDirective(); 2207 } 2208 2209 /// Handle '#pragma omp ...' when OpenMP is enabled. 2210 /// 2211 void PragmaOpenMPHandler::HandlePragma(Preprocessor &PP, 2212 PragmaIntroducer Introducer, 2213 Token &FirstTok) { 2214 SmallVector<Token, 16> Pragma; 2215 Token Tok; 2216 Tok.startToken(); 2217 Tok.setKind(tok::annot_pragma_openmp); 2218 Tok.setLocation(Introducer.Loc); 2219 2220 while (Tok.isNot(tok::eod) && Tok.isNot(tok::eof)) { 2221 Pragma.push_back(Tok); 2222 PP.Lex(Tok); 2223 if (Tok.is(tok::annot_pragma_openmp)) { 2224 PP.Diag(Tok, diag::err_omp_unexpected_directive) << 0; 2225 unsigned InnerPragmaCnt = 1; 2226 while (InnerPragmaCnt != 0) { 2227 PP.Lex(Tok); 2228 if (Tok.is(tok::annot_pragma_openmp)) 2229 ++InnerPragmaCnt; 2230 else if (Tok.is(tok::annot_pragma_openmp_end)) 2231 --InnerPragmaCnt; 2232 } 2233 PP.Lex(Tok); 2234 } 2235 } 2236 SourceLocation EodLoc = Tok.getLocation(); 2237 Tok.startToken(); 2238 Tok.setKind(tok::annot_pragma_openmp_end); 2239 Tok.setLocation(EodLoc); 2240 Pragma.push_back(Tok); 2241 2242 auto Toks = std::make_unique<Token[]>(Pragma.size()); 2243 std::copy(Pragma.begin(), Pragma.end(), Toks.get()); 2244 PP.EnterTokenStream(std::move(Toks), Pragma.size(), 2245 /*DisableMacroExpansion=*/false, /*IsReinject=*/false); 2246 } 2247 2248 /// Handle '#pragma pointers_to_members' 2249 // The grammar for this pragma is as follows: 2250 // 2251 // <inheritance model> ::= ('single' | 'multiple' | 'virtual') '_inheritance' 2252 // 2253 // #pragma pointers_to_members '(' 'best_case' ')' 2254 // #pragma pointers_to_members '(' 'full_generality' [',' inheritance-model] ')' 2255 // #pragma pointers_to_members '(' inheritance-model ')' 2256 void PragmaMSPointersToMembers::HandlePragma(Preprocessor &PP, 2257 PragmaIntroducer Introducer, 2258 Token &Tok) { 2259 SourceLocation PointersToMembersLoc = Tok.getLocation(); 2260 PP.Lex(Tok); 2261 if (Tok.isNot(tok::l_paren)) { 2262 PP.Diag(PointersToMembersLoc, diag::warn_pragma_expected_lparen) 2263 << "pointers_to_members"; 2264 return; 2265 } 2266 PP.Lex(Tok); 2267 const IdentifierInfo *Arg = Tok.getIdentifierInfo(); 2268 if (!Arg) { 2269 PP.Diag(Tok.getLocation(), diag::warn_pragma_expected_identifier) 2270 << "pointers_to_members"; 2271 return; 2272 } 2273 PP.Lex(Tok); 2274 2275 LangOptions::PragmaMSPointersToMembersKind RepresentationMethod; 2276 if (Arg->isStr("best_case")) { 2277 RepresentationMethod = LangOptions::PPTMK_BestCase; 2278 } else { 2279 if (Arg->isStr("full_generality")) { 2280 if (Tok.is(tok::comma)) { 2281 PP.Lex(Tok); 2282 2283 Arg = Tok.getIdentifierInfo(); 2284 if (!Arg) { 2285 PP.Diag(Tok.getLocation(), 2286 diag::err_pragma_pointers_to_members_unknown_kind) 2287 << Tok.getKind() << /*OnlyInheritanceModels*/ 0; 2288 return; 2289 } 2290 PP.Lex(Tok); 2291 } else if (Tok.is(tok::r_paren)) { 2292 // #pragma pointers_to_members(full_generality) implicitly specifies 2293 // virtual_inheritance. 2294 Arg = nullptr; 2295 RepresentationMethod = LangOptions::PPTMK_FullGeneralityVirtualInheritance; 2296 } else { 2297 PP.Diag(Tok.getLocation(), diag::err_expected_punc) 2298 << "full_generality"; 2299 return; 2300 } 2301 } 2302 2303 if (Arg) { 2304 if (Arg->isStr("single_inheritance")) { 2305 RepresentationMethod = 2306 LangOptions::PPTMK_FullGeneralitySingleInheritance; 2307 } else if (Arg->isStr("multiple_inheritance")) { 2308 RepresentationMethod = 2309 LangOptions::PPTMK_FullGeneralityMultipleInheritance; 2310 } else if (Arg->isStr("virtual_inheritance")) { 2311 RepresentationMethod = 2312 LangOptions::PPTMK_FullGeneralityVirtualInheritance; 2313 } else { 2314 PP.Diag(Tok.getLocation(), 2315 diag::err_pragma_pointers_to_members_unknown_kind) 2316 << Arg << /*HasPointerDeclaration*/ 1; 2317 return; 2318 } 2319 } 2320 } 2321 2322 if (Tok.isNot(tok::r_paren)) { 2323 PP.Diag(Tok.getLocation(), diag::err_expected_rparen_after) 2324 << (Arg ? Arg->getName() : "full_generality"); 2325 return; 2326 } 2327 2328 SourceLocation EndLoc = Tok.getLocation(); 2329 PP.Lex(Tok); 2330 if (Tok.isNot(tok::eod)) { 2331 PP.Diag(Tok.getLocation(), diag::warn_pragma_extra_tokens_at_eol) 2332 << "pointers_to_members"; 2333 return; 2334 } 2335 2336 Token AnnotTok; 2337 AnnotTok.startToken(); 2338 AnnotTok.setKind(tok::annot_pragma_ms_pointers_to_members); 2339 AnnotTok.setLocation(PointersToMembersLoc); 2340 AnnotTok.setAnnotationEndLoc(EndLoc); 2341 AnnotTok.setAnnotationValue( 2342 reinterpret_cast<void *>(static_cast<uintptr_t>(RepresentationMethod))); 2343 PP.EnterToken(AnnotTok, /*IsReinject=*/true); 2344 } 2345 2346 /// Handle '#pragma vtordisp' 2347 // The grammar for this pragma is as follows: 2348 // 2349 // <vtordisp-mode> ::= ('off' | 'on' | '0' | '1' | '2' ) 2350 // 2351 // #pragma vtordisp '(' ['push' ','] vtordisp-mode ')' 2352 // #pragma vtordisp '(' 'pop' ')' 2353 // #pragma vtordisp '(' ')' 2354 void PragmaMSVtorDisp::HandlePragma(Preprocessor &PP, 2355 PragmaIntroducer Introducer, Token &Tok) { 2356 SourceLocation VtorDispLoc = Tok.getLocation(); 2357 PP.Lex(Tok); 2358 if (Tok.isNot(tok::l_paren)) { 2359 PP.Diag(VtorDispLoc, diag::warn_pragma_expected_lparen) << "vtordisp"; 2360 return; 2361 } 2362 PP.Lex(Tok); 2363 2364 Sema::PragmaMsStackAction Action = Sema::PSK_Set; 2365 const IdentifierInfo *II = Tok.getIdentifierInfo(); 2366 if (II) { 2367 if (II->isStr("push")) { 2368 // #pragma vtordisp(push, mode) 2369 PP.Lex(Tok); 2370 if (Tok.isNot(tok::comma)) { 2371 PP.Diag(VtorDispLoc, diag::warn_pragma_expected_punc) << "vtordisp"; 2372 return; 2373 } 2374 PP.Lex(Tok); 2375 Action = Sema::PSK_Push_Set; 2376 // not push, could be on/off 2377 } else if (II->isStr("pop")) { 2378 // #pragma vtordisp(pop) 2379 PP.Lex(Tok); 2380 Action = Sema::PSK_Pop; 2381 } 2382 // not push or pop, could be on/off 2383 } else { 2384 if (Tok.is(tok::r_paren)) { 2385 // #pragma vtordisp() 2386 Action = Sema::PSK_Reset; 2387 } 2388 } 2389 2390 2391 uint64_t Value = 0; 2392 if (Action & Sema::PSK_Push || Action & Sema::PSK_Set) { 2393 const IdentifierInfo *II = Tok.getIdentifierInfo(); 2394 if (II && II->isStr("off")) { 2395 PP.Lex(Tok); 2396 Value = 0; 2397 } else if (II && II->isStr("on")) { 2398 PP.Lex(Tok); 2399 Value = 1; 2400 } else if (Tok.is(tok::numeric_constant) && 2401 PP.parseSimpleIntegerLiteral(Tok, Value)) { 2402 if (Value > 2) { 2403 PP.Diag(Tok.getLocation(), diag::warn_pragma_expected_integer) 2404 << 0 << 2 << "vtordisp"; 2405 return; 2406 } 2407 } else { 2408 PP.Diag(Tok.getLocation(), diag::warn_pragma_invalid_action) 2409 << "vtordisp"; 2410 return; 2411 } 2412 } 2413 2414 // Finish the pragma: ')' $ 2415 if (Tok.isNot(tok::r_paren)) { 2416 PP.Diag(VtorDispLoc, diag::warn_pragma_expected_rparen) << "vtordisp"; 2417 return; 2418 } 2419 SourceLocation EndLoc = Tok.getLocation(); 2420 PP.Lex(Tok); 2421 if (Tok.isNot(tok::eod)) { 2422 PP.Diag(Tok.getLocation(), diag::warn_pragma_extra_tokens_at_eol) 2423 << "vtordisp"; 2424 return; 2425 } 2426 2427 // Enter the annotation. 2428 Token AnnotTok; 2429 AnnotTok.startToken(); 2430 AnnotTok.setKind(tok::annot_pragma_ms_vtordisp); 2431 AnnotTok.setLocation(VtorDispLoc); 2432 AnnotTok.setAnnotationEndLoc(EndLoc); 2433 AnnotTok.setAnnotationValue(reinterpret_cast<void *>( 2434 static_cast<uintptr_t>((Action << 16) | (Value & 0xFFFF)))); 2435 PP.EnterToken(AnnotTok, /*IsReinject=*/false); 2436 } 2437 2438 /// Handle all MS pragmas. Simply forwards the tokens after inserting 2439 /// an annotation token. 2440 void PragmaMSPragma::HandlePragma(Preprocessor &PP, 2441 PragmaIntroducer Introducer, Token &Tok) { 2442 Token EoF, AnnotTok; 2443 EoF.startToken(); 2444 EoF.setKind(tok::eof); 2445 AnnotTok.startToken(); 2446 AnnotTok.setKind(tok::annot_pragma_ms_pragma); 2447 AnnotTok.setLocation(Tok.getLocation()); 2448 AnnotTok.setAnnotationEndLoc(Tok.getLocation()); 2449 SmallVector<Token, 8> TokenVector; 2450 // Suck up all of the tokens before the eod. 2451 for (; Tok.isNot(tok::eod); PP.Lex(Tok)) { 2452 TokenVector.push_back(Tok); 2453 AnnotTok.setAnnotationEndLoc(Tok.getLocation()); 2454 } 2455 // Add a sentinel EoF token to the end of the list. 2456 TokenVector.push_back(EoF); 2457 // We must allocate this array with new because EnterTokenStream is going to 2458 // delete it later. 2459 auto TokenArray = std::make_unique<Token[]>(TokenVector.size()); 2460 std::copy(TokenVector.begin(), TokenVector.end(), TokenArray.get()); 2461 auto Value = new (PP.getPreprocessorAllocator()) 2462 std::pair<std::unique_ptr<Token[]>, size_t>(std::move(TokenArray), 2463 TokenVector.size()); 2464 AnnotTok.setAnnotationValue(Value); 2465 PP.EnterToken(AnnotTok, /*IsReinject*/ false); 2466 } 2467 2468 /// Handle the Microsoft \#pragma detect_mismatch extension. 2469 /// 2470 /// The syntax is: 2471 /// \code 2472 /// #pragma detect_mismatch("name", "value") 2473 /// \endcode 2474 /// Where 'name' and 'value' are quoted strings. The values are embedded in 2475 /// the object file and passed along to the linker. If the linker detects a 2476 /// mismatch in the object file's values for the given name, a LNK2038 error 2477 /// is emitted. See MSDN for more details. 2478 void PragmaDetectMismatchHandler::HandlePragma(Preprocessor &PP, 2479 PragmaIntroducer Introducer, 2480 Token &Tok) { 2481 SourceLocation DetectMismatchLoc = Tok.getLocation(); 2482 PP.Lex(Tok); 2483 if (Tok.isNot(tok::l_paren)) { 2484 PP.Diag(DetectMismatchLoc, diag::err_expected) << tok::l_paren; 2485 return; 2486 } 2487 2488 // Read the name to embed, which must be a string literal. 2489 std::string NameString; 2490 if (!PP.LexStringLiteral(Tok, NameString, 2491 "pragma detect_mismatch", 2492 /*AllowMacroExpansion=*/true)) 2493 return; 2494 2495 // Read the comma followed by a second string literal. 2496 std::string ValueString; 2497 if (Tok.isNot(tok::comma)) { 2498 PP.Diag(Tok.getLocation(), diag::err_pragma_detect_mismatch_malformed); 2499 return; 2500 } 2501 2502 if (!PP.LexStringLiteral(Tok, ValueString, "pragma detect_mismatch", 2503 /*AllowMacroExpansion=*/true)) 2504 return; 2505 2506 if (Tok.isNot(tok::r_paren)) { 2507 PP.Diag(Tok.getLocation(), diag::err_expected) << tok::r_paren; 2508 return; 2509 } 2510 PP.Lex(Tok); // Eat the r_paren. 2511 2512 if (Tok.isNot(tok::eod)) { 2513 PP.Diag(Tok.getLocation(), diag::err_pragma_detect_mismatch_malformed); 2514 return; 2515 } 2516 2517 // If the pragma is lexically sound, notify any interested PPCallbacks. 2518 if (PP.getPPCallbacks()) 2519 PP.getPPCallbacks()->PragmaDetectMismatch(DetectMismatchLoc, NameString, 2520 ValueString); 2521 2522 Actions.ActOnPragmaDetectMismatch(DetectMismatchLoc, NameString, ValueString); 2523 } 2524 2525 /// Handle the microsoft \#pragma comment extension. 2526 /// 2527 /// The syntax is: 2528 /// \code 2529 /// #pragma comment(linker, "foo") 2530 /// \endcode 2531 /// 'linker' is one of five identifiers: compiler, exestr, lib, linker, user. 2532 /// "foo" is a string, which is fully macro expanded, and permits string 2533 /// concatenation, embedded escape characters etc. See MSDN for more details. 2534 void PragmaCommentHandler::HandlePragma(Preprocessor &PP, 2535 PragmaIntroducer Introducer, 2536 Token &Tok) { 2537 SourceLocation CommentLoc = Tok.getLocation(); 2538 PP.Lex(Tok); 2539 if (Tok.isNot(tok::l_paren)) { 2540 PP.Diag(CommentLoc, diag::err_pragma_comment_malformed); 2541 return; 2542 } 2543 2544 // Read the identifier. 2545 PP.Lex(Tok); 2546 if (Tok.isNot(tok::identifier)) { 2547 PP.Diag(CommentLoc, diag::err_pragma_comment_malformed); 2548 return; 2549 } 2550 2551 // Verify that this is one of the 5 whitelisted options. 2552 IdentifierInfo *II = Tok.getIdentifierInfo(); 2553 PragmaMSCommentKind Kind = 2554 llvm::StringSwitch<PragmaMSCommentKind>(II->getName()) 2555 .Case("linker", PCK_Linker) 2556 .Case("lib", PCK_Lib) 2557 .Case("compiler", PCK_Compiler) 2558 .Case("exestr", PCK_ExeStr) 2559 .Case("user", PCK_User) 2560 .Default(PCK_Unknown); 2561 if (Kind == PCK_Unknown) { 2562 PP.Diag(Tok.getLocation(), diag::err_pragma_comment_unknown_kind); 2563 return; 2564 } 2565 2566 if (PP.getTargetInfo().getTriple().isOSBinFormatELF() && Kind != PCK_Lib) { 2567 PP.Diag(Tok.getLocation(), diag::warn_pragma_comment_ignored) 2568 << II->getName(); 2569 return; 2570 } 2571 2572 // On PS4, issue a warning about any pragma comments other than 2573 // #pragma comment lib. 2574 if (PP.getTargetInfo().getTriple().isPS4() && Kind != PCK_Lib) { 2575 PP.Diag(Tok.getLocation(), diag::warn_pragma_comment_ignored) 2576 << II->getName(); 2577 return; 2578 } 2579 2580 // Read the optional string if present. 2581 PP.Lex(Tok); 2582 std::string ArgumentString; 2583 if (Tok.is(tok::comma) && !PP.LexStringLiteral(Tok, ArgumentString, 2584 "pragma comment", 2585 /*AllowMacroExpansion=*/true)) 2586 return; 2587 2588 // FIXME: warn that 'exestr' is deprecated. 2589 // FIXME: If the kind is "compiler" warn if the string is present (it is 2590 // ignored). 2591 // The MSDN docs say that "lib" and "linker" require a string and have a short 2592 // whitelist of linker options they support, but in practice MSVC doesn't 2593 // issue a diagnostic. Therefore neither does clang. 2594 2595 if (Tok.isNot(tok::r_paren)) { 2596 PP.Diag(Tok.getLocation(), diag::err_pragma_comment_malformed); 2597 return; 2598 } 2599 PP.Lex(Tok); // eat the r_paren. 2600 2601 if (Tok.isNot(tok::eod)) { 2602 PP.Diag(Tok.getLocation(), diag::err_pragma_comment_malformed); 2603 return; 2604 } 2605 2606 // If the pragma is lexically sound, notify any interested PPCallbacks. 2607 if (PP.getPPCallbacks()) 2608 PP.getPPCallbacks()->PragmaComment(CommentLoc, II, ArgumentString); 2609 2610 Actions.ActOnPragmaMSComment(CommentLoc, Kind, ArgumentString); 2611 } 2612 2613 // #pragma clang optimize off 2614 // #pragma clang optimize on 2615 void PragmaOptimizeHandler::HandlePragma(Preprocessor &PP, 2616 PragmaIntroducer Introducer, 2617 Token &FirstToken) { 2618 Token Tok; 2619 PP.Lex(Tok); 2620 if (Tok.is(tok::eod)) { 2621 PP.Diag(Tok.getLocation(), diag::err_pragma_missing_argument) 2622 << "clang optimize" << /*Expected=*/true << "'on' or 'off'"; 2623 return; 2624 } 2625 if (Tok.isNot(tok::identifier)) { 2626 PP.Diag(Tok.getLocation(), diag::err_pragma_optimize_invalid_argument) 2627 << PP.getSpelling(Tok); 2628 return; 2629 } 2630 const IdentifierInfo *II = Tok.getIdentifierInfo(); 2631 // The only accepted values are 'on' or 'off'. 2632 bool IsOn = false; 2633 if (II->isStr("on")) { 2634 IsOn = true; 2635 } else if (!II->isStr("off")) { 2636 PP.Diag(Tok.getLocation(), diag::err_pragma_optimize_invalid_argument) 2637 << PP.getSpelling(Tok); 2638 return; 2639 } 2640 PP.Lex(Tok); 2641 2642 if (Tok.isNot(tok::eod)) { 2643 PP.Diag(Tok.getLocation(), diag::err_pragma_optimize_extra_argument) 2644 << PP.getSpelling(Tok); 2645 return; 2646 } 2647 2648 Actions.ActOnPragmaOptimize(IsOn, FirstToken.getLocation()); 2649 } 2650 2651 namespace { 2652 /// Used as the annotation value for tok::annot_pragma_fp. 2653 struct TokFPAnnotValue { 2654 enum FlagKinds { Contract }; 2655 enum FlagValues { On, Off, Fast }; 2656 2657 FlagKinds FlagKind; 2658 FlagValues FlagValue; 2659 }; 2660 } // end anonymous namespace 2661 2662 void PragmaFPHandler::HandlePragma(Preprocessor &PP, 2663 PragmaIntroducer Introducer, Token &Tok) { 2664 // fp 2665 Token PragmaName = Tok; 2666 SmallVector<Token, 1> TokenList; 2667 2668 PP.Lex(Tok); 2669 if (Tok.isNot(tok::identifier)) { 2670 PP.Diag(Tok.getLocation(), diag::err_pragma_fp_invalid_option) 2671 << /*MissingOption=*/true << ""; 2672 return; 2673 } 2674 2675 while (Tok.is(tok::identifier)) { 2676 IdentifierInfo *OptionInfo = Tok.getIdentifierInfo(); 2677 2678 auto FlagKind = 2679 llvm::StringSwitch<llvm::Optional<TokFPAnnotValue::FlagKinds>>( 2680 OptionInfo->getName()) 2681 .Case("contract", TokFPAnnotValue::Contract) 2682 .Default(None); 2683 if (!FlagKind) { 2684 PP.Diag(Tok.getLocation(), diag::err_pragma_fp_invalid_option) 2685 << /*MissingOption=*/false << OptionInfo; 2686 return; 2687 } 2688 PP.Lex(Tok); 2689 2690 // Read '(' 2691 if (Tok.isNot(tok::l_paren)) { 2692 PP.Diag(Tok.getLocation(), diag::err_expected) << tok::l_paren; 2693 return; 2694 } 2695 PP.Lex(Tok); 2696 2697 if (Tok.isNot(tok::identifier)) { 2698 PP.Diag(Tok.getLocation(), diag::err_pragma_fp_invalid_argument) 2699 << PP.getSpelling(Tok) << OptionInfo->getName(); 2700 return; 2701 } 2702 const IdentifierInfo *II = Tok.getIdentifierInfo(); 2703 2704 auto FlagValue = 2705 llvm::StringSwitch<llvm::Optional<TokFPAnnotValue::FlagValues>>( 2706 II->getName()) 2707 .Case("on", TokFPAnnotValue::On) 2708 .Case("off", TokFPAnnotValue::Off) 2709 .Case("fast", TokFPAnnotValue::Fast) 2710 .Default(llvm::None); 2711 2712 if (!FlagValue) { 2713 PP.Diag(Tok.getLocation(), diag::err_pragma_fp_invalid_argument) 2714 << PP.getSpelling(Tok) << OptionInfo->getName(); 2715 return; 2716 } 2717 PP.Lex(Tok); 2718 2719 // Read ')' 2720 if (Tok.isNot(tok::r_paren)) { 2721 PP.Diag(Tok.getLocation(), diag::err_expected) << tok::r_paren; 2722 return; 2723 } 2724 PP.Lex(Tok); 2725 2726 auto *AnnotValue = new (PP.getPreprocessorAllocator()) 2727 TokFPAnnotValue{*FlagKind, *FlagValue}; 2728 // Generate the loop hint token. 2729 Token FPTok; 2730 FPTok.startToken(); 2731 FPTok.setKind(tok::annot_pragma_fp); 2732 FPTok.setLocation(PragmaName.getLocation()); 2733 FPTok.setAnnotationEndLoc(PragmaName.getLocation()); 2734 FPTok.setAnnotationValue(reinterpret_cast<void *>(AnnotValue)); 2735 TokenList.push_back(FPTok); 2736 } 2737 2738 if (Tok.isNot(tok::eod)) { 2739 PP.Diag(Tok.getLocation(), diag::warn_pragma_extra_tokens_at_eol) 2740 << "clang fp"; 2741 return; 2742 } 2743 2744 auto TokenArray = std::make_unique<Token[]>(TokenList.size()); 2745 std::copy(TokenList.begin(), TokenList.end(), TokenArray.get()); 2746 2747 PP.EnterTokenStream(std::move(TokenArray), TokenList.size(), 2748 /*DisableMacroExpansion=*/false, /*IsReinject=*/false); 2749 } 2750 2751 void Parser::HandlePragmaFP() { 2752 assert(Tok.is(tok::annot_pragma_fp)); 2753 auto *AnnotValue = 2754 reinterpret_cast<TokFPAnnotValue *>(Tok.getAnnotationValue()); 2755 2756 LangOptions::FPContractModeKind FPC; 2757 switch (AnnotValue->FlagValue) { 2758 case TokFPAnnotValue::On: 2759 FPC = LangOptions::FPC_On; 2760 break; 2761 case TokFPAnnotValue::Fast: 2762 FPC = LangOptions::FPC_Fast; 2763 break; 2764 case TokFPAnnotValue::Off: 2765 FPC = LangOptions::FPC_Off; 2766 break; 2767 } 2768 2769 Actions.ActOnPragmaFPContract(FPC); 2770 ConsumeAnnotationToken(); 2771 } 2772 2773 /// Parses loop or unroll pragma hint value and fills in Info. 2774 static bool ParseLoopHintValue(Preprocessor &PP, Token &Tok, Token PragmaName, 2775 Token Option, bool ValueInParens, 2776 PragmaLoopHintInfo &Info) { 2777 SmallVector<Token, 1> ValueList; 2778 int OpenParens = ValueInParens ? 1 : 0; 2779 // Read constant expression. 2780 while (Tok.isNot(tok::eod)) { 2781 if (Tok.is(tok::l_paren)) 2782 OpenParens++; 2783 else if (Tok.is(tok::r_paren)) { 2784 OpenParens--; 2785 if (OpenParens == 0 && ValueInParens) 2786 break; 2787 } 2788 2789 ValueList.push_back(Tok); 2790 PP.Lex(Tok); 2791 } 2792 2793 if (ValueInParens) { 2794 // Read ')' 2795 if (Tok.isNot(tok::r_paren)) { 2796 PP.Diag(Tok.getLocation(), diag::err_expected) << tok::r_paren; 2797 return true; 2798 } 2799 PP.Lex(Tok); 2800 } 2801 2802 Token EOFTok; 2803 EOFTok.startToken(); 2804 EOFTok.setKind(tok::eof); 2805 EOFTok.setLocation(Tok.getLocation()); 2806 ValueList.push_back(EOFTok); // Terminates expression for parsing. 2807 2808 Info.Toks = llvm::makeArrayRef(ValueList).copy(PP.getPreprocessorAllocator()); 2809 2810 Info.PragmaName = PragmaName; 2811 Info.Option = Option; 2812 return false; 2813 } 2814 2815 /// Handle the \#pragma clang loop directive. 2816 /// #pragma clang 'loop' loop-hints 2817 /// 2818 /// loop-hints: 2819 /// loop-hint loop-hints[opt] 2820 /// 2821 /// loop-hint: 2822 /// 'vectorize' '(' loop-hint-keyword ')' 2823 /// 'interleave' '(' loop-hint-keyword ')' 2824 /// 'unroll' '(' unroll-hint-keyword ')' 2825 /// 'vectorize_predicate' '(' loop-hint-keyword ')' 2826 /// 'vectorize_width' '(' loop-hint-value ')' 2827 /// 'interleave_count' '(' loop-hint-value ')' 2828 /// 'unroll_count' '(' loop-hint-value ')' 2829 /// 'pipeline' '(' disable ')' 2830 /// 'pipeline_initiation_interval' '(' loop-hint-value ')' 2831 /// 2832 /// loop-hint-keyword: 2833 /// 'enable' 2834 /// 'disable' 2835 /// 'assume_safety' 2836 /// 2837 /// unroll-hint-keyword: 2838 /// 'enable' 2839 /// 'disable' 2840 /// 'full' 2841 /// 2842 /// loop-hint-value: 2843 /// constant-expression 2844 /// 2845 /// Specifying vectorize(enable) or vectorize_width(_value_) instructs llvm to 2846 /// try vectorizing the instructions of the loop it precedes. Specifying 2847 /// interleave(enable) or interleave_count(_value_) instructs llvm to try 2848 /// interleaving multiple iterations of the loop it precedes. The width of the 2849 /// vector instructions is specified by vectorize_width() and the number of 2850 /// interleaved loop iterations is specified by interleave_count(). Specifying a 2851 /// value of 1 effectively disables vectorization/interleaving, even if it is 2852 /// possible and profitable, and 0 is invalid. The loop vectorizer currently 2853 /// only works on inner loops. 2854 /// 2855 /// The unroll and unroll_count directives control the concatenation 2856 /// unroller. Specifying unroll(enable) instructs llvm to unroll the loop 2857 /// completely if the trip count is known at compile time and unroll partially 2858 /// if the trip count is not known. Specifying unroll(full) is similar to 2859 /// unroll(enable) but will unroll the loop only if the trip count is known at 2860 /// compile time. Specifying unroll(disable) disables unrolling for the 2861 /// loop. Specifying unroll_count(_value_) instructs llvm to try to unroll the 2862 /// loop the number of times indicated by the value. 2863 void PragmaLoopHintHandler::HandlePragma(Preprocessor &PP, 2864 PragmaIntroducer Introducer, 2865 Token &Tok) { 2866 // Incoming token is "loop" from "#pragma clang loop". 2867 Token PragmaName = Tok; 2868 SmallVector<Token, 1> TokenList; 2869 2870 // Lex the optimization option and verify it is an identifier. 2871 PP.Lex(Tok); 2872 if (Tok.isNot(tok::identifier)) { 2873 PP.Diag(Tok.getLocation(), diag::err_pragma_loop_invalid_option) 2874 << /*MissingOption=*/true << ""; 2875 return; 2876 } 2877 2878 while (Tok.is(tok::identifier)) { 2879 Token Option = Tok; 2880 IdentifierInfo *OptionInfo = Tok.getIdentifierInfo(); 2881 2882 bool OptionValid = llvm::StringSwitch<bool>(OptionInfo->getName()) 2883 .Case("vectorize", true) 2884 .Case("interleave", true) 2885 .Case("unroll", true) 2886 .Case("distribute", true) 2887 .Case("vectorize_predicate", true) 2888 .Case("vectorize_width", true) 2889 .Case("interleave_count", true) 2890 .Case("unroll_count", true) 2891 .Case("pipeline", true) 2892 .Case("pipeline_initiation_interval", true) 2893 .Default(false); 2894 if (!OptionValid) { 2895 PP.Diag(Tok.getLocation(), diag::err_pragma_loop_invalid_option) 2896 << /*MissingOption=*/false << OptionInfo; 2897 return; 2898 } 2899 PP.Lex(Tok); 2900 2901 // Read '(' 2902 if (Tok.isNot(tok::l_paren)) { 2903 PP.Diag(Tok.getLocation(), diag::err_expected) << tok::l_paren; 2904 return; 2905 } 2906 PP.Lex(Tok); 2907 2908 auto *Info = new (PP.getPreprocessorAllocator()) PragmaLoopHintInfo; 2909 if (ParseLoopHintValue(PP, Tok, PragmaName, Option, /*ValueInParens=*/true, 2910 *Info)) 2911 return; 2912 2913 // Generate the loop hint token. 2914 Token LoopHintTok; 2915 LoopHintTok.startToken(); 2916 LoopHintTok.setKind(tok::annot_pragma_loop_hint); 2917 LoopHintTok.setLocation(PragmaName.getLocation()); 2918 LoopHintTok.setAnnotationEndLoc(PragmaName.getLocation()); 2919 LoopHintTok.setAnnotationValue(static_cast<void *>(Info)); 2920 TokenList.push_back(LoopHintTok); 2921 } 2922 2923 if (Tok.isNot(tok::eod)) { 2924 PP.Diag(Tok.getLocation(), diag::warn_pragma_extra_tokens_at_eol) 2925 << "clang loop"; 2926 return; 2927 } 2928 2929 auto TokenArray = std::make_unique<Token[]>(TokenList.size()); 2930 std::copy(TokenList.begin(), TokenList.end(), TokenArray.get()); 2931 2932 PP.EnterTokenStream(std::move(TokenArray), TokenList.size(), 2933 /*DisableMacroExpansion=*/false, /*IsReinject=*/false); 2934 } 2935 2936 /// Handle the loop unroll optimization pragmas. 2937 /// #pragma unroll 2938 /// #pragma unroll unroll-hint-value 2939 /// #pragma unroll '(' unroll-hint-value ')' 2940 /// #pragma nounroll 2941 /// #pragma unroll_and_jam 2942 /// #pragma unroll_and_jam unroll-hint-value 2943 /// #pragma unroll_and_jam '(' unroll-hint-value ')' 2944 /// #pragma nounroll_and_jam 2945 /// 2946 /// unroll-hint-value: 2947 /// constant-expression 2948 /// 2949 /// Loop unrolling hints can be specified with '#pragma unroll' or 2950 /// '#pragma nounroll'. '#pragma unroll' can take a numeric argument optionally 2951 /// contained in parentheses. With no argument the directive instructs llvm to 2952 /// try to unroll the loop completely. A positive integer argument can be 2953 /// specified to indicate the number of times the loop should be unrolled. To 2954 /// maximize compatibility with other compilers the unroll count argument can be 2955 /// specified with or without parentheses. Specifying, '#pragma nounroll' 2956 /// disables unrolling of the loop. 2957 void PragmaUnrollHintHandler::HandlePragma(Preprocessor &PP, 2958 PragmaIntroducer Introducer, 2959 Token &Tok) { 2960 // Incoming token is "unroll" for "#pragma unroll", or "nounroll" for 2961 // "#pragma nounroll". 2962 Token PragmaName = Tok; 2963 PP.Lex(Tok); 2964 auto *Info = new (PP.getPreprocessorAllocator()) PragmaLoopHintInfo; 2965 if (Tok.is(tok::eod)) { 2966 // nounroll or unroll pragma without an argument. 2967 Info->PragmaName = PragmaName; 2968 Info->Option.startToken(); 2969 } else if (PragmaName.getIdentifierInfo()->getName() == "nounroll" || 2970 PragmaName.getIdentifierInfo()->getName() == "nounroll_and_jam") { 2971 PP.Diag(Tok.getLocation(), diag::warn_pragma_extra_tokens_at_eol) 2972 << PragmaName.getIdentifierInfo()->getName(); 2973 return; 2974 } else { 2975 // Unroll pragma with an argument: "#pragma unroll N" or 2976 // "#pragma unroll(N)". 2977 // Read '(' if it exists. 2978 bool ValueInParens = Tok.is(tok::l_paren); 2979 if (ValueInParens) 2980 PP.Lex(Tok); 2981 2982 Token Option; 2983 Option.startToken(); 2984 if (ParseLoopHintValue(PP, Tok, PragmaName, Option, ValueInParens, *Info)) 2985 return; 2986 2987 // In CUDA, the argument to '#pragma unroll' should not be contained in 2988 // parentheses. 2989 if (PP.getLangOpts().CUDA && ValueInParens) 2990 PP.Diag(Info->Toks[0].getLocation(), 2991 diag::warn_pragma_unroll_cuda_value_in_parens); 2992 2993 if (Tok.isNot(tok::eod)) { 2994 PP.Diag(Tok.getLocation(), diag::warn_pragma_extra_tokens_at_eol) 2995 << "unroll"; 2996 return; 2997 } 2998 } 2999 3000 // Generate the hint token. 3001 auto TokenArray = std::make_unique<Token[]>(1); 3002 TokenArray[0].startToken(); 3003 TokenArray[0].setKind(tok::annot_pragma_loop_hint); 3004 TokenArray[0].setLocation(PragmaName.getLocation()); 3005 TokenArray[0].setAnnotationEndLoc(PragmaName.getLocation()); 3006 TokenArray[0].setAnnotationValue(static_cast<void *>(Info)); 3007 PP.EnterTokenStream(std::move(TokenArray), 1, 3008 /*DisableMacroExpansion=*/false, /*IsReinject=*/false); 3009 } 3010 3011 /// Handle the Microsoft \#pragma intrinsic extension. 3012 /// 3013 /// The syntax is: 3014 /// \code 3015 /// #pragma intrinsic(memset) 3016 /// #pragma intrinsic(strlen, memcpy) 3017 /// \endcode 3018 /// 3019 /// Pragma intrisic tells the compiler to use a builtin version of the 3020 /// function. Clang does it anyway, so the pragma doesn't really do anything. 3021 /// Anyway, we emit a warning if the function specified in \#pragma intrinsic 3022 /// isn't an intrinsic in clang and suggest to include intrin.h. 3023 void PragmaMSIntrinsicHandler::HandlePragma(Preprocessor &PP, 3024 PragmaIntroducer Introducer, 3025 Token &Tok) { 3026 PP.Lex(Tok); 3027 3028 if (Tok.isNot(tok::l_paren)) { 3029 PP.Diag(Tok.getLocation(), diag::warn_pragma_expected_lparen) 3030 << "intrinsic"; 3031 return; 3032 } 3033 PP.Lex(Tok); 3034 3035 bool SuggestIntrinH = !PP.isMacroDefined("__INTRIN_H"); 3036 3037 while (Tok.is(tok::identifier)) { 3038 IdentifierInfo *II = Tok.getIdentifierInfo(); 3039 if (!II->getBuiltinID()) 3040 PP.Diag(Tok.getLocation(), diag::warn_pragma_intrinsic_builtin) 3041 << II << SuggestIntrinH; 3042 3043 PP.Lex(Tok); 3044 if (Tok.isNot(tok::comma)) 3045 break; 3046 PP.Lex(Tok); 3047 } 3048 3049 if (Tok.isNot(tok::r_paren)) { 3050 PP.Diag(Tok.getLocation(), diag::warn_pragma_expected_rparen) 3051 << "intrinsic"; 3052 return; 3053 } 3054 PP.Lex(Tok); 3055 3056 if (Tok.isNot(tok::eod)) 3057 PP.Diag(Tok.getLocation(), diag::warn_pragma_extra_tokens_at_eol) 3058 << "intrinsic"; 3059 } 3060 3061 // #pragma optimize("gsty", on|off) 3062 void PragmaMSOptimizeHandler::HandlePragma(Preprocessor &PP, 3063 PragmaIntroducer Introducer, 3064 Token &Tok) { 3065 SourceLocation StartLoc = Tok.getLocation(); 3066 PP.Lex(Tok); 3067 3068 if (Tok.isNot(tok::l_paren)) { 3069 PP.Diag(Tok.getLocation(), diag::warn_pragma_expected_lparen) << "optimize"; 3070 return; 3071 } 3072 PP.Lex(Tok); 3073 3074 if (Tok.isNot(tok::string_literal)) { 3075 PP.Diag(Tok.getLocation(), diag::warn_pragma_expected_string) << "optimize"; 3076 return; 3077 } 3078 // We could syntax check the string but it's probably not worth the effort. 3079 PP.Lex(Tok); 3080 3081 if (Tok.isNot(tok::comma)) { 3082 PP.Diag(Tok.getLocation(), diag::warn_pragma_expected_comma) << "optimize"; 3083 return; 3084 } 3085 PP.Lex(Tok); 3086 3087 if (Tok.is(tok::eod) || Tok.is(tok::r_paren)) { 3088 PP.Diag(Tok.getLocation(), diag::warn_pragma_missing_argument) 3089 << "optimize" << /*Expected=*/true << "'on' or 'off'"; 3090 return; 3091 } 3092 IdentifierInfo *II = Tok.getIdentifierInfo(); 3093 if (!II || (!II->isStr("on") && !II->isStr("off"))) { 3094 PP.Diag(Tok.getLocation(), diag::warn_pragma_invalid_argument) 3095 << PP.getSpelling(Tok) << "optimize" << /*Expected=*/true 3096 << "'on' or 'off'"; 3097 return; 3098 } 3099 PP.Lex(Tok); 3100 3101 if (Tok.isNot(tok::r_paren)) { 3102 PP.Diag(Tok.getLocation(), diag::warn_pragma_expected_rparen) << "optimize"; 3103 return; 3104 } 3105 PP.Lex(Tok); 3106 3107 if (Tok.isNot(tok::eod)) { 3108 PP.Diag(Tok.getLocation(), diag::warn_pragma_extra_tokens_at_eol) 3109 << "optimize"; 3110 return; 3111 } 3112 PP.Diag(StartLoc, diag::warn_pragma_optimize); 3113 } 3114 3115 void PragmaForceCUDAHostDeviceHandler::HandlePragma( 3116 Preprocessor &PP, PragmaIntroducer Introducer, Token &Tok) { 3117 Token FirstTok = Tok; 3118 3119 PP.Lex(Tok); 3120 IdentifierInfo *Info = Tok.getIdentifierInfo(); 3121 if (!Info || (!Info->isStr("begin") && !Info->isStr("end"))) { 3122 PP.Diag(FirstTok.getLocation(), 3123 diag::warn_pragma_force_cuda_host_device_bad_arg); 3124 return; 3125 } 3126 3127 if (Info->isStr("begin")) 3128 Actions.PushForceCUDAHostDevice(); 3129 else if (!Actions.PopForceCUDAHostDevice()) 3130 PP.Diag(FirstTok.getLocation(), 3131 diag::err_pragma_cannot_end_force_cuda_host_device); 3132 3133 PP.Lex(Tok); 3134 if (!Tok.is(tok::eod)) 3135 PP.Diag(FirstTok.getLocation(), 3136 diag::warn_pragma_force_cuda_host_device_bad_arg); 3137 } 3138 3139 /// Handle the #pragma clang attribute directive. 3140 /// 3141 /// The syntax is: 3142 /// \code 3143 /// #pragma clang attribute push (attribute, subject-set) 3144 /// #pragma clang attribute push 3145 /// #pragma clang attribute (attribute, subject-set) 3146 /// #pragma clang attribute pop 3147 /// \endcode 3148 /// 3149 /// There are also 'namespace' variants of push and pop directives. The bare 3150 /// '#pragma clang attribute (attribute, subject-set)' version doesn't require a 3151 /// namespace, since it always applies attributes to the most recently pushed 3152 /// group, regardless of namespace. 3153 /// \code 3154 /// #pragma clang attribute namespace.push (attribute, subject-set) 3155 /// #pragma clang attribute namespace.push 3156 /// #pragma clang attribute namespace.pop 3157 /// \endcode 3158 /// 3159 /// The subject-set clause defines the set of declarations which receive the 3160 /// attribute. Its exact syntax is described in the LanguageExtensions document 3161 /// in Clang's documentation. 3162 /// 3163 /// This directive instructs the compiler to begin/finish applying the specified 3164 /// attribute to the set of attribute-specific declarations in the active range 3165 /// of the pragma. 3166 void PragmaAttributeHandler::HandlePragma(Preprocessor &PP, 3167 PragmaIntroducer Introducer, 3168 Token &FirstToken) { 3169 Token Tok; 3170 PP.Lex(Tok); 3171 auto *Info = new (PP.getPreprocessorAllocator()) 3172 PragmaAttributeInfo(AttributesForPragmaAttribute); 3173 3174 // Parse the optional namespace followed by a period. 3175 if (Tok.is(tok::identifier)) { 3176 IdentifierInfo *II = Tok.getIdentifierInfo(); 3177 if (!II->isStr("push") && !II->isStr("pop")) { 3178 Info->Namespace = II; 3179 PP.Lex(Tok); 3180 3181 if (!Tok.is(tok::period)) { 3182 PP.Diag(Tok.getLocation(), diag::err_pragma_attribute_expected_period) 3183 << II; 3184 return; 3185 } 3186 PP.Lex(Tok); 3187 } 3188 } 3189 3190 if (!Tok.isOneOf(tok::identifier, tok::l_paren)) { 3191 PP.Diag(Tok.getLocation(), 3192 diag::err_pragma_attribute_expected_push_pop_paren); 3193 return; 3194 } 3195 3196 // Determine what action this pragma clang attribute represents. 3197 if (Tok.is(tok::l_paren)) { 3198 if (Info->Namespace) { 3199 PP.Diag(Tok.getLocation(), 3200 diag::err_pragma_attribute_namespace_on_attribute); 3201 PP.Diag(Tok.getLocation(), 3202 diag::note_pragma_attribute_namespace_on_attribute); 3203 return; 3204 } 3205 Info->Action = PragmaAttributeInfo::Attribute; 3206 } else { 3207 const IdentifierInfo *II = Tok.getIdentifierInfo(); 3208 if (II->isStr("push")) 3209 Info->Action = PragmaAttributeInfo::Push; 3210 else if (II->isStr("pop")) 3211 Info->Action = PragmaAttributeInfo::Pop; 3212 else { 3213 PP.Diag(Tok.getLocation(), diag::err_pragma_attribute_invalid_argument) 3214 << PP.getSpelling(Tok); 3215 return; 3216 } 3217 3218 PP.Lex(Tok); 3219 } 3220 3221 // Parse the actual attribute. 3222 if ((Info->Action == PragmaAttributeInfo::Push && Tok.isNot(tok::eod)) || 3223 Info->Action == PragmaAttributeInfo::Attribute) { 3224 if (Tok.isNot(tok::l_paren)) { 3225 PP.Diag(Tok.getLocation(), diag::err_expected) << tok::l_paren; 3226 return; 3227 } 3228 PP.Lex(Tok); 3229 3230 // Lex the attribute tokens. 3231 SmallVector<Token, 16> AttributeTokens; 3232 int OpenParens = 1; 3233 while (Tok.isNot(tok::eod)) { 3234 if (Tok.is(tok::l_paren)) 3235 OpenParens++; 3236 else if (Tok.is(tok::r_paren)) { 3237 OpenParens--; 3238 if (OpenParens == 0) 3239 break; 3240 } 3241 3242 AttributeTokens.push_back(Tok); 3243 PP.Lex(Tok); 3244 } 3245 3246 if (AttributeTokens.empty()) { 3247 PP.Diag(Tok.getLocation(), diag::err_pragma_attribute_expected_attribute); 3248 return; 3249 } 3250 if (Tok.isNot(tok::r_paren)) { 3251 PP.Diag(Tok.getLocation(), diag::err_expected) << tok::r_paren; 3252 return; 3253 } 3254 SourceLocation EndLoc = Tok.getLocation(); 3255 PP.Lex(Tok); 3256 3257 // Terminate the attribute for parsing. 3258 Token EOFTok; 3259 EOFTok.startToken(); 3260 EOFTok.setKind(tok::eof); 3261 EOFTok.setLocation(EndLoc); 3262 AttributeTokens.push_back(EOFTok); 3263 3264 Info->Tokens = 3265 llvm::makeArrayRef(AttributeTokens).copy(PP.getPreprocessorAllocator()); 3266 } 3267 3268 if (Tok.isNot(tok::eod)) 3269 PP.Diag(Tok.getLocation(), diag::warn_pragma_extra_tokens_at_eol) 3270 << "clang attribute"; 3271 3272 // Generate the annotated pragma token. 3273 auto TokenArray = std::make_unique<Token[]>(1); 3274 TokenArray[0].startToken(); 3275 TokenArray[0].setKind(tok::annot_pragma_attribute); 3276 TokenArray[0].setLocation(FirstToken.getLocation()); 3277 TokenArray[0].setAnnotationEndLoc(FirstToken.getLocation()); 3278 TokenArray[0].setAnnotationValue(static_cast<void *>(Info)); 3279 PP.EnterTokenStream(std::move(TokenArray), 1, 3280 /*DisableMacroExpansion=*/false, /*IsReinject=*/false); 3281 } 3282