Ubuntu V4l2中摄像头输出Mjpeg数据在Qt中显示

被这个问题纠结了一天,我的山寨zc301摄像头只能输出mjpeg压缩的数据帧,然而网上绝大多数人都是yuv,教程大多是yuv转rgb24,mjpeg转rgb24的几乎没有,反正我是没找到,被逼的走投无路,尝试了下面的方法。

解决方法

如果你已经了解了v4l2处理usb摄像头的步骤,就会知道有一个VIDIOC_DQBUF指令可以获取当前数据帧的缓冲区编号,通过这个编号,我们就可以找到这一帧的起始地址以及长度,观察QPixmap这个类,我们会发现它的其中一个构造函数就是使用内存的起始地址以及长度,事实证明,这是可以正确转换的,目前还不知道原理,难道mjpeg有特殊优待?下面附上与之相关的代码。

QPixmap m_pixmap;
m_pixmap.loadFromData(frame_data, frame_len);
ui->label->setPixmap(m_pixmap);
  • 4
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 4
    评论
以下是一个简单的示例代码,可以在Ubuntu上使用V4L2接口来捕获摄像头图像并保存为JPEG图片。 ```c++ #include <stdio.h> #include <stdlib.h> #include <string.h> #include <fcntl.h> #include <unistd.h> #include <sys/ioctl.h> #include <sys/mman.h> #include <linux/videodev2.h> #include <jpeglib.h> #define VIDEO_DEVICE "/dev/video0" #define IMAGE_WIDTH 640 #define IMAGE_HEIGHT 480 #define IMAGE_QUALITY 80 int main(int argc, char **argv) { int fd = open(VIDEO_DEVICE, O_RDWR); if (fd == -1) { perror("Failed to open video device"); return 1; } struct v4l2_capability cap; if (ioctl(fd, VIDIOC_QUERYCAP, &cap) == -1) { perror("Failed to query video device capabilities"); close(fd); return 1; } struct v4l2_format format; format.type = V4L2_BUF_TYPE_VIDEO_CAPTURE; format.fmt.pix.width = IMAGE_WIDTH; format.fmt.pix.height = IMAGE_HEIGHT; format.fmt.pix.pixelformat = V4L2_PIX_FMT_MJPEG; format.fmt.pix.field = V4L2_FIELD_NONE; if (ioctl(fd, VIDIOC_S_FMT, &format) == -1) { perror("Failed to set video device format"); close(fd); return 1; } struct v4l2_requestbuffers reqbuf; reqbuf.type = V4L2_BUF_TYPE_VIDEO_CAPTURE; reqbuf.memory = V4L2_MEMORY_MMAP; reqbuf.count = 1; if (ioctl(fd, VIDIOC_REQBUFS, &reqbuf) == -1) { perror("Failed to request video device buffers"); close(fd); return 1; } struct v4l2_buffer buffer; buffer.type = V4L2_BUF_TYPE_VIDEO_CAPTURE; buffer.memory = V4L2_MEMORY_MMAP; buffer.index = 0; if (ioctl(fd, VIDIOC_QUERYBUF, &buffer) == -1) { perror("Failed to query video device buffer"); close(fd); return 1; } void *mem = mmap(NULL, buffer.length, PROT_READ | PROT_WRITE, MAP_SHARED, fd, buffer.m.offset); if (mem == MAP_FAILED) { perror("Failed to map video device buffer"); close(fd); return 1; } if (ioctl(fd, VIDIOC_STREAMON, &buffer.type) == -1) { perror("Failed to start video device stream"); munmap(mem, buffer.length); close(fd); return 1; } struct jpeg_compress_struct cinfo; struct jpeg_error_mgr jerr; unsigned char *jpeg_data; unsigned long jpeg_size; cinfo.err = jpeg_std_error(&jerr); jpeg_create_compress(&cinfo); jpeg_mem_dest(&cinfo, &jpeg_data, &jpeg_size); cinfo.image_width = IMAGE_WIDTH; cinfo.image_height = IMAGE_HEIGHT; cinfo.input_components = 3; cinfo.in_color_space = JCS_RGB; jpeg_set_defaults(&cinfo); jpeg_set_quality(&cinfo, IMAGE_QUALITY, true); if (ioctl(fd, VIDIOC_QBUF, &buffer) == -1) { perror("Failed to queue video device buffer"); munmap(mem, buffer.length); close(fd); return 1; } if (ioctl(fd, VIDIOC_DQBUF, &buffer) == -1) { perror("Failed to dequeue video device buffer"); munmap(mem, buffer.length); close(fd); return 1; } jpeg_start_compress(&cinfo, true); JSAMPROW row_pointer[1]; for (int y = 0; y < IMAGE_HEIGHT; y++) { row_pointer[0] = (JSAMPROW)((unsigned char *)mem + y * buffer.bytesused); jpeg_write_scanlines(&cinfo, row_pointer, 1); } jpeg_finish_compress(&cinfo); FILE *fp = fopen("image.jpg", "wb"); if (fp == NULL) { perror("Failed to open image file"); munmap(mem, buffer.length); close(fd); return 1; } fwrite(jpeg_data, jpeg_size, 1, fp); fclose(fp); jpeg_destroy_compress(&cinfo); munmap(mem, buffer.length); close(fd); return 0; } ``` 注意:这个示例代码仅仅是一个基础的例子,实际使用时需要考虑更多的情况,例如错误处理、缓冲区管理、图像分辨率、图像格式、图像质量等。此外,还需要安装libjpeg库,否则会编译错误。
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值