S3C2440上LCD驱动(FrameBuffer)开发(一)

发布者:asdfrewqpp最新更新时间:2021-10-21 来源: eefocus关键字:S3C2440  LCD驱动  FrameBuffer 手机看文章 扫描二维码
随时随地手机看文章

一、开发环境


主  机:VMWare--Fedora 9

开发板:Mini2440--64MB Nand, Kernel:2.6.30.4

编译器:arm-linux-gcc-4.3.2

二、背景知识


1. LCD工作的硬件需求:

要使一块LCD正常的显示文字或图像,不仅需要LCD驱动器,而且还需要相应的LCD控制器。在通常情况下,生产厂商把LCD驱动器会以COF/COG的形式与LCD玻璃基板制作在一起,而LCD控制器则是由外部的电路来实现,现在很多的MCU内部都集成了LCD控制器,如S3C2410/2440等。通过LCD控制器就可以产生LCD驱动器所需要的控制信号来控制STN/TFT屏了。

 

2. S3C2440内部LCD控制器结构图:


我们根据数据手册来描述一下这个集成在S3C2440内部的LCD控制器:

a:LCD控制器由REGBANK、LCDCDMA、TIMEGEN、VIDPRCS寄存器组成;

b:REGBANK由17个可编程的寄存器组和一块256*16的调色板内存组成,它们用来配置LCD控制器的;

c:LCDCDMA是一个专用的DMA,它能自动地把在侦内存中的视频数据传送到LCD驱动器,通过使用这个DMA通道,视频数据在不需要CPU的干预的情况下显示在LCD屏上;

d:VIDPRCS接收来自LCDCDMA的数据,将数据转换为合适的数据格式,比如说4/8位单扫,4位双扫显示模式,然后通过数据端口VD[23:0]传送视频数据到LCD驱动器;

e:TIMEGEN由可编程的逻辑组成,他生成LCD驱动器需要的控制信号,比如VSYNC、HSYNC、VCLK和LEND等等,而这些控制信号又与REGBANK寄存器组中的LCDCON1/2/3/4/5的配置密切相关,通过不同的配置,TIMEGEN就能产生这些信号的不同形态,从而支持不同的LCD驱动器(即不同的STN/TFT屏)。

 

3. 常见TFT屏工作时序分析:


LCD提供的外部接口信号:

VSYNC/VFRAME/STV:垂直同步信号(TFT)/帧同步信号(STN)/SEC TFT信号;

HSYNC/VLINE/CPV:水平同步信号(TFT)/行同步脉冲信号(STN)/SEC TFT信号;

VCLK/LCD_HCLK:象素时钟信号(TFT/STN)/SEC TFT信号;

VD[23:0]:LCD像素数据输出端口(TFT/STN/SEC TFT);

VDEN/VM/TP:数据使能信号(TFT)/LCD驱动交流偏置信号(STN)/SEC TFT 信号;

LEND/STH:行结束信号(TFT)/SEC TFT信号;

LCD_LPCOE:SEC TFT OE信号;

LCD_LPCREV:SEC TFT REV信号;

LCD_LPCREVB:SEC TFT REVB信号。


 

所有显示器显示图像的原理都是从上到下,从左到右的。这是什么意思呢?这么说吧,一副图像可以看做是一个矩形,由很多排列整齐的点一行一行组成,这些点称之为像素。那么这幅图在LCD上的显示原理就是:

A:显示指针从矩形左上角的第一行第一个点开始,一个点一个点的在LCD上显示,在上面的时序图上用时间线表示就为VCLK,我们称之为像素时钟信号;

B:当显示指针一直显示到矩形的右边就结束这一行,那么这一行的动作在上面的时序图中就称之为1 Line;

C:接下来显示指针又回到矩形的左边从第二行开始显示,注意,显示指针在从第一行的右边回到第二行的左边是需要一定的时间的,我们称之为行切换;

D:如此类推,显示指针就这样一行一行的显示至矩形的右下角才把一副图显示完成。因此,这一行一行的显示在时间线上看,就是时序图上的HSYNC;

E:然而,LCD的显示并不是对一副图像快速的显示一下,为了持续和稳定的在LCD上显示,就需要切换到另一幅图上(另一幅图可以和上一副图一样或者不一样,目的只是为了将图像持续的显示在LCD上)。那么这一副一副的图像就称之为帧,在时序图上就表示为1 Frame,因此从时序图上可以看出1 Line只是1 Frame中的一行;

