| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203 |
- #include "BsMonoManager.h"
- #include "CmException.h"
- #include "BsScriptMeta.h"
- #include "BsMonoAssembly.h"
- #include "BsMonoClass.h"
- #include <mono/metadata/assembly.h>
- #include <mono/metadata/mono-config.h>
- namespace BansheeEngine
- {
- const String MonoManager::MONO_LIB_DIR = "..\\..\\Mono\\lib";
- const String MonoManager::MONO_ETC_DIR = "..\\..\\Mono\\etc";
- MonoManager::MonoManager()
- :mDomain(nullptr), mIsCoreLoaded(false)
- {
- mono_set_dirs(MONO_LIB_DIR.c_str(), MONO_ETC_DIR.c_str());
- mono_config_parse(nullptr);
- }
- MonoManager::~MonoManager()
- {
- for(auto& entry : mAssemblies)
- {
- unloadAssembly(*entry.second);
- cm_delete(entry.second);
- }
- mAssemblies.clear();
- if(mDomain != nullptr)
- {
- mono_jit_cleanup(mDomain);
- mDomain = nullptr;
- }
- }
- MonoAssembly& MonoManager::loadAssembly(const String& path, const String& name)
- {
- MonoAssembly* assembly = nullptr;
- if(mDomain == nullptr)
- {
- mDomain = mono_jit_init (path.c_str());
- if(mDomain == nullptr)
- {
- CM_EXCEPT(InternalErrorException, "Cannot initialize Mono runtime.");
- }
- }
- auto iterFind = mAssemblies.find(name);
- if(iterFind != mAssemblies.end())
- {
- assembly = iterFind->second;
- }
- else
- {
- assembly = new (cm_alloc<MonoAssembly>()) MonoAssembly();
- mAssemblies[name] = assembly;
- }
-
- if(!assembly->mIsLoaded)
- {
- assembly->load(path, name);
- // Fully initialize all types that use this assembly
- Vector<ScriptMeta*>::type& mTypeMetas = getTypesToInitialize()[name];
- for(auto& meta : mTypeMetas)
- {
- meta->scriptClass = assembly->getClass(meta->ns, meta->name);
- if(meta->scriptClass == nullptr)
- CM_EXCEPT(InvalidParametersException, "Unable to find class of type: \"" + meta->ns + "::" + meta->name + "\"");
- if(meta->scriptClass->hasField("mCachedPtr"))
- meta->thisPtrField = meta->scriptClass->getField("mCachedPtr");
- else
- meta->thisPtrField = nullptr;
- meta->initCallback();
- }
- }
- if(!mIsCoreLoaded)
- {
- mIsCoreLoaded = true;
- MonoImage* existingImage = mono_image_loaded("mscorlib");
- if(existingImage != nullptr)
- {
- MonoAssembly* mscorlib = new (cm_alloc<MonoAssembly>()) MonoAssembly();
- mAssemblies["mscorlib"] = mscorlib;
- mscorlib->loadAsDependency(existingImage, "mscorlib");
- }
- else
- loadAssembly("mscorlib", "mscorlib");
- }
- return *assembly;
- }
- void MonoManager::unloadAssembly(MonoAssembly& assembly)
- {
- ::MonoAssembly* monoAssembly = assembly.mMonoAssembly;
- assembly.unload();
- if(monoAssembly)
- mono_assembly_close(monoAssembly);
- }
- MonoAssembly* MonoManager::getAssembly(const String& name) const
- {
- auto iterFind = mAssemblies.find(name);
- if(iterFind != mAssemblies.end())
- return iterFind->second;
- return nullptr;
- }
- void MonoManager::registerScriptType(ScriptMeta* metaData)
- {
- Vector<ScriptMeta*>::type& mMetas = getTypesToInitialize()[metaData->assembly];
- mMetas.push_back(metaData);
- }
- MonoClass* MonoManager::findClass(const String& ns, const String& typeName)
- {
- MonoClass* monoClass = nullptr;
- for(auto& assembly : mAssemblies)
- {
- monoClass = assembly.second->getClass(ns, typeName);
- if(monoClass != nullptr)
- return monoClass;
- }
- return nullptr;
- }
- MonoClass* MonoManager::findClass(::MonoClass* rawMonoClass)
- {
- MonoClass* monoClass = nullptr;
- for(auto& assembly : mAssemblies)
- {
- monoClass = assembly.second->getClass(rawMonoClass);
- if(monoClass != nullptr)
- return monoClass;
- }
- return nullptr;
- }
- String MonoManager::getFullTypeName(MonoObject* obj)
- {
- if(obj == nullptr)
- CM_EXCEPT(InvalidParametersException, "Object cannot be null.");
- ::MonoClass* monoClass = mono_object_get_class(obj);
- const char* nameSpaceChars = mono_class_get_namespace(monoClass);
- String namespaceStr;
- if(nameSpaceChars != nullptr)
- namespaceStr = nameSpaceChars;
- const char* typeNameChars = mono_class_get_name(monoClass);
- String typeNameStr;
- if(typeNameChars != nullptr)
- typeNameStr = typeNameChars;
- return namespaceStr + "." + typeNameStr;
- }
- String MonoManager::getNamespace(MonoObject* obj)
- {
- if(obj == nullptr)
- CM_EXCEPT(InvalidParametersException, "Object cannot be null.");
- ::MonoClass* monoClass = mono_object_get_class(obj);
- const char* nameSpaceChars = mono_class_get_namespace(monoClass);
- String namespaceStr;
- if(nameSpaceChars != nullptr)
- namespaceStr = nameSpaceChars;
- return namespaceStr;
- }
- String MonoManager::getTypeName(MonoObject* obj)
- {
- if(obj == nullptr)
- CM_EXCEPT(InvalidParametersException, "Object cannot be null.");
- ::MonoClass* monoClass = mono_object_get_class(obj);
- const char* typeNameChars = mono_class_get_name(monoClass);
- String typeNameStr;
- if(typeNameChars != nullptr)
- typeNameStr = typeNameChars;
- return typeNameStr;
- }
- }
|