FSMC在正点原子的HAL课程中用法较为单一,就是用来模拟8080时序驱动LCD屏幕,其实就是利用FSMC读写LCD模块的SRAM。当然,这个我也没有买,只学习了理论。

大部分图片来源:正点原子HAL库课程

 专栏目录:记录自己的嵌入式学习之路-CSDN博客


目录

1    用途

2    引脚

3    FSMC驱动LCD屏幕

3.1    GPIO8080和FSMC驱动的概念区别

​编辑

3.2    FSMC的五种异步时序

3.3    FSMC时序与GPIO8080时序的对比

3.4    FSMC的地址映射

3.5    FSMC的本质

3.6    8080 RS信号选择的实现

3.7    相关HAL库函数

3.8    重要结构体

3.9    实际使用


1    用途

FSMC,Flexible Static Memory Controller,灵活的静态存储控制器。

用于驱动SRAM,NOR FLASH,NAND FLASH及PC卡类型的存储器。配置好FSMC,定义一个指向这些地址的指针,通过对指针操作就可以直接修改存储单元的内容,FSMC自动完成读写命令和数据访问操作,不需要程序去实现时序。也就是无需手动去拉高、拉低信号实现选择写、读和选择读写命令、数据等东西。

F1/ F4(407)系列大容量型号,且引脚数目在100脚以上的芯片都有FSMC接口。F4/F7/H7系列就是FMC接口,区别是FMC可以直接驱动SDRAM,FSMC不行。


2    引脚

  • 引脚名称的N表示该信号是低电平有效。
  • 上面红框框着的部分是LCD要用到的。

  • 3    FSMC驱动LCD屏幕

    3.1    GPIO8080和FSMC驱动的概念区别

    3.2    FSMC的五种异步时序

    选择模式A的原因:

  • 操作屏幕实际操作的是屏幕的SRAM,因此模式B、C都不满足;
  • 而模式1在读时序时片选信号不翻转,但在8080时序中,要进行读写操作时WR和RD都是上升沿才有效,因此模式1也不满足;
  • 忽略过于高级的模式D;
  • 选择模式A;
  • 3.3    FSMC时序与GPIO8080时序的对比

    可以看到,总体来说FSMC的时序和8080是一致的,只是在实际使用中,各高低电平的脉宽需要根据8080的进行细微调整,具体需要查阅LCD屏幕的数据手册。需调整的部分时序如下图所示:

    3.4    FSMC的地址映射

    注意⚠️:这里FSMC有4个存储块,每个存储块又有4个区;存储块叫做Bank,选区其实也叫Bank。在实际使用中,一般都是通过存储块的功能来区分存储块,再通过序号区分选区,如:FSMC_NORSRAM_BANK1,这里说的就是Bank1的第4区。因此正点原子的例程写的Bank4其实是没错的,因为它已经指定了NOR的存储块了。

    这里HADDR是内部AHB地址总线,其中HADDR[25:0]来自外部存储器地址FSMC_A[25:0],而HADDR[26:27]对4个区进行寻址。

    在FSMC连接到8位的存储器时,其HADDR[25:0]就完全对应FSMC_A[25:0],但是对于16位的存储器,由于HADDR的一个位就不对应FSMC中的一个地址了,因为其一个位本应对应一个字节,但如果按一个一个来,访问到的就是外部的半个数据,因此需要将HADDR右移一位,使得HADDR的一位对应上FSMC的一个数据长度。

    3.5    FSMC的本质

    自动片选、自动发送地址信号,自动输出/输入。

    当FSMC外设被配置成正常工作,并且外部接了NOR FLASH时,若向0x60000000地址写入数据如0xABCD,FSMC会自动在各信号线上产生相应的电平 信号,写入数据。FSMC会控制片选信号NE1选择相应的NOR 芯片,然后使用地址线A[25:0]输出0x60000000,在NWE写使能信号线上发出低 电平的写使能信号,而要写入的数据信号0xABCD则从数据线D[15:0]输出,然后数据就被保存到NOR FLASH中了。

    3.6    8080 RS信号选择的实现

    要实现8080的RS选择命令/数据的效果,需要用地址线来做。

    因为RS信号线实质就是低电平是命令,高电平是数据,所以实际上它不需要选择命令,只需要有一条信号线选择数据就行。因此,在LCD的例子中,使用FSMC_A10,即地址总线的其中一根来做选择即可。

    而例子中之所以说代表命令的地址是0x6C00 0000,而代表数据的地址是0x6C00 0800(注意⚠️这里偏移了0x800是因为10号地址线是210*2,乘2是因为要将地址右移以满足16位的数据宽度),是因为总得选择一个地址嘛,所以就是用FSMC_NEx的基地址做命令地址,用FSMC_A10的地址做数据地址。那么,在将FSMC_A10接到RS线上后,根据FSMC的信号时序,我在写FSMC_A10的数据时,那么由于FSMC_A10变为高电平,那么RS也会处于高电平,相当于选择了数据。而不对FSMC_A10写数据时,其将位于低电平状态,从而选择了命令。

    3.7    相关HAL库函数

    3.8    重要结构体

    SRAM_HandleTypeDef:SRAM控制句柄

    其中:

  • Instance应为FSMC_NORSRAM_DEVICE
  • Extended应为FSMC_NORSRAM_EXTERNDEVICE
  • Init最为重要
  • 剩下的用不到
  • FSMC_NORSRAM_InitTypeDef:SRAM初始化结构体

    其中:

  • NSBank设为FSMC_NORSRAM_BANK4(因为实验中用到的是内存块4)
  • DataAddressMux设为FSMC_DATA_ADDRESS_MUX_DISABLE(因为实验中并未将地址线和数据线复用,因此不使能)
  • MemoryType设为FSMC_MEMORY_TYPE_SRAM(屏幕的显存为SRAM)
  • MemoryDataWidth设为FSMC_NORSRAM_MEM_BUS_WIDTH_16(屏幕的数据宽度为16位)
  • WriteOperation设为FSMC_WRITE_OPERATION_ENABLE(使能写操作)
  • ExtendedMode设为FSMC_EXTENDED_MODE_ENABLE(扩展模式指的是设置读写时设置的时序不一致,而LCD的8080总线需要读写时序不一致,因此需要设置为使能)
  • 其余在SRAM模式基本不需要管
  • FSMC_NORSRAM_TimingTypeDef:时序初始化结构体

    其中:

  • AddressHoldTime在模式A中并没有用到;
  • 3.9    实际使用

    在FSMC的实际使用中,其实就是定义并初始化好了FSMC之后,直接根据FSMC的基地址定义一个数据长度的两个变量,一个是命令地址,一个是数据地址。如:

    #define FSMC_ADDR_DATA          ((uint32_t) 0X6C000800)
    #define FSMC_ADDR_CMD           ((uint32_t) 0X6C000000)

    而这之后,对这两个地址写入数据就是写命令和写数据啦,如下:

    作者:Tesseract_9527

    物联沃分享整理
    物联沃-IOTWORD物联网 » 【STM32】FSMC

    发表回复