欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

STM32之FSMC-SRAM例程

程序员文章站 2024-02-23 16:09:22
...

SRAM使用的是55ns的IS62WV51216,需要先分析IS62WV51216对读写的时序要求。

读时序

STM32之FSMC-SRAM例程STM32之FSMC-SRAM例程

 分析时序图,可以提取如下信息(不分析高低字节位和使能位,因为FSMC访问模式是一直保持的):整个读周期大于55ns、地址建立时间55ns且读使能25ns才能确保输出数据

 

写时序

STM32之FSMC-SRAM例程STM32之FSMC-SRAM例程

分析时序图,可以提取如下信息(不分析高低字节位和使能位,因为FSMC访问模式是一直保持的):整个写周期大于55ns、写使能时间大于45ns(tHZWE+tSD)才能保证数据写入

 

FSMC时序配置

根据手册模式1和模式A一般用于SRAM操作。注意:根据手册,模式1具有相同的读写时序,而模式A则是独立的读写时序。

下面分别结合两种模式进行时序分析:

模式1

STM32之FSMC-SRAM例程STM32之FSMC-SRAM例程

根据提取出来的要求:整个读周期大于55ns、地址建立时间55ns且读使能25ns才能确保输出数据、整个写周期大于55ns、写使能时间大于45ns(tHZWE+tSD)才能保证数据写入。

得出:

tHCLK * ((ADDSET + 1) + (DATAST + 1) + 2) >= 55ns

tHCLK * ((ADDSET + 1) + (DATAST + 1) ) >= 55ns(不应包括2周期存储时间)

tHCLK * ((ADDSET + 1) + (DATAST + 1) ) >= 25ns(不应包括2周期存储时间)

tHCLK * ((ADDSET + 1) + (DATAST + 1) ) >= 55ns

tHCLK * DATAST >= 45ns

STM32F103的HCLK配置为72MHz,则tHCLK=13.8ns。因此,DATAST >= 4且ADDSET >= 0。

 

模式A

STM32之FSMC-SRAM例程

根据提取出来的要求:整个读周期大于55ns、地址建立时间55ns且读使能25ns才能确保输出数据

得出:

tHCLK * ((ADDSET + 1) + (DATAST + 1) + 2) >= 55ns

tHCLK * ((ADDSET + 1) + (DATAST + 1) ) >= 55ns(不应包括2周期存储时间)

tHCLK * (DATAST + 1) >= 25ns(不应包括2周期存储时间)

STM32F103的HCLK配置为72MHz,则tHCLK=13.8ns。因此,DATAST = 1且ADDSET >= 1,DATAST >= 2且ADDSET >= 0。

 

STM32之FSMC-SRAM例程

根据提取出来的要求:整个写周期大于55ns、写使能时间大于45ns(tHZWE+tSD)才能保证数据写入

tHCLK * ((ADDSET + 1) + (DATAST + 1) ) >= 55ns

tHCLK * DATAST >= 45ns

STM32F103的HCLK配置为72MHz,则tHCLK=13.8ns。因此,DATAST >= 4且ADDSET >= 0。

 

源码例程

以模式A为例

#include "stm32f10x.h"

/* RCC时钟配置 */
void RCC_config()
{ 
	ErrorStatus HSEStartUpStatus;

	/* RCC寄存器设置为默认配置 */
	RCC_DeInit();
	/* 打开外部高速时钟 */
	RCC_HSEConfig(RCC_HSE_ON);
	/* 等待外部高速时钟稳定 */
	HSEStartUpStatus = RCC_WaitForHSEStartUp();
	if(HSEStartUpStatus == SUCCESS) 
	{ 
		/* 设置HCLK = SYSCLK */
		RCC_HCLKConfig(RCC_SYSCLK_Div1);
		/* 设置PCLK2 = HCLK */
		RCC_PCLK2Config(RCC_HCLK_Div1);
		/* 设置PCLK1 = HCLK / 2 */
		RCC_PCLK1Config(RCC_HCLK_Div2);
		/* 设置FLASH代码延时 */
		FLASH_SetLatency(FLASH_Latency_2);
		/* 使能预取址缓存 */
		FLASH_PrefetchBufferCmd(FLASH_PrefetchBuffer_Enable);
		/* 设置PLL时钟源为HSE倍频9 72MHz */
		RCC_PLLConfig(RCC_PLLSource_HSE_Div1, RCC_PLLMul_9);
		/* 使能PLL */
		RCC_PLLCmd(ENABLE);
		/* 等待PLL稳定 */
		while(RCC_GetFlagStatus(RCC_FLAG_PLLRDY) == RESET);
		/* 设置PLL为系统时钟源 */
		RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK);
		/* 等待系统时钟源切换到PLL */
		while(RCC_GetSYSCLKSource() != 0x08);
	}
}

