色哟哟视频在线观看-色哟哟视频在线-色哟哟欧美15最新在线-色哟哟免费在线观看-国产l精品国产亚洲区在线观看-国产l精品国产亚洲区久久

0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

在ELF 1 開發(fā)板上實(shí)現(xiàn)讀取攝像頭視頻進(jìn)行目標(biāo)檢測

ElfBoard ? 2024-01-24 10:38 ? 次閱讀

當(dāng)前,將AI深度學(xué)習(xí)算法(如分類、目標(biāo)檢測和軌跡追蹤)部署到嵌入式設(shè)備,進(jìn)而實(shí)現(xiàn)邊緣計(jì)算,正成為輕量級(jí)深度學(xué)習(xí)算法發(fā)展的一個(gè)重要趨勢(shì)。今天將與各位小伙伴分享一個(gè)實(shí)際案例:在ELF 1開發(fā)板上成功部署深度學(xué)習(xí)模型的項(xiàng)目,該項(xiàng)目能夠?qū)崟r(shí)讀取攝像頭視頻流并實(shí)現(xiàn)對(duì)畫面中的物體進(jìn)行精準(zhǔn)的目標(biāo)檢測

項(xiàng)目所需的硬件設(shè)備:1、基于NXP(恩智浦)i.MX6ULL的ELF 1開發(fā)板,2、網(wǎng)線,3、USB攝像頭。


獲取開發(fā)板攝像頭文件路徑

本次項(xiàng)目開發(fā)使用的為普通的USB攝像頭,將攝像頭插在開發(fā)板任一USB口均可。

wKgaomWwc62AdHkFAA84k1aIyk4789.png

Linux開發(fā)板中使用USB攝像頭,通常會(huì)涉及到一些基本的命令行操作。這些操作主要是通過 Video4Linux (V4L2)內(nèi)核框架API進(jìn)行的。以下是一些常用的命令和概念:

1. 列出所有攝像頭設(shè)備: 使用 ls /dev/video* 命令可以列出所有已連接的視頻設(shè)備。這些設(shè)備通常顯示為 /dev/video0 , /dev/video1 等。如下圖,開發(fā)板中對(duì)應(yīng)的攝像頭為/dev/video2(插入哪個(gè)USB口都是一樣的)。

wKgZomWwc9aAS0PnAAFY7oKBqK0923.png

2. 查看攝像頭信息: 使用v4l2-ctl --all -d /dev/video2可以查看特定攝像頭(例如/dev/video2)的所有信息,包括支持的格式、幀率等。可以看到圖象尺寸為640×480,為了和后續(xù)的目標(biāo)檢測輸入圖像大小匹配,需要在程序中進(jìn)行resize。

wKgZomWwc_yADzB3AAMBo4JfDFE785.png

編寫程序,讀取取攝像頭視頻進(jìn)行檢測

并傳遞檢測結(jié)果到上位機(jī)

編寫在開發(fā)板中運(yùn)行的程序,開發(fā)板中運(yùn)行的程序主要有兩個(gè)功能:1. 讀取攝像頭捕捉的視頻并進(jìn)行檢測,2. 將檢測結(jié)果通過網(wǎng)絡(luò)通信傳遞到上位機(jī)中。

首先是第一個(gè)功能,因?yàn)橐贿呉x取視頻,一邊要進(jìn)行圖片檢測,為了提高檢測速度,使用多線程來編寫相應(yīng)的程序。線程間通訊采用隊(duì)列,為避免多線程間的競態(tài),在訪問共享資源時(shí)需要添加互斥鎖。

第二個(gè)功能,采用socket通信,將檢測后的圖像發(fā)送到上位機(jī)中即可。

下面是完整的程序?qū)崿F(xiàn):

/*命名為 squeezenetssd_thread.cpp */ #include "net.h" #include

#include

#include #include #include #include #include #include

#include /*增加多線程代碼*/ #include #include #include #include /*隊(duì)列通信,全局變量*/ std::queue frameQueue; std::mutex queueMutex; std::condition_variable queueCondVar; bool finished = false; const size_t MAX_QUEUE_SIZE = 2; // 設(shè)為兩個(gè),因?yàn)闄z測速度實(shí)在太慢,多了意義不大

