260 lines
		
	
	
		
			6.1 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
			
		
		
	
	
			260 lines
		
	
	
		
			6.1 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
| // Copyright David Abrahams 2002.
 | |
| // Distributed under the Boost Software License, Version 1.0. (See
 | |
| // accompanying file LICENSE_1_0.txt or copy at
 | |
| // http://www.boost.org/LICENSE_1_0.txt)
 | |
| #ifndef EXTRACT_DWA200265_HPP
 | |
| # define EXTRACT_DWA200265_HPP
 | |
| 
 | |
| # include <boost/python/detail/prefix.hpp>
 | |
| 
 | |
| # include <boost/python/converter/object_manager.hpp>
 | |
| # include <boost/python/converter/from_python.hpp>
 | |
| # include <boost/python/converter/rvalue_from_python_data.hpp>
 | |
| # include <boost/python/converter/registered.hpp>
 | |
| # include <boost/python/converter/registered_pointee.hpp>
 | |
| 
 | |
| # include <boost/python/object_core.hpp>
 | |
| # include <boost/python/refcount.hpp>
 | |
| 
 | |
| # include <boost/python/detail/copy_ctor_mutates_rhs.hpp>
 | |
| # include <boost/python/detail/void_ptr.hpp>
 | |
| # include <boost/python/detail/void_return.hpp>
 | |
| # include <boost/call_traits.hpp>
 | |
| 
 | |
| #if BOOST_WORKAROUND(BOOST_INTEL_WIN, <= 900)
 | |
| # define BOOST_EXTRACT_WORKAROUND ()
 | |
| #else
 | |
| # define BOOST_EXTRACT_WORKAROUND
 | |
| #endif
 | |
| 
 | |
