[开发工具] FSMC扩展外部SRAM

[复制链接]
594|7
 楼主| mollylawrence 发表于 2023-3-30 18:11 | 显示全部楼层 |阅读模式
软件主要涉及的就是FSMC的配置工作,涉及到几个结构体(这里都是以NOR和SRAM为例):
【FSMC_NORSRAMInitTypeDef】:前 13 个基本类型(unit32_t)的成员变量用来配置片选控制寄存器 FSMC_BCRx,后面两个SMC_NORSRAMTimingInitTypeDef 指针类型的成员变量分别用来配置寄存器 FSMC_BTRx 和 FSMC_BWTRx,设置读写时序参数。
  1. typedef struct
  2. {
  3.   uint32_t FSMC_Bank;               //设置使用到的存储块标号和区号
  4.   uint32_t FSMC_DataAddressMux;     //设置地址/数据复用使能,若设置为使能,那么地址的低 16 位
  5. 和数据将共用数据总线,仅对 NOR 和 PSRAM 有效
  6.   uint32_t FSMC_MemoryType;          //设置存储器类型
  7.   uint32_t FSMC_MemoryDataWidth;     //设置数据宽度
  8.   uint32_t FSMC_BurstAccessMode;     //成组模式同步模式才需要设置
  9.   uint32_t FSMC_AsynchronousWait;    //成组模式同步模式才需要设置
  10.   uint32_t FSMC_WaitSignalPolarity;  //成组模式同步模式才需要设置
  11.   uint32_t FSMC_WrapMode;            //成组模式同步模式才需要设置
  12.   uint32_t FSMC_WaitSignalActive;    //成组模式同步模式才需要设置
  13.   uint32_t FSMC_WriteOperation;      //设置写使能
  14.   uint32_t FSMC_WaitSignal;          //成组模式同步模式才需要设置
  15.   uint32_t FSMC_ExtendedMode;        //设置扩展模式使能位,也就是是否允许读写不同的时序
  16.   uint32_t FSMC_WriteBurst;          //成组模式同步模式才需要设置

  17.   FSMC_NORSRAMTimingInitTypeDef* FSMC_ReadWriteTimingStruct;//初始化片选控制寄存器FSMC_BTRx
  18.   FSMC_NORSRAMTimingInitTypeDef* FSMC_WriteTimingStruct;   //初始化写操作时序控制寄存器FSMC_BWTRx
  19. }FSMC_NORSRAMInitTypeDef;
【FSMC_NORSRAMTimingInitTypeDef】:
  1. typedef struct
  2. {
  3.   uint32_t FSMC_AddressSetupTime;      //地址建立保持时间
  4.   uint32_t FSMC_AddressHoldTime;       //地址保持时间
  5.   uint32_t FSMC_DataSetupTime;         //数据建立时间

  6.   uint32_t FSMC_BusTurnAroundDuration;  //总线周转期
  7.   uint32_t FSMC_CLKDivision;           //分频系数
  8.   uint32_t FSMC_DataLatency;           //

  9.   uint32_t FSMC_AccessMode;            //模式
  10. }FSMC_NORSRAMTimingInitTypeDef;
  C程序这里仅给出sram.c程序,在STM32F103平台使用的话加入.h 函数声明和FSMC 固件库文件 stm32f10x_fsmc.c、stm32f10x_fsmc.h 文件即可。
