Lcd帧缓冲设备

news/2024/7/8 5:09:56 标签: struct, linux, 嵌入式, 图形, 工作, 编程

帧缓冲(framebuffer)是Linux为显示设备提供的一个接口,是把显存抽象后的一种设备,它允许上层应用程序在图形模式下直接对显示缓冲区进行读写操作,这种操作是抽象的,统一的。用户不必关心物理显存的位置,换页机制等等具体细节。这些都由Framebuffer设备驱动程序完成的。帧缓冲驱动应用广泛,在linux的桌面系统中,Xwindow服务器就是利用帧缓冲进行窗口绘制的。尤其是通过帧缓冲设备可显示汉字点阵,成为linux汉化的唯一可行方案。
    帧缓冲设备对应的设备文件为/dev/fb*,如果系统有多个显卡,Linux下还可支持多个帧缓冲设备,最多可达32个,分别为/dev/fb0~/dev/fb31,而/dev/fb则为当前缺省的帧缓冲设备,通常指向/dev/fb0。当然在嵌入式系统中支持一个显示设备就够了。帧缓冲设备为标准字符设备,主设备号为29,次设备号为0到31,分别对应/dev/fb0~/dev/fb31

通过/dev/fb,应用程序的操作主要有一下几种:
    1.读/写(read/write)/dev/fb
      相当于读/写屏幕缓冲区。
      cp /dev/fb0 tmp 将当前屏幕的内容拷贝到一个文件中
      cp tmp > /dev/fb0 则将图形文件tmp显示在屏幕上
    2.映射(map)操作
      由于Linux工作在保护模式,每个应用程序都有自己的虚拟地址空间,在应用程序不能直接访问物理缓冲区地址的。为此,linux在文件操作file_operations结构中提供了mmap函数,可将文件的内容应身到用户空间。对于帧缓冲设备,则可通过映射操作,可将屏幕缓冲区的物理地址映射到用户空间的一段虚拟地址中,之后用户就可以通过读写这段虚拟地址访问屏幕缓冲区,在屏幕上绘图了。而且若干个进程可以映射到同一显示缓冲区。实际上,使用帧缓冲设备的应用程序都是通过映射操作来显示图形的。
    3.I/O控制
     对于帧缓冲设备,对设备文件的ioctl操作可读取/设置显示设备及屏幕参数,如分辨率,显示颜色数,屏幕大小等等。ioctl的操作是有底层的驱动程序来完成的。

在应用程序中,操作/dev/fb的一般步骤如下:
    1.打开/dev/fb设备文件
    2.用ioctrl操作取得当前显示屏幕参数,根据屏幕参数可计算屏幕缓冲区的大小。
    3.将屏幕缓冲区映射到用户空间。
    4.映射后就可以直接读写屏幕缓冲区,进行绘图和图片显示了

备注:memset:作用是在一段内存块中填充某个给定的值,它对较大的结构体或数组进行清零操作的一种最快方法。
     memcpy:作用是把一块内存中的字节,不管其中的内容是什么,从内存的一个区域复制到另一个区域。

  12月9日晚22:40于实验室

____________________________________________________________________

典型程序段分析:

   #include
int main(){
 int fbfd = 0;
        struct fb_var_screeninfo vimnfo; 
        struct fb_var_screeninfo finfo;
        long int screensize = 0; 
      
        /*open the device*/ 
        fbfd = open ("/dev/fb0",O_RDWR);
 
        ioctl(fbfd,FBIOGET_FSCREENINFO,&finfo);
        ioctl(fbfd,FBIOGET_FSCREENINFO,&vimnfo);
       
        screensize = vimnfo.xres * vimnifo.yres * vimnfo.bit_per_pixel /8 ;
       
        fbp=(char*)mmap(0,screensize,PROT_READ|PROT_WRITE,MAP_SHARED,fbfd,0);
......
}
帧缓冲驱动的编写

    帧缓冲设备属于字符设备,其目的就是通过配置MX1寄存器,在一段制定的内存与LCD之间建立一个自动传输的通道。这样,任何程序只要修改这段内存中的数 据,就可以改变LCD上的显示内容。帧缓冲设备驱动也采用“文件层-驱动层”的接口方式。在文件层次上,Linux为其定义了:

     static struct file_operation fb_fops={

         owner:THIS_MODULE
         read:fb_read, 
/*读操作*/
         write:fb_write, 
/*写操作*/
         ioctl:fb_ioctl, 
/*控制操作*/
         mmap:fb_mmap, 
/*映射操作*/
         open:fb_open, 
/*打开操作*/
         release:fb_release, /*关闭操作*/

  }

    其中的成员函数都在文件linux/dev/vimdeo/fbmem.c中定义。

    由于显示设备的特殊性,在驱动层的接口中不但要定义底层函数,还要有一些纪录设备状态的函数。Linux为帧缓冲设备定义的驱动测借口为struct fb_info结构, 在include/linux/fb.h中定义。

                                 12月10日上午于实验室