| namespace boost { namespace python {
 | |
| 
 | |
| namespace api
 | |
| {
 | |
|   class object;
 | |
| }
 | |
| 
 | |
| namespace converter
 | |
| {
 | |
|   template <class Ptr>
 | |
|   struct extract_pointer
 | |
|   {
 | |
|       typedef Ptr result_type;
 | |
|       extract_pointer(PyObject*);
 | |
|       
 | |
|       bool check() const;
 | |
|       Ptr operator()() const;
 | |
|       
 | |
|    private:
 | |
|       PyObject* m_source;
 | |
|       void* m_result;
 | |
|   };
 | |
|   
 | |
|   template <class Ref>
 | |
|   struct extract_reference
 | |
|   {
 | |
|       typedef Ref result_type;
 | |
|       extract_reference(PyObject*);
 | |
|       
 | |
|       bool check() const;
 | |
|       Ref operator()() const;
 | |
|       
 | |
|    private:
 | |
|       PyObject* m_source;
 | |
|       void* m_result;
 | |
|   };
 | |
|   
 | |
|   template <class T>
 | |
|   struct extract_rvalue : private noncopyable
 | |
|   {
 | |
|       typedef typename mpl::if_<
 | |
|           python::detail::copy_ctor_mutates_rhs<T>
 | |
|         , T&
 | |
|         , typename call_traits<T>::param_type
 | |
|       >::type result_type;
 | |
| 
 | |
|       extract_rvalue(PyObject*);
 | |
| 
 | |
|       bool check() const;
 | |
|       result_type operator()() const;
 | |
|    private:
 | |
|       PyObject* m_source;
 | |
|       mutable rvalue_from_python_data<T> m_data;
 | |
|   };
 | |
|   
 | |
|   template <class T>
 | |
|   struct extract_object_manager
 | |
|   {
 | |
|       typedef T result_type;
 | |
|       extract_object_manager(PyObject*);
 | |
| 
 | |
|       bool check() const;
 | |
|       result_type operator()() const;
 | |
|    private:
 | |
|       PyObject* m_source;
 | |
|   };
 | |
|   
 | |
|   template <class T>
 | |
|   struct select_extract
 | |
|   {
 | |
|       BOOST_STATIC_CONSTANT(
 | |
|           bool, obj_mgr = is_object_manager<T>::value);
 | |
| 
 | |
|       BOOST_STATIC_CONSTANT(
 | |
|           bool, ptr = is_pointer<T>::value);
 | |
|     
 | |
|       BOOST_STATIC_CONSTANT(
 | |
|           bool, ref = is_reference<T>::value);
 | |
| 
 | |
|       typedef typename mpl::if_c<
 | |
|           obj_mgr
 | |
|           , extract_object_manager<T>
 | |
|           , typename mpl::if_c<
 | |
|               ptr
 | |
|               , extract_pointer<T>
 | |
|               , typename mpl::if_c<
 | |
|                   ref
 | |
|                   , extract_reference<T>
 | |
|                   , extract_rvalue<T>
 | |
|                 >::type
 | |
|             >::type
 | |
|          >::type type;
 | |
|   };
 | |
| }
 | |
| 
 | |
| template <class T>
 | |
| struct extract
 | |
|     : converter::select_extract<T>::type
 | |
| {
 | |
|  private:
 | |
|     typedef typename converter::select_extract<T>::type base;
 | |
|  public:
 | |
|     typedef typename base::result_type result_type;
 | |
|     
 | |
|     operator result_type() const
 | |
|     {
 | |
|         return (*this)();
 | |
|     }
 | |
|     
 | |
|     extract(PyObject*);
 | |
|     extract(api::object const&);
 | |
| };
 | |
| 
 | |
| //
 | |
| // Implementations
 | |
| //
 | |
| template <class T>
 | |
| inline extract<T>::extract(PyObject* o)
 | |
|     : base(o)
 | |
| {
 | |
| }
 | |
| 
 | |
| template <class T>
 | |
| inline extract<T>::extract(api::object const& o)
 | |
|     : base(o.ptr())
 | |
| {
 | |
| }
 | |
| 
 | |
| namespace converter
 | |
| {
 | |
|   template <class T>
 | |
|   inline extract_rvalue<T>::extract_rvalue(PyObject* x)
 | |
|       : m_source(x)
 | |
|       , m_data(
 | |
|           (rvalue_from_python_stage1)(x, registered<T>::converters)
 | |
|           )
 | |
|   {
 | |
|   }
 | |
|   
 | |
|   template <class T>
 | |
|   inline bool
 | |
|   extract_rvalue<T>::check() const
 | |
|   {
 | |
|       return m_data.stage1.convertible;
 | |
|   }
 | |
| 
 | |
|   template <class T>
 | |
|   inline typename extract_rvalue<T>::result_type
 | |
|   extract_rvalue<T>::operator()() const
 | |
|   {
 | |
|       return *(T*)(
 | |
|           // Only do the stage2 conversion once
 | |
|           m_data.stage1.convertible ==  m_data.storage.bytes
 | |
|              ? m_data.storage.bytes
 | |
|              : (rvalue_from_python_stage2)(m_source, m_data.stage1, registered<T>::converters)
 | |
|           );
 | |
|   }
 | |
| 
 | |
|   template <class Ref>
 | |
|   inline extract_reference<Ref>::extract_reference(PyObject* obj)
 | |
|       : m_source(obj)
 | |
|       , m_result(
 | |
|           (get_lvalue_from_python)(obj, registered<Ref>::converters)
 | |
|           )
 | |
|   {
 | |
|   }
 | |
| 
 | |
|   template <class Ref>
 | |
|   inline bool extract_reference<Ref>::check() const
 | |
|   {
 | |
|       return m_result != 0;
 | |
|   }
 | |
| 
 | |
|   template <class Ref>
 | |
|   inline Ref extract_reference<Ref>::operator()() const
 | |
|   {
 | |
|       if (m_result == 0)
 | |
|           (throw_no_reference_from_python)(m_source, registered<Ref>::converters);
 | |
|       
 | |
|       return python::detail::void_ptr_to_reference(m_result, (Ref(*)())0);
 | |
|   }
 | |
| 
 | |
|   template <class Ptr>
 | |
|   inline extract_pointer<Ptr>::extract_pointer(PyObject* obj)
 | |
|       : m_source(obj)
 | |
|       , m_result(
 | |
|           obj == Py_None ? 0 : (get_lvalue_from_python)(obj, registered_pointee<Ptr>::converters)
 | |
|           )
 | |
|   {
 | |
|   }
 | |
| 
 | |
|   template <class Ptr>
 | |
|   inline bool extract_pointer<Ptr>::check() const
 | |
|   {
 | |
|       return m_source == Py_None || m_result != 0;
 | |
|   }
 | |
| 
 | |
|   template <class Ptr>
 | |
|   inline Ptr extract_pointer<Ptr>::operator()() const
 | |
|   {
 | |
|       if (m_result == 0 && m_source != Py_None)
 | |
|           (throw_no_pointer_from_python)(m_source, registered_pointee<Ptr>::converters);
 | |
|       
 | |
|       return Ptr(m_result);
 | |
|   }
 | |
| 
 | |
|   template <class T>
 | |
|   inline extract_object_manager<T>::extract_object_manager(PyObject* obj)
 | |
|       : m_source(obj)
 | |
|   {
 | |
|   }
 | |
| 
 | |
|   template <class T>
 | |
|   inline bool extract_object_manager<T>::check() const
 | |
|   {
 | |
|       return object_manager_traits<T>::check(m_source);
 | |
|   }
 | |
| 
 | |
|   template <class T>
 | |
|   inline T extract_object_manager<T>::operator()() const
 | |
|   {
 | |
|       return T(
 | |
|           object_manager_traits<T>::adopt(python::incref(m_source))
 | |
|           );
 | |
|   }
 | |
| }
 | |
|   
 | |
| }} // namespace boost::python::converter
 | |
| 
 | |
| #endif // EXTRACT_DWA200265_HPP
 | 
