memorycleaner汉化版(v4l2 userptr)

本文链接:https://blog.csdn.net/coroutines/article/details/70141086可参考:http://www.it610.com/article/4522348.htm//v4l2官方翻译基于V4L2的应用,通常面临着大块数据的读取与拷贝等问题。尤其在嵌入式系统中,对于实时性能要求较高的应用,拷贝会花上几十个ms…

大家好,又见面了,我是你们的朋友全栈君。

本文链接:https://blog.csdn.net/coroutines/article/details/70141086

可参考:http://www.it610.com/article/4522348.htm   //v4l2官方翻译

            

基于V4L2的应用,通常面临着大块数据的读取与拷贝等问题。尤其在嵌入式系统中,对于实时性能要求较高的应用,拷贝会花上几十个ms的时间,这通常轻则造成用户体验差,重则导致产品质量不达标。V4L2 Framework定义了几种不同的方式,用于从设备中读取数据,这篇文章简要介绍下在Streaming I/O模式下,如何使用这几种数据的获取与使用方法。Streaming I/O设计的目的就是为了减少在数据处理的各个环节中,拷贝的次数,从而实现各阶段硬件的无缝配合。
本文针对的是USB Camera (Capture)设备。
1. 设备支持

对于设备特性来说,需要设备支持Streaming能力,这个需要通过V4L2的Capability来判断,方式如下:

struct v4l2_capability _cap;
ioctl( _fd, VIDIOC_QUERYCAP, &_cap );

其中,_fd是打开的V4L2设备的描述符,通过:

if( (_cap.capabilities & V4L2_CAP_STREAMING) != 0 ) {

    printf( ” V4L2_CAP_STREAMING” );
}

判断是否支持Streaming方式访问。
2 Memory Map

通过Memory Map访问V4L2设备驱动中分配的内存。设备收到的数据存在驱动内的Buffer中,通过Map方式,将内存Map到用户空间。使用这种方式只有指向这段内存的用户空间指针在各个处理环节中传递,不会发生真实的数据拷贝。

struct v4l2_requestbuffers reqbuf;
struct {

    void *start;
    size_t length;
} *buffers;
unsigned int i;

memset(&reqbuf, 0, sizeof(reqbuf));
reqbuf.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
reqbuf.memory = V4L2_MEMORY_MMAP;
reqbuf.count = 20;

if (-1 == ioctl (fd, VIDIOC_REQBUFS, &reqbuf)) {

    if (errno == EINVAL)
        printf(“Video capturing or mmap-streaming is not supported\\n”);
    else
        perror(“VIDIOC_REQBUFS”);

    exit(EXIT_FAILURE);
}

/* We want at least five buffers. */

if (reqbuf.count < 5) {

    /* You may need to free the buffers here. */
    printf(“Not enough buffer memory\\n”);
    exit(EXIT_FAILURE);
}

buffers = calloc(reqbuf.count, sizeof(*buffers));
assert(buffers != NULL);

for (i = 0; i < reqbuf.count; i++) {

    struct v4l2_buffer buffer;

    memset(&buffer, 0, sizeof(buffer));
    buffer.type = reqbuf.type;
    buffer.memory = V4L2_MEMORY_MMAP;
    buffer.index = i;

    if (-1 == ioctl (fd, VIDIOC_QUERYBUF, &buffer)) {

        perror(“VIDIOC_QUERYBUF”);
        exit(EXIT_FAILURE);
    }

    buffers[i].length = buffer.length; /* remember for munmap() */

    buffers[i].start = mmap(NULL, buffer.length,
                PROT_READ | PROT_WRITE, /* recommended */
                MAP_SHARED,             /* recommended */
                fd, buffer.m.offset);

    if (MAP_FAILED == buffers[i].start) {

        /* If you do not exit here you should unmap() and free()
           the buffers mapped so far. */
        perror(“mmap”);
        exit(EXIT_FAILURE);
    }
}

/* Cleanup. */

for (i = 0; i < reqbuf.count; i++)
    munmap(buffers[i].start, buffers[i].length);

用户程序首先需要通过VIDIOC_REQBUFS,通知驱动进行驱动内内存分配;之后通过VIDIOC_QUERYBUF取得驱动中各内存块的基本信息,主要是Buffer长度和Offset;取得这些信息后,通过mmap,将内存map到User Space中使用。在驱动中,V4L2对于Buffer的使用是队列形式,Buffer出队后,再次入队之前,驱动无法再使用这个Buffer,因此申请的Buffer个数通常是多个,避免数据丢失。
在使用过程中,需要通过poll操作,判断是否有数据到达,然后通过VIDIOC_DQBUF,取得当前有数据的Buffer,通过Buffer的Index属性,找到对应的User Space指针,交由下个环节处理;处理完成后,通过VIDIOC_QBUF,将Buffer重新入队。
3 User Pointers

