【驱动】第4课、LCD驱动之学习笔记

开发环境

主   机:VMWare--Ubuntu-16.04.2-x64-100ask
开发板:Mini2440--256M NandFlash, 2M NorFlash, 64M SDRAM, LCD-TD35;
    bootlorder:u-boot1.16, Kernel:2.6.22.6;
编译器:arm-linux-gcc-3.4.5


 目录

1、LCD驱动程序分析方法

2、本课程LCD驱动源码程序的缺陷

3、本LCD驱动程序与裸机LCD程序框架的异同

4、LCD驱动程序设计扩展猜想

5、笔记

6、LCD驱动学习难点是以下三个函数的使用

7、问题:在bootloader阶段、内核空间、用户空间对指针指向的空间是如何定义和引用的

  其一:郝斌C语言课程关于指针常见错误的笔记

  其二:谭浩强《C程序设计》第八章_善于利用指针

  7.2 内核空间指针和指针变量的引用的方法?或者说对内核空间内存单元的操作方法

  总结

8、寄存器位运算详解

9.测试


 

1、驱动程序分析,一般从入口函数(xx_init()函数)或设备的 file_operations 结构体入手,一路走下去,

重点分析对象: 1>赋值语句; 2>函数调用; 3>
不特别在意的东西: 1>参数的判断的语句; 2>看不懂的语句;

2、本LCD驱动程序的缺陷:对于LCD的参数的使用是直接查看LCD手册,然后填写到驱动的帧缓冲器结构体、硬件寄存器的配置、
对于不同的LCD操作不具有可移植性,

3、本LCD驱动程序与裸机LCD程序框架的异同:
相同:
不同: 1)本LCD驱动多了帧缓冲器信息结构体struct fb_info *s3c_lcd 的初始化和配置;
驱动程序中,根据s3c_lcd = framebuffer_alloc(0, NULL);分配帧缓冲器的各项信息,
I: 内容上类同于裸板LCD结构体的定义,但是本质和功能完全不同,该驱动是帧缓冲器信息结构体s3c_lcd,裸板是某款LCD的各项时间参数结构体lcd_3_5_params等;
帧缓冲器信息结构体s3c_lcd作用:定义可供内核调用某款LCD的使用的LCD帧缓冲器;
LCD参数结构体lcd_3_5_params作用:给LCD控制器的初始化配置提供数据;
II:FB_BASE显存地址的分配方式不同:
驱动上是>> /* 3.3 分配显存(framebuffer), 并把地址告诉LCD控制器 */
s3c_lcd->screen_base = dma_alloc_writecombine(NULL, s3c_lcd->fix.smem_len, &s3c_lcd->fix.smem_start, GFP_KERNEL);
裸板上是>> #define LCD_3_5_FB_BASE 0x33b00000 //lcd_3.5.c文件;

2)本LCD驱动少了裸板对于LCD结构体的定义、具体某一款LCD的定义初始化;

4、LCD驱动程序设计扩展猜想:可否把裸板的分离分层程序和该章的框架结合起来?甚至把前两章学习的分离、分层结合起来?
编写一个具备良好移植兼容性的LCD驱动模块。

5、笔记:
1)对于寄存器或者映射内存,需要遵循先定义后使用的原则:
gpbcon = ioremap(0x56000010, 8);
lcdcon1 = (4<<8) | (3<<5) | (0x0c<<1);

2)硬件寄存器的设置上,本驱动和我的mini2440的寄存器配置的数据有什么不同?
I:GPIO口——
II:LCD控制器——

3)LCD背光灯、LCD本身、触摸屏是三个独立的系统,只不过集成到一个LCD屏幕上工作而已。
4)裸板LCD的主要工作是配置LCD及其控制器,并实现分离分层,然后是关于测试LCD的应用程序:在帧缓冲器中填充数据以完成不同的图案或字符的输出;
驱动LCD的主要工作是配置帧缓冲器结构体struct fb_info *s3c_lcd ,然后是LCD相关寄存器的配置。
5)驱动中调色板(假调色板)终究没有用到?只是在fb_ops结构体中定义好元素.fb_setcolreg(即对应LCD的调色板配置函数s3c_lcdfb_setcolreg),
以兼容以往的程序?

