本文是一個(gè)小系列的第四篇,MCU部署OpenCV的“進(jìn)階篇”,已經(jīng)發(fā)表了“先跑篇”、"配置篇"和“實(shí)戰(zhàn)篇”稍后會(huì)陸續(xù)還有“優(yōu)化篇”,帶您牽手OpenCV,進(jìn)入OpenCV的廣闊世界。
上一期小編帶著大家簡(jiǎn)單過(guò)了一下OpenCV的API的基本使用方法,并最終在MCU上實(shí)際跑了下。相信大家看的一定不是很過(guò)癮,本期小編將和大家一起完全從0開(kāi)始構(gòu)建一個(gè)基于MCUXPresso的OpenCV測(cè)試工程,并部署到MCU上。
提到這兒,可能有朋友會(huì)提出質(zhì)疑:上期不是說(shuō)過(guò)了,基于HelloWorld例程,把那5個(gè)可愛(ài)的庫(kù)一股腦放進(jìn)去就可以了嗎?
可能事情并不是這么簡(jiǎn)單喲!不然,小編可就有湊字+湊篇幅的嫌疑了啊。還請(qǐng)聽(tīng)小編娓娓道來(lái)。
我們知道OpenCV是基于C++編寫(xiě)的項(xiàng)目,而我們所常用的Hello_World例程實(shí)際上是一個(gè)C工程。這樣一來(lái),原生的Hello_World的C工程是無(wú)法兼容,所編譯出的OpenCV庫(kù)工程的,或者說(shuō),工程本身不能支持構(gòu)建C++工程。
因此,在開(kāi)始部署之前,要針對(duì)性地對(duì)工程本身進(jìn)行一些小小的改造,以添加對(duì)于C++的支持。
具體步驟如下:
一 所謂站在巨人的肩膀看得遠(yuǎn),先做一些準(zhǔn)備工作:
-
SDK代碼包:2.11.0 for i.MX RT1170
-
MCUXPresso IDE:11.5.0
-
選取參考例程:SDK_rootoardsevkmimxrt1170demo_appshello_world_demo_cm7
-
準(zhǔn)備好5個(gè)靜態(tài)庫(kù):libopencv_world, libopenjp2, libjpeg-turbo, libpng, zlib
二通過(guò)Quickstart Panel導(dǎo)入hello world例程:
三犀利的操作:添加C++支持,找到工程目錄下的.project文件,并添加:
修改好之后,使用MCUXPresso IDE重新打開(kāi)工程即可開(kāi)啟C++屬性。
四此時(shí)的C++工程屬性還是空的,首先是針對(duì)MCU C++ Compiler的配置,包括頭文件以及預(yù)編譯符號(hào)的添加:
頭文件路徑如下:
"${workspace_loc:/${ProjName}/drivers}"
"${workspace_loc:/${ProjName}/board}"
"${workspace_loc:/${ProjName}/source}"
"${workspace_loc:/${ProjName}/utilities}"
"${workspace_loc:/${ProjName}/drivers}"
"${workspace_loc:/${ProjName}/device}"
"${workspace_loc:/${ProjName}/component/uart}"
"${workspace_loc:/${ProjName}/component/lists}"
"${workspace_loc:/${ProjName}/startup}"
"${workspace_loc:/${ProjName}/xip}"
"${workspace_loc:/${ProjName}/CMSIS}"
"${workspace_loc:/${ProjName}/utilities}"
"${workspace_loc:/${ProjName}/device}"
"your_cv_pathopencvuild"
" your_cv_path opencvinclude"
" your_cv_path opencvmodulescoreinclude"
" your_cv_path opencvmodulesimgcodecsinclude"
" your_cv_path opencvmodulesimgprocinclude"
" your_cv_path opencvmodulesworldinclude"
" your_cv_path opencvmoduleshighguiinclude"
" your_cv_path opencvmodulesfeatures2dinclude"
" your_cv_path opencvmodulesmlinclude"
" your_cv_path opencvmodulesvideoinclude"
預(yù)編譯符號(hào):
OPENCV_DISABLE_THREAD_SUPPORT=1
__NEWLIB__
CPU_MIMXRT1176DVMAA
CPU_MIMXRT1176DVMAA_cm7
XIP_BOOT_HEADER_DCD_ENABLE=1
USE_SDRAM
DATA_SECTION_IS_CACHEABLE=1
SDK_DEBUGCONSOLE=1
XIP_EXTERNAL_FLASH=1
XIP_BOOT_HEADER_ENABLE=1
PRINTF_FLOAT_ENABLE=0
SCANF_FLOAT_ENABLE=0
PRINTF_ADVANCED_ENABLE=0
SCANF_ADVANCED_ENABLE=0
FSL_SDK_DRIVER_QUICK_ACCESS_ENABLE=1
MCUXPRESSO_SDK
CR_INTEGER_PRINTF
__MCUXPRESSO
__USE_CMSIS
DEBUG
接下來(lái)是MCU C++ Linker的配置,包括所引用的庫(kù)名字以及庫(kù)搜索路徑:
要注意,庫(kù)的搜索路徑就是存放上面那5個(gè)庫(kù)的位置。
五 開(kāi)啟C++編程模式,問(wèn):C文件切換成C++文件需要幾步?答:只需一步!重命名hello_world.c->hello_world.cc即可。內(nèi)容可以保存不變。
六導(dǎo)入測(cè)試數(shù)據(jù),包括壓縮格式(jpeg,PNG)或是其他未經(jīng)壓縮的原始數(shù)據(jù)。
考慮到MCU平臺(tái)一般沒(méi)有片上的文件系統(tǒng),或者說(shuō)沒(méi)有集成文件系統(tǒng)。那么我們的測(cè)試數(shù)據(jù)就要以RO data的形式直接集成到鏡像中。
為了實(shí)現(xiàn)這一需求,介紹給大家一個(gè)很好用的匯編指令:.incbin,顧名思義,指令本身就好像在隱隱地告訴我們,我就是用來(lái)include bin文件的,快點(diǎn)用我。
既然是匯編指令,就要新建一個(gè)匯編文件到我們的工程中。新建匯編文件放到哪里,沒(méi)有特殊要求,但是最好放到和hello_world.cc文件同一級(jí)目錄下:
添枝加葉:
.global img_start
.global img_end
img_start:
.incbin "data/lena.jpg"
img_end:
具體測(cè)試圖片的名字可以任意指定,只不過(guò)要注意一點(diǎn)。如果想要使用相對(duì)路徑的話(huà),要保證存儲(chǔ)圖片的位置要和hello_world.cc的位置一致。
七至此,MCUXPresso工程就準(zhǔn)備完畢了,下一步就是編寫(xiě)測(cè)試代碼。要注意,因?yàn)?/span>我們已經(jīng)切換到了C++編程模式,就要順著C++的脾氣來(lái)。
比較重要的一條是:如果不想重命名hello_world.c,說(shuō):我就看.c尾綴舒服。沒(méi)問(wèn)題,但是請(qǐng)不要忘了,在聲明函數(shù)的時(shí)候,不要忘了用extern “C”來(lái)修飾。否則,會(huì)有千千萬(wàn)萬(wàn)個(gè)link error向你撲面而來(lái)。
接下來(lái)開(kāi)始正式編寫(xiě)測(cè)試代碼:
一包含頭文件,你只需要一行即可,如此和諧友善:
#include "opencv2opencv.hpp"
二OpenCV使用cv::Mat來(lái)表征數(shù)據(jù),首先我們要聲明并初始化cv::Mat實(shí)例。
考慮到?jīng)]有片上文件系統(tǒng)的支持,上文也提到直接使用.incbin導(dǎo)入圖片數(shù)據(jù)。這里,我們就可以使用在匯編文件中所定義的符號(hào)對(duì)這些數(shù)據(jù)進(jìn)行訪(fǎng)問(wèn)。如果是壓縮后的圖片,需要首先進(jìn)行解碼操作;如果是源數(shù)據(jù)的話(huà),可以直接使用:
extern uint8_t img_start[];
extern uint8_t img_end[];
#define IMG_LEN (img_end - img_start)
// compressed data
std::vector data(img_start, img_start + IMG_LEN);
cv::Mat img = cv::Mat(data), IMREAD_UNCHANGED);
// raw data, need to aware the shape, and also the depth, such as rgb == CV_8UC3, equal to
// each pixel has 3 items, and each item is 8bits
Mat img(Size(480, 360), CV_8UC3);
memcpy(img.data, img_start, IMG_LEN);
三尋找物體輪廓并畫(huà)出:
vector> contours;
vector hierarchy;
findContours(dst, contours, hierarchy, RETR_EXTERNAL, CHAIN_APPROX_SIMPLE);
// To display the contours
Mat resultImage = Mat ::zeros(dst.size(),CV_8U);
drawContours(resultImage, contours, -1, Scalar(255, 0, 255));
四接下來(lái)是一個(gè)更加復(fù)雜的任務(wù),尋找矩形:
參考代碼如下:
// returns sequence of squares detected on the image.
static void findSquares( const Mat& image, vector >& squares )
{
squares.clear();
Mat pyr, timg, gray0(image.size(), CV_8U), gray;
// down-scale and upscale the image to filter out the noise
pyrDown(image, pyr, Size(image.cols/2, image.rows/2));
pyrUp(pyr, timg, image.size());
vector > contours;
for( int c = 0; c < 3; c++ )
{
int ch[] = {c, 0};
mixChannels(&timg, 1, &gray0, 1, ch, 1);
// try several threshold levels
for( int l = 0; l < N; l++ )
{
if( l == 0 )
{
Canny(gray0, gray, 0, thresh, 5);
dilate(gray, gray, Mat(), Point(-1,-1));
}
else
{
gray = gray0 >= (l+1)*255/N;
}
findContours(gray, contours, RETR_LIST, CHAIN_APPROX_SIMPLE);
vector approx;
// test each contour
for( size_t i = 0; i < contours.size(); i++ )
{
// approximate contour with accuracy proportional to the contour perimeter
approxPolyDP(contours[i], approx, arcLength(contours[i], true)*0.02, true);
// square contours should have 4 vertices after approximation
if( approx.size() == 4 &&
fabs(contourArea(approx)) > 1000 &&
isContourConvex(approx) )
{
double maxCosine = 0;
for( int j = 2; j < 5; j++ )
{
// find the maximum cosine of the angle between joint edges
double cosine = fabs(angle(approx[j%4], approx[j-2], approx[j-1]));
maxCosine = MAX(maxCosine, cosine);
}
if( maxCosine < 0.3 )
squares.push_back(approx);
}
}
}
}
}
五編碼圖像,這里我們選擇利用調(diào)試器將編碼后的數(shù)據(jù)從內(nèi)存download到我們的PC上。關(guān)于如何在MCUXPress中進(jìn)行數(shù)據(jù)保存的操作,在上一篇文章中已有介紹。
std::vector decoded_img;
cv::imencode(".jpeg", img, decoded_img);
uchar *data = decoded_img.data();
不過(guò),這里有個(gè)小坑要提醒給大家,在傳遞編碼格式時(shí)候,請(qǐng)不要忘記那個(gè)人見(jiàn)人愛(ài)的句點(diǎn)“.” 。也就是說(shuō),編碼格式要寫(xiě)成.jpeg而不是jpeg。小編可是在這上面吃過(guò)虧的。
至此,本期小編就給大家介紹了如何從0開(kāi)始新建一個(gè)MCUXPresso工程,并編寫(xiě)OpenCV測(cè)試代碼進(jìn)行測(cè)試。感興趣的小伙伴們快動(dòng)起手來(lái)吧!
審核編輯 :李倩
-
mcu
+關(guān)注
關(guān)注
146文章
17718瀏覽量
358368 -
C++
+關(guān)注
關(guān)注
22文章
2116瀏覽量
74588 -
OpenCV
+關(guān)注
關(guān)注
31文章
642瀏覽量
42253
原文標(biāo)題:這個(gè)秋天,OpenCV和MCU更配喲(進(jìn)階篇)
文章出處:【微信號(hào):NXP_SMART_HARDWARE,微信公眾號(hào):恩智浦MCU加油站】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。
發(fā)布評(píng)論請(qǐng)先 登錄
相關(guān)推薦
快速部署!米爾全志T527開(kāi)發(fā)板的OpenCV行人檢測(cè)方案指南
為L(zhǎng)SDK構(gòu)建opencv時(shí)遇到的問(wèn)題求解
零基礎(chǔ)開(kāi)發(fā)小安派-Eyes-S1 進(jìn)階篇 ——通過(guò)屏幕輸入連接 Wi-Fi

AI模型部署邊緣設(shè)備的奇妙之旅:如何在邊緣端部署OpenCV
AI模型部署邊緣設(shè)備的奇妙之旅:如何在邊緣端部署OpenCV

迅為iTOP-RK3568開(kāi)發(fā)板驅(qū)動(dòng)開(kāi)發(fā)指南-第十八篇 PWM
文檔更新 | 迅為RK3568驅(qū)動(dòng)指南-第十七篇(串口)
【《大語(yǔ)言模型應(yīng)用指南》閱讀體驗(yàn)】+ 俯瞰全書(shū)
OpenCV圖像識(shí)別C++代碼
opencv-python和opencv一樣嗎
opencv的主要功能有哪些
文檔更新 |迅為 RK3568開(kāi)發(fā)板驅(qū)動(dòng)指南-第十五/十六篇
I.MX6ULL-飛凌 ElfBoard ELF1板卡 - 如何在Ubuntu中編譯OpenCV庫(kù)(X86架構(gòu))
嵌入式學(xué)習(xí)-飛凌ElfBoard ELF 1板卡 - 如何在Ubuntu中編譯OpenCV庫(kù)
ELF 1技術(shù)貼|如何在Ubuntu中編譯OpenCV庫(kù)

評(píng)論