基于stm32f103zet6之使用FSMC驅(qū)動TFT的學(xué)習(xí)
在完成IO驅(qū)動彩屏的試驗(yàn)后,就準(zhǔn)備著手使用FSMC來驅(qū)動彩屏,先了解一下預(yù)備知識
一、所謂的FSMC機(jī)制
簡單介紹FSMC在這篇博文里面很清楚,推薦一下http://blog.csdn.net/king_bingge/article/details/8718566
然后還有就是這篇學(xué)習(xí)筆記,也還行http://www.cnblogs.com/hduxyc/archive/2011/05/17/2048099.html
個人覺得有了這兩篇博文再加上我們的參考手冊足夠搞定FSMC驅(qū)動彩屏了
二、FSMC之我見
開始只是談到別人對FSMC的理解,注意這里只討論FSMC控制TFT,也就是在FSMC的NORPSRAM模式控制LCD,所以我們以下的分析都是基于這種模式的。
1、我們之前通過使用GPIO來模擬8080/6800時序從而達(dá)到驅(qū)動彩屏的,同樣需要明白的一點(diǎn)就是我們也只是使用FSMC來模擬8080/6800時序,只不過這個讀寫速度有些快(使用了總線嘛),僅此而已!如果不明白8080/6800時序是怎樣的或許在這個文庫里面能找到你想要的http://wenku.baidu.com/view/a8c98600cc1755270722083e.html
簡單一點(diǎn)就是:8080是通過“讀使能(RE)”和“寫使能(WE)”兩條控制線進(jìn)行讀寫操作。 6800是通過“總使能(E)”和“讀寫選擇(W/R)”兩條控制線進(jìn)行
2、那么了解到FSMC的三總線如下!
數(shù)據(jù)線:這個可以分為8位的和16位,這個不難理解,就是之指一次穿上紅8位還是16位數(shù)據(jù),我的是16位的,8位的有一個懶得用。
地址線:既然我們訪問的外NOR FLASH,那么一定會有相應(yīng)的地址線,那么這些地址線在哪里呢?肯定是通過GPIO引腳復(fù)用的。有A0 -- A23 24根,能夠控制訪問16M的空間,也就是一個子bank;
控制總線:它的控制總線只有三根:讀使能信號,寫使能信號,片選信號。所以這里和我們8080時序相比,少了復(fù)位信號線和數(shù)據(jù)/命令控制線,怎么辦?繼續(xù)看!
3、了解了FSMC會有這三總線的概念,那么接下來就是如何轉(zhuǎn)化為我們需要的時序了。
對比一下FSMC訪問外nor flash和8080訪問時序如下
差別似乎很小是吧,簡單說就是在數(shù)據(jù)/指令選擇和復(fù)位信號上的區(qū)別。
4、在這里我們使用的軟件方法來完善FSMC轉(zhuǎn)化為8080的讀寫時序
在參考手冊上的存儲系統(tǒng)能找到,芯片留給我們外擴(kuò)的存儲器(NOR FLASH、PSRAM這類可直接尋址的器
件)地址是從0x60000000開始的,意思就是當(dāng)我們訪問0x60000000的時候,那就是相當(dāng)于訪問外部nor flash了(我們只討論這種情況),那么他就會自動產(chǎn)生FSMC的時序
在這里,我們所需要添加的就是D/C選擇信號,如何實(shí)現(xiàn)呢?我們是通過,一根地址線來實(shí)現(xiàn)的,當(dāng)我們把A0多對應(yīng)的GPIOF0(引腳默認(rèn)復(fù)用)接到TFT的RS端,
然后執(zhí)行訪問0x60000000的指令,那么RS是否就是低電平選擇為數(shù)據(jù)呢?又加入我們訪問的地址是0x60000001的時候,那么RS是否就是高電平,從而選擇的就是指令傳送呢?答案當(dāng)然是肯定的!所以我們就解決了這個問題,復(fù)位信號就更好解決了,直接和開發(fā)板的復(fù)位引腳接在一起就好了,就這么簡單!
三、說了這么久的理論,來個實(shí)例分析更加形象了,首先硬件連線要明白
在原理圖或者開發(fā)手冊上面能夠確定引腳復(fù)用問題
地址引腳:
(A0-A5 :PF0 - PF5) (A6-A9: F12-F15 ) (A10-A15:PG0-PG5)
(A16-A18:PD11-PD13) (A19-A22:PE3-PE6) (A23-PE2)
片選信號(NEx:PG12)因?yàn)槲疫x擇的是block4
寫使能(NWR:PD5)
讀使能(NOE:PD4)
至此控制引腳基本完成
下面就是數(shù)據(jù)引腳:
PD14-FSMC-D0 ----LCD-DB0
PD15-FSMC-D1 ----LCD-DB1
PD0-FSMC-D2 ----LCD-DB2
PD1-FSMC-D3 ----LCD-DB3
PE7-FSMC-D4 ----LCD-DB4
PE8-FSMC-D5 ----LCD-DB5
PE9-FSMC-D6 ----LCD-DB6
PE10-FSMC-D7 ----LCD-DB7
PE11-FSMC-D8 ----LCD-DB8
PE12-FSMC-D9 ----LCD-DB9
PE13-FSMC-D10 ----LCD-DB10
PE14-FSMC-D11 ----LCD-DB11
PE15-FSMC-D12 ----LCD-DB12
PD8-FSMC-D13 ----LCD-DB13
PD9-FSMC-D14 ----LCD-DB14
PD10-FSMC-D15 ----LCD-DB15
有了這些硬件連線是沒有任何問題的
四、正式分析程序
1、硬件引腳配置函數(shù)
voidLCD_CtrlLinesConfig(void)
{
GPIO_InitTypeDefGPIO_InitStructure;
/*EnableFSMC,GPIOD,GPIOE,GPIOF,GPIOGandAFIOclocks*/
RCC_AHBPeriphClockCmd(RCC_AHBPeriph_FSMC,ENABLE);//使能FSMC
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOD|RCC_APB2Periph_GPIOE|
RCC_APB2Periph_GPIOF|RCC_APB2Periph_GPIOG|
RCC_APB2Periph_AFIO,ENABLE);
//IO口復(fù)用功能時鐘
/*SetPD.00(D2),PD.01(D3),PD.04(NOE),PD.05(NWE),PD.08(D13),PD.09(D14),
PD.10(D15),PD.14(D0),PD.15(D1)asalternate
functionpushpull*/
/*D端口初始化*/
GPIO_InitStructure.GPIO_Pin=GPIO_Pin_0|GPIO_Pin_1|GPIO_Pin_4|GPIO_Pin_5|
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);
/*E端口初始化*/
/*SetPE.07(D4),PE.08(D5),PE.09(D6),PE.10(D7),PE.11(D8),PE.12(D9),PE.13(D10),
PE.14(D11),PE.15(D12)asalternatefunctionpushpull*/
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_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;
GPIO_InitStructure.GPIO_Mode=GPIO_Mode_AF_PP;
GPIO_Init(GPIOE,&GPIO_InitStructure);//將配置寫入GPIOE管腳
/*A0地址線*/
GPIO_InitStructure.GPIO_Pin=GPIO_Pin_0;
GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;
GPIO_InitStructure.GPIO_Mode=GPIO_Mode_AF_PP;
GPIO_Init(GPIOF,&GPIO_InitStructure);
/*SetPG.12(NE4(LCD/CS))asalternatefunctionpushpull-CE3(LCD/CS)*/
GPIO_InitStructure.GPIO_Pin=GPIO_Pin_12;
GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;
GPIO_InitStructure.GPIO_Mode=GPIO_Mode_AF_PP;
GPIO_Init(GPIOG,&GPIO_InitStructure);