--- /dev/null
+////////////////////////////////////////////////////////////////////////////////\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
+////////////////////////////////////////////////////////////////////////////////\r
+#include "persistent_int.hpp"\r
+#include "persistent_pointer.hpp"\r
+#include "persistent_callback.hpp"\r
+#include "persistent_interface.hpp"\r
+\r
+namespace stlplus\r
+{\r
+\r
+ ////////////////////////////////////////////////////////////////////////////////\r
+\r
+ template<typename T, typename DE>\r
+ void dump_simple_ptr(dump_context& context, const simple_ptr<T>& data,\r
+ DE dump_element)\r
+ throw(persistent_dump_failed)\r
+ {\r
+ // Many smart pointers can point to the same object.\r
+ // I could have used the address of the object to differentiate, \r
+ // but that would not have differentiated between different null smart pointers\r
+ // so I use the address of the count to differentiate between different objects.\r
+ // get a magic key for the substructure - this also returns a flag saying whether its been seen before\r
+ std::pair<bool,unsigned> mapping = context.pointer_map(data._count());\r
+ // dump the magic key for the count\r
+ dump_unsigned(context,mapping.second);\r
+ // dump the contents always - this is because I need to rely on the pointer routines dumping a second magic key\r
+ // use the existing routines for ordinary pointers to dump the contents\r
+ dump_pointer(context,data._pointer(),dump_element);\r
+ }\r
+\r
+ template<typename T, typename RE>\r
+ void restore_simple_ptr(restore_context& context, simple_ptr<T>& data,\r
+ RE restore_element)\r
+ throw(persistent_restore_failed)\r
+ {\r
+ // get the old counter magic key\r
+ unsigned magic = 0;\r
+ restore_unsigned(context,magic);\r
+ // lookup this magic number to see if we have seen this already\r
+ std::pair<bool,void*> mapping = context.pointer_map(magic);\r
+ if (mapping.first)\r
+ {\r
+ // this holder has already been restored\r
+ // now restore the object and rely on the pointer routines to return the existing object\r
+ T* value = 0;\r
+ restore_pointer(context,value,restore_element);\r
+ // dealias the existing holder and replace it with the seen-before holder to make this object an alias of it\r
+ data._make_alias(value, (unsigned*)mapping.second);\r
+ }\r
+ else\r
+ {\r
+ // this is the first contact with this holder\r
+ // make sure this smart pointer is unique to prevent side-effects\r
+ data.clear_unique();\r
+ // map the magic key onto this structure's holder\r
+ // do this before restoring the object so that self-referential structures restore correctly\r
+ context.pointer_add(magic,data._count());\r
+ // now restore the object\r
+ T* value = 0;\r
+ restore_pointer(context,value,restore_element);\r
+ // and add it to the pointer\r
+ data.set(value);\r
+ }\r
+ }\r
+\r
+ ////////////////////////////////////////////////////////////////////////////////\r
+ // simple_ptr_clone using callbacks\r
+\r
+ template<typename T>\r
+ void dump_simple_ptr_clone_callback(dump_context& context, const simple_ptr_clone<T>& data)\r
+ throw(persistent_dump_failed)\r
+ {\r
+ std::pair<bool,unsigned> mapping = context.pointer_map(data._count());\r
+ dump_unsigned(context,mapping.second);\r
+ dump_callback(context,data._pointer());\r
+ }\r
+\r
+ template<typename T>\r
+ void restore_simple_ptr_clone_callback(restore_context& context, simple_ptr_clone<T>& data)\r
+ throw(persistent_restore_failed)\r
+ {\r
+ unsigned magic = 0;\r
+ restore_unsigned(context,magic);\r
+ std::pair<bool,void*> mapping = context.pointer_map(magic);\r
+ if (mapping.first)\r
+ {\r
+ T* value = 0;\r
+ restore_callback(context,value);\r
+ data._make_alias(value, (unsigned*)mapping.second);\r
+ }\r
+ else\r
+ {\r
+ data.clear_unique();\r
+ context.pointer_add(magic,data._count());\r
+ T* value = 0;\r
+ restore_callback(context,value);\r
+ data.set(value);\r
+ }\r
+ }\r
+\r
+ ////////////////////////////////////////////////////////////////////////////////\r
+ // simple_ptr_clone using interface\r
+\r
+ template<typename T>\r
+ void dump_simple_ptr_clone_interface(dump_context& context, const simple_ptr_clone<T>& data)\r
+ throw(persistent_dump_failed)\r
+ {\r
+ std::pair<bool,unsigned> mapping = context.pointer_map(data._count());\r
+ dump_unsigned(context,mapping.second);\r
+ dump_interface(context,data._pointer());\r
+ }\r
+\r
+ template<typename T>\r
+ void restore_simple_ptr_clone_interface(restore_context& context, simple_ptr_clone<T>& data)\r
+ throw(persistent_restore_failed)\r
+ {\r
+ unsigned magic = 0;\r
+ restore_unsigned(context,magic);\r
+ std::pair<bool,void*> mapping = context.pointer_map(magic);\r
+ if (mapping.first)\r
+ {\r
+ T* value = 0;\r
+ restore_interface(context,value);\r
+ data._make_alias(value, (unsigned*)mapping.second);\r
+ }\r
+ else\r
+ {\r
+ data.clear_unique();\r
+ context.pointer_add(magic,data._count());\r
+ T* value = 0;\r
+ restore_interface(context,value);\r
+ data.set(value);\r
+ }\r
+ }\r
+\r
+ ////////////////////////////////////////////////////////////////////////////////\r
+\r
+} // end namespace stlplus\r