struct Object { cv::Rect_ rect; int label; float prob; }; ncnn::Net squeezenet; int client_sock; static int detect_squeezenet(const cv::Mat& bgr, std::vector& objects)

{ const int target_size = 300; int img_w = bgr.cols; int img_h = bgr.rows; ncnn::Mat in = ncnn::Mat::from_pixels_resize(bgr.data, ncnn::Mat::PIXEL_BGR, bgr.cols, bgr.rows, target_size, target_size);

const float mean_vals[3] = {104.f, 117.f, 123.f};

in.substract_mean_normalize(mean_vals, 0); ncnn::Extractor ex = squeezenet.create_extractor(); ex.input("data", in); ncnn::Mat out; ex.extract("detection_out", out); // printf("%d %d %d\n", out.w, out.h, out.c); objects.clear(); for (int i = 0; i < out.h; i++)?

{ const float* values = out.row(i); Object object; object.label = values[0]; object.prob = values[1];

object.rect.x = values[2] * img_w; object.rect.y = values[3] * img_h; object.rect.width = values[4] * img_w - object.rect.x; object.rect.height = values[5] * img_h - object.rect.y; objects.push_back(object); } return 0; } static void draw_objects(cv::Mat& bgr, const std::vector& objects)

{ static const char* class_names[] = {"background", "aeroplane", "bicycle", "bird", "boat", "bottle", "bus", "car", "cat", "chair", "cow", "diningtable", "dog", "horse", "motorbike", "person", "pottedplant", "sheep", "sofa", "train", "tvmonitor" }; //cv::Mat image = bgr.clone(); //cv::Mat& image = bgr; for (size_t i = 0; i < objects.size(); i++) { const Object& obj = objects[i];?

fprintf(stderr, "%d = %.5f at %.2f %.2f %.2f x %.2f\n", obj.label, obj.prob, obj.rect.x, obj.rect.y, obj.rect.width, obj.rect.height);

cv::rectangle(bgr, obj.rect, cv::Scalar(255, 0, 0)); char text[256];

sprintf(text, "%s %.1f%%", class_names[obj.label], obj.prob * 100); int baseLine = 0; cv::Size label_size = cv::getTextSize(text, cv::FONT_HERSHEY_SIMPLEX, 0.5, 1, &baseLine); int x = obj.rect.x; int y = obj.rect.y - label_size.height - baseLine; if (y < 0) y = 0; if (x + label_size.width > bgr.cols) x = bgr.cols - label_size.width; cv::rectangle(bgr, cv::Rect(cv::Point(x, y), cv::Size(label_size.width, label_size.height + baseLine)), cv::Scalar(255, 255, 255), -1); cv::putText(bgr, text, cv::Point(x, y + label_size.height), cv::FONT_HERSHEY_SIMPLEX, 0.5, cv::Scalar(0, 0, 0)); } // cv::imshow("image", image); // cv::waitKey(0); } void send_to_client(const cv::Mat& image, int client_sock) { std::vector buffer; std::vector params = {cv::IMWRITE_JPEG_QUALITY, 80}; cv::imencode(".jpg", image, buffer, params); uint32_t len = htonl(buffer.size()); send(client_sock, &len, sizeof(len), 0);

send(client_sock, buffer.data(), buffer.size(), 0); } /*線程1工作函數(shù),此線程是用來采集相機(jī)圖像的*/ static void captureThreadFunction(cv::VideoCapture& cap) { while (true) { cv::Mat frame; cap >> frame; if (frame.empty()) { finished = true; queueCondVar.notify_all(); break; } cv::rotate(frame, frame, cv::ROTATE_90_COUNTERCLOCKWISE); // 圖像旋轉(zhuǎn)90度 std::unique_lock lock(queueMutex); if (frameQueue.size() >= MAX_QUEUE_SIZE) { frameQueue.pop(); // 丟棄最舊的幀

} frameQueue.push(frame); queueCondVar.notify_one(); } } /* 線程2工作函數(shù),此線程是用來檢測圖像*/