6、LCD驱动学习难点是以下三个函数的使用:
I. s3c_lcdfb = framebuffer_alloc(0, NULL);
功能:分配一个帧缓冲器,即分配一段连续的物理内存并 set to zero 给帧缓冲器结构体指针 s3c_lcdfb使用。
问题1:以前分配一个结构体时,直接定义然后初始化结构体内的元素即可。这次的xx_fb结构体,定义时(或之后),
需要在初始化之前使用专用的函数分配一段内存给结构体xx_fb。是因为这个结构体太大了吗?
以前裸机全局变量是直接在.bss段分配内存给其使用,由于.bss段在程序有效数据的最后面,只要总的内存还够,就可以直接
顺序分配内存空间给全局变量。如此,既有问题2:
问题2:整个linux系统(bootlorder, linux内核, fs根文件系统)中,关于内存是如何管理、使用和分配的?具体到驱动程序的全局变量和应用程序
的全局变量的内存又是怎样分配的?
答:百度查到的博文有说,内存管理系统可以分为两部分,分别是内核空间内存管理和用户空间内存管理。具体以后再看。
问题3:framebuffer_alloc()函数的使用是因为在其之前只定义了帧缓冲器结构体指针 s3c_lcdfb的原因还是 因为结构体太大需要用专用函数
分配一段内存给该结构体使用?对于结构体的定义是怎么样的?
补充:以前裸机的全局变量,若是一般全局变量分配到.data段,const全局变量(常变量)分配到.rodata段(只读数据段),初值为0/无初值的全局变量
分配到.bss段。这些段在内存中的地址分配如下:
SECTIONS{
. = 0x30000000; /*SDRAM在CPU统一编址中的基地址*/
__code_start = .;
. = ALIGN(4);
.text : {*(.text);}
. = ALIGN(4);
.rodata : {*(.rodata);}
. = ALIGN(4);
.data : {*(.data);}
. = ALIGN(4);
.bss : {
_bss_start = . ;
*(.bss) *(.COMMON);
}
. = ALIGN(4);
_end = . ;
}
注意:kzalloc申请过程做了初始化工作并在其后的过程中给指针的成员device做了赋值工作。如果暂时不要给device成员赋值,参数可以写NULL。
II. s3c_lcdfb->screen_base = dma_alloc_writecombine(NULL, s3c_lcdfb->fix.smem_len, s3c_lcdfb->fix.smem_start, GFP_KERNEL); /* 3.3 分配显存,并把显存物理地址告诉LCD控制器 */
III. register_framebuffer(s3c_lcdfb); /* 4.注册帧缓冲器设备 */

--------------------------------------------------------------------------------------
7、问题:在bootloader阶段、内核空间、用户空间对指针指向的空间是如何定义和引用的?
答:我的困惑源于对各层级指针操作仍存在错误的模糊认识!关于bootloader空间、内核空间、用户空间指针的用法如下。

7.0 <1>旧问题:内核中,对于结构体变量的是如何定义和分配内存的?
<2>定义结构体指针变量时,有没有同时分配存储空间啊?
<3>看到结构体的数组定义好以后就直接可以用了。但是结构体指针在链表中还要malloc()申请空间。这是为什么啊?
答:<1>两种方法,一种是静态分配内存(即直接定义):struct Student stu1; 第二种是通过该结构体指针动态分配一段内存空间:struct Student *pst2=malloc(sizeof(struct Student));
<2>定义结构体指针变量 p,并不代表定义了结构体指针指向的变量 *p,系统分配的内存是给指针变量p用的,对于指针p指向的空间未定义不分配内存空间;
<3>结构体数组struct Student pstu3[]={{...}, {...}};即同时定义了两个结构体变量pstu3[0], pstu3[1], 只不过他们是作为数组pstu3的元素而存在的。
链表一般是动态分配内存空间,可随时插入和删除节点,作为指针域的指针,需要动态分配一个节点的内存空间作为其指向的插入的节点。