____________________________________________________________________

 
对驱动程序中要用到的几个关键成员的说明
     fb_info:纪录了帧缓冲设备的全部信息,包括设备的参数,状态以及操作函数指针。每一个帧缓冲设备都必须对应一个fb_info结构。 其中成员变量modename为设备名称,fontname为显示字体,fboops为指向底层操作的函数的指针,这些函数是需要驱动程序人员编写的。成 员fb_var_screeninfo和fb_var_screeninfo也是结构体。其中var_screeninfo纪录用户可修改的显示控制器参 数,包括屏幕分辨率和每个象素的比特数。fb_var_screeninfo中的xres定义屏幕一行有多少个点,yres定义屏幕一列有多少个 点,bits_per_pixel定义每个点用多少个字节表示。而fb_fix_screeninfo中纪录用户不能修改的显示控制器的参数,如屏幕缓冲 区的物理地址,长度。当对缓冲区设备进行映射操作时,就是从fb_fix_screeninfo中取得缓冲区相应的物理地址的。上面所说的数据成员都是需 要在驱动程序中初始化和设置的。
     
编写帧缓冲驱动程序的步骤为:
     1.编写初始化函数:初始化函数首先初始化LCD控制器,通过写寄存器设置的显示模式和显示颜色数,然后分配LCD显示缓冲区。在Linux可通过kmalloc函数分配一片连续空间。
      例如LCD显示方式为320*240,16位彩色,需要分配的显示缓冲区位320*240*2=150K字节,缓冲区通常分配在大容量的片外SDRAM中,起始地址保存在LCD控制寄存器中。
      最后是初始化一个fb_info结构,填充其中的成员变量,并调用register_framebuffer(&fb_info),将fb_info登记入内核。
      2.编写结构fb_info中函数指针fb_ops对应的成员函数:对于嵌入式系统的简单实现,只需要下列三个函数就可以了:
       struct fb_ops{
       ......
       int(*fb_get_fix)(struct fb_fix_screeninfo *fix,int con,struct fb_info *info);  
       int(*fb_get_var)(struct fb_var_screeninfo *fix,int con,struct fb_info *info);   
       int(*fb_set_fix)(struct fb_fix_screeninfo *fix,int con,struct fb_info *info); 
       ......   
}
       struct fb_ops在include/linux/fb.h中定义。这些函数都是用来设置/获取fb_info结构中的成员变量的。当应用程序对设备文件进行 Ioctl操作时会调用它们,例如,对于fb_get_fix(),应用程序传入的是fb_fix_screeninfo结构,在函数中对其成员变量赋 值,主要是smem_start(缓冲区其实地址)和smem_len(缓冲区长度),最终返回给应用程序。而fb_set_var函数的传入参数是 fb_var_screeninfo,函数中,需要对xres,yres和bits_per_pixel赋值。驱动程序编写完成后,开发者可以选择将其编 译位动态加载模块,或者静态地编译入内核中。
 
LCD API
     LCD API是一套应用层编程入口,它将屏幕缓冲区的物理地址映射到用户空间的一段虚拟地址中,之后用户就可以通过这段虚拟地址访问屏幕缓冲区,在屏幕上绘图了。
     lcd_initalize完成屏幕缓冲区的物理地址映射到用户空间的一段虚拟地址中,以后用户就可以通过读写这段虚拟地址访问屏幕缓冲区,在屏幕上绘图了。
     lcd_putpixel是画图的基础,直线,矩形,椭圆,圆都是在此继承上的算法。
     lcd_textout在使用这个函数显示字符的时候,需要将/usr/local/src/example//bin/hz16字符库文件。
      
 
LCD显示原理:
     LCD是基于液晶光电效应的显示器件。液晶显示器的工作原理就是利用液晶的物理特性,在通电时,液晶排列变得有次序,使光线容易通过;不通电时排列变得混 乱,阻止光线的通过。即液晶工作时,使用的是外部的光线,自己本身并不发光,所以与CRT相比,液晶显示器的耗电量较低。
    LCD中使用的位液晶照明的主要有两种:传送式和反射式。传送式屏幕要使用外加光源照明,成为背光(backlight),照明光源安装在LCD背后。传送式LCD在正常光线及暗光线下,显示效果很好,但在户外,尤其在日光下,很难辨清显示内容。
    反射式屏幕,则不需要外加照明电源,使用周围环境的光线,这样,反射式屏幕就没有背光,所以这种屏幕在户外或光线充足的室内,才会有出色的显示效果,但在一般室内光线下,这种显示屏的显示效果不及背光传送式的。
 
