mirror of
https://gitlab.com/OpenMW/openmw.git
synced 2025-01-08 09:37:53 +00:00
b88f0d2dbd
Mostly to avoid string literal lookup by index with iteration over all preciding literals and calling strlen. This is very inefficient. In genral this makes code much more straightforward but also makes it portable since now int and float of different sizes are properly supported.
98 lines
2.6 KiB
C++
98 lines
2.6 KiB
C++
#include "scriptparser.hpp"
|
|
|
|
#include "errorhandler.hpp"
|
|
#include "scanner.hpp"
|
|
#include "skipparser.hpp"
|
|
|
|
namespace Compiler
|
|
{
|
|
ScriptParser::ScriptParser(ErrorHandler& errorHandler, const Context& context, Locals& locals, bool end)
|
|
: Parser(errorHandler, context)
|
|
, mOutput(locals)
|
|
, mLineParser(errorHandler, context, locals, mOutput.getLiterals(), mOutput.getCode())
|
|
, mControlParser(errorHandler, context, locals, mOutput.getLiterals())
|
|
, mEnd(end)
|
|
{
|
|
}
|
|
|
|
Interpreter::Program ScriptParser::getProgram() const
|
|
{
|
|
return mOutput.getProgram();
|
|
}
|
|
|
|
bool ScriptParser::parseName(const std::string& name, const TokenLoc& loc, Scanner& scanner)
|
|
{
|
|
mLineParser.reset();
|
|
if (mLineParser.parseName(name, loc, scanner))
|
|
scanner.scan(mLineParser);
|
|
|
|
return true;
|
|
}
|
|
|
|
bool ScriptParser::parseKeyword(int keyword, const TokenLoc& loc, Scanner& scanner)
|
|
{
|
|
if (keyword == Scanner::K_while || keyword == Scanner::K_if || keyword == Scanner::K_elseif)
|
|
{
|
|
mControlParser.reset();
|
|
if (mControlParser.parseKeyword(keyword, loc, scanner))
|
|
scanner.scan(mControlParser);
|
|
|
|
mControlParser.appendCode(mOutput.getCode());
|
|
|
|
return true;
|
|
}
|
|
|
|
/// \todo add an option to disable this nonsense
|
|
if (keyword == Scanner::K_endif)
|
|
{
|
|
// surplus endif
|
|
getErrorHandler().warning("endif without matching if/elseif", loc);
|
|
|
|
SkipParser skip(getErrorHandler(), getContext());
|
|
scanner.scan(skip);
|
|
return true;
|
|
}
|
|
|
|
if (keyword == Scanner::K_end && mEnd)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
mLineParser.reset();
|
|
if (mLineParser.parseKeyword(keyword, loc, scanner))
|
|
scanner.scan(mLineParser);
|
|
|
|
return true;
|
|
}
|
|
|
|
bool ScriptParser::parseSpecial(int code, const TokenLoc& loc, Scanner& scanner)
|
|
{
|
|
if (code == Scanner::S_newline) // empty line
|
|
return true;
|
|
|
|
if (code == Scanner::S_open) /// \todo Option to switch this off
|
|
{
|
|
scanner.putbackSpecial(code, loc);
|
|
return parseKeyword(Scanner::K_if, loc, scanner);
|
|
}
|
|
|
|
mLineParser.reset();
|
|
if (mLineParser.parseSpecial(code, loc, scanner))
|
|
scanner.scan(mLineParser);
|
|
|
|
return true;
|
|
}
|
|
|
|
void ScriptParser::parseEOF(Scanner& scanner)
|
|
{
|
|
if (mEnd)
|
|
Parser::parseEOF(scanner);
|
|
}
|
|
|
|
void ScriptParser::reset()
|
|
{
|
|
mLineParser.reset();
|
|
mOutput.clear();
|
|
}
|
|
}
|