// 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) 2020 Intel Corporation #include "../test_precomp.hpp" #include #include #include "../src/backends/common/gbackend.hpp" namespace opencv_test { using cv::GMatDesc; using View = cv::RMat::View; using cv::Mat; using cv::gimpl::asMat; using cv::gimpl::asView; using namespace ::testing; static void expect_eq_desc(const GMatDesc& desc, const View& view) { EXPECT_EQ(desc.size, view.size()); EXPECT_EQ(desc.dims, view.dims()); EXPECT_EQ(desc.size.width, view.cols()); EXPECT_EQ(desc.size.height, view.rows()); EXPECT_EQ(desc.depth, view.depth()); EXPECT_EQ(desc.chan, view.chan()); EXPECT_EQ(desc.depth, view.depth()); EXPECT_EQ(desc.chan, view.chan()); } TEST(RMatView, TestDefaultConstruction) { View view; GMatDesc desc{}; expect_eq_desc(desc, view); EXPECT_EQ(nullptr, view.ptr()); EXPECT_EQ(0u, view.step()); } struct RMatViewTest : public TestWithParam{}; TEST_P(RMatViewTest, ConstructionFromMat) { auto type = GetParam(); Mat mat(8,8,type); const auto desc = cv::descr_of(mat); View view = asView(mat); expect_eq_desc(desc, view); EXPECT_EQ(mat.ptr(), view.ptr()); EXPECT_EQ(mat.step, view.step()); } TEST(RMatView, TestConstructionFromMatND) { std::vector dims(4, 8); Mat mat(dims, CV_8UC1); const auto desc = cv::descr_of(mat); View view(cv::descr_of(mat), mat.ptr()); expect_eq_desc(desc, view); EXPECT_EQ(mat.ptr(), view.ptr()); } TEST_P(RMatViewTest, DefaultStep) { auto type = GetParam(); GMatDesc desc; desc.chan = CV_MAT_CN(type); desc.depth = CV_MAT_DEPTH(type); desc.size = {8,8}; std::vector data(desc.size.width*desc.size.height*CV_ELEM_SIZE(type)); View view(desc, data.data()); EXPECT_EQ(static_cast(desc.size.width)*CV_ELEM_SIZE(type), view.step()); } struct RMatViewNDTest : public TestWithParam< std::tuple>{}; TEST_P(RMatViewNDTest, DefaultStep) { int depth = 0, ndims = 0; std::tie(depth, ndims) = GetParam(); std::vector dims(ndims, 12); GMatDesc desc; desc.dims = dims; desc.depth = depth; GAPI_Assert(desc.chan == -1); auto elemSize = CV_ELEM_SIZE(depth); auto total = std::accumulate(dims.begin(), dims.end(), elemSize, std::multiplies()); std::vector data(total); View view(desc, data.data()); auto step = static_cast(total/dims[0]); EXPECT_EQ(step, view.step(0)); for (int i = 1; i < ndims; i++) { step /= dims[i]; EXPECT_EQ(step, view.step(i)); } } TEST_P(RMatViewNDTest, StepFromMat) { int depth = 0, ndims = 0; std::tie(depth, ndims) = GetParam(); std::vector dims(ndims, 12); cv::Mat mat(dims, depth); auto view = asView(mat); EXPECT_EQ(mat.ptr(), view.ptr()); for (int i = 0; i < ndims; i++) { EXPECT_EQ(mat.step[i], view.step(i)); } } TEST_P(RMatViewNDTest, StepFromView) { int depth = 0, ndims = 0; std::tie(depth, ndims) = GetParam(); std::vector dims(ndims, 12); std::vector aligned(ndims, 16); GMatDesc desc; desc.dims = dims; desc.depth = depth; GAPI_Assert(desc.chan == -1); auto elemSize = CV_ELEM_SIZE(depth); auto total = std::accumulate(aligned.begin(), aligned.end(), elemSize, std::multiplies()); std::vector data(total); View::stepsT steps(ndims); auto step = static_cast(total/aligned[0]); steps[0] = step; for (int i = 1; i < ndims; i++) { step /= aligned[i]; steps[i] = step; } View view(desc, data.data(), steps); auto mat = asMat(view); EXPECT_EQ(mat.ptr(), view.ptr()); for (int i = 0; i < ndims; i++) { EXPECT_EQ(mat.step[i], view.step(i)); } } INSTANTIATE_TEST_CASE_P(Test, RMatViewNDTest, Combine(Values(CV_8U, CV_32F), // depth Values(1,2,3,4,7))); // ndims struct RMatViewNDTestNegative : public TestWithParam< std::tuple>{}; TEST_P(RMatViewNDTestNegative, DefaultStep) { int depth = 0, chan = 0, ndims = 0; std::tie(depth, chan, ndims) = GetParam(); std::vector dims(ndims, 12); GMatDesc desc; desc.dims = dims; desc.depth = depth; desc.chan = chan; auto elemSize = CV_ELEM_SIZE(depth); auto total = std::accumulate(dims.begin(), dims.end(), elemSize, std::multiplies()); std::vector data(total); EXPECT_ANY_THROW(View view(desc, data.data())); } INSTANTIATE_TEST_CASE_P(Test, RMatViewNDTestNegative, Combine(Values(CV_8U, CV_32F), // depth Values(1,2,3,4), // chan Values(2,4,7))); // ndims TEST_P(RMatViewTest, NonDefaultStepInput) { auto type = GetParam(); Mat bigMat(16,16,type); cv::randn(bigMat, cv::Scalar::all(127), cv::Scalar::all(40)); Mat mat = bigMat(cv::Rect{4,4,8,8}); View view = asView(mat); const auto viewMat = asMat(view); Mat ref, out; cv::Size ksize{1,1}; cv::blur(viewMat, out, ksize); cv::blur( mat, ref, ksize); EXPECT_EQ(0, cvtest::norm(ref, out, NORM_INF)); } TEST_P(RMatViewTest, NonDefaultStepOutput) { auto type = GetParam(); Mat mat(8,8,type); cv::randn(mat, cv::Scalar::all(127), cv::Scalar::all(40)); Mat bigMat = Mat::zeros(16,16,type); Mat out = bigMat(cv::Rect{4,4,8,8}); View view = asView(out); auto viewMat = asMat(view); Mat ref; cv::Size ksize{1,1}; cv::blur(mat, viewMat, ksize); cv::blur(mat, ref, ksize); EXPECT_EQ(0, cvtest::norm(ref, out, NORM_INF)); } TEST_P(RMatViewTest, NonDefaultStep2DInput) { auto type = GetParam(); Mat bigMat(16,16,type); cv::randn(bigMat, cv::Scalar::all(127), cv::Scalar::all(40)); Mat mat = bigMat(cv::Rect{4,4,8,8}); View view(cv::descr_of(mat), mat.data, mat.step); const auto viewMat = asMat(view); Mat ref, out; cv::Size ksize{1,1}; cv::blur(viewMat, out, ksize); cv::blur( mat, ref, ksize); EXPECT_EQ(0, cvtest::norm(ref, out, NORM_INF)); } TEST_P(RMatViewTest, NonDefaultStep2DOutput) { auto type = GetParam(); Mat mat(8,8,type); cv::randn(mat, cv::Scalar::all(127), cv::Scalar::all(40)); Mat bigMat = Mat::zeros(16,16,type); Mat out = bigMat(cv::Rect{4,4,8,8}); View view(cv::descr_of(out), out.data, out.step); auto viewMat = asMat(view); Mat ref; cv::Size ksize{1,1}; cv::blur(mat, viewMat, ksize); cv::blur(mat, ref, ksize); EXPECT_EQ(0, cvtest::norm(ref, out, NORM_INF)); } INSTANTIATE_TEST_CASE_P(Test, RMatViewTest, Values(CV_8UC1, CV_8UC3, CV_32FC1)); struct RMatViewCallbackTest : public ::testing::Test { RMatViewCallbackTest() : mat(8,8,CV_8UC1) { cv::randn(mat, cv::Scalar::all(127), cv::Scalar::all(40)); } View getView() { return asView(mat, [this](){ callbackCalls++; }); } int callbackCalls = 0; Mat mat; }; TEST_F(RMatViewCallbackTest, MoveCtor) { { View copy(getView()); cv::util::suppress_unused_warning(copy); EXPECT_EQ(0, callbackCalls); } EXPECT_EQ(1, callbackCalls); } TEST_F(RMatViewCallbackTest, MoveCopy) { { View copy; copy = getView(); cv::util::suppress_unused_warning(copy); EXPECT_EQ(0, callbackCalls); } EXPECT_EQ(1, callbackCalls); } static int firstElement(const View& view) { return *view.ptr(); } static void setFirstElement(View& view, uchar value) { *view.ptr() = value; } TEST_F(RMatViewCallbackTest, MagazineInteraction) { cv::gimpl::magazine::Class mag; constexpr int rc = 1; constexpr uchar value = 11; mag.slot()[rc] = getView(); { auto& mag_view = mag.slot()[rc]; setFirstElement(mag_view, value); auto mag_el = firstElement(mag_view); EXPECT_EQ(value, mag_el); } { const auto& mag_view = mag.slot()[rc]; auto mag_el = firstElement(mag_view); EXPECT_EQ(value, mag_el); } EXPECT_EQ(0, callbackCalls); mag.slot().erase(rc); EXPECT_EQ(1, callbackCalls); } TEST(RMatView, Access1DMat) { cv::Mat m({1}, CV_32FC1); m.dims = 1; auto rmat = cv::make_rmat(m); auto view = rmat.access(cv::RMat::Access::R); auto out = cv::gimpl::asMat(view); EXPECT_EQ(1, out.dims); } } // namespace opencv_test