一、FSMC之我見
開始只是談到別人對FSMC的理解,注意這里只討論FSMC控制TFT,也就是在FSMC的NOR\PSRAM模式控制LCD,所以我們以下的分析都是基于這種模式的。
1、我們之前通過使用GPIO來模擬8080/6800時序從而達到驅動彩屏的,同樣需要明白的一點就是我們也只是使用FSMC來模擬8080/6800時序,只不過這個讀寫速度有些快(使用了總線嘛),僅此而已!
簡單一點就是:8080是通過“讀使能(RE)”和“寫使能(WE)”兩條控制線進行讀寫操作。 6800是通過“總使能(E)”和“讀寫選擇(W/R)”兩條控制線進行
2、那么了解到FSMC的三總線如下!
數據線:這個可以分為8位的和16位,這個不難理解,就是之指一次穿上紅8位還是16位數據,我的是16位的,8位的有一個懶得用。
地址線:既然我們訪問的外NOR FLASH,那么一定會有相應的地址線,那么這些地址線在哪里呢?肯定是通過GPIO引腳復用的。有A0 -- A23 24根,能夠控制訪問16M的空間,也就是一個子bank;
控制總線:它的控制總線只有三根:讀使能信號,寫使能信號,片選信號。所以這里和我們8080時序相比,少了復位信號線和數據/命令控制線,怎么辦?繼續看!
3、了解了FSMC會有這三總線的概念,那么接下來就是如何轉化為我們需要的時序了。
對比一下FSMC訪問外nor flash和8080訪問時序如下
差別似乎很小是吧,簡單說就是在數據/指令選擇和復位信號上的區別。
4、在這里我們使用的軟件方法來完善FSMC轉化為8080的讀寫時序
在參考手冊上的存儲系統能找到,芯片留給我們外擴的存儲器(NOR FLASH、PSRAM這類可直接尋址的器
件)地址是從0x60000000開始的,意思就是當我們訪問0x60000000的時候,那就是相當于訪問外部nor flash了(我們只討論這種情況),那么他就會自動產生FSMC的時序
在這里,我們所需要添加的就是D/C選擇信號,如何實現呢?我們是通過,一根地址線來實現的,當我們把A0多對應的GPIOF0(引腳默認復用)接到TFT的RS端,
然后執行訪問0x60000000的指令,那么RS是否就是低電平選擇為數據呢?又加入我們訪問的地址是0x60000001的時候,那么RS是否就是高電平,從而選擇的就是指令傳送呢?答案當然是肯定的!所以我們就解決了這個問題,復位信號就更好解決了,直接和開發板的復位引腳接在一起就好了,就這么簡單!
二、說了這么久的理論,來個實例分析更加形象了,首先硬件連線要明白
在原理圖或者開發手冊上面能夠確定引腳復用問題
地址引腳:
(A0-A5 :PF0 - PF5) (A6-A9: F12-F15 ) (A10-A15:PG0-PG5)
(A16-A18:PD11-PD13) (A19-A22:PE3-PE6) (A23-PE2)
片選信號(NEx:PG12)因為我選擇的是block4
寫使能(NWR:PD5)
讀使能(NOE:PD4)
至此控制引腳基本完成
下面就是數據引腳:
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、硬件引腳配置函數
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口復用功能時鐘
/*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);
評論
查看更多