Skip to content

Commit 2820b30

Browse files
Add samples and tutorials
Co-authored-by: CaoMengqing <[email protected]>
1 parent 68a050f commit 2820b30

13 files changed

+232
-56
lines changed

modules/cannops/CMakeLists.txt

+1
Original file line numberDiff line numberDiff line change
@@ -14,3 +14,4 @@ ocv_include_directories(${CMAKE_SOURCE_DIR}/modules/ts/include)
1414

1515
ocv_add_accuracy_tests(DEPENDS_ON opencv_cannops)
1616
ocv_add_perf_tests(DEPENDS_ON opencv_cannops)
17+
ocv_add_samples(opencv_cannops)

modules/cannops/include/opencv2/cann_private.hpp

+1-1
Original file line numberDiff line numberDiff line change
@@ -31,4 +31,4 @@ double threshold(AscendMat& src, AscendMat& dst, double thresh, double maxval, i
3131
} // namespace cann
3232
} // namespace cv
3333

34-
#endif // OPENCV_CANNOPS_CANN_PRIVATE_HPP
34+
#endif // OPENCV_CANNOPS_CANN_PRIVATE_HPP
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,59 @@
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+
#include <iostream>
6+
#include <opencv2/imgcodecs.hpp>
7+
#include <opencv2/cann.hpp>
8+
#include <opencv2/cann_interface.hpp>
9+
10+
int main(int argc, char *argv[])
11+
{
12+
cv::CommandLineParser parser(argc, argv,
13+
"{@input|puppy.png|path to input image}"
14+
"{@output|output.png|path to output image}"
15+
"{help||show help}"
16+
);
17+
parser.about("This is a sample for image processing with Ascend NPU. \n");
18+
if (argc != 3 || parser.has("help"))
19+
{
20+
parser.printMessage();
21+
return 0;
22+
}
23+
24+
std::string imagePath = parser.get<std::string>(0);
25+
std::string outputPath = parser.get<std::string>(1);
26+
27+
// read input image and generate guass noise
28+
//! [input_noise]
29+
cv::Mat img = cv::imread(imagePath);
30+
// Generate gauss noise that will be added into the input image
31+
cv::Mat gaussNoise(img.rows, img.cols,img.type());
32+
cv::RNG rng;
33+
rng.fill(gaussNoise, cv::RNG::NORMAL, 0, 25);
34+
//! [input_noise]
35+
36+
//setup cann
37+
//! [setup]
38+
cv::cann::initAcl();
39+
cv::cann::setDevice(0);
40+
//! [setup]
41+
42+
//! [image-process]
43+
cv::Mat output;
44+
// add gauss noise to the image
45+
cv::cann::add(img, gaussNoise, output);
46+
// rotate the image with a certain mode (0, 1 and 2, correspond to rotation of 90, 180 and 270 degrees clockwise respectively)
47+
cv::cann::rotate(output, output, 0);
48+
// flip the image with a certain mode (0, positive and negative number, correspond to flipping around the x-axis, y-axis and both axes respectively)
49+
cv::cann::flip(output, output, 0);
50+
//! [image-process]
51+
52+
cv::imwrite(outputPath, output);
53+
54+
//! [tear-down-cann]
55+
cv::cann::resetDevice();
56+
cv::cann::finalizeAcl();
57+
//! [tear-down-cann]
58+
return 0;
59+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,40 @@
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+
import numpy as np
6+
import cv2
7+
import argparse
8+
9+
parser = argparse.ArgumentParser(description='This is a sample for image processing with Ascend NPU.')
10+
parser.add_argument('image', help='path to input image')
11+
parser.add_argument('output', help='path to output image')
12+
args = parser.parse_args()
13+
14+
# read input image and generate guass noise
15+
#! [input_noise]
16+
img = cv2.imread(args.image)
17+
# Generate gauss noise that will be added into the input image
18+
gaussNoise = np.random.normal(0, 25,(img.shape[0], img.shape[1], img.shape[2])).astype(img.dtype)
19+
#! [input_noise]
20+
21+
# setup cann
22+
#! [setup]
23+
cv2.cann.initAcl()
24+
cv2.cann.setDevice(0)
25+
#! [setup]
26+
27+
#! [image-process]
28+
# add gauss noise to the image
29+
output = cv2.cann.add(img, gaussNoise)
30+
# rotate the image with a certain mode (0, 1 and 2, correspond to rotation of 90, 180 and 270 degrees clockwise respectively)
31+
output = cv2.cann.rotate(output, 0)
32+
# flip the image with a certain mode (0, positive and negative number, correspond to flipping around the x-axis, y-axis and both axes respectively)
33+
output = cv2.cann.flip(output, 0)
34+
#! [image-process]
35+
36+
cv2.imwrite(args.output, output)
37+
38+
#! [tear-down-cann]
39+
cv2.cann.finalizeAcl()
40+
#! [tear-down-cann]

modules/cannops/samples/sample.cpp

-33
This file was deleted.

modules/cannops/samples/sample.py

-20
This file was deleted.

modules/cannops/src/core.cpp

-1
Original file line numberDiff line numberDiff line change
@@ -156,7 +156,6 @@ void flip(InputArray _src, OutputArray _dst, int flipCode, AscendStream& stream)
156156

157157
void rotate(InputArray _src, OutputArray _dst, int rotateMode, AscendStream& stream)
158158
{
159-
CV_Assert(_src.dims() <= 2);
160159
AscendMat src = getInputMat(_src, stream), dst, tempMat;
161160
switch (rotateMode)
162161
{

modules/cannops/test/test_cvtcolor.cpp

+1-1
Original file line numberDiff line numberDiff line change
@@ -69,4 +69,4 @@ TEST(CVT_COLOR, YUV2BGR_DC4) { cvtColorTest(COLOR_YUV2BGR, 3, 4, 10.0f); }
6969
TEST(CVT_COLOR, YUV2RGB_DC4) { cvtColorTest(COLOR_YUV2RGB, 3, 4, 10.0f); }
7070

7171
} // namespace
72-
} // namespace opencv_test
72+
} // namespace opencv_test
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,130 @@
1+
Ascend NPU Image Processing {#tutorial_ascend_npu_image_processing}
2+
==========================================================
3+
4+
## Goal
5+
6+
In this guide, you will gain insights into the thread safety of Ascend operators already in use, as well as discover how to effectively employ Ascend operators for image preprocessing and understand their usage limitations.
7+
8+
## Preface
9+
10+
We provide a suite of common matrix operation operators that support the [Ascend NPU](https://www.hiascend.com/en/) within OpenCV. For user convenience, the new 'AscendMat' structure and its associated operators maintain compatibility with the 'Mat' interface in OpenCV. These operators encompass a wide range of frequently used functions, including arithmetic operations, image processing operations, and image color space conversion. All of these operators are implemented utilizing [CANN](https://www.hiascend.com/en/software/cann)(Compute Architecture of Neural Networks). The Ascend operator facilitates accelerated operations on the NPU by making use of CANN. This acceleration effect is particularly noticeable when working with larger images, such as those with dimensions like 2048x2048, 3840x2160, 7680x4320, etc.
11+
12+
13+
## Instructions on Thread Safety
14+
15+
Our stream function is implemented by invoking the CANN operators. In the same stream, tasks are executed sequentially, while across different streams, tasks are executed in parallel. The use of event mechanisms ensures synchronization of tasks between streams, please refer to the [**Stream Management**](https://www.hiascend.com/document/detail/en/CANNCommunityEdition/600alphaX/infacldevg/aclcppdevg/aclcppdevg_000147.html) documentation for details.
16+
17+
18+
## Example for Image Preprocessing
19+
20+
In this section, you will discover how to use Ascend operators for image preprocessing, including functions below:
21+
22+
- Add
23+
- Rotate
24+
- Flip
25+
26+
27+
### code
28+
29+
@add_toggle_cpp
30+
@include opencv_contrib/modules/cannops/samples/image_processing.cpp
31+
@end_toggle
32+
33+
@add_toggle_python
34+
@include opencv_contrib/modules/cannops/samples/image_processing.py
35+
@end_toggle
36+
37+
### Explanation
38+
39+
**Input Image**
40+
41+
@add_toggle_cpp
42+
@snippet opencv_contrib/modules/cannops/samples/image_processing.cpp input_noise
43+
@end_toggle
44+
45+
@add_toggle_python
46+
47+
```python
48+
# Read the input image
49+
img = cv2.imread("/path/to/img")
50+
# Generate gauss noise that will be added into the input image
51+
gaussNoise = np.random.normal(mean=0,sigma=25,(img.shape[0],img.shape[1],img.shape[2])).astype(img.dtype)
52+
```
53+
54+
@end_toggle
55+
56+
**Setup CANN**
57+
58+
@add_toggle_cpp
59+
60+
@snippet opencv_contrib/modules/cannops/samples/image_processing.cpp setup
61+
62+
@end_toggle
63+
64+
@add_toggle_python
65+
66+
@snippet opencv_contrib/modules/cannops/samples/image_processing.py setup
67+
68+
@end_toggle
69+
**Image Preprocessing Example**
70+
71+
@add_toggle_cpp
72+
73+
@snippet opencv_contrib/modules/cannops/samples/image_processing.cpp image-process
74+
75+
@end_toggle
76+
77+
@add_toggle_python
78+
79+
@snippet opencv_contrib/modules/cannops/samples/image_processing.py image-process
80+
81+
@end_toggle
82+
83+
**Tear down CANN**
84+
85+
@add_toggle_cpp
86+
@snippet opencv_contrib/modules/cannops/samples/image_processing.cpp tear-down-cann
87+
88+
@end_toggle
89+
90+
@add_toggle_python
91+
92+
@snippet opencv_contrib/modules/cannops/samples/image_processing.py tear-down-cann
93+
94+
@end_toggle
95+
Results
96+
97+
1. The original RGB input image with dimensions of (480, 640, 3):
98+
99+
![puppy](./puppy.png)
100+
101+
2. After introducing Gaussian noise, we obtain the following result:
102+
103+
![puppy_noisy](./puppy_noisy.png)
104+
105+
3. When applying the rotate operation with a rotation code of 0 (90 degrees clockwise), we obtain this result:
106+
107+
![puppy_noisy_rotate](./puppy_noisy_rotate.png)
108+
109+
4. Upon applying the flip operation with a flip code of 0 (flipping around the x-axis), we achieve the final result:
110+
111+
![puppy_processed_normalized](./puppy_processed.png)
112+
113+
114+
115+
## Usage Limitations
116+
117+
While Ascend supports most commonly used operators, there are still some limitations that need to be addressed.
118+
119+
- There is no strict limit on the size of the input image used for encoding; however, it depends on the available RAM size of your device.
120+
- Please note that not all data types (dtypes) are supported by every operator. The current dtype limitations are outlined in the following table. We are actively working on addressing these limitations through automatic dtype conversion in an upcoming commit.
121+
122+
123+
| Operator | Supported Dtype |
124+
| ---------------------- | ------------------------------------------------------------ |
125+
| multiply (with scale) | float16,float32,int32 |
126+
| divide (with scale) | float16,float,int32,int8,uint8 |
127+
| bitwise add/or/xor/not | int32,int16,uint16 |
128+
| flip | float16,float,int64,int32,int16,uint16 |
129+
| transpose | float16,float,int64,int32,int16,int8,uint64,uint32,uint16,uint8,bool |
130+
| rotate | float16,float,int64,int32,int16,uint16 |

modules/cannops/tutorials/puppy.png

342 KB
Loading
822 KB
Loading
Loading
820 KB
Loading

0 commit comments

Comments
 (0)