/*==LICENSE==* CyanWorlds.com Engine - MMOG client, server and tools Copyright (C) 2011 Cyan Worlds, Inc. This program is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program. If not, see <http://www.gnu.org/licenses/>. Additional permissions under GNU GPL version 3 section 7 If you modify this Program, or any covered work, by linking or combining it with any of RAD Game Tools Bink SDK, Autodesk 3ds Max SDK, NVIDIA PhysX SDK, Microsoft DirectX SDK, OpenSSL library, Independent JPEG Group JPEG library, Microsoft Windows Media SDK, or Apple QuickTime SDK (or a modified version of those libraries), containing parts covered by the terms of the Bink SDK EULA, 3ds Max EULA, PhysX SDK EULA, DirectX SDK EULA, OpenSSL and SSLeay licenses, IJG JPEG Library README, Windows Media SDK EULA, or QuickTime SDK EULA, the licensors of this Program grant you additional permission to convey the resulting work. Corresponding Source for a non-source form of such a combination shall include the source code for the parts of OpenSSL and IJG JPEG Library used as well as that of the covered work. You can contact Cyan Worlds, Inc. by email legal@cyan.com or by snail mail at: Cyan Worlds, Inc. 14617 N Newport Hwy Mead, WA 99021 *==LICENSE==*/ #ifndef PL_SYNCHEDVALUE_inc #define PL_SYNCHEDVALUE_inc #include "plSynchedObject.h" #ifdef USE_SYNCHED_VALUES #include "hsTypes.h" #include "hsStream.h" #include "hsBitVector.h" #include "hsMemory.h" // // Defines a class for variable types which need to automatically be synchronized // (replicated) over the network. // current size is 8 bytes // // // ----------------------------------------------------- // SYNCHED VALUE BASE CLASS // ----------------------------------------------------- // class hsResMgr; class plSceneNode; class plSceneObject; class plCoordinateInterface; class plSynchedValueBase { public: enum Flags // 16 bits { kValueIsDirty = 0x1, kValueSendOnlyOnce = 0x2, // perm flag kValueHasBeenSent = 0x4, // perm flag kRegistered = 0x8, // perm flag kDontDirty = 0x10 // perm flag }; protected: Int16 fSynchedObjectAddrOffset; // this could represent dwords instead of byte offsets UInt16 fFlags; void IConstruct() // too bad this can't be virtual (because it's called from ctor) { // The synchMgr for the class that owns us is constructed first and the staticMgr // is set to his address so we can automatically get at it during construction. fFlags=0; Int32 off = (Int32)plSynchedObject::GetStaticSynchedObject() - (Int32)this; if ( hsABS(off) < (1<<(sizeof(fSynchedObjectAddrOffset)<<3)) ) fSynchedObjectAddrOffset = (Int16)off; else fSynchedObjectAddrOffset=-1; } hsBool32 IOKToDirty() { if (fFlags & (kDontDirty | kValueIsDirty)) return false; return GetSynchedObject() ? GetSynchedObject()->IOKToDirty() : false; } virtual void ISaveOrLoad(hsBool32 save, hsStream* stream, hsResMgr* mgr) = 0; // save/load methods for different types const plKey ISaveOrLoad(const plKey key, hsBool32 save, hsStream* stream, hsResMgr* mgr); hsKeyedObject* ISaveOrLoad(hsKeyedObject* obj, hsBool32 save, hsStream* stream, hsResMgr* mgr); plSceneNode* ISaveOrLoad(plSceneNode* obj, hsBool32 save, hsStream* stream, hsResMgr* mgr); plSceneObject* ISaveOrLoad(plSceneObject* obj, hsBool32 save, hsStream* stream, hsResMgr* mgr); Int32 ISaveOrLoad(Int32 v, hsBool32 save, hsStream* stream, hsResMgr* mgr); UInt32 ISaveOrLoad(UInt32 v, hsBool32 save, hsStream* stream, hsResMgr* mgr); bool ISaveOrLoad(bool v, hsBool32 save, hsStream* stream, hsResMgr* mgr); int ISaveOrLoad(int v, hsBool32 save, hsStream* stream, hsResMgr* mgr); // or hsBool32 hsScalar ISaveOrLoad(hsScalar v, hsBool32 save, hsStream* stream, hsResMgr* mgr); double ISaveOrLoad(double v, hsBool32 save, hsStream* stream, hsResMgr* mgr); hsBitVector ISaveOrLoad(hsBitVector& v, hsBool32 save, hsStream* stream, hsResMgr* mgr); plCoordinateInterface* ISaveOrLoad(const plCoordinateInterface* cInt, hsBool32 save, hsStream* stream, hsResMgr* mgr); public: plSynchedValueBase() { IConstruct(); } virtual ~plSynchedValueBase() {} // getters virtual plSynchedObject* GetSynchedObject() { hsAssert(fSynchedObjectAddrOffset!=-1, "invalid synchedObject address offset"); plSynchedObject* so = fSynchedObjectAddrOffset == -1 ? nil : (plSynchedObject*)((Int32)this+fSynchedObjectAddrOffset); if (!(fFlags & kRegistered) && so) { so->RegisterSynchedValue(this); fFlags |= kRegistered; } return so; } UInt16 GetFlags() { return fFlags; } // setters void SetFlags(UInt16 f) { fFlags=f; } void MakeDirty() { SetFlags(GetFlags() | kValueIsDirty); } void MakeClean() { SetFlags(GetFlags() & ~kValueIsDirty); } void DirtyIfNecessary() { if (IOKToDirty()) { MakeDirty(); // dirty value if (GetSynchedObject()) GetSynchedObject()->DirtySynchState(nil, 0); // dirty owner } } // save/load static void Save(plSynchedValueBase& obj, hsStream* stream, hsResMgr* mgr) { obj.ISaveOrLoad(true, stream, mgr); } static void Load(plSynchedValueBase& obj, hsStream* stream, hsResMgr* mgr) { obj.ISaveOrLoad(false, stream, mgr); } }; // // ----------------------------------- // SYNCHED VALUE TEMPLATE // ----------------------------------- // template <class T> class plSynchedValue : public plSynchedValueBase { protected: T fValue; void ISaveOrLoad(hsBool32 save, hsStream* stream, hsResMgr* mgr) { fValue=(T)plSynchedValueBase::ISaveOrLoad(fValue, save, stream, mgr); } // default method public: plSynchedValue() {} plSynchedValue(const T& v) : plSynchedValueBase() { fValue=v; } plSynchedValue(const plSynchedValue<T>& pRHS) : plSynchedValueBase() { fValue=pRHS.GetValue(); } // copy ctor // conversion operators operator T() const { return fValue; } T* operator &() const { return &fValue; } T* operator &() { return &fValue; } // equality hsBool32 operator==(const T& other) const { return fValue==(T)other; } hsBool32 operator!=(const T& other) const { return !(*this == other); } // other operators T operator++() { DirtyIfNecessary(); return ++fValue; } T operator++(int) { DirtyIfNecessary(); return fValue++; } // postfix T operator--() { DirtyIfNecessary(); return --fValue; } T operator--(int) { DirtyIfNecessary(); return fValue--; } // postfix T operator+=(const T& other) { return SetValue(fValue+other); } T operator*=(const T& other) { return SetValue(fValue*other); } T operator/=(const T& other) { return SetValue(fValue/other); } // these return reference in the event they are bitvector types T& operator&=(const T& other) { return SetValue(fValue&other); } T& operator|=(const T& other) { return SetValue(fValue|other); } T& operator^=(const T& other) { return SetValue(fValue^other); } T& operator-=(const T& other) { return SetValue(fValue-other); } const T& operator=(const T& v){ return SetValue(v); } #if HS_BUILD_FOR_WIN32 #pragma warning( push ) #pragma warning( disable : 4284 ) // disable annoying warnings in release build for non pointer types #endif // for pointer types, which are allowed to change the object pointed to T operator->(void) { return fValue; } #if HS_BUILD_FOR_WIN32 #pragma warning( pop ) #endif // asignment, can use instead of setvalue const T& operator=(const plSynchedValue<T>& pRHS ) { return SetValue(pRHS.GetValue()); } // setters T& SetValue(const T& v) // return true if changed value { if (v != fValue) // dont dirty unless value changes { fValue=v; DirtyIfNecessary(); } return fValue; } // getters const T& GetValue() const { return fValue; } // for hsBitVector hsBool32 IsBitSet(UInt32 which) const { return fValue.IsBitSet(which); } hsBool32 SetBit(UInt32 which, hsBool32 on = true) { hsBool32 bitSet = IsBitSet(which); if ( (on && !bitSet) || (!on && bitSet) ) DirtyIfNecessary(); return fValue.SetBit(which, on); } void Read(hsStream* s) { fValue.Read(s); } void Write(hsStream* s) const { fValue.Write(s); } void Clear() { DirtyIfNecessary(); fValue.Clear(); } hsBool32 ClearBit(UInt32 which) { if (fValue.IsBitSet(which)) DirtyIfNecessary(); return fValue.ClearBit(which); } void Reset() { if (fValue.GetSize()!=0) DirtyIfNecessary(); fValue.Reset(); } hsBool32 ToggleBit(UInt32 which) { DirtyIfNecessary(); return fValue.ToggleBit(which); } UInt32 GetSize() { return fValue.GetSize(); } }; ////////////////////////////////////// // Synched Value Friend - allows a synched value to be contained // in an object which is not a synchedObject. Uses a pointer instead // of an computer the addr offset of it's associated synchedObject. // This one is 4 bytes bigger than regular synched values. ////////////////////////////////////// template <class T> class plSynchedValueFriend : public plSynchedValue<T> { protected: plSynchedObject** fSynchedObject; public: plSynchedValueFriend() : fSynchedObject(nil) { } // this is explicit so it won't be invoked instead of operator()= explicit plSynchedValueFriend(const T& v) : plSynchedValue<T>(v),fSynchedObject(nil) { } plSynchedValueFriend(const plSynchedValueFriend<T>& pRHS) : plSynchedValue<T>(pRHS) { fSynchedObject = pRHS.fSynchedObject; } ~plSynchedValueFriend() { if (GetSynchedObject()) GetSynchedObject()->RemoveSynchedValue(this); } // this isn't inherited for some reason const T& operator=(const T& v) { return SetValue(v); } plSynchedObject* GetSynchedObject() { hsAssert(fSynchedObject, "nil synched object, need to SetSynchedObjectPtrAddr?"); if (*fSynchedObject && !(fFlags & kRegistered)) { (*fSynchedObject)->RegisterSynchedValueFriend(this); fFlags |= kRegistered; } return *fSynchedObject; } void SetSynchedObjectPtrAddr(plSynchedObject** so) { hsAssert(!(fFlags & kRegistered), "SynchedValueFriend already registered?"); fSynchedObject=so; } }; ///////////////////////////////////// // Synched TArray Template ///////////////////////////////////// #include "hsTemplates.h" template <class T> class plSynchedTArray : public plSynchedValueBase { private: hsTArray<T> fValueList; void ISaveOrLoad(hsBool32 save, hsStream* stream, hsResMgr* mgr); public: enum { kMissingIndex = hsTArray<T>::kMissingIndex }; plSynchedTArray() {} ~plSynchedTArray() {} // conversion operators operator T() const { return fValueList; } // common methods const T& operator[](int i) const { return Get(i); } const T& Get(int i) const { return fValueList.Get(i); } void Set(int i, const T& item) { if (fValueList[i] != item) DirtyIfNecessary(); fValueList[i]=item; } void Append(const T& item) { fValueList.Append(item); DirtyIfNecessary(); } T* Insert(int index) { fValueList.Insert(index); DirtyIfNecessary(); } void Remove(int index) { fValueList.Remove(index); DirtyIfNecessary(); } int Count() const { return fValueList.Count(); } int GetCount() const { return Count(); } void Reset() { if (fValueList.GetCount() != 0) DirtyIfNecessary(); fValueList.Reset(); } void SetCountAndZero(int count) { if (count || GetCount()) DirtyIfNecessary(); fValueList.SetCountAndZero(count); } void SetCount(int count) { if (count || GetCount()) DirtyIfNecessary(); fValueList.SetCount(count); } void ExpandAndZero(int count) { if (count || GetCount()) DirtyIfNecessary(); fValueList.ExpandAndZero(count); } int Find(const T& item) const { return fValueList.Find(item); } T* Push() { DirtyIfNecessary(); return fValueList.Push(); } void Push(const T& item) { DirtyIfNecessary(); return fValueList.Push(item); } T Pop() { DirtyIfNecessary(); return fValueList.Pop(); } const T& Peek() const { return fValue.Peek(); } T* DetachArray() { DirtyIfNecessary(); return fValueList.DetachArray(); } T* AcquireArray() { DirtyIfNecessary(); return fValueList.AcquireArray(); } }; // // inlines // template <class T> inline void plSynchedTArray<T>::ISaveOrLoad(hsBool32 save, hsStream* stream, hsResMgr* mgr) { if (save) { // write out size of array Int32 i, num = fValueList.GetCount(); stream->WriteSwap(num); for(i=0;i<num;i++) { plSynchedValueBase::ISaveOrLoad(fValueList[i], save, stream, mgr); } } else { // clear array fValueList.Reset(); // read in size of array Int32 i, num; stream->ReadSwap(&num); for(i=0;i<num;i++) { T v; HSMemory::ClearMemory(&v, sizeof(v)); v=(T)plSynchedValueBase::ISaveOrLoad(v, save, stream, mgr); fValueList.Append(v); } } } ////////////////////////////////////// // Variation on synchedTArray. See plSynchedValueFriend above for more info ////////////////////////////////////// template <class T> class plSynchedTArrayFriend : public plSynchedTArray<T> { protected: plSynchedObject** fSynchedObject; public: plSynchedTArrayFriend() : fSynchedObject(nil) { } plSynchedObject* GetSynchedObject() { hsAssert(fSynchedObject, "nil synched object, need to SetSynchedObjectPtrAddr?"); if (*fSynchedObject && !(fFlags & kRegistered)) { (*fSynchedObject)->RegisterSynchedValueFriend(this); fFlags |= kRegistered; } return *fSynchedObject; } void SetSynchedObjectPtrAddr(plSynchedObject** so) { hsAssert(!(fFlags & kRegistered), "SynchedValueTArrayFriend already registered?"); fSynchedObject=so; } #if 0 // // redefine operators since they are not inherited // // conversion operators operator T() const { return fValueList; } // common methods const T& operator[](int i) const { return Get(i); } #endif }; #endif // USE_SYNCHED_VALUES #endif // PL_SYNCHEDVALUE_inc