Skip to content

Commit 0e3c39f

Browse files
MengqingCaorurban
authored andcommitted
Merge pull request opencv#3608 from MengqingCao:dvpp_support
Add additional image processing operators for Ascend NPU by utilizing DVPP opencv#3608 The user base for [Ascend NPU](https://www.hiascend.com/en/) and programming with CANN is increasing rapidly, with a growing number of users joining each day. To facilitate the use of these users, this PR provides more support for Ascend backend operators. All operators this PR offers are using use DVPP as the computational unit. Digital Vision Pre-Processing (DVPP) is an image processing unit built into the Ascend AI processor. Its main functions include image and video encoding/decoding, as well as image cropping and scaling. The high-frequency operators with NPU as the backend and basic data structure AscendMat has been provided in opencv#3552, while it still lacks many image processing operators. Moreover, only two interpolation algorithms for the resize operator are supported in opencv#3552. In this PR, the bilinear interpolation algorithm and nearest neighbour interpolation algorithm are implemented for the resize operator, as well as the Ascend implementation of the copyMakeBorder operator. In addition, the serialization of image processing operations is widely used in the preprocessing and post-processing stages of computer vision deep learning methods. Therefore, providing integrated operators is very meaningful for improving the convenience of use for OpenCV and deep learning crossover users. For example, torchvision also provides similar operators: [RESIZED_CROP](https://pytorch.org/vision/stable/generated/torchvision.transforms.functional.resized_crop.html?highlight=resizedcrop). Thus, this PR also provides two serialization processing operators: cropResize and cropResizeMakeBorder. ### 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 - [x] The PR is proposed to the proper branch - [N/A] There is a reference to the original bug report and related work - [x] There is accuracy test, performance test and test data in opencv_extra repository, if applicable Patch to opencv_extra has the same branch name. - [x] The feature is well documented and sample code can be built with the project CMake
1 parent 261cfc7 commit 0e3c39f

12 files changed

+1126
-69
lines changed

modules/cannops/include/opencv2/cann.hpp

+15
Original file line numberDiff line numberDiff line change
@@ -318,6 +318,21 @@ CV_EXPORTS_W void initAcl();
318318
*/
319319
CV_EXPORTS_W void finalizeAcl();
320320

321+
/**
322+
* @brief init DVPP system.
323+
* @note The DVPP interfaces used are all version V2.
324+
* Supported devices: Atlas Inference Series products, Atlas 200/500 A2 Inference products and
325+
* Atlas A2 Training Series products/Atlas 300I A2 Inference products
326+
*/
327+
CV_EXPORTS_W void initDvpp();
328+
329+
/**
330+
* @brief finalize DVPP system.
331+
* @note Supported devices: Atlas Inference Series products, Atlas 200/500 A2 Inference products and
332+
* Atlas A2 Training Series products/Atlas 300I A2 Inference products
333+
*/
334+
CV_EXPORTS_W void finalizeDvpp();
335+
321336
//! @} cann_init
322337

323338
} // namespace cann

modules/cannops/include/opencv2/cann_interface.hpp

