通过Framebuffer刷屏使得屏幕显示红色

文章的开始,想要申明一下:我的目的是在整个屏幕上显示红色。关于Framebuffer的原理知识我很少涉及,请读者自己查阅相关书籍或者上网查找资料。

开发环境:

物理机(不是虚拟机)Ubuntu 11.04


最近在作一个视频采集显示的东西,要用到Framebuffer,就看了点相关的知识。在这里稍微总结一下。

       Framebuffer对应的文件是/dev/fb0。

       这里我说明两点:

       1.如果在/dev目录下没有fb0文件,那就重新启动试试。一开始我的电脑上也没有fb0文件,我就瞎捣鼓了一下Nvidia X Server Settings,修改了显示器设置参数,又重新启动了电脑就出现fb0文件了,具体原因我不知道(根据我的做事原则-用最简单、最有效、最直接的方法解决问题。我的目的是要显示器显示红色,既然我搞出来fb0文件了,离目的又进了一步,也就没有深究为什么)。

        2.修改下fb0文件的权限,否则会在以后运行程序时出现无法打开文件的错误。

           sudo chmod 666 /dev/fb0


       操作Framebuffer的步骤如下:

       1.打开文件:open函数

       2.映射帧缓冲到应用程序空间:mmap函数

       3.操作映射后的内存空间,在这里就是将红色对应的RGB和透析度的数值写到相关的内存空间内。

       4.释放映射空间和关闭文件描述符。


    程序总共两个:

  

1.第一个程序如下:

很简单的一个程序,就是打印出Framebuffer的一些相关信息,就是fb_var_screeninfo 和fb_fix_screeninfo 两个结构体主要成员的打印。

#include 
#include 
#include 
#include 
#include 
#include 
#include 

int 
main (int argc,char **argv ){
	int fb_fd=-1;
	struct fb_var_screeninfo vinfo;
	struct fb_fix_screeninfo finfo;
	fb_fd = open("/dev/fb0",O_RDWR);
	if (fb_fd < 0){
		printf("Error : Can not open framebuffer device\n");
		exit(EXIT_FAILURE);
	}

	if(ioctl(fb_fd,FBIOGET_FSCREENINFO,&finfo)){
		printf("Error reading fixed information\n");
		exit(EXIT_FAILURE);
	}

	if (ioctl(fb_fd,FBIOGET_VSCREENINFO,&vinfo)){
		printf("Error reading variable information\n");
		exit(3);
	}

	printf("the fixed information is as follow:\n");
	printf("id=%s\n",finfo.id);
	printf("sem_start=%lx\n",finfo.smem_start);
	printf("smem_len=%u\n",finfo.smem_len);
	printf("type=%u\n",finfo.type);
	printf("line_length=%u\n",finfo.line_length);
	printf("mmio_start=%lu\n",finfo.mmio_start);
	printf("mmio_len=%d\n",finfo.mmio_len);

	printf("variable information is as follow:\n");
	printf("The xres is :%u\n",vinfo.xres);
	printf("The yres is :%u\n",vinfo.yres);
	printf("xres_virtual=%u\n",vinfo.xres_virtual);
	printf("yres_virtual=%u\n",vinfo.yres_virtual);
	printf("xoffset=%u\n",vinfo.xoffset);
	printf("yoffset=%u\n",vinfo.yoffset);
	printf("bits_per_pixel is :%u\n",vinfo.bits_per_pixel);
	printf("red.offset=%u\n",vinfo.red.offset);
	printf("red.length=%u\n",vinfo.red.length);
	printf("red.msb_right=%u\n",vinfo.red.msb_right);
	printf("green.offset=%d\n",vinfo.green.offset);
	printf("green.length=%d\n",vinfo.green.length);
	printf("green.msb_right=%d\n",vinfo.green.msb_right);
	printf("blue.offset=%d\n",vinfo.blue.offset);
	printf("blue.length=%d\n",vinfo.blue.length);
	printf("blue.msb_right=%d\n",vinfo.blue.msb_right);
	printf("transp.offset=%d\n",vinfo.transp.offset);
	printf("transp.length=%d\n",vinfo.transp.length);
	printf("transp.msb_right=%d\n",vinfo.transp.msb_right);
	printf("height=%x\n",vinfo.height);
	printf("width=%x\n",vinfo.width);
	
	close (fb_fd);
	return 0;
}