F:同样的,在帧与帧切换之间也是需要一定的时间的,我们称之为帧切换,那么LCD整个显示的过程在时间线上看,就可表示为时序图上的VSYNC。


 

上面时序图上各时钟延时参数的含义如下:(这些参数的值,LCD产生厂商会提供相应的数据手册)

VBPD(vertical back porch):表示在一帧图像开始时,垂直同步信号以后的无效的行数,对应驱动中的upper_margin;

VFBD(vertical front porch):表示在一帧图像结束后,垂直同步信号以前的无效的行数,对应驱动中的lower_margin;

VSPW(vertical sync pulse width):表示垂直同步脉冲的宽度,用行数计算,对应驱动中的vsync_len;

HBPD(horizontal back porch):表示从水平同步信号开始到一行的有效数据开始之间的VCLK的个数,对应驱动中的left_margin;

HFPD(horizontal front porth):表示一行的有效数据结束到下一个水平同步信号开始之间的VCLK的个数,对应驱动中的right_margin;

HSPW(horizontal sync pulse width):表示水平同步信号的宽度,用VCLK计算,对应驱动中的hsync_len;


 

对于以上这些参数的值将分别保存到REGBANK寄存器组中的LCDCON1/2/3/4/5寄存器中:(对寄存器的操作请查看S3c2440数据手册LCD部分)

LCDCON1:17 - 8位CLKVAL 

          6 - 5位扫描模式(对于STN屏:4位单/双扫、8位单扫) 

          4 - 1位色位模式(1BPP、8BPP、16BPP等)


LCDCON2:31 - 24位VBPD 

         23 - 14位LINEVAL 

         13 - 6位VFPD 

          5 - 0位VSPW


LCDCON3:25 - 19位HBPD 

         18 - 8位HOZVAL 

          7 - 0位HFPD


LCDCON4: 7 - 0位HSPW


LCDCON5:


 

4. 帧缓冲(FrameBuffer):

   帧缓冲是Linux为显示设备提供的一个接口,它把一些显示设备描述成一个缓冲区,允许应用程序通过FrameBuffer定义好的接口访问这些图形设备,从而不用去关心具体的硬件细节。对于帧缓冲设备而言,只要在显示缓冲区与显示点对应的区域写入颜色值,对应的颜色就会自动的在屏幕上显示。下面来看一下在不同色位模式下缓冲区与显示点的对应关系:



三、帧缓冲(FrameBuffer)设备驱动结构:

帧缓冲设备为标准的字符型设备,在Linux中主设备号29,定义在/include/linux/major.h中的FB_MAJOR,次设备号定义帧缓冲的个数,最大允许有32个FrameBuffer,定义在/include/linux/fb.h中的FB_MAX,对应于文件系统下/dev/fb%d设备文件。

1. 帧缓冲设备驱动在Linux子系统中的结构如下:


我们从上面这幅图看,帧缓冲设备在Linux中也可以看做是一个完整的子系统,大体由fbmem.c和xxxfb.c组成。向上给应用程序提供完善的设备文件操作接口(即对FrameBuffer设备进行read、write、ioctl等操作),接口在Linux提供的fbmem.c文件中实现;向下提供了硬件操作的接口,只是这些接口Linux并没有提供实现,因为这要根据具体的LCD控制器硬件进行设置,所以这就是我们要做的事情了(即xxxfb.c部分的实现)。


2. 帧缓冲相关的重要数据结构:从帧缓冲设备驱动程序结构看,该驱动主要跟fb_info结构体有关,该结构体记录了帧缓冲设备的全部信息,包括设备的设置参数、状态以及对底层硬件操作的函数指针。在Linux中,每一个帧缓冲设备都必须对应一个fb_info,fb_info在/linux/fb.h中的定义如下:(只列出重要的一些)


struct fb_info {

    int node;

    int flags;

    struct fb_var_screeninfo var;/*LCD可变参数结构体*/

    struct fb_fix_screeninfo fix;/*LCD固定参数结构体*/

    struct fb_monspecs monspecs; /*LCD显示器标准*/

    struct work_struct queue;    /*帧缓冲事件队列*/

