summaryrefslogtreecommitdiff
path: root/libodb/odb/wrapper-traits.hxx
diff options
context:
space:
mode:
Diffstat (limited to 'libodb/odb/wrapper-traits.hxx')
-rw-r--r--libodb/odb/wrapper-traits.hxx276
1 files changed, 276 insertions, 0 deletions
diff --git a/libodb/odb/wrapper-traits.hxx b/libodb/odb/wrapper-traits.hxx
new file mode 100644
index 0000000..d31425d
--- /dev/null
+++ b/libodb/odb/wrapper-traits.hxx
@@ -0,0 +1,276 @@
+// file : odb/wrapper-traits.hxx
+// license : GNU GPL v2; see accompanying LICENSE file
+
+#ifndef ODB_WRAPPER_TRAITS_HXX
+#define ODB_WRAPPER_TRAITS_HXX
+
+#include <odb/pre.hxx>
+
+#include <memory> // std::auto_ptr, std::unique_ptr, std::shared_ptr/weak_ptr
+
+#include <odb/nullable.hxx>
+
+#include <odb/details/config.hxx> // ODB_CXX11
+#include <odb/details/meta/remove-const.hxx>
+
+namespace odb
+{
+ template <typename T>
+ class wrapper_traits;
+
+ // Sample specialization for raw pointers. It is not enabled by default
+ // since it makes many assumptions that may not always hold true (such
+ // as that instances are allocated with new and freed with delete).
+ // This makes it too dangerous to be enabled unconditionally. If you
+ // need this functionality, you can copy the below code into your
+ // application. Also consider changing it to only specialize for
+ // specific types instead of for any pointer (it will almost always
+ // do the wrong thing for char*).
+ //
+#if 0
+ template <typename T>
+ class wrapper_traits<T*>
+ {
+ public:
+ typedef T wrapped_type;
+ typedef T* wrapper_type;
+
+ // T can be const.
+ //
+ typedef
+ typename details::meta::remove_const<T>::result
+ unrestricted_wrapped_type;
+
+ static const bool null_handler = true;
+ static const bool null_default = false;
+
+ static bool
+ get_null (const wrapper_type& p)
+ {
+ return p == 0;
+ }
+
+ static void
+ set_null (wrapper_type& p)
+ {
+ delete p;
+ p = 0;
+ }
+
+ static const wrapped_type&
+ get_ref (const wrapper_type& p)
+ {
+ return *p;
+ }
+
+ static unrestricted_wrapped_type&
+ set_ref (wrapper_type& p)
+ {
+ if (p == 0)
+ p = new unrestricted_wrapped_type;
+
+ return const_cast<unrestricted_wrapped_type&> (*p);
+ }
+ };
+#endif
+
+ // Specialization for std::auto_ptr.
+ //
+#ifndef ODB_CXX11
+ template <typename T>
+ class wrapper_traits< std::auto_ptr<T> >
+ {
+ public:
+ // T can be const.
+ //
+ typedef T wrapped_type;
+ typedef std::auto_ptr<T> wrapper_type;
+
+ // T can be const.
+ //
+ typedef
+ typename odb::details::meta::remove_const<T>::result
+ unrestricted_wrapped_type;
+
+ static const bool null_handler = true;
+ static const bool null_default = false;
+
+ static bool
+ get_null (const wrapper_type& p)
+ {
+ return p.get () == 0;
+ }
+
+ static void
+ set_null (wrapper_type& p)
+ {
+ p.reset ();
+ }
+
+ static const wrapped_type&
+ get_ref (const wrapper_type& p)
+ {
+ return *p;
+ }
+
+ static unrestricted_wrapped_type&
+ set_ref (wrapper_type& p)
+ {
+ if (p.get () == 0)
+ p.reset (new unrestricted_wrapped_type ());
+
+ return const_cast<unrestricted_wrapped_type&> (*p);
+ }
+ };
+#endif
+
+#ifdef ODB_CXX11
+
+ // Specialization for C++11 std::unique_ptr.
+ //
+ template <typename T, typename D>
+ class wrapper_traits<std::unique_ptr<T, D>>
+ {
+ public:
+ // T can be const.
+ //
+ typedef T wrapped_type;
+ typedef std::unique_ptr<T, D> wrapper_type;
+
+ // T can be const.
+ //
+ typedef
+ typename odb::details::meta::remove_const<T>::result
+ unrestricted_wrapped_type;
+
+ static const bool null_handler = true;
+ static const bool null_default = false;
+
+ static bool
+ get_null (const wrapper_type& p)
+ {
+ return !p;
+ }
+
+ static void
+ set_null (wrapper_type& p)
+ {
+ p.reset ();
+ }
+
+ static const wrapped_type&
+ get_ref (const wrapper_type& p)
+ {
+ return *p;
+ }
+
+ static unrestricted_wrapped_type&
+ set_ref (wrapper_type& p)
+ {
+ if (!p)
+ p.reset (new unrestricted_wrapped_type ());
+
+ return const_cast<unrestricted_wrapped_type&> (*p);
+ }
+ };
+
+ // Specialization for C++11 std::shared_ptr.
+ //
+ template <typename T>
+ class wrapper_traits<std::shared_ptr<T>>
+ {
+ public:
+ typedef T wrapped_type;
+ typedef std::shared_ptr<T> wrapper_type;
+
+ // T can be const.
+ //
+ typedef
+ typename odb::details::meta::remove_const<T>::result
+ unrestricted_wrapped_type;
+
+ static const bool null_handler = true;
+ static const bool null_default = false;
+
+ static bool
+ get_null (const wrapper_type& p)
+ {
+ return !p;
+ }
+
+ static void
+ set_null (wrapper_type& p)
+ {
+ p.reset ();
+ }
+
+ static const wrapped_type&
+ get_ref (const wrapper_type& p)
+ {
+ return *p;
+ }
+
+ static unrestricted_wrapped_type&
+ set_ref (wrapper_type& p)
+ {
+ if (!p)
+ p.reset (new unrestricted_wrapped_type);
+
+ return const_cast<unrestricted_wrapped_type&> (*p);
+ }
+ };
+
+#endif // ODB_CXX11
+
+ // Specialization for odb::nullable.
+ //
+ template <typename T>
+ class wrapper_traits< nullable<T> >
+ {
+ public:
+ // T can be const.
+ //
+ typedef T wrapped_type;
+ typedef nullable<T> wrapper_type;
+
+ // T can be const.
+ //
+ typedef
+ typename odb::details::meta::remove_const<T>::result
+ unrestricted_wrapped_type;
+
+ static const bool null_handler = true;
+ static const bool null_default = true;
+
+ static bool
+ get_null (const wrapper_type& n)
+ {
+ return n.null ();
+ }
+
+ static void
+ set_null (wrapper_type& n)
+ {
+ n.reset ();
+ }
+
+ static const wrapped_type&
+ get_ref (const wrapper_type& n)
+ {
+ return *n;
+ }
+
+ static unrestricted_wrapped_type&
+ set_ref (wrapper_type& n)
+ {
+ if (n.null ())
+ n = unrestricted_wrapped_type ();
+
+ return const_cast<unrestricted_wrapped_type&> (*n);
+ }
+ };
+}
+
+#include <odb/post.hxx>
+
+#endif // ODB_WRAPPER_TRAITS_HXX