V4L2在VIDIOC_DQBUF处阻塞以及视频花屏残影问题

V4L2在VIDIOC_DQBUF处阻塞以及视频花屏残影问题

V4L2在VIDIOC_DQBUF处阻塞以及视频花屏残影问题

1.问题一:使用默认分辨率(640*480)打开视频出现花屏并阻塞问题:

程序在打开副头时出现花屏,同时在ioctl(fd, VIDIOC_DQBUF, &buffer)处阻塞住导致UI界面没有反应,调试时发现副头打开正常,可读取一帧数据后显示出来的却是不正常的图像,之后再次读取一帧数据时便会阻塞住,本来想使用非阻塞模式来open设备,后来想起来非阻塞模式下回出现大量的"Resource temporarily unavailable"警告,因此还是考虑在阻塞模式下解决;使用cheese去打开视频,发现它就能正常的显示,而为什么我的程序里打开其他设备没有问题,偏偏只有这一个副头出现这种情况呢?我猜测应该不是设备的问题,还是程序的问题,经过尝试,发现设置分辨率为160120时程序即可正常运行,之后切换到640480也没问题,此时回来想想原因,总觉得是缓存帧的大小问题,不过不清楚具体的原因,只能先在打开视频前设置一下分辨率解决眼下的问题再说,后续有时间再看看吧…

2.问题二:MJPG格式下视频出现残影、抖动问题

由于设备具有不同的视频格式,比如会同时存在YUYV和MJPG格式,此时需要对不同的格式做不同的处理,在我这里YUYV原始数据需要手动转RGB888,之后将转换后的一帧图像数据传递给QImage去加载,而MJPG压缩数据最初我是直接将这一帧数据起始地址传递给QImage,使用QImage::loadFromData去加载这一帧数据,之前这两种格式也没出问题。后来换了个设备后,在MJPG格式下,画面出现不停的抖动、显示残像,看起来跟抽风了一样,报Corrupt JPEG data: premature end of data segment警告,很难受。静下来想想,考虑到可能是直接使用缓存区那一帧数据的地址去加载图像数据的问题,因此在中间进行一次转换,先使用memcpy将那一帧数据拷贝到,在使用QImage去加载拷贝的数据,果然正常了 。这个原因应该是用户程序直接从映射出来的缓存区取数据造成的,取到的数据可能不完整因此才有JPEG那里的警告。

这里的

静下来想想,考虑到可能是直接使用缓存区那一帧数据的地址去加载图像数据的问题,因此在中间进行一次转换,先使用memcpy将那一帧数据拷贝到,在使用QImage去加载拷贝的数据,果然正常了 。

提示很有作用。

reference

虾米小小小 V4L2在VIDIOC_DQBUF处阻塞以及视频花屏残影问题

  • 1
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
v4l2 VIDIOC_EXPBUF是用于向驱动程序请求分配视频缓冲区的ioctl命令。它用于在视频捕获设备上分配一个或多个视频缓冲区,以便在视频流中存储捕获的帧。 以下是使用v4l2 VIDIOC_EXPBUF的示例代码[^1]: ```c struct v4l2_requestbuffers reqbuf; memset(&reqbuf, 0, sizeof(reqbuf)); reqbuf.count = 1; reqbuf.type = V4L2_BUF_TYPE_VIDEO_CAPTURE; reqbuf.memory = V4L2_MEMORY_MMAP; if (ioctl(fd, VIDIOC_REQBUFS, &reqbuf) == -1) { perror("VIDIOC_REQBUFS"); exit(EXIT_FAILURE); } struct v4l2_buffer buffer; memset(&buffer, 0, sizeof(buffer)); buffer.type = V4L2_BUF_TYPE_VIDEO_CAPTURE; buffer.memory = V4L2_MEMORY_MMAP; buffer.index = 0; if (ioctl(fd, VIDIOC_QUERYBUF, &buffer) == -1) { perror("VIDIOC_QUERYBUF"); exit(EXIT_FAILURE); } buffer.length = buffer.bytesused; buffer.start = mmap(NULL, buffer.length, PROT_READ | PROT_WRITE, MAP_SHARED, fd, buffer.m.offset); if (buffer.start == MAP_FAILED) { perror("mmap"); exit(EXIT_FAILURE); } if (ioctl(fd, VIDIOC_QBUF, &buffer) == -1) { perror("VIDIOC_QBUF"); exit(EXIT_FAILURE); } enum v4l2_buf_type type = V4L2_BUF_TYPE_VIDEO_CAPTURE; if (ioctl(fd, VIDIOC_STREAMON, &type) == -1) { perror("VIDIOC_STREAMON"); exit(EXIT_FAILURE); } ``` 在上述代码中,我们首先使用VIDIOC_REQBUFS命令请求分配一个视频缓冲区。然后,我们使用VIDIOC_QUERYBUF命令查询缓冲区的信息,并使用mmap函数将缓冲区映射到用户空间。接下来,我们使用VIDIOC_QBUF命令将缓冲区放入队列中。最后,我们使用VIDIOC_STREAMON命令启动视频流。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值