User Pointers方式会将用户空间分配的内存指针及长度传递给V4L2驱动(虽然是用户空间分配,但不一定是在堆空间上分配的内存,可以是通过其它方式映射出来的内存,比如从另一个设备驱动中),这样数据到达后,可以直接传递这个指针到下个环节中处理。初始化方式如下:

struct v4l2_requestbuffers reqbuf;

memset (&reqbuf, 0, sizeof (reqbuf));
reqbuf.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
reqbuf.memory = V4L2_MEMORY_USERPTR;

if (ioctl (fd, VIDIOC_REQBUFS, &reqbuf) == -1) {

    if (errno == EINVAL)
        printf (“Video capturing or user pointer streaming is not supported\\n”);
    else
        perror (“VIDIOC_REQBUFS”);

    exit (EXIT_FAILURE);
}

4 DMA buffer importing

DMA方式的初始化方式如下:

struct v4l2_requestbuffers reqbuf;

memset(&reqbuf, 0, sizeof (reqbuf));
reqbuf.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
reqbuf.memory = V4L2_MEMORY_DMABUF;
reqbuf.count = 1;

if (ioctl(fd, VIDIOC_REQBUFS, &reqbuf) == -1) {

    if (errno == EINVAL)
        printf(“Video capturing or DMABUF streaming is not supported\\n”);
    else
        perror(“VIDIOC_REQBUFS”);

    exit(EXIT_FAILURE);
}

DMA方式的传输基于文件描述符进行,fd的传递是通过VIDIOC_QBUF中的描述符设置:

int buffer_queue(int v4lfd, int index, int dmafd)
{

    struct v4l2_buffer buf;

    memset(&buf, 0, sizeof buf);
    buf.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
    buf.memory = V4L2_MEMORY_DMABUF;
    buf.index = index;
    buf.m.fd = dmafd;

    if (ioctl(v4lfd, VIDIOC_QBUF, &buf) == -1) {

        perror(“VIDIOC_QBUF”);
        return -1;
    }

    return 0;
}

poll操作返回后,可以通过这个dmafd进行下一步处理。
 

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/128001.html原文链接:https://javaforall.net

(0)
上一篇 2022年4月16日 上午11:00
下一篇 2022年4月16日 上午11:00


相关推荐

  • 阿里千问发布最新旗舰模型Qwen3-Max-Thinking,性能据称超GPT-5.2

    阿里千问发布最新旗舰模型Qwen3-Max-Thinking,性能据称超GPT-5.2

    2026年3月13日
    2
  • 效率狂飙:用十分钟时间,解锁 MCP+Cursor 开发姿势

    效率狂飙:用十分钟时间,解锁 MCP+Cursor 开发姿势

    2026年3月16日
    3
  • 清除浮动的最常用的四种方法,以及优缺点

    清除浮动的最常用的四种方法,以及优缺点为什么要清除浮动 清除浮动主要是为了解决 父元素因为子级元素浮动引起的内部高度为 0 的问题 1 如下 我给父盒子设置一个 boder 内部放两个盒子一个 big 一个 small 未给 big 和 small 设置浮动 则他们会默认撑开父盒子 2 当我给内部两个盒子加上 float 属性的时候顶部深蓝色盒子就会顶上来 然后父盒子因为没设置高度 变成一条线 big 和 small 已经浮动了 nbsp 总结

    2026年3月19日
    2
  • srvctl start_执行命令ls>c

    srvctl start_执行命令ls>c==查看数据库信息srvctlconfigdatabase-dorcl-a==数据库随CRS启动而启动srvctlenabledatabase-dorcl==禁止服务在某个实例上运行srvctldisableservice-dorcl-sservicename-iorcl1srvctlconfigservic

    2025年10月30日
    5
  • MQTT–入门「建议收藏」

    MQTT–入门「建议收藏」一、简述 MQTT(MessageQueuingTelemetryTransport,消息队列遥测传输协议),是一种基于发布/订阅(publish/subscribe)模式的“轻量级”通讯协议,该协议构建于TCP/IP协议上,由IBM在1999年发布。MQTT最大优点在于,可以以极少的代码和有限的带宽,为连接远程设备提供实时可靠的消息服务。作为一种低开销、低带宽占用的即时通讯协议,使其在物联网

    2022年5月11日
    46
  • 内存分析利器purify简介

    内存分析利器purify简介1 内存问题的原因及分类在 C C 程序中 有关内存使用的问题是最难发现和解决的 这些问题可能导致程序莫名其妙地停止 崩溃 或者不断消耗内存直至资源耗尽 由于 C C 语言本身的特质和历史原因 程序员使用内存需要注意的事项较多 而且语言本身也不提供类似 Java 的垃圾清理机制 编程人员使用一定的工具来查找和调试内存相关问题是十分必要的 总的说来 与内存有关的问题可以分成两类 内存访问错误和内

    2026年3月17日
    2

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号