void processThreadFunction() { int frameCount = 0; while (true) { cv::Mat frame; { std::unique_lock lock(queueMutex); queueCondVar.wait(lock, []{ return !frameQueue.empty() || finished; }); if (finished && frameQueue.empty()) { // 退出前釋放鎖 return; // 使用 return 替代 break 來確保在持有鎖時(shí)不退出循環(huán)

} frame = frameQueue.front(); frameQueue.pop(); } // 鎖在這里被釋放 // 檢測代碼...

std::vector objects; // if (++frameCount % 5 == 0) { // detect_squeezenet(frame, objects); // frameCount = 0; // } detect_squeezenet(frame,objects); draw_objects(frame, objects); send_to_client(frame, client_sock); if (cv::waitKey(1) >= 0)

{ break; } } } int main() { int server_sock = socket(AF_INET, SOCK_STREAM, 0); if (server_sock < 0)?

{ perror("socket 創(chuàng)建失敗"); return -1; } struct sockaddr_in server_addr; server_addr.sin_family = AF_INET; server_addr.sin_port = htons(12345); server_addr.sin_addr.s_addr = INADDR_ANY; if (bind(server_sock, (struct sockaddr*)&server_addr, sizeof(server_addr)) < 0)?

{ perror("bind 失敗"); close(server_sock); return -1; } if (listen(server_sock, 1) < 0) { perror("listen 失敗");

close(server_sock); return -1; } printf("等待客戶端連接...\n"); struct sockaddr_in client_addr; socklen_t client_len = sizeof(client_addr); client_sock = accept(server_sock, (struct sockaddr*)&client_addr, &client_len); if (client_sock < 0) { perror("accept 失敗"); close(server_sock); return -1; }?

printf("客戶端已連接\n"); // ... [模型加載和初始化代碼] ... cv::VideoCapture cap("/dev/video2");

if (!cap.isOpened()) { fprintf(stderr, "攝像頭打開失敗\n"); return -1; } squeezenet.opt.use_vulkan_compute = true; // original pretrained model from https://github.com/chuanqi305/SqueezeNetSSD // squeezenet_ssd_voc_deploy.prototxt // https://drive.google.com/open?id=0B3gersZ2cHIxdGpyZlZnbEQ5Snc // the ncnn model https://github.com/nihui/ncnn-assets/tree/master/models if (squeezenet.load_param("squeezenet_ssd_voc.param")) exit(-1);

if (squeezenet.load_model("squeezenet_ssd_voc.bin")) exit(-1);

std::thread captureThread(captureThreadFunction, std::ref(cap)); std::thread processThread(processThreadFunction); captureThread.join(); processThread.join(); cap.release(); close(client_sock); close(server_sock); return 0; }

將上述程序,拷貝到ncnn目錄下,并更改CMakeLists.txt文件。

1、拷貝程序

wKgaomWwdPiAS4oeAALOyg8ZGRs995.png

2、更改CMakeLists.txt文件

wKgaomWwdR-AWsvVAAMdliYh20M564.png

wKgaomWwdTuAP9lYAAFc-1-BiPg034.pngwKgaomWwdU6AIqG7AAISuue24A4519.png

做好以上工作后,我們直接進(jìn)入ncnn-master/build/examples/ 文件夾下進(jìn)行編譯。編譯之前直接切換到ncnn-master/build 目錄輸入:

cmake -DCMAKE_TOOLCHAIN_FILE=../toolchains/arm-linux-gnueabihf.toolchain.cmake - DNCNN_SIMPLEOCV=ON -DNCNN_BUILD_EXAMPLES=ON -DCMAKE_BUILD_TYPE=Release ..

然后切換到ncnn-master/build/examples/ 目錄下輸入 make -j4 即可。

wKgZomWwd2-AM6VeAALgCsFSxBo812.png

可見編譯成功,拷貝到開發(fā)板中就行。

wKgaomWwd4uAbCsxAAESAXakkr4044.png

編寫上位機(jī)軟件

上位機(jī)軟件較為簡單,使用ChatGPT生成源碼即可,下面附上源碼:

