openmw-tes3coop/apps/openmw/mwscript/scriptmanager.cpp

120 lines
3.3 KiB
C++
Raw Normal View History

#include "scriptmanager.hpp"
2010-07-02 15:21:27 +00:00
#include <cassert>
#include <iostream>
2010-07-02 15:21:27 +00:00
#include <sstream>
#include <exception>
#include <components/esm/loadscpt.hpp>
#include <components/esm_store/store.hpp>
2010-07-02 16:08:00 +00:00
#include <components/compiler/scanner.hpp>
2010-07-03 10:12:13 +00:00
#include <components/compiler/context.hpp>
2010-07-02 16:08:00 +00:00
#include "extensions.hpp"
namespace MWScript
{
2010-07-02 15:21:27 +00:00
ScriptManager::ScriptManager (const ESMS::ESMStore& store, bool verbose,
Compiler::Context& compilerContext)
: mErrorHandler (std::cerr), mStore (store), mVerbose (verbose),
mCompilerContext (compilerContext), mParser (mErrorHandler, mCompilerContext),
mOpcodesInstalled (false)
{}
2010-07-02 15:21:27 +00:00
bool ScriptManager::compile (const std::string& name)
{
mParser.reset();
mErrorHandler.reset();
bool Success = true;
if (const ESM::Script *script = mStore.scripts.find (name))
{
2010-07-02 15:21:27 +00:00
if (mVerbose)
std::cout << "compiling script: " << name << std::endl;
2010-07-02 15:21:27 +00:00
try
{
2010-07-02 15:21:27 +00:00
std::istringstream input (script->scriptText);
2010-07-03 10:12:13 +00:00
Compiler::Scanner scanner (mErrorHandler, input, mCompilerContext.getExtensions());
2010-07-02 15:21:27 +00:00
scanner.scan (mParser);
2010-07-02 15:21:27 +00:00
if (!mErrorHandler.isGood())
Success = false;
}
catch (...)
2010-07-02 15:21:27 +00:00
{
Success = false;
}
if (!Success && mVerbose)
{
std::cerr
<< "compiling failed: " << name << std::endl
<< script->scriptText
<< std::endl << std::endl;
}
2010-07-02 15:21:27 +00:00
if (Success)
{
std::vector<Interpreter::Type_Code> code;
mParser.getCode (code);
mScripts.insert (std::make_pair (name, code));
2010-07-02 15:21:27 +00:00
// TODO sanity check on generated locals
2010-07-02 15:21:27 +00:00
return true;
}
}
return false;
2010-07-02 15:21:27 +00:00
}
2010-07-02 16:08:00 +00:00
void ScriptManager::run (const std::string& name, Interpreter::Context& interpreterContext)
{
2010-07-02 15:21:27 +00:00
// compile script
std::map<std::string, std::vector<Interpreter::Type_Code> >::iterator iter =
mScripts.find (name);
2010-07-02 15:21:27 +00:00
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;
}
2010-07-02 15:21:27 +00:00
iter = mScripts.find (name);
assert (iter!=mScripts.end());
}
2010-07-02 15:21:27 +00:00
// execute script
2010-07-02 16:08:00 +00:00
if (!iter->second.empty())
try
{
if (!mOpcodesInstalled)
{
installOpcodes (mInterpreter);
mOpcodesInstalled = true;
}
mInterpreter.run (&iter->second[0], iter->second.size(), interpreterContext);
2010-07-02 16:08:00 +00:00
}
catch (const std::exception& e)
2010-07-02 16:08:00 +00:00
{
std::cerr << "exeution of script " << name << " failed." << std::endl;
if (mVerbose)
std::cerr << "(" << e.what() << ")" << std::endl;
2010-07-02 16:08:00 +00:00
iter->second.clear(); // don't execute again.
}
}
}