| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242 |
- //********************************** Banshee Engine (www.banshee3d.com) **************************************************//
- //**************** Copyright (c) 2016 Marko Pintera ([email protected]). All rights reserved. **********************//
- #include "Serialization/BsManagedSerializableList.h"
- #include "RTTI/BsManagedSerializableListRTTI.h"
- #include "BsMonoManager.h"
- #include "Serialization/BsScriptAssemblyManager.h"
- #include "Serialization/BsManagedSerializableField.h"
- #include "BsMonoClass.h"
- #include "BsMonoMethod.h"
- #include "BsMonoProperty.h"
- #include "BsMonoUtil.h"
- namespace bs
- {
- ManagedSerializableList::ManagedSerializableList(const ConstructPrivately& dummy)
- {
- }
- ManagedSerializableList::ManagedSerializableList(const ConstructPrivately& dummy,
- const SPtr<ManagedSerializableTypeInfoList>& typeInfo, MonoObject* managedInstance)
- :mListTypeInfo(typeInfo)
- {
- mGCHandle = MonoUtil::newGCHandle(managedInstance, false);
- MonoClass* listClass = MonoManager::instance().findClass(MonoUtil::getClass(managedInstance));
- if(listClass == nullptr)
- return;
- initMonoObjects(listClass);
- mNumElements = getLengthInternal();
- }
- ManagedSerializableList::~ManagedSerializableList()
- {
- if(mGCHandle != 0)
- {
- MonoUtil::freeGCHandle(mGCHandle);
- mGCHandle = 0;
- }
- }
- SPtr<ManagedSerializableList> ManagedSerializableList::createFromExisting(MonoObject* managedInstance,
- const SPtr<ManagedSerializableTypeInfoList>& typeInfo)
- {
- if(managedInstance == nullptr)
- return nullptr;
- String elementNs;
- String elementTypeName;
- MonoUtil::getClassName(managedInstance, elementNs, elementTypeName);
- String fullName = elementNs + "." + elementTypeName;
- if(ScriptAssemblyManager::instance().getBuiltinClasses().systemGenericListClass->getFullName() != fullName)
- return nullptr;
- return bs_shared_ptr_new<ManagedSerializableList>(ConstructPrivately(), typeInfo, managedInstance);
- }
- SPtr<ManagedSerializableList> ManagedSerializableList::createNew(const SPtr<ManagedSerializableTypeInfoList>& typeInfo, UINT32 size)
- {
- return bs_shared_ptr_new<ManagedSerializableList>(ConstructPrivately(), typeInfo, createManagedInstance(typeInfo, size));
- }
- MonoObject* ManagedSerializableList::createManagedInstance(const SPtr<ManagedSerializableTypeInfoList>& typeInfo, UINT32 size)
- {
- if (!typeInfo->isTypeLoaded())
- return nullptr;
- ::MonoClass* listMonoClass = typeInfo->getMonoClass();
- MonoClass* listClass = MonoManager::instance().findClass(listMonoClass);
- if (listClass == nullptr)
- return nullptr;
- void* params[1] = { &size };
- MonoObject* instance = listClass->createInstance("int", params);
-
- ScriptArray tempArray(typeInfo->mElementType->getMonoClass(), size);
- params[0] = tempArray.getInternal();
- MonoMethod* addRangeMethod = listClass->getMethod("AddRange", 1);
- addRangeMethod->invoke(instance, params);
- return instance;
- }
- SPtr<ManagedSerializableList> ManagedSerializableList::createEmpty()
- {
- return bs_shared_ptr_new<ManagedSerializableList>(ConstructPrivately());
- }
- MonoObject* ManagedSerializableList::getManagedInstance() const
- {
- if(mGCHandle != 0)
- return MonoUtil::getObjectFromGCHandle(mGCHandle);
- return nullptr;
- }
- void ManagedSerializableList::setFieldData(UINT32 arrayIdx, const SPtr<ManagedSerializableFieldData>& val)
- {
- if (mGCHandle != 0)
- {
- MonoObject* managedInstance = MonoUtil::getObjectFromGCHandle(mGCHandle);
- setFieldData(managedInstance, arrayIdx, val);
- }
- else
- mCachedEntries[arrayIdx] = val;
- }
- void ManagedSerializableList::setFieldData(MonoObject* obj, UINT32 arrayIdx, const SPtr<ManagedSerializableFieldData>& val)
- {
- mItemProp->setIndexed(obj, arrayIdx, val->getValue(mListTypeInfo->mElementType));
- }
- void ManagedSerializableList::addFieldDataInternal(const SPtr<ManagedSerializableFieldData>& val)
- {
- MonoObject* managedInstance = MonoUtil::getObjectFromGCHandle(mGCHandle);
- void* params[1];
- params[0] = val->getValue(mListTypeInfo->mElementType);
- mAddMethod->invoke(managedInstance, params);
- }
- SPtr<ManagedSerializableFieldData> ManagedSerializableList::getFieldData(UINT32 arrayIdx)
- {
- if (mGCHandle != 0)
- {
- MonoObject* managedInstance = MonoUtil::getObjectFromGCHandle(mGCHandle);
- MonoObject* obj = mItemProp->getIndexed(managedInstance, arrayIdx);
- return ManagedSerializableFieldData::create(mListTypeInfo->mElementType, obj);
- }
- else
- return mCachedEntries[arrayIdx];
- }
- void ManagedSerializableList::resize(UINT32 newSize)
- {
- if (mGCHandle != 0)
- {
- ScriptArray tempArray(mListTypeInfo->mElementType->getMonoClass(), newSize);
- UINT32 minSize = std::min(mNumElements, newSize);
- UINT32 dummy = 0;
- void* params[4];
- params[0] = &dummy;;
- params[1] = tempArray.getInternal();
- params[2] = &dummy;
- params[3] = &minSize;
- mCopyToMethod->invoke(getManagedInstance(), params);
- mClearMethod->invoke(getManagedInstance(), nullptr);
- params[0] = tempArray.getInternal();
- mAddRangeMethod->invoke(getManagedInstance(), params);
- }
- else
- {
- mCachedEntries.resize(newSize);
- }
- mNumElements = newSize;
- }
- void ManagedSerializableList::serialize()
- {
- if (mGCHandle == 0)
- return;
- mNumElements = getLengthInternal();
- mCachedEntries = Vector<SPtr<ManagedSerializableFieldData>>(mNumElements);
- for (UINT32 i = 0; i < mNumElements; i++)
- mCachedEntries[i] = getFieldData(i);
- // Serialize children
- for (auto& fieldEntry : mCachedEntries)
- fieldEntry->serialize();
- MonoUtil::freeGCHandle(mGCHandle);
- mGCHandle = 0;
- }
- MonoObject* ManagedSerializableList::deserialize()
- {
- MonoObject* managedInstance = createManagedInstance(mListTypeInfo, mNumElements);
- if (managedInstance == nullptr)
- return nullptr;
- MonoClass* listClass = MonoManager::instance().findClass(mListTypeInfo->getMonoClass());
- initMonoObjects(listClass);
- // Deserialize children
- for (auto& fieldEntry : mCachedEntries)
- fieldEntry->deserialize();
- UINT32 idx = 0;
- for (auto& entry : mCachedEntries)
- {
- setFieldData(managedInstance, idx, entry);
- idx++;
- }
- return managedInstance;
- }
- UINT32 ManagedSerializableList::getLengthInternal() const
- {
- MonoObject* managedInstance = MonoUtil::getObjectFromGCHandle(mGCHandle);
- MonoObject* length = mCountProp->get(managedInstance);
- if(length == nullptr)
- return 0;
- return *(UINT32*)MonoUtil::unbox(length);
- }
- void ManagedSerializableList::initMonoObjects(MonoClass* listClass)
- {
- mItemProp = listClass->getProperty("Item");
- mCountProp = listClass->getProperty("Count");
- mAddMethod = listClass->getMethod("Add", 1);
- mAddRangeMethod = listClass->getMethod("AddRange", 1);
- mClearMethod = listClass->getMethod("Clear");
- mCopyToMethod = listClass->getMethod("CopyTo", 4);
- }
- RTTITypeBase* ManagedSerializableList::getRTTIStatic()
- {
- return ManagedSerializableListRTTI::instance();
- }
- RTTITypeBase* ManagedSerializableList::getRTTI() const
- {
- return ManagedSerializableList::getRTTIStatic();
- }
- }
|