STM8单片机对LCD模块的驱动 下载本文

内容发布更新时间 : 2024/5/13 22:45:44星期一 下面是文章的全部内容请认真阅读。

STM8单片机对LCD模块的驱动

STM8L152XX系列带有片上段式LCD驱动程序,这为低成本应用和高密度

系统设计提供了保证,利用片上LCD驱动模块,可以有效的控制系统整体功耗,简化系统结构,从整体来说可靠性得到提高。

此处不介绍LCD驱动模块的原理以及驱动时序,请参考STM8原版英文说明文档,已描述的很详细,以下介绍其寄存器的配置方法以及编程方法。 时钟,系统时钟同样用来产生LCD驱动时钟,通过时钟模块配置: CLK_PCKENR2|=S3; //LCD 使能LCD模块时钟

CLK_CRTCR=S7|S6|S5|S1; //RTC\以上配置根据实际时钟进行调整,我在此处采用FCLK=HSI=16MHZ,所以LCDclk=16M/128=125KHZ

我的LCD为六个数字的段式LCD,1/3偏压方式,4根COM线,12根COM线,这两个参数请读者自己查找自己的LCD资料找到,对于驱动LCD来说这两个参数最重要,以下为寄存器配置:

LCD_CR1=S5|S2|S1;//1/3偏压1/4占空比 LCD_CR2=S6|S4|S0;//3.3V

LCD_FRQ=5《《4;//FCK=125000/2??*16=128000/512=244Frame=244/4=61HZ LCD_PM0=0xFF; LCD_PM1=0x0F; LCD_CR3|=S6;

首先由偏压方式决定了驱动到LCD段码上的电压种类,占空比(标准并非如此翻译)Duty值决定扫过每根COM线的时序比例,由于我将VLCD与VCC接在一起了,所以选择外部电源参考3.3V,若选择内部,则可以进一步选择最高输出电压大小,实测发现选大些对比度可提高一些。LCD_FRQ用于配置扫描更新频率,具体计算不想说,文档里都有。最后是配置那些接在LCD上的COM线和SEG线为LCD驱动复用有效模式,否则仍可以作为IO口使用,最后开启LCD驱动模块扫描。

配置完以上寄存器之后,LCD模块已开始工作,它是通过从LCD_RAM0-LCD_RAM12

这一组寄存器来控制显示内容的,这时向LCD_RAM0-LCD_RAM12写入数据会发现有段码显示在LCD上,作为应用层,需要找到这种关系。 查手里这块LCD资料列出段码表如下所示: /*----------------------------------------------------- SEG:01234567891011

1DX22DX33DX14D4P5D5P6D6P 1E1C2E2C3E3C4E4C5E5C6E6C 1G1B2G2B3G3B4G4B5G5B6G6B 1F1A2F2A3F3A4F4A5F5A6F6A CODE:AFBGCEPD

-----------------------------------------------------*/

于是我把一个字节最高位至最低位从A段到D段按如上CODE顺序进行排列,并得到段表码如下:

constuint8LCD_CodeTable[]={0xED,0x28,0xB5,0xB9,0x78,0xD9,0xDD,0xA8,0xFD,0xF9,0xFC,0x5D,0x15,0x3D,0xD5,0xD4,0x5C,0x10,0xC5,0xA9,0x00};这些段码表分别对应于以下字符:0,1,2,3,4,5,6,7,8,9,A,b,c,d,E,F,h,-,[,],[注,最后一个为空格]为编程方便,我对字符进行编码:A,B,C,D,E,F,G,H,I,J,K,L,M,N,O,P,Q,R,S,T,U[依次和上面的字符相对应]为六个字符定义显示内容缓冲区:uint8LCD_DisplayBuffer[6];voidLCD_SetSegValue(void) {

uint16T,SEG[4]; uint8i,j,Code[6]; for(i=0;i《6;i++){

Code[5-i]=LCD_CodeTable[LCD_DisplayBuffer[i] if(LCD_DisplayBuffer[i] }

for(i=0;i《4;i++){

for(T=0,j=0;j《6;j++){ T《《=2; T|=(Code[j] Code[j]》》=2; }

SEG[i]=T; }

LCD_RAM0=(uint8)(SEG[0]);//COM0-》B[7:0] LCD_RAM1=(uint8)(SEG[0]》》8);//COM0-》B[11:8] LCD_RAM3=(uint8)(SEG[1]《《4);//COM1-》B[3:0]-》H LCD_RAM4=(uint8)(SEG[1]》》4);//COM1-》B[11:4] LCD_RAM7=(uint8)(SEG[2]);//COM2-》B[7:0] LCD_RAM8=(uint8)(SEG[2]》》8);//COM2-》B[11:8] LCD_RAM10=(uint8)(SEG[3]《《4);//COM3-》B[3:0]-》H LCD_RAM11=(uint8)(SEG[3]》》4);//CoM3-》B[11:4] }

以上这段程序将LCD_DisplayBuffer[]中的六个字符解码后写入LCD模块的显示缓冲区中,最终显示成相应字符,这其中用每个字符的最高位代表是否含有小数点位,若为高则点亮相当的小数点,否则关闭。至于LCD_RAM的更新和拆分方法,此外不再描述,文档中已相当详细。

围绕以上刷新程序,可得到如下常用方法: //清显示

voidLCD_Clear(uint8Index) { uint8i;

if(Index==0xFF)for(i=0;i《6;i++)LCD_DisplayBuffer[i]=‘U’-‘A’;