Linux基于V4L2的视频捕捉

2024-06-01 20:20
文章标签 视频 linux 捕捉 v4l2

本文主要是介绍Linux基于V4L2的视频捕捉,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

简介

linux环境使用V4l2实现摄像头捕捉,界面流畅播放并可以保存图片到本地。

代码

void VideoCapture::run()
{qDebug() << "VideoCapture start";// 打开设备int fd = open("/dev/video0", O_RDWR);if(fd < 0){qDebug("video设备打开失败\n");return;}else{qDebug("video设备打开成功\n");}//查看设备是否为视频采集设备struct v4l2_capability vcap;ioctl(fd, VIDIOC_QUERYCAP, &vcap);if (!(V4L2_CAP_VIDEO_CAPTURE & vcap.capabilities)){qDebug("No capture video device!\n");return;}// 枚举帧格式struct v4l2_fmtdesc fmtdesc;fmtdesc.index = 0;fmtdesc.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;qDebug("Video支持所有格式如下:");while(ioctl(fd, VIDIOC_ENUM_FMT, &fmtdesc) == 0){qDebug("v4l2_format %d:%s",fmtdesc.index, fmtdesc.description);fmtdesc.index++;}// 枚举分辨率struct v4l2_frmsizeenum frmsize;frmsize.index = 0;frmsize.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;qDebug("Video支持分辨率如下:");while(ioctl(fd, VIDIOC_ENUM_FRAMESIZES, &frmsize) == 0){qDebug("%d frame_size<%d*%d %d>", frmsize.index, frmsize.discrete.width, frmsize.discrete.height, frmsize.pixel_format);frmsize.index++;}// 枚举某分辨率下的帧速率int pixel_width = 1280;int pixel_height = 720;struct v4l2_frmivalenum frmival;frmival.index = 0;frmival.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;frmival.pixel_format = V4L2_PIX_FMT_MJPEG;frmival.width = pixel_width;frmival.height = pixel_height;while(ioctl(fd,VIDIOC_ENUM_FRAMEINTERVALS, &frmival) == 0){qDebug("<%d*%d> support frame_size %dfps", frmival.width, frmival.height, frmival.discrete.denominator / frmival.discrete.numerator);frmival.index++;}// 设置采集格式struct v4l2_format vfmt;vfmt.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;vfmt.fmt.pix.width = pixel_width;vfmt.fmt.pix.height = pixel_height;vfmt.fmt.pix.pixelformat = V4L2_PIX_FMT_MJPEG;if(ioctl(fd, VIDIOC_S_FMT, &vfmt) < 0){qDebug("video设置格式失败\n");return;}// 检查设置参数是否生效if(ioctl(fd, VIDIOC_G_FMT, &vfmt) < 0){qDebug("video获取格式失败\n");return;}// 获取帧信息struct v4l2_streamparm streamparm;streamparm.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;ioctl(fd, VIDIOC_G_PARM, &streamparm);qDebug("current frameRate <%d * %d>\n", streamparm.parm.capture.timeperframe.numerator, streamparm.parm.capture.timeperframe.denominator);// 设置帧信息if(V4L2_CAP_TIMEPERFRAME & streamparm.parm.capture.capability){streamparm.parm.capture.timeperframe.numerator = 1;streamparm.parm.capture.timeperframe.denominator = 30;
//        if(ioctl(fd, VIDIOC_S_PARM, &streamparm) < 0)
//        {
//            qDebug("video设置帧率失败 <%d * %d>", streamparm.parm.capture.timeperframe.numerator, streamparm.parm.capture.timeperframe.denominator);
//        }}// 申请缓冲区空间struct v4l2_requestbuffers reqbuf;reqbuf.type  = V4L2_BUF_TYPE_VIDEO_CAPTURE;reqbuf.count = m_frameCount;reqbuf.memory = V4L2_MEMORY_MMAP;if(ioctl(fd, VIDIOC_REQBUFS, &reqbuf) < 0){qDebug("video申请缓冲区失败\n");return;}// 将帧缓冲映射到进程地址空间struct v4l2_buffer buf;buf.type   = V4L2_BUF_TYPE_VIDEO_CAPTURE;buf.memory = V4L2_MEMORY_MMAP;// 将每一帧对应的缓冲区的起始地址保存在m_userVideoBuf数组中,读取采集数据时,只需直接读取映射区即可for(buf.index=0; buf.index<m_frameCount; buf.index++){ioctl(fd, VIDIOC_QUERYBUF, &buf);m_userVideoBuf[buf.index] = mmap(NULL, buf.length, PROT_READ | PROT_WRITE, MAP_SHARED, fd, buf.m.offset);m_userVideoBufSize[buf.index] = buf.length;if(m_userVideoBuf[buf.index] == MAP_FAILED){qDebug("video mmap failed\n");return;}// 入队操作if(ioctl(fd, VIDIOC_QBUF, &buf) < 0){qDebug("入队失败\n");return;}}// 开始采集enum v4l2_buf_type type = V4L2_BUF_TYPE_VIDEO_CAPTURE;if (ioctl(fd, VIDIOC_STREAMON, &type) < 0){qDebug("video开始采集失败");m_isRun = false;}//持续读取图像数据 使用select监听数据fd_set fds;struct timeval tv;tv.tv_sec  = 2;tv.tv_usec = 0;//图片文件缓存char *fileBuf = new char[3*1024*1024];while(m_isRun){FD_ZERO(&fds);FD_SET(fd, &fds);int ret = select(fd+1, &fds, NULL, NULL, &tv);if(ret < 0){qDebug("select io error\n");break;}// 读取帧struct v4l2_buffer readbuffer;readbuffer.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;readbuffer.memory = V4L2_MEMORY_MMAP;if(ioctl(fd, VIDIOC_DQBUF, &readbuffer) < 0){qDebug("读取帧失败\n");}//        qDebug() << QDateTime::currentDateTime().toString("hh:mm:ss:zzz");//重点:帧数据处理const uchar *data = (const uchar *)m_userVideoBuf[readbuffer.index];int length = readbuffer.length;//构造并显示QImage image = QImage::fromData(data, length);emit sigUpdateImage(image.copy());// 再次入队if(ioctl(fd, VIDIOC_QBUF, &readbuffer) < 0){qDebug("再次入队失败\n");}}// 停止采集if (ioctl(fd, VIDIOC_STREAMOFF, &type) < 0){qDebug("停止采集失败\n");}// 释放映射for(uint i=0; i<m_frameCount; i++){munmap(m_userVideoBuf[i], m_userVideoBufSize[i]);}//关闭文件emit sigUpdateImage(QImage());close(fd);delete[] fileBuf;qDebug() << "VideoCapture stop";
}

