| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481 |
- //********************************** Banshee Engine (www.banshee3d.com) **************************************************//
- //**************** Copyright (c) 2016 Marko Pintera ([email protected]). All rights reserved. **********************//
- #include "Serialization/BsManagedSerializableDictionary.h"
- #include "RTTI/BsManagedSerializableDictionaryRTTI.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
- {
- ManagedSerializableDictionaryKeyValue::ManagedSerializableDictionaryKeyValue(const SPtr<ManagedSerializableFieldData>& key,
- const SPtr<ManagedSerializableFieldData>& value)
- :key(key), value(value)
- {
-
- }
- RTTITypeBase* ManagedSerializableDictionaryKeyValue::getRTTIStatic()
- {
- return ManagedSerializableDictionaryKeyValueRTTI::instance();
- }
- RTTITypeBase* ManagedSerializableDictionaryKeyValue::getRTTI() const
- {
- return ManagedSerializableDictionaryKeyValue::getRTTIStatic();
- }
- inline size_t ManagedSerializableDictionary::Hash::operator()(const SPtr<ManagedSerializableFieldData>& x) const
- {
- return x->getHash();
- }
- inline bool ManagedSerializableDictionary::Equals::operator()(const SPtr<ManagedSerializableFieldData>& a, const SPtr<ManagedSerializableFieldData>& b) const
- {
- return a->equals(b);
- }
- ManagedSerializableDictionary::Enumerator::Enumerator(const ManagedSerializableDictionary* parent)
- : mIteratorInitialized(false), mParent(parent)
- {
- MonoArray* keysArray = nullptr;
- MonoArray* valuesArray = nullptr;
- if(parent->mGCHandle != 0)
- {
- MonoObject* managedInstance = MonoUtil::getObjectFromGCHandle(parent->mGCHandle);
- mNumEntries = *(UINT32*)MonoUtil::unbox(parent->mCountProp->get(managedInstance));
- MonoObject* keyCollection = parent->mKeysProp->get(managedInstance);
- MonoObject* valueCollection = parent->mValuesProp->get(managedInstance);
- mKeyType = parent->mDictionaryTypeInfo->mKeyType->getMonoClass();
- mValueType = parent->mDictionaryTypeInfo->mValueType->getMonoClass();
- ScriptArray keys(mKeyType, mNumEntries);
- ScriptArray values(mValueType, mNumEntries);
- UINT32 offset = 0;
- void* keyParams[2] = { keys.getInternal(), &offset };
- parent->mKeysCopyTo->invoke(keyCollection, keyParams);
- void* valueParams[2] = { values.getInternal(), &offset };
- parent->mValuesCopyTo->invoke(valueCollection, valueParams);
- keysArray = keys.getInternal();
- valuesArray = values.getInternal();
- }
- else
- mNumEntries = (UINT32)parent->mCachedEntries.size();
- // Note: Handle needed since Enumerator will be on the stack? meaning the GC should be able to find the references.
- if(keysArray && valuesArray)
- {
- mKeysArrayHandle = MonoUtil::newGCHandle((MonoObject*)keysArray, false);
- mValuesArrayHandle = MonoUtil::newGCHandle((MonoObject*)valuesArray, false);
- }
- }
- ManagedSerializableDictionary::Enumerator::Enumerator(const Enumerator& other)
- : mNumEntries(other.mNumEntries), mIteratorInitialized(false), mParent(other.mParent)
- {
- if(other.mKeysArrayHandle != 0 && other.mValuesArrayHandle != 0)
- {
- MonoObject* keysArray = MonoUtil::getObjectFromGCHandle(other.mKeysArrayHandle);
- mKeysArrayHandle = MonoUtil::newGCHandle(keysArray, false);
- MonoObject* valuesArray = MonoUtil::getObjectFromGCHandle(other.mValuesArrayHandle);
- mValuesArrayHandle = MonoUtil::newGCHandle(valuesArray, false);
- mKeyType = other.mKeyType;
- mValueType = other.mValueType;
- }
- }
- ManagedSerializableDictionary::Enumerator::~Enumerator()
- {
- if(mKeysArrayHandle != 0)
- MonoUtil::freeGCHandle(mKeysArrayHandle);
- if(mValuesArrayHandle != 0)
- MonoUtil::freeGCHandle(mValuesArrayHandle);
- }
- ManagedSerializableDictionary::Enumerator&
- ManagedSerializableDictionary::Enumerator::operator=(const Enumerator& other)
- {
- mNumEntries = other.mNumEntries;
- mIteratorInitialized = false;
- mParent = other.mParent;
- mKeyType = nullptr;
- mValueType = nullptr;
- if(mKeysArrayHandle != 0)
- {
- MonoUtil::freeGCHandle(mKeysArrayHandle);
- mKeysArrayHandle = 0;
- }
- if(mValuesArrayHandle != 0)
- {
- MonoUtil::freeGCHandle(mValuesArrayHandle);
- mValuesArrayHandle = 0;
- }
- if(other.mKeysArrayHandle != 0 && other.mValuesArrayHandle != 0)
- {
- MonoObject* keysArray = MonoUtil::getObjectFromGCHandle(other.mKeysArrayHandle);
- mKeysArrayHandle = MonoUtil::newGCHandle(keysArray, false);
- MonoObject* valuesArray = MonoUtil::getObjectFromGCHandle(other.mValuesArrayHandle);
- mValuesArrayHandle = MonoUtil::newGCHandle(valuesArray, false);
- mKeyType = other.mKeyType;
- mValueType = other.mValueType;
- }
- return *this;
- }
- SPtr<ManagedSerializableFieldData> ManagedSerializableDictionary::Enumerator::getKey() const
- {
- if (mKeysArrayHandle != 0)
- {
- MonoArray* keysArray = (MonoArray*)MonoUtil::getObjectFromGCHandle(mKeysArrayHandle);
- ScriptArray keys(keysArray);
- if(mCurrentIdx != (UINT32)-1)
- {
- void* val = (void*)keys.getRaw(mCurrentIdx, keys.elementSize());
- MonoObject* obj = nullptr;
- if (MonoUtil::isValueType(mKeyType))
- {
- if (val != nullptr)
- obj = MonoUtil::box(mKeyType, val);
- }
- else
- obj = *(MonoObject**)val;
- return ManagedSerializableFieldData::create(mParent->mDictionaryTypeInfo->mKeyType, obj);
- }
- else
- return nullptr;
- }
- else
- {
- return mCachedIter->first;
- }
- }
- SPtr<ManagedSerializableFieldData> ManagedSerializableDictionary::Enumerator::getValue() const
- {
- if (mValuesArrayHandle != 0)
- {
- MonoArray* valuesArray = (MonoArray*)MonoUtil::getObjectFromGCHandle(mValuesArrayHandle);
- ScriptArray values(valuesArray);
- if(mCurrentIdx != (UINT32)-1)
- {
- void* val = (void*)values.getRaw(mCurrentIdx, values.elementSize());
- MonoObject* obj = nullptr;
- if (MonoUtil::isValueType(mValueType))
- {
- if (val != nullptr)
- obj = MonoUtil::box(mValueType, val);
- }
- else
- obj = *(MonoObject**)val;
- return ManagedSerializableFieldData::create(mParent->mDictionaryTypeInfo->mValueType, obj);
- }
- else
- return nullptr;
- }
- else
- {
- return mCachedIter->second;
- }
- }
- bool ManagedSerializableDictionary::Enumerator::moveNext()
- {
- if (mKeysArrayHandle != 0 && mValuesArrayHandle != 0)
- {
- if((mCurrentIdx + 1) < mNumEntries)
- {
- mCurrentIdx++;
- return true;
- }
- return false;
- }
- else
- {
- if (!mIteratorInitialized)
- {
- mCachedIter = mParent->mCachedEntries.begin();
- mIteratorInitialized = true;
- }
- else
- ++mCachedIter;
- return mCachedIter != mParent->mCachedEntries.end();
- }
- }
- ManagedSerializableDictionary::ManagedSerializableDictionary(const ConstructPrivately& dummy)
- { }
- ManagedSerializableDictionary::ManagedSerializableDictionary(const ConstructPrivately& dummy, const SPtr<ManagedSerializableTypeInfoDictionary>& typeInfo, MonoObject* managedInstance)
- : mDictionaryTypeInfo(typeInfo)
- {
- mGCHandle = MonoUtil::newGCHandle(managedInstance, false);
- MonoClass* dictClass = MonoManager::instance().findClass(MonoUtil::getClass(managedInstance));
- if (dictClass == nullptr)
- return;
- initMonoObjects(dictClass);
- }
- ManagedSerializableDictionary::~ManagedSerializableDictionary()
- {
- if(mGCHandle != 0)
- {
- MonoUtil::freeGCHandle(mGCHandle);
- mGCHandle = 0;
- }
- }
- SPtr<ManagedSerializableDictionary> ManagedSerializableDictionary::createFromExisting(MonoObject* managedInstance,
- const SPtr<ManagedSerializableTypeInfoDictionary>& typeInfo)
- {
- if(managedInstance == nullptr)
- return nullptr;
- String elementNs;
- String elementTypeName;
- MonoUtil::getClassName(managedInstance, elementNs, elementTypeName);
- String fullName = elementNs + "." + elementTypeName;
- if(ScriptAssemblyManager::instance().getSystemGenericDictionaryClass()->getFullName() != fullName)
- return nullptr;
- return bs_shared_ptr_new<ManagedSerializableDictionary>(ConstructPrivately(), typeInfo, managedInstance);
- }
- SPtr<ManagedSerializableDictionary> ManagedSerializableDictionary::createNew(const SPtr<ManagedSerializableTypeInfoDictionary>& typeInfo)
- {
- return bs_shared_ptr_new<ManagedSerializableDictionary>(ConstructPrivately(), typeInfo, createManagedInstance(typeInfo));
- }
- MonoObject* ManagedSerializableDictionary::createManagedInstance(const SPtr<ManagedSerializableTypeInfoDictionary>& typeInfo)
- {
- if (!typeInfo->isTypeLoaded())
- return nullptr;
- ::MonoClass* dictionaryMonoClass = typeInfo->getMonoClass();
- MonoClass* dictionaryClass = MonoManager::instance().findClass(dictionaryMonoClass);
- if (dictionaryClass == nullptr)
- return nullptr;
- return dictionaryClass->createInstance();
- }
- SPtr<ManagedSerializableDictionary> ManagedSerializableDictionary::createEmpty()
- {
- return bs_shared_ptr_new<ManagedSerializableDictionary>(ConstructPrivately());
- }
- MonoObject* ManagedSerializableDictionary::getManagedInstance() const
- {
- if(mGCHandle != 0)
- return MonoUtil::getObjectFromGCHandle(mGCHandle);
- return nullptr;
- }
- void ManagedSerializableDictionary::serialize()
- {
- if (mGCHandle == 0)
- return;
- MonoObject* managedInstance = MonoUtil::getObjectFromGCHandle(mGCHandle);
- MonoClass* dictionaryClass = MonoManager::instance().findClass(MonoUtil::getClass(managedInstance));
- if (dictionaryClass == nullptr)
- return;
- initMonoObjects(dictionaryClass);
- mCachedEntries.clear();
- Enumerator enumerator = getEnumerator();
- while (enumerator.moveNext())
- {
- SPtr<ManagedSerializableFieldData> key = enumerator.getKey();
- mCachedEntries.insert(std::make_pair(key, enumerator.getValue()));
- }
- // Serialize children
- for (auto& fieldEntry : mCachedEntries)
- {
- fieldEntry.first->serialize();
- fieldEntry.second->serialize();
- }
- MonoUtil::freeGCHandle(mGCHandle);
- mGCHandle = 0;
- }
- MonoObject* ManagedSerializableDictionary::deserialize()
- {
- MonoObject* managedInstance = createManagedInstance(mDictionaryTypeInfo);
- if (managedInstance == nullptr)
- return nullptr;
- ::MonoClass* dictionaryMonoClass = mDictionaryTypeInfo->getMonoClass();
- MonoClass* dictionaryClass = MonoManager::instance().findClass(dictionaryMonoClass);
- if (dictionaryClass == nullptr)
- return nullptr;
- initMonoObjects(dictionaryClass);
- // Deserialize children
- for (auto& fieldEntry : mCachedEntries)
- {
- fieldEntry.first->deserialize();
- fieldEntry.second->deserialize();
- }
- UINT32 idx = 0;
- for (auto& entry : mCachedEntries)
- {
- setFieldData(managedInstance, entry.first, entry.second);
- idx++;
- }
- return managedInstance;
- }
- SPtr<ManagedSerializableFieldData> ManagedSerializableDictionary::getFieldData(const SPtr<ManagedSerializableFieldData>& key)
- {
- if (mGCHandle != 0)
- {
- MonoObject* value = nullptr;
- void* params[2];
- params[0] = key->getValue(mDictionaryTypeInfo->mKeyType);
- params[1] = &value;
- MonoObject* managedInstance = MonoUtil::getObjectFromGCHandle(mGCHandle);
- mTryGetValueMethod->invoke(managedInstance, params);
- MonoObject* boxedValue = value;
- ::MonoClass* valueTypeClass = mDictionaryTypeInfo->mValueType->getMonoClass();
- if (MonoUtil::isValueType(valueTypeClass))
- {
- if (value != nullptr)
- boxedValue = MonoUtil::box(valueTypeClass, &value);
- }
- return ManagedSerializableFieldData::create(mDictionaryTypeInfo->mValueType, boxedValue);
- }
- else
- {
- return mCachedEntries[key];
- }
- }
- void ManagedSerializableDictionary::setFieldData(const SPtr<ManagedSerializableFieldData>& key, const SPtr<ManagedSerializableFieldData>& val)
- {
- if (mGCHandle != 0)
- {
- MonoObject* managedInstance = MonoUtil::getObjectFromGCHandle(mGCHandle);
- setFieldData(managedInstance, key, val);
- }
- else
- {
- mCachedEntries[key] = val;
- }
- }
- void ManagedSerializableDictionary::setFieldData(MonoObject* obj, const SPtr<ManagedSerializableFieldData>& key, const SPtr<ManagedSerializableFieldData>& val)
- {
- void* params[2];
- params[0] = key->getValue(mDictionaryTypeInfo->mKeyType);
- params[1] = val->getValue(mDictionaryTypeInfo->mValueType);
- mAddMethod->invoke(obj, params);
- }
- void ManagedSerializableDictionary::removeFieldData(const SPtr<ManagedSerializableFieldData>& key)
- {
- if (mGCHandle != 0)
- {
- void* params[1];
- params[0] = key->getValue(mDictionaryTypeInfo->mKeyType);
- MonoObject* managedInstance = MonoUtil::getObjectFromGCHandle(mGCHandle);
- mRemoveMethod->invoke(managedInstance, params);
- }
- else
- {
- auto findIter = mCachedEntries.find(key);
- if (findIter != mCachedEntries.end())
- mCachedEntries.erase(findIter);
- }
- }
- bool ManagedSerializableDictionary::contains(const SPtr<ManagedSerializableFieldData>& key) const
- {
- if (mGCHandle != 0)
- {
- void* params[1];
- params[0] = key->getValue(mDictionaryTypeInfo->mKeyType);
- MonoObject* managedInstance = MonoUtil::getObjectFromGCHandle(mGCHandle);
- MonoObject* returnVal = mContainsKeyMethod->invoke(managedInstance, params);
- return *(bool*)MonoUtil::unbox(returnVal);
- }
- else
- return mCachedEntries.find(key) != mCachedEntries.end();
- }
- ManagedSerializableDictionary::Enumerator ManagedSerializableDictionary::getEnumerator() const
- {
- return Enumerator(this);
- }
- void ManagedSerializableDictionary::initMonoObjects(MonoClass* dictionaryClass)
- {
- mAddMethod = dictionaryClass->getMethod("Add", 2);
- mRemoveMethod = dictionaryClass->getMethod("Remove", 1);
- mTryGetValueMethod = dictionaryClass->getMethod("TryGetValue", 2);
- mContainsKeyMethod = dictionaryClass->getMethod("ContainsKey", 1);
- mCountProp = dictionaryClass->getProperty("Count");
- mKeysProp = dictionaryClass->getProperty("Keys");
- mValuesProp = dictionaryClass->getProperty("Values");
- MonoClass* keyCollectionClass = mKeysProp->getReturnType();
- mKeysCopyTo = keyCollectionClass->getMethod("CopyTo", 2);
- MonoClass* valueCollectionClass = mValuesProp->getReturnType();
- mValuesCopyTo = valueCollectionClass->getMethod("CopyTo", 2);
- }
- RTTITypeBase* ManagedSerializableDictionary::getRTTIStatic()
- {
- return ManagedSerializableDictionaryRTTI::instance();
- }
- RTTITypeBase* ManagedSerializableDictionary::getRTTI() const
- {
- return ManagedSerializableDictionary::getRTTIStatic();
- }
- }
|