Skip to content

Commit a00b3f3

Browse files
authored
Merge pull request #3845 from CodeLinaro:adsha_2ndPost
FastCV Extension code for OpenCV 2ndpost-2 #3845 ### Description: - Add support for cv::fastcv::calcHist Depends on: [#3844](#3844) Depends on: [opencv/opencv#26619](opencv/opencv#26619) Requires binary from: [opencv/opencv_3rdparty#90](opencv/opencv_3rdparty#90) ### Pull Request Readiness Checklist See details at https://github.com/opencv/opencv/wiki/How_to_contribute#making-a-good-pull-request - [x] I agree to contribute to the project under Apache 2 License. - [x] To the best of my knowledge, the proposed patch is not based on a code under GPL or another license that is incompatible with OpenCV - [ ] The PR is proposed to the proper branch - [ ] There is a reference to the original bug report and related work - [ ] There is accuracy test, performance test and test data in opencv_extra repository, if applicable Patch to opencv_extra has the same branch name. - [ ] The feature is well documented and sample code can be built with the project CMake
1 parent 67815e9 commit a00b3f3

10 files changed

+217
-81
lines changed

modules/fastcv/include/opencv2/fastcv.hpp

+1
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,7 @@
1616
#include "opencv2/fastcv/edges.hpp"
1717
#include "opencv2/fastcv/fast10.hpp"
1818
#include "opencv2/fastcv/fft.hpp"
19+
#include "opencv2/fastcv/histogram.hpp"
1920
#include "opencv2/fastcv/hough.hpp"
2021
#include "opencv2/fastcv/ipptransform.hpp"
2122
#include "opencv2/fastcv/moments.hpp"
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,29 @@
1+
/*
2+
* Copyright (c) 2024 Qualcomm Innovation Center, Inc. All rights reserved.
3+
* SPDX-License-Identifier: Apache-2.0
4+
*/
5+
6+
#ifndef OPENCV_FASTCV_HISTOGRAM_HPP
7+
#define OPENCV_FASTCV_HISTOGRAM_HPP
8+
9+
#include <opencv2/core.hpp>
10+
11+
namespace cv {
12+
namespace fastcv {
13+
14+
//! @addtogroup fastcv
15+
//! @{
16+
17+
/**
18+
* @brief Calculates histogram of input image. This function implements specific use case of
19+
* 256-bin histogram calculation for 8u single channel images in an optimized way.
20+
* @param _src Intput image with type CV_8UC1
21+
* @param _hist Output histogram of type int of 256 bins
22+
*/
23+
CV_EXPORTS_W void calcHist( InputArray _src, OutputArray _hist );
24+
//! @}
25+
26+
} // fastcv::
27+
} // cv::
28+
29+
#endif // OPENCV_FASTCV_HISTOGRAM_HPP

modules/fastcv/include/opencv2/fastcv/moments.hpp