    struct fb_pixmap pixmap;     /*图像硬件mapper*/

    struct fb_pixmap sprite;     /*光标硬件mapper*/

    struct fb_cmap cmap;         /*当前的颜色表*/

    struct fb_videomode *mode;   /*当前的显示模式*/


#ifdef CONFIG_FB_BACKLIGHT

    struct backlight_device *bl_dev;/*对应的背光设备*/

    struct mutex bl_curve_mutex;

    u8 bl_curve[FB_BACKLIGHT_LEVELS];/*背光调整*/

#endif

#ifdef CONFIG_FB_DEFERRED_IO

    struct delayed_work deferred_work;

    struct fb_deferred_io *fbdefio;

#endif


    struct fb_ops *fbops; /*对底层硬件操作的函数指针*/

    struct device *device;

    struct device *dev;   /*fb设备*/

    int class_flag;    

#ifdef CONFIG_FB_TILEBLITTING

    struct fb_tile_ops *tileops; /*图块Blitting*/

#endif

    char __iomem *screen_base;   /*虚拟基地址*/

    unsigned long screen_size;   /*LCD IO映射的虚拟内存大小*/ 

    void *pseudo_palette;        /*伪16色颜色表*/ 

#define FBINFO_STATE_RUNNING    0

#define FBINFO_STATE_SUSPENDED  1

    u32 state;  /*LCD的挂起或恢复状态*/

    void *fbcon_par;

    void *par;    

};


其中,比较重要的成员有struct fb_var_screeninfo var、struct fb_fix_screeninfo fix和struct fb_ops *fbops,他们也都是结构体。下面我们一个一个的来看。


fb_var_screeninfo结构体主要记录用户可以修改的控制器的参数,比如屏幕的分辨率和每个像素的比特数等,该结构体定义如下:


struct fb_var_screeninfo {

    __u32 xres;                /*可见屏幕一行有多少个像素点*/

    __u32 yres;                /*可见屏幕一列有多少个像素点*/

    __u32 xres_virtual;        /*虚拟屏幕一行有多少个像素点*/        

    __u32 yres_virtual;        /*虚拟屏幕一列有多少个像素点*/

    __u32 xoffset;             /*虚拟到可见屏幕之间的行偏移*/

    __u32 yoffset;             /*虚拟到可见屏幕之间的列偏移*/

    __u32 bits_per_pixel;      /*每个像素的位数即BPP*/

    __u32 grayscale;           /*非0时,指的是灰度*/


    struct fb_bitfield red;    /*fb缓存的R位域*/

    struct fb_bitfield green;  /*fb缓存的G位域*/

    struct fb_bitfield blue;   /*fb缓存的B位域*/

    struct fb_bitfield transp; /*透明度*/    


    __u32 nonstd;              /* != 0 非标准像素格式*/

    __u32 activate;                

    __u32 height;              /*高度*/

    __u32 width;               /*宽度*/

    __u32 accel_flags;    


    /*定时:除了pixclock本身外,其他的都以像素时钟为单位*/

    __u32 pixclock;            /*像素时钟(皮秒)*/

    __u32 left_margin;         /*行切换,从同步到绘图之间的延迟*/

    __u32 right_margin;        /*行切换,从绘图到同步之间的延迟*/

    __u32 upper_margin;        /*帧切换,从同步到绘图之间的延迟*/

    __u32 lower_margin;        /*帧切换,从绘图到同步之间的延迟*/

    __u32 hsync_len;           /*水平同步的长度*/

    __u32 vsync_len;           /*垂直同步的长度*/

    __u32 sync;

    __u32 vmode;

    __u32 rotate;

    __u32 reserved[5];         /*保留*/

};


而fb_fix_screeninfo结构体又主要记录用户不可以修改的控制器的参数,比如屏幕缓冲区的物理地址和长度等,该结构体的定义如下:


struct fb_fix_screeninfo {

    char id[16];                /*字符串形式的标示符 */

    unsigned long smem_start;   /*fb缓存的开始位置 */

    __u32 smem_len;             /*fb缓存的长度 */

    __u32 type;                 /*看FB_TYPE_* */

    __u32 type_aux;             /*分界*/

    __u32 visual;               /*看FB_VISUAL_* */ 

    __u16 xpanstep;             /*如果没有硬件panning就赋值为0 */

