355 lines
13 KiB
C++
355 lines
13 KiB
C++
|
// 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 <opencv2/gapi/opencv_includes.hpp>
|
||
|
#include <opencv2/gapi/own/types.hpp>
|
||
|
#include <opencv2/gapi/own/scalar.hpp>
|
||
|
#include <opencv2/gapi/own/saturate.hpp>
|
||
|
#include <opencv2/gapi/own/assert.hpp>
|
||
|
|
||
|
#include <memory> //std::shared_ptr
|
||
|
#include <cstring> //std::memcpy
|
||
|
#include <numeric> //std::accumulate
|
||
|
#include <vector>
|
||
|
#include <opencv2/gapi/util/throw.hpp>
|
||
|
|
||
|
namespace cv { namespace gapi { namespace own {
|
||
|
namespace detail {
|
||
|
template <typename T, unsigned char channels>
|
||
|
void assign_row(void* ptr, int cols, Scalar const& s)
|
||
|
{
|
||
|
auto p = static_cast<T*>(ptr);
|
||
|
for (int c = 0; c < cols; c++)
|
||
|
{
|
||
|
for (int ch = 0; ch < channels; ch++)
|
||
|
{
|
||
|
p[c * channels + ch] = saturate<T>(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<int> &_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<int> 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<int> &_dims, int _type, void* _data)
|
||
|
: MatHeader (_dims, _type, _data)
|
||
|
{}
|
||
|
|
||
|
Mat(std::vector<int> &&_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<type, 1>, assign_row<type, 2>, assign_row<type, 3>, assign_row<type, 4>}
|
||
|
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<unsigned int>(this->depth());
|
||
|
GAPI_Assert(depth < sizeof(func_tbl)/sizeof(func_tbl[0]));
|
||
|
|
||
|
if (dims.empty())
|
||
|
{
|
||
|
const auto channels = static_cast<unsigned int>(this->channels());
|
||
|
GAPI_Assert(channels <= max_channels);
|
||
|
|
||
|
auto* f = func_tbl[depth][channels - 1];
|
||
|
for (int r = 0; r < rows; ++r)
|
||
|
{
|
||
|
(*f)(static_cast<void *>(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<void *>(data), static_cast<int>(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<int> &_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<int>());
|
||
|
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<std::size_t>(rows) * cols)
|
||
|
: std::accumulate(dims.begin(), dims.end(), static_cast<std::size_t>(1), std::multiplies<size_t>());
|
||
|
}
|
||
|
|
||
|
/** @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<uchar*>(const_cast<const Mat*>(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<uchar> memory;
|
||
|
};
|
||
|
|
||
|
} //namespace own
|
||
|
} //namespace gapi
|
||
|
} //namespace cv
|
||
|
|
||
|
#endif /* OPENCV_GAPI_OWN_MAT_HPP */
|