在使用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读回来并将屏幕图像保存到文件中。它看起来像这样的乱码:
代码:
#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:~#
答案 0 :(得分:1)
finfo.line_length
给出实际物理扫描线的大小(以字节为单位)。它不一定与屏幕宽度乘以像素大小相同,因为可以填充扫描线。
但是,您使用的QImage构造函数假设没有填充。
如果xoffset为零,则应该可以使用带有 bytesPerLine 参数的构造函数直接从帧缓冲区数据构造QImage。否则有两种选择:
答案 1 :(得分:0)
如果您正在使用DRM,那么/ dev / fb0可能指向完全不同的缓冲区(不是当前可见的缓冲区)或格式不同。
fbdev真的只适用于尚未移植DRM / KMS的旧版本 并且只具有非常有限的modsetting功能。
BTW:您使用的是哪个内核?希望不是那个古老而破碎的TI供应商内核......