// This file is part of OpenCV project. // It is subject to the license terms in the LICENSE file found in the top-level directory // of this distribution and at http://opencv.org/license.html. // // Copyright (C) 2018 Intel Corporation #ifndef OPENCV_GAPI_OWN_MAT_HPP #define OPENCV_GAPI_OWN_MAT_HPP #include #include #include #include #include #include //std::shared_ptr #include //std::memcpy #include //std::accumulate #include #include namespace cv { namespace gapi { namespace own { namespace detail { template void assign_row(void* ptr, int cols, Scalar const& s) { auto p = static_cast(ptr); for (int c = 0; c < cols; c++) { for (int ch = 0; ch < channels; ch++) { p[c * channels + ch] = saturate(s[ch], roundd); } } } inline size_t default_step(int type, int cols) { return CV_ELEM_SIZE(type) * cols; } //Matrix header, i.e. fields that are unique to each Mat object. //Devoted class is needed to implement custom behavior on move (erasing state of moved from object) struct MatHeader{ enum { AUTO_STEP = 0}; enum { TYPE_MASK = 0x00000FFF }; MatHeader() = default; MatHeader(int _rows, int _cols, int type, void* _data, size_t _step) : flags((type & TYPE_MASK)), rows(_rows), cols(_cols), data((uchar*)_data), step(_step == AUTO_STEP ? detail::default_step(type, _cols) : _step) {} MatHeader(const std::vector &_dims, int type, void* _data) : flags((type & TYPE_MASK)), data((uchar*)_data), step(0), dims(_dims) {} MatHeader(const MatHeader& ) = default; MatHeader(MatHeader&& src) : MatHeader(src) // reuse copy constructor here { MatHeader empty; //give it a name to call copy(not move) assignment below src = empty; } MatHeader& operator=(const MatHeader& ) = default; MatHeader& operator=(MatHeader&& src) { *this = src; //calling a copy assignment here, not move one MatHeader empty; //give it a name to call copy(not move) assignment below src = empty; return *this; } /*! includes several bit-fields: - depth - number of channels */ int flags = 0; //! the number of rows and columns or (-1, -1) when the matrix has more than 2 dimensions int rows = 0, cols = 0; //! pointer to the data uchar* data = nullptr; size_t step = 0; //! dimensions (ND-case) std::vector dims; }; } // namespace detail //concise version of cv::Mat suitable for GAPI needs (used when no dependence on OpenCV is required) class Mat : public detail::MatHeader{ public: Mat() = default; /** @overload @param _rows Number of rows in a 2D array. @param _cols Number of columns in a 2D array. @param _type Array type. Use CV_8UC1, ..., CV_64FC4 to create 1-4 channel matrices, or CV_8UC(n), ..., CV_64FC(n) to create multi-channel (up to CV_CN_MAX channels) matrices. @param _data Pointer to the user data. Matrix constructors that take data and step parameters do not allocate matrix data. Instead, they just initialize the matrix header that points to the specified data, which means that no data is copied. This operation is very efficient and can be used to process external data using OpenCV functions. The external data is not automatically deallocated, so you should take care of it. @param _step Number of bytes each matrix row occupies. The value should include the padding bytes at the end of each row, if any. If the parameter is missing (set to AUTO_STEP ), no padding is assumed and the actual step is calculated as cols*elemSize(). See Mat::elemSize. */ Mat(int _rows, int _cols, int _type, void* _data, size_t _step = AUTO_STEP) : MatHeader (_rows, _cols, _type, _data, _step) {} Mat(const std::vector &_dims, int _type, void* _data) : MatHeader (_dims, _type, _data) {} Mat(std::vector &&_dims, int _type, void* _data) : MatHeader (std::move(_dims), _type, _data) {} Mat(Mat const& src, const Rect& roi ) : Mat(src) { rows = roi.height; cols = roi.width; data = ptr(roi.y, roi.x); } Mat(Mat const& ) = default; Mat(Mat&& ) = default; Mat& operator=(Mat const& ) = default; Mat& operator=(Mat&& ) = default; /** @brief Sets all or some of the array elements to the specified value. @param s Assigned scalar converted to the actual array type. */ Mat& operator = (const Scalar& s) { constexpr unsigned max_channels = 4; //Scalar can't fit more than 4 using func_p_t = void (*)(void*, int, Scalar const&); using detail::assign_row; #define TABLE_ENTRY(type) {assign_row, assign_row, assign_row, assign_row} static constexpr func_p_t func_tbl[][max_channels] = { TABLE_ENTRY(uchar), TABLE_ENTRY(schar), TABLE_ENTRY(ushort), TABLE_ENTRY(short), TABLE_ENTRY(int), TABLE_ENTRY(float), TABLE_ENTRY(double) }; #undef TABLE_ENTRY static_assert(CV_8U == 0 && CV_8S == 1 && CV_16U == 2 && CV_16S == 3 && CV_32S == 4 && CV_32F == 5 && CV_64F == 6, "OCV type ids used as indexes to array, thus exact numbers are important!" ); const auto depth = static_cast(this->depth()); GAPI_Assert(depth < sizeof(func_tbl)/sizeof(func_tbl[0])); if (dims.empty()) { const auto channels = static_cast(this->channels()); GAPI_Assert(channels <= max_channels); auto* f = func_tbl[depth][channels - 1]; for (int r = 0; r < rows; ++r) { (*f)(static_cast(ptr(r)), cols, s ); } } else { auto* f = func_tbl[depth][0]; // FIXME: better to refactor assign_row to use std::size_t by default (*f)(static_cast(data), static_cast(total()), s); } return *this; } /** @brief Returns the matrix element size in bytes. The method returns the matrix element size in bytes. For example, if the matrix type is CV_16SC3 , the method returns 3\*sizeof(short) or 6. */ size_t elemSize() const { return CV_ELEM_SIZE(type()); } /** @brief Returns the type of a matrix element. The method returns a matrix element type. This is an identifier compatible with the CvMat type system, like CV_16SC3 or 16-bit signed 3-channel array, and so on. */ int type() const {return CV_MAT_TYPE(flags);} /** @brief Returns the depth of a matrix element. The method returns the identifier of the matrix element depth (the type of each individual channel). For example, for a 16-bit signed element array, the method returns CV_16S . A complete list of matrix types contains the following values: - CV_8U - 8-bit unsigned integers ( 0..255 ) - CV_8S - 8-bit signed integers ( -128..127 ) - CV_16U - 16-bit unsigned integers ( 0..65535 ) - CV_16S - 16-bit signed integers ( -32768..32767 ) - CV_32S - 32-bit signed integers ( -2147483648..2147483647 ) - CV_32F - 32-bit floating-point numbers ( -FLT_MAX..FLT_MAX, INF, NAN ) - CV_64F - 64-bit floating-point numbers ( -DBL_MAX..DBL_MAX, INF, NAN ) */ int depth() const {return CV_MAT_DEPTH(flags);} /** @brief Returns the number of matrix channels. The method returns the number of matrix channels. If matrix is N-dimensional, -1 is returned. */ int channels() const {return dims.empty() ? CV_MAT_CN(flags) : -1;} /** @param _rows New number of rows. @param _cols New number of columns. @param _type New matrix type. */ void create(int _rows, int _cols, int _type) { create(Size{_cols, _rows}, _type); } /** @overload @param _size Alternative new matrix size specification: Size(cols, rows) @param _type New matrix type. */ void create(Size _size, int _type) { GAPI_Assert(_size.height >= 0 && _size.width >= 0); if (_size != Size{cols, rows} ) { Mat tmp{_size.height, _size.width, _type, nullptr}; tmp.memory.reset(new uchar[ tmp.step * tmp.rows], [](uchar * p){delete[] p;}); tmp.data = tmp.memory.get(); *this = std::move(tmp); } } void create(const std::vector &_dims, int _type) { // FIXME: make a proper reallocation-on-demands // WARNING: no tensor views, so no strides Mat tmp{_dims, _type, nullptr}; // FIXME: this accumulate duplicates a lot const auto sz = std::accumulate(_dims.begin(), _dims.end(), 1, std::multiplies()); tmp.memory.reset(new uchar[CV_ELEM_SIZE(_type)*sz], [](uchar * p){delete[] p;}); tmp.data = tmp.memory.get(); *this = std::move(tmp); } /** @brief Creates a full copy of the matrix and the underlying data. The method creates a full copy of the matrix. The original step[] is not taken into account. So, the copy has a continuous buffer occupying total() * elemSize() bytes. */ Mat clone() const { Mat m; copyTo(m); return m; } /** @brief Copies the matrix to another one. The method copies the matrix data to another matrix. Before copying the data, the method invokes : @code m.create(this->size(), this->type()); @endcode so that the destination matrix is reallocated if needed. While m.copyTo(m); works flawlessly, the function does not handle the case of a partial overlap between the source and the destination matrices. */ void copyTo(Mat& dst) const { if (dims.empty()) { dst.create(rows, cols, type()); for (int r = 0; r < rows; ++r) { std::copy_n(ptr(r), detail::default_step(type(),cols), dst.ptr(r)); } } else { dst.create(dims, depth()); std::copy_n(data, total()*elemSize(), data); } } /** @brief Returns true if the array has no elements. The method returns true if Mat::total() is 0 or if Mat::data is NULL. Because of pop_back() and resize() methods `M.total() == 0` does not imply that `M.data == NULL`. */ bool empty() const { return data == 0 || total() == 0; } /** @brief Returns the total number of array elements. The method returns the number of array elements (a number of pixels if the array represents an image). */ size_t total() const { return dims.empty() ? (static_cast(rows) * cols) : std::accumulate(dims.begin(), dims.end(), static_cast(1), std::multiplies()); } /** @overload @param roi Extracted submatrix specified as a rectangle. */ Mat operator()( const Rect& roi ) const { return Mat{*this, roi}; } /** @brief Returns a pointer to the specified matrix row. The methods return `uchar*` or typed pointer to the specified matrix row. See the sample in Mat::isContinuous to know how to use these methods. @param row Index along the dimension 0 @param col Index along the dimension 1 */ uchar* ptr(int row, int col = 0) { return const_cast(const_cast(this)->ptr(row,col)); } /** @overload */ const uchar* ptr(int row, int col = 0) const { return data + step * row + CV_ELEM_SIZE(type()) * col; } private: //actual memory allocated for storage, or nullptr if object is non owning view to over memory std::shared_ptr memory; }; } //namespace own } //namespace gapi } //namespace cv #endif /* OPENCV_GAPI_OWN_MAT_HPP */