mplane方式v4l2应用分析 -- VIDIOC_DQBUF(缓存放入队列)

本文介绍了Linuxv4l2架构中的VIDIOC_DQBUF和VIDIOC_QBUFioctl调用,用于处理视频缓冲区的队列操作。内容涉及如何填充v4l2_buffer结构,包括时间戳、标志、缓冲区数据等,并展示了应用层如何使用这些接口进行多媒体数据的捕获和回送。代码示例演示了如何在用户空间中处理多平面缓冲区。
摘要由CSDN通过智能技术生成

Linux v4l2架构学习总链接

gitee源码

VIDIOC_DQBUF

同样的先去看从应用调用vivi驱动分析v4l2 – 缓存放入队列(VIDIOC_QBUF)
那到缓冲区信息的代码可以看__fill_v4l2_buffer

static void __fill_v4l2_buffer(struct vb2_buffer *vb, void *pb) { struct v4l2_buffer *b = pb; struct vb2_v4l2_buffer *vbuf = to_vb2_v4l2_buffer(vb); struct vb2_queue *q = vb->vb2_queue; unsigned int plane;
/* Copy back data such as timestamp, flags, etc. */
b->index = vb->index;
b->type = vb->type;
b->memory = vb->memory;
b->bytesused = 0;

b->flags = vbuf->flags;
b->field = vbuf->field;
b->timestamp = ns_to_timeval(vb->timestamp);
b->timecode = vbuf->timecode;
b->sequence = vbuf->sequence;
b->reserved2 = 0;
b->reserved = 0;

if (q->is_multiplanar) {
	/*
	 * Fill in plane-related data if userspace provided an array
	 * for it. The caller has already verified memory and size.
	 */
	b->length = vb->num_planes;
	for (plane = 0; plane < vb->num_planes; ++plane) {
		struct v4l2_plane *pdst = &b->m.planes[plane];
		struct vb2_plane *psrc = &vb->planes[plane];

		pdst->bytesused = psrc->bytesused;
		pdst->length = psrc->length;
		if (q->memory == VB2_MEMORY_MMAP)
			pdst->m.mem_offset = psrc->m.offset;
		else if (q->memory == VB2_MEMORY_USERPTR)
			pdst->m.userptr = psrc->m.userptr;
		else if (q->memory == VB2_MEMORY_DMABUF)
			pdst->m.fd = psrc->m.fd;
		pdst->data_offset = psrc->data_offset;
		memset(pdst->reserved, 0, sizeof(pdst->reserved));
	}
} else { ... }

这里可以看到,本次ioctl实际上是填充了所有的plane的信息,也就是说当前buffer中的所有plane对应的缓冲区中都填充了framebuff的数据。

编写对应的应用代码

    struct v4l2_plane *tmp_plane;
    tmp_plane = calloc(num_planes, sizeof(*tmp_plane));

    while (1) 
    {   
        FD_ZERO (&fds);
        FD_SET(fd, &fds);
        tv.tv_sec = 5;
        tv.tv_usec = 0;

        r = select (fd + 1, &fds, NULL, NULL, &tv);

        if (-1 == r)
        {
            if (EINTR == errno)
                continue;
            printf ("select err\n");
        }
        if (0 == r)
        {
            fprintf (stderr, "select timeout\n");
            exit (EXIT_FAILURE);
        }

        memset(&buf, 0, sizeof(buf));
        buf.type = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
        buf.memory = V4L2_MEMORY_MMAP;
        buf.m.planes = tmp_plane;
        /* 注意这个length 内核代码中还是会检测的 */
        buf.length = num_planes;
        if (ioctl (fd, VIDIOC_DQBUF, &buf) < 0)
            printf("dqbuf fail\n");
    
        for (j = 0; j < num_planes; j++) {
            printf("plane[%d] start = %p, bytesused = %d\n", j, ((buffers + buf.index)->plane_start + j)->start, (tmp_plane + j)->bytesused);
        }
    
        if (ioctl (fd, VIDIOC_QBUF, &buf) < 0)
            printf("failture VIDIOC_QBUF\n");
    } 

                
 
 
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
针对V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE类型,VIDIOC_REQBUFS和VIDIOC_QUERYBUF需要按照以下步骤进行: 1. VIDIOC_REQBUFS:首先,使用VIDIOC_REQBUFS命令来请求缓冲区,需要设置struct v4l2_requestbuffers结构体中的成员,如下所示: ``` struct v4l2_requestbuffers reqbuf; reqbuf.type = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE; reqbuf.memory = V4L2_MEMORY_MMAP; reqbuf.count = 4; ``` 其中,type成员表明请求的是视频捕获缓冲区,memory成员表明使用内存映射方式,count成员表明请求4个缓冲区。 2. VIDIOC_QUERYBUF:请求缓冲区之后,需要使用VIDIOC_QUERYBUF命令来查询每个缓冲区的信息,需要设置struct v4l2_buffer和struct v4l2_plane结构体中的成员,如下所示: ``` struct v4l2_buffer buf; struct v4l2_plane planes[VIDEO_MAX_PLANES]; buf.type = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE; buf.memory = V4L2_MEMORY_MMAP; buf.index = 0; buf.length = VIDEO_MAX_PLANES; buf.m.planes = planes; for (int i = 0; i < buf.length; i++) { buf.m.planes[i].length = buffer_size; buf.m.planes[i].m.mem_offset = i * buffer_size; buf.m.planes[i].bytesused = 0; } ``` 其中,type成员和memory成员同样表明请求的是视频捕获缓冲区,index成员表明查询第一个缓冲区的信息,length成员表明缓冲区中的平面数,m.planes成员表示缓冲区中的每个平面的信息,包括长度、内存偏移和已使用的字节数。需要注意的是,buffer_size是每个平面的大小,需要根据实际情况设置。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值