+141-24
Large diffs are not rendered by default.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,107 @@
1+
// This file is part of OpenCV project.
2+
// It is subject to the license terms in the LICENSE file found in the top-level directory
3+
// of this distribution and at http://opencv.org/license.html.
4+
5+
#ifndef ENABLE_DVPP_INTERFACE
6+
#define ENABLE_DVPP_INTERFACE
7+
#endif // ENABLE_DVPP_INTERFACE
8+
9+
#include <vector>
10+
#include <string>
11+
#include <acl/acl.h>
12+
#include <acl/acl_op_compiler.h>
13+
#include <acl/dvpp/hi_dvpp.h>
14+
#include "acl/acl_op.h"
15+
#include "cann_call.hpp"
16+
17+
namespace cv
18+
{
19+
namespace cann
20+
{
21+
struct AscendPicDesc
22+
{
23+
const char* name;
24+
std::shared_ptr<hi_void> data;
25+
std::vector<int64_t> batchNum;
26+
27+
size_t widthAlignment = 16;
28+
size_t heightAlignment = 1;
29+
size_t sizeAlignment = 3;
30+
size_t sizeNum = 3;
31+
32+
hi_vpc_pic_info Pic;
33+
AscendPicDesc& setMemAlign();
34+
AscendPicDesc& setPic(hi_pixel_format _picture_format);
35+
std::shared_ptr<hi_void> allocate();
36+
AscendPicDesc(){};
37+
AscendPicDesc(const AscendMat& ascendMat, hi_pixel_format _picture_format);
38+
AscendPicDesc(const Mat& mat, hi_pixel_format _picture_format);
39+
};
40+
41+
/*
42+
***************************** hi_mpi_vpc warppers ***************************
43+
The DVPP VPC interfaces here are all version v2. Only the following devices are supported: Atlas
44+
Inference Series products, Atlas 200/500 A2 Inference products and Atlas A2 Training Series
45+
products/Atlas 300I A2 Inference products.
46+
*/
47+
inline void vpcResizeWarpper(hi_vpc_chn chnId, hi_vpc_pic_info& inPic, hi_vpc_pic_info& outPic,
48+
int interpolation, uint32_t* taskID)
49+
{
50+
uint32_t ret = hi_mpi_vpc_resize(chnId, &inPic, &outPic, 0, 0, interpolation, taskID, -1);
51+
if (ret != HI_SUCCESS)
52+
CV_Error(Error::StsBadFlag, "failed to resize image");
53+
}
54+
void vpcCropResizeWarpper(hi_vpc_chn chnId, hi_vpc_pic_info& inPic, hi_vpc_pic_info& outPic,
55+
int cnt, uint32_t* taskID, const Rect& rect, Size dsize,
56+
int interpolation);
57+
58+
void vpcCropResizeMakeBorderWarpper(hi_vpc_chn chnId, std::vector<AscendPicDesc>& inPicDesc,
59+
std::vector<AscendPicDesc>& outPicDesc, int cnt,
60+
uint32_t* taskID, const Rect& rect, Size dsize,
61+
int interpolation, const int borderType, Scalar scalarV,
62+
int top, int left);
63+
void vpcCopyMakeBorderWarpper(hi_vpc_chn chnId, hi_vpc_pic_info& inPic, hi_vpc_pic_info& outPic,
64+
uint32_t* taskID, int* offsets, int bordertype, Scalar value);
65+
/*****************************************************************************/
66+
67+
/**
68+
* @brief Interface for calling DVPP operator descriptors.
69+
* The DVPP VPC interfaces here are all version v2. Supported devices: Atlas Inference Series
70+
* products, Atlas 200/500 A2 Inference products and Atlas A2 Training Series products/Atlas 300I A2
71+
* Inference products.
72+
*/
73+
class DvppOperatorDesc
74+
{
75+
private:
76+
DvppOperatorDesc& addInput(AscendPicDesc& picDesc);
77+
DvppOperatorDesc& addOutput(AscendPicDesc& picDesc);
78+
std::set<std::shared_ptr<hi_void>> holder;
79+
80+
public:
81+
DvppOperatorDesc()
82+
{
83+
chnId = 0;
84+
stChnAttr = {};
85+
createChannel();
86+
}
87+
virtual ~DvppOperatorDesc() { reset(); }
88+
DvppOperatorDesc& addInput(const AscendMat& mat);
89+
DvppOperatorDesc& addOutput(AscendMat& mat);
90+
DvppOperatorDesc& addInput(const Mat& mat);
91+
DvppOperatorDesc& addOutput(Mat& mat);
92+
93+
DvppOperatorDesc& getResult(Mat& dst, uint32_t& taskIDResult);
94+
DvppOperatorDesc& getResult(AscendMat& dst, uint32_t& taskIDResult);
95+
96+
DvppOperatorDesc& reset();
97+
DvppOperatorDesc& createChannel();
98+
99+
std::vector<AscendPicDesc> inputDesc_;
100+
std::vector<AscendPicDesc> outputDesc_;
101+
102+
hi_vpc_chn chnId;
103+
hi_vpc_chn_attr stChnAttr;
104+
};
105+
106+
} // namespace cann
107+
} // namespace cv

modules/cannops/misc/python/test/test_cannops.py

