BsScriptManagedResource.cpp 2.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990
  1. #include "BsScriptManagedResource.h"
  2. #include "BsScriptResourceManager.h"
  3. #include "BsScriptAssemblyManager.h"
  4. #include "BsScriptMeta.h"
  5. #include "BsMonoField.h"
  6. #include "BsMonoClass.h"
  7. #include "BsMonoManager.h"
  8. #include "BsManagedResource.h"
  9. #include "BsResources.h"
  10. #include "BsException.h"
  11. namespace BansheeEngine
  12. {
  13. ScriptManagedResource::ScriptManagedResource(MonoObject* instance, const HManagedResource& resource)
  14. :ScriptObject(instance), mResource(resource)
  15. {
  16. assert(instance != nullptr);
  17. ::MonoClass* monoClass = mono_object_get_class(instance);
  18. mNamespace = mono_class_get_namespace(monoClass);
  19. mType = mono_class_get_name(monoClass);
  20. }
  21. void ScriptManagedResource::initRuntimeData()
  22. {
  23. metaData.scriptClass->addInternalCall("Internal_CreateInstance", &ScriptManagedResource::internal_createInstance);
  24. }
  25. void ScriptManagedResource::internal_createInstance(MonoObject* instance)
  26. {
  27. HManagedResource resource = ManagedResource::create(instance);
  28. ScriptResourceManager::instance().createManagedResource(instance, resource);
  29. }
  30. MonoObject* ScriptManagedResource::_createManagedInstance(bool construct)
  31. {
  32. ManagedSerializableObjectInfoPtr currentObjInfo = nullptr;
  33. // See if this type even still exists
  34. if (!ScriptAssemblyManager::instance().getSerializableObjectInfo(mNamespace, mType, currentObjInfo))
  35. return nullptr;
  36. return currentObjInfo->mMonoClass->createInstance(construct);
  37. }
  38. ScriptObjectBackup ScriptManagedResource::beginRefresh()
  39. {
  40. ScriptResourceBase::beginRefresh();
  41. ScriptObjectBackup backupData;
  42. backupData.data = mResource->backup(true);
  43. return backupData;
  44. }
  45. void ScriptManagedResource::endRefresh(const ScriptObjectBackup& backupData)
  46. {
  47. ResourceBackupData resourceBackup = any_cast<ResourceBackupData>(backupData.data);
  48. mResource->restore(mManagedInstance, resourceBackup);
  49. // If we could not find resource type after refresh, treat it as if it was destroyed
  50. if (mManagedInstance == nullptr)
  51. _onManagedInstanceDeleted();
  52. ScriptResourceBase::endRefresh(backupData);
  53. }
  54. void ScriptManagedResource::_onManagedInstanceDeleted()
  55. {
  56. mManagedInstance = nullptr;
  57. if (!mRefreshInProgress)
  58. {
  59. // The only way this method should be reachable is when Resource::unload is called, which means the resource
  60. // has had to been already freed. Even if all managed instances are released ManagedResource itself holds the last
  61. // instance which is only freed on unload().
  62. // Note: During domain unload this could get called even if not all instances are released, but ManagedResourceManager
  63. // should make sure all instances are unloaded before that happens.
  64. assert(mResource == nullptr || !mResource.isLoaded());
  65. ScriptResourceManager::instance().destroyScriptResource(this);
  66. }
  67. }
  68. void ScriptManagedResource::setNativeHandle(const HResource& resource)
  69. {
  70. mResource = static_resource_cast<ManagedResource>(resource);
  71. }
  72. }