SerializableObject.h 8.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155
  1. // Jolt Physics Library (https://github.com/jrouwe/JoltPhysics)
  2. // SPDX-FileCopyrightText: 2021 Jorrit Rouwe
  3. // SPDX-License-Identifier: MIT
  4. #pragma once
  5. #include <Jolt/ObjectStream/ObjectStream.h>
  6. JPH_NAMESPACE_BEGIN
  7. //////////////////////////////////////////////////////////////////////////////////////////
  8. // Helper macros
  9. //////////////////////////////////////////////////////////////////////////////////////////
  10. // JPH_DECLARE_SERIALIZATION_FUNCTIONS
  11. #define JPH_DECLARE_SERIALIZATION_FUNCTIONS(linkage, prefix, class_name) \
  12. linkage prefix bool OSReadData(IObjectStreamIn &ioStream, class_name &inInstance); \
  13. linkage prefix bool OSReadData(IObjectStreamIn &ioStream, class_name *&inPointer); \
  14. linkage prefix bool OSIsType(class_name *, int inArrayDepth, EOSDataType inDataType, const char *inClassName); \
  15. linkage prefix bool OSIsType(class_name **, int inArrayDepth, EOSDataType inDataType, const char *inClassName); \
  16. linkage prefix void OSWriteData(IObjectStreamOut &ioStream, const class_name &inInstance); \
  17. linkage prefix void OSWriteData(IObjectStreamOut &ioStream, class_name *const &inPointer); \
  18. linkage prefix void OSWriteDataType(IObjectStreamOut &ioStream, class_name *); \
  19. linkage prefix void OSWriteDataType(IObjectStreamOut &ioStream, class_name **);
  20. // JPH_IMPLEMENT_SERIALIZATION_FUNCTIONS
  21. #define JPH_IMPLEMENT_SERIALIZATION_FUNCTIONS(class_name) \
  22. bool OSReadData(IObjectStreamIn &ioStream, class_name &inInstance) \
  23. { \
  24. return ioStream.ReadClassData(#class_name, (void *)&inInstance); \
  25. } \
  26. bool OSReadData(IObjectStreamIn &ioStream, class_name *&inPointer) \
  27. { \
  28. return ioStream.ReadPointerData(JPH_RTTI(class_name), (void **)&inPointer); \
  29. } \
  30. bool OSIsType(class_name *, int inArrayDepth, EOSDataType inDataType, const char *inClassName) \
  31. { \
  32. return inArrayDepth == 0 && inDataType == EOSDataType::Instance && strcmp(inClassName, #class_name) == 0; \
  33. } \
  34. bool OSIsType(class_name **, int inArrayDepth, EOSDataType inDataType, const char *inClassName) \
  35. { \
  36. return inArrayDepth == 0 && inDataType == EOSDataType::Pointer && strcmp(inClassName, #class_name) == 0; \
  37. } \
  38. void OSWriteData(IObjectStreamOut &ioStream, const class_name &inInstance) \
  39. { \
  40. ioStream.WriteClassData(JPH_RTTI(class_name), (void *)&inInstance); \
  41. } \
  42. void OSWriteData(IObjectStreamOut &ioStream, class_name *const &inPointer) \
  43. { \
  44. if (inPointer) \
  45. ioStream.WritePointerData(GetRTTI(inPointer), (void *)inPointer); \
  46. else \
  47. ioStream.WritePointerData(nullptr, nullptr); \
  48. } \
  49. void OSWriteDataType(IObjectStreamOut &ioStream, class_name *) \
  50. { \
  51. ioStream.WriteDataType(EOSDataType::Instance); \
  52. ioStream.WriteName(#class_name); \
  53. } \
  54. void OSWriteDataType(IObjectStreamOut &ioStream, class_name **) \
  55. { \
  56. ioStream.WriteDataType(EOSDataType::Pointer); \
  57. ioStream.WriteName(#class_name); \
  58. }
  59. //////////////////////////////////////////////////////////////////////////////////////////
  60. // Use these macros on non-virtual objects to make them serializable
  61. //////////////////////////////////////////////////////////////////////////////////////////
  62. // JPH_DECLARE_SERIALIZABLE_NON_VIRTUAL
  63. #define JPH_DECLARE_SERIALIZABLE_NON_VIRTUAL(linkage, class_name) \
  64. public: \
  65. JPH_DECLARE_RTTI_NON_VIRTUAL(linkage, class_name) \
  66. JPH_DECLARE_SERIALIZATION_FUNCTIONS(linkage, friend, class_name) \
  67. // JPH_IMPLEMENT_SERIALIZABLE_NON_VIRTUAL
  68. #define JPH_IMPLEMENT_SERIALIZABLE_NON_VIRTUAL(class_name) \
  69. JPH_IMPLEMENT_SERIALIZATION_FUNCTIONS(class_name) \
  70. JPH_IMPLEMENT_RTTI_NON_VIRTUAL(class_name) \
  71. //////////////////////////////////////////////////////////////////////////////////////////
  72. // Same as above, but when you cannot insert the declaration in the class itself
  73. //////////////////////////////////////////////////////////////////////////////////////////
  74. // JPH_DECLARE_SERIALIZABLE_OUTSIDE_CLASS
  75. #define JPH_DECLARE_SERIALIZABLE_OUTSIDE_CLASS(linkage, class_name) \
  76. JPH_DECLARE_RTTI_OUTSIDE_CLASS(linkage, class_name) \
  77. JPH_DECLARE_SERIALIZATION_FUNCTIONS(linkage, extern, class_name) \
  78. // JPH_IMPLEMENT_SERIALIZABLE_OUTSIDE_CLASS
  79. #define JPH_IMPLEMENT_SERIALIZABLE_OUTSIDE_CLASS(class_name) \
  80. JPH_IMPLEMENT_SERIALIZATION_FUNCTIONS(class_name) \
  81. JPH_IMPLEMENT_RTTI_OUTSIDE_CLASS(class_name) \
  82. //////////////////////////////////////////////////////////////////////////////////////////
  83. // Same as above, but for classes that have virtual functions
  84. //////////////////////////////////////////////////////////////////////////////////////////
  85. // JPH_DECLARE_SERIALIZABLE_VIRTUAL - Use for concrete, non-base classes
  86. #define JPH_DECLARE_SERIALIZABLE_VIRTUAL(linkage, class_name) \
  87. public: \
  88. JPH_DECLARE_RTTI_VIRTUAL(linkage, class_name) \
  89. JPH_DECLARE_SERIALIZATION_FUNCTIONS(linkage, friend, class_name) \
  90. // JPH_IMPLEMENT_SERIALIZABLE_VIRTUAL
  91. #define JPH_IMPLEMENT_SERIALIZABLE_VIRTUAL(class_name) \
  92. JPH_IMPLEMENT_SERIALIZATION_FUNCTIONS(class_name) \
  93. JPH_IMPLEMENT_RTTI_VIRTUAL(class_name) \
  94. // JPH_DECLARE_SERIALIZABLE_ABSTRACT - Use for abstract, non-base classes
  95. #define JPH_DECLARE_SERIALIZABLE_ABSTRACT(linkage, class_name) \
  96. public: \
  97. JPH_DECLARE_RTTI_ABSTRACT(linkage, class_name) \
  98. JPH_DECLARE_SERIALIZATION_FUNCTIONS(linkage, friend, class_name) \
  99. // JPH_IMPLEMENT_SERIALIZABLE_ABSTRACT
  100. #define JPH_IMPLEMENT_SERIALIZABLE_ABSTRACT(class_name) \
  101. JPH_IMPLEMENT_SERIALIZATION_FUNCTIONS(class_name) \
  102. JPH_IMPLEMENT_RTTI_ABSTRACT(class_name) \
  103. // JPH_DECLARE_SERIALIZABLE_VIRTUAL_BASE - Use for concrete base classes
  104. #define JPH_DECLARE_SERIALIZABLE_VIRTUAL_BASE(linkage, class_name) \
  105. public: \
  106. JPH_DECLARE_RTTI_VIRTUAL_BASE(linkage, class_name) \
  107. JPH_DECLARE_SERIALIZATION_FUNCTIONS(linkage, friend, class_name) \
  108. // JPH_IMPLEMENT_SERIALIZABLE_VIRTUAL_BASE
  109. #define JPH_IMPLEMENT_SERIALIZABLE_VIRTUAL_BASE(class_name) \
  110. JPH_IMPLEMENT_SERIALIZATION_FUNCTIONS(class_name) \
  111. JPH_IMPLEMENT_RTTI_VIRTUAL_BASE(class_name) \
  112. // JPH_DECLARE_SERIALIZABLE_ABSTRACT_BASE - Use for abstract base class
  113. #define JPH_DECLARE_SERIALIZABLE_ABSTRACT_BASE(linkage, class_name) \
  114. public: \
  115. JPH_DECLARE_RTTI_ABSTRACT_BASE(linkage, class_name) \
  116. JPH_DECLARE_SERIALIZATION_FUNCTIONS(linkage, friend, class_name) \
  117. // JPH_IMPLEMENT_SERIALIZABLE_ABSTRACT_BASE
  118. #define JPH_IMPLEMENT_SERIALIZABLE_ABSTRACT_BASE(class_name) \
  119. JPH_IMPLEMENT_SERIALIZATION_FUNCTIONS(class_name) \
  120. JPH_IMPLEMENT_RTTI_ABSTRACT_BASE(class_name)
  121. /// Classes must be derived from SerializableObject if you want to be able to save pointers or
  122. /// reference counting pointers to objects of this or derived classes. The type will automatically
  123. /// be determined during serialization and upon deserialization it will be restored correctly.
  124. class JPH_EXPORT SerializableObject : public NonCopyable
  125. {
  126. JPH_DECLARE_SERIALIZABLE_ABSTRACT_BASE(JPH_EXPORT, SerializableObject)
  127. public:
  128. /// Constructor
  129. virtual ~SerializableObject() = default;
  130. };
  131. JPH_NAMESPACE_END