历史上的今天
今天是:2024年11月08日(星期五)
2021年11月08日 | S3C2440裸机学习[2] - LCD驱动原理及代码分析[二]
2021-11-08 来源:eefocus
下面看看2440test里面的lcd.c文件
static void PutPixel(U32 x,U32 y,U16 c) { if(x } |
很容易发现TFT LCD上显示单个像素的函数实际上很简洁
看来似乎只需要LCD_BUFFER[(y)][(x)] = c这一句话
下面就来分析下,是如何通过这一句话来实现在LCD上显示单个像素的
先分析下LCD_Init()即LCD初始化函数
rLCDCON1 = (LCD_PIXCLOCK << 8) | (3 << 5) | (12 << 1); |
LCDCON1 0x4d000000
#define LCD_WIDTH 240
#define LCD_HEIGHT 320
#define LCD_PIXCLOCK 4
#define LCD_RIGHT_MARGIN 36
#define LCD_LEFT_MARGIN 19
#define LCD_HSYNC_LEN 5
#define LCD_UPPER_MARGIN 1
#define LCD_LOWER_MARGIN 5
#define LCD_VSYNC_LEN 1
CLKVAL[17:8] = 4
TFT: VCLK = HCLK / [(CLKVAL+1) * 2] ( CLKVAL>=0 )
MMODE[7] = 0
PNRMODE[6:5] = 11 TFT LCD panel
BPPMODE[4:1] = 1100 16bpp for TFT
ENVID[0] = 0 Disable
rLCDCON2 = (LCD_UPPER_MARGIN << 24) | ((LCD_HEIGHT - 1) << 14) | (LCD_LOWER_MARGIN << 6) | (LCD_VSYNC_LEN << 0); |
LCDCON2 0x4d000004
VBPD = 1
VBPD(vertical back porch):表示在一帧图像开始时,垂直同步信号以后的无效的行数,对应驱动中的upper_margin
LINVAL = 240 – 1
LINVAL:LCD屏的垂直大小
VFPD = 5
VFPD(vertical front porch):表示在一帧图像结束后,垂直同步信号以前的无效的行数,对应驱动中的lower_margin
VSPW = 1
VSPW(vertical sync pulse width):表示垂直同步脉冲的宽度,用行数计算,对应驱动中的vsync_len
rLCDCON3 = (LCD_RIGHT_MARGIN << 19) | ((LCD_WIDTH - 1) << 8) | (LCD_LEFT_MARGIN << 0); |
LCDCON3 0x4d000008
HBPD = 36
HBPD(horizontal back porch):表示从水平同步信号开始到一行的有效数据开始之间的VCLK的个数,对应驱动中的left_margin
HOZVAL = 320 – 1
HOZVAL:LCD屏的水平大小
HFPD = 19
HFPD(horizontal front porth):表示一行的有效数据结束到下一个水平同步信号开始之间的VCLK的个数,对应驱动中的right_margin
rLCDCON4 = (13 << 8) | (LCD_HSYNC_LEN << 0); |
LCDCON4 0x4d00000c
MVAL = 13
HSPW = 5
HSPW(horizontal sync pulse width):表示水平同步信号的宽度,用VCLK计算,对应驱动中的hsync_len
# define LCD_CON5 ((1<<11) | (1 << 9) | (1 << 8) | (1 << 3) | (1 << 0)) |
LCDCON5 0x4d000010
HWSWP = 1 Swap Enable
PWREN = 1 Enable PWREN signal
INVVFRAME = 1 VFRAME/VSYNC pulse polarity Inverted 选择负极性脉冲
INVVLINE = 1 VLINE/HSYNC pulse polarity Inverted 选择负极性脉冲
FRM565 = 1 5:6:5 Format
rLCDINTMSK |= 3; |
INT_FrSyn = 1 LCD frame synchronized interrupt Masked
INT_FiCnt = 1 LCD FIFO interrupt Masked
rTCONSEL &= (~7); rTCONSEL &= ~((1<<4) | 1); |
MODE_SEL = 0 Sync mode
RES_SEL = 0 320 x 240
LPC_EN = 0 LPC3600 Disable
rTPAL = 0x0; |
Temporary palette register enable bit Disable
volatile static unsigned short LCD_BUFFER[SCR_YSIZE][SCR_XSIZE]; #define LCD_ADDR ((U32)LCD_BUFFER) #define M5D(n) ((n)&0x1fffff) rLCDSADDR1 = ((LCD_ADDR >> 22) << 21) | ((M5D(LCD_ADDR >> 1)) << 0); rLCDSADDR2 = M5D((LCD_ADDR + LCD_WIDTH * LCD_HEIGHT * 2) >> 1); rLCDSADDR3 = LCD_WIDTH; |
LCDSADDR1 0x4d000014 帧缓冲起始寄存器1
LCDBANK[29:21] = (U32)LCD_BUFFER >> 22
These bits indicate A[30:22] of the bank location for the video buffer in the system memory. LCDBANK value cannot be changed even when moving the view port. LCD frame buffer should be within aligned 4MB region, which ensures that LCDBANK value will not be changed when moving the view port. So, care should be taken to use the malloc() Function
系统内存地址A[30:22]处的Bank位置为图像缓冲。LCDBANK的值在视图移动的值在视图移动时不能改变,LCD帧缓冲应该在4MB区域对齐,保证LCDBANK的值在移动视图时不会改变。
LCDBASEU[20:0] = ((U32)LCD_BUFFER >> 1)&0x1fffff
For dual-scan LCD : These bits indicate A[21:1] of the start address of the upper address counter, which is for the upper frame memory of dual scan LCD or the frame memory of single scan LCD.
For single-scan LCD : These bits indicate A[21:1] of the start address of the LCD frame buffer.
双扫描:表明高地址计数器的起始地址A[21:1],用于LCD双扫描的上部帧内存或者单扫描的帧内存
单扫描:表明LCD帧缓冲的起始地址A[21:1]
LCDSADDR2 0x4d000018帧缓冲起始寄存器2
LCDBASEL[20:0] = ((LCD_ADDR + LCD_WIDTH * LCD_HEIGHT * 2) >> 1)& 0x1fffff
= (LCD_ADDR >> 1 + LCD_WIDTH * LCD_HEIGHT)& 0x1fffff
For dual-scan LCD: These bits indicate A[21:1] of the start address of the lower address counter, which is used for the lower frame memory of dual scan LCD.
For single scan LCD: These bits indicate A[21:1] of the end address of the LCD frame buffer.
LCDBASEL = ((the frame end address) >>1) + 1
= LCDBASEU + (PAGEWIDTH+OFFSIZE) x (LINEVAL+1)
双扫描:表明低地址计数器的起始地址A[21:1],用于LCD双扫描的下部帧内存或者单扫描的帧内存
单扫描:表明LCD帧缓冲的结束地址A[21:1]
LCDSADDR3 0x4d00001c帧缓冲起始寄存器3
OFFSIZE = 0
PAGEWIDTH = 320 虚拟屏页宽(半字数量) 定义了帧中的视图域宽度
程序分析至此,大概已经清楚是如何通过LCD_BUFFER[(y)][(x)] = c来实现在LCD上显示单个像素了。
就是在设置好各个LCD寄存器之后,通过将LCD_BUFFER地址与LCDBANK以及LCDBASEU、LCDBASEL对应之后,通过改变LCD_BUFFER里不同单元存储的值(即像素的颜色),即可在LCD相应位置上做出显示。
那么在应用不同LCD的时候,只需对LCDCONx以及LCDSADDRx做出相应的配置,在创建一个数组,做出上述的地址映射即可。
关于VCLK计算,由于配置的是TFT,可用到公式VCLK = HCLK / [(CLKVAL+1) * 2] ( CLKVAL>=0 )
设置的FLK为400MHz,HCLK为100MHz,CLKVAL = 4,因此VLCK = 10MHz
史海拾趣
|
功率MOSFET的测试设备有那些? 我想问的是在做MOSFET的Datasheet时做精确测试的测试设备, 0 d2 {) _) z9 O9 P) _ T5 _0 c! y包括DC和AC全部,我先谢谢了!!… 查看全部问答> |
|
Verilog菜鸟请教各位大侠:为什么仿真的时候+1加到一个值后会突然乱掉 各位大侠,问个关于Verilog的问题,为什么按照下面的程序 always @(negedge RST or posedge CLK or posedge rise or posedge fall) begin if((RST == 1\'b0)||(rise == 1\'b1)||(fall ...… 查看全部问答> |
|
最近在做FIR数字滤波器,方法是通过Matlab按相应的滤波要求设计出滤波器,并用Matlab中DSP builder生成出VHDL文件,最后在ALTERA的FPGA开发板中来实现.但是在最后一步出现了问题..我的滤波器已做好,在最后生成VHDL时,Convert MDL to VHDL ...… 查看全部问答> |
|
大家好! 小弟最近在搞wince5.0的battary驱动,利用DS2786来检测电池状态,现在我不用该芯片来检测,直接就在BatteryPDDInitialize()里用 sps.ACLineStatus = AC_LI ...… 查看全部问答> |
|
因为工作方向的原因,我和一些行内的朋友都是集中在无线通信及嵌入式等领域,对于传统的Windows开发反倒不熟悉。 目前我们团队成员的经验集中在半导体(芯片原厂)、无线/有线通信(ZTE、华为、凯明等)等行业的领先企业,至于嵌入式开发,这个不 ...… 查看全部问答> |
|
我在某一2440的平台上烧写自己移植好的文件系统时出现这样的问题 :Failed to execute /linuxrc. Attempting defaults... 是什么原因导致的呢? [ 本帖最后由 heningbo 于 2011-10-8 13:35 编辑 ]… 查看全部问答> |
|
穷怕了,缺芯币,卖些钱。其实TI例程包中都有 第一步: 把BOOT_ETH烧写到芯片中。 [ 本帖最后由 hlx3012 于 2012-3-1 22:25 编辑 ]… 查看全部问答> |
|
【Launchpad 心得】μC/OS-II移植到Launchpad 小弟成功把μC/OS-II移植到Launchpad上,特提供完整的IAR工程文件供大家参考 ,创建了2个任务控制led闪烁,快要考试了潜水一段时间,如果帮助到大家请自觉祝小弟考试顺利… 查看全部问答> |
|
【学习Sitara™ AM335x】- Beaglebone环境的搭建方法 Basic Requirements ARM Cross CompilerUbuntu (11.04 (Natty Narwhal) minimum): sudo apt-get install gcc-arm-linux-gnueabiLinaro: https://launchpad.net/linaro-toolchain-binaries复制代码BootloaderDas U-Boot – the Universal Boot Load ...… 查看全部问答> |




