imx6uLL应用-v4l2

Linux V4L2 视频采集 + JPEG 解码 + LCD 显示实践

本文记录一个完整的嵌入式视频处理项目:使用 V4L2 接口从摄像头采集 MJPEG 图像,使用 libjpeg 解码为 RGB 格式,并通过 framebuffer 显示在 LCD 屏幕上。适用于使用 ARM Cortex-A 系列开发板进行嵌入式 Linux 多媒体开发的学习和实践。


开发环境

  • 操作系统:Linux(支持 V4L2 和 framebuffer)
  • 摄像头:支持 MJPEG 输出格式,分辨率 640×480
  • 显示屏:支持 framebuffer 显示,分辨率 800×480,RGB565 格式
  • 编程语言:C
  • 编译依赖:libjpeg 解码库

实现功能

  • 打开摄像头 /dev/video1,设置 MJPEG 格式采集
  • 申请并映射视频缓冲区
  • 解码采集到的 JPEG 数据为 RGB 图像
  • 将 RGB 图像转换为 RGB565 并显示在 LCD(/dev/fb0)上

关键流程

1. 打开 LCD 设备并映射 framebuffer

int lcdfd = open("/dev/fb0", O_RDWR);
lcdptr = (unsigned int *)mmap(NULL, 800*480*4, PROT_READ | PROT_WRITE, MAP_SHARED, lcdfd, 0);

2.打开摄像头设备并设置采集格式

int fd = open("/dev/video1", O_RDWR);
struct v4l2_format v4formt;
v4formt.fmt.pix.pixelformat = V4L2_PIX_FMT_MJPEG;
v4formt.fmt.pix.width = 640;
v4formt.fmt.pix.height = 480;
ioctl(fd, VIDIOC_S_FMT, &v4formt);

3.申请缓冲区并映射到用户空间

struct v4l2_requestbuffers v4rqbuffer;
v4rqbuffer.count = 4;
v4rqbuffer.memory = V4L2_MEMORY_MMAP;
ioctl(fd, VIDIOC_REQBUFS, &v4rqbuffer);

for (int i = 0; i < 4; i++) {
    ioctl(fd, VIDIOC_QUERYBUF, &v4buffer);
    mptr[i] = (unsigned char *)mmap(NULL, v4buffer.length, ...);
    ioctl(fd, VIDIOC_QBUF, &v4buffer); // 放回队列
}

4.启动采集并循环抓图

ioctl(fd, VIDIOC_STREAMON, &type);
while (1) {
    ioctl(fd, VIDIOC_DQBUF, &readbuffer); // 取帧
    read_JPEG_file(mptr[readbuffer.index], rgbdata, readbuffer.length);
    lcd_show_rgb(rgbdata, 640, 480);      // 显示图像
    ioctl(fd, VIDIOC_QBUF, &readbuffer);  // 放回队列
}

图像解码与显示

摄像头输出的是 MJPEG 格式(实质是一帧帧 JPEG 图像),我们使用 libjpeg 将其解码为 RGB888 格式(三通道,每像素 3 字节):

jpeg_mem_src(&cinfo, jpegData, size);        // 将 JPEG 数据源指向内存
jpeg_read_header(&cinfo, TRUE);              // 读取头部信息
jpeg_start_decompress(&cinfo);               // 启动解压
jpeg_read_scanlines(&cinfo, &buffer, 1);     // 逐行读取 RGB 数据

RGB → RGB565 显示

LCD framebuffer 使用的是 RGB565 格式(每像素 2 字节),我们将 RGB888 的三通道数据压缩为 RGB565,并写入 /dev/fb0

unsigned char r = rgbdata[j*3 + 0];
unsigned char g = rgbdata[j*3 + 1];
unsigned char b = rgbdata[j*3 + 2];
unsigned short color = ((r & 0xF8) << 8) | ((g & 0xFC) << 3) | (b >> 3);
ptr[j] = color;

编译方法

需要下载libjpeg源码,然后把一些库文件一直到imx6u里面。交叉编译使用命令

source /opt/fsl-imx-x11/4.1.15-2.1.0/environment-setup-cortexa7hf-neon-poky-linux-gnueabi  
${CC} -o video_show video_show.c  -I/home/zwl/linux/tool/jpeg/include -L/home/zwl/linux/tool/jpeg/lib -ljpeg -Wl,-rpath,/home/zwl/linux/tool/jpeg/lib

运行效果

image-20250504215104659
请添加图片描述
请添加图片描述

