mplane方式v4l2应用分析 -- VIDIOC_QUERYBUF(查询缓存信息)

Linux v4l2架构学习总链接

gitee源码

VIDIOC_QUERYBUF(查询缓存信息)

同样的还是先看从应用调用vivi驱动分析v4l2 – 查询(VIDIOC_QUERYBUF)缓存信息
vb2_querybuf中,之前没有分析__verify_planes_array

static int __verify_planes_array(struct vb2_buffer *vb, const struct v4l2_buffer *b)
{
	/* 非NPLANE这里直接返回 */
	if (!V4L2_TYPE_IS_MULTIPLANAR(b->type))
		return 0;

	/* Is memory for copying plane information present? */
	/* 注意:
	 * b->m.planes是用户空间传入的,用于保存mplane的信息
	 */
	if (b->m.planes == NULL) {
		dprintk(1, "multi-planar buffer passed but planes array not provided\n");
		return -EINVAL;
	}

	/* 对于length
	 * NO-MPLANE的时候 单平面缓冲区的缓冲区大小(不是其有效载荷,有效值和总大小不一定相等)
	 * MPLANE的时候:多平面缓冲区的平面数组中的元素数(应该就是num_planes的值) 
	 */
	if (b->length < vb->num_planes || b->length > VB2_MAX_PLANES) {
		dprintk(1, "incorrect planes array length, expected %d, got %d\n",
			vb->num_planes, b->length);
		return -EINVAL;
	}

	return 0;
}

所以说,对于MPLANE,这里加了一个判断,接着看__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.
		 */
		/* 对于length,这里直接被赋值 num_planes
		 * 所以说应用传入的时候要大于等于num_planes
		 */
		b->length = vb->num_planes;
		/* 应用代码中要根据planes的数量,合理申请m.planes的空间大小 */
		for (plane = 0; plane < vb->num_planes; ++plane) {
			/* pdst是用户空间传入的 */
			struct v4l2_plane *pdst = &b->m.planes[plane];
			/* psrc是内核空间已经申请的 */
			struct vb2_plane *psrc = &vb->planes[plane];

			pdst->bytesused = psrc->bytesused;
			pdst->length = psrc->length;
			if (q->memory == VB2_MEMORY_MMAP)
				/* 可以看到 offset是放在m.mem_offset中返回用户空间的 */
				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;
			/* 对于data_offset, 没有看到相关的操作 
			 * 对于这个变量,注释如下
			 * @data_offset:	offset in the plane to the start of data; usually 0,
 			 * unless there is a header in front of the data
			 */
			pdst->data_offset = psrc->data_offset;
			memset(pdst->reserved, 0, sizeof(pdst->reserved));
		}
	} else {
		... ...
	}

编写对应的应用代码

	struct v4l2_plane* planes_buffer;
	num_planes = fmt.fmt.pix_mp.num_planes;

	/* planes_buffer 的大小要根据plane地个数*/
	planes_buffer = calloc(num_planes, sizeof(*planes_buffer));
	
	for(i = 0; i < req.count; i++) {
	    memset(&buf, 0, sizeof(buf));
	    memset(planes_buffer, 0, sizeof(*planes_buffer));
	    buf.type = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
	    buf.memory = V4L2_MEMORY_MMAP;
	    buf.m.planes = planes_buffer;
	    /* 网上很多代码,直接将length置为1,这是不对的
	     * 虽然很多sensor plane个数为1
	     */
	    buf.length = num_planes;
	    buf.index = i;
	    if (-1 == ioctl (fd, VIDIOC_QUERYBUF, &buf)) {
	        printf("Querybuf fail\n");
	        goto err;
	    }
	
			/* 打印每个plane的信息 */
			/* 之前也分析过,offset的作用其实就是找到对应的buffer,mplane的话就是找到对应的plane */
	    for(j = 0; j < num_planes; j++) {
	        printf("plane[%d]: length = %d\n", j, (planes_buffer + j)->length);
	        printf("plane[%d]: offset = %d\n", j, (planes_buffer + j)->m.mem_offset);
	    }
	}
针对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
发出的红包

打赏作者

dianlong_lee

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值