您现在的位置: 主页 > MCU > 单片机技术应用 > STM32采用结构体定义FSMC的地址 -
本文所属标签:
为本文创立个标签吧:

STM32采用结构体定义FSMC的地址 -

来源: 网络用户发布,如有版权联系网管删除 2018-09-06 

[导读]LCD地址设置图中只画出了数据线与地址线其他的线我没有花,请各位脑部一下,实在不行去看原子提供的LCD的原理图。LCD有一个引脚是D/C引脚,用来区分往LCD中写入的数据是命令还是数据Data/Command比如说我要给LCD控制

LCD地址设置

本文引用地址: http://www.21ic.com/app/mcu/201807/779240.htm



图中只画出了数据线与地址线其他的线我没有花,请各位脑部一下,实在不行去看原子提供的LCD的原理图。

LCD有一个引脚是D/C引脚,用来区分往LCD中写入的数据是命令还是数据Data/Command
比如说我要给LCD控制器的X寄存器写入数据Y那么我需要先给LCD发送一个命令----X寄存器的地址,此时D/C引脚为低电平
然后再发送一个数据Y,此时D/C引脚为高电平。但是正常的FSMC总线中没有控制命令与数据的功能(或许我不知道)。
正常的FSMC是绝对地址寻址。也就是需要硬件地址线A0-AX(X为地址的长度)去控制。
所以STM32使用了一根地址线不是说往LCD控制的的某个地址写入数据。而是用这一根地址线去控制往LCD输入是命令还是数据。
那么是怎么控制的,咱们一点一点来分析。
首先数据手册上说




为什么16位的时候要右移1位?这是因为宽度为16位的时候每写入一个数据,即为两个字节,那么地址就应该加2.
所以这样的话,用FSMC配置为16位数据宽度时,每次设置地址都只改变了HADDR[25:1],那么就会保证地址为2的倍数,不会出现从半个数据宽度的地址处写入数据的情况。搞清楚这个然后再来看原子提供的LCD结构体

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)

这段代码,结构体中包含两个双字节成员LCD_REG跟LCD_RAM;
大家都知道结构体中的地址是从上到下连续增加的。
结构体的地址即为LCD_REG的地址。那么LCD_RAM的地址将为LCD_REG地址加2,因为是u16,双字节类型。
FSMC使用了NOR/SRAM的Bank1 sector4那么根据数据手册可以得知NOR/SRAM的起始地址为0x6000 0000
但是FSMC又把NOR这部分区域的地址分为了四块,在这使用的是第四块,所以HADDR[27 26] == 11
即27与26位为1,所以地址应该基地址应该为0x6C00 0000

然后再看地址线选用了A6当作LCD的D/C区分线,那么就应该是写入LCD_RAM时,第七位(还有A0嘛这反应一下)应该为1。但是设置的数据宽度为16位,STM32会自动的右移1位,所以咱们需要反过来左移一位,那么就应该是1000 0000b把这个数减2就应该是
0111 1110b即为0x7E
那么0x6C00 0000 | 0x7E这个地址将会是LCD结构体的首地址。

1

#define LCD_BASE((u32)(0x6C000000 | 0x0000007E))

而这句话,将LCD_BASE代表这个地址数据

1

#define LCD((LCD_TypeDef *) LCD_BASE)

这句话将LCD定义为一个结构体指针,而这个结构体指针的地址为LCD_BASE

现在再反过来推一遍。
定义了这个结构体指针,第一个成员的地址为0x6C00 007E,第二个成员的地址为0x6C00 0080
把这两个数写成二进制格式
0110 1100 0000 0000 0000 0000 0111 1110
0110 1100 0000 0000 0000 0000 1000 0000
通过对比是不是看出来第8位从0变为了1,即为A7的电平状态,但是由于16位数据格式,要右移一位,所以就会成为A6的电平状态。
所以通过这个结构体对不同地址写入数据,就能控制是对LCD写入的是命令还是数据。




              查看评论 回复



嵌入式交流网主页 > MCU > 单片机技术应用 > STM32采用结构体定义FSMC的地址 -
 

"STM32采用结构体定义FSMC的地址 -"的相关文章

网站地图

围观()