树莓派学习专题<11>:使用V4L2驱动获取摄像头数据--启动/停止数据流,数据捕获,缓存释放

树莓派学习专题<11>:使用V4L2驱动获取摄像头数据--启动/停止数据流,数据捕获,缓存释放

  • [1. 启动和停止数据流](#1. 启动和停止数据流)
  • [2. 捕获数据](#2. 捕获数据)
  • [3. 释放缓存](#3. 释放缓存)

1. 启动和停止数据流

使用命令 VIDIOC_STREAMON 启动摄像头数据流,使用命令 VIDIOC_STREAMOFF 停止摄像头数据流。

这两条命令的定义是:

c 复制代码
#define VIDIOC_STREAMON		 _IOW('V', 18, int)
#define VIDIOC_STREAMOFF	 _IOW('V', 19, int)

这两条命令都需要一个 enum v4l2_buf_type 类型的参数作为入参数。参数内容固定填写 V4L2_BUF_TYPE_VIDEO_CAPTURE

代码如下:

c 复制代码
/*******************************************************************************
- Function    : __StartStream
- Description : 本函数启动数据接收。
- Input       : VOID
- Output      : NULL
- Return      : VOID
- Others      :
*******************************************************************************/
void __StartStream(void)
{
    enum v4l2_buf_type type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
    
    if(-1 == ioctl(g_iFDVideo, VIDIOC_STREAMON, &type))
    {
        printf("Start stream failed.\n") ;
        exit(-1) ;
    }

    return ;
}


/*******************************************************************************
- Function    : __StopStream
- Description : 本函数停止数据接收。
- Input       : VOID
- Output      : NULL
- Return      : VOID
- Others      :
*******************************************************************************/
void __StopStream(void)
{
    enum v4l2_buf_type type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
    
    if(-1 == ioctl(g_iFDVideo, VIDIOC_STREAMOFF, &type))
    {
        printf("Stop stream failed.\n") ;
        exit(-1) ;
    }

    return ;
}

2. 捕获数据

代码如下:

c 复制代码
/*******************************************************************************
- Function    : __ReadFrame
- Description : 本函数读取摄像头数据并写入到文件。
- Input       : VOID
- Output      : NULL
- Return      : VOID
- Others      :
*******************************************************************************/
void __ReadFrame(void)
{
    struct v4l2_buffer stBuf;

    memset(&stBuf, 0, sizeof(stBuf)) ;

    stBuf.type   = V4L2_BUF_TYPE_VIDEO_CAPTURE;
    stBuf.memory = V4L2_MEMORY_MMAP;
    
    if(-1 == ioctl(g_iFDVideo, VIDIOC_DQBUF, &stBuf))
    {
        printf("De-queue buffer failed.\n") ;
        exit(-1) ;
    }

#if 1 == WRITE_TO_FILE
    if(-1 == write(g_iFDStore, g_pstBufDesc[stBuf.index].pvBufPtr, stBuf.bytesused))
    {
        printf("Write camera data failed.\n") ;
        exit(-1) ;
    }
#endif
    
    if(-1 == ioctl(g_iFDVideo, VIDIOC_QBUF, &stBuf))
    {
        printf("Re-Queue buffer failed.\n") ;
        exit(-1) ;
    }
    
    return ;
}

/*******************************************************************************
- Function    : __CaptureFrame
- Description : 本函数捕获帧数据。
- Input       : VOID
- Output      : NULL
- Return      : VOID
- Others      :
*******************************************************************************/
void __CaptureFrame(void)
{
    struct timeval stTimeVal ;
    fd_set         stFDRead ;
    int            iRetVal ;
    
    stTimeVal.tv_usec = 0 ;
    stTimeVal.tv_sec  = 2 ; 
    
    FD_ZERO(&stFDRead) ;
    FD_SET(g_iFDVideo, &stFDRead) ;

    iRetVal = select(g_iFDVideo + 1, &stFDRead, NULL, NULL, &stTimeVal) ;
    
    if(-1 == iRetVal)
    {
        perror("select") ;
        exit(-1) ;
    }
    else if(0 == iRetVal)
    {
        printf("timeout.\n") ;
        exit(-1) ;
    }
    
    __ReadFrame() ;

    return ;
}

主函数中,首先调用函数 __StartStream 来启动摄像头输出数据的接收。而后调用函数 __CaptureFrame 来捕获一帧数据。使用 select 函数监视摄像头文件描述符的读操作。一旦监视到,则调用 __ReadFrame 函数来获取数据。

对于V4L2驱动,使用 VIDIOC_DQBUF命令来从缓存队列中取出一块就绪的缓存。该命令的定义如下:

c 复制代码
#define VIDIOC_DQBUF		_IOWR('V', 17, struct v4l2_buffer)

命令需要一个struct v4l2_buffer 类型的参数。

结构体中,index 字段指示了当前是那一块缓冲。bytesused 字段指示了缓冲中有多少有效数据量。对于非编码格式,如YUV或RGB系列格式,应该是整个缓冲都用了。使用index 字段可以索引到对应缓存块在用户空间中的地址,从而获取数据。

从队列中取出缓存块后,可对缓存块中数据进行处理,例如写入文件,或者编码等。操作完毕后,还需要将缓存块重新放入到队列中。该操作使用 VIDIOC_QBUF 命令执行。

c 复制代码
#define VIDIOC_QBUF		_IOWR('V', 15, struct v4l2_buffer)

3. 释放缓存

在使用 VIDIOC_STREAMOFF 命令停止摄像头数据流后,退出程序之前,还需要释放缓存。

c 复制代码
/*******************************************************************************
- Function    : __UnmapBuffer
- Description : 本函数撤除缓冲区映射。
- Input       : VOID
- Output      : NULL
- Return      : VOID
- Others      :
*******************************************************************************/
void __UnmapBuffer(void)
{
    int iLoop ;
    
    for(iLoop = 0 ; iLoop < 4 ; iLoop++) 
    {
        munmap(g_pstBufDesc[iLoop].pvBufPtr, g_pstBufDesc[iLoop].szBufSize) ;
    }
    
    free(g_pstBufDesc) ;

    return ;
}

这里包括解除 mmap 映射关系,以及释放描述符。


相关推荐
心平愈三千疾5 小时前
学习秒杀系统-实现秒杀功能(商品列表,商品详情,基本秒杀功能实现,订单详情)
java·分布式·学习
艾莉丝努力练剑7 小时前
【数据结构与算法】数据结构初阶:详解顺序表和链表(四)——单链表(下)
c语言·开发语言·数据结构·学习·算法·链表
心疼你的一切9 小时前
Unity 多人游戏框架学习系列一
学习·游戏·unity·c#·游戏引擎
Chef_Chen11 小时前
从0开始学习R语言--Day47--Nomogram
学习
毕设源码柳学姐12 小时前
计算机毕业设计Java医学生在线学习平台系统 基于 Java 的医学生在线学习平台设计与开发 Java 医学在线教育学习系统的设计与实现
java·学习·课程设计
永日4567013 小时前
学习日记-spring-day46-7.11
java·学习·spring
Yhame.13 小时前
【 Cache 写策略学习笔记】
笔记·学习
Blossom.11814 小时前
从“炼丹”到“流水线”——如何用Prompt Engineering把LLM微调成本打下来?
人工智能·python·深度学习·神经网络·学习·机器学习·prompt
86Eric14 小时前
C# 入门教程(三):详解字段、属性、索引器及各类参数与扩展方法
学习·c#·传参·扩展方法·属性,字段·输出参数
Xudde.15 小时前
解决了困扰我的upload靶场无法解析phtml等后缀的问题
学习·安全·php