【sram.c】:
  1. //使用NOR/SRAM的 BANK 4,地址位HADDR[27,26]=10
  2. //对IS61LV25616/IS62WV25616,地址线范围为A0~A17
  3. //对IS61LV51216/IS62WV51216,地址线范围为A0~A18
  4. #define Bank1_SRAM3_ADDR    ((u32)(0x60000000 | 0x08000000))       
  5.                                                     
  6. //初始化外部SRAM
  7. void FSMC_SRAM_Init(void)
  8. {       
  9.         FSMC_NORSRAMInitTypeDef  FSMC_NORSRAMInitStructure;
  10.         FSMC_NORSRAMTimingInitTypeDef  readWriteTiming;
  11.         GPIO_InitTypeDef  GPIO_InitStructure;
  12. RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOD|RCC_APB2Periph_GPIOE|RCC_APB2Periph_GPIOF|RCC_APB2Periph_GPIOG|RCC_APB2Periph_AFIO,ENABLE);
  13.           RCC_AHBPeriphClockCmd(RCC_AHBPeriph_FSMC,ENABLE);
  14.   
  15.         GPIO_InitStructure.GPIO_Pin = 0xFF33;                                  //PORTD复用推挽输出
  16.         GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;                  //推挽输出
  17.         GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  18.         GPIO_Init(GPIOD, &GPIO_InitStructure);


  19.         GPIO_InitStructure.GPIO_Pin = 0xFF83;                                  //PORTE复用推挽输出
  20.         GPIO_Init(GPIOE, &GPIO_InitStructure);

  21.         GPIO_InitStructure.GPIO_Pin = 0xF03F;                                  //PORTD复用推挽输出
  22.         GPIO_Init(GPIOF, &GPIO_InitStructure);

  23.         GPIO_InitStructure.GPIO_Pin = 0x043F;                                  //PORTD复用推挽输出
  24.         GPIO_Init(GPIOG, &GPIO_InitStructure);

  25.                                              
  26.         readWriteTiming.FSMC_AddressSetupTime = 0x00;         //地址建立时间(ADDSET)为1个HCLK 1/36M=27ns
  27.     readWriteTiming.FSMC_AddressHoldTime = 0x00;         //地址保持时间(ADDHLD)模式A未用到       
  28.     readWriteTiming.FSMC_DataSetupTime = 0x03;                 //数据保持时间(DATAST)为3个HCLK 4/72M=55ns(对EM的SRAM芯片)         
  29.     readWriteTiming.FSMC_BusTurnAroundDuration = 0x00;
  30.     readWriteTiming.FSMC_CLKDivision = 0x00;
  31.     readWriteTiming.FSMC_DataLatency = 0x00;
  32.     readWriteTiming.FSMC_AccessMode = FSMC_AccessMode_A;         //模式A

  33.     FSMC_NORSRAMInitStructure.FSMC_Bank = FSMC_Bank1_NORSRAM3;//  这里我们使用NE3 ,也就对应BTCR[4],[5]。
  34.     FSMC_NORSRAMInitStructure.FSMC_DataAddressMux = FSMC_DataAddressMux_Disable;
  35.     FSMC_NORSRAMInitStructure.FSMC_MemoryType =FSMC_MemoryType_SRAM;//SRAM   
  36.     FSMC_NORSRAMInitStructure.FSMC_MemoryDataWidth = FSMC_MemoryDataWidth_16b;//存储器数据宽度为16bit  
  37.     FSMC_NORSRAMInitStructure.FSMC_BurstAccessMode =FSMC_BurstAccessMode_Disable;// FSMC_BurstAccessMode_Disable;
  38.     FSMC_NORSRAMInitStructure.FSMC_WaitSignalPolarity = FSMC_WaitSignalPolarity_Low;
  39.         FSMC_NORSRAMInitStructure.FSMC_AsynchronousWait=FSMC_AsynchronousWait_Disable;
  40.     FSMC_NORSRAMInitStructure.FSMC_WrapMode = FSMC_WrapMode_Disable;  
  41.     FSMC_NORSRAMInitStructure.FSMC_WaitSignalActive = FSMC_WaitSignalActive_BeforeWaitState;
  42.     FSMC_NORSRAMInitStructure.FSMC_WriteOperation = FSMC_WriteOperation_Enable;        //存储器写使能
  43.     FSMC_NORSRAMInitStructure.FSMC_WaitSignal = FSMC_WaitSignal_Disable;  
  44.     FSMC_NORSRAMInitStructure.FSMC_ExtendedMode = FSMC_ExtendedMode_Disable; // 读写使用相同的时序
  45.     FSMC_NORSRAMInitStructure.FSMC_WriteBurst = FSMC_WriteBurst_Disable;
  46.     FSMC_NORSRAMInitStructure.FSMC_ReadWriteTimingStruct = &readWriteTiming; // 读写使用相同的时序
  47.     FSMC_NORSRAMInitStructure.FSMC_WriteTimingStruct = &readWriteTiming;  // 读写使用相同的时序

  48.     FSMC_NORSRAMInit(&FSMC_NORSRAMInitStructure);  //初始化FSMC配置

  49.            FSMC_NORSRAMCmd(FSMC_Bank1_NORSRAM3, ENABLE);  // 使能BANK3                                                                                  
  50.                                                                                        
  51. }
  52.                                                                                                                             
  53. //在指定地址开始,连续写入n个字节.
  54. //pBuffer:字节指针
  55. //WriteAddr:要写入的地址
  56. //n:要写入的字节数
  57. void FSMC_SRAM_WriteBuffer(u8* pBuffer,u32 WriteAddr,u32 n)
  58. {
  59.         for(;n!=0;n--)  
  60.         {                                                                                    
  61.                 *(vu8*)(Bank1_SRAM3_ADDR+WriteAddr)=*pBuffer;          
  62.                 WriteAddr+=2;//这里需要加2,是因为STM32的FSMC地址右移一位对其.加2相当于加1.
  63.                 pBuffer++;
  64.         }   
  65. }                                                                                                                                                            
  66. //在指定地址开始,连续读出n个字节.
  67. //pBuffer:字节指针
  68. //ReadAddr:要读出的起始地址
  69. //n:要写入的字节数
  70. void FSMC_SRAM_ReadBuffer(u8* pBuffer,u32 ReadAddr,u32 n)
  71. {
  72.         for(;n!=0;n--)  
  73.         {                                                                                            
  74.                 *pBuffer++=*(vu8*)(Bank1_SRAM3_ADDR+ReadAddr);   
  75.                 ReadAddr+=2;//这里需要加2,是因为STM32的FSMC地址右移一位对其.加2相当于加1.
  76.         }  
  77. }



周半梅 发表于 2024-2-5 10:10 | 显示全部楼层

一种了解状态变化的简单方法
童雨竹 发表于 2024-2-5 13:09 | 显示全部楼层

待向GPIO(通用I/O端口)的输入从0变为1时,程序可以一定的间隔来检查GPIO的状态
Clyde011 发表于 2024-2-5 15:15 | 显示全部楼层

错误的比较大的Ⅵ乘积结果
公羊子丹 发表于 2024-2-5 16:08 | 显示全部楼层

定时器输出引脚的设定
Uriah 发表于 2024-2-5 18:14 | 显示全部楼层

使它们之间的电流通路尽可能短
您需要登录后才可以回帖 登录 | 注册

本版积分规则

50

主题

2016

帖子

1

粉丝
快速回复 在线客服 返回列表 返回顶部