ioctl VIDIOC_REQBUFS

名称
    VIDIOC_REQBUFS—开启内存映射或用户指针I/O

函数
    int ioctl(int fd, int request, struct v4l2_requestbuffers *argp);

参数
 fd:open()返回的文件描述符
 request:VIDIOC_REQBUFS
 argp

描述
    这个ioctl用于初始化内存映射或者用户指针IO,内存映射缓冲区由设备内存分配而且必须在应用程序地址空间分配之前由ioctl分配。用户空间的缓冲区由用户层自己分配,这个ioctl只是用来转换用户指针的一个驱动模块。
    为分配设备缓冲区的应用程序初始化3个v4l2_requestbuffer结构,用type来区分流或者缓冲区,count是所需buffer数量,memory必须设置为v4l2_MEMORY_MMAP,当ioctl调用一个指针结构去驱动分配buffer数量并把这个数存在count中。当驱动运行的可用内存,请求的number可以更小甚至是0。当驱动正确调用函数请求更多的buffer,number也可以更大,当内存映射i/o不支持ioctl返回EINVAL。
    应用程序可以重复调用VIDIOC_REQBUFS改变buffer数量,但是对于已经在映射的buffer不能成功。count值为0释放所有buffer,当所有DMA结束或退出,即VIDIOC_STEAMOFF。

struct v4l2_requestbuffers
__u32 count       /*请求或得到的buffer数量,这个字段只能在memory设置为V4L2_MEMORY_MMAP后才能生效*/
enum v4l2_buf_type type  /*stream或者buffer的类型,和v4l2_format结构的type字段相同,*/
enum v4l2_memory memory  /*应用程序设置这个字段用来设置V4L2_MEMORY_MMAP 或 V4L2_MEMORY_USERPTR*/
__u32 reserved[2]  /*留出扩展空间,定制buffer类型V4L2_BUF_TYPE_PRIVATE或更高*/

返回值
    成功返回0,出错返回-1且errno设为某特定值:
EBUSY
    该驱动程序支持多种开放和I / O是已经在进行中,或试图重新分配的缓冲区,虽然仍有一个或多个映射。
EINVAL
    缓冲区型(类型字段)或I / O请求方法(内存)不支持

int init_camera_attribute(int fd) { int numBufs; v4l2_std_id id; struct v4l2_format fmt; struct v4l2_requestbuffers req; struct v4l2_buffer buf; //检查当前视频设备支持的标准 ioctl(fd,VIDIOC_QUERYSTD,&id); //设置视频捕获格式 memset(&fmt,0,sizeof(fmt)); fmt.type = V4L2_BUF_TYPE_VIDEO_CAPTURE; fmt.fmt.pix.width = 640; fmt.fmt.pix.height = 480; fmt.fmt.pix.pixelformat = V4L2_PIX_FMT_YUYV; // fmt.fmt.pix.field = V4L2_FIELD_INTERLACED; if(ioctl(fd,VIDIOC_S_FMT,&fmt) == -1){ perror("set VIDIOC_S_FMT is fail"); exit(EXIT_FAILURE); } //分配内存 memset(&req,0,sizeof(req)); req.count = 4; req.type = V4L2_BUF_TYPE_VIDEO_CAPTURE; req.memory = V4L2_MEMORY_MMAP; if(ioctl(fd,VIDIOC_REQBUFS,&req) == -1){ perror("set VIDIOC_REQBUFS is fail"); exit(EXIT_FAILURE); } //获取并记录缓存的物理空间 buffers = calloc(req.count,sizeof(*buffers)); for(numBufs = 0; numBufs < req.count; numBufs ++){ memset(&buf,0,sizeof(buf)); buf.type = V4L2_BUF_TYPE_VIDEO_CAPTURE; buf.memory = V4L2_MEMORY_MMAP; buf.index = numBufs; //读取缓存 if(ioctl(fd,VIDIOC_QUERYBUF,&buf) == -1){ perror("set VIDIOC_REQBUFS is fail"); exit(EXIT_FAILURE); } // 转换成相对地址 buffers[numBufs].length = buf.length; buffers[numBufs].start = mmap(NULL,buf.length,PROT_READ|PROT_WRITE, MAP_SHARED,fd,buf.m.offset); if(buffers[numBufs].start == MAP_FAILED){ perror("mmap is fail"); exit(EXIT_FAILURE); } // 放入缓存队列 if(ioctl(fd,VIDIOC_QBUF,&buf) == -1){ perror("set VIDIOC_QBUF is fail"); exit(EXIT_FAILURE); } } return 0; }
06-12
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值