1
0
mirror of https://gitlab.com/OpenMW/openmw.git synced 2025-01-25 06:35:30 +00:00
OpenMW/apps/openmw/mwscript/scriptmanagerimp.cpp

228 lines
6.7 KiB
C++
Raw Normal View History

#include "scriptmanagerimp.hpp"
2010-07-02 17:21:27 +02:00
#include <cassert>
#include <sstream>
#include <exception>
2014-07-21 09:34:10 +02:00
#include <algorithm>
2010-07-02 17:21:27 +02:00
2018-08-14 23:05:43 +04:00
#include <components/debug/debuglog.hpp>
2010-07-02 17:21:27 +02:00
#include <components/esm/loadscpt.hpp>
#include <components/misc/stringops.hpp>
2010-07-02 18:08:00 +02:00
#include <components/compiler/scanner.hpp>
2010-07-03 12:12:13 +02:00
#include <components/compiler/context.hpp>
#include <components/compiler/exception.hpp>
#include <components/compiler/quickfileparser.hpp>
2010-07-02 18:08:00 +02:00
#include "../mwworld/esmstore.hpp"
#include "extensions.hpp"
#include "interpretercontext.hpp"
namespace MWScript
{
ScriptManager::ScriptManager (const MWWorld::ESMStore& store,
2014-07-21 09:34:10 +02:00
Compiler::Context& compilerContext, int warningsMode,
const std::vector<std::string>& scriptBlacklist)
: mErrorHandler(), mStore (store),
mCompilerContext (compilerContext), mParser (mErrorHandler, mCompilerContext),
mOpcodesInstalled (false), mGlobalScripts (store)
2014-02-02 14:09:59 +01:00
{
mErrorHandler.setWarningsMode (warningsMode);
2014-07-21 09:34:10 +02:00
mScriptBlacklist.resize (scriptBlacklist.size());
std::transform (scriptBlacklist.begin(), scriptBlacklist.end(),
mScriptBlacklist.begin(), Misc::StringUtils::lowerCase);
std::sort (mScriptBlacklist.begin(), mScriptBlacklist.end());
2014-02-02 14:09:59 +01:00
}
2010-07-02 17:21:27 +02:00
bool ScriptManager::compile (const std::string& name)
{
mParser.reset();
mErrorHandler.reset();
if (const ESM::Script *script = mStore.get<ESM::Script>().find (name))
{
mErrorHandler.setContext(name);
2014-09-26 17:12:48 +02:00
bool Success = true;
2010-07-02 17:21:27 +02:00
try
{
2012-09-17 11:37:50 +04:00
std::istringstream input (script->mScriptText);
2010-07-03 12:12:13 +02:00
Compiler::Scanner scanner (mErrorHandler, input, mCompilerContext.getExtensions());
2010-07-02 17:21:27 +02:00
scanner.scan (mParser);
2010-07-02 17:21:27 +02:00
if (!mErrorHandler.isGood())
Success = false;
}
catch (const Compiler::SourceException&)
2010-07-02 17:21:27 +02:00
{
// error has already been reported via error handler
Success = false;
}
catch (const std::exception& error)
{
2018-08-14 23:05:43 +04:00
Log(Debug::Error) << "Error: An exception has been thrown: " << error.what();
2010-07-02 17:21:27 +02:00
Success = false;
}
if (!Success)
2010-07-02 17:21:27 +02:00
{
Log(Debug::Error) << "Error: script compiling failed: " << name;
2010-07-02 17:21:27 +02:00
}
2010-07-02 17:21:27 +02:00
if (Success)
{
std::vector<Interpreter::Type_Code> code;
mParser.getCode(code);
mScripts.emplace(name, CompiledScript(code, mParser.getLocals()));
2010-07-02 17:21:27 +02:00
return true;
}
}
return false;
2010-07-02 17:21:27 +02:00
}
bool ScriptManager::run (const std::string& name, Interpreter::Context& interpreterContext)
{
2010-07-02 17:21:27 +02:00
// compile script
2012-03-05 16:56:14 +01:00
ScriptCollection::iterator iter = mScripts.find (name);
2010-07-02 17:21:27 +02:00
if (iter==mScripts.end())
{
if (!compile (name))
{
// failed -> ignore script from now on.
std::vector<Interpreter::Type_Code> empty;
mScripts.emplace(name, CompiledScript(empty, Compiler::Locals()));
return false;
2010-07-02 17:21:27 +02:00
}
2010-07-02 17:21:27 +02:00
iter = mScripts.find (name);
assert (iter!=mScripts.end());
}
2010-07-02 17:21:27 +02:00
// execute script
2021-02-14 16:25:38 +01:00
std::string target = Misc::StringUtils::lowerCase(interpreterContext.getTarget());
if (!iter->second.mByteCode.empty() && iter->second.mInactive.find(target) == iter->second.mInactive.end())
2010-07-02 18:08:00 +02:00
try
{
if (!mOpcodesInstalled)
{
installOpcodes (mInterpreter);
mOpcodesInstalled = true;
}
mInterpreter.run (&iter->second.mByteCode[0], iter->second.mByteCode.size(), interpreterContext);
return true;
}
catch (const MissingImplicitRefError& e)
{
Log(Debug::Error) << "Execution of script " << name << " failed: " << e.what();
2010-07-02 18:08:00 +02:00
}
catch (const std::exception& e)
2010-07-02 18:08:00 +02:00
{
Log(Debug::Error) << "Execution of script " << name << " failed: " << e.what();
2021-02-14 16:25:38 +01:00
iter->second.mInactive.insert(target); // don't execute again.
2010-07-02 18:08:00 +02:00
}
return false;
}
void ScriptManager::clear()
{
for (auto& script : mScripts)
{
2021-02-14 16:25:38 +01:00
script.second.mInactive.clear();
}
mGlobalScripts.clear();
}
std::pair<int, int> ScriptManager::compileAll()
{
int count = 0;
int success = 0;
for (auto& script : mStore.get<ESM::Script>())
{
2014-07-21 09:34:10 +02:00
if (!std::binary_search (mScriptBlacklist.begin(), mScriptBlacklist.end(),
Misc::StringUtils::lowerCase(script.mId)))
2014-07-21 09:34:10 +02:00
{
++count;
if (compile(script.mId))
2014-07-21 09:34:10 +02:00
++success;
}
}
return std::make_pair (count, success);
}
2012-03-05 16:56:14 +01:00
const Compiler::Locals& ScriptManager::getLocals (const std::string& name)
2012-03-05 16:56:14 +01:00
{
std::string name2 = Misc::StringUtils::lowerCase (name);
{
auto iter = mScripts.find (name2);
if (iter!=mScripts.end())
return iter->second.mLocals;
}
2012-03-05 16:56:14 +01:00
{
auto iter = mOtherLocals.find (name2);
if (iter!=mOtherLocals.end())
return iter->second;
}
2012-03-05 16:56:14 +01:00
if (const ESM::Script *script = mStore.get<ESM::Script>().search (name2))
{
Compiler::Locals locals;
2020-04-04 14:09:00 +02:00
const Compiler::ContextOverride override(mErrorHandler, name2 + "[local variables]");
std::istringstream stream (script->mScriptText);
Compiler::QuickFileParser parser (mErrorHandler, mCompilerContext, locals);
Compiler::Scanner scanner (mErrorHandler, stream, mCompilerContext.getExtensions());
try
{
scanner.scan (parser);
}
catch (const Compiler::SourceException&)
{
// error has already been reported via error handler
locals.clear();
}
catch (const std::exception& error)
{
Log(Debug::Error) << "Error: An exception has been thrown: " << error.what();
locals.clear();
}
2013-02-23 20:20:40 +01:00
auto iter = mOtherLocals.emplace(name2, locals).first;
2013-02-23 20:20:40 +01:00
return iter->second;
2012-03-05 16:56:14 +01:00
}
throw std::logic_error ("script " + name + " does not exist");
2012-03-05 16:56:14 +01:00
}
GlobalScripts& ScriptManager::getGlobalScripts()
{
return mGlobalScripts;
}
2021-07-07 18:48:25 +02:00
const Compiler::Extensions& ScriptManager::getExtensions() const
{
return *mCompilerContext.getExtensions();
}
}