2021-10-29 21:25:12 +01:00
|
|
|
// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
|
|
|
|
#include "Luau/Parser.h"
|
|
|
|
|
2023-10-06 18:31:16 +01:00
|
|
|
#include "Luau/Common.h"
|
2021-11-05 02:07:18 +00:00
|
|
|
#include "Luau/TimeTrace.h"
|
|
|
|
|
2021-10-29 21:25:12 +01:00
|
|
|
#include <algorithm>
|
|
|
|
|
2022-07-01 00:29:02 +01:00
|
|
|
#include <errno.h>
|
|
|
|
#include <limits.h>
|
|
|
|
|
2023-10-27 20:33:36 +01:00
|
|
|
LUAU_FASTINTVARIABLE(LuauRecursionLimit, 1000)
|
|
|
|
LUAU_FASTINTVARIABLE(LuauTypeLengthLimit, 1000)
|
|
|
|
LUAU_FASTINTVARIABLE(LuauParseErrorLimit, 100)
|
|
|
|
|
2021-10-29 21:25:12 +01:00
|
|
|
// Warning: If you are introducing new syntax, ensure that it is behind a separate
|
|
|
|
// flag so that we don't break production games by reverting syntax changes.
|
|
|
|
// See docs/SyntaxChanges.md for an explanation.
|
2024-02-23 18:40:00 +00:00
|
|
|
LUAU_FASTFLAGVARIABLE(DebugLuauDeferredConstraintResolution, false)
|
2024-06-07 18:09:03 +01:00
|
|
|
LUAU_FASTFLAG(LuauAttributeSyntax)
|
2024-06-14 17:38:56 +01:00
|
|
|
LUAU_FASTFLAGVARIABLE(LuauLeadingBarAndAmpersand2, false)
|
|
|
|
LUAU_FASTFLAGVARIABLE(LuauNativeAttribute, false)
|
|
|
|
LUAU_FASTFLAGVARIABLE(LuauAttributeSyntaxFunExpr, false)
|
2024-06-29 01:07:35 +01:00
|
|
|
LUAU_FASTFLAGVARIABLE(LuauDeclarationExtraPropData, false)
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
namespace Luau
|
|
|
|
{
|
|
|
|
|
2024-06-07 18:09:03 +01:00
|
|
|
struct AttributeEntry
|
|
|
|
{
|
|
|
|
const char* name;
|
|
|
|
AstAttr::Type type;
|
|
|
|
};
|
|
|
|
|
2024-06-14 17:38:56 +01:00
|
|
|
AttributeEntry kAttributeEntries[] = {{"@checked", AstAttr::Type::Checked}, {"@native", AstAttr::Type::Native}, {nullptr, AstAttr::Type::Checked}};
|
2024-06-07 18:09:03 +01:00
|
|
|
|
2021-10-29 21:25:12 +01:00
|
|
|
ParseError::ParseError(const Location& location, const std::string& message)
|
|
|
|
: location(location)
|
|
|
|
, message(message)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
const char* ParseError::what() const throw()
|
|
|
|
{
|
|
|
|
return message.c_str();
|
|
|
|
}
|
|
|
|
|
|
|
|
const Location& ParseError::getLocation() const
|
|
|
|
{
|
|
|
|
return location;
|
|
|
|
}
|
|
|
|
|
|
|
|
const std::string& ParseError::getMessage() const
|
|
|
|
{
|
|
|
|
return message;
|
|
|
|
}
|
|
|
|
|
|
|
|
// LUAU_NOINLINE is used to limit the stack cost of this function due to std::string object / exception plumbing
|
|
|
|
LUAU_NOINLINE void ParseError::raise(const Location& location, const char* format, ...)
|
|
|
|
{
|
|
|
|
va_list args;
|
|
|
|
va_start(args, format);
|
|
|
|
std::string message = vformat(format, args);
|
|
|
|
va_end(args);
|
|
|
|
|
|
|
|
throw ParseError(location, message);
|
|
|
|
}
|
|
|
|
|
|
|
|
ParseErrors::ParseErrors(std::vector<ParseError> errors)
|
|
|
|
: errors(std::move(errors))
|
|
|
|
{
|
|
|
|
LUAU_ASSERT(!this->errors.empty());
|
|
|
|
|
|
|
|
if (this->errors.size() == 1)
|
|
|
|
message = this->errors.front().what();
|
|
|
|
else
|
|
|
|
message = format("%d parse errors", int(this->errors.size()));
|
|
|
|
}
|
|
|
|
|
|
|
|
const char* ParseErrors::what() const throw()
|
|
|
|
{
|
|
|
|
return message.c_str();
|
|
|
|
}
|
|
|
|
|
|
|
|
const std::vector<ParseError>& ParseErrors::getErrors() const
|
|
|
|
{
|
|
|
|
return errors;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
TempVector<T>::TempVector(std::vector<T>& storage)
|
|
|
|
: storage(storage)
|
|
|
|
, offset(storage.size())
|
|
|
|
, size_(0)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
TempVector<T>::~TempVector()
|
|
|
|
{
|
|
|
|
LUAU_ASSERT(storage.size() == offset + size_);
|
|
|
|
storage.erase(storage.begin() + offset, storage.end());
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
const T& TempVector<T>::operator[](size_t index) const
|
|
|
|
{
|
|
|
|
LUAU_ASSERT(index < size_);
|
|
|
|
return storage[offset + index];
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
const T& TempVector<T>::front() const
|
|
|
|
{
|
|
|
|
LUAU_ASSERT(size_ > 0);
|
|
|
|
return storage[offset];
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
const T& TempVector<T>::back() const
|
|
|
|
{
|
|
|
|
LUAU_ASSERT(size_ > 0);
|
|
|
|
return storage.back();
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
bool TempVector<T>::empty() const
|
|
|
|
{
|
|
|
|
return size_ == 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
size_t TempVector<T>::size() const
|
|
|
|
{
|
|
|
|
return size_;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
void TempVector<T>::push_back(const T& item)
|
|
|
|
{
|
|
|
|
LUAU_ASSERT(storage.size() == offset + size_);
|
|
|
|
storage.push_back(item);
|
|
|
|
size_++;
|
|
|
|
}
|
|
|
|
|
2023-03-17 14:59:30 +00:00
|
|
|
static bool shouldParseTypePack(Lexer& lexer)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
if (lexer.current().type == Lexeme::Dot3)
|
|
|
|
return true;
|
|
|
|
else if (lexer.current().type == Lexeme::Name && lexer.lookahead().type == Lexeme::Dot3)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
ParseResult Parser::parse(const char* buffer, size_t bufferSize, AstNameTable& names, Allocator& allocator, ParseOptions options)
|
|
|
|
{
|
2021-11-05 02:07:18 +00:00
|
|
|
LUAU_TIMETRACE_SCOPE("Parser::parse", "Parser");
|
|
|
|
|
2022-03-18 00:06:25 +00:00
|
|
|
Parser p(buffer, bufferSize, names, allocator, options);
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
try
|
|
|
|
{
|
2022-03-18 00:06:25 +00:00
|
|
|
AstStatBlock* root = p.parseChunk();
|
2022-10-27 23:22:49 +01:00
|
|
|
size_t lines = p.lexer.current().location.end.line + (bufferSize > 0 && buffer[bufferSize - 1] != '\n');
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-10-27 23:22:49 +01:00
|
|
|
return ParseResult{root, lines, std::move(p.hotcomments), std::move(p.parseErrors), std::move(p.commentLocations)};
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
catch (ParseError& err)
|
|
|
|
{
|
|
|
|
// when catching a fatal error, append it to the list of non-fatal errors and return
|
|
|
|
p.parseErrors.push_back(err);
|
|
|
|
|
2022-10-27 23:22:49 +01:00
|
|
|
return ParseResult{nullptr, 0, {}, p.parseErrors};
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-02-18 00:41:20 +00:00
|
|
|
Parser::Parser(const char* buffer, size_t bufferSize, AstNameTable& names, Allocator& allocator, const ParseOptions& options)
|
|
|
|
: options(options)
|
|
|
|
, lexer(buffer, bufferSize, names)
|
2021-10-29 21:25:12 +01:00
|
|
|
, allocator(allocator)
|
|
|
|
, recursionCounter(0)
|
2022-08-18 22:04:33 +01:00
|
|
|
, endMismatchSuspect(Lexeme(Location(), Lexeme::Eof))
|
2021-10-29 21:25:12 +01:00
|
|
|
, localMap(AstName())
|
|
|
|
{
|
|
|
|
Function top;
|
|
|
|
top.vararg = true;
|
|
|
|
|
2022-04-21 22:04:22 +01:00
|
|
|
functionStack.reserve(8);
|
2021-10-29 21:25:12 +01:00
|
|
|
functionStack.push_back(top);
|
|
|
|
|
|
|
|
nameSelf = names.addStatic("self");
|
|
|
|
nameNumber = names.addStatic("number");
|
2022-02-18 00:41:20 +00:00
|
|
|
nameError = names.addStatic(kParseNameError);
|
2021-10-29 21:25:12 +01:00
|
|
|
nameNil = names.getOrAdd("nil"); // nil is a reserved keyword
|
|
|
|
|
|
|
|
matchRecoveryStopOnToken.assign(Lexeme::Type::Reserved_END, 0);
|
|
|
|
matchRecoveryStopOnToken[Lexeme::Type::Eof] = 1;
|
|
|
|
|
2022-03-18 00:06:25 +00:00
|
|
|
// required for lookahead() to work across a comment boundary and for nextLexeme() to work when captureComments is false
|
|
|
|
lexer.setSkipComments(true);
|
2022-02-18 00:41:20 +00:00
|
|
|
|
2022-03-18 00:06:25 +00:00
|
|
|
// read first lexeme (any hot comments get .header = true)
|
|
|
|
LUAU_ASSERT(hotcommentHeader);
|
2021-10-29 21:25:12 +01:00
|
|
|
nextLexeme();
|
2022-02-18 00:41:20 +00:00
|
|
|
|
|
|
|
// all hot comments parsed after the first non-comment lexeme are special in that they don't affect type checking / linting mode
|
|
|
|
hotcommentHeader = false;
|
2022-04-21 22:04:22 +01:00
|
|
|
|
|
|
|
// preallocate some buffers that are very likely to grow anyway; this works around std::vector's inefficient growth policy for small arrays
|
|
|
|
localStack.reserve(16);
|
|
|
|
scratchStat.reserve(16);
|
|
|
|
scratchExpr.reserve(16);
|
|
|
|
scratchLocal.reserve(16);
|
|
|
|
scratchBinding.reserve(16);
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
bool Parser::blockFollow(const Lexeme& l)
|
|
|
|
{
|
|
|
|
return l.type == Lexeme::Eof || l.type == Lexeme::ReservedElse || l.type == Lexeme::ReservedElseif || l.type == Lexeme::ReservedEnd ||
|
|
|
|
l.type == Lexeme::ReservedUntil;
|
|
|
|
}
|
|
|
|
|
|
|
|
AstStatBlock* Parser::parseChunk()
|
|
|
|
{
|
|
|
|
AstStatBlock* result = parseBlock();
|
|
|
|
|
|
|
|
if (lexer.current().type != Lexeme::Eof)
|
|
|
|
expectAndConsumeFail(Lexeme::Eof, nullptr);
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
// chunk ::= {stat [`;']} [laststat [`;']]
|
|
|
|
// block ::= chunk
|
|
|
|
AstStatBlock* Parser::parseBlock()
|
|
|
|
{
|
|
|
|
unsigned int localsBegin = saveLocals();
|
|
|
|
|
|
|
|
AstStatBlock* result = parseBlockNoScope();
|
|
|
|
|
|
|
|
restoreLocals(localsBegin);
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool isStatLast(AstStat* stat)
|
|
|
|
{
|
|
|
|
return stat->is<AstStatBreak>() || stat->is<AstStatContinue>() || stat->is<AstStatReturn>();
|
|
|
|
}
|
|
|
|
|
|
|
|
AstStatBlock* Parser::parseBlockNoScope()
|
|
|
|
{
|
|
|
|
TempVector<AstStat*> body(scratchStat);
|
|
|
|
|
|
|
|
const Position prevPosition = lexer.previousLocation().end;
|
|
|
|
|
|
|
|
while (!blockFollow(lexer.current()))
|
|
|
|
{
|
2023-10-27 20:33:36 +01:00
|
|
|
unsigned int oldRecursionCount = recursionCounter;
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
incrementRecursionCounter("block");
|
|
|
|
|
|
|
|
AstStat* stat = parseStat();
|
|
|
|
|
2023-10-27 20:33:36 +01:00
|
|
|
recursionCounter = oldRecursionCount;
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
if (lexer.current().type == ';')
|
|
|
|
{
|
|
|
|
nextLexeme();
|
|
|
|
stat->hasSemicolon = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
body.push_back(stat);
|
|
|
|
|
|
|
|
if (isStatLast(stat))
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
const Location location = Location(prevPosition, lexer.current().location.begin);
|
|
|
|
|
|
|
|
return allocator.alloc<AstStatBlock>(location, copy(body));
|
|
|
|
}
|
|
|
|
|
|
|
|
// stat ::=
|
|
|
|
// varlist `=' explist |
|
|
|
|
// functioncall |
|
|
|
|
// do block end |
|
|
|
|
// while exp do block end |
|
|
|
|
// repeat block until exp |
|
|
|
|
// if exp then block {elseif exp then block} [else block] end |
|
|
|
|
// for binding `=' exp `,' exp [`,' exp] do block end |
|
|
|
|
// for namelist in explist do block end |
|
|
|
|
// function funcname funcbody |
|
2024-06-07 18:09:03 +01:00
|
|
|
// attributes function funcname funcbody |
|
2021-10-29 21:25:12 +01:00
|
|
|
// local function Name funcbody |
|
2024-06-07 18:09:03 +01:00
|
|
|
// local attributes function Name funcbody |
|
2021-10-29 21:25:12 +01:00
|
|
|
// local namelist [`=' explist]
|
|
|
|
// laststat ::= return [explist] | break
|
|
|
|
AstStat* Parser::parseStat()
|
|
|
|
{
|
|
|
|
// guess the type from the token type
|
|
|
|
switch (lexer.current().type)
|
|
|
|
{
|
|
|
|
case Lexeme::ReservedIf:
|
|
|
|
return parseIf();
|
|
|
|
case Lexeme::ReservedWhile:
|
|
|
|
return parseWhile();
|
|
|
|
case Lexeme::ReservedDo:
|
|
|
|
return parseDo();
|
|
|
|
case Lexeme::ReservedFor:
|
|
|
|
return parseFor();
|
|
|
|
case Lexeme::ReservedRepeat:
|
|
|
|
return parseRepeat();
|
|
|
|
case Lexeme::ReservedFunction:
|
2024-06-07 18:09:03 +01:00
|
|
|
return parseFunctionStat(AstArray<AstAttr*>({nullptr, 0}));
|
2021-10-29 21:25:12 +01:00
|
|
|
case Lexeme::ReservedLocal:
|
2024-06-07 18:09:03 +01:00
|
|
|
return parseLocal(AstArray<AstAttr*>({nullptr, 0}));
|
2021-10-29 21:25:12 +01:00
|
|
|
case Lexeme::ReservedReturn:
|
|
|
|
return parseReturn();
|
|
|
|
case Lexeme::ReservedBreak:
|
|
|
|
return parseBreak();
|
2024-06-07 18:09:03 +01:00
|
|
|
case Lexeme::Attribute:
|
|
|
|
if (FFlag::LuauAttributeSyntax)
|
|
|
|
return parseAttributeStat();
|
2021-10-29 21:25:12 +01:00
|
|
|
default:;
|
|
|
|
}
|
|
|
|
|
|
|
|
Location start = lexer.current().location;
|
|
|
|
|
|
|
|
// we need to disambiguate a few cases, primarily assignment (lvalue = ...) vs statements-that-are calls
|
|
|
|
AstExpr* expr = parsePrimaryExpr(/* asStatement= */ true);
|
|
|
|
|
|
|
|
if (expr->is<AstExprCall>())
|
|
|
|
return allocator.alloc<AstStatExpr>(expr->location, expr);
|
|
|
|
|
|
|
|
// if the next token is , or =, it's an assignment (, means it's an assignment with multiple variables)
|
|
|
|
if (lexer.current().type == ',' || lexer.current().type == '=')
|
|
|
|
return parseAssignment(expr);
|
|
|
|
|
|
|
|
// if the next token is a compound assignment operator, it's a compound assignment (these don't support multiple variables)
|
|
|
|
if (std::optional<AstExprBinary::Op> op = parseCompoundOp(lexer.current()))
|
|
|
|
return parseCompoundAssignment(expr, *op);
|
|
|
|
|
|
|
|
// we know this isn't a call or an assignment; therefore it must be a context-sensitive keyword such as `type` or `continue`
|
|
|
|
AstName ident = getIdentifier(expr);
|
|
|
|
|
2023-05-05 20:57:12 +01:00
|
|
|
if (ident == "type")
|
|
|
|
return parseTypeAlias(expr->location, /* exported= */ false);
|
2023-03-17 14:59:30 +00:00
|
|
|
|
2023-05-05 20:57:12 +01:00
|
|
|
if (ident == "export" && lexer.current().type == Lexeme::Name && AstName(lexer.current().name) == "type")
|
|
|
|
{
|
|
|
|
nextLexeme();
|
|
|
|
return parseTypeAlias(expr->location, /* exported= */ true);
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
2023-05-05 20:57:12 +01:00
|
|
|
if (ident == "continue")
|
2021-10-29 21:25:12 +01:00
|
|
|
return parseContinue(expr->location);
|
|
|
|
|
2023-05-05 20:57:12 +01:00
|
|
|
if (options.allowDeclarationSyntax)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
if (ident == "declare")
|
2024-06-07 18:09:03 +01:00
|
|
|
return parseDeclaration(expr->location, AstArray<AstAttr*>({nullptr, 0}));
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// skip unexpected symbol if lexer couldn't advance at all (statements are parsed in a loop)
|
|
|
|
if (start == lexer.current().location)
|
|
|
|
nextLexeme();
|
|
|
|
|
|
|
|
return reportStatError(expr->location, copy({expr}), {}, "Incomplete statement: expected assignment or a function call");
|
|
|
|
}
|
|
|
|
|
|
|
|
// if exp then block {elseif exp then block} [else block] end
|
|
|
|
AstStat* Parser::parseIf()
|
|
|
|
{
|
|
|
|
Location start = lexer.current().location;
|
|
|
|
|
|
|
|
nextLexeme(); // if / elseif
|
|
|
|
|
|
|
|
AstExpr* cond = parseExpr();
|
|
|
|
|
|
|
|
Lexeme matchThen = lexer.current();
|
2022-02-18 00:41:20 +00:00
|
|
|
std::optional<Location> thenLocation;
|
|
|
|
if (expectAndConsume(Lexeme::ReservedThen, "if statement"))
|
|
|
|
thenLocation = matchThen.location;
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
AstStatBlock* thenbody = parseBlock();
|
|
|
|
|
|
|
|
AstStat* elsebody = nullptr;
|
|
|
|
Location end = start;
|
|
|
|
std::optional<Location> elseLocation;
|
|
|
|
|
|
|
|
if (lexer.current().type == Lexeme::ReservedElseif)
|
|
|
|
{
|
2024-01-27 02:30:40 +00:00
|
|
|
thenbody->hasEnd = true;
|
2023-10-27 20:33:36 +01:00
|
|
|
unsigned int oldRecursionCount = recursionCounter;
|
2022-01-14 16:06:31 +00:00
|
|
|
incrementRecursionCounter("elseif");
|
|
|
|
elseLocation = lexer.current().location;
|
|
|
|
elsebody = parseIf();
|
|
|
|
end = elsebody->location;
|
2023-10-27 20:33:36 +01:00
|
|
|
recursionCounter = oldRecursionCount;
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
Lexeme matchThenElse = matchThen;
|
|
|
|
|
|
|
|
if (lexer.current().type == Lexeme::ReservedElse)
|
|
|
|
{
|
2024-01-27 02:30:40 +00:00
|
|
|
thenbody->hasEnd = true;
|
2021-10-29 21:25:12 +01:00
|
|
|
elseLocation = lexer.current().location;
|
|
|
|
matchThenElse = lexer.current();
|
|
|
|
nextLexeme();
|
|
|
|
|
|
|
|
elsebody = parseBlock();
|
|
|
|
elsebody->location.begin = matchThenElse.location.end;
|
|
|
|
}
|
|
|
|
|
|
|
|
end = lexer.current().location;
|
|
|
|
|
2023-10-13 20:38:31 +01:00
|
|
|
bool hasEnd = expectMatchEndAndConsume(Lexeme::ReservedEnd, matchThenElse);
|
|
|
|
|
2024-01-27 02:30:40 +00:00
|
|
|
if (elsebody)
|
2023-10-13 20:38:31 +01:00
|
|
|
{
|
2024-01-27 02:30:40 +00:00
|
|
|
if (AstStatBlock* elseBlock = elsebody->as<AstStatBlock>())
|
|
|
|
elseBlock->hasEnd = hasEnd;
|
2023-10-13 20:38:31 +01:00
|
|
|
}
|
2024-01-27 02:30:40 +00:00
|
|
|
else
|
|
|
|
thenbody->hasEnd = hasEnd;
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
2024-01-27 02:30:40 +00:00
|
|
|
return allocator.alloc<AstStatIf>(Location(start, end), cond, thenbody, elsebody, thenLocation, elseLocation);
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// while exp do block end
|
|
|
|
AstStat* Parser::parseWhile()
|
|
|
|
{
|
|
|
|
Location start = lexer.current().location;
|
|
|
|
|
|
|
|
nextLexeme(); // while
|
|
|
|
|
|
|
|
AstExpr* cond = parseExpr();
|
|
|
|
|
|
|
|
Lexeme matchDo = lexer.current();
|
|
|
|
bool hasDo = expectAndConsume(Lexeme::ReservedDo, "while loop");
|
|
|
|
|
|
|
|
functionStack.back().loopDepth++;
|
|
|
|
|
|
|
|
AstStatBlock* body = parseBlock();
|
|
|
|
|
|
|
|
functionStack.back().loopDepth--;
|
|
|
|
|
|
|
|
Location end = lexer.current().location;
|
|
|
|
|
|
|
|
bool hasEnd = expectMatchEndAndConsume(Lexeme::ReservedEnd, matchDo);
|
2024-01-27 02:30:40 +00:00
|
|
|
body->hasEnd = hasEnd;
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2024-01-27 02:30:40 +00:00
|
|
|
return allocator.alloc<AstStatWhile>(Location(start, end), cond, body, hasDo, matchDo.location);
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// repeat block until exp
|
|
|
|
AstStat* Parser::parseRepeat()
|
|
|
|
{
|
|
|
|
Location start = lexer.current().location;
|
|
|
|
|
|
|
|
Lexeme matchRepeat = lexer.current();
|
|
|
|
nextLexeme(); // repeat
|
|
|
|
|
|
|
|
unsigned int localsBegin = saveLocals();
|
|
|
|
|
|
|
|
functionStack.back().loopDepth++;
|
|
|
|
|
|
|
|
AstStatBlock* body = parseBlockNoScope();
|
|
|
|
|
|
|
|
functionStack.back().loopDepth--;
|
|
|
|
|
|
|
|
bool hasUntil = expectMatchEndAndConsume(Lexeme::ReservedUntil, matchRepeat);
|
2024-01-27 02:30:40 +00:00
|
|
|
body->hasEnd = hasUntil;
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
AstExpr* cond = parseExpr();
|
|
|
|
|
|
|
|
restoreLocals(localsBegin);
|
|
|
|
|
|
|
|
return allocator.alloc<AstStatRepeat>(Location(start, cond->location), cond, body, hasUntil);
|
|
|
|
}
|
|
|
|
|
|
|
|
// do block end
|
|
|
|
AstStat* Parser::parseDo()
|
|
|
|
{
|
|
|
|
Location start = lexer.current().location;
|
|
|
|
|
|
|
|
Lexeme matchDo = lexer.current();
|
|
|
|
nextLexeme(); // do
|
|
|
|
|
2023-09-01 17:38:53 +01:00
|
|
|
AstStatBlock* body = parseBlock();
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
body->location.begin = start.begin;
|
|
|
|
|
2023-09-01 17:38:53 +01:00
|
|
|
body->hasEnd = expectMatchEndAndConsume(Lexeme::ReservedEnd, matchDo);
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
return body;
|
|
|
|
}
|
|
|
|
|
|
|
|
// break
|
|
|
|
AstStat* Parser::parseBreak()
|
|
|
|
{
|
|
|
|
Location start = lexer.current().location;
|
|
|
|
|
|
|
|
nextLexeme(); // break
|
|
|
|
|
|
|
|
if (functionStack.back().loopDepth == 0)
|
|
|
|
return reportStatError(start, {}, copy<AstStat*>({allocator.alloc<AstStatBreak>(start)}), "break statement must be inside a loop");
|
|
|
|
|
|
|
|
return allocator.alloc<AstStatBreak>(start);
|
|
|
|
}
|
|
|
|
|
|
|
|
// continue
|
|
|
|
AstStat* Parser::parseContinue(const Location& start)
|
|
|
|
{
|
|
|
|
if (functionStack.back().loopDepth == 0)
|
|
|
|
return reportStatError(start, {}, copy<AstStat*>({allocator.alloc<AstStatContinue>(start)}), "continue statement must be inside a loop");
|
|
|
|
|
|
|
|
// note: the token is already parsed for us!
|
|
|
|
|
|
|
|
return allocator.alloc<AstStatContinue>(start);
|
|
|
|
}
|
|
|
|
|
|
|
|
// for binding `=' exp `,' exp [`,' exp] do block end |
|
|
|
|
// for bindinglist in explist do block end |
|
|
|
|
AstStat* Parser::parseFor()
|
|
|
|
{
|
|
|
|
Location start = lexer.current().location;
|
|
|
|
|
|
|
|
nextLexeme(); // for
|
|
|
|
|
|
|
|
Binding varname = parseBinding();
|
|
|
|
|
|
|
|
if (lexer.current().type == '=')
|
|
|
|
{
|
|
|
|
nextLexeme();
|
|
|
|
|
|
|
|
AstExpr* from = parseExpr();
|
|
|
|
|
|
|
|
expectAndConsume(',', "index range");
|
|
|
|
|
|
|
|
AstExpr* to = parseExpr();
|
|
|
|
|
|
|
|
AstExpr* step = nullptr;
|
|
|
|
|
|
|
|
if (lexer.current().type == ',')
|
|
|
|
{
|
|
|
|
nextLexeme();
|
|
|
|
|
|
|
|
step = parseExpr();
|
|
|
|
}
|
|
|
|
|
|
|
|
Lexeme matchDo = lexer.current();
|
|
|
|
bool hasDo = expectAndConsume(Lexeme::ReservedDo, "for loop");
|
|
|
|
|
|
|
|
unsigned int localsBegin = saveLocals();
|
|
|
|
|
|
|
|
functionStack.back().loopDepth++;
|
|
|
|
|
|
|
|
AstLocal* var = pushLocal(varname);
|
|
|
|
|
|
|
|
AstStatBlock* body = parseBlock();
|
|
|
|
|
|
|
|
functionStack.back().loopDepth--;
|
|
|
|
|
|
|
|
restoreLocals(localsBegin);
|
|
|
|
|
|
|
|
Location end = lexer.current().location;
|
|
|
|
|
|
|
|
bool hasEnd = expectMatchEndAndConsume(Lexeme::ReservedEnd, matchDo);
|
2024-01-27 02:30:40 +00:00
|
|
|
body->hasEnd = hasEnd;
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2024-01-27 02:30:40 +00:00
|
|
|
return allocator.alloc<AstStatFor>(Location(start, end), var, from, to, step, body, hasDo, matchDo.location);
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
TempVector<Binding> names(scratchBinding);
|
|
|
|
names.push_back(varname);
|
|
|
|
|
|
|
|
if (lexer.current().type == ',')
|
|
|
|
{
|
|
|
|
nextLexeme();
|
|
|
|
|
|
|
|
parseBindingList(names);
|
|
|
|
}
|
|
|
|
|
|
|
|
Location inLocation = lexer.current().location;
|
|
|
|
bool hasIn = expectAndConsume(Lexeme::ReservedIn, "for loop");
|
|
|
|
|
|
|
|
TempVector<AstExpr*> values(scratchExpr);
|
|
|
|
parseExprList(values);
|
|
|
|
|
|
|
|
Lexeme matchDo = lexer.current();
|
|
|
|
bool hasDo = expectAndConsume(Lexeme::ReservedDo, "for loop");
|
|
|
|
|
|
|
|
unsigned int localsBegin = saveLocals();
|
|
|
|
|
|
|
|
functionStack.back().loopDepth++;
|
|
|
|
|
|
|
|
TempVector<AstLocal*> vars(scratchLocal);
|
|
|
|
|
|
|
|
for (size_t i = 0; i < names.size(); ++i)
|
|
|
|
vars.push_back(pushLocal(names[i]));
|
|
|
|
|
|
|
|
AstStatBlock* body = parseBlock();
|
|
|
|
|
|
|
|
functionStack.back().loopDepth--;
|
|
|
|
|
|
|
|
restoreLocals(localsBegin);
|
|
|
|
|
|
|
|
Location end = lexer.current().location;
|
|
|
|
|
|
|
|
bool hasEnd = expectMatchEndAndConsume(Lexeme::ReservedEnd, matchDo);
|
2024-01-27 02:30:40 +00:00
|
|
|
body->hasEnd = hasEnd;
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2024-02-16 01:25:31 +00:00
|
|
|
return allocator.alloc<AstStatForIn>(Location(start, end), copy(vars), copy(values), body, hasIn, inLocation, hasDo, matchDo.location);
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// funcname ::= Name {`.' Name} [`:' Name]
|
2022-08-25 21:55:08 +01:00
|
|
|
AstExpr* Parser::parseFunctionName(Location start, bool& hasself, AstName& debugname)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-08-25 21:55:08 +01:00
|
|
|
if (lexer.current().type == Lexeme::Name)
|
|
|
|
debugname = AstName(lexer.current().name);
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
// parse funcname into a chain of indexing operators
|
|
|
|
AstExpr* expr = parseNameExpr("function name");
|
|
|
|
|
2023-10-27 20:33:36 +01:00
|
|
|
unsigned int oldRecursionCount = recursionCounter;
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
while (lexer.current().type == '.')
|
|
|
|
{
|
|
|
|
Position opPosition = lexer.current().location.begin;
|
|
|
|
nextLexeme();
|
|
|
|
|
|
|
|
Name name = parseName("field name");
|
|
|
|
|
|
|
|
// while we could concatenate the name chain, for now let's just write the short name
|
|
|
|
debugname = name.name;
|
|
|
|
|
|
|
|
expr = allocator.alloc<AstExprIndexName>(Location(start, name.location), expr, name.name, name.location, opPosition, '.');
|
|
|
|
|
|
|
|
// note: while the parser isn't recursive here, we're generating recursive structures of unbounded depth
|
|
|
|
incrementRecursionCounter("function name");
|
|
|
|
}
|
|
|
|
|
2023-10-27 20:33:36 +01:00
|
|
|
recursionCounter = oldRecursionCount;
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
// finish with :
|
|
|
|
if (lexer.current().type == ':')
|
|
|
|
{
|
|
|
|
Position opPosition = lexer.current().location.begin;
|
|
|
|
nextLexeme();
|
|
|
|
|
|
|
|
Name name = parseName("method name");
|
|
|
|
|
|
|
|
// while we could concatenate the name chain, for now let's just write the short name
|
|
|
|
debugname = name.name;
|
|
|
|
|
|
|
|
expr = allocator.alloc<AstExprIndexName>(Location(start, name.location), expr, name.name, name.location, opPosition, ':');
|
|
|
|
|
|
|
|
hasself = true;
|
|
|
|
}
|
|
|
|
|
2022-08-25 21:55:08 +01:00
|
|
|
return expr;
|
|
|
|
}
|
|
|
|
|
|
|
|
// function funcname funcbody
|
2024-06-07 18:09:03 +01:00
|
|
|
AstStat* Parser::parseFunctionStat(const AstArray<AstAttr*>& attributes)
|
2022-08-25 21:55:08 +01:00
|
|
|
{
|
|
|
|
Location start = lexer.current().location;
|
|
|
|
|
|
|
|
Lexeme matchFunction = lexer.current();
|
|
|
|
nextLexeme();
|
|
|
|
|
|
|
|
bool hasself = false;
|
|
|
|
AstName debugname;
|
|
|
|
AstExpr* expr = parseFunctionName(start, hasself, debugname);
|
|
|
|
|
2021-10-29 21:25:12 +01:00
|
|
|
matchRecoveryStopOnToken[Lexeme::ReservedEnd]++;
|
|
|
|
|
2024-06-07 18:09:03 +01:00
|
|
|
AstExprFunction* body = parseFunctionBody(hasself, matchFunction, debugname, nullptr, attributes).first;
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
matchRecoveryStopOnToken[Lexeme::ReservedEnd]--;
|
|
|
|
|
|
|
|
return allocator.alloc<AstStatFunction>(Location(start, body->location), expr, body);
|
|
|
|
}
|
|
|
|
|
2024-06-07 18:09:03 +01:00
|
|
|
|
|
|
|
std::pair<bool, AstAttr::Type> Parser::validateAttribute(const char* attributeName, const TempVector<AstAttr*>& attributes)
|
|
|
|
{
|
|
|
|
LUAU_ASSERT(FFlag::LuauAttributeSyntax);
|
|
|
|
|
|
|
|
AstAttr::Type type;
|
|
|
|
|
|
|
|
// check if the attribute name is valid
|
|
|
|
|
|
|
|
bool found = false;
|
|
|
|
|
|
|
|
for (int i = 0; kAttributeEntries[i].name; ++i)
|
|
|
|
{
|
|
|
|
found = !strcmp(attributeName, kAttributeEntries[i].name);
|
|
|
|
if (found)
|
|
|
|
{
|
|
|
|
type = kAttributeEntries[i].type;
|
2024-06-14 17:38:56 +01:00
|
|
|
|
|
|
|
if (!FFlag::LuauNativeAttribute && type == AstAttr::Type::Native)
|
|
|
|
found = false;
|
|
|
|
|
2024-06-07 18:09:03 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!found)
|
|
|
|
{
|
|
|
|
if (strlen(attributeName) == 1)
|
|
|
|
report(lexer.current().location, "Attribute name is missing");
|
|
|
|
else
|
|
|
|
report(lexer.current().location, "Invalid attribute '%s'", attributeName);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// check that attribute is not duplicated
|
|
|
|
for (const AstAttr* attr : attributes)
|
|
|
|
{
|
|
|
|
if (attr->type == type)
|
|
|
|
{
|
|
|
|
report(lexer.current().location, "Cannot duplicate attribute '%s'", attributeName);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return {found, type};
|
|
|
|
}
|
|
|
|
|
|
|
|
// attribute ::= '@' NAME
|
|
|
|
void Parser::parseAttribute(TempVector<AstAttr*>& attributes)
|
|
|
|
{
|
|
|
|
LUAU_ASSERT(FFlag::LuauAttributeSyntax);
|
|
|
|
|
|
|
|
LUAU_ASSERT(lexer.current().type == Lexeme::Type::Attribute);
|
|
|
|
|
|
|
|
Location loc = lexer.current().location;
|
|
|
|
|
|
|
|
const char* name = lexer.current().name;
|
|
|
|
const auto [found, type] = validateAttribute(name, attributes);
|
|
|
|
|
|
|
|
nextLexeme();
|
|
|
|
|
|
|
|
if (found)
|
|
|
|
attributes.push_back(allocator.alloc<AstAttr>(loc, type));
|
|
|
|
}
|
|
|
|
|
|
|
|
// attributes ::= {attribute}
|
|
|
|
AstArray<AstAttr*> Parser::parseAttributes()
|
|
|
|
{
|
|
|
|
LUAU_ASSERT(FFlag::LuauAttributeSyntax);
|
|
|
|
|
|
|
|
Lexeme::Type type = lexer.current().type;
|
|
|
|
|
|
|
|
LUAU_ASSERT(type == Lexeme::Attribute);
|
|
|
|
|
|
|
|
TempVector<AstAttr*> attributes(scratchAttr);
|
|
|
|
|
|
|
|
while (lexer.current().type == Lexeme::Attribute)
|
|
|
|
parseAttribute(attributes);
|
|
|
|
|
|
|
|
return copy(attributes);
|
|
|
|
}
|
|
|
|
|
|
|
|
// attributes local function Name funcbody
|
|
|
|
// attributes function funcname funcbody
|
|
|
|
// attributes `declare function' Name`(' [parlist] `)' [`:` Type]
|
|
|
|
// declare Name '{' Name ':' attributes `(' [parlist] `)' [`:` Type] '}'
|
|
|
|
AstStat* Parser::parseAttributeStat()
|
|
|
|
{
|
|
|
|
LUAU_ASSERT(FFlag::LuauAttributeSyntax);
|
|
|
|
|
2024-06-14 17:38:56 +01:00
|
|
|
AstArray<AstAttr*> attributes = parseAttributes();
|
2024-06-07 18:09:03 +01:00
|
|
|
|
|
|
|
Lexeme::Type type = lexer.current().type;
|
|
|
|
|
|
|
|
switch (type)
|
|
|
|
{
|
|
|
|
case Lexeme::Type::ReservedFunction:
|
|
|
|
return parseFunctionStat(attributes);
|
|
|
|
case Lexeme::Type::ReservedLocal:
|
|
|
|
return parseLocal(attributes);
|
|
|
|
case Lexeme::Type::Name:
|
|
|
|
if (options.allowDeclarationSyntax && !strcmp("declare", lexer.current().data))
|
|
|
|
{
|
|
|
|
AstExpr* expr = parsePrimaryExpr(/* asStatement= */ true);
|
|
|
|
return parseDeclaration(expr->location, attributes);
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
return reportStatError(lexer.current().location, {}, {},
|
|
|
|
"Expected 'function', 'local function', 'declare function' or a function type declaration after attribute, but got %s intead",
|
|
|
|
lexer.current().toString().c_str());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-29 21:25:12 +01:00
|
|
|
// local function Name funcbody |
|
|
|
|
// local bindinglist [`=' explist]
|
2024-06-07 18:09:03 +01:00
|
|
|
AstStat* Parser::parseLocal(const AstArray<AstAttr*>& attributes)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
Location start = lexer.current().location;
|
|
|
|
|
|
|
|
nextLexeme(); // local
|
|
|
|
|
|
|
|
if (lexer.current().type == Lexeme::ReservedFunction)
|
|
|
|
{
|
|
|
|
Lexeme matchFunction = lexer.current();
|
|
|
|
nextLexeme();
|
|
|
|
|
|
|
|
// matchFunction is only used for diagnostics; to make it suitable for detecting missed indentation between
|
|
|
|
// `local function` and `end`, we patch the token to begin at the column where `local` starts
|
|
|
|
if (matchFunction.location.begin.line == start.begin.line)
|
|
|
|
matchFunction.location.begin.column = start.begin.column;
|
|
|
|
|
|
|
|
Name name = parseName("variable name");
|
|
|
|
|
|
|
|
matchRecoveryStopOnToken[Lexeme::ReservedEnd]++;
|
|
|
|
|
2024-06-07 18:09:03 +01:00
|
|
|
auto [body, var] = parseFunctionBody(false, matchFunction, name.name, &name, attributes);
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
matchRecoveryStopOnToken[Lexeme::ReservedEnd]--;
|
|
|
|
|
|
|
|
Location location{start.begin, body->location.end};
|
|
|
|
|
|
|
|
return allocator.alloc<AstStatLocalFunction>(location, var, body);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2024-06-07 18:09:03 +01:00
|
|
|
if (FFlag::LuauAttributeSyntax && attributes.size != 0)
|
|
|
|
{
|
|
|
|
return reportStatError(lexer.current().location, {}, {}, "Expected 'function' after local declaration with attribute, but got %s intead",
|
|
|
|
lexer.current().toString().c_str());
|
|
|
|
}
|
|
|
|
|
2021-10-29 21:25:12 +01:00
|
|
|
matchRecoveryStopOnToken['=']++;
|
|
|
|
|
|
|
|
TempVector<Binding> names(scratchBinding);
|
|
|
|
parseBindingList(names);
|
|
|
|
|
|
|
|
matchRecoveryStopOnToken['=']--;
|
|
|
|
|
|
|
|
TempVector<AstLocal*> vars(scratchLocal);
|
|
|
|
|
|
|
|
TempVector<AstExpr*> values(scratchExpr);
|
|
|
|
|
|
|
|
std::optional<Location> equalsSignLocation;
|
|
|
|
|
|
|
|
if (lexer.current().type == '=')
|
|
|
|
{
|
|
|
|
equalsSignLocation = lexer.current().location;
|
|
|
|
|
|
|
|
nextLexeme();
|
|
|
|
|
|
|
|
parseExprList(values);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (size_t i = 0; i < names.size(); ++i)
|
|
|
|
vars.push_back(pushLocal(names[i]));
|
|
|
|
|
|
|
|
Location end = values.empty() ? lexer.previousLocation() : values.back()->location;
|
|
|
|
|
|
|
|
return allocator.alloc<AstStatLocal>(Location(start, end), copy(vars), copy(values), equalsSignLocation);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// return [explist]
|
|
|
|
AstStat* Parser::parseReturn()
|
|
|
|
{
|
|
|
|
Location start = lexer.current().location;
|
|
|
|
|
|
|
|
nextLexeme();
|
|
|
|
|
|
|
|
TempVector<AstExpr*> list(scratchExpr);
|
|
|
|
|
|
|
|
if (!blockFollow(lexer.current()) && lexer.current().type != ';')
|
|
|
|
parseExprList(list);
|
|
|
|
|
|
|
|
Location end = list.empty() ? start : list.back()->location;
|
|
|
|
|
|
|
|
return allocator.alloc<AstStatReturn>(Location(start, end), copy(list));
|
|
|
|
}
|
|
|
|
|
2023-03-17 14:59:30 +00:00
|
|
|
// type Name [`<' varlist `>'] `=' Type
|
2021-10-29 21:25:12 +01:00
|
|
|
AstStat* Parser::parseTypeAlias(const Location& start, bool exported)
|
|
|
|
{
|
|
|
|
// note: `type` token is already parsed for us, so we just need to parse the rest
|
|
|
|
|
2022-02-18 00:41:20 +00:00
|
|
|
std::optional<Name> name = parseNameOpt("type name");
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
// Use error name if the name is missing
|
|
|
|
if (!name)
|
|
|
|
name = Name(nameError, lexer.current().location);
|
|
|
|
|
2022-03-04 16:19:20 +00:00
|
|
|
auto [generics, genericPacks] = parseGenericTypeList(/* withDefaultValues= */ true);
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
expectAndConsume('=', "type alias");
|
|
|
|
|
2023-03-17 14:59:30 +00:00
|
|
|
AstType* type = parseType();
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2023-01-20 12:02:39 +00:00
|
|
|
return allocator.alloc<AstStatTypeAlias>(Location(start, type->location), name->name, name->location, generics, genericPacks, type, exported);
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
AstDeclaredClassProp Parser::parseDeclaredClassMethod()
|
|
|
|
{
|
2024-06-29 01:07:35 +01:00
|
|
|
Location start;
|
|
|
|
|
|
|
|
if (FFlag::LuauDeclarationExtraPropData)
|
|
|
|
start = lexer.current().location;
|
|
|
|
|
2021-10-29 21:25:12 +01:00
|
|
|
nextLexeme();
|
2024-06-29 01:07:35 +01:00
|
|
|
|
|
|
|
if (!FFlag::LuauDeclarationExtraPropData)
|
|
|
|
start = lexer.current().location;
|
|
|
|
|
2021-10-29 21:25:12 +01:00
|
|
|
Name fnName = parseName("function name");
|
|
|
|
|
|
|
|
// TODO: generic method declarations CLI-39909
|
2022-01-14 16:06:31 +00:00
|
|
|
AstArray<AstGenericType> generics;
|
|
|
|
AstArray<AstGenericTypePack> genericPacks;
|
2021-10-29 21:25:12 +01:00
|
|
|
generics.size = 0;
|
|
|
|
generics.data = nullptr;
|
|
|
|
genericPacks.size = 0;
|
|
|
|
genericPacks.data = nullptr;
|
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
MatchLexeme matchParen = lexer.current();
|
2021-10-29 21:25:12 +01:00
|
|
|
expectAndConsume('(', "function parameter list start");
|
|
|
|
|
|
|
|
TempVector<Binding> args(scratchBinding);
|
|
|
|
|
2022-08-25 21:55:08 +01:00
|
|
|
bool vararg = false;
|
|
|
|
Location varargLocation;
|
2021-10-29 21:25:12 +01:00
|
|
|
AstTypePack* varargAnnotation = nullptr;
|
|
|
|
if (lexer.current().type != ')')
|
2022-08-25 21:55:08 +01:00
|
|
|
std::tie(vararg, varargLocation, varargAnnotation) = parseBindingList(args, /* allowDot3 */ true);
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
expectMatchAndConsume(')', matchParen);
|
|
|
|
|
2023-03-17 14:59:30 +00:00
|
|
|
AstTypeList retTypes = parseOptionalReturnType().value_or(AstTypeList{copy<AstType*>(nullptr, 0), nullptr});
|
2024-06-29 01:07:35 +01:00
|
|
|
Location end = FFlag::LuauDeclarationExtraPropData ? lexer.previousLocation() : lexer.current().location;
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2023-03-17 14:59:30 +00:00
|
|
|
TempVector<AstType*> vars(scratchType);
|
2021-10-29 21:25:12 +01:00
|
|
|
TempVector<std::optional<AstArgumentName>> varNames(scratchOptArgName);
|
|
|
|
|
|
|
|
if (args.size() == 0 || args[0].name.name != "self" || args[0].annotation != nullptr)
|
|
|
|
{
|
2024-06-29 01:07:35 +01:00
|
|
|
return AstDeclaredClassProp{fnName.name, FFlag::LuauDeclarationExtraPropData ? fnName.location : Location{},
|
|
|
|
reportTypeError(Location(start, end), {}, "'self' must be present as the unannotated first parameter"), true};
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Skip the first index.
|
|
|
|
for (size_t i = 1; i < args.size(); ++i)
|
|
|
|
{
|
|
|
|
varNames.push_back(AstArgumentName{args[i].name.name, args[i].name.location});
|
|
|
|
|
|
|
|
if (args[i].annotation)
|
|
|
|
vars.push_back(args[i].annotation);
|
|
|
|
else
|
2023-03-17 14:59:30 +00:00
|
|
|
vars.push_back(reportTypeError(Location(start, end), {}, "All declaration parameters aside from 'self' must be annotated"));
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if (vararg && !varargAnnotation)
|
|
|
|
report(start, "All declaration parameters aside from 'self' must be annotated");
|
|
|
|
|
|
|
|
AstType* fnType = allocator.alloc<AstTypeFunction>(
|
|
|
|
Location(start, end), generics, genericPacks, AstTypeList{copy(vars), varargAnnotation}, copy(varNames), retTypes);
|
|
|
|
|
2024-06-29 01:07:35 +01:00
|
|
|
return AstDeclaredClassProp{fnName.name, FFlag::LuauDeclarationExtraPropData ? fnName.location : Location{}, fnType, true,
|
|
|
|
FFlag::LuauDeclarationExtraPropData ? Location(start, end) : Location{}};
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
2024-06-07 18:09:03 +01:00
|
|
|
AstStat* Parser::parseDeclaration(const Location& start, const AstArray<AstAttr*>& attributes)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
// `declare` token is already parsed at this point
|
2024-06-07 18:09:03 +01:00
|
|
|
|
|
|
|
if (FFlag::LuauAttributeSyntax && (attributes.size != 0) && (lexer.current().type != Lexeme::ReservedFunction))
|
|
|
|
return reportStatError(lexer.current().location, {}, {}, "Expected a function type declaration after attribute, but got %s intead",
|
|
|
|
lexer.current().toString().c_str());
|
|
|
|
|
2021-10-29 21:25:12 +01:00
|
|
|
if (lexer.current().type == Lexeme::ReservedFunction)
|
|
|
|
{
|
|
|
|
nextLexeme();
|
|
|
|
|
2023-10-06 18:31:16 +01:00
|
|
|
Name globalName = parseName("global function name");
|
2022-01-14 16:06:31 +00:00
|
|
|
auto [generics, genericPacks] = parseGenericTypeList(/* withDefaultValues= */ false);
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
MatchLexeme matchParen = lexer.current();
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
expectAndConsume('(', "global function declaration");
|
|
|
|
|
|
|
|
TempVector<Binding> args(scratchBinding);
|
|
|
|
|
2022-08-25 21:55:08 +01:00
|
|
|
bool vararg = false;
|
|
|
|
Location varargLocation;
|
2021-10-29 21:25:12 +01:00
|
|
|
AstTypePack* varargAnnotation = nullptr;
|
|
|
|
|
|
|
|
if (lexer.current().type != ')')
|
2022-08-25 21:55:08 +01:00
|
|
|
std::tie(vararg, varargLocation, varargAnnotation) = parseBindingList(args, /* allowDot3= */ true);
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
expectMatchAndConsume(')', matchParen);
|
|
|
|
|
2023-03-17 14:59:30 +00:00
|
|
|
AstTypeList retTypes = parseOptionalReturnType().value_or(AstTypeList{copy<AstType*>(nullptr, 0)});
|
2021-10-29 21:25:12 +01:00
|
|
|
Location end = lexer.current().location;
|
|
|
|
|
2023-03-17 14:59:30 +00:00
|
|
|
TempVector<AstType*> vars(scratchType);
|
2021-10-29 21:25:12 +01:00
|
|
|
TempVector<AstArgumentName> varNames(scratchArgName);
|
|
|
|
|
|
|
|
for (size_t i = 0; i < args.size(); ++i)
|
|
|
|
{
|
|
|
|
if (!args[i].annotation)
|
|
|
|
return reportStatError(Location(start, end), {}, {}, "All declaration parameters must be annotated");
|
|
|
|
|
|
|
|
vars.push_back(args[i].annotation);
|
|
|
|
varNames.push_back({args[i].name.name, args[i].name.location});
|
|
|
|
}
|
|
|
|
|
|
|
|
if (vararg && !varargAnnotation)
|
|
|
|
return reportStatError(Location(start, end), {}, {}, "All declaration parameters must be annotated");
|
|
|
|
|
2024-06-29 01:07:35 +01:00
|
|
|
if (FFlag::LuauDeclarationExtraPropData)
|
|
|
|
return allocator.alloc<AstStatDeclareFunction>(Location(start, end), attributes, globalName.name, globalName.location, generics,
|
|
|
|
genericPacks, AstTypeList{copy(vars), varargAnnotation}, copy(varNames), vararg, varargLocation, retTypes);
|
|
|
|
else
|
|
|
|
return allocator.alloc<AstStatDeclareFunction>(Location(start, end), attributes, globalName.name, Location{}, generics, genericPacks,
|
|
|
|
AstTypeList{copy(vars), varargAnnotation}, copy(varNames), false, Location{}, retTypes);
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
else if (AstName(lexer.current().name) == "class")
|
|
|
|
{
|
|
|
|
nextLexeme();
|
|
|
|
Location classStart = lexer.current().location;
|
|
|
|
Name className = parseName("class name");
|
|
|
|
std::optional<AstName> superName = std::nullopt;
|
|
|
|
|
|
|
|
if (AstName(lexer.current().name) == "extends")
|
|
|
|
{
|
|
|
|
nextLexeme();
|
|
|
|
superName = parseName("superclass name").name;
|
|
|
|
}
|
|
|
|
|
|
|
|
TempVector<AstDeclaredClassProp> props(scratchDeclaredClassProps);
|
2023-06-16 18:01:18 +01:00
|
|
|
AstTableIndexer* indexer = nullptr;
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
while (lexer.current().type != Lexeme::ReservedEnd)
|
|
|
|
{
|
|
|
|
// There are two possibilities: Either it's a property or a function.
|
|
|
|
if (lexer.current().type == Lexeme::ReservedFunction)
|
|
|
|
{
|
|
|
|
props.push_back(parseDeclaredClassMethod());
|
|
|
|
}
|
2024-02-16 01:25:31 +00:00
|
|
|
else if (lexer.current().type == '[' && (lexer.lookahead().type == Lexeme::RawString || lexer.lookahead().type == Lexeme::QuotedString))
|
2022-10-21 18:33:43 +01:00
|
|
|
{
|
|
|
|
const Lexeme begin = lexer.current();
|
|
|
|
nextLexeme(); // [
|
|
|
|
|
2024-06-29 01:07:35 +01:00
|
|
|
if (FFlag::LuauDeclarationExtraPropData)
|
|
|
|
{
|
|
|
|
const Location nameBegin = lexer.current().location;
|
|
|
|
std::optional<AstArray<char>> chars = parseCharArray();
|
2022-10-21 18:33:43 +01:00
|
|
|
|
2024-06-29 01:07:35 +01:00
|
|
|
const Location nameEnd = lexer.previousLocation();
|
2022-10-21 18:33:43 +01:00
|
|
|
|
2024-06-29 01:07:35 +01:00
|
|
|
expectMatchAndConsume(']', begin);
|
|
|
|
expectAndConsume(':', "property type annotation");
|
|
|
|
AstType* type = parseType();
|
2022-10-21 18:33:43 +01:00
|
|
|
|
2024-06-29 01:07:35 +01:00
|
|
|
// since AstName contains a char*, it can't contain null
|
|
|
|
bool containsNull = chars && (strnlen(chars->data, chars->size) < chars->size);
|
|
|
|
|
|
|
|
if (chars && !containsNull)
|
|
|
|
props.push_back(AstDeclaredClassProp{
|
|
|
|
AstName(chars->data), Location(nameBegin, nameEnd), type, false, Location(begin.location, lexer.previousLocation())});
|
|
|
|
else
|
|
|
|
report(begin.location, "String literal contains malformed escape sequence or \\0");
|
|
|
|
}
|
2022-10-21 18:33:43 +01:00
|
|
|
else
|
2024-06-29 01:07:35 +01:00
|
|
|
{
|
|
|
|
std::optional<AstArray<char>> chars = parseCharArray();
|
|
|
|
|
|
|
|
expectMatchAndConsume(']', begin);
|
|
|
|
expectAndConsume(':', "property type annotation");
|
|
|
|
AstType* type = parseType();
|
|
|
|
|
|
|
|
// since AstName contains a char*, it can't contain null
|
|
|
|
bool containsNull = chars && (strnlen(chars->data, chars->size) < chars->size);
|
|
|
|
|
|
|
|
if (chars && !containsNull)
|
|
|
|
props.push_back(AstDeclaredClassProp{AstName(chars->data), Location{}, type, false});
|
|
|
|
else
|
|
|
|
report(begin.location, "String literal contains malformed escape sequence or \\0");
|
|
|
|
}
|
2022-10-21 18:33:43 +01:00
|
|
|
}
|
2023-11-10 18:05:48 +00:00
|
|
|
else if (lexer.current().type == '[')
|
2023-06-16 18:01:18 +01:00
|
|
|
{
|
|
|
|
if (indexer)
|
|
|
|
{
|
|
|
|
// maybe we don't need to parse the entire badIndexer...
|
|
|
|
// however, we either have { or [ to lint, not the entire table type or the bad indexer.
|
2024-01-19 15:13:08 +00:00
|
|
|
AstTableIndexer* badIndexer = parseTableIndexer(AstTableAccess::ReadWrite, std::nullopt);
|
2023-06-16 18:01:18 +01:00
|
|
|
|
|
|
|
// we lose all additional indexer expressions from the AST after error recovery here
|
|
|
|
report(badIndexer->location, "Cannot have more than one class indexer");
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2024-01-19 15:13:08 +00:00
|
|
|
indexer = parseTableIndexer(AstTableAccess::ReadWrite, std::nullopt);
|
2023-06-16 18:01:18 +01:00
|
|
|
}
|
|
|
|
}
|
2024-06-29 01:07:35 +01:00
|
|
|
else if (FFlag::LuauDeclarationExtraPropData)
|
|
|
|
{
|
|
|
|
Location propStart = lexer.current().location;
|
|
|
|
Name propName = parseName("property name");
|
|
|
|
expectAndConsume(':', "property type annotation");
|
|
|
|
AstType* propType = parseType();
|
|
|
|
props.push_back(
|
|
|
|
AstDeclaredClassProp{propName.name, propName.location, propType, false, Location(propStart, lexer.previousLocation())});
|
|
|
|
}
|
2021-10-29 21:25:12 +01:00
|
|
|
else
|
|
|
|
{
|
|
|
|
Name propName = parseName("property name");
|
|
|
|
expectAndConsume(':', "property type annotation");
|
2023-03-17 14:59:30 +00:00
|
|
|
AstType* propType = parseType();
|
2024-06-29 01:07:35 +01:00
|
|
|
props.push_back(AstDeclaredClassProp{propName.name, Location{}, propType, false});
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Location classEnd = lexer.current().location;
|
|
|
|
nextLexeme(); // skip past `end`
|
|
|
|
|
2023-06-16 18:01:18 +01:00
|
|
|
return allocator.alloc<AstStatDeclareClass>(Location(classStart, classEnd), className.name, superName, copy(props), indexer);
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
2022-02-18 00:41:20 +00:00
|
|
|
else if (std::optional<Name> globalName = parseNameOpt("global variable name"))
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
expectAndConsume(':', "global variable declaration");
|
|
|
|
|
2023-10-06 18:31:16 +01:00
|
|
|
AstType* type = parseType(/* in declaration context */ true);
|
2024-06-29 01:07:35 +01:00
|
|
|
return allocator.alloc<AstStatDeclareGlobal>(
|
|
|
|
Location(start, type->location), globalName->name, FFlag::LuauDeclarationExtraPropData ? globalName->location : Location{}, type);
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return reportStatError(start, {}, {}, "declare must be followed by an identifier, 'function', or 'class'");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool isExprLValue(AstExpr* expr)
|
|
|
|
{
|
|
|
|
return expr->is<AstExprLocal>() || expr->is<AstExprGlobal>() || expr->is<AstExprIndexExpr>() || expr->is<AstExprIndexName>();
|
|
|
|
}
|
|
|
|
|
|
|
|
// varlist `=' explist
|
|
|
|
AstStat* Parser::parseAssignment(AstExpr* initial)
|
|
|
|
{
|
|
|
|
if (!isExprLValue(initial))
|
|
|
|
initial = reportExprError(initial->location, copy({initial}), "Assigned expression must be a variable or a field");
|
|
|
|
|
|
|
|
TempVector<AstExpr*> vars(scratchExpr);
|
|
|
|
vars.push_back(initial);
|
|
|
|
|
|
|
|
while (lexer.current().type == ',')
|
|
|
|
{
|
|
|
|
nextLexeme();
|
|
|
|
|
2022-02-03 23:09:37 +00:00
|
|
|
AstExpr* expr = parsePrimaryExpr(/* asStatement= */ true);
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
if (!isExprLValue(expr))
|
|
|
|
expr = reportExprError(expr->location, copy({expr}), "Assigned expression must be a variable or a field");
|
|
|
|
|
|
|
|
vars.push_back(expr);
|
|
|
|
}
|
|
|
|
|
|
|
|
expectAndConsume('=', "assignment");
|
|
|
|
|
|
|
|
TempVector<AstExpr*> values(scratchExprAux);
|
|
|
|
parseExprList(values);
|
|
|
|
|
|
|
|
return allocator.alloc<AstStatAssign>(Location(initial->location, values.back()->location), copy(vars), copy(values));
|
|
|
|
}
|
|
|
|
|
|
|
|
// var [`+=' | `-=' | `*=' | `/=' | `%=' | `^=' | `..='] exp
|
|
|
|
AstStat* Parser::parseCompoundAssignment(AstExpr* initial, AstExprBinary::Op op)
|
|
|
|
{
|
|
|
|
if (!isExprLValue(initial))
|
|
|
|
{
|
|
|
|
initial = reportExprError(initial->location, copy({initial}), "Assigned expression must be a variable or a field");
|
|
|
|
}
|
|
|
|
|
|
|
|
nextLexeme();
|
|
|
|
|
|
|
|
AstExpr* value = parseExpr();
|
|
|
|
|
|
|
|
return allocator.alloc<AstStatCompoundAssign>(Location(initial->location, value->location), op, initial, value);
|
|
|
|
}
|
|
|
|
|
2022-08-25 21:55:08 +01:00
|
|
|
std::pair<AstLocal*, AstArray<AstLocal*>> Parser::prepareFunctionArguments(const Location& start, bool hasself, const TempVector<Binding>& args)
|
|
|
|
{
|
|
|
|
AstLocal* self = nullptr;
|
|
|
|
|
|
|
|
if (hasself)
|
|
|
|
self = pushLocal(Binding(Name(nameSelf, start), nullptr));
|
|
|
|
|
|
|
|
TempVector<AstLocal*> vars(scratchLocal);
|
|
|
|
|
|
|
|
for (size_t i = 0; i < args.size(); ++i)
|
|
|
|
vars.push_back(pushLocal(args[i]));
|
|
|
|
|
|
|
|
return {self, copy(vars)};
|
|
|
|
}
|
|
|
|
|
2021-10-29 21:25:12 +01:00
|
|
|
// funcbody ::= `(' [parlist] `)' [`:' ReturnType] block end
|
|
|
|
// parlist ::= bindinglist [`,' `...'] | `...'
|
|
|
|
std::pair<AstExprFunction*, AstLocal*> Parser::parseFunctionBody(
|
2024-06-07 18:09:03 +01:00
|
|
|
bool hasself, const Lexeme& matchFunction, const AstName& debugname, const Name* localName, const AstArray<AstAttr*>& attributes)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
Location start = matchFunction.location;
|
|
|
|
|
2022-01-14 16:06:31 +00:00
|
|
|
auto [generics, genericPacks] = parseGenericTypeList(/* withDefaultValues= */ false);
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
MatchLexeme matchParen = lexer.current();
|
2021-10-29 21:25:12 +01:00
|
|
|
expectAndConsume('(', "function");
|
|
|
|
|
|
|
|
TempVector<Binding> args(scratchBinding);
|
|
|
|
|
2022-08-25 21:55:08 +01:00
|
|
|
bool vararg = false;
|
|
|
|
Location varargLocation;
|
2021-10-29 21:25:12 +01:00
|
|
|
AstTypePack* varargAnnotation = nullptr;
|
|
|
|
|
|
|
|
if (lexer.current().type != ')')
|
2022-08-25 21:55:08 +01:00
|
|
|
std::tie(vararg, varargLocation, varargAnnotation) = parseBindingList(args, /* allowDot3= */ true);
|
|
|
|
|
|
|
|
std::optional<Location> argLocation;
|
|
|
|
|
|
|
|
if (matchParen.type == Lexeme::Type('(') && lexer.current().type == Lexeme::Type(')'))
|
|
|
|
argLocation = Location(matchParen.position, lexer.current().location.end);
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
expectMatchAndConsume(')', matchParen, true);
|
|
|
|
|
2023-03-17 14:59:30 +00:00
|
|
|
std::optional<AstTypeList> typelist = parseOptionalReturnType();
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
AstLocal* funLocal = nullptr;
|
|
|
|
|
|
|
|
if (localName)
|
|
|
|
funLocal = pushLocal(Binding(*localName, nullptr));
|
|
|
|
|
|
|
|
unsigned int localsBegin = saveLocals();
|
|
|
|
|
|
|
|
Function fun;
|
2022-08-25 21:55:08 +01:00
|
|
|
fun.vararg = vararg;
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-08-25 21:55:08 +01:00
|
|
|
functionStack.emplace_back(fun);
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-08-25 21:55:08 +01:00
|
|
|
auto [self, vars] = prepareFunctionArguments(start, hasself, args);
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
AstStatBlock* body = parseBlock();
|
|
|
|
|
|
|
|
functionStack.pop_back();
|
|
|
|
|
|
|
|
restoreLocals(localsBegin);
|
|
|
|
|
|
|
|
Location end = lexer.current().location;
|
|
|
|
|
|
|
|
bool hasEnd = expectMatchEndAndConsume(Lexeme::ReservedEnd, matchFunction);
|
2024-01-27 02:30:40 +00:00
|
|
|
body->hasEnd = hasEnd;
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2024-06-07 18:09:03 +01:00
|
|
|
return {allocator.alloc<AstExprFunction>(Location(start, end), attributes, generics, genericPacks, self, vars, vararg, varargLocation, body,
|
2024-01-27 02:30:40 +00:00
|
|
|
functionStack.size(), debugname, typelist, varargAnnotation, argLocation),
|
2021-10-29 21:25:12 +01:00
|
|
|
funLocal};
|
|
|
|
}
|
|
|
|
|
|
|
|
// explist ::= {exp `,'} exp
|
|
|
|
void Parser::parseExprList(TempVector<AstExpr*>& result)
|
|
|
|
{
|
|
|
|
result.push_back(parseExpr());
|
|
|
|
|
|
|
|
while (lexer.current().type == ',')
|
|
|
|
{
|
|
|
|
nextLexeme();
|
|
|
|
|
2022-12-02 10:46:05 +00:00
|
|
|
if (lexer.current().type == ')')
|
2022-10-07 00:55:58 +01:00
|
|
|
{
|
|
|
|
report(lexer.current().location, "Expected expression after ',' but got ')' instead");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2021-10-29 21:25:12 +01:00
|
|
|
result.push_back(parseExpr());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Parser::Binding Parser::parseBinding()
|
|
|
|
{
|
2022-02-18 00:41:20 +00:00
|
|
|
std::optional<Name> name = parseNameOpt("variable name");
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
// Use placeholder if the name is missing
|
|
|
|
if (!name)
|
|
|
|
name = Name(nameError, lexer.current().location);
|
|
|
|
|
2023-03-17 14:59:30 +00:00
|
|
|
AstType* annotation = parseOptionalType();
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
return Binding(*name, annotation);
|
|
|
|
}
|
|
|
|
|
|
|
|
// bindinglist ::= (binding | `...') [`,' bindinglist]
|
2022-08-25 21:55:08 +01:00
|
|
|
std::tuple<bool, Location, AstTypePack*> Parser::parseBindingList(TempVector<Binding>& result, bool allowDot3)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
while (true)
|
|
|
|
{
|
|
|
|
if (lexer.current().type == Lexeme::Dot3 && allowDot3)
|
|
|
|
{
|
|
|
|
Location varargLocation = lexer.current().location;
|
|
|
|
nextLexeme();
|
|
|
|
|
|
|
|
AstTypePack* tailAnnotation = nullptr;
|
|
|
|
if (lexer.current().type == ':')
|
|
|
|
{
|
|
|
|
nextLexeme();
|
2023-03-17 14:59:30 +00:00
|
|
|
tailAnnotation = parseVariadicArgumentTypePack();
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
2022-08-25 21:55:08 +01:00
|
|
|
return {true, varargLocation, tailAnnotation};
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
result.push_back(parseBinding());
|
|
|
|
|
|
|
|
if (lexer.current().type != ',')
|
|
|
|
break;
|
|
|
|
nextLexeme();
|
|
|
|
}
|
|
|
|
|
2022-08-25 21:55:08 +01:00
|
|
|
return {false, Location(), nullptr};
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
2023-03-17 14:59:30 +00:00
|
|
|
AstType* Parser::parseOptionalType()
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2023-05-05 20:57:12 +01:00
|
|
|
if (lexer.current().type == ':')
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
nextLexeme();
|
2023-03-17 14:59:30 +00:00
|
|
|
return parseType();
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2023-03-17 14:59:30 +00:00
|
|
|
// TypeList ::= Type [`,' TypeList] | ...Type
|
2021-10-29 21:25:12 +01:00
|
|
|
AstTypePack* Parser::parseTypeList(TempVector<AstType*>& result, TempVector<std::optional<AstArgumentName>>& resultNames)
|
|
|
|
{
|
|
|
|
while (true)
|
|
|
|
{
|
2023-03-17 14:59:30 +00:00
|
|
|
if (shouldParseTypePack(lexer))
|
|
|
|
return parseTypePack();
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
if (lexer.current().type == Lexeme::Name && lexer.lookahead().type == ':')
|
|
|
|
{
|
|
|
|
// Fill in previous argument names with empty slots
|
|
|
|
while (resultNames.size() < result.size())
|
|
|
|
resultNames.push_back({});
|
|
|
|
|
|
|
|
resultNames.push_back(AstArgumentName{AstName(lexer.current().name), lexer.current().location});
|
2022-02-11 18:43:14 +00:00
|
|
|
nextLexeme();
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
expectAndConsume(':');
|
|
|
|
}
|
|
|
|
else if (!resultNames.empty())
|
|
|
|
{
|
|
|
|
// If we have a type with named arguments, provide elements for all types
|
|
|
|
resultNames.push_back({});
|
|
|
|
}
|
|
|
|
|
2023-03-17 14:59:30 +00:00
|
|
|
result.push_back(parseType());
|
2021-10-29 21:25:12 +01:00
|
|
|
if (lexer.current().type != ',')
|
|
|
|
break;
|
2022-10-07 00:55:58 +01:00
|
|
|
|
2021-10-29 21:25:12 +01:00
|
|
|
nextLexeme();
|
2022-10-07 00:55:58 +01:00
|
|
|
|
2022-12-02 10:46:05 +00:00
|
|
|
if (lexer.current().type == ')')
|
2022-10-07 00:55:58 +01:00
|
|
|
{
|
|
|
|
report(lexer.current().location, "Expected type after ',' but got ')' instead");
|
|
|
|
break;
|
|
|
|
}
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2023-03-17 14:59:30 +00:00
|
|
|
std::optional<AstTypeList> Parser::parseOptionalReturnType()
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2023-05-05 20:57:12 +01:00
|
|
|
if (lexer.current().type == ':' || lexer.current().type == Lexeme::SkinnyArrow)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-07-08 02:05:31 +01:00
|
|
|
if (lexer.current().type == Lexeme::SkinnyArrow)
|
2022-06-17 01:52:23 +01:00
|
|
|
report(lexer.current().location, "Function return type annotations are written after ':' instead of '->'");
|
|
|
|
|
2021-10-29 21:25:12 +01:00
|
|
|
nextLexeme();
|
|
|
|
|
|
|
|
unsigned int oldRecursionCount = recursionCounter;
|
|
|
|
|
2023-03-17 14:59:30 +00:00
|
|
|
auto [_location, result] = parseReturnType();
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
// At this point, if we find a , character, it indicates that there are multiple return types
|
|
|
|
// in this type annotation, but the list wasn't wrapped in parentheses.
|
|
|
|
if (lexer.current().type == ',')
|
|
|
|
{
|
|
|
|
report(lexer.current().location, "Expected a statement, got ','; did you forget to wrap the list of return types in parentheses?");
|
|
|
|
|
|
|
|
nextLexeme();
|
|
|
|
}
|
|
|
|
|
|
|
|
recursionCounter = oldRecursionCount;
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
return std::nullopt;
|
|
|
|
}
|
|
|
|
|
2023-03-17 14:59:30 +00:00
|
|
|
// ReturnType ::= Type | `(' TypeList `)'
|
|
|
|
std::pair<Location, AstTypeList> Parser::parseReturnType()
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
incrementRecursionCounter("type annotation");
|
|
|
|
|
|
|
|
Lexeme begin = lexer.current();
|
|
|
|
|
|
|
|
if (lexer.current().type != '(')
|
|
|
|
{
|
2023-03-17 14:59:30 +00:00
|
|
|
if (shouldParseTypePack(lexer))
|
|
|
|
{
|
|
|
|
AstTypePack* typePack = parseTypePack();
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2023-03-17 14:59:30 +00:00
|
|
|
return {typePack->location, AstTypeList{{}, typePack}};
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
AstType* type = parseType();
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2023-03-17 14:59:30 +00:00
|
|
|
return {type->location, AstTypeList{copy(&type, 1), nullptr}};
|
|
|
|
}
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
nextLexeme();
|
|
|
|
|
|
|
|
Location innerBegin = lexer.current().location;
|
|
|
|
|
|
|
|
matchRecoveryStopOnToken[Lexeme::SkinnyArrow]++;
|
|
|
|
|
2023-03-17 14:59:30 +00:00
|
|
|
TempVector<AstType*> result(scratchType);
|
|
|
|
TempVector<std::optional<AstArgumentName>> resultNames(scratchOptArgName);
|
|
|
|
AstTypePack* varargAnnotation = nullptr;
|
|
|
|
|
2021-10-29 21:25:12 +01:00
|
|
|
// possibly () -> ReturnType
|
|
|
|
if (lexer.current().type != ')')
|
|
|
|
varargAnnotation = parseTypeList(result, resultNames);
|
|
|
|
|
|
|
|
const Location location{begin.location, lexer.current().location};
|
|
|
|
|
|
|
|
expectMatchAndConsume(')', begin, true);
|
|
|
|
|
|
|
|
matchRecoveryStopOnToken[Lexeme::SkinnyArrow]--;
|
|
|
|
|
|
|
|
if (lexer.current().type != Lexeme::SkinnyArrow && resultNames.empty())
|
|
|
|
{
|
|
|
|
// If it turns out that it's just '(A)', it's possible that there are unions/intersections to follow, so fold over it.
|
|
|
|
if (result.size() == 1)
|
|
|
|
{
|
2023-03-17 14:59:30 +00:00
|
|
|
AstType* returnType = parseTypeSuffix(result[0], innerBegin);
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2023-03-17 14:59:30 +00:00
|
|
|
// If parseType parses nothing, then returnType->location.end only points at the last non-type-pack
|
2023-03-03 13:45:38 +00:00
|
|
|
// type to successfully parse. We need the span of the whole annotation.
|
|
|
|
Position endPos = result.size() == 1 ? location.end : returnType->location.end;
|
|
|
|
|
|
|
|
return {Location{location.begin, endPos}, AstTypeList{copy(&returnType, 1), varargAnnotation}};
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return {location, AstTypeList{copy(result), varargAnnotation}};
|
|
|
|
}
|
|
|
|
|
2024-06-07 18:09:03 +01:00
|
|
|
AstType* tail = parseFunctionTypeTail(begin, {nullptr, 0}, {}, {}, copy(result), copy(resultNames), varargAnnotation);
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2023-03-17 14:59:30 +00:00
|
|
|
return {Location{location, tail->location}, AstTypeList{copy(&tail, 1), varargAnnotation}};
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
2023-03-17 14:59:30 +00:00
|
|
|
// TableIndexer ::= `[' Type `]' `:' Type
|
2024-01-19 15:13:08 +00:00
|
|
|
AstTableIndexer* Parser::parseTableIndexer(AstTableAccess access, std::optional<Location> accessLocation)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
const Lexeme begin = lexer.current();
|
|
|
|
nextLexeme(); // [
|
|
|
|
|
2023-03-17 14:59:30 +00:00
|
|
|
AstType* index = parseType();
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
expectMatchAndConsume(']', begin);
|
|
|
|
|
|
|
|
expectAndConsume(':', "table field");
|
|
|
|
|
2023-03-17 14:59:30 +00:00
|
|
|
AstType* result = parseType();
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2024-01-19 15:13:08 +00:00
|
|
|
return allocator.alloc<AstTableIndexer>(AstTableIndexer{index, result, Location(begin.location, result->location), access, accessLocation});
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
2023-03-17 14:59:30 +00:00
|
|
|
// TableProp ::= Name `:' Type
|
2021-10-29 21:25:12 +01:00
|
|
|
// TablePropOrIndexer ::= TableProp | TableIndexer
|
|
|
|
// PropList ::= TablePropOrIndexer {fieldsep TablePropOrIndexer} [fieldsep]
|
2023-03-17 14:59:30 +00:00
|
|
|
// TableType ::= `{' PropList `}'
|
2023-10-06 18:31:16 +01:00
|
|
|
AstType* Parser::parseTableType(bool inDeclarationContext)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
incrementRecursionCounter("type annotation");
|
|
|
|
|
|
|
|
TempVector<AstTableProp> props(scratchTableTypeProps);
|
|
|
|
AstTableIndexer* indexer = nullptr;
|
|
|
|
|
|
|
|
Location start = lexer.current().location;
|
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
MatchLexeme matchBrace = lexer.current();
|
2021-10-29 21:25:12 +01:00
|
|
|
expectAndConsume('{', "table type");
|
|
|
|
|
|
|
|
while (lexer.current().type != '}')
|
|
|
|
{
|
2024-01-19 15:13:08 +00:00
|
|
|
AstTableAccess access = AstTableAccess::ReadWrite;
|
|
|
|
std::optional<Location> accessLocation;
|
|
|
|
|
2024-05-10 17:17:09 +01:00
|
|
|
if (lexer.current().type == Lexeme::Name && lexer.lookahead().type != ':')
|
2024-01-19 15:13:08 +00:00
|
|
|
{
|
2024-05-10 17:17:09 +01:00
|
|
|
if (AstName(lexer.current().name) == "read")
|
2024-01-19 15:13:08 +00:00
|
|
|
{
|
2024-05-10 17:17:09 +01:00
|
|
|
accessLocation = lexer.current().location;
|
|
|
|
access = AstTableAccess::Read;
|
|
|
|
lexer.next();
|
|
|
|
}
|
|
|
|
else if (AstName(lexer.current().name) == "write")
|
|
|
|
{
|
|
|
|
accessLocation = lexer.current().location;
|
|
|
|
access = AstTableAccess::Write;
|
|
|
|
lexer.next();
|
2024-01-19 15:13:08 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-24 21:49:08 +00:00
|
|
|
if (lexer.current().type == '[' && (lexer.lookahead().type == Lexeme::RawString || lexer.lookahead().type == Lexeme::QuotedString))
|
2021-11-18 22:21:07 +00:00
|
|
|
{
|
|
|
|
const Lexeme begin = lexer.current();
|
|
|
|
nextLexeme(); // [
|
|
|
|
std::optional<AstArray<char>> chars = parseCharArray();
|
|
|
|
|
|
|
|
expectMatchAndConsume(']', begin);
|
|
|
|
expectAndConsume(':', "table field");
|
|
|
|
|
2023-03-17 14:59:30 +00:00
|
|
|
AstType* type = parseType();
|
2021-11-18 22:21:07 +00:00
|
|
|
|
2023-08-25 16:25:09 +01:00
|
|
|
// since AstName contains a char*, it can't contain null
|
2021-11-18 22:21:07 +00:00
|
|
|
bool containsNull = chars && (strnlen(chars->data, chars->size) < chars->size);
|
|
|
|
|
|
|
|
if (chars && !containsNull)
|
2024-01-19 15:13:08 +00:00
|
|
|
props.push_back(AstTableProp{AstName(chars->data), begin.location, type, access, accessLocation});
|
2021-11-18 22:21:07 +00:00
|
|
|
else
|
2023-08-25 16:25:09 +01:00
|
|
|
report(begin.location, "String literal contains malformed escape sequence or \\0");
|
2021-11-18 22:21:07 +00:00
|
|
|
}
|
|
|
|
else if (lexer.current().type == '[')
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
if (indexer)
|
|
|
|
{
|
|
|
|
// maybe we don't need to parse the entire badIndexer...
|
|
|
|
// however, we either have { or [ to lint, not the entire table type or the bad indexer.
|
2024-01-19 15:13:08 +00:00
|
|
|
AstTableIndexer* badIndexer = parseTableIndexer(access, accessLocation);
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
// we lose all additional indexer expressions from the AST after error recovery here
|
|
|
|
report(badIndexer->location, "Cannot have more than one table indexer");
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2024-01-19 15:13:08 +00:00
|
|
|
indexer = parseTableIndexer(access, accessLocation);
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (props.empty() && !indexer && !(lexer.current().type == Lexeme::Name && lexer.lookahead().type == ':'))
|
|
|
|
{
|
2023-03-17 14:59:30 +00:00
|
|
|
AstType* type = parseType();
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
// array-like table type: {T} desugars into {[number]: T}
|
2023-04-21 22:41:03 +01:00
|
|
|
AstType* index = allocator.alloc<AstTypeReference>(type->location, std::nullopt, nameNumber, std::nullopt, type->location);
|
2024-01-19 15:13:08 +00:00
|
|
|
indexer = allocator.alloc<AstTableIndexer>(AstTableIndexer{index, type, type->location, access, accessLocation});
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2022-02-18 00:41:20 +00:00
|
|
|
std::optional<Name> name = parseNameOpt("table field");
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
if (!name)
|
|
|
|
break;
|
|
|
|
|
|
|
|
expectAndConsume(':', "table field");
|
|
|
|
|
2023-10-06 18:31:16 +01:00
|
|
|
AstType* type = parseType(inDeclarationContext);
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2024-01-19 15:13:08 +00:00
|
|
|
props.push_back(AstTableProp{name->name, name->location, type, access, accessLocation});
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if (lexer.current().type == ',' || lexer.current().type == ';')
|
|
|
|
{
|
|
|
|
nextLexeme();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (lexer.current().type != '}')
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Location end = lexer.current().location;
|
|
|
|
|
|
|
|
if (!expectMatchAndConsume('}', matchBrace))
|
|
|
|
end = lexer.previousLocation();
|
|
|
|
|
|
|
|
return allocator.alloc<AstTypeTable>(Location(start, end), copy(props), indexer);
|
|
|
|
}
|
|
|
|
|
2023-03-17 14:59:30 +00:00
|
|
|
// ReturnType ::= Type | `(' TypeList `)'
|
|
|
|
// FunctionType ::= [`<' varlist `>'] `(' [TypeList] `)' `->` ReturnType
|
2024-06-07 18:09:03 +01:00
|
|
|
AstTypeOrPack Parser::parseFunctionType(bool allowPack, const AstArray<AstAttr*>& attributes)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
incrementRecursionCounter("type annotation");
|
|
|
|
|
2022-07-01 00:29:02 +01:00
|
|
|
bool forceFunctionType = lexer.current().type == '<';
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
Lexeme begin = lexer.current();
|
|
|
|
|
2022-01-14 16:06:31 +00:00
|
|
|
auto [generics, genericPacks] = parseGenericTypeList(/* withDefaultValues= */ false);
|
2021-11-12 02:12:39 +00:00
|
|
|
|
|
|
|
Lexeme parameterStart = lexer.current();
|
|
|
|
|
|
|
|
expectAndConsume('(', "function parameters");
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
matchRecoveryStopOnToken[Lexeme::SkinnyArrow]++;
|
|
|
|
|
2023-03-17 14:59:30 +00:00
|
|
|
TempVector<AstType*> params(scratchType);
|
2021-10-29 21:25:12 +01:00
|
|
|
TempVector<std::optional<AstArgumentName>> names(scratchOptArgName);
|
|
|
|
AstTypePack* varargAnnotation = nullptr;
|
|
|
|
|
|
|
|
if (lexer.current().type != ')')
|
|
|
|
varargAnnotation = parseTypeList(params, names);
|
|
|
|
|
2021-11-12 02:12:39 +00:00
|
|
|
expectMatchAndConsume(')', parameterStart, true);
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
matchRecoveryStopOnToken[Lexeme::SkinnyArrow]--;
|
|
|
|
|
2021-11-05 02:07:18 +00:00
|
|
|
AstArray<AstType*> paramTypes = copy(params);
|
|
|
|
|
2023-01-13 20:36:28 +00:00
|
|
|
if (!names.empty())
|
2022-07-01 00:29:02 +01:00
|
|
|
forceFunctionType = true;
|
|
|
|
|
2022-07-08 02:05:31 +01:00
|
|
|
bool returnTypeIntroducer = lexer.current().type == Lexeme::SkinnyArrow || lexer.current().type == ':';
|
2022-06-17 01:52:23 +01:00
|
|
|
|
2021-11-05 02:07:18 +00:00
|
|
|
// Not a function at all. Just a parenthesized type. Or maybe a type pack with a single element
|
2022-07-08 02:05:31 +01:00
|
|
|
if (params.size() == 1 && !varargAnnotation && !forceFunctionType && !returnTypeIntroducer)
|
2021-11-05 02:07:18 +00:00
|
|
|
{
|
|
|
|
if (allowPack)
|
|
|
|
return {{}, allocator.alloc<AstTypePackExplicit>(begin.location, AstTypeList{paramTypes, nullptr})};
|
|
|
|
else
|
|
|
|
return {params[0], {}};
|
|
|
|
}
|
|
|
|
|
2022-07-08 02:05:31 +01:00
|
|
|
if (!forceFunctionType && !returnTypeIntroducer && allowPack)
|
2021-11-05 02:07:18 +00:00
|
|
|
return {{}, allocator.alloc<AstTypePackExplicit>(begin.location, AstTypeList{paramTypes, varargAnnotation})};
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
AstArray<std::optional<AstArgumentName>> paramNames = copy(names);
|
|
|
|
|
2024-06-07 18:09:03 +01:00
|
|
|
return {parseFunctionTypeTail(begin, attributes, generics, genericPacks, paramTypes, paramNames, varargAnnotation), {}};
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
2024-06-07 18:09:03 +01:00
|
|
|
AstType* Parser::parseFunctionTypeTail(const Lexeme& begin, const AstArray<AstAttr*>& attributes, AstArray<AstGenericType> generics,
|
|
|
|
AstArray<AstGenericTypePack> genericPacks, AstArray<AstType*> params, AstArray<std::optional<AstArgumentName>> paramNames,
|
|
|
|
AstTypePack* varargAnnotation)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
incrementRecursionCounter("type annotation");
|
|
|
|
|
2022-07-08 02:05:31 +01:00
|
|
|
if (lexer.current().type == ':')
|
2022-06-17 01:52:23 +01:00
|
|
|
{
|
|
|
|
report(lexer.current().location, "Return types in function type annotations are written after '->' instead of ':'");
|
|
|
|
lexer.next();
|
|
|
|
}
|
2021-10-29 21:25:12 +01:00
|
|
|
// Users occasionally write '()' as the 'unit' type when they actually want to use 'nil', here we'll try to give a more specific error
|
2022-06-17 01:52:23 +01:00
|
|
|
else if (lexer.current().type != Lexeme::SkinnyArrow && generics.size == 0 && genericPacks.size == 0 && params.size == 0)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
report(Location(begin.location, lexer.previousLocation()), "Expected '->' after '()' when parsing function type; did you mean 'nil'?");
|
|
|
|
|
2023-04-21 22:41:03 +01:00
|
|
|
return allocator.alloc<AstTypeReference>(begin.location, std::nullopt, nameNil, std::nullopt, begin.location);
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
expectAndConsume(Lexeme::SkinnyArrow, "function type");
|
|
|
|
}
|
|
|
|
|
2023-03-17 14:59:30 +00:00
|
|
|
auto [endLocation, returnTypeList] = parseReturnType();
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
AstTypeList paramTypes = AstTypeList{params, varargAnnotation};
|
2023-10-06 18:31:16 +01:00
|
|
|
return allocator.alloc<AstTypeFunction>(
|
2024-06-07 18:09:03 +01:00
|
|
|
Location(begin.location, endLocation), attributes, generics, genericPacks, paramTypes, paramNames, returnTypeList);
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
2023-03-17 14:59:30 +00:00
|
|
|
// Type ::=
|
2021-10-29 21:25:12 +01:00
|
|
|
// nil |
|
|
|
|
// Name[`.' Name] [`<' namelist `>'] |
|
|
|
|
// `{' [PropList] `}' |
|
|
|
|
// `(' [TypeList] `)' `->` ReturnType
|
2023-03-17 14:59:30 +00:00
|
|
|
// `typeof` Type
|
|
|
|
AstType* Parser::parseTypeSuffix(AstType* type, const Location& begin)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2023-03-17 14:59:30 +00:00
|
|
|
TempVector<AstType*> parts(scratchType);
|
2024-06-07 18:09:03 +01:00
|
|
|
|
2024-06-14 17:38:56 +01:00
|
|
|
if (!FFlag::LuauLeadingBarAndAmpersand2 || type != nullptr)
|
2024-06-07 18:09:03 +01:00
|
|
|
{
|
|
|
|
parts.push_back(type);
|
|
|
|
}
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
incrementRecursionCounter("type annotation");
|
|
|
|
|
|
|
|
bool isUnion = false;
|
|
|
|
bool isIntersection = false;
|
2023-10-27 20:33:36 +01:00
|
|
|
bool hasOptional = false;
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
Location location = begin;
|
|
|
|
|
|
|
|
while (true)
|
|
|
|
{
|
|
|
|
Lexeme::Type c = lexer.current().type;
|
|
|
|
if (c == '|')
|
|
|
|
{
|
|
|
|
nextLexeme();
|
2023-10-27 20:33:36 +01:00
|
|
|
|
|
|
|
unsigned int oldRecursionCount = recursionCounter;
|
2023-03-17 14:59:30 +00:00
|
|
|
parts.push_back(parseSimpleType(/* allowPack= */ false).type);
|
2023-11-10 18:05:48 +00:00
|
|
|
recursionCounter = oldRecursionCount;
|
2023-10-27 20:33:36 +01:00
|
|
|
|
2021-10-29 21:25:12 +01:00
|
|
|
isUnion = true;
|
|
|
|
}
|
|
|
|
else if (c == '?')
|
|
|
|
{
|
2024-06-14 17:38:56 +01:00
|
|
|
LUAU_ASSERT(parts.size() >= 1);
|
|
|
|
|
2021-10-29 21:25:12 +01:00
|
|
|
Location loc = lexer.current().location;
|
|
|
|
nextLexeme();
|
2023-10-27 20:33:36 +01:00
|
|
|
|
2023-11-10 18:05:48 +00:00
|
|
|
if (!hasOptional)
|
2023-10-27 20:33:36 +01:00
|
|
|
parts.push_back(allocator.alloc<AstTypeReference>(loc, std::nullopt, nameNil, std::nullopt, loc));
|
|
|
|
|
2021-10-29 21:25:12 +01:00
|
|
|
isUnion = true;
|
2023-10-27 20:33:36 +01:00
|
|
|
hasOptional = true;
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
else if (c == '&')
|
|
|
|
{
|
|
|
|
nextLexeme();
|
2023-10-27 20:33:36 +01:00
|
|
|
|
|
|
|
unsigned int oldRecursionCount = recursionCounter;
|
2023-03-17 14:59:30 +00:00
|
|
|
parts.push_back(parseSimpleType(/* allowPack= */ false).type);
|
2023-11-10 18:05:48 +00:00
|
|
|
recursionCounter = oldRecursionCount;
|
2023-10-27 20:33:36 +01:00
|
|
|
|
2021-10-29 21:25:12 +01:00
|
|
|
isIntersection = true;
|
|
|
|
}
|
2022-05-06 00:52:48 +01:00
|
|
|
else if (c == Lexeme::Dot3)
|
2022-04-07 21:53:47 +01:00
|
|
|
{
|
|
|
|
report(lexer.current().location, "Unexpected '...' after type annotation");
|
|
|
|
nextLexeme();
|
|
|
|
}
|
2021-10-29 21:25:12 +01:00
|
|
|
else
|
|
|
|
break;
|
2023-10-27 20:33:36 +01:00
|
|
|
|
2023-11-10 18:05:48 +00:00
|
|
|
if (parts.size() > unsigned(FInt::LuauTypeLengthLimit) + hasOptional)
|
2023-10-27 20:33:36 +01:00
|
|
|
ParseError::raise(parts.back()->location, "Exceeded allowed type length; simplify your type annotation to make the code compile");
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if (parts.size() == 1)
|
2024-06-14 17:38:56 +01:00
|
|
|
return FFlag::LuauLeadingBarAndAmpersand2 ? parts[0] : type;
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
if (isUnion && isIntersection)
|
|
|
|
{
|
2023-03-17 14:59:30 +00:00
|
|
|
return reportTypeError(Location(begin, parts.back()->location), copy(parts),
|
2021-10-29 21:25:12 +01:00
|
|
|
"Mixing union and intersection types is not allowed; consider wrapping in parentheses.");
|
|
|
|
}
|
|
|
|
|
|
|
|
location.end = parts.back()->location.end;
|
|
|
|
|
|
|
|
if (isUnion)
|
|
|
|
return allocator.alloc<AstTypeUnion>(location, copy(parts));
|
|
|
|
|
|
|
|
if (isIntersection)
|
|
|
|
return allocator.alloc<AstTypeIntersection>(location, copy(parts));
|
|
|
|
|
|
|
|
LUAU_ASSERT(false);
|
|
|
|
ParseError::raise(begin, "Composite type was not an intersection or union.");
|
|
|
|
}
|
|
|
|
|
2023-03-17 14:59:30 +00:00
|
|
|
AstTypeOrPack Parser::parseTypeOrPack()
|
2021-11-05 02:07:18 +00:00
|
|
|
{
|
|
|
|
unsigned int oldRecursionCount = recursionCounter;
|
2023-10-27 20:33:36 +01:00
|
|
|
// recursion counter is incremented in parseSimpleType
|
2021-11-05 02:07:18 +00:00
|
|
|
|
|
|
|
Location begin = lexer.current().location;
|
|
|
|
|
2023-03-17 14:59:30 +00:00
|
|
|
auto [type, typePack] = parseSimpleType(/* allowPack= */ true);
|
2021-11-05 02:07:18 +00:00
|
|
|
|
|
|
|
if (typePack)
|
|
|
|
{
|
|
|
|
LUAU_ASSERT(!type);
|
|
|
|
return {{}, typePack};
|
|
|
|
}
|
|
|
|
|
|
|
|
recursionCounter = oldRecursionCount;
|
|
|
|
|
2023-03-17 14:59:30 +00:00
|
|
|
return {parseTypeSuffix(type, begin), {}};
|
2021-11-05 02:07:18 +00:00
|
|
|
}
|
|
|
|
|
2023-10-06 18:31:16 +01:00
|
|
|
AstType* Parser::parseType(bool inDeclarationContext)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
unsigned int oldRecursionCount = recursionCounter;
|
2024-06-07 18:09:03 +01:00
|
|
|
// recursion counter is incremented in parseSimpleType and/or parseTypeSuffix
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
Location begin = lexer.current().location;
|
|
|
|
|
2024-06-14 17:38:56 +01:00
|
|
|
if (FFlag::LuauLeadingBarAndAmpersand2)
|
2024-06-07 18:09:03 +01:00
|
|
|
{
|
|
|
|
AstType* type = nullptr;
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2024-06-07 18:09:03 +01:00
|
|
|
Lexeme::Type c = lexer.current().type;
|
|
|
|
if (c != '|' && c != '&')
|
|
|
|
{
|
|
|
|
type = parseSimpleType(/* allowPack= */ false, /* in declaration context */ inDeclarationContext).type;
|
|
|
|
recursionCounter = oldRecursionCount;
|
|
|
|
}
|
|
|
|
|
|
|
|
AstType* typeWithSuffix = parseTypeSuffix(type, begin);
|
|
|
|
recursionCounter = oldRecursionCount;
|
|
|
|
|
|
|
|
return typeWithSuffix;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
AstType* type = parseSimpleType(/* allowPack= */ false, /* in declaration context */ inDeclarationContext).type;
|
|
|
|
|
|
|
|
recursionCounter = oldRecursionCount;
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2024-06-07 18:09:03 +01:00
|
|
|
return parseTypeSuffix(type, begin);
|
|
|
|
}
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
2023-03-17 14:59:30 +00:00
|
|
|
// Type ::= nil | Name[`.' Name] [ `<' Type [`,' ...] `>' ] | `typeof' `(' expr `)' | `{' [PropList] `}'
|
2021-10-29 21:25:12 +01:00
|
|
|
// | [`<' varlist `>'] `(' [TypeList] `)' `->` ReturnType
|
2023-10-06 18:31:16 +01:00
|
|
|
AstTypeOrPack Parser::parseSimpleType(bool allowPack, bool inDeclarationContext)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
incrementRecursionCounter("type annotation");
|
|
|
|
|
2022-09-02 00:00:14 +01:00
|
|
|
Location start = lexer.current().location;
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2024-06-07 18:09:03 +01:00
|
|
|
AstArray<AstAttr*> attributes{nullptr, 0};
|
|
|
|
|
|
|
|
if (lexer.current().type == Lexeme::Attribute)
|
|
|
|
{
|
|
|
|
if (!inDeclarationContext || !FFlag::LuauAttributeSyntax)
|
|
|
|
{
|
|
|
|
return {reportTypeError(start, {}, "attributes are not allowed in declaration context")};
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
attributes = Parser::parseAttributes();
|
|
|
|
return parseFunctionType(allowPack, attributes);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (lexer.current().type == Lexeme::ReservedNil)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
nextLexeme();
|
2023-04-21 22:41:03 +01:00
|
|
|
return {allocator.alloc<AstTypeReference>(start, std::nullopt, nameNil, std::nullopt, start), {}};
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
2022-03-24 21:49:08 +00:00
|
|
|
else if (lexer.current().type == Lexeme::ReservedTrue)
|
2021-11-18 22:21:07 +00:00
|
|
|
{
|
|
|
|
nextLexeme();
|
2022-09-02 00:00:14 +01:00
|
|
|
return {allocator.alloc<AstTypeSingletonBool>(start, true)};
|
2021-11-18 22:21:07 +00:00
|
|
|
}
|
2022-03-24 21:49:08 +00:00
|
|
|
else if (lexer.current().type == Lexeme::ReservedFalse)
|
2021-11-18 22:21:07 +00:00
|
|
|
{
|
|
|
|
nextLexeme();
|
2022-09-02 00:00:14 +01:00
|
|
|
return {allocator.alloc<AstTypeSingletonBool>(start, false)};
|
2021-11-18 22:21:07 +00:00
|
|
|
}
|
2022-03-24 21:49:08 +00:00
|
|
|
else if (lexer.current().type == Lexeme::RawString || lexer.current().type == Lexeme::QuotedString)
|
2021-11-18 22:21:07 +00:00
|
|
|
{
|
|
|
|
if (std::optional<AstArray<char>> value = parseCharArray())
|
|
|
|
{
|
|
|
|
AstArray<char> svalue = *value;
|
2022-09-02 00:00:14 +01:00
|
|
|
return {allocator.alloc<AstTypeSingletonString>(start, svalue)};
|
2021-11-18 22:21:07 +00:00
|
|
|
}
|
|
|
|
else
|
2023-03-17 14:59:30 +00:00
|
|
|
return {reportTypeError(start, {}, "String literal contains malformed escape sequence")};
|
2021-11-18 22:21:07 +00:00
|
|
|
}
|
2022-08-25 21:55:08 +01:00
|
|
|
else if (lexer.current().type == Lexeme::InterpStringBegin || lexer.current().type == Lexeme::InterpStringSimple)
|
|
|
|
{
|
|
|
|
parseInterpString();
|
|
|
|
|
2023-03-17 14:59:30 +00:00
|
|
|
return {reportTypeError(start, {}, "Interpolated string literals cannot be used as types")};
|
2022-08-25 21:55:08 +01:00
|
|
|
}
|
2022-03-24 21:49:08 +00:00
|
|
|
else if (lexer.current().type == Lexeme::BrokenString)
|
2021-11-18 22:21:07 +00:00
|
|
|
{
|
|
|
|
nextLexeme();
|
2023-08-25 16:25:09 +01:00
|
|
|
return {reportTypeError(start, {}, "Malformed string; did you forget to finish it?")};
|
2021-11-18 22:21:07 +00:00
|
|
|
}
|
2021-10-29 21:25:12 +01:00
|
|
|
else if (lexer.current().type == Lexeme::Name)
|
|
|
|
{
|
|
|
|
std::optional<AstName> prefix;
|
2023-04-21 22:41:03 +01:00
|
|
|
std::optional<Location> prefixLocation;
|
2021-10-29 21:25:12 +01:00
|
|
|
Name name = parseName("type name");
|
|
|
|
|
|
|
|
if (lexer.current().type == '.')
|
|
|
|
{
|
|
|
|
Position pointPosition = lexer.current().location.begin;
|
|
|
|
nextLexeme();
|
|
|
|
|
|
|
|
prefix = name.name;
|
2023-04-21 22:41:03 +01:00
|
|
|
prefixLocation = name.location;
|
2021-10-29 21:25:12 +01:00
|
|
|
name = parseIndexName("field name", pointPosition);
|
|
|
|
}
|
2022-05-06 00:52:48 +01:00
|
|
|
else if (lexer.current().type == Lexeme::Dot3)
|
2022-04-07 21:53:47 +01:00
|
|
|
{
|
|
|
|
report(lexer.current().location, "Unexpected '...' after type name; type pack is not allowed in this context");
|
|
|
|
nextLexeme();
|
|
|
|
}
|
2021-10-29 21:25:12 +01:00
|
|
|
else if (name.name == "typeof")
|
|
|
|
{
|
|
|
|
Lexeme typeofBegin = lexer.current();
|
|
|
|
expectAndConsume('(', "typeof type");
|
|
|
|
|
|
|
|
AstExpr* expr = parseExpr();
|
|
|
|
|
|
|
|
Location end = lexer.current().location;
|
|
|
|
|
|
|
|
expectMatchAndConsume(')', typeofBegin);
|
|
|
|
|
2022-09-02 00:00:14 +01:00
|
|
|
return {allocator.alloc<AstTypeTypeof>(Location(start, end), expr), {}};
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
2021-12-02 23:20:08 +00:00
|
|
|
bool hasParameters = false;
|
|
|
|
AstArray<AstTypeOrPack> parameters{};
|
2021-11-05 02:07:18 +00:00
|
|
|
|
2021-12-02 23:20:08 +00:00
|
|
|
if (lexer.current().type == '<')
|
2021-11-05 02:07:18 +00:00
|
|
|
{
|
2021-12-02 23:20:08 +00:00
|
|
|
hasParameters = true;
|
|
|
|
parameters = parseTypeParams();
|
|
|
|
}
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2021-12-02 23:20:08 +00:00
|
|
|
Location end = lexer.previousLocation();
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2023-04-21 22:41:03 +01:00
|
|
|
return {
|
|
|
|
allocator.alloc<AstTypeReference>(Location(start, end), prefix, name.name, prefixLocation, name.location, hasParameters, parameters), {}};
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
else if (lexer.current().type == '{')
|
|
|
|
{
|
2023-10-06 18:31:16 +01:00
|
|
|
return {parseTableType(/* inDeclarationContext */ inDeclarationContext), {}};
|
|
|
|
}
|
2021-11-12 02:12:39 +00:00
|
|
|
else if (lexer.current().type == '(' || lexer.current().type == '<')
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2024-06-07 18:09:03 +01:00
|
|
|
return parseFunctionType(allowPack, AstArray<AstAttr*>({nullptr, 0}));
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
2022-08-18 22:04:33 +01:00
|
|
|
else if (lexer.current().type == Lexeme::ReservedFunction)
|
2022-05-26 21:33:48 +01:00
|
|
|
{
|
|
|
|
nextLexeme();
|
|
|
|
|
2023-03-17 14:59:30 +00:00
|
|
|
return {reportTypeError(start, {},
|
2022-05-26 21:33:48 +01:00
|
|
|
"Using 'function' as a type annotation is not supported, consider replacing with a function type annotation e.g. '(...any) -> "
|
|
|
|
"...any'"),
|
|
|
|
{}};
|
|
|
|
}
|
2021-10-29 21:25:12 +01:00
|
|
|
else
|
|
|
|
{
|
2022-10-27 23:22:49 +01:00
|
|
|
// For a missing type annotation, capture 'space' between last token and the next one
|
|
|
|
Location astErrorlocation(lexer.previousLocation().end, start.begin);
|
|
|
|
// The parse error includes the next lexeme to make it easier to display where the error is (e.g. in an IDE or a CLI error message).
|
|
|
|
// Including the current lexeme also makes the parse error consistent with other parse errors returned by Luau.
|
|
|
|
Location parseErrorLocation(lexer.previousLocation().end, start.end);
|
2023-03-17 14:59:30 +00:00
|
|
|
return {reportMissingTypeError(parseErrorLocation, astErrorlocation, "Expected type, got %s", lexer.current().toString().c_str()), {}};
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-17 14:59:30 +00:00
|
|
|
AstTypePack* Parser::parseVariadicArgumentTypePack()
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
// Generic: a...
|
|
|
|
if (lexer.current().type == Lexeme::Name && lexer.lookahead().type == Lexeme::Dot3)
|
|
|
|
{
|
|
|
|
Name name = parseName("generic name");
|
|
|
|
Location end = lexer.current().location;
|
|
|
|
|
|
|
|
// This will not fail because of the lookahead guard.
|
|
|
|
expectAndConsume(Lexeme::Dot3, "generic type pack annotation");
|
|
|
|
return allocator.alloc<AstTypePackGeneric>(Location(name.location, end), name.name);
|
|
|
|
}
|
|
|
|
// Variadic: T
|
|
|
|
else
|
|
|
|
{
|
2023-03-17 14:59:30 +00:00
|
|
|
AstType* variadicAnnotation = parseType();
|
2021-10-29 21:25:12 +01:00
|
|
|
return allocator.alloc<AstTypePackVariadic>(variadicAnnotation->location, variadicAnnotation);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-17 14:59:30 +00:00
|
|
|
AstTypePack* Parser::parseTypePack()
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
// Variadic: ...T
|
|
|
|
if (lexer.current().type == Lexeme::Dot3)
|
|
|
|
{
|
|
|
|
Location start = lexer.current().location;
|
|
|
|
nextLexeme();
|
2023-03-17 14:59:30 +00:00
|
|
|
AstType* varargTy = parseType();
|
2021-10-29 21:25:12 +01:00
|
|
|
return allocator.alloc<AstTypePackVariadic>(Location(start, varargTy->location), varargTy);
|
|
|
|
}
|
|
|
|
// Generic: a...
|
|
|
|
else if (lexer.current().type == Lexeme::Name && lexer.lookahead().type == Lexeme::Dot3)
|
|
|
|
{
|
|
|
|
Name name = parseName("generic name");
|
|
|
|
Location end = lexer.current().location;
|
|
|
|
|
|
|
|
// This will not fail because of the lookahead guard.
|
|
|
|
expectAndConsume(Lexeme::Dot3, "generic type pack annotation");
|
|
|
|
return allocator.alloc<AstTypePackGeneric>(Location(name.location, end), name.name);
|
|
|
|
}
|
|
|
|
|
2023-08-25 16:25:09 +01:00
|
|
|
// TODO: shouldParseTypePack can be removed and parseTypePack can be called unconditionally instead
|
|
|
|
LUAU_ASSERT(!"parseTypePack can't be called if shouldParseTypePack() returned false");
|
2021-10-29 21:25:12 +01:00
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::optional<AstExprUnary::Op> Parser::parseUnaryOp(const Lexeme& l)
|
|
|
|
{
|
|
|
|
if (l.type == Lexeme::ReservedNot)
|
|
|
|
return AstExprUnary::Not;
|
|
|
|
else if (l.type == '-')
|
|
|
|
return AstExprUnary::Minus;
|
|
|
|
else if (l.type == '#')
|
|
|
|
return AstExprUnary::Len;
|
|
|
|
else
|
|
|
|
return std::nullopt;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::optional<AstExprBinary::Op> Parser::parseBinaryOp(const Lexeme& l)
|
|
|
|
{
|
|
|
|
if (l.type == '+')
|
|
|
|
return AstExprBinary::Add;
|
|
|
|
else if (l.type == '-')
|
|
|
|
return AstExprBinary::Sub;
|
|
|
|
else if (l.type == '*')
|
|
|
|
return AstExprBinary::Mul;
|
|
|
|
else if (l.type == '/')
|
|
|
|
return AstExprBinary::Div;
|
2023-09-01 17:38:53 +01:00
|
|
|
else if (l.type == Lexeme::FloorDiv)
|
|
|
|
return AstExprBinary::FloorDiv;
|
2021-10-29 21:25:12 +01:00
|
|
|
else if (l.type == '%')
|
|
|
|
return AstExprBinary::Mod;
|
|
|
|
else if (l.type == '^')
|
|
|
|
return AstExprBinary::Pow;
|
|
|
|
else if (l.type == Lexeme::Dot2)
|
|
|
|
return AstExprBinary::Concat;
|
|
|
|
else if (l.type == Lexeme::NotEqual)
|
|
|
|
return AstExprBinary::CompareNe;
|
|
|
|
else if (l.type == Lexeme::Equal)
|
|
|
|
return AstExprBinary::CompareEq;
|
|
|
|
else if (l.type == '<')
|
|
|
|
return AstExprBinary::CompareLt;
|
|
|
|
else if (l.type == Lexeme::LessEqual)
|
|
|
|
return AstExprBinary::CompareLe;
|
|
|
|
else if (l.type == '>')
|
|
|
|
return AstExprBinary::CompareGt;
|
|
|
|
else if (l.type == Lexeme::GreaterEqual)
|
|
|
|
return AstExprBinary::CompareGe;
|
|
|
|
else if (l.type == Lexeme::ReservedAnd)
|
|
|
|
return AstExprBinary::And;
|
|
|
|
else if (l.type == Lexeme::ReservedOr)
|
|
|
|
return AstExprBinary::Or;
|
|
|
|
else
|
|
|
|
return std::nullopt;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::optional<AstExprBinary::Op> Parser::parseCompoundOp(const Lexeme& l)
|
|
|
|
{
|
|
|
|
if (l.type == Lexeme::AddAssign)
|
|
|
|
return AstExprBinary::Add;
|
|
|
|
else if (l.type == Lexeme::SubAssign)
|
|
|
|
return AstExprBinary::Sub;
|
|
|
|
else if (l.type == Lexeme::MulAssign)
|
|
|
|
return AstExprBinary::Mul;
|
|
|
|
else if (l.type == Lexeme::DivAssign)
|
|
|
|
return AstExprBinary::Div;
|
2023-09-01 17:38:53 +01:00
|
|
|
else if (l.type == Lexeme::FloorDivAssign)
|
|
|
|
return AstExprBinary::FloorDiv;
|
2021-10-29 21:25:12 +01:00
|
|
|
else if (l.type == Lexeme::ModAssign)
|
|
|
|
return AstExprBinary::Mod;
|
|
|
|
else if (l.type == Lexeme::PowAssign)
|
|
|
|
return AstExprBinary::Pow;
|
|
|
|
else if (l.type == Lexeme::ConcatAssign)
|
|
|
|
return AstExprBinary::Concat;
|
|
|
|
else
|
|
|
|
return std::nullopt;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::optional<AstExprUnary::Op> Parser::checkUnaryConfusables()
|
|
|
|
{
|
|
|
|
const Lexeme& curr = lexer.current();
|
|
|
|
|
|
|
|
// early-out: need to check if this is a possible confusable quickly
|
|
|
|
if (curr.type != '!')
|
|
|
|
return {};
|
|
|
|
|
|
|
|
// slow path: possible confusable
|
|
|
|
Location start = curr.location;
|
|
|
|
|
|
|
|
if (curr.type == '!')
|
|
|
|
{
|
2023-08-25 16:25:09 +01:00
|
|
|
report(start, "Unexpected '!'; did you mean 'not'?");
|
2021-10-29 21:25:12 +01:00
|
|
|
return AstExprUnary::Not;
|
|
|
|
}
|
|
|
|
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
|
|
|
std::optional<AstExprBinary::Op> Parser::checkBinaryConfusables(const BinaryOpPriority binaryPriority[], unsigned int limit)
|
|
|
|
{
|
|
|
|
const Lexeme& curr = lexer.current();
|
|
|
|
|
|
|
|
// early-out: need to check if this is a possible confusable quickly
|
|
|
|
if (curr.type != '&' && curr.type != '|' && curr.type != '!')
|
|
|
|
return {};
|
|
|
|
|
|
|
|
// slow path: possible confusable
|
|
|
|
Location start = curr.location;
|
|
|
|
Lexeme next = lexer.lookahead();
|
|
|
|
|
|
|
|
if (curr.type == '&' && next.type == '&' && curr.location.end == next.location.begin && binaryPriority[AstExprBinary::And].left > limit)
|
|
|
|
{
|
|
|
|
nextLexeme();
|
2023-08-25 16:25:09 +01:00
|
|
|
report(Location(start, next.location), "Unexpected '&&'; did you mean 'and'?");
|
2021-10-29 21:25:12 +01:00
|
|
|
return AstExprBinary::And;
|
|
|
|
}
|
|
|
|
else if (curr.type == '|' && next.type == '|' && curr.location.end == next.location.begin && binaryPriority[AstExprBinary::Or].left > limit)
|
|
|
|
{
|
|
|
|
nextLexeme();
|
2023-08-25 16:25:09 +01:00
|
|
|
report(Location(start, next.location), "Unexpected '||'; did you mean 'or'?");
|
2021-10-29 21:25:12 +01:00
|
|
|
return AstExprBinary::Or;
|
|
|
|
}
|
|
|
|
else if (curr.type == '!' && next.type == '=' && curr.location.end == next.location.begin &&
|
|
|
|
binaryPriority[AstExprBinary::CompareNe].left > limit)
|
|
|
|
{
|
|
|
|
nextLexeme();
|
2023-08-25 16:25:09 +01:00
|
|
|
report(Location(start, next.location), "Unexpected '!='; did you mean '~='?");
|
2021-10-29 21:25:12 +01:00
|
|
|
return AstExprBinary::CompareNe;
|
|
|
|
}
|
|
|
|
|
|
|
|
return std::nullopt;
|
|
|
|
}
|
|
|
|
|
|
|
|
// subexpr -> (asexp | unop subexpr) { binop subexpr }
|
|
|
|
// where `binop' is any binary operator with a priority higher than `limit'
|
|
|
|
AstExpr* Parser::parseExpr(unsigned int limit)
|
|
|
|
{
|
|
|
|
static const BinaryOpPriority binaryPriority[] = {
|
2023-09-01 17:38:53 +01:00
|
|
|
{6, 6}, {6, 6}, {7, 7}, {7, 7}, {7, 7}, {7, 7}, // `+' `-' `*' `/' `//' `%'
|
|
|
|
{10, 9}, {5, 4}, // power and concat (right associative)
|
|
|
|
{3, 3}, {3, 3}, // equality and inequality
|
|
|
|
{3, 3}, {3, 3}, {3, 3}, {3, 3}, // order
|
|
|
|
{2, 2}, {1, 1} // logical (and/or)
|
2021-10-29 21:25:12 +01:00
|
|
|
};
|
2023-09-01 17:38:53 +01:00
|
|
|
static_assert(sizeof(binaryPriority) / sizeof(binaryPriority[0]) == size_t(AstExprBinary::Op__Count), "binaryPriority needs an entry per op");
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2023-10-27 20:33:36 +01:00
|
|
|
unsigned int oldRecursionCount = recursionCounter;
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
// this handles recursive calls to parseSubExpr/parseExpr
|
|
|
|
incrementRecursionCounter("expression");
|
|
|
|
|
|
|
|
const unsigned int unaryPriority = 8;
|
|
|
|
|
|
|
|
Location start = lexer.current().location;
|
|
|
|
|
|
|
|
AstExpr* expr;
|
|
|
|
|
|
|
|
std::optional<AstExprUnary::Op> uop = parseUnaryOp(lexer.current());
|
|
|
|
|
|
|
|
if (!uop)
|
|
|
|
uop = checkUnaryConfusables();
|
|
|
|
|
|
|
|
if (uop)
|
|
|
|
{
|
|
|
|
nextLexeme();
|
|
|
|
|
|
|
|
AstExpr* subexpr = parseExpr(unaryPriority);
|
|
|
|
|
|
|
|
expr = allocator.alloc<AstExprUnary>(Location(start, subexpr->location), *uop, subexpr);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
expr = parseAssertionExpr();
|
|
|
|
}
|
|
|
|
|
|
|
|
// expand while operators have priorities higher than `limit'
|
|
|
|
std::optional<AstExprBinary::Op> op = parseBinaryOp(lexer.current());
|
|
|
|
|
|
|
|
if (!op)
|
|
|
|
op = checkBinaryConfusables(binaryPriority, limit);
|
|
|
|
|
|
|
|
while (op && binaryPriority[*op].left > limit)
|
|
|
|
{
|
|
|
|
nextLexeme();
|
|
|
|
|
|
|
|
// read sub-expression with higher priority
|
|
|
|
AstExpr* next = parseExpr(binaryPriority[*op].right);
|
|
|
|
|
|
|
|
expr = allocator.alloc<AstExprBinary>(Location(start, next->location), *op, expr, next);
|
|
|
|
op = parseBinaryOp(lexer.current());
|
|
|
|
|
|
|
|
if (!op)
|
|
|
|
op = checkBinaryConfusables(binaryPriority, limit);
|
|
|
|
|
|
|
|
// note: while the parser isn't recursive here, we're generating recursive structures of unbounded depth
|
|
|
|
incrementRecursionCounter("expression");
|
|
|
|
}
|
|
|
|
|
2023-10-27 20:33:36 +01:00
|
|
|
recursionCounter = oldRecursionCount;
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
return expr;
|
|
|
|
}
|
|
|
|
|
|
|
|
// NAME
|
|
|
|
AstExpr* Parser::parseNameExpr(const char* context)
|
|
|
|
{
|
2022-02-18 00:41:20 +00:00
|
|
|
std::optional<Name> name = parseNameOpt(context);
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
if (!name)
|
|
|
|
return allocator.alloc<AstExprError>(lexer.current().location, copy<AstExpr*>({}), unsigned(parseErrors.size() - 1));
|
|
|
|
|
|
|
|
AstLocal* const* value = localMap.find(name->name);
|
|
|
|
|
|
|
|
if (value && *value)
|
|
|
|
{
|
|
|
|
AstLocal* local = *value;
|
|
|
|
|
|
|
|
return allocator.alloc<AstExprLocal>(name->location, local, local->functionDepth != functionStack.size() - 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
return allocator.alloc<AstExprGlobal>(name->location, name->name);
|
|
|
|
}
|
|
|
|
|
|
|
|
// prefixexp -> NAME | '(' expr ')'
|
|
|
|
AstExpr* Parser::parsePrefixExpr()
|
|
|
|
{
|
|
|
|
if (lexer.current().type == '(')
|
|
|
|
{
|
2022-08-18 22:04:33 +01:00
|
|
|
Position start = lexer.current().location.begin;
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
MatchLexeme matchParen = lexer.current();
|
2021-10-29 21:25:12 +01:00
|
|
|
nextLexeme();
|
|
|
|
|
|
|
|
AstExpr* expr = parseExpr();
|
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
Position end = lexer.current().location.end;
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
if (lexer.current().type != ')')
|
|
|
|
{
|
|
|
|
const char* suggestion = (lexer.current().type == '=') ? "; did you mean to use '{' when defining a table?" : nullptr;
|
|
|
|
|
|
|
|
expectMatchAndConsumeFail(static_cast<Lexeme::Type>(')'), matchParen, suggestion);
|
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
end = lexer.previousLocation().end;
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
nextLexeme();
|
|
|
|
}
|
|
|
|
|
|
|
|
return allocator.alloc<AstExprGroup>(Location(start, end), expr);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return parseNameExpr("expression");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// primaryexp -> prefixexp { `.' NAME | `[' exp `]' | `:' NAME funcargs | funcargs }
|
|
|
|
AstExpr* Parser::parsePrimaryExpr(bool asStatement)
|
|
|
|
{
|
2022-08-18 22:04:33 +01:00
|
|
|
Position start = lexer.current().location.begin;
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
AstExpr* expr = parsePrefixExpr();
|
|
|
|
|
2023-10-27 20:33:36 +01:00
|
|
|
unsigned int oldRecursionCount = recursionCounter;
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
while (true)
|
|
|
|
{
|
|
|
|
if (lexer.current().type == '.')
|
|
|
|
{
|
|
|
|
Position opPosition = lexer.current().location.begin;
|
|
|
|
nextLexeme();
|
|
|
|
|
|
|
|
Name index = parseIndexName(nullptr, opPosition);
|
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
expr = allocator.alloc<AstExprIndexName>(Location(start, index.location.end), expr, index.name, index.location, opPosition, '.');
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
else if (lexer.current().type == '[')
|
|
|
|
{
|
2022-08-18 22:04:33 +01:00
|
|
|
MatchLexeme matchBracket = lexer.current();
|
2021-10-29 21:25:12 +01:00
|
|
|
nextLexeme();
|
|
|
|
|
|
|
|
AstExpr* index = parseExpr();
|
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
Position end = lexer.current().location.end;
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
expectMatchAndConsume(']', matchBracket);
|
|
|
|
|
|
|
|
expr = allocator.alloc<AstExprIndexExpr>(Location(start, end), expr, index);
|
|
|
|
}
|
|
|
|
else if (lexer.current().type == ':')
|
|
|
|
{
|
|
|
|
Position opPosition = lexer.current().location.begin;
|
|
|
|
nextLexeme();
|
|
|
|
|
|
|
|
Name index = parseIndexName("method name", opPosition);
|
2022-08-18 22:04:33 +01:00
|
|
|
AstExpr* func = allocator.alloc<AstExprIndexName>(Location(start, index.location.end), expr, index.name, index.location, opPosition, ':');
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
expr = parseFunctionArgs(func, true);
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
else if (lexer.current().type == '(')
|
|
|
|
{
|
|
|
|
// This error is handled inside 'parseFunctionArgs' as well, but for better error recovery we need to break out the current loop here
|
|
|
|
if (!asStatement && expr->location.end.line != lexer.current().location.begin.line)
|
|
|
|
{
|
2022-08-18 22:04:33 +01:00
|
|
|
reportAmbiguousCallError();
|
2021-10-29 21:25:12 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
expr = parseFunctionArgs(expr, false);
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
else if (lexer.current().type == '{' || lexer.current().type == Lexeme::RawString || lexer.current().type == Lexeme::QuotedString)
|
|
|
|
{
|
2022-08-18 22:04:33 +01:00
|
|
|
expr = parseFunctionArgs(expr, false);
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// note: while the parser isn't recursive here, we're generating recursive structures of unbounded depth
|
|
|
|
incrementRecursionCounter("expression");
|
|
|
|
}
|
|
|
|
|
2023-10-27 20:33:36 +01:00
|
|
|
recursionCounter = oldRecursionCount;
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
return expr;
|
|
|
|
}
|
|
|
|
|
2023-03-17 14:59:30 +00:00
|
|
|
// asexp -> simpleexp [`::' Type]
|
2021-10-29 21:25:12 +01:00
|
|
|
AstExpr* Parser::parseAssertionExpr()
|
|
|
|
{
|
|
|
|
Location start = lexer.current().location;
|
|
|
|
AstExpr* expr = parseSimpleExpr();
|
|
|
|
|
2023-05-05 20:57:12 +01:00
|
|
|
if (lexer.current().type == Lexeme::DoubleColon)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
nextLexeme();
|
2023-03-17 14:59:30 +00:00
|
|
|
AstType* annotation = parseType();
|
2021-10-29 21:25:12 +01:00
|
|
|
return allocator.alloc<AstExprTypeAssertion>(Location(start, annotation->location), expr, annotation);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
return expr;
|
|
|
|
}
|
|
|
|
|
2022-08-04 22:27:28 +01:00
|
|
|
static ConstantNumberParseResult parseInteger(double& result, const char* data, int base)
|
|
|
|
{
|
|
|
|
LUAU_ASSERT(base == 2 || base == 16);
|
|
|
|
|
|
|
|
char* end = nullptr;
|
|
|
|
unsigned long long value = strtoull(data, &end, base);
|
|
|
|
|
|
|
|
if (*end != 0)
|
|
|
|
return ConstantNumberParseResult::Malformed;
|
|
|
|
|
|
|
|
result = double(value);
|
|
|
|
|
|
|
|
if (value == ULLONG_MAX && errno == ERANGE)
|
|
|
|
{
|
|
|
|
// 'errno' might have been set before we called 'strtoull', but we don't want the overhead of resetting a TLS variable on each call
|
|
|
|
// so we only reset it when we get a result that might be an out-of-range error and parse again to make sure
|
|
|
|
errno = 0;
|
|
|
|
value = strtoull(data, &end, base);
|
|
|
|
|
|
|
|
if (errno == ERANGE)
|
|
|
|
return base == 2 ? ConstantNumberParseResult::BinOverflow : ConstantNumberParseResult::HexOverflow;
|
|
|
|
}
|
|
|
|
|
2023-12-02 02:04:44 +00:00
|
|
|
if (value >= (1ull << 53) && static_cast<unsigned long long>(result) != value)
|
|
|
|
return ConstantNumberParseResult::Imprecise;
|
2023-11-03 19:47:28 +00:00
|
|
|
|
2022-08-04 22:27:28 +01:00
|
|
|
return ConstantNumberParseResult::Ok;
|
|
|
|
}
|
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
static ConstantNumberParseResult parseDouble(double& result, const char* data)
|
2022-08-04 22:27:28 +01:00
|
|
|
{
|
|
|
|
// binary literal
|
|
|
|
if (data[0] == '0' && (data[1] == 'b' || data[1] == 'B') && data[2])
|
|
|
|
return parseInteger(result, data + 2, 2);
|
|
|
|
|
|
|
|
// hexadecimal literal
|
|
|
|
if (data[0] == '0' && (data[1] == 'x' || data[1] == 'X') && data[2])
|
|
|
|
return parseInteger(result, data, 16); // pass in '0x' prefix, it's handled by 'strtoull'
|
|
|
|
|
|
|
|
char* end = nullptr;
|
|
|
|
double value = strtod(data, &end);
|
|
|
|
|
2023-12-02 02:04:44 +00:00
|
|
|
// trailing non-numeric characters
|
|
|
|
if (*end != 0)
|
|
|
|
return ConstantNumberParseResult::Malformed;
|
2023-11-03 19:47:28 +00:00
|
|
|
|
2023-12-02 02:04:44 +00:00
|
|
|
result = value;
|
2023-11-03 19:47:28 +00:00
|
|
|
|
2023-12-02 02:04:44 +00:00
|
|
|
// for linting, we detect integer constants that are parsed imprecisely
|
|
|
|
// since the check is expensive we only perform it when the number is larger than the precise integer range
|
|
|
|
if (value >= double(1ull << 53) && strspn(data, "0123456789") == strlen(data))
|
2023-11-03 19:47:28 +00:00
|
|
|
{
|
2023-12-02 02:04:44 +00:00
|
|
|
char repr[512];
|
|
|
|
snprintf(repr, sizeof(repr), "%.0f", value);
|
|
|
|
|
|
|
|
if (strcmp(repr, data) != 0)
|
|
|
|
return ConstantNumberParseResult::Imprecise;
|
2023-11-03 19:47:28 +00:00
|
|
|
}
|
2023-12-02 02:04:44 +00:00
|
|
|
|
|
|
|
return ConstantNumberParseResult::Ok;
|
2022-08-04 22:27:28 +01:00
|
|
|
}
|
|
|
|
|
2024-06-14 17:38:56 +01:00
|
|
|
// simpleexp -> NUMBER | STRING | NIL | true | false | ... | constructor | [attributes] FUNCTION body | primaryexp
|
2021-10-29 21:25:12 +01:00
|
|
|
AstExpr* Parser::parseSimpleExpr()
|
|
|
|
{
|
|
|
|
Location start = lexer.current().location;
|
|
|
|
|
2024-06-14 17:38:56 +01:00
|
|
|
AstArray<AstAttr*> attributes{nullptr, 0};
|
|
|
|
|
|
|
|
if (FFlag::LuauAttributeSyntax && FFlag::LuauAttributeSyntaxFunExpr && lexer.current().type == Lexeme::Attribute)
|
|
|
|
{
|
|
|
|
attributes = parseAttributes();
|
|
|
|
|
|
|
|
if (lexer.current().type != Lexeme::ReservedFunction)
|
|
|
|
{
|
|
|
|
return reportExprError(
|
|
|
|
start, {}, "Expected 'function' declaration after attribute, but got %s intead", lexer.current().toString().c_str());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-29 21:25:12 +01:00
|
|
|
if (lexer.current().type == Lexeme::ReservedNil)
|
|
|
|
{
|
|
|
|
nextLexeme();
|
|
|
|
|
|
|
|
return allocator.alloc<AstExprConstantNil>(start);
|
|
|
|
}
|
|
|
|
else if (lexer.current().type == Lexeme::ReservedTrue)
|
|
|
|
{
|
|
|
|
nextLexeme();
|
|
|
|
|
|
|
|
return allocator.alloc<AstExprConstantBool>(start, true);
|
|
|
|
}
|
|
|
|
else if (lexer.current().type == Lexeme::ReservedFalse)
|
|
|
|
{
|
|
|
|
nextLexeme();
|
|
|
|
|
|
|
|
return allocator.alloc<AstExprConstantBool>(start, false);
|
|
|
|
}
|
|
|
|
else if (lexer.current().type == Lexeme::ReservedFunction)
|
|
|
|
{
|
|
|
|
Lexeme matchFunction = lexer.current();
|
|
|
|
nextLexeme();
|
|
|
|
|
2024-06-14 17:38:56 +01:00
|
|
|
return parseFunctionBody(false, matchFunction, AstName(), nullptr, attributes).first;
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
else if (lexer.current().type == Lexeme::Number)
|
|
|
|
{
|
2022-08-18 22:04:33 +01:00
|
|
|
return parseNumber();
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
2022-09-02 00:00:14 +01:00
|
|
|
else if (lexer.current().type == Lexeme::RawString || lexer.current().type == Lexeme::QuotedString ||
|
2023-01-27 21:28:45 +00:00
|
|
|
lexer.current().type == Lexeme::InterpStringSimple)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
return parseString();
|
|
|
|
}
|
2023-01-27 21:28:45 +00:00
|
|
|
else if (lexer.current().type == Lexeme::InterpStringBegin)
|
2022-08-25 21:55:08 +01:00
|
|
|
{
|
|
|
|
return parseInterpString();
|
|
|
|
}
|
2021-10-29 21:25:12 +01:00
|
|
|
else if (lexer.current().type == Lexeme::BrokenString)
|
|
|
|
{
|
|
|
|
nextLexeme();
|
2023-08-25 16:25:09 +01:00
|
|
|
return reportExprError(start, {}, "Malformed string; did you forget to finish it?");
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
2022-08-25 21:55:08 +01:00
|
|
|
else if (lexer.current().type == Lexeme::BrokenInterpDoubleBrace)
|
|
|
|
{
|
|
|
|
nextLexeme();
|
2023-08-25 16:25:09 +01:00
|
|
|
return reportExprError(start, {}, "Double braces are not permitted within interpolated strings; did you mean '\\{'?");
|
2022-08-25 21:55:08 +01:00
|
|
|
}
|
2021-10-29 21:25:12 +01:00
|
|
|
else if (lexer.current().type == Lexeme::Dot3)
|
|
|
|
{
|
|
|
|
if (functionStack.back().vararg)
|
|
|
|
{
|
|
|
|
nextLexeme();
|
|
|
|
|
|
|
|
return allocator.alloc<AstExprVarargs>(start);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
nextLexeme();
|
|
|
|
|
|
|
|
return reportExprError(start, {}, "Cannot use '...' outside of a vararg function");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (lexer.current().type == '{')
|
|
|
|
{
|
|
|
|
return parseTableConstructor();
|
|
|
|
}
|
2022-01-14 16:06:31 +00:00
|
|
|
else if (lexer.current().type == Lexeme::ReservedIf)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
return parseIfElseExpr();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return parsePrimaryExpr(/* asStatement= */ false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// args ::= `(' [explist] `)' | tableconstructor | String
|
2022-08-18 22:04:33 +01:00
|
|
|
AstExpr* Parser::parseFunctionArgs(AstExpr* func, bool self)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
if (lexer.current().type == '(')
|
|
|
|
{
|
|
|
|
Position argStart = lexer.current().location.end;
|
|
|
|
if (func->location.end.line != lexer.current().location.begin.line)
|
2022-08-18 22:04:33 +01:00
|
|
|
reportAmbiguousCallError();
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
MatchLexeme matchParen = lexer.current();
|
2021-10-29 21:25:12 +01:00
|
|
|
nextLexeme();
|
|
|
|
|
|
|
|
TempVector<AstExpr*> args(scratchExpr);
|
|
|
|
|
|
|
|
if (lexer.current().type != ')')
|
|
|
|
parseExprList(args);
|
|
|
|
|
|
|
|
Location end = lexer.current().location;
|
|
|
|
Position argEnd = end.end;
|
|
|
|
|
|
|
|
expectMatchAndConsume(')', matchParen);
|
|
|
|
|
|
|
|
return allocator.alloc<AstExprCall>(Location(func->location, end), func, copy(args), self, Location(argStart, argEnd));
|
|
|
|
}
|
|
|
|
else if (lexer.current().type == '{')
|
|
|
|
{
|
|
|
|
Position argStart = lexer.current().location.end;
|
|
|
|
AstExpr* expr = parseTableConstructor();
|
|
|
|
Position argEnd = lexer.previousLocation().end;
|
|
|
|
|
|
|
|
return allocator.alloc<AstExprCall>(Location(func->location, expr->location), func, copy(&expr, 1), self, Location(argStart, argEnd));
|
|
|
|
}
|
|
|
|
else if (lexer.current().type == Lexeme::RawString || lexer.current().type == Lexeme::QuotedString)
|
|
|
|
{
|
|
|
|
Location argLocation = lexer.current().location;
|
|
|
|
AstExpr* expr = parseString();
|
|
|
|
|
|
|
|
return allocator.alloc<AstExprCall>(Location(func->location, expr->location), func, copy(&expr, 1), self, argLocation);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2022-08-18 22:04:33 +01:00
|
|
|
return reportFunctionArgsError(func, self);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
LUAU_NOINLINE AstExpr* Parser::reportFunctionArgsError(AstExpr* func, bool self)
|
|
|
|
{
|
|
|
|
if (self && lexer.current().location.begin.line != func->location.end.line)
|
|
|
|
{
|
|
|
|
return reportExprError(func->location, copy({func}), "Expected function call arguments after '('");
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return reportExprError(Location(func->location.begin, lexer.current().location.begin), copy({func}),
|
|
|
|
"Expected '(', '{' or <string> when parsing function call, got %s", lexer.current().toString().c_str());
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
LUAU_NOINLINE void Parser::reportAmbiguousCallError()
|
|
|
|
{
|
|
|
|
report(lexer.current().location, "Ambiguous syntax: this looks like an argument list for a function call, but could also be a start of "
|
|
|
|
"new statement; use ';' to separate statements");
|
|
|
|
}
|
|
|
|
|
2021-10-29 21:25:12 +01:00
|
|
|
// tableconstructor ::= `{' [fieldlist] `}'
|
|
|
|
// fieldlist ::= field {fieldsep field} [fieldsep]
|
|
|
|
// field ::= `[' exp `]' `=' exp | Name `=' exp | exp
|
|
|
|
// fieldsep ::= `,' | `;'
|
|
|
|
AstExpr* Parser::parseTableConstructor()
|
|
|
|
{
|
|
|
|
TempVector<AstExprTable::Item> items(scratchItem);
|
|
|
|
|
|
|
|
Location start = lexer.current().location;
|
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
MatchLexeme matchBrace = lexer.current();
|
2021-10-29 21:25:12 +01:00
|
|
|
expectAndConsume('{', "table literal");
|
2022-11-04 17:02:37 +00:00
|
|
|
unsigned lastElementIndent = 0;
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
while (lexer.current().type != '}')
|
|
|
|
{
|
2022-12-02 10:46:05 +00:00
|
|
|
lastElementIndent = lexer.current().location.begin.column;
|
2022-11-04 17:02:37 +00:00
|
|
|
|
2021-10-29 21:25:12 +01:00
|
|
|
if (lexer.current().type == '[')
|
|
|
|
{
|
2022-08-18 22:04:33 +01:00
|
|
|
MatchLexeme matchLocationBracket = lexer.current();
|
2021-10-29 21:25:12 +01:00
|
|
|
nextLexeme();
|
|
|
|
|
|
|
|
AstExpr* key = parseExpr();
|
|
|
|
|
|
|
|
expectMatchAndConsume(']', matchLocationBracket);
|
|
|
|
|
|
|
|
expectAndConsume('=', "table field");
|
|
|
|
|
|
|
|
AstExpr* value = parseExpr();
|
|
|
|
|
|
|
|
items.push_back({AstExprTable::Item::General, key, value});
|
|
|
|
}
|
|
|
|
else if (lexer.current().type == Lexeme::Name && lexer.lookahead().type == '=')
|
|
|
|
{
|
|
|
|
Name name = parseName("table field");
|
|
|
|
|
|
|
|
expectAndConsume('=', "table field");
|
|
|
|
|
|
|
|
AstArray<char> nameString;
|
|
|
|
nameString.data = const_cast<char*>(name.name.value);
|
|
|
|
nameString.size = strlen(name.name.value);
|
|
|
|
|
2023-08-25 16:25:09 +01:00
|
|
|
AstExpr* key = allocator.alloc<AstExprConstantString>(name.location, nameString, AstExprConstantString::Unquoted);
|
2021-10-29 21:25:12 +01:00
|
|
|
AstExpr* value = parseExpr();
|
|
|
|
|
2022-03-24 21:49:08 +00:00
|
|
|
if (AstExprFunction* func = value->as<AstExprFunction>())
|
|
|
|
func->debugname = name.name;
|
2022-02-18 00:41:20 +00:00
|
|
|
|
2021-10-29 21:25:12 +01:00
|
|
|
items.push_back({AstExprTable::Item::Record, key, value});
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
AstExpr* expr = parseExpr();
|
|
|
|
|
|
|
|
items.push_back({AstExprTable::Item::List, nullptr, expr});
|
|
|
|
}
|
|
|
|
|
|
|
|
if (lexer.current().type == ',' || lexer.current().type == ';')
|
|
|
|
{
|
|
|
|
nextLexeme();
|
|
|
|
}
|
2022-12-02 10:46:05 +00:00
|
|
|
else if ((lexer.current().type == '[' || lexer.current().type == Lexeme::Name) && lexer.current().location.begin.column == lastElementIndent)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-11-04 17:02:37 +00:00
|
|
|
report(lexer.current().location, "Expected ',' after table constructor element");
|
|
|
|
}
|
|
|
|
else if (lexer.current().type != '}')
|
|
|
|
{
|
|
|
|
break;
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Location end = lexer.current().location;
|
|
|
|
|
|
|
|
if (!expectMatchAndConsume('}', matchBrace))
|
|
|
|
end = lexer.previousLocation();
|
|
|
|
|
|
|
|
return allocator.alloc<AstExprTable>(Location(start, end), copy(items));
|
|
|
|
}
|
|
|
|
|
|
|
|
AstExpr* Parser::parseIfElseExpr()
|
|
|
|
{
|
|
|
|
bool hasElse = false;
|
|
|
|
Location start = lexer.current().location;
|
|
|
|
|
|
|
|
nextLexeme(); // skip if / elseif
|
|
|
|
|
|
|
|
AstExpr* condition = parseExpr();
|
|
|
|
|
|
|
|
bool hasThen = expectAndConsume(Lexeme::ReservedThen, "if then else expression");
|
|
|
|
|
|
|
|
AstExpr* trueExpr = parseExpr();
|
|
|
|
AstExpr* falseExpr = nullptr;
|
|
|
|
|
|
|
|
if (lexer.current().type == Lexeme::ReservedElseif)
|
|
|
|
{
|
|
|
|
unsigned int oldRecursionCount = recursionCounter;
|
|
|
|
incrementRecursionCounter("expression");
|
|
|
|
hasElse = true;
|
|
|
|
falseExpr = parseIfElseExpr();
|
|
|
|
recursionCounter = oldRecursionCount;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
hasElse = expectAndConsume(Lexeme::ReservedElse, "if then else expression");
|
|
|
|
falseExpr = parseExpr();
|
|
|
|
}
|
|
|
|
|
|
|
|
Location end = falseExpr->location;
|
|
|
|
|
|
|
|
return allocator.alloc<AstExprIfElse>(Location(start, end), condition, hasThen, trueExpr, hasElse, falseExpr);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Name
|
|
|
|
std::optional<Parser::Name> Parser::parseNameOpt(const char* context)
|
|
|
|
{
|
|
|
|
if (lexer.current().type != Lexeme::Name)
|
|
|
|
{
|
|
|
|
reportNameError(context);
|
|
|
|
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
|
|
|
Name result(AstName(lexer.current().name), lexer.current().location);
|
|
|
|
|
|
|
|
nextLexeme();
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
Parser::Name Parser::parseName(const char* context)
|
|
|
|
{
|
2022-02-18 00:41:20 +00:00
|
|
|
if (std::optional<Name> name = parseNameOpt(context))
|
2021-10-29 21:25:12 +01:00
|
|
|
return *name;
|
|
|
|
|
|
|
|
Location location = lexer.current().location;
|
|
|
|
location.end = location.begin;
|
|
|
|
|
|
|
|
return Name(nameError, location);
|
|
|
|
}
|
|
|
|
|
|
|
|
Parser::Name Parser::parseIndexName(const char* context, const Position& previous)
|
|
|
|
{
|
2022-02-18 00:41:20 +00:00
|
|
|
if (std::optional<Name> name = parseNameOpt(context))
|
2021-10-29 21:25:12 +01:00
|
|
|
return *name;
|
|
|
|
|
|
|
|
// If we have a reserved keyword next at the same line, assume it's an incomplete name
|
|
|
|
if (lexer.current().type >= Lexeme::Reserved_BEGIN && lexer.current().type < Lexeme::Reserved_END &&
|
|
|
|
lexer.current().location.begin.line == previous.line)
|
|
|
|
{
|
|
|
|
Name result(AstName(lexer.current().name), lexer.current().location);
|
|
|
|
|
|
|
|
nextLexeme();
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
Location location = lexer.current().location;
|
|
|
|
location.end = location.begin;
|
|
|
|
|
|
|
|
return Name(nameError, location);
|
|
|
|
}
|
|
|
|
|
2022-01-14 16:06:31 +00:00
|
|
|
std::pair<AstArray<AstGenericType>, AstArray<AstGenericTypePack>> Parser::parseGenericTypeList(bool withDefaultValues)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-01-14 16:06:31 +00:00
|
|
|
TempVector<AstGenericType> names{scratchGenericTypes};
|
|
|
|
TempVector<AstGenericTypePack> namePacks{scratchGenericTypePacks};
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
if (lexer.current().type == '<')
|
|
|
|
{
|
|
|
|
Lexeme begin = lexer.current();
|
|
|
|
nextLexeme();
|
|
|
|
|
|
|
|
bool seenPack = false;
|
2022-01-14 16:06:31 +00:00
|
|
|
bool seenDefault = false;
|
|
|
|
|
2021-10-29 21:25:12 +01:00
|
|
|
while (true)
|
|
|
|
{
|
2022-01-14 16:06:31 +00:00
|
|
|
Location nameLocation = lexer.current().location;
|
2021-10-29 21:25:12 +01:00
|
|
|
AstName name = parseName().name;
|
2022-02-24 23:15:41 +00:00
|
|
|
if (lexer.current().type == Lexeme::Dot3 || seenPack)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
seenPack = true;
|
2022-01-14 16:06:31 +00:00
|
|
|
|
2022-02-24 23:15:41 +00:00
|
|
|
if (lexer.current().type != Lexeme::Dot3)
|
2022-01-14 16:06:31 +00:00
|
|
|
report(lexer.current().location, "Generic types come before generic type packs");
|
|
|
|
else
|
|
|
|
nextLexeme();
|
|
|
|
|
|
|
|
if (withDefaultValues && lexer.current().type == '=')
|
|
|
|
{
|
|
|
|
seenDefault = true;
|
|
|
|
nextLexeme();
|
|
|
|
|
2023-03-17 14:59:30 +00:00
|
|
|
if (shouldParseTypePack(lexer))
|
2022-01-14 16:06:31 +00:00
|
|
|
{
|
2023-03-17 14:59:30 +00:00
|
|
|
AstTypePack* typePack = parseTypePack();
|
2022-01-14 16:06:31 +00:00
|
|
|
|
|
|
|
namePacks.push_back({name, nameLocation, typePack});
|
|
|
|
}
|
2023-05-05 20:57:12 +01:00
|
|
|
else
|
2022-11-10 22:04:44 +00:00
|
|
|
{
|
2023-03-17 14:59:30 +00:00
|
|
|
auto [type, typePack] = parseTypeOrPack();
|
2022-11-10 22:04:44 +00:00
|
|
|
|
|
|
|
if (type)
|
|
|
|
report(type->location, "Expected type pack after '=', got type");
|
|
|
|
|
|
|
|
namePacks.push_back({name, nameLocation, typePack});
|
|
|
|
}
|
2022-01-14 16:06:31 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (seenDefault)
|
|
|
|
report(lexer.current().location, "Expected default type pack after type pack name");
|
|
|
|
|
|
|
|
namePacks.push_back({name, nameLocation, nullptr});
|
|
|
|
}
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2022-01-14 16:06:31 +00:00
|
|
|
if (withDefaultValues && lexer.current().type == '=')
|
|
|
|
{
|
|
|
|
seenDefault = true;
|
|
|
|
nextLexeme();
|
|
|
|
|
2023-03-17 14:59:30 +00:00
|
|
|
AstType* defaultType = parseType();
|
2022-01-14 16:06:31 +00:00
|
|
|
|
|
|
|
names.push_back({name, nameLocation, defaultType});
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (seenDefault)
|
|
|
|
report(lexer.current().location, "Expected default type after type name");
|
|
|
|
|
|
|
|
names.push_back({name, nameLocation, nullptr});
|
|
|
|
}
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if (lexer.current().type == ',')
|
2022-10-07 00:55:58 +01:00
|
|
|
{
|
2021-10-29 21:25:12 +01:00
|
|
|
nextLexeme();
|
2022-10-07 00:55:58 +01:00
|
|
|
|
2022-12-02 10:46:05 +00:00
|
|
|
if (lexer.current().type == '>')
|
2022-10-07 00:55:58 +01:00
|
|
|
{
|
|
|
|
report(lexer.current().location, "Expected type after ',' but got '>' instead");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2021-10-29 21:25:12 +01:00
|
|
|
else
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
expectMatchAndConsume('>', begin);
|
|
|
|
}
|
|
|
|
|
2022-01-14 16:06:31 +00:00
|
|
|
AstArray<AstGenericType> generics = copy(names);
|
|
|
|
AstArray<AstGenericTypePack> genericPacks = copy(namePacks);
|
2021-10-29 21:25:12 +01:00
|
|
|
return {generics, genericPacks};
|
|
|
|
}
|
|
|
|
|
2021-11-05 02:07:18 +00:00
|
|
|
AstArray<AstTypeOrPack> Parser::parseTypeParams()
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2023-03-17 14:59:30 +00:00
|
|
|
TempVector<AstTypeOrPack> parameters{scratchTypeOrPack};
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
if (lexer.current().type == '<')
|
|
|
|
{
|
|
|
|
Lexeme begin = lexer.current();
|
|
|
|
nextLexeme();
|
|
|
|
|
|
|
|
while (true)
|
|
|
|
{
|
2023-03-17 14:59:30 +00:00
|
|
|
if (shouldParseTypePack(lexer))
|
2021-12-02 23:20:08 +00:00
|
|
|
{
|
2023-03-17 14:59:30 +00:00
|
|
|
AstTypePack* typePack = parseTypePack();
|
2021-12-02 23:20:08 +00:00
|
|
|
|
|
|
|
parameters.push_back({{}, typePack});
|
|
|
|
}
|
|
|
|
else if (lexer.current().type == '(')
|
2021-11-05 02:07:18 +00:00
|
|
|
{
|
2023-03-17 14:59:30 +00:00
|
|
|
auto [type, typePack] = parseTypeOrPack();
|
2021-12-02 23:20:08 +00:00
|
|
|
|
|
|
|
if (typePack)
|
|
|
|
parameters.push_back({{}, typePack});
|
2021-11-05 02:07:18 +00:00
|
|
|
else
|
2021-12-02 23:20:08 +00:00
|
|
|
parameters.push_back({type, {}});
|
|
|
|
}
|
|
|
|
else if (lexer.current().type == '>' && parameters.empty())
|
|
|
|
{
|
|
|
|
break;
|
2021-11-05 02:07:18 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2023-03-17 14:59:30 +00:00
|
|
|
parameters.push_back({parseType(), {}});
|
2021-11-05 02:07:18 +00:00
|
|
|
}
|
|
|
|
|
2021-10-29 21:25:12 +01:00
|
|
|
if (lexer.current().type == ',')
|
|
|
|
nextLexeme();
|
|
|
|
else
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
expectMatchAndConsume('>', begin);
|
|
|
|
}
|
|
|
|
|
2021-11-05 02:07:18 +00:00
|
|
|
return copy(parameters);
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
2021-11-18 22:21:07 +00:00
|
|
|
std::optional<AstArray<char>> Parser::parseCharArray()
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-09-02 00:00:14 +01:00
|
|
|
LUAU_ASSERT(lexer.current().type == Lexeme::QuotedString || lexer.current().type == Lexeme::RawString ||
|
|
|
|
lexer.current().type == Lexeme::InterpStringSimple);
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2024-06-07 18:09:03 +01:00
|
|
|
scratchData.assign(lexer.current().data, lexer.current().getLength());
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-08-25 21:55:08 +01:00
|
|
|
if (lexer.current().type == Lexeme::QuotedString || lexer.current().type == Lexeme::InterpStringSimple)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
if (!Lexer::fixupQuotedString(scratchData))
|
|
|
|
{
|
|
|
|
nextLexeme();
|
2021-11-18 22:21:07 +00:00
|
|
|
return std::nullopt;
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
Lexer::fixupMultilineString(scratchData);
|
|
|
|
}
|
|
|
|
|
|
|
|
AstArray<char> value = copy(scratchData);
|
|
|
|
nextLexeme();
|
2021-11-18 22:21:07 +00:00
|
|
|
return value;
|
|
|
|
}
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2021-11-18 22:21:07 +00:00
|
|
|
AstExpr* Parser::parseString()
|
|
|
|
{
|
|
|
|
Location location = lexer.current().location;
|
|
|
|
if (std::optional<AstArray<char>> value = parseCharArray())
|
|
|
|
return allocator.alloc<AstExprConstantString>(location, *value);
|
|
|
|
else
|
|
|
|
return reportExprError(location, {}, "String literal contains malformed escape sequence");
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
2022-08-25 21:55:08 +01:00
|
|
|
AstExpr* Parser::parseInterpString()
|
|
|
|
{
|
|
|
|
TempVector<AstArray<char>> strings(scratchString);
|
|
|
|
TempVector<AstExpr*> expressions(scratchExpr);
|
|
|
|
|
|
|
|
Location startLocation = lexer.current().location;
|
2022-11-18 18:45:14 +00:00
|
|
|
Location endLocation;
|
2022-08-25 21:55:08 +01:00
|
|
|
|
2022-09-02 00:00:14 +01:00
|
|
|
do
|
|
|
|
{
|
2022-08-25 21:55:08 +01:00
|
|
|
Lexeme currentLexeme = lexer.current();
|
2022-09-02 00:00:14 +01:00
|
|
|
LUAU_ASSERT(currentLexeme.type == Lexeme::InterpStringBegin || currentLexeme.type == Lexeme::InterpStringMid ||
|
|
|
|
currentLexeme.type == Lexeme::InterpStringEnd || currentLexeme.type == Lexeme::InterpStringSimple);
|
2022-08-25 21:55:08 +01:00
|
|
|
|
2022-11-18 18:45:14 +00:00
|
|
|
endLocation = currentLexeme.location;
|
2022-08-25 21:55:08 +01:00
|
|
|
|
2024-06-07 18:09:03 +01:00
|
|
|
scratchData.assign(currentLexeme.data, currentLexeme.getLength());
|
2022-08-25 21:55:08 +01:00
|
|
|
|
|
|
|
if (!Lexer::fixupQuotedString(scratchData))
|
|
|
|
{
|
|
|
|
nextLexeme();
|
2022-11-18 18:45:14 +00:00
|
|
|
return reportExprError(Location{startLocation, endLocation}, {}, "Interpolated string literal contains malformed escape sequence");
|
2022-08-25 21:55:08 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
AstArray<char> chars = copy(scratchData);
|
|
|
|
|
|
|
|
nextLexeme();
|
|
|
|
|
|
|
|
strings.push_back(chars);
|
|
|
|
|
|
|
|
if (currentLexeme.type == Lexeme::InterpStringEnd || currentLexeme.type == Lexeme::InterpStringSimple)
|
|
|
|
{
|
2022-11-18 18:45:14 +00:00
|
|
|
break;
|
2022-08-25 21:55:08 +01:00
|
|
|
}
|
|
|
|
|
2022-11-18 18:45:14 +00:00
|
|
|
bool errorWhileChecking = false;
|
|
|
|
|
|
|
|
switch (lexer.current().type)
|
|
|
|
{
|
|
|
|
case Lexeme::InterpStringMid:
|
|
|
|
case Lexeme::InterpStringEnd:
|
|
|
|
{
|
|
|
|
errorWhileChecking = true;
|
|
|
|
nextLexeme();
|
|
|
|
expressions.push_back(reportExprError(endLocation, {}, "Malformed interpolated string, expected expression inside '{}'"));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case Lexeme::BrokenString:
|
|
|
|
{
|
|
|
|
errorWhileChecking = true;
|
|
|
|
nextLexeme();
|
2023-08-25 16:25:09 +01:00
|
|
|
expressions.push_back(reportExprError(endLocation, {}, "Malformed interpolated string; did you forget to add a '`'?"));
|
2022-11-18 18:45:14 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
expressions.push_back(parseExpr());
|
|
|
|
}
|
2022-08-25 21:55:08 +01:00
|
|
|
|
2022-11-18 18:45:14 +00:00
|
|
|
if (errorWhileChecking)
|
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
2022-08-25 21:55:08 +01:00
|
|
|
|
|
|
|
switch (lexer.current().type)
|
|
|
|
{
|
|
|
|
case Lexeme::InterpStringBegin:
|
|
|
|
case Lexeme::InterpStringMid:
|
|
|
|
case Lexeme::InterpStringEnd:
|
|
|
|
break;
|
|
|
|
case Lexeme::BrokenInterpDoubleBrace:
|
|
|
|
nextLexeme();
|
2023-08-25 16:25:09 +01:00
|
|
|
return reportExprError(endLocation, {}, "Double braces are not permitted within interpolated strings; did you mean '\\{'?");
|
2022-08-25 21:55:08 +01:00
|
|
|
case Lexeme::BrokenString:
|
|
|
|
nextLexeme();
|
2023-08-25 16:25:09 +01:00
|
|
|
return reportExprError(endLocation, {}, "Malformed interpolated string; did you forget to add a '}'?");
|
2022-08-25 21:55:08 +01:00
|
|
|
default:
|
2022-11-18 18:45:14 +00:00
|
|
|
return reportExprError(endLocation, {}, "Malformed interpolated string, got %s", lexer.current().toString().c_str());
|
2022-08-25 21:55:08 +01:00
|
|
|
}
|
|
|
|
} while (true);
|
2022-11-18 18:45:14 +00:00
|
|
|
|
|
|
|
AstArray<AstArray<char>> stringsArray = copy(strings);
|
|
|
|
AstArray<AstExpr*> expressionsArray = copy(expressions);
|
|
|
|
return allocator.alloc<AstExprInterpString>(Location{startLocation, endLocation}, stringsArray, expressionsArray);
|
2022-08-25 21:55:08 +01:00
|
|
|
}
|
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
AstExpr* Parser::parseNumber()
|
|
|
|
{
|
|
|
|
Location start = lexer.current().location;
|
|
|
|
|
2024-06-07 18:09:03 +01:00
|
|
|
scratchData.assign(lexer.current().data, lexer.current().getLength());
|
2022-08-18 22:04:33 +01:00
|
|
|
|
|
|
|
// Remove all internal _ - they don't hold any meaning and this allows parsing code to just pass the string pointer to strtod et al
|
|
|
|
if (scratchData.find('_') != std::string::npos)
|
|
|
|
{
|
|
|
|
scratchData.erase(std::remove(scratchData.begin(), scratchData.end(), '_'), scratchData.end());
|
|
|
|
}
|
|
|
|
|
2022-09-23 19:32:10 +01:00
|
|
|
double value = 0;
|
|
|
|
ConstantNumberParseResult result = parseDouble(value, scratchData.c_str());
|
|
|
|
nextLexeme();
|
2022-08-18 22:04:33 +01:00
|
|
|
|
2022-09-23 19:32:10 +01:00
|
|
|
if (result == ConstantNumberParseResult::Malformed)
|
|
|
|
return reportExprError(start, {}, "Malformed number");
|
2022-08-18 22:04:33 +01:00
|
|
|
|
2022-09-23 19:32:10 +01:00
|
|
|
return allocator.alloc<AstExprConstantNumber>(start, value, result);
|
2022-08-18 22:04:33 +01:00
|
|
|
}
|
|
|
|
|
2021-10-29 21:25:12 +01:00
|
|
|
AstLocal* Parser::pushLocal(const Binding& binding)
|
|
|
|
{
|
|
|
|
const Name& name = binding.name;
|
|
|
|
AstLocal*& local = localMap[name.name];
|
|
|
|
|
|
|
|
local = allocator.alloc<AstLocal>(
|
|
|
|
name.name, name.location, /* shadow= */ local, functionStack.size() - 1, functionStack.back().loopDepth, binding.annotation);
|
|
|
|
|
|
|
|
localStack.push_back(local);
|
|
|
|
|
|
|
|
return local;
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned int Parser::saveLocals()
|
|
|
|
{
|
|
|
|
return unsigned(localStack.size());
|
|
|
|
}
|
|
|
|
|
|
|
|
void Parser::restoreLocals(unsigned int offset)
|
|
|
|
{
|
|
|
|
for (size_t i = localStack.size(); i > offset; --i)
|
|
|
|
{
|
|
|
|
AstLocal* l = localStack[i - 1];
|
|
|
|
|
|
|
|
localMap[l->name] = l->shadow;
|
|
|
|
}
|
|
|
|
|
|
|
|
localStack.resize(offset);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Parser::expectAndConsume(char value, const char* context)
|
|
|
|
{
|
|
|
|
return expectAndConsume(static_cast<Lexeme::Type>(static_cast<unsigned char>(value)), context);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Parser::expectAndConsume(Lexeme::Type type, const char* context)
|
|
|
|
{
|
|
|
|
if (lexer.current().type != type)
|
|
|
|
{
|
|
|
|
expectAndConsumeFail(type, context);
|
|
|
|
|
|
|
|
// check if this is an extra token and the expected token is next
|
|
|
|
if (lexer.lookahead().type == type)
|
|
|
|
{
|
|
|
|
// skip invalid and consume expected
|
|
|
|
nextLexeme();
|
|
|
|
nextLexeme();
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
nextLexeme();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// LUAU_NOINLINE is used to limit the stack cost of this function due to std::string objects, and to increase caller performance since this code is
|
|
|
|
// cold
|
|
|
|
LUAU_NOINLINE void Parser::expectAndConsumeFail(Lexeme::Type type, const char* context)
|
|
|
|
{
|
|
|
|
std::string typeString = Lexeme(Location(Position(0, 0), 0), type).toString();
|
|
|
|
std::string currLexemeString = lexer.current().toString();
|
|
|
|
|
|
|
|
if (context)
|
|
|
|
report(lexer.current().location, "Expected %s when parsing %s, got %s", typeString.c_str(), context, currLexemeString.c_str());
|
|
|
|
else
|
|
|
|
report(lexer.current().location, "Expected %s, got %s", typeString.c_str(), currLexemeString.c_str());
|
|
|
|
}
|
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
bool Parser::expectMatchAndConsume(char value, const MatchLexeme& begin, bool searchForMissing)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
Lexeme::Type type = static_cast<Lexeme::Type>(static_cast<unsigned char>(value));
|
|
|
|
|
|
|
|
if (lexer.current().type != type)
|
|
|
|
{
|
|
|
|
expectMatchAndConsumeFail(type, begin);
|
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
return expectMatchAndConsumeRecover(value, begin, searchForMissing);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
nextLexeme();
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
LUAU_NOINLINE bool Parser::expectMatchAndConsumeRecover(char value, const MatchLexeme& begin, bool searchForMissing)
|
|
|
|
{
|
|
|
|
Lexeme::Type type = static_cast<Lexeme::Type>(static_cast<unsigned char>(value));
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
if (searchForMissing)
|
|
|
|
{
|
|
|
|
// previous location is taken because 'current' lexeme is already the next token
|
|
|
|
unsigned currentLine = lexer.previousLocation().end.line;
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
// search to the end of the line for expected token
|
|
|
|
// we will also stop if we hit a token that can be handled by parsing function above the current one
|
|
|
|
Lexeme::Type lexemeType = lexer.current().type;
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
while (currentLine == lexer.current().location.begin.line && lexemeType != type && matchRecoveryStopOnToken[lexemeType] == 0)
|
|
|
|
{
|
|
|
|
nextLexeme();
|
|
|
|
lexemeType = lexer.current().type;
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
2022-08-18 22:04:33 +01:00
|
|
|
|
|
|
|
if (lexemeType == type)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-08-18 22:04:33 +01:00
|
|
|
nextLexeme();
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
return true;
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2022-08-18 22:04:33 +01:00
|
|
|
// check if this is an extra token and the expected token is next
|
|
|
|
if (lexer.lookahead().type == type)
|
|
|
|
{
|
|
|
|
// skip invalid and consume expected
|
|
|
|
nextLexeme();
|
|
|
|
nextLexeme();
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
return true;
|
|
|
|
}
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
2022-08-18 22:04:33 +01:00
|
|
|
|
|
|
|
return false;
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// LUAU_NOINLINE is used to limit the stack cost of this function due to std::string objects, and to increase caller performance since this code is
|
|
|
|
// cold
|
2022-08-18 22:04:33 +01:00
|
|
|
LUAU_NOINLINE void Parser::expectMatchAndConsumeFail(Lexeme::Type type, const MatchLexeme& begin, const char* extra)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
std::string typeString = Lexeme(Location(Position(0, 0), 0), type).toString();
|
2022-08-18 22:04:33 +01:00
|
|
|
std::string matchString = Lexeme(Location(Position(0, 0), 0), begin.type).toString();
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
if (lexer.current().location.begin.line == begin.position.line)
|
|
|
|
report(lexer.current().location, "Expected %s (to close %s at column %d), got %s%s", typeString.c_str(), matchString.c_str(),
|
|
|
|
begin.position.column + 1, lexer.current().toString().c_str(), extra ? extra : "");
|
2021-10-29 21:25:12 +01:00
|
|
|
else
|
2022-08-18 22:04:33 +01:00
|
|
|
report(lexer.current().location, "Expected %s (to close %s at line %d), got %s%s", typeString.c_str(), matchString.c_str(),
|
|
|
|
begin.position.line + 1, lexer.current().toString().c_str(), extra ? extra : "");
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
bool Parser::expectMatchEndAndConsume(Lexeme::Type type, const MatchLexeme& begin)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
if (lexer.current().type != type)
|
|
|
|
{
|
|
|
|
expectMatchEndAndConsumeFail(type, begin);
|
|
|
|
|
|
|
|
// check if this is an extra token and the expected token is next
|
|
|
|
if (lexer.lookahead().type == type)
|
|
|
|
{
|
|
|
|
// skip invalid and consume expected
|
|
|
|
nextLexeme();
|
|
|
|
nextLexeme();
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// If the token matches on a different line and a different column, it suggests misleading indentation
|
|
|
|
// This can be used to pinpoint the problem location for a possible future *actual* mismatch
|
2022-09-02 00:00:14 +01:00
|
|
|
if (lexer.current().location.begin.line != begin.position.line && lexer.current().location.begin.column != begin.position.column &&
|
2022-08-18 22:04:33 +01:00
|
|
|
endMismatchSuspect.position.line < begin.position.line) // Only replace the previous suspect with more recent suspects
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
endMismatchSuspect = begin;
|
|
|
|
}
|
|
|
|
|
|
|
|
nextLexeme();
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// LUAU_NOINLINE is used to limit the stack cost of this function due to std::string objects, and to increase caller performance since this code is
|
|
|
|
// cold
|
2022-08-18 22:04:33 +01:00
|
|
|
LUAU_NOINLINE void Parser::expectMatchEndAndConsumeFail(Lexeme::Type type, const MatchLexeme& begin)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-08-18 22:04:33 +01:00
|
|
|
if (endMismatchSuspect.type != Lexeme::Eof && endMismatchSuspect.position.line > begin.position.line)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-08-18 22:04:33 +01:00
|
|
|
std::string matchString = Lexeme(Location(Position(0, 0), 0), endMismatchSuspect.type).toString();
|
|
|
|
std::string suggestion = format("; did you forget to close %s at line %d?", matchString.c_str(), endMismatchSuspect.position.line + 1);
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
expectMatchAndConsumeFail(type, begin, suggestion.c_str());
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
expectMatchAndConsumeFail(type, begin);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
AstArray<T> Parser::copy(const T* data, size_t size)
|
|
|
|
{
|
|
|
|
AstArray<T> result;
|
|
|
|
|
|
|
|
result.data = size ? static_cast<T*>(allocator.allocate(sizeof(T) * size)) : nullptr;
|
|
|
|
result.size = size;
|
|
|
|
|
|
|
|
// This is equivalent to std::uninitialized_copy, but without the exception guarantee
|
|
|
|
// since our types don't have destructors
|
|
|
|
for (size_t i = 0; i < size; ++i)
|
|
|
|
new (result.data + i) T(data[i]);
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
AstArray<T> Parser::copy(const TempVector<T>& data)
|
|
|
|
{
|
|
|
|
return copy(data.empty() ? nullptr : &data[0], data.size());
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
AstArray<T> Parser::copy(std::initializer_list<T> data)
|
|
|
|
{
|
|
|
|
return copy(data.size() == 0 ? nullptr : data.begin(), data.size());
|
|
|
|
}
|
|
|
|
|
|
|
|
AstArray<char> Parser::copy(const std::string& data)
|
|
|
|
{
|
|
|
|
AstArray<char> result = copy(data.c_str(), data.size() + 1);
|
|
|
|
|
|
|
|
result.size = data.size();
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Parser::incrementRecursionCounter(const char* context)
|
|
|
|
{
|
|
|
|
recursionCounter++;
|
|
|
|
|
|
|
|
if (recursionCounter > unsigned(FInt::LuauRecursionLimit))
|
|
|
|
{
|
|
|
|
ParseError::raise(lexer.current().location, "Exceeded allowed recursion depth; simplify your %s to make the code compile", context);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Parser::report(const Location& location, const char* format, va_list args)
|
|
|
|
{
|
|
|
|
// To reduce number of errors reported to user for incomplete statements, we skip multiple errors at the same location
|
|
|
|
// For example, consider 'local a = (((b + ' where multiple tokens haven't been written yet
|
|
|
|
if (!parseErrors.empty() && location == parseErrors.back().getLocation())
|
|
|
|
return;
|
|
|
|
|
|
|
|
std::string message = vformat(format, args);
|
|
|
|
|
|
|
|
// when limited to a single error, behave as if the error recovery is disabled
|
|
|
|
if (FInt::LuauParseErrorLimit == 1)
|
|
|
|
throw ParseError(location, message);
|
|
|
|
|
|
|
|
parseErrors.emplace_back(location, message);
|
|
|
|
|
|
|
|
if (parseErrors.size() >= unsigned(FInt::LuauParseErrorLimit))
|
|
|
|
ParseError::raise(location, "Reached error limit (%d)", int(FInt::LuauParseErrorLimit));
|
|
|
|
}
|
|
|
|
|
|
|
|
void Parser::report(const Location& location, const char* format, ...)
|
|
|
|
{
|
|
|
|
va_list args;
|
|
|
|
va_start(args, format);
|
|
|
|
report(location, format, args);
|
|
|
|
va_end(args);
|
|
|
|
}
|
|
|
|
|
|
|
|
LUAU_NOINLINE void Parser::reportNameError(const char* context)
|
|
|
|
{
|
|
|
|
if (context)
|
|
|
|
report(lexer.current().location, "Expected identifier when parsing %s, got %s", context, lexer.current().toString().c_str());
|
|
|
|
else
|
|
|
|
report(lexer.current().location, "Expected identifier, got %s", lexer.current().toString().c_str());
|
|
|
|
}
|
|
|
|
|
|
|
|
AstStatError* Parser::reportStatError(
|
|
|
|
const Location& location, const AstArray<AstExpr*>& expressions, const AstArray<AstStat*>& statements, const char* format, ...)
|
|
|
|
{
|
|
|
|
va_list args;
|
|
|
|
va_start(args, format);
|
|
|
|
report(location, format, args);
|
|
|
|
va_end(args);
|
|
|
|
|
|
|
|
return allocator.alloc<AstStatError>(location, expressions, statements, unsigned(parseErrors.size() - 1));
|
|
|
|
}
|
|
|
|
|
|
|
|
AstExprError* Parser::reportExprError(const Location& location, const AstArray<AstExpr*>& expressions, const char* format, ...)
|
|
|
|
{
|
|
|
|
va_list args;
|
|
|
|
va_start(args, format);
|
|
|
|
report(location, format, args);
|
|
|
|
va_end(args);
|
|
|
|
|
|
|
|
return allocator.alloc<AstExprError>(location, expressions, unsigned(parseErrors.size() - 1));
|
|
|
|
}
|
|
|
|
|
2023-03-17 14:59:30 +00:00
|
|
|
AstTypeError* Parser::reportTypeError(const Location& location, const AstArray<AstType*>& types, const char* format, ...)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
va_list args;
|
|
|
|
va_start(args, format);
|
|
|
|
report(location, format, args);
|
|
|
|
va_end(args);
|
|
|
|
|
2022-10-27 23:22:49 +01:00
|
|
|
return allocator.alloc<AstTypeError>(location, types, false, unsigned(parseErrors.size() - 1));
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
2023-03-17 14:59:30 +00:00
|
|
|
AstTypeError* Parser::reportMissingTypeError(const Location& parseErrorLocation, const Location& astErrorLocation, const char* format, ...)
|
2022-09-02 00:00:14 +01:00
|
|
|
{
|
|
|
|
va_list args;
|
|
|
|
va_start(args, format);
|
|
|
|
report(parseErrorLocation, format, args);
|
|
|
|
va_end(args);
|
|
|
|
|
|
|
|
return allocator.alloc<AstTypeError>(astErrorLocation, AstArray<AstType*>{}, true, unsigned(parseErrors.size() - 1));
|
|
|
|
}
|
|
|
|
|
2022-02-18 00:41:20 +00:00
|
|
|
void Parser::nextLexeme()
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-07-29 04:41:13 +01:00
|
|
|
Lexeme::Type type = lexer.next(/* skipComments= */ false, true).type;
|
2022-07-14 23:39:35 +01:00
|
|
|
|
2022-07-29 04:41:13 +01:00
|
|
|
while (type == Lexeme::BrokenComment || type == Lexeme::Comment || type == Lexeme::BlockComment)
|
|
|
|
{
|
|
|
|
const Lexeme& lexeme = lexer.current();
|
2022-02-18 00:41:20 +00:00
|
|
|
|
2022-07-29 04:41:13 +01:00
|
|
|
if (options.captureComments)
|
|
|
|
commentLocations.push_back(Comment{lexeme.type, lexeme.location});
|
2022-02-18 00:41:20 +00:00
|
|
|
|
2022-07-29 04:41:13 +01:00
|
|
|
// Subtlety: Broken comments are weird because we record them as comments AND pass them to the parser as a lexeme.
|
|
|
|
// The parser will turn this into a proper syntax error.
|
|
|
|
if (lexeme.type == Lexeme::BrokenComment)
|
|
|
|
return;
|
2022-02-18 00:41:20 +00:00
|
|
|
|
2022-07-29 04:41:13 +01:00
|
|
|
// Comments starting with ! are called "hot comments" and contain directives for type checking / linting / compiling
|
2024-06-07 18:09:03 +01:00
|
|
|
if (lexeme.type == Lexeme::Comment && lexeme.getLength() && lexeme.data[0] == '!')
|
2022-07-29 04:41:13 +01:00
|
|
|
{
|
|
|
|
const char* text = lexeme.data;
|
2022-02-18 00:41:20 +00:00
|
|
|
|
2024-06-07 18:09:03 +01:00
|
|
|
unsigned int end = lexeme.getLength();
|
2022-07-29 04:41:13 +01:00
|
|
|
while (end > 0 && isSpace(text[end - 1]))
|
|
|
|
--end;
|
2022-03-18 00:06:25 +00:00
|
|
|
|
2022-07-29 04:41:13 +01:00
|
|
|
hotcomments.push_back({hotcommentHeader, lexeme.location, std::string(text + 1, text + end)});
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
2022-07-29 04:41:13 +01:00
|
|
|
|
|
|
|
type = lexer.next(/* skipComments= */ false, /* updatePrevLocation= */ false).type;
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace Luau
|