]> Dogcows Code - chaz/yoink/blobdiff - src/stlplus/persistence/persistent_interface.tpp
testing new non-autotools build system
[chaz/yoink] / src / stlplus / persistence / persistent_interface.tpp
diff --git a/src/stlplus/persistence/persistent_interface.tpp b/src/stlplus/persistence/persistent_interface.tpp
new file mode 100644 (file)
index 0000000..a684aa3
--- /dev/null
@@ -0,0 +1,99 @@
+////////////////////////////////////////////////////////////////////////////////\r
+\r
+//   Author:    Andy Rushton\r
+//   Copyright: (c) Southampton University 1999-2004\r
+//              (c) Andy Rushton           2004-2009\r
+//   License:   BSD License, see ../docs/license.html\r
+\r
+//   Polymorphous classes using the interface approach\r
+\r
+//   format: magic [ key data ]\r
+\r
+////////////////////////////////////////////////////////////////////////////////\r
+#include "persistent_int.hpp"\r
+\r
+namespace stlplus\r
+{\r
+\r
+  ////////////////////////////////////////////////////////////////////////////////\r
+\r
+  template<typename T>\r
+  void dump_interface(dump_context& context, const T* const data)\r
+    throw(persistent_dump_failed)\r
+  {\r
+    try\r
+    {\r
+      // register the address and get the magic key for it\r
+      std::pair<bool,unsigned> mapping = context.pointer_map(data);\r
+      dump_unsigned(context,mapping.second);\r
+      // if the address is null, then that is all that we need to do\r
+      // however, if it is non-null and this is the first sight of the address, dump the contents\r
+      if (data && !mapping.first)\r
+      {\r
+        // interface method\r
+        // the lookup just finds the magic key and the type has a dump method\r
+        // this will throw persistent_illegal_type if the type is not registered\r
+        unsigned key = context.lookup_interface(typeid(*data));\r
+        // dump the magic key for the type\r
+        dump_unsigned(context, key);\r
+        // now call the dump method defined by the interface\r
+        data->dump(context);\r
+      }\r
+    }\r
+    catch (const persistent_illegal_type& except)\r
+    {\r
+      // convert this to a simpler dump failed exception\r
+      throw persistent_dump_failed(except.what());\r
+    }\r
+  }\r
+\r
+  ////////////////////////////////////////////////////////////////////////////////\r
+\r
+  template<typename T>\r
+  void restore_interface(restore_context& context, T*& data)\r
+    throw(persistent_restore_failed)\r
+  {\r
+    try\r
+    {\r
+      // first delete any previous object pointed to since the restore creates the object of the right subclass\r
+      if (data)\r
+      {\r
+        delete data;\r
+        data = 0;\r
+      }\r
+      // get the magic key\r
+      unsigned magic = 0;\r
+      restore_unsigned(context,magic);\r
+      // now lookup the magic key to see if this pointer has already been restored\r
+      // null pointers are always flagged as already restored\r
+      std::pair<bool,void*> address = context.pointer_map(magic);\r
+      if (address.first)\r
+      {\r
+        // seen before, so simply map it to the existing address\r
+        data = (T*)address.second;\r
+      }\r
+      else\r
+      {\r
+        // now restore the magic key that denotes the particular subclass\r
+        unsigned key = 0;\r
+        restore_unsigned(context, key);\r
+        // interface approach\r
+        // first clone the sample object stored in the map - lookup_interface can throw persistent_illegal_type\r
+        data = (T*)(context.lookup_interface(key)->clone());\r
+        // add this pointer to the set of already seen objects\r
+        // do this before restoring the object so that self-referential structures restore correctly\r
+        context.pointer_add(magic,data);\r
+        // now restore the contents using the object's method\r
+        data->restore(context);\r
+      }\r
+    }\r
+    catch (const persistent_illegal_type& exception)\r
+    {\r
+      // convert this to a simpler dump failed exception\r
+      throw persistent_restore_failed(exception.what());\r
+    }\r
+  }\r
+\r
+  ////////////////////////////////////////////////////////////////////////////////\r
+\r
+} // end namespace stlplus\r
This page took 0.02212 seconds and 4 git commands to generate.