/*==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 . 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_SDL_inc #define PL_SDL_inc // // Code for the State Description Language (SDL) // #include #include "plSDLDescriptor.h" #include "pnFactory/plCreatable.h" #include "pnKeyedObject/plKey.h" #include "pnKeyedObject/plUoid.h" #include "plUnifiedTime/plUnifiedTime.h" namespace plSDL { typedef std::list DescriptorList; enum GeneralPurpose { kLatestVersion = -1, // used when requesting the latest version of a state descriptor kMaxListSize = 9999 // maximum size of var lists }; enum ContentsFlags // or saveFlags, 16 bits { kHasUoid = 0x1, kHasNotificationInfo = 0x2, kHasTimeStamp = 0x4, kSameAsDefault = 0x8, kHasDirtyFlag = 0x10, kWantTimeStamp = 0x20, kAddedVarLengthIO = 0x8000, // using to establish a new version in the header, can delete in 8/03 kHasMaximumValue= 0xffff, }; enum RWOptions { kDirtyOnly = 1<< 0, // write option kSkipNotificationInfo = 1<< 1, // read/write option kBroadcast = 1<< 2, // send option kWriteTimeStamps = 1<< 3, // write out time stamps kTimeStampOnRead = 1<< 4, // read: timestamp each var when it gets read. write: request that the reader timestamp the dirty vars. kTimeStampOnWrite = 1<< 5, // read: n/a. write: timestamp each var when it gets written. kKeepDirty = 1<< 6, // don't clear dirty flag on read kDontWriteDirtyFlag = 1<< 7, // write option. don't write var dirty flag. kMakeDirty = 1<< 8, // read/write: set dirty flag on var read/write. kDirtyNonDefaults = 1<< 9, // dirty the var if non default value. kForceConvert = 1<<10, // always try to convert rec to latest on read }; enum BehaviorFlags { kDisallowTimeStamping = 0x1, }; extern const plString kAgeSDLObjectName; void VariableLengthRead(hsStream* s, int size, int* val); void VariableLengthWrite(hsStream* s, int size, int val); }; class plStateVarNotificationInfo { private: plString fHintString; public: void SetHintString(const plString& c) { fHintString=c; } plString GetHintString() const { return fHintString; } void Read(hsStream* s, uint32_t readOptions); void Write(hsStream* s, uint32_t writeOptions) const; }; // // Base class for a state variable. // A state var is a var descriptor and it's value (contents) // class plSimpleStateVariable; class plSDStateVariable; class plStateVariable { public: enum Flags { kDirty = 0x1, // true when someone sets the value using Set(...), can be cleared after writing kUsed = 0x2 // true when it contains some value (either by Set(...) or Read() ) }; protected: uint32_t fFlags; plStateVarNotificationInfo fNotificationInfo; public: plStateVariable() : fFlags(0) {} virtual ~plStateVariable() {} plString GetName() const { return GetVarDescriptor()->GetName(); } bool IsNamed(const char* n) const { return (n && !GetName().CompareI(n)); } virtual int GetCount() const = 0; // conversion ops virtual plSimpleStateVariable* GetAsSimpleStateVar() = 0; virtual plSDStateVariable* GetAsSDStateVar() = 0; virtual plVarDescriptor* GetVarDescriptor() = 0; virtual const plVarDescriptor* GetVarDescriptor() const = 0; virtual void Alloc(int cnt=-1 /* -1 means don't change count */) = 0; virtual bool IsDirty() const { return (fFlags & kDirty) != 0; } virtual bool IsUsed() const { return (fFlags & kUsed) != 0; } void SetDirty(bool d) { if (d) fFlags |= kDirty; else fFlags &= ~kDirty; } void SetUsed(bool d) { if (d) fFlags |= kUsed; else fFlags &= ~kUsed; } virtual void SetFromDefaults(bool timeStampNow) = 0; virtual void TimeStamp( const plUnifiedTime & ut=plUnifiedTime::GetCurrent() ) = 0; virtual const plUnifiedTime& GetTimeStamp() const = 0; plStateVarNotificationInfo& GetNotificationInfo() { return fNotificationInfo; } const plStateVarNotificationInfo& GetNotificationInfo() const { return fNotificationInfo; } virtual void DumpToObjectDebugger(bool dirtyOnly, int level) const {} virtual void DumpToStream(hsStream* stream, bool dirtyOnly, int level) const {} // IO virtual bool ReadData(hsStream* s, float timeConvert, uint32_t readOptions); virtual bool WriteData(hsStream* s, float timeConvert, uint32_t writeOptions) const; }; // // Change Notifier. // When a plSimpleStateVariable changes it's value by more than the given delta value, // a notification msg will be sent to the objects that registered interest. // class plStateChangeNotifier { friend class plSimpleStateVariable; private: float fDelta; typedef std::vector KeyList; KeyList fKeys; // the objects to notify on change>delta static uint32_t fCurrentPlayerID; void IAddKey(plKey k); int IRemoveKey(plKey k); public: plStateChangeNotifier(); plStateChangeNotifier(float i, plKey k); void AddNotificationKey(plKey k) { IAddKey(k); } void AddNotificationKeys(KeyList keys); int RemoveNotificationKey(plKey k); // returns number of keys left after removal int RemoveNotificationKeys(KeyList keys); // returns number of keys left after removal void SendNotificationMsg(const plSimpleStateVariable* srcVar, const plSimpleStateVariable* dstVar, const plString& sdlName); bool GetValue(float* i) const; bool SetValue(float i); static uint32_t GetCurrentPlayerID() { return fCurrentPlayerID; } static void SetCurrentPlayerID(uint32_t p) { fCurrentPlayerID=p; } bool operator==(const plStateChangeNotifier &) const; }; // // A (non-nested) variable descriptor and its data contents. // class plUoid; class plKey; class plClientUnifiedTime; class plSimpleStateVariable : public plStateVariable { protected: union { int* fI; // array of int short* fS; // array of short uint8_t* fBy; // array of byte float* fF; // array of float double* fD; // array of double bool* fB; // array of bool plUoid* fU; // array of uoid plCreatable** fC; // array of plCreatable ptrs plVarDescriptor::String32* fS32; // array of strings plClientUnifiedTime* fT; // array of Times }; mutable plUnifiedTime fTimeStamp; // the last time the var was changed plSimpleVarDescriptor fVar; typedef std::vector StateChangeNotifiers; StateChangeNotifiers fChangeNotifiers; void IDeAlloc(); void IInit(); // initize vars void IVarSet(bool timeStampNow=false); // converter fxns bool IConvertFromBool(plVarDescriptor::Type newType); bool IConvertFromInt(plVarDescriptor::Type newType); bool IConvertFromByte(plVarDescriptor::Type newType); bool IConvertFromShort(plVarDescriptor::Type newType); bool IConvertFromFloat(plVarDescriptor::Type newType); bool IConvertFromDouble(plVarDescriptor::Type newType); bool IConvertFromString(plVarDescriptor::Type newType); bool IConvertFromRGB(plVarDescriptor::Type newType); bool IConvertFromRGBA(plVarDescriptor::Type newType); bool IConvertFromRGB8(plVarDescriptor::Type newType); bool IConvertFromRGBA8(plVarDescriptor::Type newType); bool IReadData(hsStream* s, float timeConvert, int idx, uint32_t readOptions); bool IWriteData(hsStream* s, float timeConvert, int idx, uint32_t writeOptions) const; public: plSimpleStateVariable() { IInit(); } plSimpleStateVariable(plVarDescriptor* vd) { IInit(); CopyFrom(vd); } ~plSimpleStateVariable() { IDeAlloc(); } // conversion ops plSimpleStateVariable* GetAsSimpleStateVar() { return this; } plSDStateVariable* GetAsSDStateVar() { return nil; } bool operator==(const plSimpleStateVariable &other) const; // assumes matching var descriptors void TimeStamp( const plUnifiedTime & ut=plUnifiedTime::GetCurrent() ); void CopyFrom(plVarDescriptor* v); void CopyData(const plSimpleStateVariable* other, uint32_t writeOptions=0); bool SetFromString(const plString& value, int idx, bool timeStampNow); // set value from string, type. return false on err plString GetAsString(int idx) const; bool ConvertTo(plSimpleVarDescriptor* toVar, bool force=false); // return false on err void Alloc(int cnt=-1 /* -1 means don't change count */); // alloc memory after setting type void Reset(); // setters bool Set(float v, int idx=0); bool Set(float* v, int idx=0); // floatVector bool Set(double v, int idx=0); bool Set(double* v, int idx=0); // doubleVector bool Set(int v, int idx=0); bool Set(int* v, int idx=0) { return Set(*v, idx); } // helper since there is no int vec type bool Set(uint8_t v, int idx=0); bool Set(uint8_t* v, int idx=0); // uint8_tVector bool Set(short v, int idx=0); bool Set(short* v, int idx=0) { return Set(*v, idx); } // helper since there is no short vec type bool Set(bool v, int idx=0); bool Set(bool* v, int idx=0) { return Set(*v, idx); } // helper since there is no bool vec type bool Set(const char* v, int idx=0); // string bool Set(const plKey& v, int idx=0); bool Set(plCreatable*, int idx=0); // only SDL generated by the server is allowed to use this type. void SetFromDefaults(bool timeStampNow); // getters bool Get(int* value, int idx=0) const; bool Get(short* value, int idx=0) const; bool Get(uint8_t* value, int idx=0) const; // returns uint8_t or uint8_tVector bool Get(float* value, int idx=0) const; // returns float or floatVector bool Get(double* value, int idx=0) const; // returns double or doubleVector bool Get(bool* value, int idx=0) const; bool Get(char value[], int idx=0) const; bool Get(plKey* value, int idx=0) const; bool Get(plCreatable** value, int idx=0) const; const plUnifiedTime& GetTimeStamp() const { return fTimeStamp; } // Special backdoor so the KI Manager can get the key name without having a ResMgr plString GetKeyName(int idx=0) const; int GetCount() const { return fVar.GetCount(); } // helper plVarDescriptor* GetVarDescriptor() { return &fVar; } plSimpleVarDescriptor* GetSimpleVarDescriptor() { return fVar.GetAsSimpleVarDescriptor(); } const plVarDescriptor* GetVarDescriptor() const { return &fVar; } const plSimpleVarDescriptor* GetSimpleVarDescriptor() const { return fVar.GetAsSimpleVarDescriptor(); } // State Change Notification void AddStateChangeNotification(plStateChangeNotifier& n); void RemoveStateChangeNotification(plKey notificationObj); // remove all with this key void RemoveStateChangeNotification(plStateChangeNotifier n); // remove ones which match void NotifyStateChange(const plSimpleStateVariable* other, const plString& sdlName); // send notification msg if necessary, internal use void DumpToObjectDebugger(bool dirtyOnly, int level) const; void DumpToStream(hsStream* stream, bool dirtyOnly, int level) const; // IO bool ReadData(hsStream* s, float timeConvert, uint32_t readOptions); bool WriteData(hsStream* s, float timeConvert, uint32_t writeOptions) const; }; // // A list of state data records, all of which are the same kind. // Corresponds to a SD var descriptor. // class plStateDataRecord; class plSDStateVariable : public plStateVariable { public: typedef std::vector ConstDataRecList; protected: typedef std::vector DataRecList; DataRecList fDataRecList; plSDVarDescriptor* fVarDescriptor; void IDeInit(); public: plSDStateVariable(plSDVarDescriptor* sdvd); ~plSDStateVariable(); // delete all records // conversion ops plSimpleStateVariable* GetAsSimpleStateVar() { return nil; } plSDStateVariable* GetAsSDStateVar() { return this; } bool operator==(const plSDStateVariable &other) const; // assumes matching var descriptors void ConvertTo(plSDStateVariable* otherSDVar, bool force=false); void CopyFrom(plSDStateVariable* other, uint32_t writeOptions=0); void UpdateFrom(plSDStateVariable* other, uint32_t writeOptions=0); void AddStateDataRecord(plStateDataRecord *sdr) { fDataRecList.push_back(sdr); SetDirty(true); SetUsed(true); } void InsertStateDataRecord(plStateDataRecord *sdr, int i) { fDataRecList[i] = sdr; SetDirty(true); SetUsed(true);} void SetFromDefaults(bool timeStampNow); void TimeStamp( const plUnifiedTime & ut=plUnifiedTime::GetCurrent() ); const plUnifiedTime& GetTimeStamp() const { static plUnifiedTime foo; return foo; } void Alloc(int cnt=-1 /* -1 means don't change count */); // wipe and re-create void Alloc(plSDVarDescriptor* sdvd, int cnt=-1); // wipe and re-create void Resize(int cnt); bool IsDirty() const; bool IsUsed() const; // getters plStateDataRecord* GetStateDataRecord(int i) { return fDataRecList[i]; } const plStateDataRecord* GetStateDataRecord(int i) const { return fDataRecList[i]; } int GetCount() const { return fDataRecList.size(); } int GetUsedCount() const; int GetDirtyCount() const; void GetUsedDataRecords(ConstDataRecList*) const; void GetDirtyDataRecords(ConstDataRecList*) const; plVarDescriptor* GetVarDescriptor() { return fVarDescriptor; } plSDVarDescriptor* GetSDVarDescriptor() { return fVarDescriptor->GetAsSDVarDescriptor(); } const plVarDescriptor* GetVarDescriptor() const { return fVarDescriptor; } const plSDVarDescriptor* GetSDVarDescriptor() const { return fVarDescriptor->GetAsSDVarDescriptor(); } void FlagNewerState(const plSDStateVariable&, bool respectAlwaysNew); void FlagAlwaysNewState(); void DumpToObjectDebugger(bool dirtyOnly, int level) const; void DumpToStream(hsStream* stream, bool dirtyOnly, int level) const; // IO bool ReadData(hsStream* s, float timeConvert, uint32_t readOptions); bool WriteData(hsStream* s, float timeConvert, uint32_t writeOptions) const; }; // // Contains the actual data contents and points to its associated descriptor // class plNetMsgSDLState; class plStateDataRecord : public plCreatable { public: typedef std::vector SimpleVarsList; typedef std::vector SDVarsList; enum Flags { kVolatile = 0x1 }; protected: typedef std::vector VarsList; const plStateDescriptor* fDescriptor; plUoid fAssocObject; // optional VarsList fVarsList; // list of variables VarsList fSDVarsList; // list of nested data records uint32_t fFlags; static const uint8_t kIOVersion; // I/O Version void IDeleteVarsList(VarsList& vars); void IInitDescriptor(const plString& name, int version); // or plSDL::kLatestVersion void IInitDescriptor(const plStateDescriptor* sd); void IReadHeader(hsStream* s); void IWriteHeader(hsStream* s) const; bool IConvertVar(plSimpleStateVariable* fromVar, plSimpleStateVariable* toVar, bool force); plStateVariable* IFindVar(const VarsList& vars, const plString& name) const; int IGetNumUsedVars(const VarsList& vars) const; int IGetUsedVars(const VarsList& varsOut, VarsList *varsIn) const; // build a list of vars that have data bool IHasUsedVars(const VarsList& vars) const; int IGetNumDirtyVars(const VarsList& vars) const; int IGetDirtyVars(const VarsList& varsOut, VarsList *varsIn) const; // build a list of vars that are dirty bool IHasDirtyVars(const VarsList& vars) const; public: CLASSNAME_REGISTER( plStateDataRecord ); GETINTERFACE_ANY( plStateDataRecord, plCreatable); plStateDataRecord(const plString& sdName, int version=plSDL::kLatestVersion); plStateDataRecord(plStateDescriptor* sd); plStateDataRecord(const plStateDataRecord &other, uint32_t writeOptions=0 ):fFlags(0) { CopyFrom(other, writeOptions); } plStateDataRecord() : fDescriptor(nil), fFlags(0) {} ~plStateDataRecord(); bool ConvertTo(plStateDescriptor* other, bool force=false ); bool operator==(const plStateDataRecord &other) const; // assumes matching state descriptors uint32_t GetFlags() const { return fFlags; } void SetFlags(uint32_t f) { fFlags =f; } plSimpleStateVariable* FindVar(const plString& name) const { return (plSimpleStateVariable*)IFindVar(fVarsList, name); } plSDStateVariable* FindSDVar(const plString& name) const { return (plSDStateVariable*)IFindVar(fSDVarsList, name); } plStateDataRecord& operator=(const plStateDataRecord& other) { CopyFrom(other); return *this; } void CopyFrom(const plStateDataRecord& other, uint32_t writeOptions=0); void UpdateFrom(const plStateDataRecord& other, uint32_t writeOptions=0); void SetFromDefaults(bool timeStampNow); void TimeStampDirtyVars(); int GetNumVars() const { return fVarsList.size(); } plSimpleStateVariable* GetVar(int i) const { return (plSimpleStateVariable*)fVarsList[i]; } int GetNumSDVars() const { return fSDVarsList.size(); } plSDStateVariable* GetSDVar(int i) const { return (plSDStateVariable*)fSDVarsList[i]; } // Used vars bool IsUsed() const { return (HasUsedVars() || HasUsedSDVars()); } int GetNumUsedVars() const { return IGetNumUsedVars(fVarsList); } int GetUsedVars(SimpleVarsList *vars) const { return IGetUsedVars(fVarsList, (VarsList*)vars); } // build a list of vars that have data bool HasUsedVars() const { return IHasUsedVars(fVarsList); } int GetNumUsedSDVars() const { return IGetNumUsedVars(fSDVarsList); } int GetUsedSDVars(SDVarsList *vars) const { return IGetUsedVars(fSDVarsList, (VarsList*)vars); } // build a list of SD vars that have data bool HasUsedSDVars() const { return IHasUsedVars(fSDVarsList); } // Dirty Vars bool IsDirty() const { return (HasDirtyVars() || HasDirtySDVars()); } int GetNumDirtyVars() const { return IGetNumDirtyVars(fVarsList); } int GetDirtyVars(SimpleVarsList *vars) const { return IGetDirtyVars(fVarsList, (VarsList*)vars); } // build a list of vars that are dirty bool HasDirtyVars() const { return IHasDirtyVars(fVarsList); } int GetNumDirtySDVars() const { return IGetNumDirtyVars(fSDVarsList); } int GetDirtySDVars(SDVarsList *vars) const { return IGetDirtyVars(fSDVarsList, (VarsList*)vars); }; // build a list of Sdvars that are dirty bool HasDirtySDVars() const { return IHasDirtyVars(fSDVarsList); } const plStateDescriptor* GetDescriptor() const { return fDescriptor; } void SetDescriptor(const plString& sdName, int version); plNetMsgSDLState* PrepNetMsg(float timeConvert, uint32_t writeOptions) const; // create/prep a net msg with this data void SetAssocObject(const plUoid& u) { fAssocObject=u; } // optional plUoid* GetAssocObject() { return &fAssocObject; } // optional const plUoid* GetAssocObject() const { return &fAssocObject; } // optional // utils void FlagDifferentState(const plStateDataRecord& other); // mark items which differ from 'other' as dirty void FlagNewerState(const plStateDataRecord& other, bool respectAlwaysNew=false); // mark items which are newer than 'other' as dirty void FlagAlwaysNewState(); // mark 'alwaysNew' items as dirty void DumpToObjectDebugger(const char* msg, bool dirtyOnly=false, int level=0) const; void DumpToStream(hsStream* stream, const char* msg, bool dirtyOnly=false, int level=0) const; // IO bool Read(hsStream* s, float timeConvert, uint32_t readOptions=0); void Write(hsStream* s, float timeConvert, uint32_t writeOptions=0) const; static bool ReadStreamHeader(hsStream* s, plString* name, int* version, plUoid* objUoid=nil); void WriteStreamHeader(hsStream* s, plUoid* objUoid=nil) const; }; // // Simple SDL parser // class plSDLMgr; class plSDLParser { private: bool IReadDescriptors() const; bool ILoadSDLFile(const plFileName& fileName) const; bool IParseVarDesc(const plFileName& fileName, hsStream* stream, char token[], plStateDescriptor*& curDesc, plVarDescriptor*& curVar) const; bool IParseStateDesc(const plFileName& fileName, hsStream* stream, char token[], plStateDescriptor*& curDesc) const; void DebugMsg(const plString& msg) const; hsDeprecated("plSDLParser::DebugMsg with format is deprecated -- use plFormat instead") void DebugMsg(const char* fmt, ...) const; void DebugMsgV(const char* fmt, va_list args) const; public: bool Parse() const; // reads sdl folder, creates descriptor list }; // // Holds, loads and unloads all state descriptors from sdl files. // Singleton. // class plNetApp; class plSDLMgr { friend class plSDLParser; private: plFileName fSDLDir; plSDL::DescriptorList fDescriptors; plNetApp* fNetApp; uint32_t fBehaviorFlags; void IDeleteDescriptors(plSDL::DescriptorList* dl); public: plSDLMgr(); ~plSDLMgr(); static plSDLMgr* GetInstance(); plStateDescriptor* FindDescriptor(const plString& name, int version, const plSDL::DescriptorList * dl=nil) const; // version or kLatestVersion const plSDL::DescriptorList * GetDescriptors( void ) const { return &fDescriptors;} void SetSDLDir(const plFileName& s) { fSDLDir=s; } plFileName GetSDLDir() const { return fSDLDir; } void SetNetApp(plNetApp* a) { fNetApp=a; } plNetApp* GetNetApp() const { return fNetApp; } bool Init( uint32_t behaviorFlags=0 ); // parse sdl folder void DeInit(); uint32_t GetBehaviorFlags() const { return fBehaviorFlags; } void SetBehaviorFlags(uint32_t v) { fBehaviorFlags=v; } bool AllowTimeStamping() const { return ! ( fBehaviorFlags&plSDL::kDisallowTimeStamping ); } // I/O - return # of bytes read/written int Write(hsStream* s, const plSDL::DescriptorList* dl=nil); // write descriptors to a stream int Read(hsStream* s, plSDL::DescriptorList* dl=nil); // read descriptors into provided list (use legacyList if nil) }; #endif // PL_SDL_inc