blob: 40f96e95500640933369b77e79c3925a57ed1ec1 [file] [log] [blame]
// Copyright (c) 2015, the Dart project authors. Please see the AUTHORS file
// for details. All rights reserved. Use of this source code is governed by a
// BSD-style license that can be found in the LICENSE file.
library fasta.scanner.precedence;
import '../../scanner/token.dart';
import 'token_constants.dart';
class PrecedenceInfo implements TokenType {
final String value;
final String name;
final int precedence;
final int kind;
final bool isOperator;
final bool isUserDefinableOperator;
const PrecedenceInfo(this.value, this.name, this.precedence, this.kind,
{this.isOperator: false, this.isUserDefinableOperator: false});
toString() => 'PrecedenceInfo($value, $precedence, $kind)';
static const List<PrecedenceInfo> all = const <PrecedenceInfo>[
BACKPING_INFO,
BACKSLASH_INFO,
PERIOD_PERIOD_PERIOD_INFO,
PERIOD_PERIOD_INFO,
BANG_INFO,
COLON_INFO,
INDEX_INFO,
MINUS_MINUS_INFO,
PLUS_PLUS_INFO,
TILDE_INFO,
FUNCTION_INFO,
HASH_INFO,
INDEX_EQ_INFO,
SEMICOLON_INFO,
COMMA_INFO,
AT_INFO,
AMPERSAND_EQ_INFO,
BAR_EQ_INFO,
CARET_EQ_INFO,
EQ_INFO,
GT_GT_EQ_INFO,
LT_LT_EQ_INFO,
MINUS_EQ_INFO,
PERCENT_EQ_INFO,
PLUS_EQ_INFO,
SLASH_EQ_INFO,
STAR_EQ_INFO,
TILDE_SLASH_EQ_INFO,
QUESTION_QUESTION_EQ_INFO,
QUESTION_INFO,
QUESTION_QUESTION_INFO,
BAR_BAR_INFO,
AMPERSAND_AMPERSAND_INFO,
BAR_INFO,
CARET_INFO,
AMPERSAND_INFO,
BANG_EQ_EQ_INFO,
BANG_EQ_INFO,
EQ_EQ_EQ_INFO,
EQ_EQ_INFO,
GT_EQ_INFO,
GT_INFO,
IS_INFO,
AS_INFO,
LT_EQ_INFO,
LT_INFO,
GT_GT_INFO,
LT_LT_INFO,
MINUS_INFO,
PLUS_INFO,
PERCENT_INFO,
SLASH_INFO,
STAR_INFO,
TILDE_SLASH_INFO,
PERIOD_INFO,
QUESTION_PERIOD_INFO,
KEYWORD_INFO,
EOF_INFO,
IDENTIFIER_INFO,
BAD_INPUT_INFO,
OPEN_PAREN_INFO,
CLOSE_PAREN_INFO,
OPEN_CURLY_BRACKET_INFO,
CLOSE_CURLY_BRACKET_INFO,
INT_INFO,
STRING_INFO,
OPEN_SQUARE_BRACKET_INFO,
CLOSE_SQUARE_BRACKET_INFO,
DOUBLE_INFO,
STRING_INTERPOLATION_INFO,
STRING_INTERPOLATION_IDENTIFIER_INFO,
HEXADECIMAL_INFO,
SINGLE_LINE_COMMENT_INFO,
MULTI_LINE_COMMENT_INFO,
];
@override
bool get isAdditiveOperator => precedence == ADDITIVE_PRECEDENCE;
@override
bool get isAssignmentOperator => precedence == ASSIGNMENT_PRECEDENCE;
@override
bool get isAssociativeOperator =>
this == AMPERSAND_INFO ||
this == AMPERSAND_AMPERSAND_INFO ||
this == BAR_INFO ||
this == BAR_BAR_INFO ||
this == CARET_INFO ||
this == PLUS_INFO ||
this == STAR_INFO;
@override
bool get isEqualityOperator => this == BANG_EQ_INFO || this == EQ_EQ_INFO;
@override
bool get isIncrementOperator =>
this == PLUS_PLUS_INFO || this == MINUS_MINUS_INFO;
@override
bool get isMultiplicativeOperator => precedence == MULTIPLICATIVE_PRECEDENCE;
@override
bool get isRelationalOperator =>
this == LT_INFO ||
this == LT_EQ_INFO ||
this == GT_INFO ||
this == GT_EQ_INFO;
@override
bool get isShiftOperator => precedence == SHIFT_PRECEDENCE;
@override
bool get isUnaryPostfixOperator => precedence == POSTFIX_PRECEDENCE;
@override
bool get isUnaryPrefixOperator =>
precedence == PREFIX_PRECEDENCE ||
this == PLUS_PLUS_INFO ||
this == MINUS_MINUS_INFO;
@override
String get lexeme => value;
}
// TODO(ahe): The following are not tokens in Dart.
const PrecedenceInfo BACKPING_INFO =
const PrecedenceInfo('`', 'BACKPING', 0, BACKPING_TOKEN);
const PrecedenceInfo BACKSLASH_INFO =
const PrecedenceInfo('\\', 'BACKSLASH', 0, BACKSLASH_TOKEN);
const PrecedenceInfo PERIOD_PERIOD_PERIOD_INFO = const PrecedenceInfo(
'...', 'PERIOD_PERIOD_PERIOD', 0, PERIOD_PERIOD_PERIOD_TOKEN);
/**
* The cascade operator has the lowest precedence of any operator
* except assignment.
*/
const int CASCADE_PRECEDENCE = 2;
const PrecedenceInfo PERIOD_PERIOD_INFO = const PrecedenceInfo(
'..', 'PERIOD_PERIOD', CASCADE_PRECEDENCE, PERIOD_PERIOD_TOKEN,
isOperator: true);
const int PREFIX_PRECEDENCE = 15;
const PrecedenceInfo BANG_INFO = const PrecedenceInfo(
'!', 'BANG', PREFIX_PRECEDENCE, BANG_TOKEN,
isOperator: true);
const PrecedenceInfo COLON_INFO =
const PrecedenceInfo(':', 'COLON', 0, COLON_TOKEN);
const PrecedenceInfo INDEX_INFO = const PrecedenceInfo(
'[]', 'INDEX', 0, INDEX_TOKEN,
isOperator: true, isUserDefinableOperator: true);
const PrecedenceInfo MINUS_MINUS_INFO = const PrecedenceInfo(
'--', 'MINUS_MINUS', POSTFIX_PRECEDENCE, MINUS_MINUS_TOKEN,
isOperator: true);
const PrecedenceInfo PLUS_PLUS_INFO = const PrecedenceInfo(
'++', 'PLUS_PLUS', POSTFIX_PRECEDENCE, PLUS_PLUS_TOKEN,
isOperator: true);
const PrecedenceInfo TILDE_INFO = const PrecedenceInfo(
'~', 'TILDE', PREFIX_PRECEDENCE, TILDE_TOKEN,
isOperator: true, isUserDefinableOperator: true);
const PrecedenceInfo FUNCTION_INFO =
const PrecedenceInfo('=>', 'FUNCTION', 0, FUNCTION_TOKEN);
const PrecedenceInfo HASH_INFO =
const PrecedenceInfo('#', 'HASH', 0, HASH_TOKEN);
const PrecedenceInfo INDEX_EQ_INFO = const PrecedenceInfo(
'[]=', 'INDEX_EQ', 0, INDEX_EQ_TOKEN,
isOperator: true, isUserDefinableOperator: true);
const PrecedenceInfo SEMICOLON_INFO =
const PrecedenceInfo(';', 'SEMICOLON', 0, SEMICOLON_TOKEN);
const PrecedenceInfo COMMA_INFO =
const PrecedenceInfo(',', 'COMMA', 0, COMMA_TOKEN);
const PrecedenceInfo AT_INFO = const PrecedenceInfo('@', 'AT', 0, AT_TOKEN);
// Assignment operators.
const int ASSIGNMENT_PRECEDENCE = 1;
const PrecedenceInfo AMPERSAND_EQ_INFO = const PrecedenceInfo(
'&=', 'AMPERSAND_EQ', ASSIGNMENT_PRECEDENCE, AMPERSAND_EQ_TOKEN,
isOperator: true);
const PrecedenceInfo BAR_EQ_INFO = const PrecedenceInfo(
'|=', 'BAR_EQ', ASSIGNMENT_PRECEDENCE, BAR_EQ_TOKEN,
isOperator: true);
const PrecedenceInfo CARET_EQ_INFO = const PrecedenceInfo(
'^=', 'CARET_EQ', ASSIGNMENT_PRECEDENCE, CARET_EQ_TOKEN,
isOperator: true);
const PrecedenceInfo EQ_INFO = const PrecedenceInfo(
'=', 'EQ', ASSIGNMENT_PRECEDENCE, EQ_TOKEN,
isOperator: true);
const PrecedenceInfo GT_GT_EQ_INFO = const PrecedenceInfo(
'>>=', 'GT_GT_EQ', ASSIGNMENT_PRECEDENCE, GT_GT_EQ_TOKEN,
isOperator: true);
const PrecedenceInfo LT_LT_EQ_INFO = const PrecedenceInfo(
'<<=', 'LT_LT_EQ', ASSIGNMENT_PRECEDENCE, LT_LT_EQ_TOKEN,
isOperator: true);
const PrecedenceInfo MINUS_EQ_INFO = const PrecedenceInfo(
'-=', 'MINUS_EQ', ASSIGNMENT_PRECEDENCE, MINUS_EQ_TOKEN,
isOperator: true);
const PrecedenceInfo PERCENT_EQ_INFO = const PrecedenceInfo(
'%=', 'PERCENT_EQ', ASSIGNMENT_PRECEDENCE, PERCENT_EQ_TOKEN,
isOperator: true);
const PrecedenceInfo PLUS_EQ_INFO = const PrecedenceInfo(
'+=', 'PLUS_EQ', ASSIGNMENT_PRECEDENCE, PLUS_EQ_TOKEN,
isOperator: true);
const PrecedenceInfo SLASH_EQ_INFO = const PrecedenceInfo(
'/=', 'SLASH_EQ', ASSIGNMENT_PRECEDENCE, SLASH_EQ_TOKEN,
isOperator: true);
const PrecedenceInfo STAR_EQ_INFO = const PrecedenceInfo(
'*=', 'STAR_EQ', ASSIGNMENT_PRECEDENCE, STAR_EQ_TOKEN,
isOperator: true);
const PrecedenceInfo TILDE_SLASH_EQ_INFO = const PrecedenceInfo(
'~/=', 'TILDE_SLASH_EQ', ASSIGNMENT_PRECEDENCE, TILDE_SLASH_EQ_TOKEN,
isOperator: true);
const PrecedenceInfo QUESTION_QUESTION_EQ_INFO = const PrecedenceInfo('??=',
'QUESTION_QUESTION_EQ', ASSIGNMENT_PRECEDENCE, QUESTION_QUESTION_EQ_TOKEN,
isOperator: true);
const PrecedenceInfo QUESTION_INFO =
const PrecedenceInfo('?', 'QUESTION', 3, QUESTION_TOKEN, isOperator: true);
const PrecedenceInfo QUESTION_QUESTION_INFO = const PrecedenceInfo(
'??', 'QUESTION_QUESTION', 4, QUESTION_QUESTION_TOKEN,
isOperator: true);
const PrecedenceInfo BAR_BAR_INFO =
const PrecedenceInfo('||', 'BAR_BAR', 5, BAR_BAR_TOKEN, isOperator: true);
const PrecedenceInfo AMPERSAND_AMPERSAND_INFO = const PrecedenceInfo(
'&&', 'AMPERSAND_AMPERSAND', 6, AMPERSAND_AMPERSAND_TOKEN,
isOperator: true);
const PrecedenceInfo BAR_INFO = const PrecedenceInfo('|', 'BAR', 9, BAR_TOKEN,
isOperator: true, isUserDefinableOperator: true);
const PrecedenceInfo CARET_INFO = const PrecedenceInfo(
'^', 'CARET', 10, CARET_TOKEN,
isOperator: true, isUserDefinableOperator: true);
const PrecedenceInfo AMPERSAND_INFO = const PrecedenceInfo(
'&', 'AMPERSAND', 11, AMPERSAND_TOKEN,
isOperator: true, isUserDefinableOperator: true);
// Equality operators.
const int EQUALITY_PRECEDENCE = 7;
const PrecedenceInfo BANG_EQ_EQ_INFO = const PrecedenceInfo(
'!==', 'BANG_EQ_EQ', EQUALITY_PRECEDENCE, BANG_EQ_EQ_TOKEN);
const PrecedenceInfo BANG_EQ_INFO = const PrecedenceInfo(
'!=', 'BANG_EQ', EQUALITY_PRECEDENCE, BANG_EQ_TOKEN,
isOperator: true);
const PrecedenceInfo EQ_EQ_EQ_INFO = const PrecedenceInfo(
'===', 'EQ_EQ_EQ', EQUALITY_PRECEDENCE, EQ_EQ_EQ_TOKEN);
const PrecedenceInfo EQ_EQ_INFO = const PrecedenceInfo(
'==', 'EQ_EQ', EQUALITY_PRECEDENCE, EQ_EQ_TOKEN,
isOperator: true, isUserDefinableOperator: true);
// Relational operators.
const int RELATIONAL_PRECEDENCE = 8;
const PrecedenceInfo GT_EQ_INFO = const PrecedenceInfo(
'>=', 'GT_EQ', RELATIONAL_PRECEDENCE, GT_EQ_TOKEN,
isOperator: true, isUserDefinableOperator: true);
const PrecedenceInfo GT_INFO = const PrecedenceInfo(
'>', 'GT', RELATIONAL_PRECEDENCE, GT_TOKEN,
isOperator: true, isUserDefinableOperator: true);
const PrecedenceInfo IS_INFO =
const PrecedenceInfo('is', 'IS', RELATIONAL_PRECEDENCE, KEYWORD_TOKEN);
const PrecedenceInfo AS_INFO =
const PrecedenceInfo('as', 'AS', RELATIONAL_PRECEDENCE, KEYWORD_TOKEN);
const PrecedenceInfo LT_EQ_INFO = const PrecedenceInfo(
'<=', 'LT_EQ', RELATIONAL_PRECEDENCE, LT_EQ_TOKEN,
isOperator: true, isUserDefinableOperator: true);
const PrecedenceInfo LT_INFO = const PrecedenceInfo(
'<', 'LT', RELATIONAL_PRECEDENCE, LT_TOKEN,
isOperator: true, isUserDefinableOperator: true);
// Shift operators.
const int SHIFT_PRECEDENCE = 12;
const PrecedenceInfo GT_GT_INFO = const PrecedenceInfo(
'>>', 'GT_GT', SHIFT_PRECEDENCE, GT_GT_TOKEN,
isOperator: true, isUserDefinableOperator: true);
const PrecedenceInfo LT_LT_INFO = const PrecedenceInfo(
'<<', 'LT_LT', SHIFT_PRECEDENCE, LT_LT_TOKEN,
isOperator: true, isUserDefinableOperator: true);
// Additive operators.
const int ADDITIVE_PRECEDENCE = 13;
const PrecedenceInfo MINUS_INFO = const PrecedenceInfo(
'-', 'MINUS', ADDITIVE_PRECEDENCE, MINUS_TOKEN,
isOperator: true, isUserDefinableOperator: true);
const PrecedenceInfo PLUS_INFO = const PrecedenceInfo(
'+', 'PLUS', ADDITIVE_PRECEDENCE, PLUS_TOKEN,
isOperator: true, isUserDefinableOperator: true);
// Multiplicative operators.
const int MULTIPLICATIVE_PRECEDENCE = 14;
const PrecedenceInfo PERCENT_INFO = const PrecedenceInfo(
'%', 'PERCENT', MULTIPLICATIVE_PRECEDENCE, PERCENT_TOKEN,
isOperator: true, isUserDefinableOperator: true);
const PrecedenceInfo SLASH_INFO = const PrecedenceInfo(
'/', 'SLASH', MULTIPLICATIVE_PRECEDENCE, SLASH_TOKEN,
isOperator: true, isUserDefinableOperator: true);
const PrecedenceInfo STAR_INFO = const PrecedenceInfo(
'*', 'STAR', MULTIPLICATIVE_PRECEDENCE, STAR_TOKEN,
isOperator: true, isUserDefinableOperator: true);
const PrecedenceInfo TILDE_SLASH_INFO = const PrecedenceInfo(
'~/', 'TILDE_SLASH', MULTIPLICATIVE_PRECEDENCE, TILDE_SLASH_TOKEN,
isOperator: true, isUserDefinableOperator: true);
const int POSTFIX_PRECEDENCE = 16;
const PrecedenceInfo PERIOD_INFO =
const PrecedenceInfo('.', 'PERIOD', POSTFIX_PRECEDENCE, PERIOD_TOKEN);
const PrecedenceInfo QUESTION_PERIOD_INFO = const PrecedenceInfo(
'?.', 'QUESTION_PERIOD', POSTFIX_PRECEDENCE, QUESTION_PERIOD_TOKEN,
isOperator: true);
const PrecedenceInfo KEYWORD_INFO =
const PrecedenceInfo('keyword', 'KEYWORD', 0, KEYWORD_TOKEN);
const PrecedenceInfo EOF_INFO = const PrecedenceInfo('', 'EOF', 0, EOF_TOKEN);
/// Precedence info used by synthetic tokens that are created during parser
/// recovery (non-analyzer use case).
const PrecedenceInfo RECOVERY_INFO =
const PrecedenceInfo('recovery', 'RECOVERY', 0, RECOVERY_TOKEN);
const PrecedenceInfo IDENTIFIER_INFO =
const PrecedenceInfo('identifier', 'STRING_INT', 0, IDENTIFIER_TOKEN);
const PrecedenceInfo SCRIPT_INFO =
const PrecedenceInfo('script', 'SCRIPT_TAG', 0, SCRIPT_TOKEN);
const PrecedenceInfo BAD_INPUT_INFO =
const PrecedenceInfo('malformed input', 'BAD_INPUT', 0, BAD_INPUT_TOKEN);
const PrecedenceInfo OPEN_PAREN_INFO = const PrecedenceInfo(
'(', 'OPEN_PAREN', POSTFIX_PRECEDENCE, OPEN_PAREN_TOKEN);
const PrecedenceInfo CLOSE_PAREN_INFO =
const PrecedenceInfo(')', 'CLOSE_PAREN', 0, CLOSE_PAREN_TOKEN);
const PrecedenceInfo OPEN_CURLY_BRACKET_INFO = const PrecedenceInfo(
'{', 'OPEN_CURLY_BRACKET', 0, OPEN_CURLY_BRACKET_TOKEN);
const PrecedenceInfo CLOSE_CURLY_BRACKET_INFO = const PrecedenceInfo(
'}', 'CLOSE_CURLY_BRACKET', 0, CLOSE_CURLY_BRACKET_TOKEN);
const PrecedenceInfo INT_INFO =
const PrecedenceInfo('int', 'INT', 0, INT_TOKEN);
const PrecedenceInfo STRING_INFO =
const PrecedenceInfo('string', 'STRING', 0, STRING_TOKEN);
const PrecedenceInfo OPEN_SQUARE_BRACKET_INFO = const PrecedenceInfo(
'[', 'OPEN_SQUARE_BRACKET', POSTFIX_PRECEDENCE, OPEN_SQUARE_BRACKET_TOKEN);
const PrecedenceInfo CLOSE_SQUARE_BRACKET_INFO = const PrecedenceInfo(
']', 'CLOSE_SQUARE_BRACKET', 0, CLOSE_SQUARE_BRACKET_TOKEN);
const PrecedenceInfo DOUBLE_INFO =
const PrecedenceInfo('double', 'DOUBLE', 0, DOUBLE_TOKEN);
const PrecedenceInfo STRING_INTERPOLATION_INFO = const PrecedenceInfo(
'\${', 'STRING_INTERPOLATION_EXPRESSION', 0, STRING_INTERPOLATION_TOKEN);
const PrecedenceInfo STRING_INTERPOLATION_IDENTIFIER_INFO =
const PrecedenceInfo('\$', 'STRING_INTERPOLATION_IDENTIFIER', 0,
STRING_INTERPOLATION_IDENTIFIER_TOKEN);
const PrecedenceInfo HEXADECIMAL_INFO =
const PrecedenceInfo('hexadecimal', 'HEXADECIMAL', 0, HEXADECIMAL_TOKEN);
const PrecedenceInfo SINGLE_LINE_COMMENT_INFO =
const PrecedenceInfo('comment', 'SINGLE_LINE_COMMENT', 0, COMMENT_TOKEN);
const PrecedenceInfo MULTI_LINE_COMMENT_INFO =
const PrecedenceInfo('comment', 'MULTI_LINE_COMMENT', 0, COMMENT_TOKEN);