+2-1
Original file line numberDiff line numberDiff line change
@@ -16,7 +16,8 @@ namespace fastcv {
1616

1717
/**
1818
* @brief Calculates all of the moments up to the third order of the image pixels' intensities
19-
The results are returned in the structure cv::Moments.
19+
* The results are returned in the structure cv::Moments. This function cv::fastcv::moments()
20+
* calculate the moments using floating point calculations whereas cv::moments() calculate moments using double.
2021
* @param _src Input image with type CV_8UC1, CV_32SC1, CV_32FC1
2122
* @param binary If true, assumes the image to be binary (0x00 for black, 0xff for white), otherwise assumes the image to be
2223
* grayscale.

modules/fastcv/perf/perf_bilateral.cpp

+1-1
Original file line numberDiff line numberDiff line change
@@ -52,7 +52,7 @@ PERF_TEST_P(BilateralPerfTest, run,
5252

5353
RNG& rng = cv::theRNG();
5454
Mat src(size, CV_8UC1);
55-
cvtest::randUni(rng, src, Scalar::all(0), Scalar::all(256));
55+
cvtest::randUni(rng, src, Scalar::all(0), Scalar::all(255));
5656
Mat dst;
5757

5858
while (next())
+36
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,36 @@
1+
/*
2+
* Copyright (c) 2024 Qualcomm Innovation Center, Inc. All rights reserved.
3+
* SPDX-License-Identifier: Apache-2.0
4+
*/
5+
6+
#include "perf_precomp.hpp"
7+
8+
namespace opencv_test {
9+
10+
typedef std::tuple<cv::Size> HistogramPerfParams;
11+
typedef perf::TestBaseWithParam<HistogramPerfParams> HistogramPerfTest;
12+
13+
14+
PERF_TEST_P(HistogramPerfTest, run,
15+
testing::Values(perf::szQVGA, perf::szVGA, perf::sz720p, perf::sz1080p)
16+
)
17+
{
18+
auto p = GetParam();
19+
cv::Size size = std::get<0>(p);
20+
21+
RNG& rng = cv::theRNG();
22+
Mat src(size, CV_8UC1);
23+
cvtest::randUni(rng, src, Scalar::all(0), Scalar::all(255));
24+
Mat hist(1, 256, CV_32SC1);
25+
26+
while (next())
27+
{
28+
startTimer();
29+
cv::fastcv::calcHist(src, hist);
30+
stopTimer();
31+
}
32+
33+
SANITY_CHECK_NOTHING();
34+
}
35+
36+
} // namespace

modules/fastcv/src/bilateralFilter.cpp

+13-17
Original file line numberDiff line numberDiff line change
@@ -22,13 +22,13 @@ class FcvFilterLoop_Invoker : public cv::ParallelLoopBody
2222
{
2323
int height_ = range.end - range.start;
2424
int width_ = width;
25-
cv::Mat src_;
26-
int n = knl/2;
25+
cv::Mat src_;
26+
int n = knl/2;
2727

2828
src_ = cv::Mat(height_ + 2 * n, width_ + 2 * n, CV_8U);
2929
if (range.start == 0 && range.end == height)
3030
{
31-
cv::copyMakeBorder(src, src_, n, n, n, n, bdr);
31+
cv::copyMakeBorder(src(cv::Rect(0, 0, width, height)), src_, n, n, n, n, bdr);
3232
}
3333
else if (range.start == 0)
3434
{
@@ -43,7 +43,7 @@ class FcvFilterLoop_Invoker : public cv::ParallelLoopBody
4343
cv::copyMakeBorder(src(cv::Rect(0, range.start - n, width_, height_ + 2 * n)), src_, 0, 0, n, n, bdr);
4444
}
4545

46-
cv::Mat dst_padded = cv::Mat(height_ + 2*n, width_ + 2*n, CV_8U);
46+
cv::Mat dst_padded = cv::Mat(height_ + 2*n, width_ + 2*n, CV_8U);
4747

4848
auto func = (knl == 5) ? fcvBilateralFilter5x5u8_v3 :
4949
(knl == 7) ? fcvBilateralFilter7x7u8_v3 :
@@ -52,10 +52,10 @@ class FcvFilterLoop_Invoker : public cv::ParallelLoopBody
5252
func(src_.data, width_ + 2 * n, height_ + 2 * n, width_ + 2 * n,
5353
dst_padded.data, width_ + 2 * n, sigma_color, sigma_space, 0);
5454

55-
cv::Mat dst_temp1 = dst_padded(cv::Rect(n, n, width_, height_));
56-
cv::Mat dst_temp2 = dst(cv::Rect(0, range.start, width_, height_));
57-
dst_temp1.copyTo(dst_temp2);
58-
}
55+
cv::Mat dst_temp1 = dst_padded(cv::Rect(n, n, width_, height_));
56+
cv::Mat dst_temp2 = dst(cv::Rect(0, range.start, width_, height_));
57+
dst_temp1.copyTo(dst_temp2);
58+
}
5959

