#pragma once #include #include #include #include #include template struct CV_TYPE { static const int DEPTH; }; template<> static const int CV_TYPE::DEPTH = CV_32F; template<> static const int CV_TYPE::DEPTH = CV_64F; template<> static const int CV_TYPE::DEPTH = CV_32S; template<> static const int CV_TYPE::DEPTH = CV_8U; template<> static const int CV_TYPE::DEPTH = CV_8S; template<> static const int CV_TYPE::DEPTH = CV_16U; template<> static const int CV_TYPE::DEPTH = CV_16S; template struct step_functor : public thrust::unary_function { int columns; int step; int channels; __host__ __device__ step_functor(int columns_, int step_, int channels_ = 1) : columns(columns_), step(step_), channels(channels_) { }; __host__ step_functor(cv::cuda::GpuMat& mat) { CV_Assert(mat.depth() == CV_TYPE::DEPTH); columns = mat.cols; step = mat.step / sizeof(T); channels = mat.channels(); } __host__ __device__ int operator()(int x) const { int row = x / columns; int idx = (row * step) + (x % columns)*channels; return idx; } }; /* @Brief GpuMatBeginItr returns a thrust compatible iterator to the beginning of a GPU mat's memory. @Param mat is the input matrix @Param channel is the channel of the matrix that the iterator is accessing. If set to -1, the iterator will access every element in sequential order */ template thrust::permutation_iterator, thrust::transform_iterator, thrust::counting_iterator>> GpuMatBeginItr(cv::cuda::GpuMat mat, int channel = 0) { if (channel == -1) mat = mat.reshape(1); CV_Assert(mat.depth() == CV_TYPE::DEPTH); CV_Assert(channel < mat.channels()); return thrust::make_permutation_iterator(thrust::device_pointer_cast(mat.ptr(0) + channel), thrust::make_transform_iterator(thrust::make_counting_iterator(0), step_functor(mat.cols, mat.step / sizeof(T), mat.channels()))); } /* @Brief GpuMatEndItr returns a thrust compatible iterator to the end of a GPU mat's memory. @Param mat is the input matrix @Param channel is the channel of the matrix that the iterator is accessing. If set to -1, the iterator will access every element in sequential order */ template thrust::permutation_iterator, thrust::transform_iterator, thrust::counting_iterator>> GpuMatEndItr(cv::cuda::GpuMat mat, int channel = 0) { if (channel == -1) mat = mat.reshape(1); CV_Assert(mat.depth() == CV_TYPE::DEPTH); CV_Assert(channel < mat.channels()); return thrust::make_permutation_iterator(thrust::device_pointer_cast(mat.ptr(0) + channel), thrust::make_transform_iterator(thrust::make_counting_iterator(mat.rows*mat.cols), step_functor(mat.cols, mat.step / sizeof(T), mat.channels()))); }