Linux DRM(DRI)无法像以前一样使用FBDEV屏幕刮/ dev / fb0

时间:2016-05-16 16:53:50

标签: linux qt embedded-linux framebuffer drm

在使用FBDEV驱动程序(Raspberry Pi等)的其他Linux机器上,我可以mmap / dev / fb0设备并直接创建一个保存屏幕上的内容的BMP文件。

现在,我正在尝试在TI Sitara AM57XX(Beagleboard X-15)上使用DRM执行相同的操作。过去使用FBDEV的代码如下所示。

这个mmap似乎不再适用于DRM​​。我在Qt平台linuxfb插件中使用了一个非常简单的Qt5应用程序。它可以很好地绘制到/ dev / fb0并在屏幕上正确显示,但是我无法使用内存映射指针从/ dev / fb0读回来并将屏幕图像保存到文件中。它看起来像这样的乱码:

enter image description here

代码:

#ifdef FRAMEBUFFER_CAPTURE

    repaint();
    QCoreApplication::processEvents();

    // Setup framebuffer to desired format
    struct fb_var_screeninfo var;
    struct fb_fix_screeninfo finfo;
    memset(&finfo, 0, sizeof(finfo));
    memset(&var, 0, sizeof(var));
    /* Get variable screen information. Variable screen information
    * gives information like size of the image, bites per pixel,
    * virtual size of the image etc. */
    int fbFd = open("/dev/fb0", O_RDWR);
    int fdRet = ioctl(fbFd, FBIOGET_VSCREENINFO, &var);
    if (fdRet < 0) {
        qDebug() << "Error opening /dev/fb0!";
        close(fbFd);
        return -1;
    }

    if (ioctl(fbFd, FBIOPUT_VSCREENINFO, &var)<0) {
        qDebug() << "Error setting up framebuffer!";
        close(fbFd);
        return -1;
    } else {
        qDebug() << "Success setting up framebuffer!";
    }

    //Get fixed screen information
    if (ioctl(fbFd, FBIOGET_FSCREENINFO, &finfo) < 0) {
        qDebug() << "Error getting fixed screen information!";
        close(fbFd);
        return -1;
    } else {
        qDebug() << "Success getting fixed screen information!";
    }

    //int screensize = var.xres * var.yres * var.bits_per_pixel / 8;
    //int screensize = var.yres_virtual * finfo.line_length;
    //int screensize = finfo.smem_len;
    int screensize = finfo.line_length * var.yres_virtual;
    qDebug() << "Framebuffer size is: " << var.xres << var.yres << var.bits_per_pixel << screensize;
    int linuxFbWidth = var.xres;
    int linuxFbHeight = var.yres;

    int location = (var.xoffset) * (var.bits_per_pixel/8) +
                           (var.yoffset) * finfo.line_length;

    // Perform memory mapping of linux framebuffer
    char* frameBufferMmapPixels = (char *)mmap(0, screensize, PROT_READ | PROT_WRITE, MAP_SHARED, fbFd, 0);
    assert(frameBufferMmapPixels != MAP_FAILED);

    QImage toSave((uchar*)frameBufferMmapPixels,linuxFbWidth,linuxFbHeight,QImage::Format_ARGB32);
    toSave.save("/usr/bin/test.bmp");
    sync();

#endif

以下是代码运行时的输出:

Success setting up framebuffer!
Success getting fixed screen information!
Framebuffer size is:  800 480 32 1966080

这是fbset的输出,显示像素格式:

mode "800x480"
    geometry 800 480 800 480 32
    timings 0 0 0 0 0 0 0
    accel true
    rgba 8/16,8/8,8/0,8/24
endmode

root@am57xx-evm:~#

2 个答案:

答案 0 :(得分:1)

finfo.line_length给出实际物理扫描线的大小(以字节为单位)。它不一定与屏幕宽度乘以像素大小相同,因为可以填充扫描线。

但是,您使用的QImage构造函数假设没有填充。

如果xoffset为零,则应该可以使用带有 bytesPerLine 参数的构造函数直接从帧缓冲区数据构造QImage。否则有两种选择:

  • 分配一个单独的缓冲区并仅将每条扫描线的可见部分复制到其中
  • 从整个缓冲区(包括填充)创建一个图像然后裁剪

答案 1 :(得分:0)

如果您正在使用DRM,那么/ dev / fb0可能指向完全不同的缓冲区(不是当前可见的缓冲区)或格式不同。

fbdev真的只适用于尚未移植DRM / KMS的旧版本 并且只具有非常有限的modsetting功能。

BTW:您使用的是哪个内核?希望不是那个古老而破碎的TI供应商内核......