通过v4l2读取摄像头图片

这篇博客介绍了如何在树莓派上通过v4l2库读取摄像头图片,利用libjpeg存储RGB图像,并借助libudev自动查找视频设备。在设置图像格式时,需要注意检查VIDIOC_S_FMT和VIDIOC_G_FMT的设置是否成功,以避免因图片格式过大或其他不支持的问题导致的错误。
摘要由CSDN通过智能技术生成

之前网络上找到的v4l2读取图片的代码基本可以用,但是代码冗余,这个代码在树莓派上测试通过,其它系统上未测试
代码中使用的库:
libjpeg实现rgb图像存储成图片
libudev实现了自动查找视频设备的功能
说明:
VIDIOC_S_FMT设置要读取的格式后要用VIDIOC_G_FMT再获取下,查检是否设置成功了,比如设置的图片太大,fmt.fmt.pix.pixelformat不支持等问题

#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdio.h>
#include <sys/ioctl.h>
#include <stdlib.h>
#include <sys/mman.h>
#include <linux/types.h>
#include <linux/videodev2.h>
#include <setjmp.h>
#include <libudev.h>
#include <stdint.h>
#include "jpeglib.h"



#define JPEG_QUALITY 100

#define  IMAGEWIDTH 352
#define  IMAGEHEIGHT 288

static char dev_path[128];
static int fd;
static struct v4l2_capability cap;
struct v4l2_fmtdesc fmtdesc;
struct v4l2_format fmt, fmtack;
struct v4l2_requestbuffers req;
struct v4l2_buffer buf;
enum v4l2_buf_type type;
unsigned char frame_buffer[IMAGEWIDTH * IMAGEHEIGHT * 3];

struct buffer {
    void *start;
    unsigned int length;
} *buffers;



int device_find()
{
    struct udev *udev;
    struct udev_enumerate *enumerate;
    struct udev_list_entry *devices, *dev_list_entry;
    struct udev_device *dev;

    udev = udev_new();
    if (!udev) {
        printf("Can't create udev\n");
        return -1;
    }

    enumerate = udev_enumerate_new(udev);
    udev_enumerate_add_match_subsystem(enumerate, "video4linux");
    udev_enumerate_scan_devices(enumerate);
    devices = udev_enumerate_get_list_entry(enumerate);

    udev_list_entry_foreach(dev_list_entry, devices) {
        const char *path;
        path = udev_list_entry_get_name(dev_list_entry);
        dev = udev_device_new_from_syspath(udev, path);
        sprintf(dev_path, "%s",  udev_device_get_
  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
要在Linux上使用v4l2读取IMX415摄像头的数据并保存到内存中,你可以按照以下步骤进行: 1. 打开摄像头设备: ```c int fd = open("/dev/video0", O_RDWR); if (fd == -1) { perror("无法打开设备"); return EXIT_FAILURE; } ``` 2. 设置摄像头参数: ```c struct v4l2_format format; format.type = V4L2_BUF_TYPE_VIDEO_CAPTURE; format.fmt.pix.width = <设置摄像头图像宽度>; format.fmt.pix.height = <设置摄像头图像高度>; format.fmt.pix.pixelformat = V4L2_PIX_FMT_YUYV; // 根据摄像头支持的格式进行设置 format.fmt.pix.field = V4L2_FIELD_NONE; if (ioctl(fd, VIDIOC_S_FMT, &format) == -1) { perror("无法设置摄像头参数"); close(fd); return EXIT_FAILURE; } ``` 3. 请求帧缓冲: ```c struct v4l2_requestbuffers req; req.count = <设置帧缓冲个数>; req.type = V4L2_BUF_TYPE_VIDEO_CAPTURE; req.memory = V4L2_MEMORY_MMAP; if (ioctl(fd, VIDIOC_REQBUFS, &req) == -1) { perror("无法请求帧缓冲"); close(fd); return EXIT_FAILURE; } ``` 4. 映射帧缓冲到用户空间: ```c struct v4l2_buffer buf; for (int i = 0; i < req.count; i++) { buf.type = V4L2_BUF_TYPE_VIDEO_CAPTURE; buf.memory = V4L2_MEMORY_MMAP; buf.index = i; if (ioctl(fd, VIDIOC_QUERYBUF, &buf) == -1) { perror("无法查询帧缓冲"); close(fd); return EXIT_FAILURE; } void* addr = mmap(NULL, buf.length, PROT_READ | PROT_WRITE, MAP_SHARED, fd, buf.m.offset); if (addr == MAP_FAILED) { perror("无法映射帧缓冲"); close(fd); return EXIT_FAILURE; } } ``` 5. 入队帧缓冲: ```c for (int i = 0; i < req.count; i++) { buf.type = V4L2_BUF_TYPE_VIDEO_CAPTURE; buf.memory = V4L2_MEMORY_MMAP; buf.index = i; if (ioctl(fd, VIDIOC_QBUF, &buf) == -1) { perror("无法入队帧缓冲"); close(fd); return EXIT_FAILURE; } } ``` 6. 开始视频流: ```c enum v4l2_buf_type type = V4L2_BUF_TYPE_VIDEO_CAPTURE; if (ioctl(fd, VIDIOC_STREAMON, &type) == -1) { perror("无法开始视频流"); close(fd); return EXIT_FAILURE; } ``` 7. 循环读取帧数据并保存到内存: ```c while (<终止条件>) { fd_set fds; FD_ZERO(&fds); FD_SET(fd, &fds); // 设置超时时间 struct timeval timeout; timeout.tv_sec = <设置超时秒数>; timeout.tv_usec = <设置超时微秒数>; // 等待帧缓冲可用 int ret = select(fd + 1, &fds, NULL, NULL, &timeout); if (ret == -1) { perror("select"); break; } else if (ret == 0) { printf("select timeout\n"); continue; } // 出队帧缓冲 if (ioctl(fd, VIDIOC_DQBUF, &buf) == -1) { perror("无法出队帧缓冲"); break; } // 处理帧数据 // 将buf.start指向的数据保存到内存中 // 入队帧缓冲 if (ioctl(fd, VIDIOC_QBUF, &buf) == -1) { perror("无法入队帧缓冲"); break; } } ``` 8. 停止视频流和清理资源: ```c enum v4l2_buf_type type = V4L2_BUF_TYPE_VIDEO_CAPTURE; if (ioctl(fd, VIDIOC_STREAMOFF, &type) == -1) { perror("无法停止视频流"); } for (int i = 0; i < req.count; i++) { munmap(addr[i], buf.length); } close(fd); ``` 请注意,上述代码仅为简单示例,实际使用时可能需要根据自己的需求进行适当的修改和完善。同时,还需了解更多关于v4l2库的使用和摄像头设备的特定参数设置。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值