2012-10-02 09:12:39 +02:00
|
|
|
/****************************************************************************
|
2010-01-18 13:13:34 +01:00
|
|
|
**
|
2016-01-15 14:58:39 +01:00
|
|
|
** Copyright (C) 2016 The Qt Company Ltd.
|
|
|
|
|
** Contact: https://www.qt.io/licensing/
|
2010-01-18 13:13:34 +01:00
|
|
|
**
|
2012-10-02 09:12:39 +02:00
|
|
|
** This file is part of Qt Creator.
|
2010-01-18 13:13:34 +01:00
|
|
|
**
|
2012-10-02 09:12:39 +02:00
|
|
|
** Commercial License Usage
|
|
|
|
|
** Licensees holding valid commercial Qt licenses may use this file in
|
|
|
|
|
** accordance with the commercial license agreement provided with the
|
|
|
|
|
** Software or, alternatively, in accordance with the terms contained in
|
2016-01-15 14:58:39 +01:00
|
|
|
** a written agreement between you and The Qt Company. For licensing terms
|
|
|
|
|
** and conditions see https://www.qt.io/terms-conditions. For further
|
|
|
|
|
** information use the contact form at https://www.qt.io/contact-us.
|
2011-05-12 13:25:35 +02:00
|
|
|
**
|
2016-01-15 14:58:39 +01:00
|
|
|
** GNU General Public License Usage
|
|
|
|
|
** Alternatively, this file may be used under the terms of the GNU
|
|
|
|
|
** General Public License version 3 as published by the Free Software
|
|
|
|
|
** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT
|
|
|
|
|
** included in the packaging of this file. Please review the following
|
|
|
|
|
** information to ensure the GNU General Public License requirements will
|
|
|
|
|
** be met: https://www.gnu.org/licenses/gpl-3.0.html.
|
2010-01-18 13:13:34 +01:00
|
|
|
**
|
2012-10-02 09:12:39 +02:00
|
|
|
****************************************************************************/
|
2010-01-18 13:13:34 +01:00
|
|
|
|
2016-03-18 07:55:01 +01:00
|
|
|
#pragma once
|
2010-01-18 13:13:34 +01:00
|
|
|
|
|
|
|
|
//
|
|
|
|
|
// W A R N I N G
|
|
|
|
|
// -------------
|
|
|
|
|
//
|
|
|
|
|
// This file is not part of the Qt API. It exists purely as an
|
|
|
|
|
// implementation detail. This header file may change from version to
|
|
|
|
|
// version without notice, or even be removed.
|
|
|
|
|
//
|
|
|
|
|
// We mean it.
|
|
|
|
|
//
|
|
|
|
|
|
|
|
|
|
#include "qmljsglobal_p.h"
|
2011-09-13 08:42:52 +02:00
|
|
|
#include "qmljsgrammar_p.h"
|
2013-11-06 14:17:23 +01:00
|
|
|
|
|
|
|
|
#include <QtCore/qstring.h>
|
2018-10-16 15:32:58 +02:00
|
|
|
#include <QtCore/qstack.h>
|
2010-01-18 13:13:34 +01:00
|
|
|
|
|
|
|
|
QT_QML_BEGIN_NAMESPACE
|
|
|
|
|
|
|
|
|
|
namespace QmlJS {
|
|
|
|
|
|
|
|
|
|
class Engine;
|
2016-04-29 11:00:30 +02:00
|
|
|
class DiagnosticMessage;
|
2018-10-16 15:32:58 +02:00
|
|
|
class Directives;
|
2011-09-19 14:09:26 +02:00
|
|
|
|
2011-09-13 08:42:52 +02:00
|
|
|
class QML_PARSER_EXPORT Lexer: public QmlJSGrammar
|
2010-01-18 13:13:34 +01:00
|
|
|
{
|
|
|
|
|
public:
|
2011-09-13 08:42:52 +02:00
|
|
|
enum {
|
|
|
|
|
T_ABSTRACT = T_RESERVED_WORD,
|
|
|
|
|
T_BOOLEAN = T_RESERVED_WORD,
|
|
|
|
|
T_BYTE = T_RESERVED_WORD,
|
|
|
|
|
T_CHAR = T_RESERVED_WORD,
|
|
|
|
|
T_DOUBLE = T_RESERVED_WORD,
|
|
|
|
|
T_FINAL = T_RESERVED_WORD,
|
|
|
|
|
T_FLOAT = T_RESERVED_WORD,
|
|
|
|
|
T_GOTO = T_RESERVED_WORD,
|
|
|
|
|
T_IMPLEMENTS = T_RESERVED_WORD,
|
|
|
|
|
T_INT = T_RESERVED_WORD,
|
|
|
|
|
T_INTERFACE = T_RESERVED_WORD,
|
|
|
|
|
T_LONG = T_RESERVED_WORD,
|
|
|
|
|
T_NATIVE = T_RESERVED_WORD,
|
|
|
|
|
T_PACKAGE = T_RESERVED_WORD,
|
|
|
|
|
T_PRIVATE = T_RESERVED_WORD,
|
|
|
|
|
T_PROTECTED = T_RESERVED_WORD,
|
|
|
|
|
T_SHORT = T_RESERVED_WORD,
|
|
|
|
|
T_SYNCHRONIZED = T_RESERVED_WORD,
|
|
|
|
|
T_THROWS = T_RESERVED_WORD,
|
|
|
|
|
T_TRANSIENT = T_RESERVED_WORD,
|
2013-01-22 11:15:23 +01:00
|
|
|
T_VOLATILE = T_RESERVED_WORD
|
2011-09-13 08:42:52 +02:00
|
|
|
};
|
2010-01-18 13:13:34 +01:00
|
|
|
|
|
|
|
|
enum Error {
|
|
|
|
|
NoError,
|
|
|
|
|
IllegalCharacter,
|
2018-10-16 15:32:58 +02:00
|
|
|
IllegalNumber,
|
2010-01-18 13:13:34 +01:00
|
|
|
UnclosedStringLiteral,
|
|
|
|
|
IllegalEscapeSequence,
|
|
|
|
|
IllegalUnicodeEscapeSequence,
|
|
|
|
|
UnclosedComment,
|
|
|
|
|
IllegalExponentIndicator,
|
2013-11-06 14:17:23 +01:00
|
|
|
IllegalIdentifier,
|
|
|
|
|
IllegalHexadecimalEscapeSequence
|
2010-01-18 13:13:34 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
enum RegExpBodyPrefix {
|
|
|
|
|
NoPrefix,
|
|
|
|
|
EqualPrefix
|
|
|
|
|
};
|
|
|
|
|
|
2011-09-19 14:09:26 +02:00
|
|
|
enum RegExpFlag {
|
|
|
|
|
RegExp_Global = 0x01,
|
|
|
|
|
RegExp_IgnoreCase = 0x02,
|
2018-10-16 15:32:58 +02:00
|
|
|
RegExp_Multiline = 0x04,
|
|
|
|
|
RegExp_Unicode = 0x08,
|
|
|
|
|
RegExp_Sticky = 0x10
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
enum ParseModeFlags {
|
|
|
|
|
QmlMode = 0x1,
|
|
|
|
|
YieldIsKeyword = 0x2,
|
|
|
|
|
StaticIsKeyword = 0x4
|
2011-09-19 14:09:26 +02:00
|
|
|
};
|
|
|
|
|
|
2011-09-13 08:42:52 +02:00
|
|
|
public:
|
|
|
|
|
Lexer(Engine *engine);
|
|
|
|
|
|
2018-10-16 15:32:58 +02:00
|
|
|
int parseModeFlags() const {
|
|
|
|
|
int flags = 0;
|
|
|
|
|
if (qmlMode())
|
|
|
|
|
flags |= QmlMode|StaticIsKeyword;
|
|
|
|
|
if (yieldIsKeyWord())
|
|
|
|
|
flags |= YieldIsKeyword;
|
|
|
|
|
if (_staticIsKeyword)
|
|
|
|
|
flags |= StaticIsKeyword;
|
|
|
|
|
return flags;
|
|
|
|
|
}
|
|
|
|
|
|
2011-12-07 11:16:26 +01:00
|
|
|
bool qmlMode() const;
|
2018-10-16 15:32:58 +02:00
|
|
|
bool yieldIsKeyWord() const { return _generatorLevel != 0; }
|
|
|
|
|
void setStaticIsKeyword(bool b) { _staticIsKeyword = b; }
|
2011-12-07 11:16:26 +01:00
|
|
|
|
2011-09-13 08:42:52 +02:00
|
|
|
QString code() const;
|
|
|
|
|
void setCode(const QString &code, int lineno, bool qmlMode = true);
|
|
|
|
|
|
|
|
|
|
int lex();
|
|
|
|
|
|
2010-01-18 13:13:34 +01:00
|
|
|
bool scanRegExp(RegExpBodyPrefix prefix = NoPrefix);
|
2016-04-29 11:00:30 +02:00
|
|
|
bool scanDirectives(Directives *directives, DiagnosticMessage *error);
|
2010-01-18 13:13:34 +01:00
|
|
|
|
2011-09-13 08:42:52 +02:00
|
|
|
int regExpFlags() const { return _patternFlags; }
|
|
|
|
|
QString regExpPattern() const { return _tokenText; }
|
|
|
|
|
|
2012-07-31 10:12:26 +02:00
|
|
|
int tokenKind() const { return _tokenKind; }
|
|
|
|
|
int tokenOffset() const { return _tokenStartPtr - _code.unicode(); }
|
|
|
|
|
int tokenLength() const { return _tokenLength; }
|
2010-01-18 13:13:34 +01:00
|
|
|
|
2012-07-31 10:12:26 +02:00
|
|
|
int tokenStartLine() const { return _tokenLine; }
|
2018-10-16 15:32:58 +02:00
|
|
|
int tokenStartColumn() const { return _tokenColumn; }
|
2010-01-18 13:13:34 +01:00
|
|
|
|
2012-07-31 10:12:26 +02:00
|
|
|
inline QStringRef tokenSpell() const { return _tokenSpell; }
|
2019-06-19 19:42:15 +02:00
|
|
|
inline QStringRef rawString() const { return _rawString; }
|
2012-07-31 10:12:26 +02:00
|
|
|
double tokenValue() const { return _tokenValue; }
|
2011-09-13 08:42:52 +02:00
|
|
|
QString tokenText() const;
|
|
|
|
|
|
|
|
|
|
Error errorCode() const;
|
|
|
|
|
QString errorMessage() const;
|
|
|
|
|
|
|
|
|
|
bool prevTerminator() const;
|
2011-09-19 14:09:26 +02:00
|
|
|
bool followsClosingBrace() const;
|
|
|
|
|
bool canInsertAutomaticSemicolon(int token) const;
|
2011-09-13 08:42:52 +02:00
|
|
|
|
|
|
|
|
enum ParenthesesState {
|
|
|
|
|
IgnoreParentheses,
|
|
|
|
|
CountParentheses,
|
|
|
|
|
BalancedParentheses
|
|
|
|
|
};
|
2010-01-18 13:13:34 +01:00
|
|
|
|
2018-10-16 15:32:58 +02:00
|
|
|
void enterGeneratorBody() { ++_generatorLevel; }
|
|
|
|
|
void leaveGeneratorBody() { --_generatorLevel; }
|
|
|
|
|
|
2011-12-07 11:16:26 +01:00
|
|
|
protected:
|
2018-10-16 15:32:58 +02:00
|
|
|
static int classify(const QChar *s, int n, int parseModeFlags);
|
2011-12-07 11:16:26 +01:00
|
|
|
|
2010-01-18 13:13:34 +01:00
|
|
|
private:
|
2011-09-13 08:42:52 +02:00
|
|
|
inline void scanChar();
|
|
|
|
|
int scanToken();
|
2012-07-31 10:12:26 +02:00
|
|
|
int scanNumber(QChar ch);
|
2018-10-16 15:32:58 +02:00
|
|
|
enum ScanStringMode {
|
|
|
|
|
SingleQuote = '\'',
|
|
|
|
|
DoubleQuote = '"',
|
|
|
|
|
TemplateHead = '`',
|
|
|
|
|
TemplateContinuation = 0
|
|
|
|
|
};
|
|
|
|
|
int scanString(ScanStringMode mode);
|
2011-09-13 08:42:52 +02:00
|
|
|
|
2010-01-18 13:13:34 +01:00
|
|
|
bool isLineTerminator() const;
|
2013-01-22 11:15:23 +01:00
|
|
|
unsigned isLineTerminatorSequence() const;
|
2011-09-13 08:42:52 +02:00
|
|
|
static bool isIdentLetter(QChar c);
|
2010-01-18 13:13:34 +01:00
|
|
|
static bool isDecimalDigit(ushort c);
|
2011-09-13 08:42:52 +02:00
|
|
|
static bool isHexDigit(QChar c);
|
|
|
|
|
static bool isOctalDigit(ushort c);
|
2010-01-18 13:13:34 +01:00
|
|
|
|
2011-09-13 08:42:52 +02:00
|
|
|
void syncProhibitAutomaticSemicolon();
|
2018-10-16 15:32:58 +02:00
|
|
|
uint decodeUnicodeEscapeCharacter(bool *ok);
|
2013-11-06 14:17:23 +01:00
|
|
|
QChar decodeHexEscapeCharacter(bool *ok);
|
2010-01-18 13:13:34 +01:00
|
|
|
|
|
|
|
|
private:
|
2011-09-13 08:42:52 +02:00
|
|
|
Engine *_engine;
|
2010-01-18 13:13:34 +01:00
|
|
|
|
2011-09-13 08:42:52 +02:00
|
|
|
QString _code;
|
|
|
|
|
QString _tokenText;
|
|
|
|
|
QString _errorMessage;
|
|
|
|
|
QStringRef _tokenSpell;
|
2019-06-19 19:42:15 +02:00
|
|
|
QStringRef _rawString;
|
2010-01-18 13:13:34 +01:00
|
|
|
|
2011-09-13 08:42:52 +02:00
|
|
|
const QChar *_codePtr;
|
2013-11-06 14:17:23 +01:00
|
|
|
const QChar *_endPtr;
|
2011-09-13 08:42:52 +02:00
|
|
|
const QChar *_tokenStartPtr;
|
2010-01-18 13:13:34 +01:00
|
|
|
|
2011-09-13 08:42:52 +02:00
|
|
|
QChar _char;
|
|
|
|
|
Error _errorCode;
|
|
|
|
|
|
|
|
|
|
int _currentLineNumber;
|
2018-10-16 15:32:58 +02:00
|
|
|
int _currentColumnNumber;
|
2011-09-13 08:42:52 +02:00
|
|
|
double _tokenValue;
|
|
|
|
|
|
|
|
|
|
// parentheses state
|
|
|
|
|
ParenthesesState _parenthesesState;
|
|
|
|
|
int _parenthesesCount;
|
2010-01-18 13:13:34 +01:00
|
|
|
|
2018-10-16 15:32:58 +02:00
|
|
|
// template string stack
|
|
|
|
|
QStack<int> _outerTemplateBraceCount;
|
|
|
|
|
int _bracesCount = -1;
|
|
|
|
|
|
2011-09-13 08:42:52 +02:00
|
|
|
int _stackToken;
|
2010-01-18 13:13:34 +01:00
|
|
|
|
2011-09-13 08:42:52 +02:00
|
|
|
int _patternFlags;
|
|
|
|
|
int _tokenKind;
|
|
|
|
|
int _tokenLength;
|
|
|
|
|
int _tokenLine;
|
2018-10-16 15:32:58 +02:00
|
|
|
int _tokenColumn;
|
2010-01-18 13:13:34 +01:00
|
|
|
|
2011-09-13 08:42:52 +02:00
|
|
|
bool _validTokenText;
|
|
|
|
|
bool _prohibitAutomaticSemicolon;
|
|
|
|
|
bool _restrictedKeyword;
|
|
|
|
|
bool _terminator;
|
2011-09-19 14:09:26 +02:00
|
|
|
bool _followsClosingBrace;
|
2011-09-13 08:42:52 +02:00
|
|
|
bool _delimited;
|
|
|
|
|
bool _qmlMode;
|
2019-06-19 19:42:15 +02:00
|
|
|
bool _skipLinefeed = false;
|
2018-10-16 15:32:58 +02:00
|
|
|
int _generatorLevel = 0;
|
|
|
|
|
bool _staticIsKeyword = false;
|
2010-01-18 13:13:34 +01:00
|
|
|
};
|
|
|
|
|
|
2011-09-13 08:42:52 +02:00
|
|
|
} // end of namespace QmlJS
|
2010-01-18 13:13:34 +01:00
|
|
|
|
|
|
|
|
QT_QML_END_NAMESPACE
|
2017-03-16 17:13:46 +01:00
|
|
|
|