2010-06-27 17:20:21 +00:00
|
|
|
#include "fileparser.hpp"
|
|
|
|
|
|
|
|
#include "scanner.hpp"
|
|
|
|
#include "tokenloc.hpp"
|
|
|
|
|
|
|
|
namespace Compiler
|
|
|
|
{
|
|
|
|
FileParser::FileParser(ErrorHandler& errorHandler, Context& context)
|
2010-08-22 10:47:56 +00:00
|
|
|
: Parser(errorHandler, context)
|
2010-06-28 11:28:50 +00:00
|
|
|
, mScriptParser(errorHandler, context, mLocals, true)
|
2010-06-28 09:38:04 +00:00
|
|
|
, mState(BeginState)
|
2010-06-27 17:20:21 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2010-06-27 22:50:48 +00:00
|
|
|
std::string FileParser::getName() const
|
2010-06-27 17:20:21 +00:00
|
|
|
{
|
2010-06-27 22:50:48 +00:00
|
|
|
return mName;
|
2010-06-27 17:20:21 +00:00
|
|
|
}
|
2010-08-22 10:47:56 +00:00
|
|
|
|
2023-01-10 03:10:18 +00:00
|
|
|
Interpreter::Program FileParser::getProgram() const
|
2010-06-28 12:07:55 +00:00
|
|
|
{
|
2023-01-10 03:10:18 +00:00
|
|
|
return mScriptParser.getProgram();
|
2010-06-28 12:07:55 +00:00
|
|
|
}
|
2010-08-22 10:47:56 +00:00
|
|
|
|
2010-06-28 12:17:50 +00:00
|
|
|
const Locals& FileParser::getLocals() const
|
|
|
|
{
|
|
|
|
return mLocals;
|
|
|
|
}
|
2010-06-27 17:20:21 +00:00
|
|
|
|
|
|
|
bool FileParser::parseName(const std::string& name, const TokenLoc& loc, Scanner& scanner)
|
|
|
|
{
|
2010-06-27 22:50:48 +00:00
|
|
|
if (mState == NameState)
|
|
|
|
{
|
|
|
|
mName = name;
|
|
|
|
mState = BeginCompleteState;
|
|
|
|
return true;
|
|
|
|
}
|
2010-08-22 10:47:56 +00:00
|
|
|
|
2010-06-27 22:50:48 +00:00
|
|
|
if (mState == EndNameState)
|
|
|
|
{
|
|
|
|
// optional repeated name after end statement
|
|
|
|
if (mName != name)
|
|
|
|
reportWarning("Names for script " + mName + " do not match", loc);
|
2010-08-22 10:47:56 +00:00
|
|
|
|
2010-06-27 22:50:48 +00:00
|
|
|
mState = EndCompleteState;
|
2013-02-25 09:32:38 +00:00
|
|
|
return false; // we are stopping here, because there might be more garbage on the end line,
|
|
|
|
// that we must ignore.
|
|
|
|
//
|
|
|
|
/// \todo allow this workaround to be disabled for newer scripts
|
2010-06-27 22:50:48 +00:00
|
|
|
}
|
2010-08-22 10:47:56 +00:00
|
|
|
|
2014-07-18 10:29:20 +00:00
|
|
|
if (mState == BeginCompleteState)
|
|
|
|
{
|
|
|
|
reportWarning("Stray string (" + name + ") after begin statement", loc);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2010-06-27 22:50:48 +00:00
|
|
|
return Parser::parseName(name, loc, scanner);
|
2010-06-27 17:20:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool FileParser::parseKeyword(int keyword, const TokenLoc& loc, Scanner& scanner)
|
|
|
|
{
|
2010-06-27 22:50:48 +00:00
|
|
|
if (mState == BeginState && keyword == Scanner::K_begin)
|
|
|
|
{
|
|
|
|
mState = NameState;
|
2018-01-12 15:37:06 +00:00
|
|
|
scanner.enableTolerantNames(); /// \todo disable
|
2010-06-27 22:50:48 +00:00
|
|
|
return true;
|
|
|
|
}
|
2010-08-22 10:47:56 +00:00
|
|
|
|
2010-07-05 08:58:10 +00:00
|
|
|
if (mState == NameState)
|
|
|
|
{
|
|
|
|
// keywords can be used as script names too. Thank you Morrowind for another
|
|
|
|
// syntactic perversity :(
|
|
|
|
mName = loc.mLiteral;
|
|
|
|
mState = BeginCompleteState;
|
|
|
|
return true;
|
|
|
|
}
|
2010-08-22 10:47:56 +00:00
|
|
|
|
2013-04-27 12:06:23 +00:00
|
|
|
if (mState == EndNameState)
|
|
|
|
{
|
|
|
|
// optional repeated name after end statement
|
|
|
|
if (mName != loc.mLiteral)
|
|
|
|
reportWarning("Names for script " + mName + " do not match", loc);
|
|
|
|
|
|
|
|
mState = EndCompleteState;
|
|
|
|
return false; // we are stopping here, because there might be more garbage on the end line,
|
|
|
|
// that we must ignore.
|
|
|
|
//
|
|
|
|
/// \todo allow this workaround to be disabled for newer scripts
|
|
|
|
}
|
|
|
|
|
2010-06-27 22:50:48 +00:00
|
|
|
return Parser::parseKeyword(keyword, loc, scanner);
|
2010-06-27 17:20:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool FileParser::parseSpecial(int code, const TokenLoc& loc, Scanner& scanner)
|
|
|
|
{
|
2019-01-15 21:08:42 +00:00
|
|
|
if (code == Scanner::S_newline)
|
|
|
|
{
|
2022-12-27 13:59:56 +00:00
|
|
|
if (mState == BeginState)
|
|
|
|
return true;
|
2010-06-27 22:50:48 +00:00
|
|
|
if (mState == BeginCompleteState)
|
|
|
|
{
|
2010-06-28 09:38:04 +00:00
|
|
|
// parse the script body
|
|
|
|
mScriptParser.reset();
|
2010-08-22 10:47:56 +00:00
|
|
|
|
2010-06-28 09:38:04 +00:00
|
|
|
scanner.scan(mScriptParser);
|
2010-08-22 10:47:56 +00:00
|
|
|
|
2010-06-28 09:38:04 +00:00
|
|
|
mState = EndNameState;
|
2010-06-27 22:50:48 +00:00
|
|
|
return true;
|
|
|
|
}
|
2010-08-22 10:47:56 +00:00
|
|
|
|
2010-06-27 22:50:48 +00:00
|
|
|
if (mState == EndCompleteState || mState == EndNameState)
|
|
|
|
{
|
|
|
|
// we are done here -> ignore the rest of the script
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2010-08-22 10:47:56 +00:00
|
|
|
|
2010-06-27 22:50:48 +00:00
|
|
|
return Parser::parseSpecial(code, loc, scanner);
|
2010-06-27 17:20:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void FileParser::parseEOF(Scanner& scanner)
|
|
|
|
{
|
2010-06-27 22:50:48 +00:00
|
|
|
if (mState != EndNameState && mState != EndCompleteState)
|
|
|
|
Parser::parseEOF(scanner);
|
|
|
|
}
|
2010-08-22 10:47:56 +00:00
|
|
|
|
2010-06-27 22:50:48 +00:00
|
|
|
void FileParser::reset()
|
|
|
|
{
|
|
|
|
mState = BeginState;
|
|
|
|
mName.clear();
|
2010-06-28 09:38:04 +00:00
|
|
|
mScriptParser.reset();
|
2010-08-22 10:47:56 +00:00
|
|
|
Parser::reset();
|
2010-06-27 17:20:21 +00:00
|
|
|
}
|
|
|
|
}
|