/[svn]/libgig/trunk/src/Serialization.cpp
ViewVC logotype

Diff of /libgig/trunk/src/Serialization.cpp

Parent Directory Parent Directory | Revision Log Revision Log | View Patch Patch

revision 3153 by schoenebeck, Sat May 6 13:43:43 2017 UTC revision 3169 by schoenebeck, Wed May 10 21:17:10 2017 UTC
# Line 30  Line 30 
30    
31  #include "helper.h"  #include "helper.h"
32    
33    #define LIBGIG_EPOCH_TIME ((time_t)0)
34    
35  namespace Serialization {  namespace Serialization {
36    
37      // *************** DataType ***************      // *************** DataType ***************
38      // *      // *
39    
40      static UID _createNullUID() {      static UID _createNullUID() {
41          return (UID) { NULL, 0 };          const UID uid = { NULL, 0 };
42            return uid;
43      }      }
44    
45      const UID NO_UID = _createNullUID();      const UID NO_UID = _createNullUID();
# Line 277  namespace Serialization { Line 280  namespace Serialization {
280          m_operation = OPERATION_NONE;          m_operation = OPERATION_NONE;
281          m_root = NO_UID;          m_root = NO_UID;
282          m_isModified = false;          m_isModified = false;
283            m_timeCreated = m_timeModified = LIBGIG_EPOCH_TIME;
284      }      }
285    
286      Archive::Archive(const RawData& data) {      Archive::Archive(const RawData& data) {
287          m_operation = OPERATION_NONE;          m_operation = OPERATION_NONE;
288          m_root = NO_UID;          m_root = NO_UID;
289          m_isModified = false;          m_isModified = false;
290            m_timeCreated = m_timeModified = LIBGIG_EPOCH_TIME;
291          decode(m_rawData);          decode(m_rawData);
292      }      }
293    
# Line 290  namespace Serialization { Line 295  namespace Serialization {
295          m_operation = OPERATION_NONE;          m_operation = OPERATION_NONE;
296          m_root = NO_UID;          m_root = NO_UID;
297          m_isModified = false;          m_isModified = false;
298            m_timeCreated = m_timeModified = LIBGIG_EPOCH_TIME;
299          decode(data, size);          decode(data, size);
300      }      }
301    
# Line 311  namespace Serialization { Line 317  namespace Serialization {
317          return _encodeBlob(s);          return _encodeBlob(s);
318      }      }
319    
320        static String _encode(const time_t& time) {
321            return _encodeBlob(ToString(time));
322        }
323    
324      static String _encode(const DataType& type) {      static String _encode(const DataType& type) {
325          String s;          String s;
326          s += _encodeBlob(type.baseTypeName());          s += _encodeBlob(type.baseTypeName());
# Line 392  namespace Serialization { Line 402  namespace Serialization {
402          return s;          return s;
403      }      }
404    
405        template<typename T>
406        static T _primitiveObjectValueToNumber(const Object& obj) {
407            T value = 0;
408            const DataType& type = obj.type();
409            const ID& id = obj.uid().id;
410            void* ptr = obj.m_data.empty() ? (void*)id : (void*)&obj.m_data[0];
411            if (!obj.m_data.empty())
412                assert(type.size() == obj.m_data.size());
413            if (type.isPrimitive() && !type.isPointer()) {
414                if (type.isInteger() || type.isEnum()) {
415                    if (type.isSigned()) {
416                        if (type.size() == 1)
417                            value = (T)*(int8_t*)ptr;
418                        else if (type.size() == 2)
419                            value = (T)*(int16_t*)ptr;
420                        else if (type.size() == 4)
421                            value = (T)*(int32_t*)ptr;
422                        else if (type.size() == 8)
423                            value = (T)*(int64_t*)ptr;
424                        else
425                            assert(false /* unknown signed int type size */);
426                    } else {
427                        if (type.size() == 1)
428                            value = (T)*(uint8_t*)ptr;
429                        else if (type.size() == 2)
430                            value = (T)*(uint16_t*)ptr;
431                        else if (type.size() == 4)
432                            value = (T)*(uint32_t*)ptr;
433                        else if (type.size() == 8)
434                            value = (T)*(uint64_t*)ptr;
435                        else
436                            assert(false /* unknown unsigned int type size */);
437                    }
438                } else if (type.isReal()) {
439                    if (type.size() == sizeof(float))
440                        value = (T)*(float*)ptr;
441                    else if (type.size() == sizeof(double))
442                        value = (T)*(double*)ptr;
443                    else
444                        assert(false /* unknown floating point type */);
445                } else if (type.isBool()) {
446                    value = (T)*(bool*)ptr;
447                } else {
448                    assert(false /* unknown primitive type */);
449                }
450            }
451            return value;
452        }
453    
454      static String _encodePrimitiveValue(const Object& obj) {      static String _encodePrimitiveValue(const Object& obj) {
455          return _encodeBlob( _primitiveObjectValueToString(obj) );          return _encodeBlob( _primitiveObjectValueToString(obj) );
456      }      }
# Line 426  namespace Serialization { Line 485  namespace Serialization {
485          s += _encodeBlob(ToString(ENCODING_FORMAT_MINOR_VERSION));          s += _encodeBlob(ToString(ENCODING_FORMAT_MINOR_VERSION));
486          s += _encode(m_root);          s += _encode(m_root);
487          s += _encode(m_allObjects);          s += _encode(m_allObjects);
488            s += _encodeBlob(m_name);
489            s += _encodeBlob(m_comment);
490            s += _encode(m_timeCreated);
491            s += _encode(m_timeModified);
492          return _encodeBlob(s);          return _encodeBlob(s);
493      }      }
494    
495      void Archive::encode() {      void Archive::encode() {
496          m_rawData.clear();          m_rawData.clear();
497          String s = MAGIC_START;          String s = MAGIC_START;
498            m_timeModified = time(NULL);
499            if (m_timeCreated == LIBGIG_EPOCH_TIME)
500                m_timeCreated = m_timeModified;
501          s += _encodeRootBlob();          s += _encodeRootBlob();
502          m_rawData.resize(s.length() + 1);          m_rawData.resize(s.length() + 1);
503          memcpy(&m_rawData[0], &s[0], s.length() + 1);          memcpy(&m_rawData[0], &s[0], s.length() + 1);
# Line 444  namespace Serialization { Line 510  namespace Serialization {
510      };      };
511    
512      static _Blob _decodeBlob(const char* p, const char* end, bool bThrow = true) {      static _Blob _decodeBlob(const char* p, const char* end, bool bThrow = true) {
513          if (!bThrow && p >= end)          if (!bThrow && p >= end) {
514              return (_Blob) { p, end };              const _Blob blob =  { p, end };
515                return blob;
516            }
517          size_t sz = 0;          size_t sz = 0;
518          for (; true; ++p) {          for (; true; ++p) {
519              if (p >= end)              if (p >= end)
# Line 460  namespace Serialization { Line 528  namespace Serialization {
528          ++p;          ++p;
529          if (p + sz > end)          if (p + sz > end)
530              throw Exception("Decode Error: Premature end of blob");              throw Exception("Decode Error: Premature end of blob");
531          return (_Blob) { p, p + sz };          const _Blob blob = { p, p + sz };
532            return blob;
533      }      }
534    
535      template<typename T_int>      template<typename T_int>
# Line 535  namespace Serialization { Line 604  namespace Serialization {
604          return s;          return s;
605      }      }
606    
607        static time_t _popTimeBlob(const char*& p, const char* end) {
608            const uint64_t i = _popIntBlob<uint64_t>(p, end);
609            return (time_t) i;
610        }
611    
612      DataType _popDataTypeBlob(const char*& p, const char* end) {      DataType _popDataTypeBlob(const char*& p, const char* end) {
613          _Blob blob = _decodeBlob(p, end);          _Blob blob = _decodeBlob(p, end);
614          p   = blob.p;          p   = blob.p;
# Line 559  namespace Serialization { Line 633  namespace Serialization {
633          const ID id = (ID) _popIntBlob<size_t>(p, end);          const ID id = (ID) _popIntBlob<size_t>(p, end);
634          const size_t size = _popIntBlob<size_t>(p, end);          const size_t size = _popIntBlob<size_t>(p, end);
635    
636          return (UID) { id, size };          const UID uid = { id, size };
637            return uid;
638      }      }
639    
640      static UIDChain _popUIDChainBlob(const char*& p, const char* end) {      static UIDChain _popUIDChainBlob(const char*& p, const char* end) {
# Line 711  namespace Serialization { Line 786  namespace Serialization {
786          _popObjectsBlob(p, end);          _popObjectsBlob(p, end);
787          if (!m_allObjects[m_root])          if (!m_allObjects[m_root])
788              throw Exception("Decode Error: Missing declared root object");              throw Exception("Decode Error: Missing declared root object");
789    
790            m_name = _popStringBlob(p, end);
791            m_comment = _popStringBlob(p, end);
792            m_timeCreated = _popTimeBlob(p, end);
793            m_timeModified = _popTimeBlob(p, end);
794      }      }
795    
796      void Archive::decode(const RawData& data) {      void Archive::decode(const RawData& data) {
797          m_rawData = data;          m_rawData = data;
798          m_allObjects.clear();          m_allObjects.clear();
799          m_isModified = false;          m_isModified = false;
800            m_timeCreated = m_timeModified = LIBGIG_EPOCH_TIME;
801          const char* p   = (const char*) &data[0];          const char* p   = (const char*) &data[0];
802          const char* end = p + data.size();          const char* end = p + data.size();
803          if (memcmp(p, MAGIC_START, std::min(strlen(MAGIC_START), data.size())))          if (memcmp(p, MAGIC_START, std::min(strlen(MAGIC_START), data.size())))
# Line 751  namespace Serialization { Line 832  namespace Serialization {
832          m_root = NO_UID;          m_root = NO_UID;
833          m_rawData.clear();          m_rawData.clear();
834          m_isModified = false;          m_isModified = false;
835            m_timeCreated = m_timeModified = LIBGIG_EPOCH_TIME;
836        }
837    
838        String Archive::name() const {
839            return m_name;
840        }
841    
842        void Archive::setName(String name) {
843            if (m_name == name) return;
844            m_name = name;
845            m_isModified = true;
846        }
847    
848        String Archive::comment() const {
849            return m_comment;
850        }
851    
852        void Archive::setComment(String comment) {
853            if (m_comment == comment) return;
854            m_comment = comment;
855            m_isModified = true;
856        }
857    
858        static tm _convertTimeStamp(const time_t& time, time_base_t base) {
859            tm* pTm;
860            switch (base) {
861                case LOCAL_TIME:
862                    pTm = localtime(&time);
863                    break;
864                case UTC_TIME:
865                    pTm = gmtime(&time);
866                    break;
867                default:
868                    throw Exception("Time stamp with unknown time base (" + ToString((int64_t)base) + ") requested");
869            }
870            if (!pTm)
871                throw Exception("Failed assembling time stamp structure");
872            return *pTm;
873        }
874    
875        time_t Archive::timeStampCreated() const {
876            return m_timeCreated;
877        }
878    
879        time_t Archive::timeStampModified() const {
880            return m_timeModified;
881        }
882    
883        tm Archive::dateTimeCreated(time_base_t base) const {
884            return _convertTimeStamp(m_timeCreated, base);
885        }
886    
887        tm Archive::dateTimeModified(time_base_t base) const {
888            return _convertTimeStamp(m_timeModified, base);
889      }      }
890    
891      void Archive::removeMember(Object& parent, const Member& member) {      void Archive::removeMember(Object& parent, const Member& member) {
# Line 908  namespace Serialization { Line 1043  namespace Serialization {
1043          return _primitiveObjectValueToString(*pObject);          return _primitiveObjectValueToString(*pObject);
1044      }      }
1045    
1046        int64_t Archive::valueAsInt(const Object& object) {
1047            if (!object)
1048                throw Exception("Invalid object");
1049            if (!object.type().isInteger() && !object.type().isEnum())
1050                throw Exception("Object is neither an integer nor an enum");
1051            const Object* pObject = &object;
1052            if (object.type().isPointer()) {
1053                const Object& obj = objectByUID(object.uid(1));
1054                if (!obj) return 0;
1055                pObject = &obj;
1056            }
1057            return _primitiveObjectValueToNumber<int64_t>(*pObject);
1058        }
1059    
1060        double Archive::valueAsReal(const Object& object) {
1061            if (!object)
1062                throw Exception("Invalid object");
1063            if (!object.type().isReal())
1064                throw Exception("Object is not an real type");
1065            const Object* pObject = &object;
1066            if (object.type().isPointer()) {
1067                const Object& obj = objectByUID(object.uid(1));
1068                if (!obj) return 0;
1069                pObject = &obj;
1070            }
1071            return _primitiveObjectValueToNumber<double>(*pObject);
1072        }
1073    
1074        bool Archive::valueAsBool(const Object& object) {
1075            if (!object)
1076                throw Exception("Invalid object");
1077            if (!object.type().isBool())
1078                throw Exception("Object is not a bool");
1079            const Object* pObject = &object;
1080            if (object.type().isPointer()) {
1081                const Object& obj = objectByUID(object.uid(1));
1082                if (!obj) return 0;
1083                pObject = &obj;
1084            }
1085            return _primitiveObjectValueToNumber<bool>(*pObject);
1086        }
1087    
1088      // *************** Archive::Syncer ***************      // *************** Archive::Syncer ***************
1089      // *      // *
1090    

Legend:
Removed from v.3153  
changed lines
  Added in v.3169

  ViewVC Help
Powered by ViewVC