这篇关于Linux基于V4L2的视频捕捉的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1022025

相关文章

ElasticSearch+Kibana通过Docker部署到Linux服务器中操作方法

《ElasticSearch+Kibana通过Docker部署到Linux服务器中操作方法》本文介绍了Elasticsearch的基本概念,包括文档和字段、索引和映射,还详细描述了如何通过Docker... 目录1、ElasticSearch概念2、ElasticSearch、Kibana和IK分词器部署

Linux流媒体服务器部署流程

《Linux流媒体服务器部署流程》文章详细介绍了流媒体服务器的部署步骤,包括更新系统、安装依赖组件、编译安装Nginx和RTMP模块、配置Nginx和FFmpeg,以及测试流媒体服务器的搭建... 目录流媒体服务器部署部署安装1.更新系统2.安装依赖组件3.解压4.编译安装(添加RTMP和openssl模块

linux下多个硬盘划分到同一挂载点问题

《linux下多个硬盘划分到同一挂载点问题》在Linux系统中,将多个硬盘划分到同一挂载点需要通过逻辑卷管理(LVM)来实现,首先,需要将物理存储设备(如硬盘分区)创建为物理卷,然后,将这些物理卷组成... 目录linux下多个硬盘划分到同一挂载点需要明确的几个概念硬盘插上默认的是非lvm总结Linux下多

linux进程D状态的解决思路分享

《linux进程D状态的解决思路分享》在Linux系统中,进程在内核模式下等待I/O完成时会进入不间断睡眠状态(D状态),这种状态下,进程无法通过普通方式被杀死,本文通过实验模拟了这种状态,并分析了如... 目录1. 问题描述2. 问题分析3. 实验模拟3.1 使用losetup创建一个卷作为pv的磁盘3.

Java如何获取视频文件的视频时长

《Java如何获取视频文件的视频时长》文章介绍了如何使用Java获取视频文件的视频时长,包括导入maven依赖和代码案例,同时,也讨论了在运行过程中遇到的SLF4J加载问题,并给出了解决方案... 目录Java获取视频文件的视频时长1、导入maven依赖2、代码案例3、SLF4J: Failed to lo

Python实现多路视频多窗口播放功能

《Python实现多路视频多窗口播放功能》这篇文章主要为大家详细介绍了Python实现多路视频多窗口播放功能的相关知识,文中的示例代码讲解详细,有需要的小伙伴可以跟随小编一起学习一下... 目录一、python实现多路视频播放功能二、代码实现三、打包代码实现总结一、python实现多路视频播放功能服务端开

Python实现视频转换为音频的方法详解

《Python实现视频转换为音频的方法详解》这篇文章主要为大家详细Python如何将视频转换为音频并将音频文件保存到特定文件夹下,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录1. python需求的任务2. Python代码的实现3. 代码修改的位置4. 运行结果5. 注意事项

Linux环境变量&&进程地址空间详解

《Linux环境变量&&进程地址空间详解》本文介绍了Linux环境变量、命令行参数、进程地址空间以及Linux内核进程调度队列的相关知识,环境变量是系统运行环境的参数,命令行参数用于传递给程序的参数,... 目录一、初步认识环境变量1.1常见的环境变量1.2环境变量的基本概念二、命令行参数2.1通过命令编程

Linux之进程状态&&进程优先级详解

《Linux之进程状态&&进程优先级详解》文章介绍了操作系统中进程的状态,包括运行状态、阻塞状态和挂起状态,并详细解释了Linux下进程的具体状态及其管理,此外,文章还讨论了进程的优先级、查看和修改进... 目录一、操作系统的进程状态1.1运行状态1.2阻塞状态1.3挂起二、linux下具体的状态三、进程的

Linux编译器--gcc/g++使用方式

《Linux编译器--gcc/g++使用方式》文章主要介绍了C/C++程序的编译过程,包括预编译、编译、汇编和链接四个阶段,并详细解释了每个阶段的作用和具体操作,同时,还介绍了调试和发布版本的概念... 目录一、预编译指令1.1预处理功能1.2指令1.3问题扩展二、编译(生成汇编)三、汇编(生成二进制机器语