Lover雪儿
想念时,就看看天空,无论距离有多远,我们总在同一片天空下!

LCD液晶屏驱动程序

2015年1月28日 晚 20:00

一.帧缓冲FrameBuffxer

帧缓冲FrameBuffer,从本质上讲是图形设备的硬件抽象,是一块显示缓存,往显示缓存中写入特定格式的数据就意味着向屏幕输出内容,通过不断的向frame buffer中写入数据,显示控制器就自动从frame buffer中取数据并显示出来。

    一般来说是 /dev/fb0. 主设备号为29,次设备号0-31

 

二.实例演示

(1):清除LCD

        dd if=/dev/zero of=/dev/fb0 bs=240 count=320

(2):运行应用程序,画图

        ./LCD

(3):清除LCD

        dd if=/dev/zero of=/dev/fb0 bs=240 count=320

(4):显示图片

        cat 7.bmp > /dev/fb0

 

三.LCD驱动程序设计

 

1.设备描述:

    struct fb_info{    //描述帧缓冲设备

        struct fb_var_screeninfo var;        /*可变参数*/

        struct fb_fix_screeninfo fix;        /*固定参数*/

        struct fb_ops *fbops;                /*帧缓冲操作*/

        ……        };

2. struct fb_var_screeninfo记录了用户可以修改的参数显示,包括分辨率等

3. struct fb_fix_screeninfo 记录用户不能修改的显示控制器参数,如显存物理地址

4. struct fb_ops 包含了对控制器进行操作的函数指针

    例如 fb_open,fb_read,fb_write,等函数指针,如果用户没实现该函数,则内核会自动实现

5.设备注册,注册一个帧缓冲设备:

    int register_framebuffer(struct fb_info *fb_info);

同样:注销一个帧缓冲设备

    int unregister_framebuffer(struct fb_info *fb_info);

 

 

 

 

6.驱动程序分析(平台设备类型)

(1): 初始化

static struct platform_driver fb_driver = {

    .probe = fb_probe,

    .remove = fb_remove,

    .suspend=fb_suspend,

    .driver = {

        .name = 'LCD',

        .owner=THIS_MODULE,

},

}

int __init fb_init(void){

    int ret = platform_driver_register(fb_driver);    //平台设备以平台总线的方式来分类, 平台总线是 将根据 设备的名字 和 驱动程序的名字来匹配,匹配后就会调用fb_probe这个函数

    return ret;

}

 

stativ char driver_name[]="fb";

static int __init fb_probe(struct platform_device *pdev,enum s3c_drv_type drv_type)

{

    mach_info = pdev->dev.platform_data;

    irq = platform_get_irq(pdev,0);        //从平台设备中获取中断号

    fbinfo = framebuffer_alloc(sizeof(struct fb_info),&pdev->dev);

//分配一个fb_info帧结构

    res = platform_get_resource(pdev,IORESOURCE_MEM,0); //获取平台资源

            //两种资源:中断,地址(IO)资源

    info->mem = request_mem_region(res->start,size,pdev->name);

            //申请上面获取的res 地址,此处为物理地址

    info->io = ioremap(res->start,size);    

//将上面的物理地址转化为虚拟地址    

 

    ret = request_irq(irq,fb_irq,IRQF_DISABLED,pdev->name,info);

            //注册中断处理程序

    ret = register_framebuffer(fbinfo);

            //注册帧缓冲设备 framebuffer

}

 

7.内核配置

 

Graphics support ----> Support for frame buffer devices ----> LCD framebuffer support

 

 

 

三.LCD驱动程序设计

#include <unistd.h>

#include <stdio.h>

#include <fcntl.h>

#include <linux/fb.h>

#include <sys/mman.h>

 

int main () {

int fp=0;

struct fb_var_screeninfo vinfo;

struct fb_fix_screeninfo finfo;

long screensize=0;

char *fbp = 0;

int x = 0, y = 0;

long location = 0;

fp = open ("/dev/fb0",O_RDWR);

 

if (fp < 0){

printf("Error : Can not open framebuffer device\n");

exit(1);

}

 

if (ioctl(fp,FBIOGET_FSCREENINFO,&finfo)){

printf("Error reading fixed information\n");

exit(2);

}

if (ioctl(fp,FBIOGET_VSCREENINFO,&vinfo)){

printf("Error reading variable information\n");

exit(3);

}

 

screensize = vinfo.xres * vinfo.yres * vinfo.bits_per_pixel / 8; //单帧画面空间

/*这就是把fp所指的文件中从开始到screensize大小的内容给映射出来,得到一个指向这块空间的指针*/

fbp =(char *) mmap (0, screensize, PROT_READ | PROT_WRITE, MAP_SHARED, fp,0);

if ((int) fbp == -1)

{

printf ("Error: failed to map framebuffer device to memory.\n");

exit (4);

}

/*这是你想画的点的位置坐标,(0,0)点在屏幕左上角*/

for(x=100;x<150;x++)

{

for(y=100;y<150;y++)

{

location = x * (vinfo.bits_per_pixel / 8) + y * finfo.line_length;

 

*(fbp + location) = 255; /* 蓝色的色深 */

/*直接赋值来改变屏幕上某点的颜色*/

*(fbp + location + 1) = 0;/* 绿色的色深*/

/*注明:这几个赋值是针对每像素四字节来设置的,如果针对每像素2字节,*/

*(fbp + location + 2) = 0; /* 红色的色深*/

/*比如RGB565,则需要进行转化*/

*(fbp + location + 3) = 0; /* 是否透明*/

}

}

munmap (fbp, screensize); /*解除映射*/

close (fp); /*关闭文件*/

return 0;

 

}

 

 

 

 

posted on 2015-02-05 12:26  Lover雪儿  阅读(1354)  评论(0编辑  收藏  举报