+45
Original file line numberDiff line numberDiff line change
@@ -24,6 +24,7 @@ def genMask(mask, listx, listy):
2424
class cannop_test(NewOpenCVTests):
2525
def test_ascend(self):
2626
cv.cann.initAcl()
27+
cv.cann.initDvpp()
2728
cv.cann.getDevice()
2829
cv.cann.setDevice(0)
2930
stream = cv.cann.AscendStream_Null()
@@ -275,6 +276,50 @@ def test_imgproc(self):
275276
aclMat, 127, 255, tType)
276277
self.assertTrue(np.allclose(cvThresh, cannThresh.download()))
277278
self.assertTrue(np.allclose(cvRet, cannRet))
279+
280+
npMat = (np.random.random((1280, 1024, 3)) * 255).astype(np.uint8)
281+
w_off, h_off, crop_w, crop_h = 0, 0, 512, 384
282+
roi = [w_off, h_off, crop_w, crop_h]
283+
aclMat = cv.cann.AscendMat()
284+
aclMat.upload(npMat)
285+
286+
# resize
287+
dstSize = np.array([crop_w, crop_h])
288+
self.assertTrue(np.allclose(cv.cann.resize(npMat, dstSize, 0, 0, 1),
289+
cv.resize(npMat, dstSize, 0, 0, 1)))
290+
self.assertTrue(np.allclose(cv.cann.resize(aclMat, dstSize, 0, 0, 1).download(),
291+
cv.resize(npMat, dstSize, 0, 0, 1)))
292+
# cropResize
293+
self.assertTrue(np.allclose(cv.cann.cropResize(npMat, roi, dstSize, 0, 0, 1),
294+
cv.resize(npMat[h_off:crop_h, w_off:crop_w], dstSize, 0, 0, 1)), 0)
295+
self.assertTrue(np.allclose(cv.cann.cropResize(aclMat, roi, dstSize, 0, 0, 1).download(),
296+
cv.resize(npMat[h_off:crop_h, w_off:crop_w], dstSize, 0, 0, 1)), 0)
297+
298+
# cropResizeMakeBorder
299+
# TODO cv.copyMakeBorder ignores borderColorValue param; find the reason and fix it
300+
borderColorValue = (100, 0, 255)
301+
top, bottom, left, right = 32, 0, 10, 0
302+
borderTypes = [0, 1]
303+
304+
for borderType in borderTypes:
305+
self.assertTrue(np.allclose(cv.cann.cropResizeMakeBorder(npMat, roi, dstSize,
306+
0, 0, 1, top, left, borderType),
307+
cv.copyMakeBorder(cv.resize(npMat[h_off:crop_h, w_off:crop_w],
308+
dstSize, 0, 0, 1), top, bottom, left, right, borderType), 1))
309+
self.assertTrue(np.allclose(cv.cann.cropResizeMakeBorder(aclMat, roi, dstSize,
310+
0, 0, 1, top, left, borderType).download(),
311+
cv.copyMakeBorder(cv.resize(npMat[h_off:crop_h, w_off:crop_w],
312+
dstSize, 0, 0, 1), top, bottom, left, right, borderType), 1))
313+
314+
# copyMakeBorder
315+
for borderType in borderTypes:
316+
self.assertTrue(np.allclose(cv.cann.copyMakeBorder(npMat, top, bottom, left, right,
317+
borderType),
318+
cv.copyMakeBorder(npMat, top, bottom, left, right, borderType)))
319+
self.assertTrue(np.allclose(cv.cann.copyMakeBorder(aclMat, top, bottom, left, right,
320+
borderType).download(),
321+
cv.copyMakeBorder(npMat, top, bottom, left, right, borderType)))
322+
278323
cv.cann.resetDevice()
279324

280325
if __name__ == '__main__':

modules/cannops/perf/perf_core.cpp

