217 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
		
		
			
		
	
	
			217 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
| 
								 | 
							
								//---------------------------------------------------------------------------//
							 | 
						||
| 
								 | 
							
								// Copyright (c) 2013-2014 Kyle Lutz <kyle.r.lutz@gmail.com>
							 | 
						||
| 
								 | 
							
								//
							 | 
						||
| 
								 | 
							
								// 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
							 | 
						||
| 
								 | 
							
								//
							 | 
						||
| 
								 | 
							
								// See http://boostorg.github.com/compute for more information.
							 | 
						||
| 
								 | 
							
								//---------------------------------------------------------------------------//
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#ifndef BOOST_COMPUTE_DETAIL_GET_OBJECT_INFO_HPP
							 | 
						||
| 
								 | 
							
								#define BOOST_COMPUTE_DETAIL_GET_OBJECT_INFO_HPP
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#include <string>
							 | 
						||
| 
								 | 
							
								#include <vector>
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#include <boost/preprocessor/seq/for_each.hpp>
							 | 
						||
| 
								 | 
							
								#include <boost/preprocessor/tuple/elem.hpp>
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#include <boost/throw_exception.hpp>
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#include <boost/compute/cl.hpp>
							 | 
						||
| 
								 | 
							
								#include <boost/compute/exception/opencl_error.hpp>
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								namespace boost {
							 | 
						||
| 
								 | 
							
								namespace compute {
							 | 
						||
| 
								 | 
							
								namespace detail {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								template<class Function, class Object, class AuxInfo>
							 | 
						||
| 
								 | 
							
								struct bound_info_function
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								    bound_info_function(Function function, Object object, AuxInfo aux_info)
							 | 
						||
| 
								 | 
							
								        : m_function(function),
							 | 
						||
| 
								 | 
							
								          m_object(object),
							 | 
						||
| 
								 | 
							
								          m_aux_info(aux_info)
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    template<class Info>
							 | 
						||
| 
								 | 
							
								    cl_int operator()(Info info, size_t size, void *value, size_t *size_ret) const
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        return m_function(m_object, m_aux_info, info, size, value, size_ret);
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    Function m_function;
							 | 
						||
| 
								 | 
							
								    Object m_object;
							 | 
						||
| 
								 | 
							
								    AuxInfo m_aux_info;
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								template<class Function, class Object>
							 | 
						||
| 
								 | 
							
								struct bound_info_function<Function, Object, void>
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								    bound_info_function(Function function, Object object)
							 | 
						||
| 
								 | 
							
								        : m_function(function),
							 | 
						||
| 
								 | 
							
								          m_object(object)
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    template<class Info>
							 | 
						||
| 
								 | 
							
								    cl_int operator()(Info info, size_t size, void *value, size_t *size_ret) const
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        return m_function(m_object, info, size, value, size_ret);
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    Function m_function;
							 | 
						||
| 
								 | 
							
								    Object m_object;
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								template<class Function, class Object>
							 | 
						||
| 
								 | 
							
								inline bound_info_function<Function, Object, void>
							 | 
						||
| 
								 | 
							
								bind_info_function(Function f, Object o)
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								    return bound_info_function<Function, Object, void>(f, o);
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								template<class Function, class Object, class AuxInfo>
							 | 
						||
| 
								 | 
							
								inline bound_info_function<Function, Object, AuxInfo>
							 | 
						||
| 
								 | 
							
								bind_info_function(Function f, Object o, AuxInfo j)
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								    return bound_info_function<Function, Object, AuxInfo>(f, o, j);
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								// default implementation
							 | 
						||
| 
								 | 
							
								template<class T>
							 | 
						||
| 
								 | 
							
								struct get_object_info_impl
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								    template<class Function, class Info>
							 | 
						||
| 
								 | 
							
								    T operator()(Function function, Info info) const
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        T value;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        cl_int ret = function(info, sizeof(T), &value, 0);
							 | 
						||
| 
								 | 
							
								        if(ret != CL_SUCCESS){
							 | 
						||
| 
								 | 
							
								            BOOST_THROW_EXCEPTION(opencl_error(ret));
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        return value;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								// specialization for bool
							 | 
						||
| 
								 | 
							
								template<>
							 | 
						||
| 
								 | 
							
								struct get_object_info_impl<bool>
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								    template<class Function, class Info>
							 | 
						||
| 
								 | 
							
								    bool operator()(Function function, Info info) const
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        cl_bool value;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        cl_int ret = function(info, sizeof(cl_bool), &value, 0);
							 | 
						||
| 
								 | 
							
								        if(ret != CL_SUCCESS){
							 | 
						||
| 
								 | 
							
								            BOOST_THROW_EXCEPTION(opencl_error(ret));
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        return value == CL_TRUE;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								// specialization for std::string
							 | 
						||
| 
								 | 
							
								template<>
							 | 
						||
| 
								 | 
							
								struct get_object_info_impl<std::string>
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								    template<class Function, class Info>
							 | 
						||
| 
								 | 
							
								    std::string operator()(Function function, Info info) const
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        size_t size = 0;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        cl_int ret = function(info, 0, 0, &size);
							 | 
						||
| 
								 | 
							
								        if(ret != CL_SUCCESS){
							 | 
						||
| 
								 | 
							
								            BOOST_THROW_EXCEPTION(opencl_error(ret));
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if(size == 0){
							 | 
						||
| 
								 | 
							
								            return std::string();
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        std::string value(size - 1, 0);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        ret = function(info, size, &value[0], 0);
							 | 
						||
| 
								 | 
							
								        if(ret != CL_SUCCESS){
							 | 
						||
| 
								 | 
							
								            BOOST_THROW_EXCEPTION(opencl_error(ret));
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        return value;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								// specialization for std::vector<T>
							 | 
						||
| 
								 | 
							
								template<class T>
							 | 
						||
| 
								 | 
							
								struct get_object_info_impl<std::vector<T> >
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								    template<class Function, class Info>
							 | 
						||
| 
								 | 
							
								    std::vector<T> operator()(Function function, Info info) const
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        size_t size = 0;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        cl_int ret = function(info, 0, 0, &size);
							 | 
						||
| 
								 | 
							
								        if(ret != CL_SUCCESS){
							 | 
						||
| 
								 | 
							
								            BOOST_THROW_EXCEPTION(opencl_error(ret));
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        std::vector<T> vector(size / sizeof(T));
							 | 
						||
| 
								 | 
							
								        ret = function(info, size, &vector[0], 0);
							 | 
						||
| 
								 | 
							
								        if(ret != CL_SUCCESS){
							 | 
						||
| 
								 | 
							
								            BOOST_THROW_EXCEPTION(opencl_error(ret));
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        return vector;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								// returns the value (of type T) from the given clGet*Info() function call.
							 | 
						||
| 
								 | 
							
								template<class T, class Function, class Object, class Info>
							 | 
						||
| 
								 | 
							
								inline T get_object_info(Function f, Object o, Info i)
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								    return get_object_info_impl<T>()(bind_info_function(f, o), i);
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								template<class T, class Function, class Object, class Info, class AuxInfo>
							 | 
						||
| 
								 | 
							
								inline T get_object_info(Function f, Object o, Info i, AuxInfo j)
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								    return get_object_info_impl<T>()(bind_info_function(f, o, j), i);
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								// returns the value type for the clGet*Info() call on Object with Enum.
							 | 
						||
| 
								 | 
							
								template<class Object, int Enum>
							 | 
						||
| 
								 | 
							
								struct get_object_info_type;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								// defines the object::get_info<Enum>() specialization
							 | 
						||
| 
								 | 
							
								#define BOOST_COMPUTE_DETAIL_DEFINE_GET_INFO_SPECIALIZATION(object_type, result_type, value) \
							 | 
						||
| 
								 | 
							
								    namespace detail { \
							 | 
						||
| 
								 | 
							
								        template<> struct get_object_info_type<object_type, value> { typedef result_type type; }; \
							 | 
						||
| 
								 | 
							
								    } \
							 | 
						||
| 
								 | 
							
								    template<> inline result_type object_type::get_info<value>() const \
							 | 
						||
| 
								 | 
							
								    { \
							 | 
						||
| 
								 | 
							
								        return get_info<result_type>(value); \
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								// used by BOOST_COMPUTE_DETAIL_DEFINE_GET_INFO_SPECIALIZATIONS()
							 | 
						||
| 
								 | 
							
								#define BOOST_COMPUTE_DETAIL_DEFINE_GET_INFO_IMPL(r, data, elem) \
							 | 
						||
| 
								 | 
							
								    BOOST_COMPUTE_DETAIL_DEFINE_GET_INFO_SPECIALIZATION( \
							 | 
						||
| 
								 | 
							
								        data, BOOST_PP_TUPLE_ELEM(2, 0, elem), BOOST_PP_TUPLE_ELEM(2, 1, elem) \
							 | 
						||
| 
								 | 
							
								    )
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								// defines the object::get_info<Enum>() specialization for each
							 | 
						||
| 
								 | 
							
								// (result_type, value) tuple in seq for object_type.
							 | 
						||
| 
								 | 
							
								#define BOOST_COMPUTE_DETAIL_DEFINE_GET_INFO_SPECIALIZATIONS(object_type, seq) \
							 | 
						||
| 
								 | 
							
								    BOOST_PP_SEQ_FOR_EACH( \
							 | 
						||
| 
								 | 
							
								        BOOST_COMPUTE_DETAIL_DEFINE_GET_INFO_IMPL, object_type, seq \
							 | 
						||
| 
								 | 
							
								    )
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								} // end detail namespace
							 | 
						||
| 
								 | 
							
								} // end compute namespace
							 | 
						||
| 
								 | 
							
								} // end boost namespace
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#endif // BOOST_COMPUTE_DETAIL_GET_OBJECT_INFO_HPP
							 |