1. LCD/LCM的基本概念
液晶顯示器(Liquid Crystal Display: LCD)的構(gòu)造是在兩片平行的玻璃當(dāng)中放置液態(tài)的晶體,兩片玻璃中間有許多垂直和水平的細(xì)小電線(xiàn),透過(guò)通電與否來(lái)控制桿狀水晶分子改變方向,將光線(xiàn)折射出來(lái)產(chǎn)生畫(huà)面。
LCM(LCD Module)即LCD顯示模組、液晶模塊,是指將液晶顯示器件,連接件,控制與驅(qū)動(dòng)等外圍電路,PCB電路板,背光源,結(jié)構(gòu)件等裝配在一起的組件。
在平時(shí)的學(xué)習(xí)開(kāi)發(fā)中,我們一般使用的是LCM,帶有驅(qū)動(dòng)IC和LCD屏幕等多個(gè)模塊。
2. FSMC的基本概念
在STM32上開(kāi)發(fā)LCD顯示,可以有兩種方式來(lái)對(duì)LCD進(jìn)行操作,一種是通過(guò)普通的IO口,連接LCM的相應(yīng)引腳來(lái)進(jìn)行操作,第2種是通過(guò)FSMC來(lái)進(jìn)行操作。
可變靜態(tài)存儲(chǔ)控制器(Flexible Static Memory Controller: FSMC) 是STM32系列中內(nèi)部集成256 KB以上FlaSh,后綴為xC、xD和xE的高存儲(chǔ)密度微控制器特有的存儲(chǔ)控制機(jī)制。之所以稱(chēng)為“可變”,是由于通過(guò)對(duì)特殊功能寄存器的設(shè)置,F(xiàn)SMC能夠根據(jù)不同的外部存儲(chǔ)器類(lèi)型,發(fā)出相應(yīng)的數(shù)據(jù)/地址/控制信號(hào)類(lèi)型以匹配信號(hào)的速度,從而使得STM32系列微控制器不僅能夠應(yīng)用各種不同類(lèi)型、不同速度的外部靜態(tài)存儲(chǔ)器,而且能夠在不增加外部器件的情況下同時(shí)擴(kuò)展多種不同類(lèi)型的靜態(tài)存儲(chǔ)器,滿(mǎn)足系統(tǒng)設(shè)計(jì)對(duì)存儲(chǔ)容量、產(chǎn)品體積以及成本的綜合要求。
FSMC有很多優(yōu)點(diǎn):
1. 支持多種靜態(tài)存儲(chǔ)器類(lèi)型。STM32通過(guò)FSMC可以與SRAM、ROM、PSRAM、NOR Flash和NANDFlash存儲(chǔ)器的引腳直接相連。
2. 支持豐富的存儲(chǔ)操作方法。FSMC不僅支持多種數(shù)據(jù)寬度的異步讀/寫(xiě)操作,而且支持對(duì)NOR、PSRAM、NAND存儲(chǔ)器的同步突發(fā)訪(fǎng)問(wèn)方式。
3. 支持同時(shí)擴(kuò)展多種存儲(chǔ)器。FSMC的映射地址空間中,不同的BANK是獨(dú)立的,可用于擴(kuò)展不同類(lèi)型的存儲(chǔ)器。當(dāng)系統(tǒng)中擴(kuò)展和使用多個(gè)外部存儲(chǔ)器時(shí),F(xiàn)SMC會(huì)通過(guò)總線(xiàn)懸空延遲時(shí)間參數(shù)的設(shè)置,防止各存儲(chǔ)器對(duì)總線(xiàn)的訪(fǎng)問(wèn)沖突。
4. 支持更為廣泛的存儲(chǔ)器型號(hào)。通過(guò)對(duì)FSMC的時(shí)間參數(shù)設(shè)置,擴(kuò)大了系統(tǒng)中可用存儲(chǔ)器的速度范圍,為用戶(hù)提供了靈活的存儲(chǔ)芯片選擇空間。
5. 支持代碼從FSMC擴(kuò)展的外部存儲(chǔ)器中直接運(yùn)行,而不需要首先調(diào)入內(nèi)部SRAM。
FSMC包含兩類(lèi)控制器:
1. 1個(gè)NOR閃存/SRAM控制器,可以與NOR閃存、SRAM和PSRAM存儲(chǔ)器接口。
2. 1個(gè)NAND閃存/PC卡控制器,可以與NAND閃存、PC卡,CF卡和CF+存儲(chǔ)器接口。
控制器產(chǎn)生所有驅(qū)動(dòng)這些存儲(chǔ)器的信號(hào)時(shí)序:
1. 16位數(shù)據(jù)線(xiàn),用于連接8位或16位的存儲(chǔ)器;
2. 26位地址線(xiàn),最多可連續(xù)64MB的存儲(chǔ)器(這里不包括片選線(xiàn));
3. 5位獨(dú)立的片選信號(hào)線(xiàn);
4. 1組適合不同類(lèi)型存儲(chǔ)器的控制信號(hào)線(xiàn):
- 控制讀/寫(xiě)操作
- 與存儲(chǔ)器通信,提供就緒/繁忙信號(hào)和中斷信號(hào)
- 與所用配置的PC卡接口:PC存儲(chǔ)卡、PC I/O卡和真正的IDE接口
從FSMC的角度看,可以把外部存儲(chǔ)器劃分為固定大小為256MB的4個(gè)存儲(chǔ)塊
· 存儲(chǔ)塊1用于訪(fǎng)問(wèn)最多4個(gè)NOR閃存或者PSRAM存儲(chǔ)設(shè)備。這個(gè)存儲(chǔ)區(qū)被劃分為4個(gè)NOR/PSRAM區(qū),并有4個(gè)專(zhuān)用的片選。
· 存儲(chǔ)塊2和3用于訪(fǎng)問(wèn)NAND閃存設(shè)備,每個(gè)存儲(chǔ)塊連接一個(gè)NAND閃存。
· 存儲(chǔ)塊4用于訪(fǎng)問(wèn)PC卡設(shè)備
每一個(gè)存儲(chǔ)塊上的存儲(chǔ)器類(lèi)型是由用戶(hù)在配置寄存器中定義的。
注意:FSMC只是提供了一個(gè)控制器,并不提供相應(yīng)的存儲(chǔ)設(shè)備,至于外設(shè)接的是什么設(shè)備,完全是由用戶(hù)自己選擇,只要能用于FSMC控制,就可以,像本次實(shí)驗(yàn)中,我們接的就是LCM。
3. 本例中FSMC的使用
由于本例只是利用FSMC對(duì)LCM進(jìn)行操作,因此不用完全懂得FSMC的所有功能,而是懂得一部分相應(yīng)的操作即可。
1. FSMC包括哪幾個(gè)部分
FSMC包含以下4個(gè)模塊:
· AHB接口(包含F(xiàn)SMC配置寄存器)
· NOR閃存和PSRAM控制器
· NAND閃存和PC卡控制器
· 外部設(shè)備接口
需要注意的是,F(xiàn)SMC可以請(qǐng)求AHB進(jìn)行數(shù)據(jù)寬度操作。如果AHB操作的數(shù)據(jù)寬度大于外部設(shè)備(NOR或NAND或LCD)的寬度,此時(shí)FSMC將AHB操作分割成幾個(gè)連續(xù)的較小的數(shù)據(jù)寬度,以適應(yīng)外部設(shè)備的數(shù)據(jù)寬度。
2. FSMC對(duì)外部設(shè)備的地址映像
FSMC對(duì)外部設(shè)備的地址映像從0x6000 0000開(kāi)始,到0x9FFF FFFF結(jié)束,一共4個(gè)地址塊,每個(gè)地址塊256MB,而每個(gè)地址塊又分成4個(gè)分地址塊,大小為64MB。對(duì)于NOR的地址映像來(lái)說(shuō),我們可以通過(guò)選擇HADDR[27:26] 來(lái)確定當(dāng)前使用的是哪個(gè)64M的分地址塊。而這四個(gè)分存儲(chǔ)塊的片選,則使用 NE[4:1]來(lái)選擇。數(shù)據(jù)線(xiàn)/地址線(xiàn)/控制線(xiàn)是共享的。
這里的HADDR 是需要轉(zhuǎn)換到外部設(shè)備的內(nèi)部AHB地址線(xiàn),每個(gè)地址對(duì)應(yīng)一個(gè)字節(jié)單元。因此,若外部設(shè)備的地址寬度是8位的,則HADDR[25:0]與STM32的CPU引腳 FSMC_A[25:0]一一對(duì)應(yīng),最大可以訪(fǎng)問(wèn)64M字節(jié)的空間。若外部設(shè)備的地址寬度是16位的,則是HADDR[25:1]與STM32的CPU引腳FSMC_A[24:0]一一對(duì)應(yīng)。在應(yīng)用的時(shí)候,可以將FSMC_A總線(xiàn)連接到存儲(chǔ)器或其他外設(shè)的地址總線(xiàn)引腳上。
4. ILI9325
由于我們使用的是奮斗STM32 V3開(kāi)發(fā)板,其內(nèi)部自帶的是一個(gè)LCM,產(chǎn)品的編號(hào)是:QD024CPS25-36AV0,其中的詳細(xì)規(guī)格參數(shù)可以參考QD024CPS25-36AV0規(guī)格書(shū)中的記載。而LCM中的驅(qū)動(dòng)IC就是采用的ILI9325。
ILI9325的功能很多,在此無(wú)法一一說(shuō)明,但是參考ILI9325的Datasheet我們發(fā)現(xiàn)有幾個(gè)引腳還是非常重要的,而只要操作好了這幾個(gè)引腳,基本上就可以實(shí)現(xiàn)簡(jiǎn)單的對(duì)LCM的控制了。
nCS: IC的片選信號(hào)。如果是低電平,則ILI9325是被選中,并且可以進(jìn)行操作,如果是高電平,這不被選中。
RS: 寄存器選擇信號(hào)。如果是低電平,則選擇的是索引或者狀態(tài)寄存器,如果是高電平,則選擇控制寄存器。
nWR/SCL: 寫(xiě)使能信號(hào),低電平有效。
nRD: 讀使能信號(hào),低電平有效。
以上內(nèi)容是從ILI9325的Datasheet里面找到的,但是根據(jù)我的實(shí)際操作發(fā)現(xiàn),似乎高電平也是有效的。而且,不管是高電平還是低電平,都可以成功驅(qū)動(dòng)LCD,如果有了解情況的可以討論一下。
ILI9325的寄存器非常多,詳細(xì)的各個(gè)寄存器的功能請(qǐng)參考ILI9325的Datasheet。在對(duì)ILI9325進(jìn)行操作時(shí),應(yīng)該先寫(xiě)地址,然后再寫(xiě)數(shù)據(jù),設(shè)置好各個(gè)寄存器之后,ILI9325就可以開(kāi)始工作了。
5. 電路設(shè)計(jì)
1. 信號(hào)線(xiàn)的連接
STM32F10x FSMC有4個(gè)不同的banks,每一個(gè)64MB,可支持NOR以及其他類(lèi)似的存儲(chǔ)器。這些外部設(shè)備的地址線(xiàn)、數(shù)據(jù)線(xiàn)和控制線(xiàn)是共享的。每個(gè)設(shè)備的訪(fǎng)問(wèn)時(shí)通過(guò)片選信號(hào)來(lái)決定的,而每次只能訪(fǎng)問(wèn)一個(gè)設(shè)備。我們的LCM就是連接在NOR的bank上面。
FSMC_D[15:0]:16bit的數(shù)據(jù)總線(xiàn),連接ILI9325的數(shù)據(jù)線(xiàn);
FSMC_NEx:分配給NOR的256MB的地址空間還可以分為4個(gè)banks,每一個(gè)區(qū)用來(lái)分配一個(gè)外設(shè),這4個(gè)外設(shè)分別就是NE1-NE4;
FSMC_NOE:輸出使能,連接ILI9325的nRD引腳;
FSMC_NWE:寫(xiě)使能,連接ILI9325的nWR引腳;
FSMC_Ax:用在LCD顯示RAM和寄存器之間進(jìn)行選擇的地址線(xiàn),這個(gè)和ILI9325的RS引腳相連。該線(xiàn)可用任意一根地址線(xiàn),范圍是FSMC_A[25:0]。當(dāng)RS=0時(shí),表示讀寫(xiě)寄存器,RS=1時(shí),表示讀寫(xiě)數(shù)據(jù)RAM。
其實(shí)關(guān)于RS的表述也并不完全準(zhǔn)確,應(yīng)該這么理解,RS=0的時(shí)候,向這個(gè)地址寫(xiě)的數(shù)表示了選擇什么寄存器進(jìn)行操作,然而要對(duì)寄存器進(jìn)行什么操作,則要看當(dāng)RS=1時(shí),送入的數(shù)據(jù)了。
關(guān)于地址的計(jì)算,如果我們選擇NOR的第一個(gè)存儲(chǔ)區(qū),并且使用FSMC_A16來(lái)控制ILI9325的RS引腳,則如果要訪(fǎng)問(wèn)寄存器地址(RS=0),那么地址是0x6000 0000(起始地址),如果要訪(fǎng)問(wèn)數(shù)據(jù)區(qū)(RS=1),那么基地址應(yīng)該是0x6002 0000。
有人會(huì)問(wèn),為什么不是0x6001 0000呢?因?yàn)镕SMC_A16=1。因?yàn)樵谇拔闹幸呀?jīng)說(shuō)過(guò),若外部設(shè)備的地址寬度是16位的,則是HADDR[25:1]與STM32的CPU引腳FSMC_A[24:0]一一對(duì)應(yīng)。也就是說(shuō),內(nèi)部產(chǎn)生的地址應(yīng)該要左移一位,F(xiàn)SMC_A16=1,代表著第17位為1,而不是第16位為1。如果外部設(shè)備的地址寬度是8位的話(huà),則不會(huì)出現(xiàn)這個(gè)問(wèn)題。
再舉一個(gè)例子,如果選擇NOR的第4個(gè)存儲(chǔ)區(qū),使用FSMC_A0來(lái)控制RS引腳,則訪(fǎng)問(wèn)數(shù)據(jù)區(qū)的地址為0x6000 0002,訪(fǎng)問(wèn)LCD寄存器的地址為:0x6000 0000。
2. 時(shí)序問(wèn)題
一般使用模式2來(lái)做LCD的接口控制,不使用外擴(kuò)模式。并且讀寫(xiě)操作的時(shí)序一樣。此種情況下,我們需要使用3個(gè)參數(shù):ADDSET、DATAST、ADDHOLD。時(shí)序的計(jì)算需要根據(jù)NOR閃存存儲(chǔ)器的特性和STM32F10x的時(shí)鐘HCLK來(lái)計(jì)算這些參數(shù)。
寫(xiě)或讀訪(fǎng)問(wèn)時(shí)序是存儲(chǔ)器片選信號(hào)的下降沿與上升沿之間的時(shí)間,這個(gè)時(shí)間可以由FSMC時(shí)序參數(shù)的函數(shù)計(jì)算得到:
寫(xiě)/讀訪(fǎng)問(wèn)時(shí)間 = ((ADDSET + 1) + (DATAST + 1)) × HCLK
在寫(xiě)操作中,DATAST用于衡量寫(xiě)信號(hào)的下降沿與上升沿之間的時(shí)間參數(shù):
寫(xiě)使能信號(hào)從低變高的時(shí)間 = t WP = DATAST × HCLK
為了得到正確的FSMC時(shí)序配置,下列時(shí)序應(yīng)予以考慮:
最大的讀/寫(xiě)訪(fǎng)問(wèn)時(shí)間、不同的FSMC內(nèi)部延遲、不同的存儲(chǔ)器內(nèi)部延遲
因此得到:
((ADDSET + 1) + (DATAST + 1)) × HCLK = max (t WC , t RC )
DATAST × HCLK = tWP
DATAST必須滿(mǎn)足:
DATAST = (tAVQV+ tsu(Data_NE) + tv(A_NE) )/HCLK – ADDSET – 4
由于我沒(méi)有找到ILI9325的這些時(shí)序的參數(shù),所以就參考了一些以前別人寫(xiě)的程序里面的時(shí)序配置:
當(dāng) HCLK 的頻率是 72MHZ,使用模式 B,則有如下時(shí)序:
地址建立時(shí)間:0x1
地址保持時(shí)間:0x0
數(shù)據(jù)建立時(shí)間:0x5
6. 程序編寫(xiě)步驟
對(duì)于程序的編寫(xiě),一般步驟是:
1. 初始化RCC;
2. 初始化GPIO;
3. 初始化FSMC;
4. 初始化LCD;
5. 往GRAM里面寫(xiě)入顯示數(shù)據(jù)。
其中RCC、GPIO、FSMC的初始化函數(shù)在STM32的固件庫(kù)中已經(jīng)有相應(yīng)的函數(shù),在此就不一一贅述了,如果有不懂的,可以參考以前我寫(xiě)的學(xué)習(xí)筆記。FSMC的初始化參數(shù)很多,而且基本上可以通用,因此在此也不對(duì)每一個(gè)參數(shù)具體有什么用進(jìn)行解釋了,一般來(lái)說(shuō),用通用參數(shù)就足夠普通的開(kāi)發(fā)了。
而對(duì)LCD的初始化,則需要自己編寫(xiě)相應(yīng)的代碼?;驹瓌t是,首先向寄存器地址寫(xiě)入需要操作的寄存器地址(代碼),然后再根據(jù)Datasheet,向數(shù)據(jù)區(qū)地址寫(xiě)入相應(yīng)的數(shù)據(jù),以實(shí)現(xiàn)某些操作。具體的操作在ILI9325的Datasheet 第8節(jié)Register Descriptions中,有詳細(xì)的解釋。而LCD的初始化只要按照Datasheet里面的,把每一個(gè)寄存器都給配置好了,就沒(méi)有問(wèn)題了。而這些寄存器的配置,大部分都是通用的,只是有一些屏幕方向選擇,坐標(biāo)系等會(huì)略有差別。
LCD配置好之后,就可以往GRAM里面寫(xiě)入圖像數(shù)據(jù)了,在這里推薦一個(gè)軟件“Image2LCD”,這個(gè)軟件能讀取圖像,然后生成C代碼的數(shù)據(jù),只要將這些生成的代碼直接寫(xiě)入GRAM中,就可以顯示出圖像了。不過(guò)要記住,在圖像轉(zhuǎn)換的時(shí)候,輸出數(shù)據(jù)類(lèi)型選擇“C語(yǔ)言數(shù)組”,掃描模式選擇“水平掃描”,輸出灰度“16位真彩色”,最大寬度和高度“320”“240”勾選“高位在前(MSB First)”。這些配置都是和ILI9325的寄存器配置相對(duì)應(yīng)的,如果說(shuō)ILI9325的配置和本文中的不一樣,則需要相應(yīng)的選擇其他的選項(xiàng)。
7. 程序源代碼
main.c文件中的代碼:
#include "stm32f10x_lib.h"
#include "stm32f10x_lcd.h"
extern unsigned char LCD_Image_BIT[];
extern unsigned char LCD_Image_HIT[];
void RCC_cfg();
void FSMC_cfg();
void LCD_cfg();
void GPIO_cfg();
void LCD_Show(unsigned char * LCD_Image);
int main()
{
RCC_cfg();
GPIO_cfg();
FSMC_cfg();
LCD_cfg();
while(1)
{
LCD_Show(LCD_Image_HIT);
Delay(100000000);
LCD_Show(LCD_Image_BIT);
Delay(100000000);
}
}
//RCC時(shí)鐘配置
void RCC_cfg()
{
//定義錯(cuò)誤狀態(tài)變量
ErrorStatus HSEStartUpStatus;
//將RCC寄存器重新設(shè)置為默認(rèn)值
RCC_DeInit();
//打開(kāi)外部高速時(shí)鐘晶振
RCC_HSEConfig(RCC_HSE_ON);
//等待外部高速時(shí)鐘晶振工作
HSEStartUpStatus = RCC_WaitForHSEStartUp();
if(HSEStartUpStatus == SUCCESS)
{
//設(shè)置AHB時(shí)鐘(HCLK)為系統(tǒng)時(shí)鐘
RCC_HCLKConfig(RCC_SYSCLK_Div1);
//設(shè)置高速AHB時(shí)鐘(APB2)為HCLK時(shí)鐘
RCC_PCLK2Config(RCC_HCLK_Div1);
//設(shè)置低速AHB時(shí)鐘(APB1)為HCLK的2分頻
RCC_PCLK1Config(RCC_HCLK_Div2);
//設(shè)置FLASH代碼延時(shí)
FLASH_SetLatency(FLASH_Latency_2);
//使能預(yù)取指緩存
FLASH_PrefetchBufferCmd(FLASH_PrefetchBuffer_Enable);
//設(shè)置PLL時(shí)鐘,為HSE的9倍頻 8MHz * 9 = 72MHz
RCC_PLLConfig(RCC_PLLSource_HSE_Div1, RCC_PLLMul_9);
//使能PLL
RCC_PLLCmd(ENABLE);
//等待PLL準(zhǔn)備就緒
while(RCC_GetFlagStatus(RCC_FLAG_PLLRDY) == RESET);
//設(shè)置PLL為系統(tǒng)時(shí)鐘源
RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK);
//判斷PLL是否是系統(tǒng)時(shí)鐘
while(RCC_GetSYSCLKSource() != 0x08);
}
//打開(kāi)GPIO時(shí)鐘,復(fù)用功能
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA | RCC_APB2Periph_GPIOB | RCC_APB2Periph_GPIOC | RCC_APB2Periph_GPIOD | RCC_APB2Periph_GPIOE | RCC_APB2Periph_AFIO, ENABLE);
//打開(kāi)FSMC時(shí)鐘
RCC_AHBPeriphClockCmd(RCC_AHBPeriph_FSMC, ENABLE);
}
//FSMC配置
void FSMC_cfg()
{
FSMC_NORSRAMInitTypeDef FSMC_NORSRAMInitStructure;
FSMC_NORSRAMTimingInitTypeDef p;
//設(shè)置地址建立時(shí)間
p.FSMC_AddressSetupTime = 0x02;
//設(shè)置地址保持時(shí)間
p.FSMC_AddressHoldTime = 0x00;
//設(shè)置數(shù)據(jù)建立時(shí)間
p.FSMC_DataSetupTime = 0x05;
//總線(xiàn)返轉(zhuǎn)時(shí)間
p.FSMC_BusTurnAroundDuration = 0x00;
//時(shí)鐘分頻
p.FSMC_CLKDivision = 0x00;
//數(shù)據(jù)保持時(shí)間
p.FSMC_DataLatency = 0x00;
//設(shè)置FSMC訪(fǎng)問(wèn)模式
p.FSMC_AccessMode = FSMC_AccessMode_B;
//選擇設(shè)置的BANK以及片選信號(hào)(BANK1中的第一個(gè)block)
FSMC_NORSRAMInitStructure.FSMC_Bank = FSMC_Bank1_NORSRAM1;
//設(shè)置是否數(shù)據(jù)地址總線(xiàn)時(shí)分復(fù)用(No)
FSMC_NORSRAMInitStructure.FSMC_DataAddressMux = FSMC_DataAddressMux_Disable;
//設(shè)置存儲(chǔ)器類(lèi)型(NOR)
FSMC_NORSRAMInitStructure.FSMC_MemoryType = FSMC_MemoryType_NOR;
//設(shè)置數(shù)據(jù)寬度(16bit)
FSMC_NORSRAMInitStructure.FSMC_MemoryDataWidth = FSMC_MemoryDataWidth_16b;
//設(shè)置是否使用迸發(fā)訪(fǎng)問(wèn)模式(連續(xù)讀寫(xiě)模式)(No)
FSMC_NORSRAMInitStructure.FSMC_BurstAccessMode = FSMC_BurstAccessMode_Disable;
//設(shè)置WAIT信號(hào)的有效電平(低電平有效)
FSMC_NORSRAMInitStructure.FSMC_WaitSignalPolarity = FSMC_WaitSignalPolarity_Low;
//設(shè)置是否使用還回模式(No)
FSMC_NORSRAMInitStructure.FSMC_WrapMode = FSMC_WrapMode_Disable;
//設(shè)置WAIT信號(hào)有效時(shí)機(jī)(在wait狀態(tài)之前)
FSMC_NORSRAMInitStructure.FSMC_WaitSignalActive = FSMC_WaitSignalActive_BeforeWaitState;
//設(shè)置是否使能寫(xiě)操作(Yes)
FSMC_NORSRAMInitStructure.FSMC_WriteOperation = FSMC_WriteOperation_Enable;
//設(shè)置是否使用WAIT信號(hào)(No)
FSMC_NORSRAMInitStructure.FSMC_WaitSignal = FSMC_WaitSignal_Disable;
//設(shè)置是否使用擴(kuò)展模式(讀寫(xiě)時(shí)序相互獨(dú)立)(No)
FSMC_NORSRAMInitStructure.FSMC_ExtendedMode = FSMC_ExtendedMode_Disable;
//設(shè)置是否使用異步等待信號(hào)(No)
FSMC_NORSRAMInitStructure.FSMC_AsyncWait = FSMC_AsyncWait_Disable;
//設(shè)置是否使用迸發(fā)寫(xiě)模式(No)
FSMC_NORSRAMInitStructure.FSMC_WriteBurst = FSMC_WriteBurst_Disable;
//設(shè)定讀寫(xiě)時(shí)序
FSMC_NORSRAMInitStructure.FSMC_ReadWriteTimingStruct = &p;
//設(shè)定寫(xiě)時(shí)序
FSMC_NORSRAMInitStructure.FSMC_WriteTimingStruct = &p;
FSMC_NORSRAMInit(&FSMC_NORSRAMInitStructure);
//使能Bank1中的block1
FSMC_NORSRAMCmd(FSMC_Bank1_NORSRAM1, ENABLE);
}
//GPIO配置
void GPIO_cfg()
{
GPIO_InitTypeDef GPIO_InitStructure;
//背光控制
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_13;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(GPIOD, &GPIO_InitStructure);
//LCD復(fù)位
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1;
GPIO_Init(GPIOE, &GPIO_InitStructure);
//打開(kāi)FSMC的數(shù)據(jù)端口D[15:0]
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1 | GPIO_Pin_8 | GPIO_Pin_9 |
GPIO_Pin_10 | GPIO_Pin_14 | GPIO_Pin_15;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
GPIO_Init(GPIOD, &GPIO_InitStructure);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_7 | GPIO_Pin_8 | GPIO_Pin_9 | GPIO_Pin_10 |
GPIO_Pin_11 | GPIO_Pin_12 | GPIO_Pin_13 | GPIO_Pin_14 |
GPIO_Pin_15;
GPIO_Init(GPIOE, &GPIO_InitStructure);
//打開(kāi)FSMC功能端口,PD.4=RD(nOE);PD.5=WR(nWE)
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4 | GPIO_Pin_5;
GPIO_Init(GPIOD, &GPIO_InitStructure);
//打開(kāi)NE1設(shè)置
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_7;
GPIO_Init(GPIOD, &GPIO_InitStructure);
//打開(kāi)RS設(shè)置
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_11 ;
GPIO_Init(GPIOD, &GPIO_InitStructure);
//NE1=1
GPIO_SetBits(GPIOD, GPIO_Pin_7);
//LCD_RESET=0
GPIO_ResetBits(GPIOE, GPIO_Pin_1);
//LCD_RD=1(nOE)
GPIO_SetBits(GPIOD, GPIO_Pin_4);
//LCD_WR=1(nWE)
GPIO_SetBits(GPIOD, GPIO_Pin_5);
//背光LIGHT=1
GPIO_SetBits(GPIOD, GPIO_Pin_13);
}
//LCD初始化
void LCD_cfg()
{
//復(fù)位LCD
LCD_rst();
//LCD初始化
LCD_Init();
}
//LCD顯示圖片
//根據(jù)LCD_Init里面的配置,LCD的原點(diǎn)在左下角,終點(diǎn)在右上角;先縱向增長(zhǎng),再橫向增長(zhǎng)
void LCD_Show(unsigned char * LCD_Image)
{
u32 n = 0;
u16 temp = 0;
//設(shè)置進(jìn)入模式
//AM=1:地址在水平寫(xiě)入方向上更新
//I/D[1:0]=01:水平方向遞增,垂直方向遞減
//BGR=1:RGB數(shù)據(jù)轉(zhuǎn)換為BGR數(shù)據(jù)
//TRI=0;DFM=0;
//詳細(xì)信息參考ILI9325 8.2.5 Entry Mode
LCD_WR_CMD(0x0003, 0x1018);
//GRAM的水平地址
//8.2.18 GRAM Horizontal/Vertical Address Set
LCD_WR_CMD(0x0020, 0x0000);
//GRAM的垂直地址
LCD_WR_CMD(0x0021, 0x013F);
//水平方向開(kāi)始地址
LCD_WR_CMD(0x0050, 0x0000);
//水平方向結(jié)束地址(0-239)
LCD_WR_CMD(0x0051, 0x00EF);
//垂直方向開(kāi)始地址
LCD_WR_CMD(0x0052, 0x0000);
//垂直方向結(jié)束地址(0-319)
LCD_WR_CMD(0x0053, 0x013F);
//寫(xiě)數(shù)據(jù)地址
//因?yàn)槭?6bits一起寫(xiě)入,而圖像數(shù)據(jù)數(shù)組中是每個(gè)數(shù)據(jù)8bits,
//所以是2個(gè)8bits的數(shù)據(jù)合并成一個(gè)16bits的數(shù)據(jù),再寫(xiě)入GRAM
LCD_WR_ADD(0x0022);
while(n<153600)
{
temp = (u16)(LCD_Image[n]<<8) + LCD_Image[n+1];
LCD_WR_DATA(temp);
n += 2;
}
}
stm32f10x_lcd.c中的代碼
#include "stm32f10x_lcd.h"
//數(shù)據(jù)區(qū)地址
#define Bank1_LCD_Data ((u32)0x60020000)
//寄存器區(qū)地址
#define Bank1_LCD_Reg ((u32)0x60000000)
u32 color1 = 0;
//延時(shí)函數(shù)
void Delay(u32 nCount)
{
for(; nCount != 0; nCount--);
}
//LCD復(fù)位
void LCD_rst()
{
//PE.1連接LCD的reset引腳
GPIO_ResetBits(GPIOE, GPIO_Pin_1);
Delay(0xAFFFFf);
GPIO_SetBits(GPIOE, GPIO_Pin_1 );
Delay(0xAFFFFf);
}
//LCD寫(xiě)寄存器地址函數(shù)
void LCD_WR_ADD(u16 index)
{
*(vu16 *)(Bank1_LCD_Reg) = index;
}
//LCD寫(xiě)數(shù)據(jù)函數(shù)
void LCD_WR_DATA(u16 val)
{
*(vu16 *)(Bank1_LCD_Data) = val;
}
//LCD寫(xiě)寄存器命令函數(shù),先將命令地址寫(xiě)到Reg中,然后再將命令的數(shù)值寫(xiě)到Data中
//具體地址和配置參照ILI9325的Datasheet
void LCD_WR_CMD(u16 index, u16 val)
{
*(vu16 *)(Bank1_LCD_Reg) = index;
*(vu16 *)(Bank1_LCD_Data) = val;
}
void LCD_Init()
{
//設(shè)置內(nèi)部時(shí)鐘
LCD_WR_CMD(0x00E3, 0x3008);
LCD_WR_CMD(0x00E7, 0x0012);
LCD_WR_CMD(0x00EF, 0x1231);
//啟動(dòng)振蕩,ILI9325可以不要這一句
LCD_WR_CMD(0x0000, 0x0001);
//設(shè)置驅(qū)動(dòng)器輸出控制,SS=1,SM=0
//當(dāng)SS=0時(shí),源輸出信號(hào)從S1開(kāi)始至S720結(jié)束;
//當(dāng)SS=1時(shí),源輸出信號(hào)從S720開(kāi)始至S1結(jié)束。
//SM和GS搭配使用,具體查看ILI9325 8.2.3 Driver Output Contorl
LCD_WR_CMD(0x0001, 0x0100);
//LCD波形控制
//B/C=1:行反轉(zhuǎn);
//EOR=1和B/C=1:設(shè)置行反轉(zhuǎn)
//8.2.4 LCD Driving Wave Control
LCD_WR_CMD(0x0002, 0x0700);
//設(shè)置進(jìn)入模式
//AM=1:地址在水平寫(xiě)入方向上更新
//I/D[1:0]=01:水平方向遞增,垂直方向遞減
//BGR=1:RGB數(shù)據(jù)轉(zhuǎn)換為BGR數(shù)據(jù)
//TRI=0;DFM=0;
//詳細(xì)信息參考ILI9325 8.2.5 Entry Mode
LCD_WR_CMD(0x0003, 0x1018);
//重新調(diào)整控制寄存器大小
//8.2.6 Resizing Control Register
LCD_WR_CMD(0x0004, 0x0000);
//顯示器控制2
//FP[3:0]=0010;
//BP[3:0]=0010;
//詳細(xì)信息參考ILI9325 8.2.8 Display Control 2
LCD_WR_CMD(0x0008, 0x0202);
//顯示器控制3
//設(shè)置非顯示區(qū)域刷新
//8.2.9 Display Control 3
LCD_WR_CMD(0x0009, 0x0000);
//顯示器控制4
//FMARK信號(hào)設(shè)置
//8.2.10 Display Control 4
LCD_WR_CMD(0x000A, 0x0000);
//RGB顯示接口控制1
//8.2.11 RGB Display Interface Control 1
LCD_WR_CMD(0x000C, 0x0000);
//幀標(biāo)記位置
//8.2.12 Frame Marker Position
LCD_WR_CMD(0x000D, 0x0000);
//RGB顯示接口控制2
//8.2.13 RGB Display Interface Control 2
LCD_WR_CMD(0x000F, 0x0000);
//功率控制1
//8.2.14 Power Control 1
LCD_WR_CMD(0x0010, 0x0000);
//功率控制2
//8.2.15 Power Control 2
//VC[2:0]=111:參考電壓為Vci
LCD_WR_CMD(0x0011, 0x0007);
//功率控制3
//8.2.16 Power Control 3
LCD_WR_CMD(0x0012, 0x0000);
//功率控制4
//8.2.17 Power Control 4
LCD_WR_CMD(0x0013, 0x0000);
//延時(shí),放電
Delay(200);
//功率控制1
//SAP=1:源驅(qū)動(dòng)程序被啟動(dòng)
//BT[2:0]=110:
//APE=1:開(kāi)始供應(yīng)電力
//AP[2:0]=001:伽馬驅(qū)動(dòng)放大和源驅(qū)動(dòng)放大
LCD_WR_CMD(0x0010, 0x1690);
//功率控制2
//DC1[2:0]=010:選擇升壓電路2工作頻率Fosc/16
//DC0[2:0]=010:選擇升壓電路1工作頻率Fosc/4
//VC[2:0]=111:參考電壓為Vci
LCD_WR_CMD(0x0011, 0x0227);
//延時(shí)
Delay(50);
//功率控制3
//PON=1:控制線(xiàn)路3(VGL)開(kāi)啟
//VRH[3:0]=1100:設(shè)置外部參考電壓
LCD_WR_CMD(0x0012, 0x001C);
//延時(shí)
Delay(50);
//功率控制4
//VDV[4:0]=11000:設(shè)置Vcom的電壓振幅交替
LCD_WR_CMD(0x0013, 0x1800);
//功率控制7
//8.2.21 Power Control 7
//VCM[5:0]=011100:設(shè)置內(nèi)部VcomH電壓
LCD_WR_CMD(0x0029, 0x001C);
//幀速率和色彩控制
//8.2.22 Frame Rate and Color Control
//FRS[3:0]:1101:幀率128
LCD_WR_CMD(0x002B, 0x000D);
//延時(shí)
Delay(50);
//GRAM的水平地址
//8.2.18 GRAM Horizontal/Vertical Address Set
LCD_WR_CMD(0x0020, 0x0000);
//GRAM的垂直地址
LCD_WR_CMD(0x0021, 0x0000);
//伽馬控制
//8.2.23 Gamma Control
LCD_WR_CMD(0x0030, 0x0007);
LCD_WR_CMD(0x0031, 0x0302);
LCD_WR_CMD(0x0032, 0x0105);
LCD_WR_CMD(0x0035, 0x0206);
LCD_WR_CMD(0x0036, 0x0808);
LCD_WR_CMD(0x0037, 0x0206);
LCD_WR_CMD(0x0038, 0x0504);
LCD_WR_CMD(0x0039, 0x0007);
LCD_WR_CMD(0x003C, 0x0105);
LCD_WR_CMD(0x003D, 0x0808);
//水平和垂直位置的RAM地址
//8.2.24 Horizontal and Vertical RAM Address Position
//水平方向開(kāi)始地址
LCD_WR_CMD(0x0050, 0x0000);
//水平方向結(jié)束地址(0-239)
LCD_WR_CMD(0x0051, 0x00EF);
//垂直方向開(kāi)始地址
LCD_WR_CMD(0x0052, 0x0000);
//垂直方向結(jié)束地址(0-319)
LCD_WR_CMD(0x0053, 0x013F);
//門(mén)掃描控制
//8.2.25 Gate Scan Control
//GS=1:掃描方向是從G320到G1
//NL[5:0]=100111
LCD_WR_CMD(0x0060, 0xA700);
//NDL=0:在非顯示區(qū)域設(shè)置源驅(qū)動(dòng)器的輸出極
//VLE=0:垂直滾動(dòng)顯示不可用
//REV=1:圖像灰度反轉(zhuǎn)
LCD_WR_CMD(0x0061, 0x0001);
//VL[8:0]=0
LCD_WR_CMD(0x006A, 0x0000);
//局部影像1顯示位置
//8.2.26 Partial Image 1 Display Position
LCD_WR_CMD(0x0080, 0x0000);
//局部影像1RAM開(kāi)始/結(jié)束地址
//8.2.27 Partial Image 1 RAM Start/End Address
LCD_WR_CMD(0x0081, 0x0000);
LCD_WR_CMD(0x0082, 0x0000);
//局部影像2顯示位置
//8.2.28. Partial Image 2 Display Position
LCD_WR_CMD(0x0083, 0x0000);
//局部影像2RAM開(kāi)始/結(jié)束地址
//8.2.29 Partial Image 2 RAM Start/End Address
LCD_WR_CMD(0x0084, 0x0000);
LCD_WR_CMD(0x0085, 0x0000);
//平板接口控制1
//8.2.30 Panel Interface Control 1
//RTNI[4:0]=10000:設(shè)置內(nèi)部時(shí)鐘運(yùn)行模式中1線(xiàn)時(shí)鐘的數(shù)目:16個(gè)
LCD_WR_CMD(0x0090, 0x0010);
//平板接口控制2
//8.2.31 Panel Interface Control 2
LCD_WR_CMD(0x0092, 0x0000);
LCD_WR_CMD(0x0093, 0x0003);
//平板接口控制4
//8.2.32 Panel Interface Control 4
LCD_WR_CMD(0x0095, 0x0110);
LCD_WR_CMD(0x0097, 0x0000);
LCD_WR_CMD(0x0098, 0x0000);
//顯示控制1
//8.2.7 Display Control 1
//BASEE=1:顯示基本圖像
//GON=1 DTE=1:正常顯示
//D[1:0]=11:打開(kāi)顯示面板
LCD_WR_CMD(0x0007, 0x0133);
//GRAM寫(xiě)入數(shù)據(jù),用黑色清屏
LCD_WR_ADD(0x0022);
for(color1=0;color1<320*240;color1++)
{
LCD_WR_DATA(0x0000); //
}
color1=0;
}
stm32f10x_lcd.h中的代碼
#include "stm32f10x_lib.h"
//LCD復(fù)位函數(shù)
void LCD_rst();
//LCD初始化函數(shù)
void LCD_Init();
//延時(shí)函數(shù)
void Delay(u32 nCount);
//LCD寫(xiě)寄存器地址函數(shù)
void LCD_WR_ADD(u16 index);
//LCD寫(xiě)數(shù)據(jù)函數(shù)
void LCD_WR_DATA(u16 val);
pic_bit.c中的代碼和pic_hit.c中的代碼由于太長(zhǎng)了,所以并沒(méi)有貼出來(lái),其實(shí)就是根據(jù)某一幅圖片用Image2LCD生成的,其數(shù)組名分別叫
const unsigned char LCD_Image_BIT[153600]
const unsigned char LCD_Image_HIT[153600]