59d1ed5b20
and preserve our customizations, where necessary.
436 lines
14 KiB
C++
436 lines
14 KiB
C++
//===--- RAIIObjectsForParser.h - RAII helpers for the parser ---*- C++ -*-===//
|
|
//
|
|
// The LLVM Compiler Infrastructure
|
|
//
|
|
// This file is distributed under the University of Illinois Open Source
|
|
// License. See LICENSE.TXT for details.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// This file defines and implements the some simple RAII objects that are used
|
|
// by the parser to manage bits in recursion.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#ifndef LLVM_CLANG_PARSE_RAII_OBJECTS_FOR_PARSER_H
|
|
#define LLVM_CLANG_PARSE_RAII_OBJECTS_FOR_PARSER_H
|
|
|
|
#include "clang/Parse/ParseDiagnostic.h"
|
|
#include "clang/Parse/Parser.h"
|
|
#include "clang/Sema/DelayedDiagnostic.h"
|
|
#include "clang/Sema/Sema.h"
|
|
|
|
namespace clang {
|
|
// TODO: move ParsingClassDefinition here.
|
|
// TODO: move TentativeParsingAction here.
|
|
|
|
/// \brief A RAII object used to temporarily suppress access-like
|
|
/// checking. Access-like checks are those associated with
|
|
/// controlling the use of a declaration, like C++ access control
|
|
/// errors and deprecation warnings. They are contextually
|
|
/// dependent, in that they can only be resolved with full
|
|
/// information about what's being declared. They are also
|
|
/// suppressed in certain contexts, like the template arguments of
|
|
/// an explicit instantiation. However, those suppression contexts
|
|
/// cannot necessarily be fully determined in advance; for
|
|
/// example, something starting like this:
|
|
/// template <> class std::vector<A::PrivateType>
|
|
/// might be the entirety of an explicit instantiation:
|
|
/// template <> class std::vector<A::PrivateType>;
|
|
/// or just an elaborated type specifier:
|
|
/// template <> class std::vector<A::PrivateType> make_vector<>();
|
|
/// Therefore this class collects all the diagnostics and permits
|
|
/// them to be re-delayed in a new context.
|
|
class SuppressAccessChecks {
|
|
Sema &S;
|
|
sema::DelayedDiagnosticPool DiagnosticPool;
|
|
Sema::ParsingDeclState State;
|
|
bool Active;
|
|
|
|
public:
|
|
/// Begin suppressing access-like checks
|
|
SuppressAccessChecks(Parser &P, bool activate = true)
|
|
: S(P.getActions()), DiagnosticPool(nullptr) {
|
|
if (activate) {
|
|
State = S.PushParsingDeclaration(DiagnosticPool);
|
|
Active = true;
|
|
} else {
|
|
Active = false;
|
|
}
|
|
}
|
|
|
|
void done() {
|
|
assert(Active && "trying to end an inactive suppression");
|
|
S.PopParsingDeclaration(State, nullptr);
|
|
Active = false;
|
|
}
|
|
|
|
void redelay() {
|
|
assert(!Active && "redelaying without having ended first");
|
|
if (!DiagnosticPool.pool_empty())
|
|
S.redelayDiagnostics(DiagnosticPool);
|
|
assert(DiagnosticPool.pool_empty());
|
|
}
|
|
|
|
~SuppressAccessChecks() {
|
|
if (Active) done();
|
|
}
|
|
};
|
|
|
|
/// \brief RAII object used to inform the actions that we're
|
|
/// currently parsing a declaration. This is active when parsing a
|
|
/// variable's initializer, but not when parsing the body of a
|
|
/// class or function definition.
|
|
class ParsingDeclRAIIObject {
|
|
Sema &Actions;
|
|
sema::DelayedDiagnosticPool DiagnosticPool;
|
|
Sema::ParsingDeclState State;
|
|
bool Popped;
|
|
|
|
ParsingDeclRAIIObject(const ParsingDeclRAIIObject &) LLVM_DELETED_FUNCTION;
|
|
void operator=(const ParsingDeclRAIIObject &) LLVM_DELETED_FUNCTION;
|
|
|
|
public:
|
|
enum NoParent_t { NoParent };
|
|
ParsingDeclRAIIObject(Parser &P, NoParent_t _)
|
|
: Actions(P.getActions()), DiagnosticPool(nullptr) {
|
|
push();
|
|
}
|
|
|
|
/// Creates a RAII object whose pool is optionally parented by another.
|
|
ParsingDeclRAIIObject(Parser &P,
|
|
const sema::DelayedDiagnosticPool *parentPool)
|
|
: Actions(P.getActions()), DiagnosticPool(parentPool) {
|
|
push();
|
|
}
|
|
|
|
/// Creates a RAII object and, optionally, initialize its
|
|
/// diagnostics pool by stealing the diagnostics from another
|
|
/// RAII object (which is assumed to be the current top pool).
|
|
ParsingDeclRAIIObject(Parser &P, ParsingDeclRAIIObject *other)
|
|
: Actions(P.getActions()),
|
|
DiagnosticPool(other ? other->DiagnosticPool.getParent() : nullptr) {
|
|
if (other) {
|
|
DiagnosticPool.steal(other->DiagnosticPool);
|
|
other->abort();
|
|
}
|
|
push();
|
|
}
|
|
|
|
~ParsingDeclRAIIObject() {
|
|
abort();
|
|
}
|
|
|
|
sema::DelayedDiagnosticPool &getDelayedDiagnosticPool() {
|
|
return DiagnosticPool;
|
|
}
|
|
const sema::DelayedDiagnosticPool &getDelayedDiagnosticPool() const {
|
|
return DiagnosticPool;
|
|
}
|
|
|
|
/// Resets the RAII object for a new declaration.
|
|
void reset() {
|
|
abort();
|
|
push();
|
|
}
|
|
|
|
/// Signals that the context was completed without an appropriate
|
|
/// declaration being parsed.
|
|
void abort() {
|
|
pop(nullptr);
|
|
}
|
|
|
|
void complete(Decl *D) {
|
|
assert(!Popped && "ParsingDeclaration has already been popped!");
|
|
pop(D);
|
|
}
|
|
|
|
/// Unregister this object from Sema, but remember all the
|
|
/// diagnostics that were emitted into it.
|
|
void abortAndRemember() {
|
|
pop(nullptr);
|
|
}
|
|
|
|
private:
|
|
void push() {
|
|
State = Actions.PushParsingDeclaration(DiagnosticPool);
|
|
Popped = false;
|
|
}
|
|
|
|
void pop(Decl *D) {
|
|
if (!Popped) {
|
|
Actions.PopParsingDeclaration(State, D);
|
|
Popped = true;
|
|
}
|
|
}
|
|
};
|
|
|
|
/// A class for parsing a DeclSpec.
|
|
class ParsingDeclSpec : public DeclSpec {
|
|
ParsingDeclRAIIObject ParsingRAII;
|
|
|
|
public:
|
|
ParsingDeclSpec(Parser &P)
|
|
: DeclSpec(P.getAttrFactory()),
|
|
ParsingRAII(P, ParsingDeclRAIIObject::NoParent) {}
|
|
ParsingDeclSpec(Parser &P, ParsingDeclRAIIObject *RAII)
|
|
: DeclSpec(P.getAttrFactory()),
|
|
ParsingRAII(P, RAII) {}
|
|
|
|
const sema::DelayedDiagnosticPool &getDelayedDiagnosticPool() const {
|
|
return ParsingRAII.getDelayedDiagnosticPool();
|
|
}
|
|
|
|
void complete(Decl *D) {
|
|
ParsingRAII.complete(D);
|
|
}
|
|
|
|
void abort() {
|
|
ParsingRAII.abort();
|
|
}
|
|
};
|
|
|
|
/// A class for parsing a declarator.
|
|
class ParsingDeclarator : public Declarator {
|
|
ParsingDeclRAIIObject ParsingRAII;
|
|
|
|
public:
|
|
ParsingDeclarator(Parser &P, const ParsingDeclSpec &DS, TheContext C)
|
|
: Declarator(DS, C), ParsingRAII(P, &DS.getDelayedDiagnosticPool()) {
|
|
}
|
|
|
|
const ParsingDeclSpec &getDeclSpec() const {
|
|
return static_cast<const ParsingDeclSpec&>(Declarator::getDeclSpec());
|
|
}
|
|
|
|
ParsingDeclSpec &getMutableDeclSpec() const {
|
|
return const_cast<ParsingDeclSpec&>(getDeclSpec());
|
|
}
|
|
|
|
void clear() {
|
|
Declarator::clear();
|
|
ParsingRAII.reset();
|
|
}
|
|
|
|
void complete(Decl *D) {
|
|
ParsingRAII.complete(D);
|
|
}
|
|
};
|
|
|
|
/// A class for parsing a field declarator.
|
|
class ParsingFieldDeclarator : public FieldDeclarator {
|
|
ParsingDeclRAIIObject ParsingRAII;
|
|
|
|
public:
|
|
ParsingFieldDeclarator(Parser &P, const ParsingDeclSpec &DS)
|
|
: FieldDeclarator(DS), ParsingRAII(P, &DS.getDelayedDiagnosticPool()) {
|
|
}
|
|
|
|
const ParsingDeclSpec &getDeclSpec() const {
|
|
return static_cast<const ParsingDeclSpec&>(D.getDeclSpec());
|
|
}
|
|
|
|
ParsingDeclSpec &getMutableDeclSpec() const {
|
|
return const_cast<ParsingDeclSpec&>(getDeclSpec());
|
|
}
|
|
|
|
void complete(Decl *D) {
|
|
ParsingRAII.complete(D);
|
|
}
|
|
};
|
|
|
|
/// ExtensionRAIIObject - This saves the state of extension warnings when
|
|
/// constructed and disables them. When destructed, it restores them back to
|
|
/// the way they used to be. This is used to handle __extension__ in the
|
|
/// parser.
|
|
class ExtensionRAIIObject {
|
|
ExtensionRAIIObject(const ExtensionRAIIObject &) LLVM_DELETED_FUNCTION;
|
|
void operator=(const ExtensionRAIIObject &) LLVM_DELETED_FUNCTION;
|
|
|
|
DiagnosticsEngine &Diags;
|
|
public:
|
|
ExtensionRAIIObject(DiagnosticsEngine &diags) : Diags(diags) {
|
|
Diags.IncrementAllExtensionsSilenced();
|
|
}
|
|
|
|
~ExtensionRAIIObject() {
|
|
Diags.DecrementAllExtensionsSilenced();
|
|
}
|
|
};
|
|
|
|
/// ColonProtectionRAIIObject - This sets the Parser::ColonIsSacred bool and
|
|
/// restores it when destroyed. This says that "foo:" should not be
|
|
/// considered a possible typo for "foo::" for error recovery purposes.
|
|
class ColonProtectionRAIIObject {
|
|
Parser &P;
|
|
bool OldVal;
|
|
public:
|
|
ColonProtectionRAIIObject(Parser &p, bool Value = true)
|
|
: P(p), OldVal(P.ColonIsSacred) {
|
|
P.ColonIsSacred = Value;
|
|
}
|
|
|
|
/// restore - This can be used to restore the state early, before the dtor
|
|
/// is run.
|
|
void restore() {
|
|
P.ColonIsSacred = OldVal;
|
|
}
|
|
|
|
~ColonProtectionRAIIObject() {
|
|
restore();
|
|
}
|
|
};
|
|
|
|
/// \brief RAII object that makes '>' behave either as an operator
|
|
/// or as the closing angle bracket for a template argument list.
|
|
class GreaterThanIsOperatorScope {
|
|
bool &GreaterThanIsOperator;
|
|
bool OldGreaterThanIsOperator;
|
|
public:
|
|
GreaterThanIsOperatorScope(bool >IO, bool Val)
|
|
: GreaterThanIsOperator(GTIO), OldGreaterThanIsOperator(GTIO) {
|
|
GreaterThanIsOperator = Val;
|
|
}
|
|
|
|
~GreaterThanIsOperatorScope() {
|
|
GreaterThanIsOperator = OldGreaterThanIsOperator;
|
|
}
|
|
};
|
|
|
|
class InMessageExpressionRAIIObject {
|
|
bool &InMessageExpression;
|
|
bool OldValue;
|
|
|
|
public:
|
|
InMessageExpressionRAIIObject(Parser &P, bool Value)
|
|
: InMessageExpression(P.InMessageExpression),
|
|
OldValue(P.InMessageExpression) {
|
|
InMessageExpression = Value;
|
|
}
|
|
|
|
~InMessageExpressionRAIIObject() {
|
|
InMessageExpression = OldValue;
|
|
}
|
|
};
|
|
|
|
/// \brief RAII object that makes sure paren/bracket/brace count is correct
|
|
/// after declaration/statement parsing, even when there's a parsing error.
|
|
class ParenBraceBracketBalancer {
|
|
Parser &P;
|
|
unsigned short ParenCount, BracketCount, BraceCount;
|
|
public:
|
|
ParenBraceBracketBalancer(Parser &p)
|
|
: P(p), ParenCount(p.ParenCount), BracketCount(p.BracketCount),
|
|
BraceCount(p.BraceCount) { }
|
|
|
|
~ParenBraceBracketBalancer() {
|
|
P.ParenCount = ParenCount;
|
|
P.BracketCount = BracketCount;
|
|
P.BraceCount = BraceCount;
|
|
}
|
|
};
|
|
|
|
class PoisonSEHIdentifiersRAIIObject {
|
|
PoisonIdentifierRAIIObject Ident_AbnormalTermination;
|
|
PoisonIdentifierRAIIObject Ident_GetExceptionCode;
|
|
PoisonIdentifierRAIIObject Ident_GetExceptionInfo;
|
|
PoisonIdentifierRAIIObject Ident__abnormal_termination;
|
|
PoisonIdentifierRAIIObject Ident__exception_code;
|
|
PoisonIdentifierRAIIObject Ident__exception_info;
|
|
PoisonIdentifierRAIIObject Ident___abnormal_termination;
|
|
PoisonIdentifierRAIIObject Ident___exception_code;
|
|
PoisonIdentifierRAIIObject Ident___exception_info;
|
|
public:
|
|
PoisonSEHIdentifiersRAIIObject(Parser &Self, bool NewValue)
|
|
: Ident_AbnormalTermination(Self.Ident_AbnormalTermination, NewValue),
|
|
Ident_GetExceptionCode(Self.Ident_GetExceptionCode, NewValue),
|
|
Ident_GetExceptionInfo(Self.Ident_GetExceptionInfo, NewValue),
|
|
Ident__abnormal_termination(Self.Ident__abnormal_termination, NewValue),
|
|
Ident__exception_code(Self.Ident__exception_code, NewValue),
|
|
Ident__exception_info(Self.Ident__exception_info, NewValue),
|
|
Ident___abnormal_termination(Self.Ident___abnormal_termination, NewValue),
|
|
Ident___exception_code(Self.Ident___exception_code, NewValue),
|
|
Ident___exception_info(Self.Ident___exception_info, NewValue) {
|
|
}
|
|
};
|
|
|
|
/// \brief RAII class that helps handle the parsing of an open/close delimiter
|
|
/// pair, such as braces { ... } or parentheses ( ... ).
|
|
class BalancedDelimiterTracker : public GreaterThanIsOperatorScope {
|
|
Parser& P;
|
|
tok::TokenKind Kind, Close, FinalToken;
|
|
SourceLocation (Parser::*Consumer)();
|
|
SourceLocation LOpen, LClose;
|
|
|
|
unsigned short &getDepth() {
|
|
switch (Kind) {
|
|
case tok::l_brace: return P.BraceCount;
|
|
case tok::l_square: return P.BracketCount;
|
|
case tok::l_paren: return P.ParenCount;
|
|
default: llvm_unreachable("Wrong token kind");
|
|
}
|
|
}
|
|
|
|
enum { MaxDepth = 256 };
|
|
|
|
bool diagnoseOverflow();
|
|
bool diagnoseMissingClose();
|
|
|
|
public:
|
|
BalancedDelimiterTracker(Parser& p, tok::TokenKind k,
|
|
tok::TokenKind FinalToken = tok::semi)
|
|
: GreaterThanIsOperatorScope(p.GreaterThanIsOperator, true),
|
|
P(p), Kind(k), FinalToken(FinalToken)
|
|
{
|
|
switch (Kind) {
|
|
default: llvm_unreachable("Unexpected balanced token");
|
|
case tok::l_brace:
|
|
Close = tok::r_brace;
|
|
Consumer = &Parser::ConsumeBrace;
|
|
break;
|
|
case tok::l_paren:
|
|
Close = tok::r_paren;
|
|
Consumer = &Parser::ConsumeParen;
|
|
break;
|
|
|
|
case tok::l_square:
|
|
Close = tok::r_square;
|
|
Consumer = &Parser::ConsumeBracket;
|
|
break;
|
|
}
|
|
}
|
|
|
|
SourceLocation getOpenLocation() const { return LOpen; }
|
|
SourceLocation getCloseLocation() const { return LClose; }
|
|
SourceRange getRange() const { return SourceRange(LOpen, LClose); }
|
|
|
|
bool consumeOpen() {
|
|
if (!P.Tok.is(Kind))
|
|
return true;
|
|
|
|
if (getDepth() < P.getLangOpts().BracketDepth) {
|
|
LOpen = (P.*Consumer)();
|
|
return false;
|
|
}
|
|
|
|
return diagnoseOverflow();
|
|
}
|
|
|
|
bool expectAndConsume(unsigned DiagID = diag::err_expected,
|
|
const char *Msg = "",
|
|
tok::TokenKind SkipToTok = tok::unknown);
|
|
bool consumeClose() {
|
|
if (P.Tok.is(Close)) {
|
|
LClose = (P.*Consumer)();
|
|
return false;
|
|
}
|
|
|
|
return diagnoseMissingClose();
|
|
}
|
|
void skipToEnd();
|
|
};
|
|
|
|
} // end namespace clang
|
|
|
|
#endif
|