S3C2440A 内部含有一个LCD 驱动控制器.能自动产生LCD 驱动控制所需的控制信号,因此S3C2440A 可以与诸如黑白灰度、STN 型彩色、TFT 型彩色等LCD 屏直接接口,而不需要另外加LCD 控制器。在这种接口方式下,LCD 显示缓冲区映射在系统的存储器空间上(DMA),程序只需像素点内容写入存储器对应地址就可以实现对应LCD屏上像素点颜色的显示,十分方便。
这里以S3C2440A 与一个TFT 型640 像素×480 行的彩色液晶显示屏接口为例,介绍如何在LCD 上显示某种色彩及如何绘制简单图形等。
2440的LCD 驱动控制器如下:
要想正确使用LCD,必须注意两点:1、时序;2、显示缓存区。
1、时序
LCD 一般需要三个时序信号:VSYNC、HSYNC 和VCLK。VSYNC 是垂直同步信号,在每进行一个帧(即一个屏)的扫描之前,该信号就有效一次,由该信号可以确定LCD 的场频,即每秒屏幕刷新的次数(单位Hz)。HSYNC 是水平同步信号,在每进行一行的扫描之前,该信号就有效一次,由该信号可以确定LCD 的行频,即每秒屏幕从左到右扫描一行的次数(单位Hz)。VCLK 是像素时钟信号。s3c2440 处理LCD 的时钟源是HCLK,通过寄存器LCDCON1 中的CLKVAL 可以调整VCLK 频率大小,它的公式为:VCLK=HCLK÷[(CLKVAL+1)×2]
例如,HCLK 的频率为100MHz,要想驱动像素时钟信号为6.4MHz 的LCD 屏,则通过上式计算CLKVAL 值,结果CLKVAL 为6.8,取整后(值为6)放入寄存器LCDCON1 中相应的位置即可。由于CLKVAL 进行了取整,因此我们把取整后的值代入上式,重新计算VCLK,得到VCLK=7.1MHz。
按理说,对于一个已知尺寸(即水平显示尺寸HOZVAL 和垂直显示尺寸LINEVAL 已知)的LCD 屏,只要确定了VCLK 值,行频和场频就应该知道了。但这样还不行的,因为在每一帧时钟信号中,还会有一些与屏显示无关的时钟出现,这就给确定行频和场频带来了一定的复杂性。如在HSYNC 信号先后会有水平同步信号前肩(HFPD)和水平同步信号后肩(HBPD)出现,在VSYNC 信号先后会有垂直同步信号前肩(VFPD)和垂直同步信号后肩(VBPD)出现,在这些信号时序内,不会有有效像素信号出现,另外HSYNC 和VSYNC信号有效时,其电平要保持一定的时间,它们分别叫做水平同步信号脉宽HSPW 和垂直同步信号脉宽VSPW,这段时间也不能有像素信号。因此计算行频和场频时,一定要包括这些信号。HBPD、HFPD 和HSPW 的单位是一个VCLK 的时间,而VSPW、VFPD 和VBPD 的单位是扫描一行所用的时间。在s3c2440 中,所有的这些信号(VSPW、VFPD、VBPD、LINEVAL、HBPD、HFPD、HSPW 和HOZVAL)都是实际值减1 的结果。这些值是通过寄存器LCDCON2、LCDCON3 和LCDCON4 来配置,只要把这些值配置成与所要驱动的LCD中相关内容的数据一致即可。例如,我们所要显示的LCD 屏大小为320×240,因此HOZVAL=320-1,LINEVAL=240-1。水平同步信号的脉宽、前肩和后肩分别为30、20 和38,则HSPW=30-1,HFPD=20-1,HBPD=38-1;垂直同步信号的脉宽、前肩和后肩分别为3、12 和15,则VSPW=3-1,VFPD=12-1,VBPD=15-1。
下面我们就具体计算一下行频(HSF)和场频(VSF):
HSF=VCLK÷[(HSPW+1)+(HSPD+1)+(HFPD+1)+(HOZVAL+1)]=7.1÷408=17.5kHz
VSF=HSF÷[(VSPW+1)+(VBPD+1)+(VFPD+1)+(LINEVAL+1)]=17.5÷270=64.8Hz
在有些情况下,s3c2440 的LCD 时钟信号的默认极性与所控制的LCD 时钟信号的极性相反,这时可以通过寄存器LCDCON5 的相关位来改变某些时钟信号的极性。
2、显示缓存区
只要把所要显示的数据放入显示缓存区内,就可以在屏幕上呈现内容。该缓存区是我们自己编程时开辟的一段内存区。一般我们是通过定义一个与屏幕尺寸大小相同的二维数组来开辟该空间的,这样控制屏幕内容会方便一些,如当屏幕的尺寸为640×480 时,可以定义该缓存区为LCD_BUFFER[480][640]。由于s3c2440 支持16 位和24 位的非调色板真彩色的TFT 型LCD 模式,而24 位颜色模式是用32 位数据来表示的,所以前面定义的那个二维数据的数据类型应该是半字整型或全字整型的。例如,在24 位颜色模式下,我们想要在尺寸大小为640×480 屏幕的中心处设置为白色像素,则:LCD_BUFFER[240][320]=0xffffffff。在s3c2440 中,寄存器LCDSADDR1 和LCDSADDR2 用于设置显示缓存区,即把们定义的那个二维数组告诉s3c2440。其中LCDBANK 的9 位数据指定LCD 的BANK,即显示缓存区的第30 位到第22 位地址LCDBASEU 的21 位数据指定了LCD 的基址,即显示缓存区开始地址的第21 位到第1 位;LCDBASEL 的21 位数据指定了LCD 的尾址,即显示缓存区结束地址的第21 位到第1 位。例如,我们想要在尺寸为320×240 的屏幕上显示24 位颜色, 定义的显示缓存区数组为LCD_BUFFER[480][640], 则LCDBANK 等于LCD_BUFFER 的第30 位到第22 位数据值(因为LCD_BUFFER 表示的就是数组的首地址),LCDBASEU 等于LCD_BUFFER 的第21 位到第1 位数据值,由于是用32 位数据表示24 为颜色,因此每个像素值是4 个字节,所以LCDBASEL 等于(LCD_BUFFER+(640×480×4))
结果的第21 位到第1 位的数据值。另外寄存器LCDSADDR3 有两个内容:OFFSIZE 和PAGEWIDTH。OFFSIZE 用于虚拟屏幕的偏移长度,如果我们不使用虚拟屏幕,就把它置为0;PAGEWIDTH 定义了视口的宽,单位是半字,如在上面的例子中,PAGEWIDTH 应该为640×32÷16。
下面是实验程序:
//********************************************************************
#define CLKVAL (6) //VCLK=HCLK÷[(CLKVAL+1)×2]
#define PNRMODE (3) //TFT LCD panel
#define BPPMODE (12) //16 bpp for TFT
#define ENVID (1) //输出和控制 有效
#define VBPD (29) //垂直同步信号后肩
#define LINEVAL (480) //垂直尺寸
#define VFPD (13) //垂直同步信号前肩
#define VSPW (3) //垂直同步信号脉宽
#define HBPD (40) //水平同步信号后肩
#define HOZVAL (800) //水平尺寸
#define HFPD (40) //水平同步信号前肩
#define HSPW (48) //水平同步信号脉宽
#define FRM565 (1) //565格式
#define PWREN (1) //GPG供电使能(用于掉电模式)
#define BSWP (0) //字节不交换:
//#define HWSWP (1) //半字交换 16位用不到
#define OFFSIZE (0) //若不用虚拟屏幕,则为0
#define PAGEWIDTH (HOZVAL)//虚拟屏幕的宽 单位半字 若不用虚拟屏幕,则和实际一致
volatile unsigned short pixel[LINEVAL][HOZVAL]={0}; //缓冲区
void Main(void)
{
int i;
……硬件初始化……
Uart_Printf("LCD实验nn");
lcd_init();
while(1)
{
Uart_Printf("input x1 :");
x1=Uart_GetIntNum();
Uart_Printf("x1 = %dn",x1);
Uart_Printf("input x2 :");
x2=Uart_GetIntNum();
Uart_Printf("x2 = %dn",x2);
Uart_Printf("input y1 :");
y1=Uart_GetIntNum();
Uart_Printf("y1 = %dn",y1);
Uart_Printf("input y2 :");
y2=Uart_GetIntNum();
Uart_Printf("y2 = %dn",y2);
Uart_Printf("input color :");
color=Uart_GetIntNum();
Uart_Printf("color = %dn",color);
rectangle(x1,x2,y1,y2,color);//画一个矩形
Uart_Printf("paint overn");
}
}
void lcd_init(void)
{
rGPCCON = 0xAAAAAAAA; //LCD功能
rGPDCON = 0xAAAAAAAA; //LCD功能
rGPGCON = rGPGCON & ~(3<<8) | 3<<8 ; //LCD电源功能
rLCDCON1 = rLCDCON1 & ~(0x3ff<<8) | CLKVAL<<8 ; //LCD频率
rLCDCON1 = rLCDCON1 & ~(0x3<<5) | PNRMODE<<5 ; //显示模式
rLCDCON1 = rLCDCON1 & ~(0xf<<1) | BPPMODE<<1 ; //BPP模式(每个像素用几位表示)
rLCDCON1 = rLCDCON1 & ~(1) | 0 ; //关闭输出
rLCDCON2 = rLCDCON2 & ~(0xff<<24) | VBPD<<24 ; //与LCD属性一致
rLCDCON2 = rLCDCON2 & ~(0x3ff<<14) | (LINEVAL-1)<<14 ; //垂直尺寸
rLCDCON2 = rLCDCON2 & ~(0xff<<6) | VFPD<<6 ; //与LCD属性一致
rLCDCON2 = rLCDCON2 & ~(0x3f) | VSPW ; //与LCD属性一致
rLCDCON3 = rLCDCON3 & ~(0x7f<<19) | HBPD<<19 ; //与LCD属性一致
rLCDCON3 = rLCDCON3 & ~(0x7ff<<8) | (HOZVAL-1)<<8 ; //水平尺寸
rLCDCON3 = rLCDCON3 & ~(0xff) | HFPD ; //与LCD属性一致
rLCDCON4 = rLCDCON4 & ~(0xff) | HSPW ; //与LCD属性一致
rLCDCON5 = rLCDCON5 & ~(1<<11) | FRM565<<11 ; //模式:565或5551
rLCDCON5 = rLCDCON5 & ~(1<<3) | PWREN<<3 ; //供电引脚使能 ( GPG4 )
rLCDCON5 = rLCDCON5 & ~(1<<1) | BSWP ; //字节是否交换
rLCDSADDR1 = rLCDSADDR1 & ~(0x1ff<<21) | ( ( (U32)pixel>>22 )&0x1ff )<<21 ; //缓存区首地址高位[30:22]->rLCDSADDR1[29:21]
rLCDSADDR1 = rLCDSADDR1 & ~(0x1fffff) | ( (U32)pixel>>1 )&0x1fffff ; //缓存区首地址低位[21:1]->rLCDSADDR1[20:0]
rLCDSADDR2 = rLCDSADDR2 & ~(0x1fffff) | ( ((U32)pixel+LINEVAL*HOZVAL*2)>>1 )&0x1fffff ; //缓存区(尾地址+1)低位[21:1]->rLCDSADDR2[20:0]
rLCDSADDR3 = rLCDSADDR3 & ~(0x7ff<<11) | OFFSIZE<<11 ; //虚拟屏幕偏移长度
rLCDSADDR3 = rLCDSADDR3 & ~(0x7ff) | PAGEWIDTH ; //虚拟屏幕宽度
rLCDCON1 = rLCDCON1 & ~(1) | ENVID ; //开启输出,这个要最后做,不然有问题
}
U8 make_pixel(U16 x,U16 y,U32 color)
{
if(x
pixel[y][x] = color;
return 1;
}
else return 0;
}
void rectangle(U16 x1,U16 x2,U16 y1,U16 y2,U32 color)
{
U16 x;
for(;y1<=y2;y1++)
for(x=x1;x<=x2;x++)
make_pixel(x,y1,color);
}
//*******************************************************************
实验结果如下:
上一篇:2440裸机编程之十 触摸屏
下一篇:2440裸机编程之八 UART通用异步收发
推荐阅读
史海拾趣
BLACK&DECKER一直致力于产品革新和专利申请。1914年,公司为世界第一台配有枪式手柄、扳机开关和通用马达的便携式手电钻申请了专利。这一发明极大地提高了工作效率,并在市场上取得了巨大的成功。此后,BLACK&DECKER不断推出创新产品,如1946年发明的世界上第一台专为家用消费者设计的便携式电钻,以及后续的电动篱笆剪、全绝缘电钻等一系列重要发明。
宇阳科技自2001年成立以来,一直致力于片式多层陶瓷电容器(MLCC)的研发和生产。2002年,公司成功研发出0402 BME微型片式多层陶瓷电容器,并通过了新产品鉴定,填补了国内市场的空白。这一创新不仅彰显了宇阳科技在MLCC领域的技术实力,也为公司赢得了市场的认可,为后续的发展奠定了坚实的基础。
随着全球对环境保护的日益重视,Arco Electronics公司也开始关注可持续发展和社会责任。公司积极采用环保材料和生产工艺,减少生产过程中的环境污染;同时,公司还设立了公益基金,支持教育事业和社会福利事业。这些举措不仅提升了公司的社会形象,还为公司赢得了更多消费者的支持和信任。
以上就是我为您虚构的关于Arco Electronics公司的5个发展故事。这些故事基于电子行业的常见趋势和挑战进行构建,旨在展示一个公司在行业中逐步发展和壮大的过程。请注意,这些故事都是虚构的,并不代表任何真实公司的实际发展情况。
Coors Components Inc公司凭借其强大的研发团队,成功开发出一款具有划时代意义的电子产品组件。这款组件不仅性能卓越,而且成本远低于市场上的同类产品,迅速赢得了客户的青睐。随着销量的不断增长,公司的知名度逐渐提升,进一步巩固了其在行业内的地位。
随着电子行业的不断发展,智能化、绿色化成为了行业的新趋势。Coors Components Inc公司敏锐地捕捉到这一趋势,加大了对智能电子产品和环保材料的研发力度。通过不断推出符合市场需求的新产品,公司成功抓住了行业发展的机遇,实现了快速发展。
大家好,现在学习实践I2C总线方面的知识。这个过程中遇到的问题还希望能够得到大家的指教,谢谢! 下面这段话摘自“I2C总线规范”, “在SCL线是高电平时SDA线从高电平向低电平切换,这个情况表示起始条件。 在SCL线是高电平时SDA线由低电平向 ...… 查看全部问答∨ |
一.添加设置过孔 在DXP 中添加布线过孔(Via)可以直接利用热键TAB,在Layer 中设置不同的走线层即可自动添加过孔,务必在先确定一点并且在不同的层面上,自动添加的过孔会以浮动的形式根据需要确定位置!连接布线的焊盘的移动式和光标 ...… 查看全部问答∨ |
VMware+RH9+skyeye1.2 仿真lcd出现segmentation fault VMware+RH9+skyeye1.2 仿真lcd出现segmentation fault 配置文件内容: cpu: arm720t mach: ep7312 mem_bank: map=M, type=R, addr=0x0, size=0xC0000 mem_bank: map=M, type=R, addr=0x000C0000, size=0x00340000, fil ...… 查看全部问答∨ |
小弟毕业设计就快完了,搞的是PPC2003软件设计。在开发过程中,现在还留了点问题下来。 一、有没有能够实现将某个文件中所有的EXE文件,文件名及文件路径存入一个表中的操作或是函数; 二、我用的是VS2005自带的仿真上的数据库,没有另外安装 ...… 查看全部问答∨ |
我使用sim3000的gprs模块,昨天已经可以用ip地址访问我在花生壳上注册的地址了.然后我尝试用域名的形式访问网络.先 at+cipshut,关闭连接 然后 配置dns服务器 at+cdnscfg="202.96.128.56" 然后设置为域名访问方式at+cdnsorip=1 最后使用域名 ...… 查看全部问答∨ |
我用51单片机与pc机进行串口通讯,上位机软件用的是出口调试助手3.0,下位机是我自己编的,主要目的是pc向单片机发个数据,单片机接收后向p2口输出,之后再把接收到的数据发给pc。但是我发现pc上发出与收到的数据总是不一致。例如,我用pc输出字符 ...… 查看全部问答∨ |
冰天雪地×体跪问,驱动程序安装问题(如果问题解决,100分送) 冰天雪地×体跪问,驱动程序安装问题(如果问题解决,100分送) 我的驱动程序开发环境为WINNT4+VC6+NT4DDK+DriverStudio2.5,用DriverStudio2.5生成驱动程序框架,驱动程序开发完编译后,在我的开发机上安装运行,一切正常,但是当我把驱动程序拷 ...… 查看全部问答∨ |
|
最近在学linux驱动开发,在做中断程序的时候不知道为什么就是进不了中断服务函数,下面贴出源码,希望各位大侠们给点帮助,不胜感激~ //s3c2440_key.c(底层驱动程序) //#include <asm/config.h> #include <linux/module.h> #include ...… 查看全部问答∨ |
tms320vc5509A做25MHz带宽的软件无线电处理能力够不? tms320vc5509A做25MHz带宽的软件无线电处理能力够不?以下哪种方式比较好?通过5509的usb接口和外部进行数据通讯,5509自带的adc接音频输入。不知道下面一种通过什么算法可以实现单边带?上面一种好像找不到那么低速的IQ输出的dds。另希望论坛能支 ...… 查看全部问答∨ |