输出结果如下:


the fixed information is as follow:
id=VESA VGA
sem_start=cf000000
smem_len=1245184
type=0
line_length=2560
mmio_start=0
mmio_len=0
variable information is as follow:
The xres is :640
The yres is :480
xres_virtual=640
yres_virtual=480
xoffset=0
yoffset=0
bits_per_pixel is :32
red.offset=16
red.length=8
red.msb_right=0
green.offset=8
green.length=8
green.msb_right=0
blue.offset=0
blue.length=8
blue.msb_right=0
transp.offset=24
transp.length=8
transp.msb_right=0
height=ffffffff
width=ffffffff

有几个输出要重视:

The xres is :640

The yres is :480//虚拟控制台屏幕的分辨率(虚拟控制台,即:Ctrl+Alt+Fn(n=1,2,3,4,5,6)时进入的控制台。我的机器是物理机,虚拟机没有尝试过)


bits_per_pixel is :32//一个像素点占用32位,也就是说明像素点对应的内存中存放的RGB32(R、G、B各占8位,剩下的八位用来存放透析度(不知道是什么东西,也没管它))数据。这里有个问题很重要啊:到底这32位的数据怎么存放的的呢?????最低的八位放R、G、B还是透析度数据呢。。。。。。接着往下看---------


red.offset=16//看来R数据放在32数据中的16-23上

red.length=8

red.msb_right=0//指的是数据的最高有效位在最左边


green.offset=8//看来G数据放在32数据中的8-15上

green.length=8

green.msb_right=0


blue.offset=0//看来B数据放在32数据中的0-7上

blue.length=8

blue.msb_right=0


transp.offset=24//看来透析度数据放在32数据中的24-31上

transp.length=8

transp.msb_right=0


以上内容一定要明白,否则先别往下看,查查资料先把理论搞明白再说。



2.第二个程序如下:

#include 
#include 
#include 
#include 
#include 
#include 
#include 

#define IMAGE_W 640  //对应上一个程序的xres
#define IMAGE_H 480   //对应上一个程序的yres

typedef struct {
	char red;
	char blue;
	char green;
	char trans;
} PIXEL;         //自定义的结构体用于存放一个像素点的颜色数据,这里比较简单就是存放红色的RGB和透析度数据

int 
main(){
	PIXEL pixel;
	pixel.trans=0;
	pixel.red=255;
	pixel.green=0;
	pixel.blue=0;   //查表得到。。。。程序后面有插图



	int fb_fd=0,i=0,j=0;
	unsigned long int *fbp,*mmap_start;
	unsigned long buffersize;
	char * cur_pix;

	struct fb_var_screeninfo vinfo;
	struct fb_fix_screeninfo finfo;
	fb_fd = open("/dev/fb0",O_RDWR);
	if (fb_fd < 0){
		printf("Error : Can not open framebuffer device\n");
		exit(1);
	}

	if(ioctl(fb_fd,FBIOGET_FSCREENINFO,&finfo)){
		printf("Error reading fixed information\n");
		exit(2);
	}

	if (ioctl(fb_fd,FBIOGET_VSCREENINFO,&vinfo)){
		printf("Error reading variable information\n");
		exit(3);
	}

	/*
	printf("The xres is :%d\n",vinfo.xres);
	printf("The yres is :%d\n",vinfo.yres);
	printf("bits_per_pixel is :%d\n",vinfo.bits_per_pixel);
	*/
	buffersize=vinfo.xres*vinfo.yres*vinfo.bits_per_pixel/8;
	mmap_start=(unsigned long int *)mmap(0,buffersize,PROT_READ | PROT_WRITE,MAP_SHARED,fb_fd,0);
	
	if((long int )mmap_start==-1){
		perror("mmap error!\n");
		exit(EXIT_FAILURE);
	}
	fbp=mmap_start;
	for(i=0;i



查表可得红色对应的RGB数据是0xFF0000 。


现在请按下组合键(Ctrl+Alt+Fn(n=1,2,3,4,5,6))进入虚拟控制台,通过cd命令转到存放源程序的目录下,编译后运行,是不是屏幕全变成红色了呢?



你可能感兴趣的:(Linux)