/* GPIO配置 */
void GPIO_config(void)
{
	GPIO_InitTypeDef GPIO_InitStructure;
	
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOD | RCC_APB2Periph_GPIOE | RCC_APB2Periph_GPIOF | RCC_APB2Periph_GPIOG, ENABLE);
  
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1 | GPIO_Pin_4 | GPIO_Pin_5 | 
																GPIO_Pin_8| GPIO_Pin_9 | GPIO_Pin_10 | GPIO_Pin_11 | 
																GPIO_Pin_12 | GPIO_Pin_13 | GPIO_Pin_14 | GPIO_Pin_15;
 	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
 	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
 	GPIO_Init(GPIOD, &GPIO_InitStructure);

	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1 | GPIO_Pin_7 | GPIO_Pin_8 | 
																GPIO_Pin_9 | GPIO_Pin_10 | GPIO_Pin_11 | GPIO_Pin_12 | 
																GPIO_Pin_13 | GPIO_Pin_14 | GPIO_Pin_15;
 	GPIO_Init(GPIOE, &GPIO_InitStructure);
	
 	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1 | GPIO_Pin_2 | GPIO_Pin_3 | 
																GPIO_Pin_4 | GPIO_Pin_5 | GPIO_Pin_12 | GPIO_Pin_13 | 
																GPIO_Pin_14 | GPIO_Pin_15;
 	GPIO_Init(GPIOF, &GPIO_InitStructure);

	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1 | GPIO_Pin_2 | GPIO_Pin_3 | 
																GPIO_Pin_4 | GPIO_Pin_5 | GPIO_Pin_10;
 	GPIO_Init(GPIOG, &GPIO_InitStructure);
}

/* FSMC配置 */
void FSMC_SRAM_Init(void)
{
	FSMC_NORSRAMInitTypeDef FSMC_NORSRAMInitStructure;
	FSMC_NORSRAMTimingInitTypeDef readTiming;
	FSMC_NORSRAMTimingInitTypeDef WriteTiming;
	
	RCC_AHBPeriphClockCmd(RCC_AHBPeriph_FSMC, ENABLE);
	
	FSMC_NORSRAMInitStructure.FSMC_Bank = FSMC_Bank1_NORSRAM3;																//Bank
	FSMC_NORSRAMInitStructure.FSMC_DataAddressMux = FSMC_DataAddressMux_Disable;							//数据和地址线是否复用
	FSMC_NORSRAMInitStructure.FSMC_MemoryType = FSMC_MemoryType_SRAM;													//存储器类型
	FSMC_NORSRAMInitStructure.FSMC_MemoryDataWidth = FSMC_MemoryDataWidth_16b;								//存储器数据宽度  
	FSMC_NORSRAMInitStructure.FSMC_BurstAccessMode = FSMC_BurstAccessMode_Disable;						//是否进行突发模式访问
	FSMC_NORSRAMInitStructure.FSMC_WaitSignalPolarity = FSMC_WaitSignalPolarity_Low;					//等待信号极性(突发模式下)
	FSMC_NORSRAMInitStructure.FSMC_AsynchronousWait = FSMC_AsynchronousWait_Disable;					//保留位
	FSMC_NORSRAMInitStructure.FSMC_WrapMode = FSMC_WrapMode_Disable;   												//是否支持非对齐操作(突发模式下)
	FSMC_NORSRAMInitStructure.FSMC_WaitSignalActive = FSMC_WaitSignalActive_BeforeWaitState;  //WAIT信号有效时机(突发模式下)
	FSMC_NORSRAMInitStructure.FSMC_WriteOperation = FSMC_WriteOperation_Enable;								//是否允许写操作
	FSMC_NORSRAMInitStructure.FSMC_WaitSignal = FSMC_WaitSignal_Disable;  										//是否使用WAIT信号(突发模式下)
	FSMC_NORSRAMInitStructure.FSMC_ExtendedMode = FSMC_ExtendedMode_Enable; 									//是否使用读写不同时序
	FSMC_NORSRAMInitStructure.FSMC_WriteBurst = FSMC_WriteBurst_Disable;											//是否允许突发写操作

	readTiming.FSMC_AddressSetupTime = 0x00;					//地址建立时间
	readTiming.FSMC_AddressHoldTime = 0x00;					//地址保持时间
	readTiming.FSMC_DataSetupTime = 0x02;		 				//数据保持时间
	readTiming.FSMC_BusTurnAroundDuration = 0x00;
	readTiming.FSMC_CLKDivision = 0x00;
	readTiming.FSMC_DataLatency = 0x00;
	readTiming.FSMC_AccessMode = FSMC_AccessMode_A;	//模式A
	
	WriteTiming.FSMC_AddressSetupTime = 0x00;					//地址建立时间
	WriteTiming.FSMC_AddressHoldTime = 0x00;					//地址保持时间
	WriteTiming.FSMC_DataSetupTime = 0x04;		 				//数据保持时间
	WriteTiming.FSMC_BusTurnAroundDuration = 0x00;
	WriteTiming.FSMC_CLKDivision = 0x00;
	WriteTiming.FSMC_DataLatency = 0x00;
	WriteTiming.FSMC_AccessMode = FSMC_AccessMode_A;	//模式A	

	FSMC_NORSRAMInitStructure.FSMC_ReadWriteTimingStruct = &readTiming;
	FSMC_NORSRAMInitStructure.FSMC_WriteTimingStruct = &WriteTiming;

	FSMC_NORSRAMInit(&FSMC_NORSRAMInitStructure);
	FSMC_NORSRAMCmd(FSMC_Bank1_NORSRAM3, ENABLE);	
}

uint8_t testsram[1000000] __attribute__((at(0x68000000)));
uint8_t test;

int main(void)
{	
	uint8_t count = 0;
	
	/* 时钟配置 */
	RCC_config();
	
	/* GPIO配置 */
	GPIO_config();
	
	/* FSMC配置 */
	FSMC_SRAM_Init();
	
	for(uint32_t ts = 0; ts < 1000000; ts++)
		testsram[ts] = count++;
	
	for(uint32_t ts = 0; ts < 1000000; ts++)
		test = testsram[ts];
	
	while(1)
	{

	}
}