红联Linux门户
Linux帮助

Ubuntu Framebuffer学习笔记

发布时间:2014-12-19 10:32:50来源:linux网站作者:weimingtom

一、环境搭建

1. 直接在Ubuntu上运行Framebuffer

默认Ubuntu是直接进入X视窗,如果要使用Framebuffer,

需要修改内核引导参数:

$ sudo gedit /etc/default/grub

查找

GRUB_CMDLINE_LINUX_DEFAULT="quiet splash"

把它改为

GRUB_CMDLINE_LINUX_DEFAULT="quiet splash text vga=0x311"

这里text表示进入文本模式,vga=0x311表示使用Framebuffer显示驱动,

0x311是指示色深和分辨率的参数

|640x480 800x600 1024x768 1280x1024

----+-------------------------------------

256 | 0x301   0x303 0x305 0x307

32k | 0x310   0x313 0x316 0x319

64k | 0x311   0x314 0x317 0x31A

16M | 0x312   0x315 0x318 0x31B

如果使用vga=0x311参数,必须使用后面提到的vesafb模块,并且取消黑名单,

否则无法进入系统,需要光盘启动删除vga参数以还原

$ sudo update-grub

写入到/boot/grub/grub.cfg

$ sudo gedit /etc/initramfs-tools/modules

在其中加入:vesafb

$ sudo gedit /etc/modprobe.d/blacklist-framebuffer.conf

用#注释以下行

# blacklist vesafb

$ sudo update-initramfs -u

(生成新的initrd)

然后重启机器,即可进入Framebuffer

如果要切换回X11,可以输入:

$ startx

有时候/boot/grub/grub.cfg的引导参数不正确导致系统无法引导,

可以用光盘引导系统,挂载硬盘后直接修改/boot/grub/grub.cfg文件

这样就可以跳过update-grub这一步。然后还原原有的引导参数进入X Window

2. 使用qemu虚拟Linux

需要编译Linux内核和busybox。

此外还需要libncurses-dev和qemu。

由于qemu可以直接加载内核和initrd,指定引导参数,

所以不需要修改grub配置。

(1)编译内核和安装qemu

$ tar xjf linux-2.6.39.2.tar.bz2

$ cd linux-2.6.39.2/

$ make help

$ make i386_defconf

$ sudo apt-get install libncurses-dev

$ make menuconfig

$ make

$ sudo apt-get install qemu

$ qemu --help

$ qemu -kernel arch/x86/boot/bzImage

$ qemu -kernel arch/x86/boot/bzImage -append "noapic"

有时候内核会这样崩溃:

MP-BIOS BUG 8254 timer not connected

trying to set up timer as Virtual Wire IRQ

所以需要添加-append "noapic"参数

(2) 修改内核配置,然后重新编译内核。

注意,不同内核版本的配置不一样,

我的内核配置作如下改动(用空格切换为*,不要切换为M):

$ make menuconfig

Device Drivers  ---> 

Graphics support  --->  

-*- Support for frame buffer devices  ---> 

[*]   VESA VGA graphics support

因为VESA支持彩色色深的显示。

默认是不选的,只能是黑白控制台。

Input device support  --->

[*]     Provide legacy /dev/psaux device

有些库如SDL在识别USB接口的鼠标时会寻找/dev/input/mice和/dev/psaux,

我发现我编译的内核没有前者,所以用这个选项制造出/dev/psaux设备。

File systems  ---> 

[*] Miscellaneous filesystems  --->

<*>   Compressed ROM file system support (cramfs)

个人喜欢cramfs,不过不是必须的,可以用这个开关编译cramfs驱动,

测试initramfs是否正常

General setup  --->

[*]   Support initial ramdisks compressed using gzip

[*] Embedded system

默认i86内核的配置不支持gzip压缩的cpio格式initrd,所以需要手动打开它。

最后重新编译内核:

$ make

(3) 编译busybox

$ tar xjf busybox-1.18.5.tar.bz2

$ cd busybox-1.18.5/

$ make defconfig

$ make menuconfig

设置修改如下:

Busybox Settings  ---> 

Build Options  --->

[*] Build BusyBox as a static binary (no shared libs) 

$ make

$ make install

默认文件安装在当前目录的_install目录下。

(4) 制作cpio封包gzip压缩的initrd

$ cd ../busybox-1.18.5/_install/

$ mkdir proc sys dev etc etc/init.d tmp root usr lib

$ gedit etc/init.d/rcS

#!/bin/sh

mount -t proc none /proc

mount -t sysfs none /sys

/sbin/mdev -s

$ chmod +x etc/init.d/rcS

$ cd ../../linux-2.6.39.2/

$ gedit prerun.sh

#!/bin/sh

cd ../busybox-1.18.5/_install

find . | cpio -o --format=newc > ../rootfs.img

cd ..

gzip -c rootfs.img > rootfs.img.gz

cd ../linux-2.6.39.2/

$ . prerun.sh

$ gedit run.sh

#!/bin/sh

qemu -kernel ./arch/i386/boot/bzImage -initrd ../busybox-1.18.5/rootfs.img.gz  -append "root=/dev/ram rdinit=/sbin/init vga=0x312 noapic"

