| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183 |
- // Part of the Carbon Language project, under the Apache License v2.0 with LLVM
- // Exceptions. See /LICENSE for license information.
- // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
- #ifndef CARBON_TOOLCHAIN_PARSE_PRECEDENCE_H_
- #define CARBON_TOOLCHAIN_PARSE_PRECEDENCE_H_
- #include <optional>
- #include "toolchain/lex/token_kind.h"
- namespace Carbon::Parse {
- // Given two operators `$` and `@`, and an expression `a $ b @ c`, how should
- // the expression be parsed?
- enum class OperatorPriority : int8_t {
- // The left operator has higher precedence: `(a $ b) @ c`.
- LeftFirst = -1,
- // The expression is ambiguous.
- Ambiguous = 0,
- // The right operator has higher precedence: `a $ (b @ c)`.
- RightFirst = 1,
- };
- enum class Associativity : int8_t {
- LeftToRight = -1,
- None = 0,
- RightToLeft = 1
- };
- // A precedence group associated with an operator or expression.
- class PrecedenceGroup {
- public:
- struct Trailing;
- // Objects of this type should only be constructed using the static factory
- // functions below.
- PrecedenceGroup() = delete;
- // Get the sentinel precedence level for a postfix expression. All operators
- // have lower precedence than this.
- static auto ForPostfixExpr() -> PrecedenceGroup;
- // Get the precedence level for a top-level or parenthesized expression. All
- // expression operators have higher precedence than this.
- static auto ForTopLevelExpr() -> PrecedenceGroup;
- // Get the sentinel precedence level for a statement context. All operators,
- // including statement operators like `=` and `++`, have higher precedence
- // than this.
- static auto ForExprStatement() -> PrecedenceGroup;
- // Get the precedence level at which to parse a type expression. All type
- // operators have higher precedence than this.
- static auto ForType() -> PrecedenceGroup;
- // Get the precedence level at which to parse the type expression between
- // `impl` and `as`.
- static auto ForImplAs() -> PrecedenceGroup;
- // Get the precedence level at which to parse expressions in requirements
- // after `where` or `require`.
- static auto ForRequirements() -> PrecedenceGroup;
- // Look up the operator information of the given prefix operator token, or
- // return std::nullopt if the given token is not a prefix operator.
- static auto ForLeading(Lex::TokenKind kind) -> std::optional<PrecedenceGroup>;
- // Look up the operator information of the given infix or postfix operator
- // token, or return std::nullopt if the given token is not an infix or postfix
- // operator. `infix` indicates whether this is a valid infix operator, but is
- // only considered if the same operator symbol is available as both infix and
- // postfix.
- static auto ForTrailing(Lex::TokenKind kind, bool infix)
- -> std::optional<Trailing>;
- friend auto operator==(PrecedenceGroup lhs, PrecedenceGroup rhs) -> bool {
- return lhs.level_ == rhs.level_;
- }
- // Compare the precedence levels for two adjacent operators.
- static auto GetPriority(PrecedenceGroup left, PrecedenceGroup right)
- -> OperatorPriority;
- // Get the associativity of this precedence group.
- auto GetAssociativity() const -> Associativity {
- return static_cast<Associativity>(GetPriority(*this, *this));
- }
- private:
- enum PrecedenceLevel : int8_t;
- struct OperatorPriorityTable;
- static const int8_t NumPrecedenceLevels;
- // We rely on implicit conversions via `int8_t` for enumerators defined in the
- // implementation.
- // NOLINTNEXTLINE(google-explicit-constructor)
- PrecedenceGroup(int8_t level) : level_(level) {}
- // The precedence level.
- int8_t level_;
- };
- // Precedence information for a trailing operator.
- struct PrecedenceGroup::Trailing {
- // The precedence level.
- PrecedenceGroup level;
- // `true` if this is an infix binary operator, `false` if this is a postfix
- // unary operator.
- bool is_binary;
- };
- ////////////////////////////////////////////////////////////////////////////////
- //
- // Only implementation details below this point.
- //
- ////////////////////////////////////////////////////////////////////////////////
- enum PrecedenceGroup::PrecedenceLevel : int8_t {
- // Sentinel representing the absence of any operator.
- Highest,
- // Terms.
- TermPrefix,
- // Numeric.
- IncrementDecrement,
- NumericPrefix,
- Modulo,
- Multiplicative,
- Additive,
- // Bitwise.
- BitwisePrefix,
- BitwiseAnd,
- BitwiseOr,
- BitwiseXor,
- BitShift,
- // Type formation.
- TypePrefix,
- TypePostfix,
- // `where` keyword.
- Where,
- // Casts.
- As,
- // Logical.
- LogicalPrefix,
- Relational,
- LogicalAnd,
- LogicalOr,
- // Conditional.
- If,
- // Assignment.
- Assignment,
- // Sentinel representing a context in which any operator can appear.
- Lowest,
- };
- inline auto PrecedenceGroup::ForPostfixExpr() -> PrecedenceGroup {
- return PrecedenceGroup(Highest);
- }
- inline auto PrecedenceGroup::ForTopLevelExpr() -> PrecedenceGroup {
- return PrecedenceGroup(If);
- }
- inline auto PrecedenceGroup::ForExprStatement() -> PrecedenceGroup {
- return PrecedenceGroup(Lowest);
- }
- inline auto PrecedenceGroup::ForType() -> PrecedenceGroup {
- return ForTopLevelExpr();
- }
- inline auto PrecedenceGroup::ForImplAs() -> PrecedenceGroup {
- return PrecedenceGroup(As);
- }
- inline auto PrecedenceGroup::ForRequirements() -> PrecedenceGroup {
- return PrecedenceGroup(Where);
- }
- } // namespace Carbon::Parse
- #endif // CARBON_TOOLCHAIN_PARSE_PRECEDENCE_H_
|