7.1 在用户空间中指针变量的引用的方法,详解如下:
其一:郝斌C语言课程关于指针常见错误的笔记
<0>指针就是地址,地址就是指针; 指针不是指针变量,指针变量不是指针。
<1>int *p; #不代表定义了一个叫做*p的变量!
<2>所有的变量、内存单元都遵循先定义再使用的规则,否则就会编译报错或程序崩溃!
例:int main(void){
int *p;
int i = 5;
// *p = i; //不屏蔽则程序运行崩溃!
printf("p = %#x\n", p); //打印:p = 0xcccccccc
// printf("*p = %d\n", *p); //不屏蔽则程序运行崩溃!
return 0;
}
/*在VC++6.0中的运行结果:
p = 0xcccccccc
Press any key to continue
*/
语句*p = i; (//不屏蔽则程序运行崩溃!)的详解:
p须先有指向,然后才能有一个值(地址)赋给它; 若p无指向,则p为空/不可知的垃圾值,因为不知道p的值是多少,所以不知*p
到底代表的是哪一个变量。而*p=i=5,就是把5赋给了一个不知道地址的单元。归根到底是因为用户程序只能访问属于本程序的内存单元,
对于不属于本程序的内存单元,指针p无权访问。
<3>为指针变量指向的内存单元分配内存空间的方法(目前只有这两种方法,其他alloc与malloc类同):
a.静态分配指针p指向的内存空间:int *p; int i; p=&i; *p=5;
b.动态分配指针p指向的内存空间:int *p = (int *)malloc(sizeof(int)); *p=5; //其中(int *)可以省略,系统会自动转换到需要的指针类型;
说明:内存的动态分配主要应用于建立程序中的动态数据结构(如链表)中。
<4>给指针变量赋值的方法只有一种: int *p; int i; p=&i; ?应该还有另外一种,如上malloc。

其二:谭浩强《C程序设计》第八章_善于利用指针
8.1 指针是什么
为了说清楚什么是指针,必须先弄清楚数据在内存中是如何存储的,又是如何读取的。
如果在程序中定义了一个变量,在对程序进行编译时,系统就会给这个变量分配内存单元。编译系统根据程序中定义的变量类型,分配一定长度的空间。
由于通过地址能找到所需的变量单元,可以说,地址指向该变量单元。
在程序中一般是通过变量名来引用变量的值,例如:printf("%d\n", i);
上面实际上,是通过变量名i找到存储单元的地址,从而对存储单元进行存取操作的。程序经过编译以后已经将变量名转换为变量的地址,对变量值的存取都是
通过地址进行的。 (对变量操作的本质>>>>>>>>>对内存操作)。
将数值3送到变量i中,有两种表达方法:
<1>直接访问:将3直接送到变量i所标志的单元中,例如“i=3;”; “scanf("%d", &i);”。
<2>间接访问:将3送到变量pi所指向的单元(即变量i的存储单元),例如“*pi=3;”, 其中*pi表pi指向的对象。
8.2 指针变量
存放地址的变量是指针变量,它用来指向另一个对象(如变量、数组、函数等)。
8.2.1 定义指针变量
类型名 * 指针变量名;
8.2.1 引用指针变量
<1>给指针变量赋值。如:p=&a; //把a的地址赋给指针变量p;
<2>引用指针变量指向的变量。如果已经执行"p=&a;",即指针变量p指向了整型变量a,则:printf("%d",*p);其作用是以整数形式输出指针变量p所指向的变量的值,即变量a的值。如果有赋值语句:*p=1;表示把整数1赋给p当前所指向的变量。
<3>引用指针变量的值。如:printf("%#x", p);
注意:要熟练掌握两个有关的运算符:<1>& 取地址运算符。<2>* 指针运算符。
8.8 动态内存分配与指向它的指针变量
8.8.1 什么是内存的动态分配
除了静态内存分配(一般变量),C语言还允许建立内存动态分配区域,以存放一些临时用的数据,这些数据不必在程序的声明部分定义,也不必等到函数结束
时才释放,而是需要时随时开辟,不需要时随时释放。由于未在声明部分定义它们为变量或数组,因此不能通过变量名或者数组名去引用这些数据,只能通过指针来引用。
8.8.2 怎样建立内存的动态分配
对内存的动态分配是通过系统提供的库函数来实现的,主要有malloc, calloc, free, realloc等函数。
内存的动态分配主要应用于建立程序中的动态数据结构(如链表)中。
【知识归纳】
指针的定义:一个从0开始的非负整数的内存单元编号。
指针变量的使用:先定义(int i;int *p;),然后赋初值(p=&i),再然后才可以对其指向的内存单元进行读/写等操作。

7.2 内核空间指针和指针变量的引用的方法?或者说对内核空间内存单元的操作方法?
分两类:一类是对硬件地址的直接引用(*(volatile unsigned int *)(0x48000000)) ,但只有读/写两种操作方式,这是在bottloader(和内核?)阶段使用的,
用户空间所不具备的应用方式。用户空间属于应用层,通过内核方可访问硬件(包括各种设备的地址)。
这个对硬件地址的直接引用的层级没有系统概念,因为系统是我们自己定义编写的。
二类是与用户空间的方法一样:定义指针变量静态分配内存,然后赋初值再引用,p=&a;*p=3; 或动态分配内存int *p=malloc(sizeof(int));*p=5;可以有读/写/算数加减(包括自加减)等运算操作。
【总结】
造成这种模糊认知是因为arm学习中,CPU对GPIO接口、协议类接口、内存类接口控制器(不包括Nandflash)进行统一编址,
对寄存器的操作(包括读/写数据)是直接用寄存器的地址(即指针)的指针运算符* Addr(例如:GPACON)作为对象进行读/写操作,而非借助
某一对象,或者说这个对象已经回归到某一具体(已知地址的)设备或内存单元。
#define __REG(x) (*(volatile unsigned int *)(x))
#define BWSCON __REG(0x48000000) //Memory Controllers:Bus width & wait status control
#define GPACON __REG(0x56000000) //I/O port:Port A control
在裸机和内核中,对于没有赋初值的变量一般自动写0,指针变量写NULL。这时,对指针变量 p 操作:*p=3;是不现实的。

----------------------------------------------------------------------------------------------------------------
8、寄存器位运算详解
<1>lcdsaddr1的设置如下,哪种方法正确?
LCDSADDR1位定义:
LCDBANK [29:21]
这些位表明系统存储器中视频缓冲器的 bank 位置的 A[30:22]。即使当
移动视口时也不能改变 LCDBANK 的值。LCD 帧缓冲器应该在 4MB
连续区域内,以保证当移动视口时不会改变 LCDBANK 的值。因此应
该谨慎使用 malloc()函数。
默认值:0x00
LCDBASEU [20:0]
对于双扫描 LCD:这些位表明递增地址计数器的开始地址的 A[21:1],
它是用于双扫描 LCD 的递增帧存储器或单扫描 LCD 的帧存储器。
对于单扫描 LCD:这些位表明 LCD 帧缓冲器的开始地址的 A[21:1]。
默认值:0x000000
LCD裸板程序该寄存器的配置:
addr = plcdparams->fb_base<<1;
addr >>= 2;
LCDSADDR1 = addr; //帧缓冲器的开始地址;
方法1:plcd_regs->lcdsaddr1 = ((s3c_lcdfb->fix.smem_start<<1)>>2);
方法2:lcd_regs->lcdsaddr1 = (s3c_lcd->fix.smem_start >> 1) & ~(3<<30);
问题:哪种可以达到正确配置要求?
答:方法2是经过LCD驱动上机验证可行的,对于方法1是裸板LCD程序使用的方法,但是对于long型数据的直接多次位操作并赋予
寄存器尚待商榷。
验证代码:
#include <stdio.h>
int main(void)
{ unsigned int addr1 = 0xffffffff;
unsigned int lcdsaddr1 = ((addr1<<1)>>2);
unsigned int lcdsaddr2 = ((0xffffffff<<1)>>2);
printf("lcdsaddr1 = %#x, lcdsaddr2 = %#x\n", lcdsaddr1, lcdsaddr2);
return 0;
}
/*在VC++6.0中的运行结果:
lcdsaddr1 = 0x3fffffff, lcdsaddr2 = 0x3fffffff
Press any key to continue
*/
VC++6.0和gcc编译器对于这类位操作的规则是否一致?为什么不在直接在arm平台测试该应用程序呢?##########
若一致,即可说明方法2也是可行的。本次LCD驱动测试使用方法2,观察LCD运行是否正常。
现象:可使得LCD正常工作!以上两种方法均可运行!


9.测试
目的:配置一个没有LCD模块的内核,用新内核去挂接到开发板0x30000000(即SDRAM)进行测试:
# nfs 30000000 192.168.1.105:/work/nfs_root/uImage_mlcd
挂接(服务器上的)网络根文件系统到(开发板根文件系统的)/mnt,从flash上启动根文件系统:
# mount -t nfs -o nolock,vers=2 192.168.1.105:/work/nfs_root/fs_second /mnt
说明:经在单板测试,若单板上的根文件系统原本就是挂接的服务器上的根文件系统(例如:fs_second),则不需再用mount命令挂接网络根文件
系统到开发板了!可直接进行驱动模块加载等试,执行命令:
# echo hello > /dev/tty1
均可在单板LCD上打印出字符串“hello”!

操作步骤:
9.1 编译一个新的没有LCD模块的内核--uImage_nolcd
$ cd /home/book/workbook/mini2440/systems/linux-2.6.22.6/
$ make menuconfig //配置内核,去掉原来的LCD驱动程序;
-> Device Drivers
-> Graphics support
<M> S3C2410 LCD framebuffer support
$ make uImage //编译生成新内核;
$ cp arch/arm/boot/uImage /work/nfs_root/uImage_nolcd
$ make modules //编译模块,是为了把fb_ops结构体的3个cfb_xx函数对应的cfb_xx.c源文件编译成.ko文件(模块),供稍后测试时使用。

9.2 用新内核启动开发板,在倒数计时结束前按下“空格”键,进入uboot菜单>
OpenJTAG> print
...(打印的内核信息..)
ipaddr=192.168.7.17
...(打印的内核信息..)
OpenJTAG> set ipaddr 192.168.1.17
OpenJTAG> save
OpenJTAG> nfs 30000000 192.168.1.105:/work/nfs_root/uImage_mlcd //uImage_mlcd是内核菜单配置时,LCD_fb配置为<M>,即模块,可以事后加载使用;
OpenJTAG> bootm 30000000 //启动新内核uImage_mlcd;
<启动内核...>
# mount -t nfs -o nolock,vers=2 192.168.1.105:/work/nfs_root/fs_second /mnt
# cd /mnt
# ls
bin driver_test lib mnt sbin usr
dev etc linuxrc proc sys

9.3 重新配置单板根文件系统的/etc/inittab文件,使得单板上的Linux系统拥有串口0终端和单板按键-LCD两个控制台
修改inittab文件:
# vi /etc/inittab
#/etc/inittab
#console::askfirst:-/bin/sh
::sysinit:/etc/init.d/rcS
s3c2410_serial0::askfirst:-/bin/sh
tty1::askfirst:-/bin/sh
::ctrlaltdel:/sbin/reboot
::shutdown:/bin/umount -a -r
查看:
# cat /etc/inittab
#/etc/inittab
#console::askfirst:-/bin/sh
::sysinit:/etc/init.d/rcS
s3c2410_serial0::askfirst:-/bin/sh
tty1::askfirst:-/bin/sh
::ctrlaltdel:/sbin/reboot
::shutdown:/bin/umount -a -r
# reboot //重启系统;
OpenJTAG> nfs 30000000 192.168.1.105:/work/nfs_root/uImage_mlcd
OpenJTAG> bootm 30000000 //启动新内核uImage_mlcd;
<启动内核...>
# mount -t nfs -o nolock,vers=2 192.168.1.105:/work/nfs_root/fs_second /mnt
# cd /mnt
# insmod cfbcopyarea.ko
# insmod cfbimgblt.ko
# insmod cfbfillrect.ko
# insmod lcd_6.ko
Console: switching to colour frame buffer device 30x40
# insmod input_keys2.ko
input: Unspecified device as /class/input/input1
<此时,LCD屏幕显示提示信息...以下都是单板按键--LCD屏幕控制台的LCD屏幕的显示信息...>
Please press Enter to activate
starting pid 768, tty '/dev/tty1': '/bin/sh'
# ls
bin dev etc driver_test ...

问题:刚加载好了LCD模块,LCD可以正常使用,但是过一会不用就黑屏了,不能再显示写入的字符串了,为什么?
答:Linux下的LCD驱动默认在无操作之后10分钟后会自动关闭屏幕。
问题:怎么唤醒LCD屏幕呢?
答:若加载了单板的输入子系统,可直接敲击单板上的指令“按键”,即可自动唤醒LCD屏幕进行指令操作。
在中断执行命令# echo wakakak > /dev/tty1 虽然仍然可以发送字符到单板LCD屏幕,但是不能唤醒LCD屏幕,需要用单板按键唤醒!

 

10、源码

  1 /*
  2 2018-12-14
  3 File: lcd_2.c
  4 */
  5 #include <linux/module.h>
  6 #include <linux/kernel.h>
  7 #include <linux/errno.h>
  8 #include <linux/string.h>
  9 #include <linux/mm.h>
 10 #include <linux/slab.h>
 11 #include <linux/delay.h>
 12 #include <linux/fb.h>
 13 #include <linux/init.h>
 14 #include <linux/dma-mapping.h>
 15 #include <linux/interrupt.h>
 16 #include <linux/workqueue.h>
 17 #include <linux/wait.h>
 18 #include <linux/platform_device.h>
 19 #include <linux/clk.h>
 20 
 21 #include <asm/io.h>
 22 #include <asm/uaccess.h>
 23 #include <asm/div64.h>
 24 
 25 #include <asm/mach/map.h>
 26 #include <asm/arch/regs-lcd.h>
 27 #include <asm/arch/regs-gpio.h>
 28 #include <asm/arch/fb.h>
 29 MODULE_LICENSE("GPL");
 30 
 31 /* 函数声明 */
 32 static int s3c_lcdfb_setcolreg(unsigned regno, unsigned red, unsigned green,
 33                 unsigned blue, unsigned transp, struct fb_info *info);    
 34 
 35 struct lcd_regs {
 36     unsigned long lcdcon1;
 37     unsigned long lcdcon2;
 38     unsigned long lcdcon3;
 39     unsigned long lcdcon4;
 40     unsigned long lcdcon5;
 41     unsigned long lcdsaddr1;
 42     unsigned long lcdsaddr2;
 43     unsigned long lcdsaddr3;
 44     unsigned long redlut;
 45     unsigned long greenlut;
 46     unsigned long bluelut;
 47     unsigned long reserved[8];
 48     unsigned long dithmode;
 49     unsigned long tpal;
 50     unsigned long lcdintpnd;
 51     unsigned long lcdsrcpnd;
 52     unsigned long lcdintmsk;
 53     unsigned long tconsel;
 54 };
 55 
 56 static struct fb_ops s3c_lcdfb_ops = {
 57     .owner = THIS_MODULE,
 58     /* set color register */
 59     .fb_setcolreg = s3c_lcdfb_setcolreg,
 60     /* Draws a rectangle */
 61     .fb_fillrect = cfb_fillrect,        /* void (*fb_fillrect) (struct fb_info *info, const struct fb_fillrect *rect); */
 62     /* Copy data from area to another*/
 63     .fb_copyarea = cfb_copyarea,        /* void (*fb_copyarea) (struct fb_info *info, const struct fb_copyarea *region);*/
 64     /* Draws a image to the display */
 65     .fb_imageblit = cfb_imageblit,        /*void (*fb_imageblit) (struct fb_info *info, const struct fb_image *image);*/
 66 };
 67 static struct fb_info *s3c_lcdfb;    /* 定义一个帧缓冲器结构体 */
 68 static volatile struct lcd_regs *plcd_regs;
 69 static volatile unsigned long *gpbcon;
 70 static volatile unsigned long *gpbdat;
 71 static volatile unsigned long *gpccon;
 72 static volatile unsigned long *gpdcon;
 73 static volatile unsigned long *gpgcon;
 74 static u32 pseudo_palette[16];
 75 
 76 /* 像素格式转换 */
 77 static inline unsigned int chan_to_field(u_int chan, struct fb_bitfield * bf)
 78 {
 79     chan &= 0xffff;                /* 保留该color的低16位 */
 80     chan >>= 16 - bf->length;    /* 裁剪到设定像素数据格式的该color长度 */
 81     return chan<<bf->offset;
 82 }
 83 /* 设置调色板 */
 84 static int s3c_lcdfb_setcolreg(unsigned regno, unsigned red, unsigned green,
 85                 unsigned blue, unsigned transp, struct fb_info *info)
 86 {
 87     unsigned int val;
 88     if(regno > 16)
 89         return 1;    /* unknown tybe */
 90     val = chan_to_field(red, &info->var.red);
 91     val |= chan_to_field(green, &info->var.green);
 92     val |= chan_to_field(blue, &info->var.blue);
 93     pseudo_palette[regno] = val;
 94     return 0;
 95 }
 96 
 97 static int lcd_init(void)
 98 {
 99     int ret = 0;
100     
101     /* 1.分配一个帧缓冲器 */
102     s3c_lcdfb = framebuffer_alloc(0, NULL);        /* sucess: return info; err: return NULL; */
103     if(!s3c_lcdfb)
104     {
105         printk(KERN_ERR "Unable to alloc s3c_lcdfb!\n");
106         goto err_fail1;
107     }
108     /* 2.设置帧缓冲器结构体 */
109     /* 2.1设置固定参数 */
110     strcpy(s3c_lcdfb->fix.id, "mylcd");    
111     //s3c_lcdfb->fix.smem_start     = ;        /* 帧缓冲器的起始地址(物理地址) */
112     s3c_lcdfb->fix.smem_len        = 320*240*16/8;            /* 帧缓冲器的长度 */
113     s3c_lcdfb->fix.type            = FB_TYPE_PACKED_PIXELS;
114     s3c_lcdfb->fix.visual        = FB_VISUAL_TRUECOLOR;    /* TFT真彩 */
115     s3c_lcdfb->fix.line_length    = 240*16/8;                /* length of a line in bytes */
116     /* 2.2设置可变参数 */
117     s3c_lcdfb->var.xres            = 240;        /* 可见分辨率 */
118     s3c_lcdfb->var.yres         = 320;
119     s3c_lcdfb->var.xres_virtual    = 240;        /* 虚拟分辨率 */
120     s3c_lcdfb->var.yres_virtual    = 320;
121     s3c_lcdfb->var.bits_per_pixel    = 16;    /* 16bpp */
122     /* RGB = 565 */
123     s3c_lcdfb->var.red.offset    = 11;        /* 开始位域偏移量 */
124     s3c_lcdfb->var.red.length    = 5;        /* 位域长度 */
125     s3c_lcdfb->var.green.offset = 5;    
126     s3c_lcdfb->var.green.length = 6;
127     s3c_lcdfb->var.blue.offset    = 0;
128     s3c_lcdfb->var.blue.length    = 5;
129     s3c_lcdfb->var.activate        = FB_ACTIVATE_NOW;
130     
131     /* 2.3设置操作函数 */
132     s3c_lcdfb->fbops    = &s3c_lcdfb_ops;
133     /* 2.4设置其余参数 */
134     //s3c_lcdfb->screen_base         = ;    /* 屏幕开始的虚拟地址 */
135     s3c_lcdfb->screen_size         = 320*240*16/8; /* Amount of ioremapped VRAM or 0 */ 
136     s3c_lcdfb->pseudo_palette    = pseudo_palette;
137     /* 3.设置硬件(LCD相关寄存器) */
138     /* 3.1设置用于LCD的GPIO引脚 */
139     /* 映射寄存器地址到虚拟内存 */
140     gpbcon = ioremap(0x56000010, 8);
141     gpbdat = gpbcon + 1;
142     gpccon = ioremap(0x56000020, 4);
143     gpdcon = ioremap(0x56000030, 4);
144     gpgcon = ioremap(0x56000060, 4);
145     /* LCD背光灯LED+-, 硬件自动连接到电源正负极,不管? 
146      * GPB1: 配置为输出引脚,高电平:背光供电;  低电平:不供电 
147      * GPBCON &= ~(3<<2);
148      * GPBCON |=  (1<<2);
149      */
150     *gpbcon &= ~(3<<2);
151     *gpbcon |=  (1<<2);
152     *gpbdat &= ~(1<<1);        /* 低电平,先不提供背光 */
153     /* S3C2440-LCD连接引脚状态控制 */
154     *gpccon = 0xaaaaaaaa;     /* GPIO管脚用于VD[7:0],LCDVF[2:0],VM,VFRAME,VLINE,VCLK,LEND */
155     *gpdcon = 0xaaaaaaaa;    /* GPIO管脚用于VD[23:8] */
156     /* GPG4: 配置为 LCD_PWREN 功能引脚 */
157     *gpgcon |= (3<<8);
158     
159     /* 3.2设置LCD控制器 */
160     plcd_regs = ioremap(0X4D000000, sizeof(struct lcd_regs));
161     /* LCDCON1位定义:
162      * CLKVAL   [17:8] 决定 VCLK 的频率和 CLKVAL[9:0]。TFT:VCLK = HCLK / [(CLKVAL + 1) × 2] (CLKVAL≥0);
163      * PNRMODE  [6:5] 11 = TFT LCD 面板;
164      * BPPMODE  [4:1] 选择 BPP(位每像素)模式:  1100 = TFT 的 16 bpp;  
165      * ENVID    [0]     0 = 禁止视频输出和 LCD 控制信号;  1 = 允许视频输出和 LCD 控制信号;
166      *
167      * //int CLKVAL = (double)HCLK/plcdparams->time_seq.vclk/2 - 1 + 0.5;
168      * //int CLKVAL = 5;        // jz2440  的4.3寸TFT屏幕;
169      * int CLKVAL = 7;            // mini2440的3.5寸TFT屏幕;
170      *  int bppmode =     plcdparams->bpp == 8?  0xb :\
171                     plcdparams->bpp == 16? 0xc :\
172                     0xd;    //0xd: 32/24bpp;
173      * LCDCON1 =  (CLKVAL<<8) | (3<<5) | (bppmode<<1);
174       */
175      plcd_regs->lcdcon1 =  (7<<8) | (3<<5) | (0xc<<1);
176     /* LCDCON2位定义:
177      * VBPD  [31:24]TFT:tvb-1, 垂直后沿为帧开始时,垂直同步周期后的的无效行数。0x00
178      * LINEVAL  [23:14]  xres-1 ,TFT/STN:此位决定了 LCD 面板的垂直尺寸。  0000000000
179      * VFPD  [13:6]TFT: tvf-1, 垂直前沿为帧结束时,垂直同步周期前的的无效行数。00000000
180      * VSPW  [5:0]TFT:  tvp-1, 通过计算无效行数垂直同步脉冲宽度决定 VSYNC 脉冲的高电平宽度。000000
181      
182      * LCDCON2 =     (plcdparams->time_seq.tvb - 1<<24) |\
183                 (plcdparams->time_seq.tvf - 1<<6)  |\
184                 (plcdparams->time_seq.tvp - 1<<0)  |\
185                 (plcdparams->yres - 1<<14);
186      */
187     plcd_regs->lcdcon2 =     (1<<24) | (319<<14) |(8<<6) | (1<<0);                
188     /* LCDCON3位定义:
189      * HBPD(TFT)[25:19] TFT:thb-1, 水平后沿为 HSYNC 的下降沿与有效数据的开始之间的 VCLK 周期数。0000000
190      * HOZVAL     [18:8]  TFT/STN:yres-1, 此位决定了 LCD 面板的水平尺寸。必须决定 HOZVAL 来满足 1 行的总字节为 4n 字节。
191      *             如果单色模式中 LCD 的 x 尺寸为 120个点,但不能支持 x=120,因为 1 行是由 16 字节(2n)所组成。
192      *            LCD面板驱动器将舍弃额外的 8 个点。00000000000
193      * HFPD(TFT)[7:0]   TFT:thf-1, 水平后沿为有效数据的结束与 HSYNC 的上升沿之间的 VCLK 周期数。0X00
194     
195      * LCDCON3 =     (plcdparams->time_seq.thb - 1<<19) |\
196                   (plcdparams->xres- 1<<8)             |\
197                   (plcdparams->time_seq.thf - 1<<0);
198       */
199      plcd_regs->lcdcon3 =     (34<<19) | (239<<8) | (39<<0);
200     /*
201      * LCDCON4位定义:
202      * HSPW(TFT)[7:0]TFT:通过计算 VCLK 的数水平同步脉冲宽度决定 HSYNC 脉冲的高电平宽度  0X00
203      
204      * LCDCON4 = (plcdparams->time_seq.thp - 1<<0);
205      */
206     plcd_regs->lcdcon4 = (4<<0);
207     /*
208      * LCDCON5位定义:
209      * FRM565     [11]TFT:此位选择 16 bpp 输出视频数据的格式, 0 = 5:5:5:1 格式  1 = 5:6:5 格式;   0
210      * INVVCLK    [10]STN/TFT:此位控制 VCLK 有效沿的极性,  0 = VCLK 下降沿取视频数据  1 = VCLK 上升沿取视频数据; 0
211      * INVVLINE   [9]STN/TFT:此位表明 VLINE/HSYNC 脉冲极性, 0 = 正常  1 = 反转;    0,
212      * INVVFRAME  [8]STN/TFT:此位表明 VFRAME/VSYNC 脉冲极性, 0 = 正常  1 = 反转;    0,
213      * INVVD      [7]STN/TFT:此位表明 VD(视频数据)脉冲极性, 0 = 正常  1 = 反转 VD;    0
214      * INVVDEN    [6]TFT:此位表明 VDEN 信号极性, 0 = 正常  1 = 反转;    0
215      * INVPWREN   [5]STN/TFT:此位表明 PWREN 信号极性, 0 = 正常  1 = 反转;    0
216      * INVLEND    [4]TFT:此位表明 LEND 信号极性, 0 = 正常  1 = 反转;    0
217      * PWREN      [3]STN/TFT:LCD_PWREN 输出信号使能/禁止, 0 = 禁止 PWREN 信号  1 = 允许 PWREN 信号;    0
218      * ENLEND     [2]TFT:LEND 输出信号使能/禁止, 0 = 禁止 LEND 信号  1 = 允许 LEND 信号;    0
219      * 
220      * BPP24BL    [12]TFT:此位决定 24 bpp 视频存储器的顺序, 0 = LSB 有效  1 = MSB 有效;    0
221      * BSWP       [1]STN/TFT:字节交换控制位, 0 = 交换禁止  1 = 交换使能;    0
222      * HWSWP      [0]STN/TFT:半字节交换控制位, 0 = 交换禁止  1 = 交换使能,    0
223      *
224      * 存储器数据格式(TFT)
225      * pixels_data_format =     plcdparams->bpp == 32 ? (0<<0) : \
226                                 plcdparams->bpp == 16 ? (1<<0) : \
227                                 (1<<1);        //8bpp;
228        LCDCON5 =     (plcdparams->pin_pol.vclk<<10) |\
229                     (plcdparams->pin_pol.hsync<<9) |\
230                     (plcdparams->pin_pol.vsync<<8) |\
231                     (plcdparams->pin_pol.rgb<<7)   |\
232                     (plcdparams->pin_pol.de<<6)    |\
233                     (plcdparams->pin_pol.pwren<<5) |\
234                     (1<<11) | pixels_data_format;
235       */
236      plcd_regs->lcdcon5 = (1<<11) | (0<<10) |(1<<9) | (1<<8) | (0<<7) |\
237                           (0<<6)  |    (0<<5) |(0<<1) | (1<<0);
238 
239     /* 3.3 配置显存 */
240     s3c_lcdfb->screen_base = dma_alloc_writecombine(NULL, s3c_lcdfb->fix.smem_len, (dma_addr_t *)&s3c_lcdfb->fix.smem_start, GFP_KERNEL);    /* 屏幕开始的虚拟地址 */
241     /* sucess: return info; err: return NULL; */
242     if(!s3c_lcdfb->screen_base)
243     {
244         printk(KERN_ERR "Unable to alloc the framebuffer of s3c_lcdfb->screen_base!\n");
245         goto err_fail2;
246     }
247     /*
248      * LCDSADDR1位定义:
249         LCDBANK  [29:21]
250         这些位表明系统存储器中视频缓冲器的 bank 位置的 A[30:22]。即使当
251         移动视口时也不能改变 LCDBANK 的值。LCD 帧缓冲器应该在 4MB
252         连续区域内,以保证当移动视口时不会改变 LCDBANK 的值。因此应
253         该谨慎使用 malloc()函数。
254         0x00
255         LCDBASEU  [20:0]
256         对于双扫描 LCD:这些位表明递增地址计数器的开始地址的 A[21:1],
257         它是用于双扫描 LCD 的递增帧存储器或单扫描 LCD 的帧存储器。
258         对于单扫描 LCD:这些位表明 LCD 帧缓冲器的开始地址的 A[21:1]。
259         0x000000
260     
261         addr = plcdparams->fb_base<<1;
262         addr >>= 2;
263         LCDSADDR1 = addr;        //帧缓冲器的开始地址;
264      */
265     plcd_regs->lcdsaddr1 = ((s3c_lcdfb->fix.smem_start<<1)>>2);
266     /*
267     LCDSADDR2  位  描述  初始状态
268      LCDBASEL  [20:0]
269      对于单扫描 LCD:这些位表明 LCD 帧缓冲器的结束地址的 A[21:1]。
270      LCDBASEL = ((帧结束地址) >> 1) + 1
271      = LCDBASEU + (PAGEWIDTH+OFFSIZE) × (LINEVAL+1);
272      0x0000
273     
274     //addr = plcdparams->fb_base + plcdparams->xres*plcdparams->yres*plcdparams->bpp/8;
275     //LCDSADDR2 = (addr>>1) + 1;        //帧缓冲器的结束地址;
276 
277     addr = plcdparams->fb_base + plcdparams->xres*plcdparams->yres*plcdparams->bpp/8;
278     addr >>= 1;
279     //addr &= 0x1fffff;    //有没有都一样!
280     LCDSADDR2 = addr; 
281     */
282     plcd_regs->lcdsaddr2 = ((s3c_lcdfb->fix.smem_start + s3c_lcdfb->fix.smem_len)>>1) & 0x1fffff;
283     plcd_regs->lcdsaddr3 = 240*16/16;    /* 虚拟屏的页宽度(单位: 半字节) */
284     /* 3.4启动LCD */
285     /* GPB0: 输出高电平,给LCD提供背光灯LED+-
286      * GPBDAT |= (1<<1);
287      */
288     *gpbdat |= (1<<1);
289     /* LCDCON1位定义:
290      * ENVID [0] 使能LCD信号输出:  0 = 禁止视频输出和 LCD 控制信号  1 = 允许视频输出和 LCD 控制信号 
291      * LCDCON1 |= (1<<0);
292      */
293     plcd_regs->lcdcon1 |= (1<<0);
294     /* LCDCON5位定义:
295      * PWREN [3] STN/TFT:LCD_PWREN 输出信号使能/禁止, 0 = 禁止 PWREN 信号  1 = 允许 PWREN 信号;    0 
296      * LCDCON5 |= (1<<3);
297      */
298     plcd_regs->lcdcon5 |= (1<<3);
299 
300     /* 4.注册帧缓冲器 */
301     ret = register_framebuffer(s3c_lcdfb);
302     if(ret < 0)
303     {
304         printk(KERN_ERR "Unable to register the s3c_lcdfb framebuffer device!\n");
305         goto err_fail3;
306     }
307     
308     return 0;
309  err_fail3:
310     unregister_framebuffer(s3c_lcdfb);
311     *gpbdat &= ~(1<<1);
312     plcd_regs->lcdcon1 &= ~(1<<0);
313     plcd_regs->lcdcon5 &= ~(1<<3);
314  err_fail2:
315     dma_free_writecombine(NULL, s3c_lcdfb->fix.smem_len, s3c_lcdfb->screen_base, s3c_lcdfb->fix.smem_start);
316     iounmap(plcd_regs);
317     iounmap(gpbcon);
318     iounmap(gpccon);
319     iounmap(gpdcon);
320     iounmap(gpgcon);
321  err_fail1:
322     framebuffer_release(s3c_lcdfb);
323     return ret;
324 }
325 
326 static void lcd_exit(void)
327 {
328     /* 1.注销帧缓冲器 */
329     unregister_framebuffer(s3c_lcdfb);
330     /* 2.禁止LCD使能 */
331     /* GPB0: 输出高电平,给LCD提供背光灯LED+-
332      * GPBDAT &= ~(1<<1);
333      */
334     *gpbdat &= ~(1<<1);
335     /* LCDCON1位定义:
336      * ENVID [0] 使能LCD信号输出:  0 = 禁止视频输出和 LCD 控制信号  1 = 允许视频输出和 LCD 控制信号 
337      * LCDCON1 &= ~(1<<0);
338      */
339     plcd_regs->lcdcon1 &= ~(1<<0);
340     /* LCDCON5位定义:
341      * PWREN [3] STN/TFT:LCD_PWREN 输出信号使能/禁止, 0 = 禁止 PWREN 信号  1 = 允许 PWREN 信号;    0     
342      * LCDCON5 &= ~(1<<3);
343      */
344     plcd_regs->lcdcon5 &= ~(1<<3);
345 
346     /* 3.释放显存 */
347     dma_free_writecombine(NULL, s3c_lcdfb->fix.smem_len, s3c_lcdfb->screen_base, s3c_lcdfb->fix.smem_start);
348     /* 4.释放寄存器映射的内存 */
349     iounmap(plcd_regs);
350     iounmap(gpbcon);
351     iounmap(gpccon);
352     iounmap(gpdcon);
353     iounmap(gpgcon);
354     /* 5.收回分配给帧缓冲器的空间 */
355     framebuffer_release(s3c_lcdfb);
356 }
357 
358 module_init(lcd_init);
359 module_exit(lcd_exit);

Makefile

 1 ifneq ($(KERNELRELEASE),)
 2     obj-m := lcd_6.o
 3 else
 4     KERN_DIR ?= /home/book/workbook/mini2440/systems/linux-2.6.22.6
 5 
 6     PWD = $(shell pwd)
 7 all:
 8     $(MAKE) -C $(KERN_DIR) M=$(PWD) modules
 9 clean:
10     $(MAKE) -C $(KERN_DIR) M=$(PWD) modules clean
11     rm -rf modules.order
12 
13 endif

 

posted @ 2018-12-21 20:10  大秦长剑  阅读(860)  评论(0编辑  收藏  举报