gpt4 book ai didi

c++ - 使用 v4l2 捕获相机图像非常慢

转载 作者:塔克拉玛干 更新时间:2023-11-03 00:34:18 26 4
gpt4 key购买 nike

我一直致力于直接使用 v4l2 在 OpenCV 中抓取相机图像。这工作得很好;这样我就可以抓取 YUYV 格式的高分辨率图像(理解帧率会下降)。我无法通过 OpenCV 实现来完成这项工作。从功能上讲,它工作得很好,但性能可能会好得多。由于这是我第一次直接使用 v4l2,所以对我来说还是有点模糊。我一直在对所有相关部分进行计时,发现 v4l2 select 方法花费的时间超过一秒。当我降低时间间隔时,选择方法花费的时间更少,但比出队花费的时间长得多(也是那一秒)。在其他功能中,相机被初始化,因此设置正确的格式等。我知道帧率会很低,没有压缩和高分辨率,但这是极低的。

下面是抓图功能。我跳过了将缓冲区转换为 Mat (YUYV -> RGB) 的代码,因为我认为它现在不相关。

有人知道如何使 v4l2 捕获图像更快吗?也许有些部分我不应该在每次抓帧时执行?

谢谢!

Mat Camera::capture_image() {
Mat returnframe(10, 10, CV_8UC3);
struct v4l2_buffer buf = {0};
buf.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
buf.memory = V4L2_MEMORY_MMAP;
buf.index = 0;
if (-1 == xioctl(fd, VIDIOC_QBUF, &buf)) {
perror("Query Buffer");
return returnframe;
}

if (-1 == xioctl(fd, VIDIOC_STREAMON, &buf.type)) {
perror("Start Capture");
return returnframe;
}

fd_set fds;
FD_ZERO(&fds);
FD_SET(fd, &fds);
struct timeval tv = {0};
tv.tv_sec = 2;
int r = select(fd + 1, &fds, NULL, NULL, &tv);
if (-1 == r) {
perror("Waiting for Frame");
return returnframe;
}

if (-1 == xioctl(fd, VIDIOC_DQBUF, &buf)) {
perror("Retrieving Frame");
return returnframe;
}

//转换为Mat的代码在这里

if (-1 == xioctl(fd, VIDIOC_STREAMOFF, &buf.type)) {
perror("Stop Capture");
return returnframe;
}

//copy Mat and free bigbuffer, to avoid memory leak
Mat returnImg = dispimg.clone();
free(bigbuffer);
return returnImg;
}

最佳答案

似乎对于每一帧你都在调用VIDIOC_STREAMONVIDIOC_STREAMOFF;这增加了很多开销(这几乎就像为每一帧重新启动您的应用程序)

正确的方法是:

  • 打开设备(仅调用一次):在您的捕获 session 开始时(例如程序启动),通过调用 VIDIOC_STREAMON 设置您的视频设备以开始流式传输

  • 捕获帧(多次调用):对于要捕获的每个帧,仅通过调用DQBUF/QBUF (这非常快,因为设备将不断数据到缓冲区队列);您仍然需要调用 select 才能知道新框架何时可用。

  • 关闭设备(仅调用一次):完成后,通过调用 VIDIOC_STREAMOFF

  • 停止流式传输

关于c++ - 使用 v4l2 捕获相机图像非常慢,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/29273508/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com