【C++】开源:Linux端V4L2视频设备库

chatgpt/2023/9/26 14:28:23

😏★,°:.☆( ̄▽ ̄)/$:.°★ 😏
这篇文章主要介绍Linux端V4L2视频设备库。
无专精则不能成,无涉猎则不能通。——梁启超
欢迎来到我的博客,一起学习,共同进步。
喜欢的朋友可以关注一下,下次更新不迷路🥞

文章目录

    • :smirk:1. 项目介绍
    • :blush:2. 环境配置
    • :satisfied:3. 使用说明

😏1. 项目介绍

Video4Linux2(V4L2)是一个用于Linux操作系统的视频设备驱动框架。它提供了一个统一的接口,用于在应用程序和视频设备之间进行通信和交互。

V4L2支持各种类型的视频设备,包括USB摄像头、摄像机、TV调谐器、网络摄像头等。通过使用V4L2,开发者可以轻松地访问和控制视频设备,以捕获视频流、调整图像参数、设置视频格式和分辨率等。

以下是V4L2的一些重要特点和概念:

1.设备节点:每个视频设备在Linux系统中都表示为一个设备节点,通常位于/dev/video*路径下。应用程序通过打开这些设备节点来访问相应的视频设备。

2.视频捕捉:V4L2允许应用程序从视频设备中捕获视频帧或图像。它提供了一系列的API函数,使应用程序能够请求存储视频帧的缓冲区,并在设备准备好时将其读取到内存中。

3.视频输出:除了捕获视频,V4L2还支持将视频数据发送到视频设备,以便在外部显示设备上进行输出。应用程序可以将视频帧写入输出缓冲区,并通过相应的IOCTL调用将其发送到视频设备。

4.控制和参数设置:V4L2允许应用程序对视频设备进行控制和配置。例如,应用程序可以设置摄像头的亮度、对比度、饱和度等参数,选择摄像头的输入源,设置视频格式和分辨率等。

5.帧缓冲管理:V4L2通过Frame Buffer子系统来管理视频帧的缓冲区。它提供了API函数来请求和管理用于存储视频帧的缓冲区,并进行帧缓冲的交换和处理。

😊2. 环境配置

下面进行环境配置:

# v4l2是linux内核的一部分,只需安装开发库
sudo apt-get install libv4l-dev
# 使用v4l2开发
# 在应用程序中使用 #include <linux/videodev2.h> 来引入V4L2的头文件,并使用相关的API函数

😆3. 使用说明

下面进行使用分析:

基于v4l2调用usb摄像头并用opencv显示示例:

#include <iostream>
#include <cstdlib>
#include <cstring>
#include <fcntl.h>
#include <unistd.h>
#include <sys/ioctl.h>
#include <sys/mman.h>	//共享内存
#include <linux/videodev2.h>
#include <opencv2/opencv.hpp>#define WIDTH 640
#define HEIGHT 480int main() {int fd;struct v4l2_capability cap;struct v4l2_format fmt;struct v4l2_requestbuffers req;struct v4l2_buffer buf;enum v4l2_buf_type type;// 打开摄像头设备fd = open("/dev/video0", O_RDWR);if (fd == -1) {std::cerr << "无法打开摄像头设备" << std::endl;return 1;}// 查询摄像头能力if (ioctl(fd, VIDIOC_QUERYCAP, &cap) == -1) {std::cerr << "无法查询摄像头能力" << std::endl;close(fd);return 1;}// 设置视频格式memset(&fmt, 0, sizeof(fmt));fmt.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;fmt.fmt.pix.width = WIDTH;fmt.fmt.pix.height = HEIGHT;fmt.fmt.pix.pixelformat = V4L2_PIX_FMT_YUYV; // YUV格式if (ioctl(fd, VIDIOC_S_FMT, &fmt) == -1) {std::cerr << "无法设置视频格式" << std::endl;close(fd);return 1;}// 请求视频缓冲区memset(&req, 0, sizeof(req));req.count = 1;req.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;req.memory = V4L2_MEMORY_MMAP;if (ioctl(fd, VIDIOC_REQBUFS, &req) == -1) {std::cerr << "无法请求视频缓冲区" << std::endl;close(fd);return 1;}// 映射视频缓冲区到用户空间struct v4l2_buffer* buffers = new v4l2_buffer[req.count];void** frame_buffers = new void*[req.count];for (int i = 0; i < req.count; i++) {memset(&buf, 0, sizeof(buf));buf.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;buf.memory = V4L2_MEMORY_MMAP;buf.index = i;if (ioctl(fd, VIDIOC_QUERYBUF, &buf) == -1) {std::cerr << "无法查询视频缓冲区" << std::endl;close(fd);return 1;}frame_buffers[i] = mmap(NULL, buf.length, PROT_READ | PROT_WRITE, MAP_SHARED, fd, buf.m.offset);if (frame_buffers[i] == MAP_FAILED) {std::cerr << "无法映射视频缓冲区到用户空间" << std::endl;close(fd);return 1;}}// 入队视频缓冲区for (int i = 0; i < req.count; i++) {memset(&buf, 0, sizeof(buf));buf.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;buf.memory = V4L2_MEMORY_MMAP;buf.index = i;if (ioctl(fd, VIDIOC_QBUF, &buf) == -1) {std::cerr << "无法入队视频缓冲区" << std::endl;close(fd);return 1;}}// 开始视频流采集type = V4L2_BUF_TYPE_VIDEO_CAPTURE;if (ioctl(fd, VIDIOC_STREAMON, &type) == -1) {std::cerr << "无法开始视频流采集" << std::endl;close(fd);return 1;}// 循环获取并显示相机数据cv::Mat frame(HEIGHT, WIDTH, CV_8UC2);cv::namedWindow("Camera", cv::WINDOW_AUTOSIZE);while (true) {// 出队视频缓冲区memset(&buf, 0, sizeof(buf));buf.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;buf.memory = V4L2_MEMORY_MMAP;if (ioctl(fd, VIDIOC_DQBUF, &buf) == -1) {std::cerr << "无法出队视频缓冲区" << std::endl;close(fd);return 1;}// 处理相机数据(这里只是简单地将YUYV格式的数据转换为RGB格式)cv::cvtColor(cv::Mat(HEIGHT, WIDTH, CV_8UC2, frame_buffers[buf.index]), frame, cv::COLOR_YUV2BGR_YUYV);// 显示相机数据cv::imshow("Camera", frame);if (cv::waitKey(1) == 27) {break; // 按下Esc键退出循环}// 再次入队视频缓冲区if (ioctl(fd, VIDIOC_QBUF, &buf) == -1) {std::cerr << "无法再次入队视频缓冲区" << std::endl;close(fd);return 1;}}// 停止视频流采集type = V4L2_BUF_TYPE_VIDEO_CAPTURE;if (ioctl(fd, VIDIOC_STREAMOFF, &type) == -1) {std::cerr << "无法停止视频流采集" << std::endl;close(fd);return 1;}// 解除映射视频缓冲区for (int i = 0; i < req.count; i++) {munmap(frame_buffers[i], buf.length);}// 关闭摄像头设备close(fd);delete[] buffers;delete[] frame_buffers;return 0;
}

