STM32采用結(jié)構(gòu)體定義FSMC的地址
LCD地址設(shè)置
圖中只畫出了數(shù)據(jù)線與地址線其他的線我沒有花,請各位腦部一下,實在不行去看原子提供的LCD的原理圖。
LCD有一個引腳是D/C引腳,用來區(qū)分往LCD中寫入的數(shù)據(jù)是命令還是數(shù)據(jù)Data/Command
比如說我要給LCD控制器的X寄存器寫入數(shù)據(jù)Y那么我需要先給LCD發(fā)送一個命令----X寄存器的地址,此時D/C引腳為低電平
然后再發(fā)送一個數(shù)據(jù)Y,此時D/C引腳為高電平。但是正常的FSMC總線中沒有控制命令與數(shù)據(jù)的功能(或許我不知道)。
正常的FSMC是絕對地址尋址。也就是需要硬件地址線A0-AX(X為地址的長度)去控制。
所以STM32使用了一根地址線不是說往LCD控制的的某個地址寫入數(shù)據(jù)。而是用這一根地址線去控制往LCD輸入是命令還是數(shù)據(jù)。
那么是怎么控制的,咱們一點一點來分析。
首先數(shù)據(jù)手冊上說
為什么16位的時候要右移1位?這是因為寬度為16位的時候每寫入一個數(shù)據(jù),即為兩個字節(jié),那么地址就應(yīng)該加2.
所以這樣的話,用FSMC配置為16位數(shù)據(jù)寬度時,每次設(shè)置地址都只改變了HADDR[25:1],那么就會保證地址為2的倍數(shù),不會出現(xiàn)從半個數(shù)據(jù)寬度的地址處寫入數(shù)據(jù)的情況。搞清楚這個然后再來看原子提供的LCD結(jié)構(gòu)體
1
2
3
4
5
6
7
8
typedef struct
{
vu16 LCD_REG;
vu16 LCD_RAM;
} LCD_TypeDef;
#define LCD_BASE((u32)(0x6C000000 | 0x0000007E))
#define LCD((LCD_TypeDef *) LCD_BASE)
這段代碼,結(jié)構(gòu)體中包含兩個雙字節(jié)成員LCD_REG跟LCD_RAM;
大家都知道結(jié)構(gòu)體中的地址是從上到下連續(xù)增加的。
結(jié)構(gòu)體的地址即為LCD_REG的地址。那么LCD_RAM的地址將為LCD_REG地址加2,因為是u16,雙字節(jié)類型。
FSMC使用了NOR/SRAM的Bank1 sector4那么根據(jù)數(shù)據(jù)手冊可以得知NOR/SRAM的起始地址為0x6000 0000
但是FSMC又把NOR這部分區(qū)域的地址分為了四塊,在這使用的是第四塊,所以HADDR[27 26] == 11
即27與26位為1,所以地址應(yīng)該基地址應(yīng)該為0x6C00 0000
然后再看地址線選用了A6當作LCD的D/C區(qū)分線,那么就應(yīng)該是寫入LCD_RAM時,第七位(還有A0嘛這反應(yīng)一下)應(yīng)該為1。但是設(shè)置的數(shù)據(jù)寬度為16位,STM32會自動的右移1位,所以咱們需要反過來左移一位,那么就應(yīng)該是1000 0000b把這個數(shù)減2就應(yīng)該是
0111 1110b即為0x7E
那么0x6C00 0000 | 0x7E這個地址將會是LCD結(jié)構(gòu)體的首地址。
1
#define LCD_BASE((u32)(0x6C000000 | 0x0000007E))
而這句話,將LCD_BASE代表這個地址數(shù)據(jù)
1
#define LCD((LCD_TypeDef *) LCD_BASE)
這句話將LCD定義為一個結(jié)構(gòu)體指針,而這個結(jié)構(gòu)體指針的地址為LCD_BASE
現(xiàn)在再反過來推一遍。
定義了這個結(jié)構(gòu)體指針,第一個成員的地址為0x6C00 007E,第二個成員的地址為0x6C00 0080
把這兩個數(shù)寫成二進制格式
0110 1100 0000 0000 0000 0000 0111 1110
0110 1100 0000 0000 0000 0000 1000 0000
通過對比是不是看出來第8位從0變?yōu)榱?,即為A7的電平狀態(tài),但是由于16位數(shù)據(jù)格式,要右移一位,所以就會成為A6的電平狀態(tài)。
所以通過這個結(jié)構(gòu)體對不同地址寫入數(shù)據(jù),就能控制是對LCD寫入的是命令還是數(shù)據(jù)。