灰度显示原理:
    实际上,LCD显示屏并不是设置成不同的亮度去驱动每一个象素的,它对象素要么显示,要么关闭。LCD显示屏的一个常用指标就是它的反应时间,反应时间指 得是一个象素从显示到关闭所花费的时间,典型的是几百ms。一种调试技术被用于驱动每个象素,即用整个固定时间周期的一部分驱动每个象素。
     LCD控制器内部有1个16周期计数器,用于产生16周期的间隔。当驱动象素时,它读帧缓冲数据所指的,在调色寄存器中的半字节数据。该数据确定位16周 期间隔里象素显示的次数。例如该值等于4,则该象素每个4个时钟周期显示1次,等于整个16周期间隔的4/16。即认为该象素以最大亮度的1/4进行显 示。
 
彩色显示原理:
    彩色显示时,每个象素点有3个子色彩象素(RGB),灰度显示的技术应用到彩色显示中,每个子彩色象素有15个浓度的感觉效果。可用RGB三种颜色的15 中浓度中的一种去驱动1个象素点。如LCD的控制器编程为4bpp模式,可支持最高15*15*15=3 375种不同的颜色;2bpp模式时,可支持64种不同的颜色;1bpp模式时,可支持8种不同的颜色;对于一个1/4VGA显示屏,实际象素数量等于 320*400*4=921600位或115200字节,小于上面提到的最大帧缓冲限制(128K)
    另外一个显示特性是刷新率,指的是整个数据帧被重新写到显示屏的频率。如果数据写的太慢,将影响到显示质量;太快则显示器的反应时间跟不上象素驱动状态的改变。大部分显示屏推荐的速率是70~80Hz

来源: http://blog.csdn.net/guyun_shine/article/details/6019621

http://www.niftyadmin.cn/n/1736426.html

相关文章

Frame Buffer

Linux的帧缓冲(Frame Buffer)之一:原理及控制参数 大家都知道Unix/Linux系统是由命令驱动的。X-Window-System是Unix/Linux上的图形系统,它是通过X-Server来控制硬件的。但有一些Linux的发行版在…

framebuffer 简介

FrameBuffer 是出现在 2.2.xx 内核当中的一种驱动程序接口。Linux 工作在保护模式下,所以用户态进程是无法象 DOS 那样使用显卡 BIOS 里提供的中断调用来实现直接写屏,Linux 抽象出 FrameBuffer 这个设备来供用户态进程实现直接写屏。Framebuffer 机制模…

对FrameBuffer的简单解释和用法示例

大家都知道Unix/Linux系统是由命令驱动的。那么最基本的系统是命令行的(就是想DOS一样的界面)。X-Window-System是Unix/Linux上的图形系统,它是通过X-Server来控制硬件的。但有一些Linux的发行版在引导的时…

mmap与内存管理

mmap与内存管理- mmap系统调用的编程实例 收藏 1 mmap的使用 mmap地址映射是Linux系统提供的一种功能强大的系统调用&#xff0c;最典型的应用是用于显卡内存的映射。同样&#xff0c;对于普通的硬盘文件也可以进行mmap系统调用。 #include <sys/mman.h> void * m…

linux spi驱动分析

关于spi的学习&#xff0c;我觉得最好的方法还是看Linux的源代码&#xff0c;主要是driver/spi/spi.c(h)&#xff0c;spidev.c(h)。spi dev的示例可以看看at25.c&#xff0c;spi总线的示例可以看omap_uwire或者spi_s3c24xx.c和spi_s3c24xx_gpio.c。在看这些代码之前&#xff0c…

小议大小端模式对C语言的共用体结构的影响

小议大小端模式对C语言的共用体结构的影响 1、一些问题 问题1 view plaincopy to clipboardprint?#include "stdio.h" union { int i; char ch[2]; }key; main() { key.i65*25666; printf("%c\t%c\n",key.ch[0],key.ch[1]); …

Linux下SPI驱动分析

Linux下SPI总线驱动有通用接口&#xff0c;一般的SPI设备驱动使用这个驱动接口实现设备驱动。分析驱动最好是先了解核心代码&#xff0c;然后从具体设备分析入手&#xff0c;然后从下至上&#xff0c;了解整个框架&#xff0c;再从上到下分析&#xff0c;理解透彻。 以下分析内…

Linux驱动修炼之道-SPI驱动框架源码分析(上)

努力成为linux kernel hacker的人李万鹏原创作品&#xff0c;为梦而战。转载请标明出处 http://blog.csdn.net/woshixingaaa/archive/2011/06/29/6574215.aspx Linux驱动修炼之道-SPI驱动框架源码分析(中) Linux驱动修炼之道-SPI驱动框架源码分析(下) SPI协议是一种同步的串…