历史上的今天
返回首页

历史上的今天

今天是:2024年10月21日(星期一)

正在发生

2021年10月21日 | S3C2440上LCD驱动(FrameBuffer)开发(一)

2021-10-21 来源:eefocus

一、开发环境


主  机: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端口开始地址

推荐阅读

史海拾趣

Connection One公司的发展小趣事

在电子行业的初期,Connection One公司凭借其创始人对半导体技术的深刻理解,成功研发出一款具有划时代意义的芯片。这款芯片不仅性能卓越,而且功耗极低,立即在市场上引起了轰动。公司因此迅速积累了第一笔资金,为后续的研发和扩展打下了坚实基础。

Graseby Infrared公司的发展小趣事
首先确认三相电源是否正常,可以使用万用表测量电源电压和相序。
High Voltage Power Solutions Inc公司的发展小趣事
首先确认三相电源是否正常,可以使用万用表测量电源电压和相序。
Cableform Inc公司的发展小趣事

2012年,对于Cableform Inc来说是一个重要的转折点。这一年,公司成功加入了HUBBELL系列,与Hubbell工业控制有限公司建立了战略合作关系。这一合作使得Cableform Inc能够借助Hubbell的品牌影响力和市场渠道,进一步拓展业务范围,提升市场地位。同时,通过与Hubbell的技术交流和资源共享,Cableform Inc在技术创新和产品升级方面也取得了显著进展。

CYAN公司的发展小趣事

随着5G技术的兴起,CYAN敏锐地捕捉到了市场的变化。公司投入大量研发资源,成功开发出一款基于5G技术的高性能网络路由器。这款路由器不仅具有超高的数据传输速度和稳定性,还具备智能管理和安全保护功能,满足了市场对高性能网络设备的需求。

台湾奇力(Chip)公司的发展小趣事

奇力公司的成功离不开与奇美集团的紧密合作。奇美集团是台湾知名的面板生产厂家,拥有强大的技术实力和市场份额。奇力公司作为奇美集团的子公司,得到了奇美集团在技术、资金和市场等方面的全力支持。通过与奇美集团的紧密合作,奇力公司得以快速发展,并在LED芯片市场上取得了显著的成绩。

问答坊 | AI 解惑

几种常用的虚拟仪表仪表

http://66.249.89.132/translate_c?hl=zh-CN&sl=en&u=http://www.abacom-online.de/uk/html/demoversionen.html&prev=/search%3Fq%3DC51%2Bcompiler%26hl%3Dzh-CN%26client%3Daff-cs-360se%26hs%3Dbtg%26sa%3DN%26start%3D50%26newwindow%3D1&usg=A ...…

查看全部问答>

为什么我的复位电平上不去?

复位芯片采用MAX811,MCU采用C8051F020,上电后,电源电压时3.3V,为什么复位电平只有1.36V? …

查看全部问答>

“喝”咖啡渣及茶渣的打印机

一台打印机,我们需要经常调换的就是墨盒了,然而用完了的墨盒还能用作什么用处?我想不出来。南韩设计师Jeon Hwan Ju为此设计了一款新的打印机墨水来源,就是利用我们经常喝过了的咖啡渣或茶渣來代替一般的墨水。 该款环保型打印机称为RITI, ...…

查看全部问答>

8*8的LED点阵如何区分共阴共阳?

8*8的LED点阵如何区分共阴共阳?…

查看全部问答>

IRP完成例程与KeWaitXxx配合出现的同步问题

我的代码是这样的: {    .......    KeInitializeEvent(&event, NotificationEvent, FALSE);    IoCopyCurrentIrpStackLocationToNext(Irp);    IoSetCompletionRoutine(Irp,PacketCompletion,&even ...…

查看全部问答>

谁能提供wince 6比5新的的详尽的特性对比?

1、谁能提供wince 6比5新的的详尽的特性对比?(google上搜到一个老外的页面的对比,不是很详细)最好是非常详细带解释的!高分求!!!!!!!! 2、谁能提供一份文字详细的wince6在2410或2440上运行的步骤包括编译连接烧写等加载! 3、以前的4 ...…

查看全部问答>

用了lut1lut的I2C函数,终于搞定,谢谢各位...

                                  …

查看全部问答>

请教DSP f2812的串行通讯错误是什么原因

我最近搞f2812时又出现了一个问题,用DSP往PC机传数据时,通过观查寄存器看到的结果是正确的,但是PC机上用串口调试助手接收到的数据却错误。我传的是0~9的一个随机数,但接收到的全是F0 00 00(用16进制显示)。我在网上查了可能是晶振或波特率设 ...…

查看全部问答>

低电平测量对测试夹具有何要求?

对用于低电平测量的测试夹具有几个重要的要求: * 绝缘电阻[1]:所有连接器、内部连线、端子和插座等的绝缘电阻都应当尽可能地高。一般地说,在高质量的测试夹具中,所有的连接器和插座都使用聚四氟乙烯绝缘材料。 * 屏蔽[2]和保护:测试夹具 ...…

查看全部问答>