+172
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,7 @@ namespace
1111
{
1212
#define TYPICAL_ASCEND_MAT_SIZES \
1313
Values(::perf::sz1080p, ::perf::sz2K, ::perf::sz2160p, ::perf::sz4320p)
14+
#define DVPP_ASCEND_MAT_SIZES Values(::perf::sz1080p, ::perf::sz2K, ::perf::sz2160p, ::perf::sz5MP)
1415
#define DEF_PARAM_TEST(name, ...) \
1516
typedef ::perf::TestBaseWithParam<testing::tuple<__VA_ARGS__>> name
1617

@@ -157,5 +158,176 @@ PERF_TEST_P(NPU, CROP_OVERLOAD, TYPICAL_ASCEND_MAT_SIZES)
157158
cv::cann::resetDevice();
158159
SANITY_CHECK_NOTHING();
159160
}
161+
162+
PERF_TEST_P(CPU, RESIZE, DVPP_ASCEND_MAT_SIZES)
163+
{
164+
Mat mat(GET_PARAM(0), CV_8UC3);
165+
Mat dst;
166+
declare.in(mat, WARMUP_RNG);
167+
Size dsize = Size(256, 256);
168+
TEST_CYCLE_N(10) { cv::resize(mat, dst, dsize, 0, 0, 1); }
169+
SANITY_CHECK_NOTHING();
170+
}
171+
172+
PERF_TEST_P(NPU, RESIZE, DVPP_ASCEND_MAT_SIZES)
173+
{
174+
Mat mat(GET_PARAM(0), CV_32FC3);
175+
AscendMat dst;
176+
AscendMat src;
177+
src.upload(mat);
178+
declare.in(mat, WARMUP_RNG);
179+
Size dsize = Size(256, 256);
180+
TEST_CYCLE_N(10) { cv::cann::resize(src, dst, dsize, 0, 0, 3); }
181+
SANITY_CHECK_NOTHING();
182+
}
183+
184+
PERF_TEST_P(NPU, THRESHOLD, TYPICAL_ASCEND_MAT_SIZES)
185+
{
186+
Mat mat(GET_PARAM(0), CV_32FC3);
187+
AscendMat dst;
188+
AscendMat src;
189+
src.upload(mat);
190+
declare.in(mat, WARMUP_RNG);
191+
TEST_CYCLE_N(10) { cv::cann::threshold(src, dst, 100.0, 255.0, cv::THRESH_BINARY); }
192+
SANITY_CHECK_NOTHING();
193+
}
194+
PERF_TEST_P(CPU, THRESHOLD, TYPICAL_ASCEND_MAT_SIZES)
195+
{
196+
Mat mat(GET_PARAM(0), CV_32FC3);
197+
Mat dst;
198+
declare.in(mat, WARMUP_RNG);
199+
TEST_CYCLE_N(10) { cv::threshold(mat, dst, 100.0, 255.0, cv::THRESH_BINARY); }
200+
SANITY_CHECK_NOTHING();
201+
}
202+
203+
PERF_TEST_P(NPU, RESIZE_INTER_NEAREST, DVPP_ASCEND_MAT_SIZES)
204+
{
205+
Mat mat(GET_PARAM(0), CV_8UC3);
206+
Mat dst;
207+
declare.in(mat, WARMUP_RNG);
208+
Size dsize = Size(256, 256);
209+
TEST_CYCLE_N(10) { cv::cann::resize(mat, dst, dsize, 0, 0, 0); }
210+
SANITY_CHECK_NOTHING();
211+
}
212+
213+
PERF_TEST_P(NPU, COPY_MAKE_BORDER, DVPP_ASCEND_MAT_SIZES)
214+
{
215+
Mat resized_cv, checker, cpuOpRet, cpuMat(GET_PARAM(0), CV_8UC3);
216+
declare.in(cpuMat, WARMUP_RNG);
217+
int top, bottom, left, right;
218+
top = (int)(20);
219+
bottom = top;
220+
left = (int)(20);
221+
right = left;
222+
int borderType = 1;
223+
float scalarV[3] = {0, 0, 255};
224+
Scalar value = {scalarV[0], scalarV[1], scalarV[2]};
225+
226+
TEST_CYCLE_N(10)
227+
{
228+
cv::cann::copyMakeBorder(cpuMat, checker, top, bottom, left, right, borderType, value);
229+
}
230+
231+
SANITY_CHECK_NOTHING();
232+
}
233+
PERF_TEST_P(CPU, COPY_MAKE_BORDER, DVPP_ASCEND_MAT_SIZES)
234+
{
235+
Mat resized_cv, checker, cpuOpRet, cpuMat(GET_PARAM(0), CV_8UC3);
236+
declare.in(cpuMat, WARMUP_RNG);
237+
int top, bottom, left, right;
238+
top = (int)(20);
239+
bottom = top;
240+
left = (int)(20);
241+
right = left;
242+
int borderType = 1;
243+
float scalarV[3] = {0, 0, 255};
244+
Scalar value = {scalarV[0], scalarV[1], scalarV[2]};
245+
246+
TEST_CYCLE_N(10)
247+
{
248+
cv::copyMakeBorder(cpuMat, checker, top, bottom, left, right, borderType, value);
249+
}
250+
251+
SANITY_CHECK_NOTHING();
252+
}
253+
254+
PERF_TEST_P(NPU, CROP_RESIZE_MAKE_BORDER, DVPP_ASCEND_MAT_SIZES)
255+
{
256+
Size size = GET_PARAM(0);
257+
Mat resized_cv, checker, cpuOpRet, cpuMat(size, CV_8UC3);
258+
declare.in(cpuMat, WARMUP_RNG);
259+
260+
const Rect b(1, 0, size.width / 2, size.height);
261+
Size dsize = Size(size.width / 4, size.height / 2);
262+
int top, left;
263+
top = (int)(20);
264+
left = (int)(20);
265+
int borderType = 0;
266+
float scalarV[3] = {1, 1, 1};
267+
Scalar value = {scalarV[0], scalarV[1], scalarV[2]};
268+
269+
TEST_CYCLE_N(10)
270+
{
271+
cv::cann::cropResizeMakeBorder(cpuMat, checker, b, dsize, 0, 0, 1, top, left, borderType,
272+
value);
273+
}
274+
275+
SANITY_CHECK_NOTHING();
276+
}
277+
278+
PERF_TEST_P(CPU, CROP_RESIZE_MAKE_BORDER, DVPP_ASCEND_MAT_SIZES)
279+
{
280+
Size size = GET_PARAM(0);
281+
Mat resized_cv, checker, cpuOpRet, cpuMat(size, CV_8UC3);
282+
declare.in(cpuMat, WARMUP_RNG);
283+
const Rect b(1, 0, size.width / 2, size.height);
284+
Size dsize = Size(size.width / 4, size.height / 2);
285+
int top, bottom, left, right;
286+
top = (int)(20);
287+
bottom = 0;
288+
left = (int)(20);
289+
right = 0;
290+
int borderType = 0;
291+
float scalarV[3] = {1, 1, 1};
292+
Scalar value = {scalarV[0], scalarV[1], scalarV[2]};
293+
294+
TEST_CYCLE_N(10)
295+
{
296+
Mat cropped_cv(cpuMat, b);
297+
cv::resize(cropped_cv, resized_cv, dsize, 0, 0, 1);
298+
cv::copyMakeBorder(resized_cv, cpuOpRet, top, bottom, left, right, borderType, value);
299+
}
300+
SANITY_CHECK_NOTHING();
301+
}
302+
303+
PERF_TEST_P(NPU, CROP_RESIZE, DVPP_ASCEND_MAT_SIZES)
304+
{
305+
Size size = GET_PARAM(0);
306+
Mat resized_cv, checker, cpuOpRet, cpuMat(size, CV_8UC3);
307+
declare.in(cpuMat, WARMUP_RNG);
308+
const Rect b(1, 0, size.width / 2, size.height);
309+
Size dsize = Size(size.width / 4, size.height / 2);
310+
311+
TEST_CYCLE_N(10) { cv::cann::cropResize(cpuMat, checker, b, dsize, 0, 0, 1); }
312+
313+
SANITY_CHECK_NOTHING();
314+
}
315+
316+
PERF_TEST_P(CPU, CROP_RESIZE, DVPP_ASCEND_MAT_SIZES)
317+
{
318+
Size size = GET_PARAM(0);
319+
Mat resized_cv, checker, cpuOpRet, cpuMat(size, CV_8UC3);
320+
declare.in(cpuMat, WARMUP_RNG);
321+
const Rect b(1, 0, size.width / 2, size.height);
322+
Size dsize = Size(size.width / 4, size.height / 2);
323+
324+
TEST_CYCLE_N(10)
325+
{
326+
Mat cropped_cv(cpuMat, b);
327+
cv::resize(cropped_cv, resized_cv, dsize, 0, 0, 1);
328+
}
329+
SANITY_CHECK_NOTHING();
330+
}
331+
160332
} // namespace
161333
} // namespace opencv_test

modules/cannops/perf/perf_main.cpp

+12-2
Original file line numberDiff line numberDiff line change
@@ -10,8 +10,18 @@ class CannEnvironment : public ::testing::Environment
1010
{
1111
public:
1212
virtual ~CannEnvironment() = default;
13-
virtual void SetUp() CV_OVERRIDE { cv::cann::initAcl(); }
14-
virtual void TearDown() CV_OVERRIDE { cv::cann::finalizeAcl(); }
13+
virtual void SetUp() CV_OVERRIDE
14+
{
15+
initAcl();
16+
cv::cann::setDevice(DEVICE_ID);
17+
initDvpp();
18+
}
19+
virtual void TearDown() CV_OVERRIDE
20+
{
21+
finalizeAcl();
22+
cv::cann::resetDevice();
23+
finalizeDvpp();
24+
}
1525
};
1626

1727
static void initTests()

0 commit comments

Comments
 (0)