1 //===--- RAIIObjectsForParser.h - RAII helpers for the parser ---*- C++ -*-===// 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 defines and implements the some simple RAII objects that are used 10 // by the parser to manage bits in recursion. 11 // 12 //===----------------------------------------------------------------------===// 13 14 #ifndef LLVM_CLANG_PARSE_RAIIOBJECTSFORPARSER_H 15 #define LLVM_CLANG_PARSE_RAIIOBJECTSFORPARSER_H 16 17 #include "clang/Parse/ParseDiagnostic.h" 18 #include "clang/Parse/Parser.h" 19 #include "clang/Sema/DelayedDiagnostic.h" 20 #include "clang/Sema/ParsedTemplate.h" 21 #include "clang/Sema/Sema.h" 22 23 namespace clang { 24 // TODO: move ParsingClassDefinition here. 25 // TODO: move TentativeParsingAction here. 26 27 /// A RAII object used to temporarily suppress access-like 28 /// checking. Access-like checks are those associated with 29 /// controlling the use of a declaration, like C++ access control 30 /// errors and deprecation warnings. They are contextually 31 /// dependent, in that they can only be resolved with full 32 /// information about what's being declared. They are also 33 /// suppressed in certain contexts, like the template arguments of 34 /// an explicit instantiation. However, those suppression contexts 35 /// cannot necessarily be fully determined in advance; for 36 /// example, something starting like this: 37 /// template <> class std::vector<A::PrivateType> 38 /// might be the entirety of an explicit instantiation: 39 /// template <> class std::vector<A::PrivateType>; 40 /// or just an elaborated type specifier: 41 /// template <> class std::vector<A::PrivateType> make_vector<>(); 42 /// Therefore this class collects all the diagnostics and permits 43 /// them to be re-delayed in a new context. 44 class SuppressAccessChecks { 45 Sema &S; 46 sema::DelayedDiagnosticPool DiagnosticPool; 47 Sema::ParsingDeclState State; 48 bool Active; 49 50 public: 51 /// Begin suppressing access-like checks 52 SuppressAccessChecks(Parser &P, bool activate = true) 53 : S(P.getActions()), DiagnosticPool(nullptr) { 54 if (activate) { 55 State = S.PushParsingDeclaration(DiagnosticPool); 56 Active = true; 57 } else { 58 Active = false; 59 } 60 } SuppressAccessChecks(SuppressAccessChecks && Other)61 SuppressAccessChecks(SuppressAccessChecks &&Other) 62 : S(Other.S), DiagnosticPool(std::move(Other.DiagnosticPool)), 63 State(Other.State), Active(Other.Active) { 64 Other.Active = false; 65 } 66 void operator=(SuppressAccessChecks &&Other) = delete; 67 done()68 void done() { 69 assert(Active && "trying to end an inactive suppression"); 70 S.PopParsingDeclaration(State, nullptr); 71 Active = false; 72 } 73 redelay()74 void redelay() { 75 assert(!Active && "redelaying without having ended first"); 76 if (!DiagnosticPool.pool_empty()) 77 S.redelayDiagnostics(DiagnosticPool); 78 assert(DiagnosticPool.pool_empty()); 79 } 80 ~SuppressAccessChecks()81 ~SuppressAccessChecks() { 82 if (Active) done(); 83 } 84 }; 85 86 /// RAII object used to inform the actions that we're 87 /// currently parsing a declaration. This is active when parsing a 88 /// variable's initializer, but not when parsing the body of a 89 /// class or function definition. 90 class ParsingDeclRAIIObject { 91 Sema &Actions; 92 sema::DelayedDiagnosticPool DiagnosticPool; 93 Sema::ParsingDeclState State; 94 bool Popped; 95 96 ParsingDeclRAIIObject(const ParsingDeclRAIIObject &) = delete; 97 void operator=(const ParsingDeclRAIIObject &) = delete; 98 99 public: 100 enum NoParent_t { NoParent }; ParsingDeclRAIIObject(Parser & P,NoParent_t _)101 ParsingDeclRAIIObject(Parser &P, NoParent_t _) 102 : Actions(P.getActions()), DiagnosticPool(nullptr) { 103 push(); 104 } 105 106 /// Creates a RAII object whose pool is optionally parented by another. ParsingDeclRAIIObject(Parser & P,const sema::DelayedDiagnosticPool * parentPool)107 ParsingDeclRAIIObject(Parser &P, 108 const sema::DelayedDiagnosticPool *parentPool) 109 : Actions(P.getActions()), DiagnosticPool(parentPool) { 110 push(); 111 } 112 113 /// Creates a RAII object and, optionally, initialize its 114 /// diagnostics pool by stealing the diagnostics from another 115 /// RAII object (which is assumed to be the current top pool). ParsingDeclRAIIObject(Parser & P,ParsingDeclRAIIObject * other)116 ParsingDeclRAIIObject(Parser &P, ParsingDeclRAIIObject *other) 117 : Actions(P.getActions()), 118 DiagnosticPool(other ? other->DiagnosticPool.getParent() : nullptr) { 119 if (other) { 120 DiagnosticPool.steal(other->DiagnosticPool); 121 other->abort(); 122 } 123 push(); 124 } 125 ~ParsingDeclRAIIObject()126 ~ParsingDeclRAIIObject() { 127 abort(); 128 } 129 getDelayedDiagnosticPool()130 sema::DelayedDiagnosticPool &getDelayedDiagnosticPool() { 131 return DiagnosticPool; 132 } getDelayedDiagnosticPool()133 const sema::DelayedDiagnosticPool &getDelayedDiagnosticPool() const { 134 return DiagnosticPool; 135 } 136 137 /// Resets the RAII object for a new declaration. reset()138 void reset() { 139 abort(); 140 push(); 141 } 142 143 /// Signals that the context was completed without an appropriate 144 /// declaration being parsed. abort()145 void abort() { 146 pop(nullptr); 147 } 148 complete(Decl * D)149 void complete(Decl *D) { 150 assert(!Popped && "ParsingDeclaration has already been popped!"); 151 pop(D); 152 } 153 154 /// Unregister this object from Sema, but remember all the 155 /// diagnostics that were emitted into it. abortAndRemember()156 void abortAndRemember() { 157 pop(nullptr); 158 } 159 160 private: push()161 void push() { 162 State = Actions.PushParsingDeclaration(DiagnosticPool); 163 Popped = false; 164 } 165 pop(Decl * D)166 void pop(Decl *D) { 167 if (!Popped) { 168 Actions.PopParsingDeclaration(State, D); 169 Popped = true; 170 } 171 } 172 }; 173 174 /// A class for parsing a DeclSpec. 175 class ParsingDeclSpec : public DeclSpec { 176 ParsingDeclRAIIObject ParsingRAII; 177 178 public: ParsingDeclSpec(Parser & P)179 ParsingDeclSpec(Parser &P) 180 : DeclSpec(P.getAttrFactory()), 181 ParsingRAII(P, ParsingDeclRAIIObject::NoParent) {} ParsingDeclSpec(Parser & P,ParsingDeclRAIIObject * RAII)182 ParsingDeclSpec(Parser &P, ParsingDeclRAIIObject *RAII) 183 : DeclSpec(P.getAttrFactory()), 184 ParsingRAII(P, RAII) {} 185 getDelayedDiagnosticPool()186 const sema::DelayedDiagnosticPool &getDelayedDiagnosticPool() const { 187 return ParsingRAII.getDelayedDiagnosticPool(); 188 } 189 complete(Decl * D)190 void complete(Decl *D) { 191 ParsingRAII.complete(D); 192 } 193 abort()194 void abort() { 195 ParsingRAII.abort(); 196 } 197 }; 198 199 /// A class for parsing a declarator. 200 class ParsingDeclarator : public Declarator { 201 ParsingDeclRAIIObject ParsingRAII; 202 203 public: ParsingDeclarator(Parser & P,const ParsingDeclSpec & DS,const ParsedAttributes & DeclarationAttrs,DeclaratorContext C)204 ParsingDeclarator(Parser &P, const ParsingDeclSpec &DS, 205 const ParsedAttributes &DeclarationAttrs, 206 DeclaratorContext C) 207 : Declarator(DS, DeclarationAttrs, C), 208 ParsingRAII(P, &DS.getDelayedDiagnosticPool()) {} 209 getDeclSpec()210 const ParsingDeclSpec &getDeclSpec() const { 211 return static_cast<const ParsingDeclSpec&>(Declarator::getDeclSpec()); 212 } 213 getMutableDeclSpec()214 ParsingDeclSpec &getMutableDeclSpec() const { 215 return const_cast<ParsingDeclSpec&>(getDeclSpec()); 216 } 217 clear()218 void clear() { 219 Declarator::clear(); 220 ParsingRAII.reset(); 221 } 222 complete(Decl * D)223 void complete(Decl *D) { 224 ParsingRAII.complete(D); 225 } 226 }; 227 228 /// A class for parsing a field declarator. 229 class ParsingFieldDeclarator : public FieldDeclarator { 230 ParsingDeclRAIIObject ParsingRAII; 231 232 public: ParsingFieldDeclarator(Parser & P,const ParsingDeclSpec & DS,const ParsedAttributes & DeclarationAttrs)233 ParsingFieldDeclarator(Parser &P, const ParsingDeclSpec &DS, 234 const ParsedAttributes &DeclarationAttrs) 235 : FieldDeclarator(DS, DeclarationAttrs), 236 ParsingRAII(P, &DS.getDelayedDiagnosticPool()) {} 237 getDeclSpec()238 const ParsingDeclSpec &getDeclSpec() const { 239 return static_cast<const ParsingDeclSpec&>(D.getDeclSpec()); 240 } 241 getMutableDeclSpec()242 ParsingDeclSpec &getMutableDeclSpec() const { 243 return const_cast<ParsingDeclSpec&>(getDeclSpec()); 244 } 245 complete(Decl * D)246 void complete(Decl *D) { 247 ParsingRAII.complete(D); 248 } 249 }; 250 251 /// ExtensionRAIIObject - This saves the state of extension warnings when 252 /// constructed and disables them. When destructed, it restores them back to 253 /// the way they used to be. This is used to handle __extension__ in the 254 /// parser. 255 class ExtensionRAIIObject { 256 ExtensionRAIIObject(const ExtensionRAIIObject &) = delete; 257 void operator=(const ExtensionRAIIObject &) = delete; 258 259 DiagnosticsEngine &Diags; 260 public: ExtensionRAIIObject(DiagnosticsEngine & diags)261 ExtensionRAIIObject(DiagnosticsEngine &diags) : Diags(diags) { 262 Diags.IncrementAllExtensionsSilenced(); 263 } 264 ~ExtensionRAIIObject()265 ~ExtensionRAIIObject() { 266 Diags.DecrementAllExtensionsSilenced(); 267 } 268 }; 269 270 /// ColonProtectionRAIIObject - This sets the Parser::ColonIsSacred bool and 271 /// restores it when destroyed. This says that "foo:" should not be 272 /// considered a possible typo for "foo::" for error recovery purposes. 273 class ColonProtectionRAIIObject { 274 Parser &P; 275 bool OldVal; 276 public: 277 ColonProtectionRAIIObject(Parser &p, bool Value = true) P(p)278 : P(p), OldVal(P.ColonIsSacred) { 279 P.ColonIsSacred = Value; 280 } 281 282 /// restore - This can be used to restore the state early, before the dtor 283 /// is run. restore()284 void restore() { 285 P.ColonIsSacred = OldVal; 286 } 287 ~ColonProtectionRAIIObject()288 ~ColonProtectionRAIIObject() { 289 restore(); 290 } 291 }; 292 293 /// Activates OpenMP parsing mode to preseve OpenMP specific annotation 294 /// tokens. 295 class ParsingOpenMPDirectiveRAII { 296 Parser &P; 297 bool OldVal; 298 299 public: 300 ParsingOpenMPDirectiveRAII(Parser &P, bool Value = true) P(P)301 : P(P), OldVal(P.OpenMPDirectiveParsing) { 302 P.OpenMPDirectiveParsing = Value; 303 } 304 305 /// This can be used to restore the state early, before the dtor 306 /// is run. restore()307 void restore() { P.OpenMPDirectiveParsing = OldVal; } 308 ~ParsingOpenMPDirectiveRAII()309 ~ParsingOpenMPDirectiveRAII() { restore(); } 310 }; 311 312 /// Activates OpenACC parsing mode to preseve OpenACC specific annotation 313 /// tokens. 314 class ParsingOpenACCDirectiveRAII { 315 Parser &P; 316 bool OldVal; 317 318 public: 319 ParsingOpenACCDirectiveRAII(Parser &P, bool Value = true) P(P)320 : P(P), OldVal(P.OpenACCDirectiveParsing) { 321 P.OpenACCDirectiveParsing = Value; 322 } 323 324 /// This can be used to restore the state early, before the dtor 325 /// is run. restore()326 void restore() { P.OpenACCDirectiveParsing = OldVal; } 327 ~ParsingOpenACCDirectiveRAII()328 ~ParsingOpenACCDirectiveRAII() { restore(); } 329 }; 330 331 /// RAII object that makes '>' behave either as an operator 332 /// or as the closing angle bracket for a template argument list. 333 class GreaterThanIsOperatorScope { 334 bool &GreaterThanIsOperator; 335 bool OldGreaterThanIsOperator; 336 public: GreaterThanIsOperatorScope(bool & GTIO,bool Val)337 GreaterThanIsOperatorScope(bool >IO, bool Val) 338 : GreaterThanIsOperator(GTIO), OldGreaterThanIsOperator(GTIO) { 339 GreaterThanIsOperator = Val; 340 } 341 ~GreaterThanIsOperatorScope()342 ~GreaterThanIsOperatorScope() { 343 GreaterThanIsOperator = OldGreaterThanIsOperator; 344 } 345 }; 346 347 class InMessageExpressionRAIIObject { 348 bool &InMessageExpression; 349 bool OldValue; 350 351 public: InMessageExpressionRAIIObject(Parser & P,bool Value)352 InMessageExpressionRAIIObject(Parser &P, bool Value) 353 : InMessageExpression(P.InMessageExpression), 354 OldValue(P.InMessageExpression) { 355 InMessageExpression = Value; 356 } 357 ~InMessageExpressionRAIIObject()358 ~InMessageExpressionRAIIObject() { 359 InMessageExpression = OldValue; 360 } 361 }; 362 363 class OffsetOfStateRAIIObject { 364 Sema::OffsetOfKind &OffsetOfState; 365 Sema::OffsetOfKind OldValue; 366 367 public: OffsetOfStateRAIIObject(Parser & P,Sema::OffsetOfKind Value)368 OffsetOfStateRAIIObject(Parser &P, Sema::OffsetOfKind Value) 369 : OffsetOfState(P.OffsetOfState), OldValue(P.OffsetOfState) { 370 OffsetOfState = Value; 371 } 372 ~OffsetOfStateRAIIObject()373 ~OffsetOfStateRAIIObject() { OffsetOfState = OldValue; } 374 }; 375 376 /// RAII object that makes sure paren/bracket/brace count is correct 377 /// after declaration/statement parsing, even when there's a parsing error. 378 class ParenBraceBracketBalancer { 379 Parser &P; 380 unsigned short ParenCount, BracketCount, BraceCount; 381 public: ParenBraceBracketBalancer(Parser & p)382 ParenBraceBracketBalancer(Parser &p) 383 : P(p), ParenCount(p.ParenCount), BracketCount(p.BracketCount), 384 BraceCount(p.BraceCount) { } 385 ~ParenBraceBracketBalancer()386 ~ParenBraceBracketBalancer() { 387 P.AngleBrackets.clear(P); 388 P.ParenCount = ParenCount; 389 P.BracketCount = BracketCount; 390 P.BraceCount = BraceCount; 391 } 392 }; 393 394 class PoisonSEHIdentifiersRAIIObject { 395 PoisonIdentifierRAIIObject Ident_AbnormalTermination; 396 PoisonIdentifierRAIIObject Ident_GetExceptionCode; 397 PoisonIdentifierRAIIObject Ident_GetExceptionInfo; 398 PoisonIdentifierRAIIObject Ident__abnormal_termination; 399 PoisonIdentifierRAIIObject Ident__exception_code; 400 PoisonIdentifierRAIIObject Ident__exception_info; 401 PoisonIdentifierRAIIObject Ident___abnormal_termination; 402 PoisonIdentifierRAIIObject Ident___exception_code; 403 PoisonIdentifierRAIIObject Ident___exception_info; 404 public: PoisonSEHIdentifiersRAIIObject(Parser & Self,bool NewValue)405 PoisonSEHIdentifiersRAIIObject(Parser &Self, bool NewValue) 406 : Ident_AbnormalTermination(Self.Ident_AbnormalTermination, NewValue), 407 Ident_GetExceptionCode(Self.Ident_GetExceptionCode, NewValue), 408 Ident_GetExceptionInfo(Self.Ident_GetExceptionInfo, NewValue), 409 Ident__abnormal_termination(Self.Ident__abnormal_termination, NewValue), 410 Ident__exception_code(Self.Ident__exception_code, NewValue), 411 Ident__exception_info(Self.Ident__exception_info, NewValue), 412 Ident___abnormal_termination(Self.Ident___abnormal_termination, NewValue), 413 Ident___exception_code(Self.Ident___exception_code, NewValue), 414 Ident___exception_info(Self.Ident___exception_info, NewValue) { 415 } 416 }; 417 418 /// RAII class that helps handle the parsing of an open/close delimiter 419 /// pair, such as braces { ... } or parentheses ( ... ). 420 class BalancedDelimiterTracker : public GreaterThanIsOperatorScope { 421 Parser& P; 422 tok::TokenKind Kind, Close, FinalToken; 423 SourceLocation (Parser::*Consumer)(); 424 SourceLocation LOpen, LClose; 425 getDepth()426 unsigned short &getDepth() { 427 switch (Kind) { 428 case tok::l_brace: return P.BraceCount; 429 case tok::l_square: return P.BracketCount; 430 case tok::l_paren: return P.ParenCount; 431 default: llvm_unreachable("Wrong token kind"); 432 } 433 } 434 435 bool diagnoseOverflow(); 436 bool diagnoseMissingClose(); 437 438 public: 439 BalancedDelimiterTracker(Parser& p, tok::TokenKind k, 440 tok::TokenKind FinalToken = tok::semi) 441 : GreaterThanIsOperatorScope(p.GreaterThanIsOperator, true), 442 P(p), Kind(k), FinalToken(FinalToken) 443 { 444 switch (Kind) { 445 default: llvm_unreachable("Unexpected balanced token"); 446 case tok::l_brace: 447 Close = tok::r_brace; 448 Consumer = &Parser::ConsumeBrace; 449 break; 450 case tok::l_paren: 451 Close = tok::r_paren; 452 Consumer = &Parser::ConsumeParen; 453 break; 454 455 case tok::l_square: 456 Close = tok::r_square; 457 Consumer = &Parser::ConsumeBracket; 458 break; 459 } 460 } 461 getOpenLocation()462 SourceLocation getOpenLocation() const { return LOpen; } getCloseLocation()463 SourceLocation getCloseLocation() const { return LClose; } getRange()464 SourceRange getRange() const { return SourceRange(LOpen, LClose); } 465 consumeOpen()466 bool consumeOpen() { 467 if (!P.Tok.is(Kind)) 468 return true; 469 470 if (getDepth() < P.getLangOpts().BracketDepth) { 471 LOpen = (P.*Consumer)(); 472 return false; 473 } 474 475 return diagnoseOverflow(); 476 } 477 478 bool expectAndConsume(unsigned DiagID = diag::err_expected, 479 const char *Msg = "", 480 tok::TokenKind SkipToTok = tok::unknown); consumeClose()481 bool consumeClose() { 482 if (P.Tok.is(Close)) { 483 LClose = (P.*Consumer)(); 484 return false; 485 } else if (P.Tok.is(tok::semi) && P.NextToken().is(Close)) { 486 SourceLocation SemiLoc = P.ConsumeToken(); 487 P.Diag(SemiLoc, diag::err_unexpected_semi) 488 << Close << FixItHint::CreateRemoval(SourceRange(SemiLoc, SemiLoc)); 489 LClose = (P.*Consumer)(); 490 return false; 491 } 492 493 return diagnoseMissingClose(); 494 } 495 void skipToEnd(); 496 }; 497 } // end namespace clang 498 499 #endif 500