    __u16 ypanstep;             /*如果没有硬件panning就赋值为0 */

    __u16 ywrapstep;            /*如果没有硬件ywrap就赋值为0 */

    __u32 line_length;          /*一行的字节数 */

    unsigned long mmio_start;   /*内存映射IO的开始位置*/

    __u32 mmio_len;             /*内存映射IO的长度*/

    __u32 accel;

    __u16 reserved[3];          /*保留*/

};


fb_ops结构体是对底层硬件操作的函数指针,该结构体中定义了对硬件的操作有:(这里只列出了常用的操作)


struct fb_ops {


    struct module *owner;


    //检查可变参数并进行设置

    int (*fb_check_var)(struct fb_var_screeninfo *var, struct fb_info *info);


    //根据设置的值进行更新,使之有效

    int (*fb_set_par)(struct fb_info *info);


    //设置颜色寄存器

    int (*fb_setcolreg)(unsigned regno, unsigned red, unsigned green,

             unsigned blue, unsigned transp, struct fb_info *info);


    //显示空白

    int (*fb_blank)(int blank, struct fb_info *info);


    //矩形填充

    void (*fb_fillrect) (struct fb_info *info, const struct fb_fillrect *rect);


    //复制数据

    void (*fb_copyarea) (struct fb_info *info, const struct fb_copyarea *region);


    //图形填充

    void (*fb_imageblit) (struct fb_info *info, const struct fb_image *image);

};


3. 帧缓冲设备作为平台设备:

在S3C2440中,LCD控制器被集成在芯片的内部作为一个相对独立的单元,所以Linux把它看做是一个平台设备,故在内核代码/arch/arm/plat-s3c24xx/devs.c中定义有LCD相关的平台设备及资源,代码如下:


/* LCD Controller */


//LCD控制器的资源信息

static struct resource s3c_lcd_resource[] = {