使用win系统下的obs打开摄像头,对比发现拍摄画质基本相似。

image-20250504215531986

源码附录

#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdlib.h>
#include <unistd.h>
#include <linux/videodev2.h>
#include <sys/ioctl.h>
#include <string.h>
#include <sys/mman.h>
#include <stdio.h>
#include <jpeglib.h>
#include <linux/fb.h>

int read_JPEG_file (const char *jpegData, char *rgbdata, int size)
{
	struct jpeg_error_mgr jerr;
	struct jpeg_decompress_struct cinfo;
	cinfo.err = jpeg_std_error(&jerr);
	//1创建解码对象并且初始化
	jpeg_create_decompress(&cinfo);
	//2.装备解码的数据
	//jpeg_stdio_src(&cinfo, infile);
	jpeg_mem_src(&cinfo,jpegData, size);
	//3.获取jpeg图片文件的参数
	(void) jpeg_read_header(&cinfo, TRUE);
	/* Step 4: set parameters for decompression */
	//5.开始解码
	(void) jpeg_start_decompress(&cinfo);
	//6.申请存储一行数据的内存空间
	int row_stride = cinfo.output_width * cinfo.output_components;
	unsigned char *buffer = malloc(row_stride);
	int i=0;
	while (cinfo.output_scanline < cinfo.output_height) {
		//printf("****%d\n",i);
		(void) jpeg_read_scanlines(&cinfo, &buffer, 1); 
		memcpy(rgbdata+i*640*3, buffer, row_stride );
		i++;
	}
	//7.解码完成
	(void) jpeg_finish_decompress(&cinfo);
	//8.释放解码对象
	jpeg_destroy_decompress(&cinfo);
	return 1;
}

int lcdfd = 0;
unsigned int *lcdptr = NULL;

void lcd_show_rgb(unsigned char *rgbdata, int w, int h)
{   
    unsigned short *ptr = (unsigned short *)lcdptr;  // 重要!!16位屏幕要用short指针!!
    for (int i = 0; i < h; i++)
    {
        for (int j = 0; j < w; j++)
        {
            unsigned char r = rgbdata[j*3 + 0];
            unsigned char g = rgbdata[j*3 + 1];
            unsigned char b = rgbdata[j*3 + 2];

            unsigned short color = ((r & 0xF8) << 8) | ((g & 0xFC) << 3) | (b >> 3);

            ptr[j] = color;
        }
        ptr += 800;         // 每行跳800列
        rgbdata += w * 3;   // 每行跳 w 个像素 * 3字节
    }
}

