[STM8L]TAB段式LCD液晶驱动程序

    STM8L152XX系列带有片上段式LCD驱动程序,这为低成本应用和高密度系统设计提供了保证,利用片上LCD驱动模块,可以有效的控制系统整体功耗,简化系统结构,从整体来说可靠性得到提高。

    此处不介绍LCD驱动模块的原理以及驱动时序,请参考STM8原版英文说明文档,已描述的很详细,以下介绍其寄存器的配置方法以及编程方法。

    时钟,系统时钟同样用来产生LCD驱动时钟,通过时钟模块配置:

    CLK_PCKENR2|=S3;   //LCD        使能LCD模块时钟

    CLK_CRTCR=S7|S6|S5|S1;    //RTC&LCD->FCLK/128      注意RTC和LCD是共同一路时钟

    以上配置根据实际时钟进行调整,我在此处采用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^5*16=128000/512=244 Frame=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:  0  1  2  3  4  5  6  7  8  9  10 11
      1D X2 2D X3 3D  X1 4D 4P 5D 5P 6D 6P
      1E 1C 2E 2C 3E  3C 4E 4C 5E 5C 6E 6C
      1G 1B 2G 2B 3G  3B 4G 4B 5G 5B 6G 6B
      1F 1A 2F 2A 3F  3A 4F 4A 5F 5A 6F 6A
CODE: A  F  B  G  C  E  P  D
-----------------------------------------------------*/
  于是我把一个字节最高位至最低位从A段到D段按如上CODE顺序进行排列,并得到段表码如下:
const uint8 LCD_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  [依次和上面的字符相对应]
   为六个字符定义显示内容缓冲区:
uint8 LCD_DisplayBuffer[6];
void LCD_SetSegValue(void)
{
 uint16 T,SEG[4];
 uint8 i,j,Code[6];
 for(i=0;i<6;i++){
  Code[5-i]=LCD_CodeTable[LCD_DisplayBuffer[i]&0x7F];
  if(LCD_DisplayBuffer[i]&0x80)Code[5-i]|=0x02;
 }
 for(i=0;i<4;i++){
  for(T=0,j=0;j<6;j++){
   T<<=2;
   T|=(Code[j]&0x03);
   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的更新和拆分方法,此外不再描述,文档中已相当详细。

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

//清显示
void LCD_Clear(uint8 Index)
{
	uint8 i;
	if(Index==0xFF)for(i=0;i<6;i++)LCD_DisplayBuffer[i]='U'-'A';
	else LCD_DisplayBuffer[Index]='U'-'A';
	LCD_SetSegValue();
}

//写显示缓冲区点
void LCD_ShowSpecial(uint8 Saddr,uint8 Char)
{
	if(Char>0)LCD_DisplayBuffer[Saddr]|=0x80;
	else      LCD_DisplayBuffer[Saddr]&=0x7F;
}

//显示字符
void LCD_ShowChar(uint8 Saddr,uint8 Char)
{
	LCD_DisplayBuffer[Saddr]&=0x80;
	LCD_DisplayBuffer[Saddr]|=Char;
	LCD_SetSegValue();
}

//显示字符串
void LCD_ShowString(uint8 Saddr,void *Text)
{
	uint8 T,P,*Str;
	Str=(uint8 *)Text;
	while(*Str>0){
		T=*Str++;
		P=LCD_DisplayBuffer[Saddr];
		LCD_DisplayBuffer[Saddr++]=(P&0x80)|(T-'A');
	}
	LCD_SetSegValue();
}

//显示数字
void LCD_ShowNumber(uint8 Saddr,uint16 Number,uint8 Length)
{
	uint8 P;
	Saddr+=Length-1;
	while(Length--){
			P=LCD_DisplayBuffer[Saddr];
			LCD_DisplayBuffer[Saddr]=(P&0x80)|(Number%10);
			Saddr--;Number/=10;			
	}
	LCD_SetSegValue();
}


    由以上函数库,可以方便的显示出如LCD_ShowString(0,"BCDEF")(显示“12345"),LCD_ShowNumber(0,1244,4)(显示"1244"),等等。配合一些简单的数据结构,便可得到一个相对复杂点的菜单操作界面。

 


你可能感兴趣的:(嵌入式软件)