最近在學習stm32的FSMC,打算整一個LCD的驅動,然後封裝的比正點原子更高層一些,方便後期直接調用 然後在學習的時候碰到了一個小問題,研究了一會之後覺得挺有意思的,所以記下來 //使用NOR/SRAM的 Bank1.sector4,地址位HADDR[27,26]=11 A6作為數據命令區分線 ...
最近在學習stm32的FSMC,打算整一個LCD的驅動,然後封裝的比正點原子更高層一些,方便後期直接調用
然後在學習的時候碰到了一個小問題,研究了一會之後覺得挺有意思的,所以記下來
//使用NOR/SRAM的 Bank1.sector4,地址位HADDR[27,26]=11 A6作為數據命令區分線 //註意設置時STM32內部會右移一位對其! 111 1110=0X7E #define LCD_BASE ((u32)(0x6C000000 | 0x0000007E)) #define LCD ((LCD_TypeDef *) LCD_BASE)
這是正點原子給的常式當中lcd.h文件的幾行代碼
他這裡對於LCD_Base的地址是怎麼來的解釋的比較粗糙,其實是這樣的
下圖是中文手冊的1194面有關FSMC的內容
可以看到FSMC存儲區域是分成了四個部分,即Bank1-Bank4
通過計算也不難發現,這四個Bank所占用的記憶體大小是一樣的
而在正點原子的常式當中,他使用的是Bank1,也就是起始地址是0x60000000
然後註釋說HADDR[27:26] = 11
根據手冊可以得出他使用的是存儲區域1的第4區
由於Bank內部的四個存儲區域的大小是一樣的(因為本質上四個區域是相同的),所以這四個存儲區域平均分了Bank1的地址,存儲區1開始與0x60,存儲區2開始於存儲區0x64,存儲區3開始於0x68,存儲區4開始於0x6C
這就能解釋通這個Bank1.sector4的起始地址是怎麼得到的了
然後就是解釋後面的0x7E
我們知道正點原子設置地址線並不是真的為了寫地址,而是一個RS信號正好連在了地址線上面,通過這個RS信號來控制寫入命令和數據
所以我們只需要關註RS信號對應的那個引腳,也就是A6引腳(Address 6),而我們的A0-A5這6個引腳是沒有用的。
0x7E經過轉換進位之後是01111110,但是stm32內部會將數據右移一位,也就是所有1往右移動1位,高位補0,即A6:A0 = 0111111,此時A6是0。
如果我們增加0x02,那麼就會得到0x80,也就是10000000,這時候右移一位就是01000000,也就是A6是1,這就實現了RS信號的0和1的控制(這是對於8位地址而言的,當是16位地址的時候,我們增加0x01,就會得到01111111,此時不會進行右移對齊,A6是1)
關於8位地址和16位地址的右移問題,可以參照手冊1195面
16位地址的時候,會進行一次右移,因為內部變化兩個位元組,外面就變化一個位元組,所以只需要增加0x01就能達到和8位地址時增加0x02一樣的效果了