forked from teamnwah/openmw-tes3coop
120 lines
3.5 KiB
C++
120 lines
3.5 KiB
C++
|
|
#include "scriptmanager.hpp"
|
|
|
|
#include <cassert>
|
|
#include <iostream>
|
|
#include <sstream>
|
|
#include <exception>
|
|
|
|
#include <components/esm/loadscpt.hpp>
|
|
#include <components/esm_store/store.hpp>
|
|
|
|
#include <components/compiler/scanner.hpp>
|
|
#include <components/compiler/context.hpp>
|
|
|
|
#include <components/interpreter/installopcodes.hpp>
|
|
#include <components/interpreter/interpreter.hpp>
|
|
|
|
#include "../mwsound/extensions.hpp"
|
|
|
|
namespace MWScript
|
|
{
|
|
ScriptManager::ScriptManager (const ESMS::ESMStore& store, bool verbose,
|
|
Compiler::Context& compilerContext)
|
|
: mErrorHandler (std::cerr), mStore (store), mVerbose (verbose),
|
|
mCompilerContext (compilerContext), mParser (mErrorHandler, mCompilerContext)
|
|
{}
|
|
|
|
bool ScriptManager::compile (const std::string& name)
|
|
{
|
|
mParser.reset();
|
|
mErrorHandler.reset();
|
|
|
|
bool Success = true;
|
|
|
|
if (const ESM::Script *script = mStore.scripts.find (name))
|
|
{
|
|
if (mVerbose)
|
|
std::cout << "compiling script: " << name << std::endl;
|
|
|
|
try
|
|
{
|
|
std::istringstream input (script->scriptText);
|
|
|
|
Compiler::Scanner scanner (mErrorHandler, input, mCompilerContext.getExtensions());
|
|
|
|
scanner.scan (mParser);
|
|
|
|
if (!mErrorHandler.isGood())
|
|
Success = false;
|
|
}
|
|
catch (const std::exception& error)
|
|
{
|
|
Success = false;
|
|
}
|
|
|
|
if (!Success && mVerbose)
|
|
{
|
|
std::cerr
|
|
<< "compiling failed: " << name << std::endl
|
|
<< script->scriptText
|
|
<< std::endl << std::endl;
|
|
}
|
|
|
|
if (Success)
|
|
{
|
|
std::vector<Interpreter::Type_Code> code;
|
|
mParser.getCode (code);
|
|
mScripts.insert (std::make_pair (name, code));
|
|
|
|
// TODO sanity check on generated locals
|
|
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
void ScriptManager::run (const std::string& name, Interpreter::Context& interpreterContext)
|
|
{
|
|
// compile script
|
|
std::map<std::string, std::vector<Interpreter::Type_Code> >::iterator iter =
|
|
mScripts.find (name);
|
|
|
|
if (iter==mScripts.end())
|
|
{
|
|
if (!compile (name))
|
|
{
|
|
// failed -> ignore script from now on.
|
|
std::vector<Interpreter::Type_Code> empty;
|
|
mScripts.insert (std::make_pair (name, empty));
|
|
return;
|
|
}
|
|
|
|
iter = mScripts.find (name);
|
|
assert (iter!=mScripts.end());
|
|
}
|
|
|
|
// execute script
|
|
if (!iter->second.empty())
|
|
try
|
|
{
|
|
Interpreter::Interpreter interpreter (interpreterContext);
|
|
installOpcodes (interpreter);
|
|
interpreter.run (&iter->second[0], iter->second.size());
|
|
}
|
|
catch (...)
|
|
{
|
|
std::cerr << "exeution of script " << name << " failed." << std::endl;
|
|
iter->second.clear(); // don't execute again.
|
|
}
|
|
}
|
|
|
|
void ScriptManager::installOpcodes (Interpreter::Interpreter& interpreter)
|
|
{
|
|
Interpreter::installOpcodes (interpreter);
|
|
MWSound::installOpcodes (interpreter);
|
|
}
|
|
}
|
|
|