int main(void)
{
    lcdfd = open("/dev/fb0", O_RDWR);
    if (lcdfd < 0)
    {
        perror("/dev/fb0打开失败\n");
        return -1;
    }

    lcdptr = (unsigned int *)mmap(NULL, 800*480*4, PROT_READ | PROT_WRITE, MAP_SHARED, lcdfd, 0);
    if(lcdptr < 0)
    {
        perror("lcd内存映射失败\n");
        return -1;
    }


//1.打开设备
    int fd = open("/dev/video1",O_RDWR);
    if (fd < 0){
        perror("video0 打开失败");
        return -1;
    }

//2.获取摄像头支持的格式
    struct v4l2_fmtdesc v4fmtdesc;
    v4fmtdesc.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
    for (int i = 0; i < 3; i++) 
    {
        v4fmtdesc.index = i;
        int ret = ioctl(fd, VIDIOC_ENUM_FMT, &v4fmtdesc);
        if (ret < 0)
        {
            perror("VIDIOC_ENUM_FMT获取结束!");
            break;
        }
        printf("index=%d\n",v4fmtdesc.index);
        printf("flags=%d\n",v4fmtdesc.flags);
        printf("description=%s\n",v4fmtdesc.description);
        unsigned char *p = (unsigned char *)&v4fmtdesc.pixelformat;
        printf("pixelformat=%C%C%C%C\n",p[0],p[1],p[2],p[3]);
        printf("reserved[0]=%d\n",v4fmtdesc.reserved[0]);      
    }
    printf("---------------设置采集格式--------------\n");

//3.设置采集格式
    struct v4l2_format v4formt;
    v4formt.type = V4L2_BUF_TYPE_VIDEO_CAPTURE; //摄像头采集
    v4formt.fmt.pix.width = 640;  //设置宽 不能任意大小
    v4formt.fmt.pix.height = 480; //设置高
    v4formt.fmt.pix.pixelformat = V4L2_PIX_FMT_MJPEG; //设置视频采集格式
    int ret = ioctl(fd, VIDIOC_S_FMT, &v4formt);
    if(ret < 0)
    {
        perror("VIDIOC_S_FMT:设置格式失败");
    }
    //验证
    memset(&v4formt, 0, sizeof(v4formt));  //清空
    v4formt.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
    ret = ioctl(fd, VIDIOC_G_FMT, &v4formt);
    if(ret < 0)
    {
        perror("获取格式失败");
    }
    else
    {
        printf("v4formt.fmt.pix.width = %d\n",v4formt.fmt.pix.width);
        printf("v4formt.fmt.pix.height = %d\n",v4formt.fmt.pix.height);
        unsigned char *p = (unsigned char *)&v4formt.fmt.pix.pixelformat;
        printf("v4formt.fmt.pix.pixelformat = %C%C%C%C\n",p[0],p[1],p[2],p[3]);
        printf("设置成功\n");
    }

    printf("---------------4.申请内核缓冲队列--------------\n");

//4.申请内核缓冲区队列
    struct v4l2_requestbuffers v4rqbuffer;
    v4rqbuffer.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
    v4rqbuffer.count = 4; //申请4个缓冲区
    v4rqbuffer.memory = V4L2_MEMORY_MMAP; //映射方式
    ret = ioctl(fd, VIDIOC_REQBUFS, &v4rqbuffer);
    if (ret < 0)
    {
        perror("申请队列空间失败");
    }
    printf("---------------5.映射队列空间到用户空间--------------\n");
//5.映射队列空间到用户空间
    unsigned char *mptr[4]; //保存映射后空间的首地址  重要!!!
    unsigned int size[4];
    struct v4l2_buffer v4buffer;
    v4buffer.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
    
    for (int i = 0; i < 4; i++)
    {
        v4buffer.index = i;
        ret = ioctl(fd, VIDIOC_QUERYBUF, &v4buffer); //从内核空间中查询一个空间做映射
        if (ret < 0)
        {
            perror("查询内核空间队列失败");
        }
        mptr[i] = (unsigned char *)mmap(NULL,v4buffer.length, PROT_READ | PROT_WRITE, MAP_SHARED, fd, v4buffer.m.offset);
        size[i] = v4buffer.length;

        //通知使用完毕--‘放回去’
        ret = ioctl(fd, VIDIOC_QBUF, &v4buffer);
        if(ret < 0)
        {
            perror("返回失败");
        }
    }

    /*  VIDIOC_STREAMON(开始采集写数据到队列中)
        VIDIOC_DQBUF(告诉内核我要某一个数据,内核不可以修改)
        VIDIOC_QBUF(告诉内核我已经使用完毕)
        VIDIOC_STREAMOFF(停止采集-不在向队列中写数据)*/
    printf("---------------6.开始采集--------------\n");   
//6.开始采集
    int type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
    ret = ioctl(fd, VIDIOC_STREAMON, &type);
    if (ret < 0)
    {
        perror("开启失败");
    }

    printf("---------------7.采集数据  从队列中提取一帧数据--------------\n");
//7.采集数据  从队列中提取一帧数据
    unsigned char rgbdata[640*480*3];  //定义一个空间存储解码后的RGB数据
    struct v4l2_buffer readbuffer;
    readbuffer.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
    while (1)
    {    
        ret = ioctl(fd, VIDIOC_DQBUF, &readbuffer);
        if (ret < 0)
        {
            perror("读取数据失败");
        }
        
        //显示在lcd上
        
        read_JPEG_file(mptr[readbuffer.index], rgbdata, readbuffer.length);//把jpeg数据解码为RGB数据
        lcd_show_rgb(rgbdata, 640, 480);

        //通知内核已经使用完毕
        ret = ioctl(fd, VIDIOC_QBUF, &readbuffer);
        if (ret < 0)
        {
            perror("放回队列失败");
        }
    }
    printf("---------------8.停止采集--------------\n");
//8.停止采集
    ret = ioctl(fd, VIDIOC_STREAMOFF, &type);
    printf("---------------9.释放映射--------------\n");
//9.释放映射
    for (int i = 0; i < 4; i++)
    {
        printf("size[%d]: %d\n",i,size[i]);
        munmap(mptr[i],size[i]);
    }

    printf("---------------10.关闭设备--------------\n");
//10.关闭设备
    close(fd);

    printf("all end\n");
    return 0;
}

请添加图片描述

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值