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);
    }
}
  • 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、付费专栏及课程。

余额充值