100 lines
		
	
	
		
			2.6 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
			
		
		
	
	
			100 lines
		
	
	
		
			2.6 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
| // (C) Copyright 2005 Matthias Troyer
 | |
| 
 | |
| // Use, modification and distribution is subject to 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)
 | |
| 
 | |
| //  Authors: Matthias Troyer
 | |
| 
 | |
| #ifndef BOOST_MPI_DETAIL_TYPE_MPI_DATATYPE_CACHE_HPP
 | |
| #define BOOST_MPI_DETAIL_TYPE_MPI_DATATYPE_CACHE_HPP
 | |
| 
 | |
| #include <boost/mpi/datatype_fwd.hpp>
 | |
| #include <boost/mpi/detail/mpi_datatype_oarchive.hpp>
 | |
| #include <boost/mpi/exception.hpp>
 | |
| #include <boost/utility/enable_if.hpp>
 | |
| #include <boost/mpl/assert.hpp>
 | |
| #include <boost/noncopyable.hpp>
 | |
| #include <typeinfo>
 | |
| 
 | |
| // The std::type_info::before function in Visual C++ 8.0 (and probably earlier)
 | |
| // incorrectly returns an "int" instead of a "bool". Then the compiler has the
 | |
| // audacity to complain when that "int" is converted to a "bool". Silence
 | |
| // this warning.
 | |
| #ifdef BOOST_MSVC
 | |
| #  pragma warning(push)
 | |
| #  pragma warning(disable : 4800)
 | |
| #endif
 | |
| 
 | |
| namespace boost { namespace mpi { namespace detail {
 | |
| 
 | |
| /// @brief comparison function object for two std::type_info pointers
 | |
| ///
 | |
| /// is implemented using the before() member function of the std::type_info
 | |
| /// class
 | |
| 
 | |
| struct type_info_compare
 | |
| {
 | |
|   bool operator()(std::type_info const* lhs, std::type_info const* rhs) const
 | |
|   {
 | |
|     return lhs->before(*rhs);
 | |
|   }
 | |
| };
 | |
| 
 | |
| 
 | |
| /// @brief a map of MPI data types, indexed by their type_info
 | |
| ///
 | |
| ///
 | |
| class BOOST_MPI_DECL mpi_datatype_map
 | |
|  : public boost::noncopyable
 | |
| {
 | |
|   struct implementation;
 | |
| 
 | |
|   implementation *impl;
 | |
| 
 | |
| public:
 | |
|   mpi_datatype_map();
 | |
|   ~mpi_datatype_map();
 | |
| 
 | |
|   template <class T>
 | |
|   MPI_Datatype datatype(const T& x = T(), typename boost::enable_if<is_mpi_builtin_datatype<T> >::type* =0)
 | |
|   {
 | |
|     return get_mpi_datatype<T>(x);
 | |
|   }
 | |
| 
 | |
|   template <class T>
 | |
|   MPI_Datatype datatype(const T& x =T(), typename boost::disable_if<is_mpi_builtin_datatype<T> >::type* =0 )
 | |
|   {
 | |
|     BOOST_MPL_ASSERT((is_mpi_datatype<T>));
 | |
| 
 | |
|     // check whether the type already exists
 | |
|     std::type_info const* t = &typeid(T);
 | |
|     MPI_Datatype datatype = get(t);
 | |
|     if (datatype == MPI_DATATYPE_NULL) {
 | |
|       // need to create a type
 | |
|       mpi_datatype_oarchive ar(x);
 | |
|       datatype = ar.get_mpi_datatype();
 | |
|       set(t, datatype);
 | |
|     }
 | |
| 
 | |
|     return datatype;
 | |
|   }
 | |
|   
 | |
|   void clear(); 
 | |
| 
 | |
| private:
 | |
|   MPI_Datatype get(const std::type_info* t);
 | |
|   void set(const std::type_info* t, MPI_Datatype datatype);
 | |
| };
 | |
| 
 | |
| /// Retrieve the MPI datatype cache
 | |
| BOOST_MPI_DECL mpi_datatype_map& mpi_datatype_cache();
 | |
| 
 | |
| } } } // end namespace boost::mpi::detail
 | |
| 
 | |
| #ifdef BOOST_MSVC
 | |
| #  pragma warning(pop)
 | |
| #endif
 | |
| 
 | |
| #endif // BOOST_MPI_DETAIL_TYPE_MPI_DATATYPE_CACHE_HPP
 | 
