[STM8L]TAB段式LCD液晶驅(qū)動程序
STM8L152XX系列帶有片上段式LCD驅(qū)動程序,這為低成本應用和高密度系統(tǒng)設計提供了保證,利用片上LCD驅(qū)動模塊,可以有效的控制系統(tǒng)整體功耗,簡化系統(tǒng)結(jié)構(gòu),從整體來說可靠性得到提高。
此處不介紹LCD驅(qū)動模塊的原理以及驅(qū)動時序,請參考STM8原版英文說明文檔,已描述的很詳細,以下介紹其寄存器的配置方法以及編程方法。
時鐘,系統(tǒng)時鐘同樣用來產(chǎn)生LCD驅(qū)動時鐘,通過時鐘模塊配置:
CLK_PCKENR2|=S3; //LCD 使能LCD模塊時鐘
CLK_CRTCR=S7|S6|S5|S1; //RTC&LCD->FCLK/128 注意RTC和LCD是共同一路時鐘
以上配置根據(jù)實際時鐘進行調(diào)整,我在此處采用FCLK=HSI=16MHZ,所以LCDclk=16M/128=125KHZ
我的LCD為六個數(shù)字的段式LCD,1/3偏壓方式,4根COM線,12根COM線,這兩個參數(shù)請讀者自己查找自己的LCD資料找到,對于驅(qū)動LCD來說這兩個參數(shù)最重要,以下為寄存器配置:
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=244Frame=244/4=61HZ
LCD_PM0=0xFF;
LCD_PM1=0x0F;
LCD_CR3|=S6;
首先由偏壓方式?jīng)Q定了驅(qū)動到LCD段碼上的電壓種類,占空比(標準并非如此翻譯)Duty值決定掃過每根COM線的時序比例,由于我將VLCD與VCC接在一起了,所以選擇外部電源參考3.3V,若選擇內(nèi)部,則可以進一步選擇最高輸出電壓大小,實測發(fā)現(xiàn)選大些對比度可提高一些。LCD_FRQ用于配置掃描更新頻率,具體計算不想說,文檔里都有。最后是配置那些接在LCD上的COM線和SEG線為LCD驅(qū)動復用有效模式,否則仍可以作為IO口使用,最后開啟LCD驅(qū)動模塊掃描。
配置完以上寄存器之后,LCD模塊已開始工作,它是通過從LCD_RAM0-LCD_RAM12這一組寄存器來控制顯示內(nèi)容的,這時向LCD_RAM0-LCD_RAM12寫入數(shù)據(jù)會發(fā)現(xiàn)有段碼顯示在LCD上,作為應用層,需要找到這種關系。
查手里這塊LCD資料列出段碼表如下所示:
/*-----------------------------------------------------
SEG:01234567891011
1DX22DX33DX14D4P5D5P6D6P
1E1C2E2C3E3C4E4C5E5C6E6C
1G1B2G2B3G3B4G4B5G5B6G6B
1F1A2F2A3F3A4F4A5F5A6F6A
CODE:AFBGCEPD
-----------------------------------------------------*/
于是我把一個字節(jié)最高位至最低位從A段到D段按如上CODE順序進行排列,并得到段表碼如下:
{
uint16T,SEG[4];
uint8i,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]
}
以上這段程序?qū)CD_DisplayBuffer[]中的六個字符解碼后寫入LCD模塊的顯示緩沖區(qū)中,最終顯示成相應字符,這其中用每個字符的最高位代表是否含有小數(shù)點位,若為高則點亮相當?shù)男?shù)點,否則關閉。至于LCD_RAM的更新和拆分方法,此外不再描述,文檔中已相當詳細。
圍繞以上刷新程序,可得到如下常用方法:
voidLCD_Clear(uint8Index)
{
uint8i;
if(Index==0xFF)for(i=0;i<6;i++)LCD_DisplayBuffer[i]='U'-'A';
elseLCD_DisplayBuffer[Index]='U'-'A';
LCD_SetSegValue();
}
//寫顯示緩沖區(qū)點
voidLCD_ShowSpecial(uint8Saddr,uint8Char)
{
if(Char>0)LCD_DisplayBuffer[Saddr]|=0x80;
elseLCD_DisplayBuffer[Saddr]&=0x7F;
}
//顯示字符
voidLCD_ShowChar(uint8Saddr,uint8Char)
{
LCD_DisplayBuffer[Saddr]&=0x80;
LCD_DisplayBuffer[Saddr]|=Char;
LCD_SetSegValue();
}
//顯示字符串
voidLCD_ShowString(uint8Saddr,void*Text)
{
uint8T,P,*Str;
Str=(uint8*)Text;
while(*Str>0){
T=*Str++;
P=LCD_DisplayBuffer[Saddr];
LCD_DisplayBuffer[Saddr++]=(P&0x80)|(T-'A');
}
LCD_SetSegValue();
}
//顯示數(shù)字
voidLCD_ShowNumber(uint8Saddr,uint16Number,uint8Length)
{
uint8P;
Saddr+=Length-1;
while(Length--){
P=LCD_DisplayBuffer[Saddr];
LCD_DisplayBuffer[Saddr]=(P&0x80)|(Number%10);
Saddr--;Number/=10;
}
LCD_SetSegValue();
}
由以上函數(shù)庫,可以方便的顯示出如LCD_ShowString(0,"BCDEF")(顯示“12345"),LCD_ShowNumber(0,1244,4)(顯示"1244"),等等。配合一些簡單的數(shù)據(jù)結(jié)構(gòu),便可得到一個相對復雜點的菜單操作界面。
<