1 //===--- RAIIObjectsForParser.h - RAII helpers for the parser ---*- C++ -*-===// 2 // 3 // The LLVM Compiler Infrastructure 4 // 5 // This file is distributed under the University of Illinois Open Source 6 // License. See LICENSE.TXT for details. 7 // 8 //===----------------------------------------------------------------------===// 9 // 10 // This file defines and implements the some simple RAII objects that are used 11 // by the parser to manage bits in recursion. 12 // 13 //===----------------------------------------------------------------------===// 14 15 #ifndef LLVM_CLANG_PARSE_RAII_OBJECTS_FOR_PARSER_H 16 #define LLVM_CLANG_PARSE_RAII_OBJECTS_FOR_PARSER_H 17 18 #include "clang/Parse/ParseDiagnostic.h" 19 #include "clang/Parse/Parser.h" 20 #include "clang/Sema/DelayedDiagnostic.h" 21 #include "clang/Sema/Sema.h" 22 23 namespace clang { 24 // TODO: move ParsingClassDefinition here. 25 // TODO: move TentativeParsingAction here. 26 27 /// \brief 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(NULL) { 54 if (activate) { 55 State = S.PushParsingDeclaration(DiagnosticPool); 56 Active = true; 57 } else { 58 Active = false; 59 } 60 } 61 done()62 void done() { 63 assert(Active && "trying to end an inactive suppression"); 64 S.PopParsingDeclaration(State, NULL); 65 Active = false; 66 } 67 redelay()68 void redelay() { 69 assert(!Active && "redelaying without having ended first"); 70 if (!DiagnosticPool.pool_empty()) 71 S.redelayDiagnostics(DiagnosticPool); 72 assert(DiagnosticPool.pool_empty()); 73 } 74 ~SuppressAccessChecks()75 ~SuppressAccessChecks() { 76 if (Active) done(); 77 } 78 }; 79 80 /// \brief RAII object used to inform the actions that we're 81 /// currently parsing a declaration. This is active when parsing a 82 /// variable's initializer, but not when parsing the body of a 83 /// class or function definition. 84 class ParsingDeclRAIIObject { 85 Sema &Actions; 86 sema::DelayedDiagnosticPool DiagnosticPool; 87 Sema::ParsingDeclState State; 88 bool Popped; 89 90 ParsingDeclRAIIObject(const ParsingDeclRAIIObject &) LLVM_DELETED_FUNCTION; 91 void operator=(const ParsingDeclRAIIObject &) LLVM_DELETED_FUNCTION; 92 93 public: 94 enum NoParent_t { NoParent }; ParsingDeclRAIIObject(Parser & P,NoParent_t _)95 ParsingDeclRAIIObject(Parser &P, NoParent_t _) 96 : Actions(P.getActions()), DiagnosticPool(NULL) { 97 push(); 98 } 99 100 /// Creates a RAII object whose pool is optionally parented by another. ParsingDeclRAIIObject(Parser & P,const sema::DelayedDiagnosticPool * parentPool)101 ParsingDeclRAIIObject(Parser &P, 102 const sema::DelayedDiagnosticPool *parentPool) 103 : Actions(P.getActions()), DiagnosticPool(parentPool) { 104 push(); 105 } 106 107 /// Creates a RAII object and, optionally, initialize its 108 /// diagnostics pool by stealing the diagnostics from another 109 /// RAII object (which is assumed to be the current top pool). ParsingDeclRAIIObject(Parser & P,ParsingDeclRAIIObject * other)110 ParsingDeclRAIIObject(Parser &P, ParsingDeclRAIIObject *other) 111 : Actions(P.getActions()), 112 DiagnosticPool(other ? other->DiagnosticPool.getParent() : NULL) { 113 if (other) { 114 DiagnosticPool.steal(other->DiagnosticPool); 115 other->abort(); 116 } 117 push(); 118 } 119 ~ParsingDeclRAIIObject()120 ~ParsingDeclRAIIObject() { 121 abort(); 122 } 123 getDelayedDiagnosticPool()124 sema::DelayedDiagnosticPool &getDelayedDiagnosticPool() { 125 return DiagnosticPool; 126 } getDelayedDiagnosticPool()127 const sema::DelayedDiagnosticPool &getDelayedDiagnosticPool() const { 128 return DiagnosticPool; 129 } 130 131 /// Resets the RAII object for a new declaration. reset()132 void reset() { 133 abort(); 134 push(); 135 } 136 137 /// Signals that the context was completed without an appropriate 138 /// declaration being parsed. abort()139 void abort() { 140 pop(0); 141 } 142 complete(Decl * D)143 void complete(Decl *D) { 144 assert(!Popped && "ParsingDeclaration has already been popped!"); 145 pop(D); 146 } 147 148 /// Unregister this object from Sema, but remember all the 149 /// diagnostics that were emitted into it. abortAndRemember()150 void abortAndRemember() { 151 pop(0); 152 } 153 154 private: push()155 void push() { 156 State = Actions.PushParsingDeclaration(DiagnosticPool); 157 Popped = false; 158 } 159 pop(Decl * D)160 void pop(Decl *D) { 161 if (!Popped) { 162 Actions.PopParsingDeclaration(State, D); 163 Popped = true; 164 } 165 } 166 }; 167 168 /// A class for parsing a DeclSpec. 169 class ParsingDeclSpec : public DeclSpec { 170 ParsingDeclRAIIObject ParsingRAII; 171 172 public: ParsingDeclSpec(Parser & P)173 ParsingDeclSpec(Parser &P) 174 : DeclSpec(P.getAttrFactory()), 175 ParsingRAII(P, ParsingDeclRAIIObject::NoParent) {} ParsingDeclSpec(Parser & P,ParsingDeclRAIIObject * RAII)176 ParsingDeclSpec(Parser &P, ParsingDeclRAIIObject *RAII) 177 : DeclSpec(P.getAttrFactory()), 178 ParsingRAII(P, RAII) {} 179 getDelayedDiagnosticPool()180 const sema::DelayedDiagnosticPool &getDelayedDiagnosticPool() const { 181 return ParsingRAII.getDelayedDiagnosticPool(); 182 } 183 complete(Decl * D)184 void complete(Decl *D) { 185 ParsingRAII.complete(D); 186 } 187 abort()188 void abort() { 189 ParsingRAII.abort(); 190 } 191 }; 192 193 /// A class for parsing a declarator. 194 class ParsingDeclarator : public Declarator { 195 ParsingDeclRAIIObject ParsingRAII; 196 197 public: ParsingDeclarator(Parser & P,const ParsingDeclSpec & DS,TheContext C)198 ParsingDeclarator(Parser &P, const ParsingDeclSpec &DS, TheContext C) 199 : Declarator(DS, C), ParsingRAII(P, &DS.getDelayedDiagnosticPool()) { 200 } 201 getDeclSpec()202 const ParsingDeclSpec &getDeclSpec() const { 203 return static_cast<const ParsingDeclSpec&>(Declarator::getDeclSpec()); 204 } 205 getMutableDeclSpec()206 ParsingDeclSpec &getMutableDeclSpec() const { 207 return const_cast<ParsingDeclSpec&>(getDeclSpec()); 208 } 209 clear()210 void clear() { 211 Declarator::clear(); 212 ParsingRAII.reset(); 213 } 214 complete(Decl * D)215 void complete(Decl *D) { 216 ParsingRAII.complete(D); 217 } 218 }; 219 220 /// A class for parsing a field declarator. 221 class ParsingFieldDeclarator : public FieldDeclarator { 222 ParsingDeclRAIIObject ParsingRAII; 223 224 public: ParsingFieldDeclarator(Parser & P,const ParsingDeclSpec & DS)225 ParsingFieldDeclarator(Parser &P, const ParsingDeclSpec &DS) 226 : FieldDeclarator(DS), ParsingRAII(P, &DS.getDelayedDiagnosticPool()) { 227 } 228 getDeclSpec()229 const ParsingDeclSpec &getDeclSpec() const { 230 return static_cast<const ParsingDeclSpec&>(D.getDeclSpec()); 231 } 232 getMutableDeclSpec()233 ParsingDeclSpec &getMutableDeclSpec() const { 234 return const_cast<ParsingDeclSpec&>(getDeclSpec()); 235 } 236 complete(Decl * D)237 void complete(Decl *D) { 238 ParsingRAII.complete(D); 239 } 240 }; 241 242 /// ExtensionRAIIObject - This saves the state of extension warnings when 243 /// constructed and disables them. When destructed, it restores them back to 244 /// the way they used to be. This is used to handle __extension__ in the 245 /// parser. 246 class ExtensionRAIIObject { 247 ExtensionRAIIObject(const ExtensionRAIIObject &) LLVM_DELETED_FUNCTION; 248 void operator=(const ExtensionRAIIObject &) LLVM_DELETED_FUNCTION; 249 250 DiagnosticsEngine &Diags; 251 public: ExtensionRAIIObject(DiagnosticsEngine & diags)252 ExtensionRAIIObject(DiagnosticsEngine &diags) : Diags(diags) { 253 Diags.IncrementAllExtensionsSilenced(); 254 } 255 ~ExtensionRAIIObject()256 ~ExtensionRAIIObject() { 257 Diags.DecrementAllExtensionsSilenced(); 258 } 259 }; 260 261 /// ColonProtectionRAIIObject - This sets the Parser::ColonIsSacred bool and 262 /// restores it when destroyed. This says that "foo:" should not be 263 /// considered a possible typo for "foo::" for error recovery purposes. 264 class ColonProtectionRAIIObject { 265 Parser &P; 266 bool OldVal; 267 public: 268 ColonProtectionRAIIObject(Parser &p, bool Value = true) P(p)269 : P(p), OldVal(P.ColonIsSacred) { 270 P.ColonIsSacred = Value; 271 } 272 273 /// restore - This can be used to restore the state early, before the dtor 274 /// is run. restore()275 void restore() { 276 P.ColonIsSacred = OldVal; 277 } 278 ~ColonProtectionRAIIObject()279 ~ColonProtectionRAIIObject() { 280 restore(); 281 } 282 }; 283 284 /// \brief RAII object that makes '>' behave either as an operator 285 /// or as the closing angle bracket for a template argument list. 286 class GreaterThanIsOperatorScope { 287 bool &GreaterThanIsOperator; 288 bool OldGreaterThanIsOperator; 289 public: GreaterThanIsOperatorScope(bool & GTIO,bool Val)290 GreaterThanIsOperatorScope(bool >IO, bool Val) 291 : GreaterThanIsOperator(GTIO), OldGreaterThanIsOperator(GTIO) { 292 GreaterThanIsOperator = Val; 293 } 294 ~GreaterThanIsOperatorScope()295 ~GreaterThanIsOperatorScope() { 296 GreaterThanIsOperator = OldGreaterThanIsOperator; 297 } 298 }; 299 300 class InMessageExpressionRAIIObject { 301 bool &InMessageExpression; 302 bool OldValue; 303 304 public: InMessageExpressionRAIIObject(Parser & P,bool Value)305 InMessageExpressionRAIIObject(Parser &P, bool Value) 306 : InMessageExpression(P.InMessageExpression), 307 OldValue(P.InMessageExpression) { 308 InMessageExpression = Value; 309 } 310 ~InMessageExpressionRAIIObject()311 ~InMessageExpressionRAIIObject() { 312 InMessageExpression = OldValue; 313 } 314 }; 315 316 /// \brief RAII object that makes sure paren/bracket/brace count is correct 317 /// after declaration/statement parsing, even when there's a parsing error. 318 class ParenBraceBracketBalancer { 319 Parser &P; 320 unsigned short ParenCount, BracketCount, BraceCount; 321 public: ParenBraceBracketBalancer(Parser & p)322 ParenBraceBracketBalancer(Parser &p) 323 : P(p), ParenCount(p.ParenCount), BracketCount(p.BracketCount), 324 BraceCount(p.BraceCount) { } 325 ~ParenBraceBracketBalancer()326 ~ParenBraceBracketBalancer() { 327 P.ParenCount = ParenCount; 328 P.BracketCount = BracketCount; 329 P.BraceCount = BraceCount; 330 } 331 }; 332 333 class PoisonSEHIdentifiersRAIIObject { 334 PoisonIdentifierRAIIObject Ident_AbnormalTermination; 335 PoisonIdentifierRAIIObject Ident_GetExceptionCode; 336 PoisonIdentifierRAIIObject Ident_GetExceptionInfo; 337 PoisonIdentifierRAIIObject Ident__abnormal_termination; 338 PoisonIdentifierRAIIObject Ident__exception_code; 339 PoisonIdentifierRAIIObject Ident__exception_info; 340 PoisonIdentifierRAIIObject Ident___abnormal_termination; 341 PoisonIdentifierRAIIObject Ident___exception_code; 342 PoisonIdentifierRAIIObject Ident___exception_info; 343 public: PoisonSEHIdentifiersRAIIObject(Parser & Self,bool NewValue)344 PoisonSEHIdentifiersRAIIObject(Parser &Self, bool NewValue) 345 : Ident_AbnormalTermination(Self.Ident_AbnormalTermination, NewValue), 346 Ident_GetExceptionCode(Self.Ident_GetExceptionCode, NewValue), 347 Ident_GetExceptionInfo(Self.Ident_GetExceptionInfo, NewValue), 348 Ident__abnormal_termination(Self.Ident__abnormal_termination, NewValue), 349 Ident__exception_code(Self.Ident__exception_code, NewValue), 350 Ident__exception_info(Self.Ident__exception_info, NewValue), 351 Ident___abnormal_termination(Self.Ident___abnormal_termination, NewValue), 352 Ident___exception_code(Self.Ident___exception_code, NewValue), 353 Ident___exception_info(Self.Ident___exception_info, NewValue) { 354 } 355 }; 356 357 /// \brief RAII class that helps handle the parsing of an open/close delimiter 358 /// pair, such as braces { ... } or parentheses ( ... ). 359 class BalancedDelimiterTracker : public GreaterThanIsOperatorScope { 360 Parser& P; 361 tok::TokenKind Kind, Close; 362 SourceLocation (Parser::*Consumer)(); 363 SourceLocation LOpen, LClose; 364 getDepth()365 unsigned short &getDepth() { 366 switch (Kind) { 367 case tok::l_brace: return P.BraceCount; 368 case tok::l_square: return P.BracketCount; 369 case tok::l_paren: return P.ParenCount; 370 default: llvm_unreachable("Wrong token kind"); 371 } 372 } 373 374 enum { MaxDepth = 256 }; 375 376 bool diagnoseOverflow(); 377 bool diagnoseMissingClose(); 378 379 public: BalancedDelimiterTracker(Parser & p,tok::TokenKind k)380 BalancedDelimiterTracker(Parser& p, tok::TokenKind k) 381 : GreaterThanIsOperatorScope(p.GreaterThanIsOperator, true), 382 P(p), Kind(k) 383 { 384 switch (Kind) { 385 default: llvm_unreachable("Unexpected balanced token"); 386 case tok::l_brace: 387 Close = tok::r_brace; 388 Consumer = &Parser::ConsumeBrace; 389 break; 390 case tok::l_paren: 391 Close = tok::r_paren; 392 Consumer = &Parser::ConsumeParen; 393 break; 394 395 case tok::l_square: 396 Close = tok::r_square; 397 Consumer = &Parser::ConsumeBracket; 398 break; 399 } 400 } 401 getOpenLocation()402 SourceLocation getOpenLocation() const { return LOpen; } getCloseLocation()403 SourceLocation getCloseLocation() const { return LClose; } getRange()404 SourceRange getRange() const { return SourceRange(LOpen, LClose); } 405 consumeOpen()406 bool consumeOpen() { 407 if (!P.Tok.is(Kind)) 408 return true; 409 410 if (getDepth() < P.getLangOpts().BracketDepth) { 411 LOpen = (P.*Consumer)(); 412 return false; 413 } 414 415 return diagnoseOverflow(); 416 } 417 418 bool expectAndConsume(unsigned DiagID, 419 const char *Msg = "", 420 tok::TokenKind SkipToTok = tok::unknown); consumeClose()421 bool consumeClose() { 422 if (P.Tok.is(Close)) { 423 LClose = (P.*Consumer)(); 424 return false; 425 } 426 427 return diagnoseMissingClose(); 428 } 429 void skipToEnd(); 430 }; 431 432 } // end namespace clang 433 434 #endif 435