注意这里用rdinit=,如果用init=就成了initramfs(内核会报告找不到合适的文件系统)

关于vga=的参数设置见前面(决定色深和分辨率)

$ . run.sh

编译程序,然后用上面写的prerun.sh打包进rootfs.img.gz,然后运行run.sh跑qemu即可。

如果程序是动态链接,需要特定的动态库,

可以把依赖的动态库复制到_install/lib目录下,打包到rootfs.img.gz中。

(5) 可以看进入qemu的效果了


二、Framebuffer的应用开发

1. 基于/lib/fb*设备和mmap

这种方法灵活性差,开发比较费时。

这里有个示例代码,代码如下:

#include <stdlib.h>  
#include <unistd.h>  
#include <stdio.h>  
#include <fcntl.h>  
#include <linux/fb.h>  
#include <sys/mman.h>  
 
int main()  
{  
int fbfd = 0;  
struct fb_var_screeninfo vinfo;  
struct fb_fix_screeninfo finfo;  
long int screensize = 0;  
char *fbp = 0;  
int x = 0, y = 0;  
long int location = 0;  
fbfd = open("/dev/fb0", O_RDWR);  
if (!fbfd) {  
printf("Error: cannot open framebuffer device.\n");  
exit(1);  
}  
printf("The framebuffer device was opened successfully.\n");  
if (ioctl(fbfd, FBIOGET_FSCREENINFO, &finfo)) {  
printf("Error reading fixed information.\n");  
exit(2);  
}  
if (ioctl(fbfd, FBIOGET_VSCREENINFO, &vinfo)) {  
printf("Error reading variable information.\n");  
exit(3);  
}  
printf("%dx%d, %dbpp\n", vinfo.xres, vinfo.yres, vinfo.bits_per_pixel );  
screensize = vinfo.xres * vinfo.yres * vinfo.bits_per_pixel / 8;  
fbp = (char *)mmap(0, screensize, PROT_READ | PROT_WRITE, MAP_SHARED,  
   fbfd, 0);  
if ((int)fbp == -1) {  
printf("Error: failed to map framebuffer device to memory.\n");  
exit(4);  
}  
printf("The framebuffer device was mapped to memory successfully.\n");  
x = 100;
y = 100;  
for ( y = 100; y < 300; y++ )  
for ( x = 100; x < 300; x++ ) {  
location = (x+vinfo.xoffset) * (vinfo.bits_per_pixel/8) +  
   (y+vinfo.yoffset) * finfo.line_length;  
if ( vinfo.bits_per_pixel == 32 ) {  
*(fbp + location) = 100;
*(fbp + location + 1) = 15+(x-100)/2;  
*(fbp + location + 2) = 200-(y-100)/5;
*(fbp + location + 3) = 0;  
} else  {  
int b = 10;  
int g = (x-100)/6;  
int r = 31-(y-100)/16;  
unsigned short int t = r<<11 | g << 5 | b;  
*((unsigned short int*)(fbp + location)) = t;  
}  
}  
munmap(fbp, screensize);  
close(fbfd);  
return 0;  

(已经在Ubuntu和qemu上测试过)

2. 基于DirectFB

DirectFB好像可以加快Framebuffer的绘画速度。

官方有5个示例代码

http://directfb.org/index.php?path=Development%2FTutorials

DirectFB有支持字体、图片和输入插件,

不过在编译前需要事先安装相关的开发库。

(已经在Ubuntu和qemu上测试过)

3. 基于SDL

SDL支持Framebuffer,因为它底层使用了DirectFB。

Ubuntu的SDL可以在Framebuffer下使用,如果自己编译SDL的代码,

需要事先编译DirectFB

(已经在Ubuntu和qemu上测试过)

4. 基于GTK+2

项目主页在:

http://www.gtk.org/

GTK+2通过Cairo库的DirectFB后端支持Framebuffer。

而且gtk库在编译时需要填加特定参数

./configure --prefix=$PREFIX --with-gdktarget=directfb --without-x

然后进行编译。

不过GTK+2的编译比较麻烦,而且directfb上GTK+2的开发代码尚不稳定。

测试的可执行文件在/bin/gtk-demo

(在Ubuntu上跑似乎有问题——不知道怎么拖动窗口和退出;未在qemu上测试)

5. 基于Qt

下载在

ftp://ftp.qt.nokia.com/qt/source/

一般需要自己编译

$ tar xzf qt-everywhere-opensource-src-4.8.0-tp.tar.gz

$ cd qt-everywhere-opensource-src-4.8.0-tp/

$ ./configure -shared -static -opensource -embedded generic

虽然-embedded这个参数没有出现在./configure --help里,

但它在这里是有效的。这里不需要添加--prefix参数

$ sudo make -j 4

$ sudo make install

默认装在/usr/local/Trolltech目录下。

示例代码和编译的二进制文件在demos/embedded目录下。

进入framebuffer下的文本模式,

然后运行sudo ./xxx -qws

运行时如果提示:

Qt/Embedded data directory is not owned by user 0:/tmp/qtembedded-0

可以运行:

sudo chown root:root /tmp/qtembedded-0

如果要在qemu上跑,需要复制/lib/fonts目录下的字体

(可以在Ubuntu上跑,在qemu上跑似乎有点问题——未解决)