基于單片機的LCD時序圖的底層驅動編寫
//=======================================================
void LCD_DataWrite(unsigned char Dat)
{
LCD_A0 = 1; //A0 置高,示意進行顯存數(shù)據(jù)操作
LCD_RW = 0; //RW 置低,示意進行寫入操作
LCD_EP = 0; //EP 先置低,以便后面產生跳變沿
LCD_CS = 0; //片選 CS 置低
DAT_PORT = Dat; //裝載數(shù)據(jù)置總線
LCD_EP = 1; //產生有效的跳變沿
LCD_CS = 1; //片選置高
}
//=====================================================
// 函數(shù): unsigned char LCD_DataRead(void)
// 描述: 從 LCD 中的顯示緩沖 RAM 當中讀一個字節(jié)的顯示數(shù)據(jù)
// 參數(shù): 無
// 返回: 讀出的數(shù)據(jù),
//=====================================================
unsigned char LCD_DataRead(void)
{
unsigned char Read_Data;
DAT_PORT = 0xff; //51 的端口想要輸入前,要先給端口全置 1
LCD_A0 = 1; //A0 置高,示意進行顯存數(shù)據(jù)操作
LCD_RW = 1; //RW 置高,示意進行讀出操作
LCD_EP = 0; //EP 先置低,以便后面產生跳變沿
LCD_CS = 0; //片選 CS 置低
LCD_EP = 1; //產生有效的跳變沿
LCD_EP = 0;
Read_Data = DAT_PORT; //讀出數(shù)據(jù)
LCD_CS = 1; //片選置高
return Read_Data; //返回讀到的數(shù)據(jù)
}
以上便是要介紹的最基本的時序操作程序,它們幾乎是整個 LCD 驅動程序當中與底層硬件打交道的代碼了,這樣的話,當要改變驅動 LCD 的 MCU 端口時或者換用別的 MCU 來驅動 LCD 時,基本上只需要在這些代碼里作一下修改即可。
關于讀 LCD 狀態(tài)
而在一般的 LCD 模塊當中,還有一個功能同樣重要,就是讀 LCD 狀態(tài);可以通過此操作獲取當前 LCD 模塊的忙狀態(tài)以及一些相關的狀態(tài)信息,當 LCD 模塊正處于忙狀態(tài)時,則不宜對它進行數(shù)據(jù)的寫入或讀出操作(有很多較老式的 LCD 控制器規(guī)定在忙的狀態(tài)下時不允許寫入或讀出數(shù)據(jù))。
所以在很多 LCD 的驅動程序當中,會在寄存器寫入、數(shù)據(jù)寫入/讀出的操作前加入讀取 LCD狀態(tài)并判別忙狀態(tài)的代碼;這點可以參考網(wǎng)上流傳的很多 LCD 驅動程序。不過,對于 MzL02這樣的較新出的 LCD 控制器來說,已經對忙狀態(tài)不是很在乎了,或者說影響已經很小甚至沒有了;所以我們在前面的代碼當中并沒有加入這樣的代碼。至于有沒有必要加讀狀態(tài)判忙的代碼,要視具體的 LCD 控制器而定。
關于時序的時間要求
時序的一個非常重要的數(shù)據(jù)就是類似上圖中標出的tAS88之類的時間長短要求,只是上圖中并沒有標出它們的具體最大最小值要求而已;但在編寫這類的時序接口程序時它們還是非常重要的,當然還要看 MCU 的端口操作速度以及 MCU 的指令執(zhí)行速度。打個比方,有的時序里就會有要求某些信號的電平保持最小寬度,而如果 MCU 的指令執(zhí)行速度以及端口操作速度非??斓脑挘托枰们樵谶B續(xù)操作端口的代碼之間加入適量的延時(通用用空操作來代替,具體多少個多少時長視具體的 MCU 以及 LCD 控制器而定)以保證該信號的脈沖寬度滿足要求。
在本文的所列出的源代碼當中,并沒有如前所述的為時序的要求而插入空操作或延時處理,因為 MCU 的速度并不是非??欤瑳r且現(xiàn)在的 LCD 控制器的總線速度都挺快的了,沒有必要加入而已。
評論