import socket import cv2 import numpy as np client_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM) client_socket.connect(('192.168.0.232', 12345)) # Connect to the server while True: # Receive size of the frame size = client_socket.recv(4) size = int.from_bytes(size, byteorder='big') # Receive the frame buffer = b'' while len(buffer) < size: buffer += client_socket.recv(size - len(buffer)) # Decode and display the frame frame = np.frombuffer(buffer, dtype=np.uint8) frame = cv2.imdecode(frame, cv2.IMREAD_COLOR) cv2.imshow('Received Frame', frame) if cv2.waitKey(1) & 0xFF == ord('q'): break client_socket.close() cv2.destroyAllWindows()

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • 嵌入式
    +關(guān)注

    關(guān)注

    5086

    文章

    19143

    瀏覽量

    306044
  • Linux
    +關(guān)注

    關(guān)注

    87

    文章

    11316

    瀏覽量

    209814
  • 攝像頭
    +關(guān)注

    關(guān)注

    60

    文章

    4850

    瀏覽量

    95884
  • 開發(fā)板
    +關(guān)注

    關(guān)注

    25

    文章

    5080

    瀏覽量

    97678
收藏 人收藏

    評(píng)論

    相關(guān)推薦

    ElfBoard技術(shù)貼|ELF 1開發(fā)板適配攝像頭詳解

    ELF1ELF1S開發(fā)板適配的OV5640攝像頭,集成了CMOS圖像傳感器,作為一款500萬像素級(jí)別的攝像頭,不僅能夠支持最高達(dá)2592x
    的頭像 發(fā)表于 12-27 10:16 ?614次閱讀
    ElfBoard技術(shù)貼|<b class='flag-5'>ELF</b> <b class='flag-5'>1</b><b class='flag-5'>開發(fā)板</b>適配<b class='flag-5'>攝像頭</b>詳解

    基于FPGA的攝像頭心率檢測裝置設(shè)計(jì)

    裝置, 通過網(wǎng)絡(luò)攝像頭獲取人臉視頻圖像,通過 FPGA 進(jìn)行分析處理,得到心跳信號(hào), 并將計(jì)算得出心率值實(shí)時(shí)顯示 HDMI 屏幕。 1.
    發(fā)表于 07-01 17:58

    基于TI AM437x 創(chuàng)龍開發(fā)板的OV2659攝像頭模塊測試

    視頻的實(shí)時(shí)采集,并將視頻保存在本地或者顯示LCD顯示屏開發(fā)板圖如下:具體測試如下:1.本地
    發(fā)表于 05-15 09:40

    通過Dragonboard 410c開發(fā)板USB攝像頭進(jìn)行移動(dòng)偵測

    將我們的USB攝像頭、鼠標(biāo)和鍵盤鏈接到dragonbaord 410c開發(fā)板即可,如果你使用的是有線的USB鼠標(biāo)和鍵盤,USB鼠標(biāo)和攝像頭可以共用一個(gè)。搭建好環(huán)境后,我們就可以來
    發(fā)表于 09-21 10:56

    【FPGA DEMO】Lab 4:攝像頭HDMI顯示(高速--HDMI&攝像頭)

    `項(xiàng)目名稱:攝像頭HDMI顯示。具體要求:攝像頭采集的視頻圖像數(shù)據(jù)通過HDMI實(shí)時(shí)顯示。 系統(tǒng)設(shè)計(jì):Perf-V開發(fā)板可以連接高速口——HDMI&
    發(fā)表于 07-30 15:21

    OV13850攝像頭開發(fā)板的配置過程怎樣的?

    OV13850攝像頭開發(fā)板的配置過程怎樣的?
    發(fā)表于 03-07 08:06

    【合宙Air105開發(fā)板試用體驗(yàn)】Ari105開發(fā)板開箱,及攝像頭使用

    ,直接上攝像頭了。要正常使用攝像頭,我們需要如下的準(zhǔn)備:1. 連接好攝像頭2. 了解開發(fā)板的撥
    發(fā)表于 05-22 21:41

    一鍵實(shí)現(xiàn)V853開發(fā)板攝像頭自由

    視頻中所實(shí)現(xiàn)攝像頭UI控制效果,是通過全志V853開發(fā)板使用InoneGUI+mpp+Ai
    發(fā)表于 09-08 13:41

    RK3288開發(fā)板安卓5.1實(shí)現(xiàn)虛擬攝像頭

    1、RK3288下如何實(shí)現(xiàn)虛擬攝像頭  提示:由于項(xiàng)目技術(shù)要求,需要在RK3288開發(fā)板安卓5.1實(shí)現(xiàn)
    發(fā)表于 11-23 17:16

    ELF 1開發(fā)板試用】+ 3.2 USB攝像頭連接測試 + Ubutu SSH連接

    連接測試 本文繼續(xù)做一下連接接口的測試,本文使用USB攝像頭進(jìn)行連接,以測試ELF 1 USB功能,以及其對(duì)應(yīng)的攝像頭
    發(fā)表于 12-06 15:37

    ELF 1開發(fā)板試用】板載資源測試3:OV5640 攝像頭測試

    飛凌嵌入式ELF1開發(fā)板(以下簡稱為“開發(fā)板”)底板設(shè)計(jì)了一個(gè)攝像頭接口,正好手里邊有一個(gè)OV5640攝像頭,但由于沒有擴(kuò)展板,手里邊也沒有
    發(fā)表于 12-15 22:49

    基于Dragonboard 410c開發(fā)板的USB攝像頭圖像保存實(shí)現(xiàn)

    前一個(gè)blog中跟大家分享了如何快速的編寫程序從連接在Dragonboard 410c開發(fā)板的USB攝像頭讀取圖像信息,給大家簡單介紹了一下如何搭建Dragonabord 410c
    發(fā)表于 02-17 10:19 ?1228次閱讀

    迅為RK3568開發(fā)板Debian系統(tǒng)使用python 進(jìn)行攝像頭開發(fā)

    迅為RK3568開發(fā)板Debian系統(tǒng)使用python 進(jìn)行攝像頭開發(fā)
    的頭像 發(fā)表于 09-14 16:58 ?1566次閱讀
    迅為RK3568<b class='flag-5'>開發(fā)板</b>Debian系統(tǒng)使用python <b class='flag-5'>進(jìn)行</b><b class='flag-5'>攝像頭</b><b class='flag-5'>開發(fā)</b>

    項(xiàng)目分享|基于ELF 1開發(fā)板的遠(yuǎn)程監(jiān)測及人臉識(shí)別項(xiàng)目

    小伙伴詳盡展示這一項(xiàng)目的相關(guān)細(xì)節(jié)。項(xiàng)目實(shí)現(xiàn)步驟1.視頻監(jiān)控這一步驟中需要實(shí)現(xiàn)兩個(gè)程序:(1
    的頭像 發(fā)表于 03-13 16:41 ?542次閱讀
    項(xiàng)目分享|基于<b class='flag-5'>ELF</b> <b class='flag-5'>1</b><b class='flag-5'>開發(fā)板</b>的遠(yuǎn)程監(jiān)測及人臉識(shí)別項(xiàng)目
    主站蜘蛛池模板: 日本50人群体交乱| 亚洲色婷婷久久精品AV蜜桃| 欧美激情视频一区二区| 欧美大片xxxxbbbb| 欧洲日韩av无线在码| 泰国淫乐园实录| 亚洲日韩在线天堂一| 607080老太太AW| www.一级毛片| 国产精品大全国产精品| 饥渴的新婚女教师| 免费毛片播放| 色欲档案之麻雀台上淫| 亚洲免费视频在线观看| 506070老熟肥妇bbwxx视频| 成年色黄APP下载| 国产午夜一级淫片| 啦啦啦影院视频在线看高清...| 欧美一区二区高清| 性奴公司 警花| 2020美女视频黄频大全视频| 岛国片免费在线观看| 精品国产福利一区二区在线| 暖暖的视频完整视频免费韩国| 无码日韩人妻精品久久蜜桃免费 | 国产国产成人人免费影院| 接吻吃胸摸下面啪啪教程| 内射爽无广熟女亚洲| 特污兔午夜影视院| 在线观看亚洲 日韩 国产| 伧理片午夜伧理片| 韩国伦理三级| 欧美在线看费视频在线| 亚洲国产AV一区二区三区四区| 777黄色片| 国产日韩精品SUV| 免费伦理片网站| 亚洲国产精品免费观看| AV一区AV久久AV无码| 果冻传媒在线观看网站| 欧美激情久久久久久久大片|