编译运行:

g++ -o main main.cpp `pkg-config --libs opencv`
./main

在这里插入图片描述

以上。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.exyb.cn/news/show-5313731.html

如若内容造成侵权/违法违规/事实不符,请联系郑州代理记账网进行投诉反馈,一经查实,立即删除!

相关文章

八大排序算法--冒泡排序(动图理解)

冒泡排序 算法思路 冒泡排序的原理是&#xff1a;从左到右&#xff0c;相邻元素进行比较。每次比较一轮&#xff0c;就会找到序列中最大的一个或最小的一个。这个数就会从序列的最右边冒出来。 以从小到大排序为例&#xff0c;第一轮比较后&#xff0c;所有数中最大的那个数就会…

自动化设备控制的基本规范(防呆必须考虑到)

1.最好用switch case语句控制&#xff0c;条理分明。 2.一个case语句对应一个动作。 3.任何报警必须停机。报警不要弹出对话框&#xff0c;只能显示在信息栏。让客户选择的选项才能弹框提示。报警信息用红色字体显示&#xff0c;报警后必须要有报警提示或者操作提示&#xff0c…

【c语言进阶】字符函数和字符串函数知识总结

字符函数和字符串函数 前期背景求字符串长度函数strlen函数strlen函数三种模拟实现 长度不受限制的字符串函数strcpy函数strcpy函数模拟实现strcat函数strcat函数模拟实现strcmp函数strcmp函数模拟实现 长度受限制的字符串函数strncpy函数strncpy函数模拟实现strncat函数strnca…

前端后端路径问题详解

加了项目名&#xff0c;访问所有页面都是 在 项目名下 出来的路径 不加项目名&#xff0c;访问所有页面都不用加项目名&#xff0c;然后前后端的加/的效果都一样&#xff0c;都是在根目录下没有项目名的路径&#xff01;&#xff01;&#xff01; 后端 一、MVC 1.不管是转发…

Netty3 和Netty4区别

Netty3 和Netty4区别 目录概述需求&#xff1a; 设计思路实现思路分析1.Netty3和Netty4区别2.demo 拓展实现 参考资料和推荐阅读 Survive by day and develop by night. talk for import biz , show your perfect code,full busy&#xff0c;skip hardness,make a better resul…

如何使用Flask-SQLAlchemy来管理数据库连接和操作数据?

首先&#xff0c;我们需要安装Flask-SQLAlchemy。你可以使用pip来安装它&#xff0c;就像这样&#xff1a; pip install Flask-SQLAlchemy好了&#xff0c;现在我们已经有了一个可以操作数据库的工具&#xff0c;接下来让我们来看看如何使用它吧&#xff01; 首先&#xff0c…

c++ inotify+epoll实现异步文件监控

需求 动态监测linux系统某一个目录下文件的变化。具体使用场景如linux下应用程序运行时产生日志文件&#xff0c;尤其在程序出现某种异常时&#xff0c;日志文件记录着错误出现的原因、时间及代码位置等信息&#xff0c;此时日志文件在增长&#xff0c;但是采用轮询的方式定时…

【软件测试】性能测试工具- LoadRunner的介绍和使用

目录 1. LoadRunner是什么2. LoadRunner环境搭建3. LoadRunner三大组件4. LoadRunner脚本录制4.1 WebTous项目介绍启动WebTous项目访问WebTous项目相关配置 4.2 脚本录制新建脚本录制脚本运行脚本 4.3 脚本加强插入事务插入集合点插入检查点插入日志字符串比较 1. LoadRunner是…
推荐文章