    [0] = {

        .start = S3C24XX_PA_LCD, //控制器IO端口开始地址

[1] [2]
关键字:S3C2440  LCD驱动  FrameBuffer 引用地址:S3C2440上LCD驱动(FrameBuffer)开发(一)

上一篇:S3C2440上LCD驱动(FrameBuffer)开发(二)
下一篇:mini2440 uboot的使用经历

推荐阅读最新更新时间:2024-10-31 21:19

s3c2440头文件之2440lib.h
//=================================================================== // File Name : 2440lib.h // Function : S3C2440 // Date : February 26, 2002 // Version : 0.0 // History // 0.0 :Feb.20.2002:SOP : Programming start // 0.01:Mar.29.2002:purnnamu: For Sleep_wake_up, the START... label is added //==================
[单片机]
s3c2440 MPLL & UPLL
S3C2440有两个PLL(phase locked loop)一个是MPLL,一个是UPLL。 MPLL用于CPU及其他外围器件,UPLL用于USB。用于产生FCLK, HCLK, PCLK三种频率,这三种频率分别有不同的用途: FCLK是CPU提供的时钟信号。 HCLK是为AHB总线提供的时钟信号, Advanced High-performance Bus,主要用于高速外设,比如内存控制器,中断控制器,LCD控制器, DMA 等。 从S3C2440的DataSheet里可以看到,S3C2440最大支持400MHz的主频,但是这并不意味着一定工作在400MHz下面,可以通过设定MPLL, UPLL寄存器来设定CPU的工作
[单片机]
S3C2440开发板上BMP图片的显示之三
三、BMP图片显示功能扩展 1、图片显示的功能扩展 (1)能够调整图片的显示区域,比如我准备了一张 480x384的图片,液晶屏大小只有它的九分之一,要通过键盘控制,“看到整张图片”。 (2)能够显示单色、16色、256色、24位真彩色图片。 (3)能够对图片进行缩小显示,比如前面那个480x384的图片,能够缩小比例显示在液晶屏上。这个实现应该不难,但是放大就比较麻烦了,要用到数学上的插值,这个稍微做一下尝试。 2、图片的移动 (1)分析 首先一个要明确的概念,我们所谓“移动图片”,实质不是在移动图片,而是在移动观测图片的 “显示窗口”。 比如一张比显示屏大的图片,现在只显示了上半部分,我们要看它的下半部分,可以按下方向键“
[单片机]
基于S3C2440嵌入式系统主板的电磁兼容性设计
作者:许凯华,董淑云,刘玉华,胡立祥 华中师范大学  随着电子设备的频率越来越高,世界各国对电子产品电磁辐射标准的执行变得越来越严格,如何保证能在有限时间很好地在设计阶段发现并解决EMI/EMC问题非常重要,而PCB往往是一个电子系统的核心构成部分,一个经仔细电磁干扰设计的PCB板,能大幅度降低阻抗不匹配、传输线问题、信号互相耦合等现象引发的信号反射、延迟等线路不稳定因素,同时也可达到降低电磁辐射发射干扰,大大提高系统的稳定性和可靠性。本文将以嵌入式系统主板为平台,运用 EMIStream仿真软件,并采用源端串联端接阻抗的方法分析了解决嵌入式高速主板存在的电磁干扰问题。 1 电磁兼容性 1.1 电磁兼容和电磁干扰
[嵌入式]
s3c2440内存类设备访问原理
在嵌入式里面所谓内存类接口有SDRAM接口,NORFLASH接口,DM9000网卡接口等 ,他们都有一个共同点就是可以直接连S3C2440内存控制器,数据总线,地址总线;这三者缺一不可。 内存类接口可以通过地址直接访问内部数据,当然他们都有自己的一段可访问的地址范围。S3C2440将其1G地址空间划分为8个bank,每个bank有128M,所以芯片提供了8个片选信号线,每个片选信号线对应一个bank,一个片选信号连接一个设备,所以每个设备可访问的地址的范围是128M空间,128M=2的27次方个字节,所以芯片上面的地址线只有27跟,最大只能到A26。理论上32位的CPU应该有32跟地址线,因为最大可以到0xffffffff=2
[单片机]
<font color='red'>s3c2440</font>内存类设备访问原理
一起学mini2440裸机开发(三)--S3C2440时钟学习
前言 首先,我们应该知道一点,mini2440开发板在没有开启时钟前,整个开发板全靠一个12MHz的外部晶振提供频率来工作运行的,也就是说CPU、内存、UART、ADC等所有需要用到时钟频率的硬件都工作在12MHz下,而S3C2440A可以正常工作在400MHz下,可想而知两者速度相差会有多大了。如果CPU工作在12MHz频率下,开发板的使用效率非常低,所有依赖系统时钟工作的硬件,其工作效率也很低,比如,我们电脑里面经常提到的超频,超频就是让CPU工作在更高的频率下,让电脑运算速度更快,虽然频率是越高越好,但是由于硬件特性决定了任何一个设备都不可能无止境的超频,电脑超频时要考虑到CPU或主板发热过大,烧坏的危险,同样开发板的主板
[单片机]
一起学mini2440裸机开发(三)--<font color='red'>S3C2440</font>时钟学习
S3C2440读取NAND Flash的总结
在网上找了一些资料,又结合自己的经历谈一下我对NAND Flash 的了解。 S3C2440 板的Nand Flash 支持由两部分组成:Nand Flash 控制器(集成在S3C2440 CPU)和Nand Flash 存储芯片(K9F1208U0B)两大部分组成。当要访问Nand Flash中的数据时,必须通过Nand Flash控制器发送命令才能完成。所以, Nand Flash相当于S3C2440的一个外设,而不位于它的内存地址区. NAND Flash 的数据是以bit 的方式保存在memory cell,一般来说,一个cell 中只能存储一个bit。这些cell 以8 个或者16 个为单位,连成bit line,
[单片机]
对<font color='red'>S3C2440</font>读取NAND Flash的总结
u-boot1.1.6在s3c2440上移植
我用的2440开发板,取名为TX2440。 第一步:U-Boot -1.1.6需要交叉编译工3.3.2,首先安装arm-linux-gcc-3.3.2.tar.bz2。 第二步:将U-Boot-1.1.6放入smb服务器,在linux下解压。 (1)到U-boot-1.1.6目录下:cd u-boot-1.1.6 修改cpu/arm920t下的config.mk文件,将其中的-msoft-float注释掉: vi cpu/arm920t/config.mk #-msoft-float(意思是不使用软浮点进行编译。我们使用 的还是硬浮点) (2)修改U-boot-1.1.6目录下Makefile: 在smdk2410_config
[单片机]
小广播
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved