1
0
mirror of https://gitlab.com/OpenMW/openmw.git synced 2025-01-12 21:39:26 +00:00
OpenMW/apps/openmw/mwscript/scriptmanagerimp.cpp

231 lines
6.5 KiB
C++
Raw Normal View History

#include "scriptmanagerimp.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>
2012-10-01 15:17:04 +00:00
#include "../mwworld/esmstore.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>
#include <components/compiler/exception.hpp>
2010-07-02 16:08:00 +00:00
#include "extensions.hpp"
namespace MWScript
{
2012-10-01 15:17:04 +00:00
ScriptManager::ScriptManager (const MWWorld::ESMStore& store, bool verbose,
2010-07-02 15:21:27 +00:00
Compiler::Context& compilerContext)
: mErrorHandler (std::cerr), mStore (store), mVerbose (verbose),
mCompilerContext (compilerContext), mParser (mErrorHandler, mCompilerContext),
mOpcodesInstalled (false), mGlobalScripts (store)
{}
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.get<ESM::Script>().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
{
2012-09-17 07:37:50 +00:00
std::istringstream input (script->mScriptText);
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 (const Compiler::SourceException&)
2010-07-02 15:21:27 +00:00
{
// error has already been reported via error handler
Success = false;
}
catch (const std::exception& error)
{
std::cerr << "An exception has been thrown: " << error.what() << std::endl;
2010-07-02 15:21:27 +00:00
Success = false;
}
if (!Success && mVerbose)
{
std::cerr
<< "compiling failed: " << name << std::endl
2012-09-17 07:37:50 +00:00
<< script->mScriptText
2010-07-02 15:21:27 +00:00
<< std::endl << std::endl;
}
2010-07-02 15:21:27 +00:00
if (Success)
{
std::vector<Interpreter::Type_Code> code;
mParser.getCode (code);
2012-03-05 15:56:14 +00:00
mScripts.insert (std::make_pair (name, std::make_pair (code, mParser.getLocals())));
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
2012-03-05 15:56:14 +00:00
ScriptCollection::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;
2012-03-05 15:56:14 +00:00
mScripts.insert (std::make_pair (name, std::make_pair (empty, Compiler::Locals())));
2010-07-02 15:21:27 +00:00
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
2012-03-05 15:56:14 +00:00
if (!iter->second.first.empty())
2010-07-02 16:08:00 +00:00
try
{
if (!mOpcodesInstalled)
{
installOpcodes (mInterpreter);
mOpcodesInstalled = true;
}
2012-03-05 15:56:14 +00:00
mInterpreter.run (&iter->second.first[0], iter->second.first.size(), interpreterContext);
2010-07-02 16:08:00 +00:00
}
catch (const std::exception& e)
2010-07-02 16:08:00 +00:00
{
2013-03-31 11:13:46 +00:00
std::cerr << "execution of script " << name << " failed." << std::endl;
if (mVerbose)
std::cerr << "(" << e.what() << ")" << std::endl;
2012-03-05 15:56:14 +00:00
iter->second.first.clear(); // don't execute again.
2010-07-02 16:08:00 +00:00
}
}
std::pair<int, int> ScriptManager::compileAll()
{
int count = 0;
int success = 0;
const MWWorld::Store<ESM::Script>& scripts = mStore.get<ESM::Script>();
MWWorld::Store<ESM::Script>::iterator it = scripts.begin();
for (; it != scripts.end(); ++it, ++count)
if (compile (it->mId))
++success;
return std::make_pair (count, success);
}
2012-03-05 15:56:14 +00:00
Compiler::Locals& ScriptManager::getLocals (const std::string& name)
{
{
ScriptCollection::iterator iter = mScripts.find (name);
if (iter!=mScripts.end())
return iter->second.second;
}
2012-03-05 15:56:14 +00:00
{
std::map<std::string, Compiler::Locals>::iterator iter = mOtherLocals.find (name);
if (iter!=mOtherLocals.end())
return iter->second;
}
2012-03-05 15:56:14 +00:00
Compiler::Locals locals;
if (const ESM::Script *script = mStore.get<ESM::Script>().find (name))
{
int index = 0;
for (int i=0; i<script->mData.mNumShorts; ++i)
locals.declare ('s', script->mVarNames[index++]);
for (int i=0; i<script->mData.mNumLongs; ++i)
locals.declare ('l', script->mVarNames[index++]);
for (int i=0; i<script->mData.mNumFloats; ++i)
locals.declare ('f', script->mVarNames[index++]);
2013-02-23 19:20:40 +00:00
std::map<std::string, Compiler::Locals>::iterator iter =
mOtherLocals.insert (std::make_pair (name, locals)).first;
return iter->second;
2012-03-05 15:56:14 +00:00
}
throw std::logic_error ("script " + name + " does not exist");
2012-03-05 15:56:14 +00:00
}
GlobalScripts& ScriptManager::getGlobalScripts()
{
return mGlobalScripts;
}
int ScriptManager::getLocalIndex (const std::string& scriptId, const std::string& variable,
char type)
{
const ESM::Script *script = mStore.get<ESM::Script>().find (scriptId);
int offset = 0;
int size = 0;
switch (type)
{
case 's':
offset = 0;
2012-09-17 07:37:50 +00:00
size = script->mData.mNumShorts;
break;
case 'l':
2012-09-17 07:37:50 +00:00
offset = script->mData.mNumShorts;
size = script->mData.mNumLongs;
break;
case 'f':
2012-09-17 07:37:50 +00:00
offset = script->mData.mNumShorts+script->mData.mNumLongs;
size = script->mData.mNumFloats;
2013-05-21 19:04:15 +00:00
break;
default:
throw std::runtime_error ("invalid variable type");
}
for (int i=0; i<size; ++i)
2012-09-17 07:37:50 +00:00
if (script->mVarNames.at (i+offset)==variable)
return i;
throw std::runtime_error ("unable to access local variable " + variable + " of " + scriptId);
}
2013-05-15 15:54:18 +00:00
void ScriptManager::resetGlobalScripts()
{
mGlobalScripts.reset();
}
}