6060
private:
6161
const size_t src_step;
@@ -67,8 +67,8 @@ class FcvFilterLoop_Invoker : public cv::ParallelLoopBody
6767
float32_t sigma_color;
6868
float32_t sigma_space;
6969
int ret;
70-
cv::Mat src;
71-
cv::Mat dst;
70+
cv::Mat src;
71+
cv::Mat dst;
7272

7373
FcvFilterLoop_Invoker(const FcvFilterLoop_Invoker &); // = delete;
7474
const FcvFilterLoop_Invoker& operator= (const FcvFilterLoop_Invoker &); // = delete;
@@ -82,24 +82,20 @@ void bilateralFilter( InputArray _src, OutputArray _dst, int d,
8282

8383
CV_Assert(!_src.empty());
8484
int type = _src.type();
85-
CV_Assert(type == CV_8UC1);
86-
CV_Assert(d == 5 || d == 7 || d == 9);
85+
CV_Assert(type == CV_8UC1);
86+
CV_Assert(d == 5 || d == 7 || d == 9);
8787

8888
Size size = _src.size();
89-
_dst.create( size, type );
89+
_dst.create( size, type );
9090
Mat src = _src.getMat();
9191
Mat dst = _dst.getMat();
9292

9393
CV_Assert(src.data != dst.data);
9494

9595
if( sigmaColor <= 0 )
96-
{
9796
sigmaColor = 1;
98-
}
9997
if( sigmaSpace <= 0 )
100-
{
10198
sigmaSpace = 1;
102-
}
10399

104100
int nStripes = (src.rows / 20 == 0) ? 1 : (src.rows / 20);
105101
cv::parallel_for_(cv::Range(0, src.rows),

modules/fastcv/src/histogram.cpp

+74
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,74 @@
1+
/*
2+
* Copyright (c) 2024 Qualcomm Innovation Center, Inc. All rights reserved.
3+
* SPDX-License-Identifier: Apache-2.0
4+
*/
5+
6+
#include "precomp.hpp"
7+
8+
namespace cv {
9+
namespace fastcv {
10+
11+
class FcvHistogramLoop_Invoker : public cv::ParallelLoopBody
12+
{
13+
public:
14+
15+
FcvHistogramLoop_Invoker(const uchar * src_data_, size_t src_step_, int width_, int height_, int32_t* gl_hist_, int stripeHeight_, cv::Mutex* histogramLock, int nStripes_):
16+
cv::ParallelLoopBody(), src_data(src_data_), src_step(src_step_), width(width_), height(height_), gl_hist(gl_hist_), stripeHeight(stripeHeight_), histogramLock_(histogramLock), nStripes(nStripes_)
17+
{
18+
}
19+
20+
virtual void operator()(const cv::Range& range) const CV_OVERRIDE
21+
{
22+
int height_ = stripeHeight;
23+
if(range.end == nStripes)
24+
height_ += (height % nStripes);
25+
const uchar* yS = src_data;
26+
int32_t l_hist[256] = {0};
27+
fcvImageIntensityHistogram(yS, src_step, 0, range.start, width, height_, l_hist);
28+
cv::AutoLock lock(*histogramLock_);
29+
30+
for( int i = 0; i < 256; i++ )
31+
gl_hist[i] += l_hist[i];
32+
}
33+
34+
private:
35+
const uchar * src_data;
36+
const size_t src_step;
37+
const int width;
38+
const int height;
39+
int32_t *gl_hist;
40+
int ret;
41+
int stripeHeight;
42+
cv::Mutex* histogramLock_;
43+
int nStripes;
44+
45+
FcvHistogramLoop_Invoker(const FcvHistogramLoop_Invoker &); // = delete;
46+
const FcvHistogramLoop_Invoker& operator= (const FcvHistogramLoop_Invoker &); // = delete;
47+
};
48+
49+
void calcHist( InputArray _src, OutputArray _hist )
50+
{
51+
INITIALIZATION_CHECK;
52+
53+
CV_Assert(!_src.empty());
54+
int type = _src.type();
55+
CV_Assert(type == CV_8UC1);
56+
57+
_hist.create( cv::Size(256, 1), CV_32SC1 );
58+
Mat src = _src.getMat();
59+
Mat hist = _hist.getMat();
60+
61+
for( int i = 0; i < 256; i++ )
62+
hist.ptr<int>()[i] = 0;
63+
64+
cv::Mutex histogramLockInstance;
65+
66+
int nStripes = cv::getNumThreads();
67+
int stripeHeight = src.rows / nStripes;
68+
69+
cv::parallel_for_(cv::Range(0, nStripes),
70+
FcvHistogramLoop_Invoker(src.data, src.step[0], src.cols, src.rows, hist.ptr<int>(), stripeHeight, &histogramLockInstance, nStripes), nStripes);
71+
}
72+
73+
} // fastcv::
74+
} // cv::

modules/fastcv/src/moments.cpp

+36-44
Original file line numberDiff line numberDiff line change
@@ -10,71 +10,63 @@ namespace fastcv {
1010

1111
cv::Moments moments(InputArray _src, bool binary)
1212
{
13-
INITIALIZATION_CHECK;
13+
INITIALIZATION_CHECK;
1414

1515
CV_Assert(!_src.empty());
1616
int type = _src.type();
17-
CV_Assert(type == CV_8UC1 || type == CV_32SC1 || type == CV_32FC1);
17+
CV_Assert(type == CV_8UC1 || type == CV_32SC1 || type == CV_32FC1);
1818

1919
Size size = _src.size();
2020
Mat src = _src.getMat();
2121

2222
cv::Moments m;
23-
if( size.width == 0 || size.height == 0 )
24-
return m;
25-
26-
fcvMoments* mFCV = new fcvMoments();
23+
fcvMoments mFCV;
2724
fcvStatus status = FASTCV_SUCCESS;
28-
if(binary)
25+
if(binary)
26+
{
27+
cv::Mat src_binary(size, CV_8UC1);
28+
cv::compare( src, 0, src_binary, cv::CMP_NE );
29+
fcvImageMomentsu8(src_binary.data, src_binary.cols,
30+
src_binary.rows, src_binary.step[0], &mFCV, binary);
31+
}
32+
else
2933
{
30-
cv::Mat src_binary(size, CV_8UC1);
31-
cv::compare( src, 0, src_binary, cv::CMP_NE );
32-
fcvImageMomentsu8(src_binary.data, src_binary.cols,
33-
src_binary.rows, src_binary.step, mFCV, binary);
34+
switch(type)
35+
{
36+
case CV_8UC1:
37+
fcvImageMomentsu8(src.data, src.cols, src.rows, src.step[0], &mFCV, binary);
38+
break;
39+
case CV_32SC1:
40+
fcvImageMomentss32(src.ptr<int>(), src.cols, src.rows, src.step[0], &mFCV, binary);
41+
break;
42+
case CV_32FC1:
43+
fcvImageMomentsf32(src.ptr<float>(), src.cols, src.rows, src.step[0], &mFCV, binary);
44+
break;
45+
}
3446
}
35-
else
36-
{
37-
switch(type)
38-
{
39-
case CV_8UC1:
40-
fcvImageMomentsu8(src.data, src.cols, src.rows,
41-
src.step, mFCV, binary);
42-
break;
43-
case CV_32SC1:
44-
fcvImageMomentss32((const int*)src.data, src.cols, src.rows,
45-
src.step, mFCV, binary);
46-
break;
47-
case CV_32FC1:
48-
fcvImageMomentsf32((const float*)src.data, src.cols, src.rows,
49-
src.step, mFCV, binary);
50-
break;
51-
}
52-
}
5347

54-
if (status != FASTCV_SUCCESS)
48+
if (status != FASTCV_SUCCESS)
5549
{
5650
CV_Error( cv::Error::StsError, cv::format("Error occurred!") );
57-
delete mFCV;
5851
return m;
5952
}
6053

61-
m.m00 = mFCV->m00; m.m10 = mFCV->m10; m.m01 = mFCV->m01;
62-
m.m20 = mFCV->m20; m.m11 = mFCV->m11; m.m02 = mFCV->m02;
63-
m.m30 = mFCV->m30; m.m21 = mFCV->m21; m.m12 = mFCV->m12;
64-
m.m03 = mFCV->m03; m.mu02 = mFCV->mu02; m.m03 = mFCV->mu03;
65-
m.mu11 = mFCV->mu11; m.mu12 = mFCV->mu12; m.mu20 = mFCV->mu20;
66-
m.mu21 = mFCV->mu21; m.mu30 = mFCV->mu30;
54+
m.m00 = mFCV.m00; m.m10 = mFCV.m10; m.m01 = mFCV.m01;
55+
m.m20 = mFCV.m20; m.m11 = mFCV.m11; m.m02 = mFCV.m02;
56+
m.m30 = mFCV.m30; m.m21 = mFCV.m21; m.m12 = mFCV.m12;
57+
m.m03 = mFCV.m03; m.mu02 = mFCV.mu02; m.m03 = mFCV.mu03;
58+
m.mu11 = mFCV.mu11; m.mu12 = mFCV.mu12; m.mu20 = mFCV.mu20;
59+
m.mu21 = mFCV.mu21; m.mu30 = mFCV.mu30;
6760

68-
float32_t inv_m00 = 1.0/mFCV->m00;
69-
float32_t inv_sqrt_m00 = mFCV->inv_sqrt_m00;
61+
float32_t inv_m00 = 1.0/mFCV.m00;
62+
float32_t inv_sqrt_m00 = mFCV.inv_sqrt_m00;
7063
float32_t s2 = inv_m00 * inv_m00, s3 = s2 * inv_sqrt_m00;
7164

72-
m.nu20 = mFCV->mu20 * s2; m.nu11 = mFCV->mu11 * s2;
73-
m.nu02 = mFCV->mu02 * s2; m.nu30 = mFCV->mu30 * s3;
74-
m.nu21 = mFCV->mu21 * s3; m.nu12 = mFCV->mu12 * s3;
75-
m.nu03 = mFCV->mu03 * s3;
65+
m.nu20 = mFCV.mu20 * s2; m.nu11 = mFCV.mu11 * s2;
66+
m.nu02 = mFCV.mu02 * s2; m.nu30 = mFCV.mu30 * s3;
67+
m.nu21 = mFCV.mu21 * s3; m.nu12 = mFCV.mu12 * s3;
68+
m.nu03 = mFCV.mu03 * s3;
7669

77-
delete mFCV;
7870
return m;
7971
}
8072

modules/fastcv/test/test_bilateral.cpp

+2-2
Original file line numberDiff line numberDiff line change
@@ -12,13 +12,13 @@ typedef testing::TestWithParam<tuple<cv::Size,int,int>> fcv_bilateralFilterTest;
1212
TEST_P(fcv_bilateralFilterTest, accuracy)
1313
{
1414
cv::Size size = get<0>(GetParam());
15-
int d = get<1>(GetParam());
15+
int d = get<1>(GetParam());
1616
double sigmaColor = get<2>(GetParam());
1717
double sigmaSpace = sigmaColor;
1818

1919
RNG& rng = cv::theRNG();
2020
Mat src(size, CV_8UC1);
21-
cvtest::randUni(rng, src, Scalar::all(0), Scalar::all(256));
21+
cvtest::randUni(rng, src, Scalar::all(0), Scalar::all(255));
2222

2323
cv::Mat dst;
2424

0 commit comments

Comments
 (0)