STM32 FMC篇-SDRAM(IS42S16400J)

news2024/11/23 19:52:46

IS42S16400J

这个东西太常见啦,长方形的。不会过多解释,详细请阅读它的数据手册。

IS42S16400J是一种高速同步动态随机存储器(SDRAM),64Mb的存储容量,采用4个bank,每个bank大小为16Mb,总线宽度为16位,工作电压为3.3V。它是一种常用的存储器芯片,广泛应用于嵌入式系统、通信设备、计算机等领域。

在STM32上使用IS42S16400J,需要先配置STM32的外部存储器控制器(FSMC、FMC)。FSMC是STM32的一种外设,用于连接外部存储器,包括NOR Flash、NAND Flash、SRAM和SDRAM等。在使用IS42S16400J时,需要将FSMC、FMC配置为SDRAM模式,并设置好时序参数。

以下是在STM32上使用IS42S16400J的基本步骤:

配置FSMC控制器,设置SDRAM模式和时序参数。

初始化SDRAM,包括对SDRAM进行预充电和初始化操作。

使用SDRAM进行数据读写操作,可以通过FSMC接口进行读写。

需要注意的是,在使用SDRAM时,需要遵循SDRAM的时序要求,包括时钟频率、时序延迟等参数。同时,还需要注意SDRAM的电源和地线连接,以确保正常工作。

总之,IS42S16400J是一种常用的存储器芯片,可以在STM32等嵌入式系统中使用。在使用时,需要配置好FSMC控制器,并遵循SDRAM的时序要求,以确保正常工作。

常见引脚定义

读写时A0~A11作为行地址线输入,A0~A7作为列地址线输入。

一个bank有12根行地址、8根列地址,总线宽度为16位,所以SDRAM容量=4*4096*256*16=64Mb=8MB。

CKE高电平时使用AUTO REFRESH,通常显高电平,只有在使用SELF REFRESH时CKE低电平。

在写入(0个时钟延迟)和读取(2个时钟延迟)期间激活或停用DQ。

在配置SDRAM工作状态时发送对应指令。

命令

SELF REFRESH COMMAND,自我刷新命令 

DRAM使用电容存储电荷,电容会放电。

自刷新:在电容存储的电荷丢失前读取存储的数据,再重新写进去。刷新周期为64ms,即64ms电容数据会丢失,所以需要保证在64ms内对bank内所有单元均完成一次自刷新。一个bank有4096行,所以需要64ms/4096=15.625us发送一个自刷新指令。

芯片内部在自我刷新操作时,需要刷新的行地址、bank和刷新间隔会自动生成。即使系统断电,自我刷新也可以在没有外部时钟时保留SDRAM数据。

CKE引脚(HIGH->LOW)启动自我刷新操作,自我刷新操作期间,无视所有引脚输入,在设备内部恢复期(tRC)前不能执行下一个命令。

一旦CKE恢复HIGH,必须发出NOP命令(至少两个时钟),以便为完成正在进行的任何内部刷新提供时间。

AUTO REFRESH COMMAND,自动刷新命令

在SELF REFRESH完成之后,由于不确定刷新的最后一行地址,因此应该立即对所有地址执行AUTO REFRESH,自动生成需要刷新的行地址、bank。

ACTIVE AOMMAND,激活命令

在读写操作前,一般需要进行激活操作。

因为SDRAM的行地址线和列地址线都是同一组地址线,分时复用。读写操作发送的是列地址,所以在读写操作之前需要确定行地址和bank。确定行地址后还需要tRCD时间才能把该行打开才可以对该行数据进行读写操作。

激活:BA0、BA1选择bank,A0~A11选择行。在向bank发出预充电指令前,该行将保持打开状态以供访问。

PRECHARGE,预充电命令

在读写操作完成后,所在行(所有行)仍处于打开状态,需要使用预充电命令关闭所在行(所有行),才能使用激活命令打开其它行进行读写操作。

预充电时A10高电平表示对所有bank预充电,低电平表示对某个bank(由BA0、BA1决定)进行预充电

预充电命令用于停用某个bank的空行或所有bank的空行。

执行预充电命令后,所选bank的下一个命令将在经过时段tRP(bank预充电所需的时段)后执行。一旦某个bank已预充电,它将处于空闲状态。

AUTO PRECHARGE,自动预充电命令

自动预充电命令可确保在脉冲串内最早的有效阶段启动预充电。此功能允许单个bank预充电,而无需显式命令。A10高电平可与读写命令一起用于启动自动预充电功能

对于单个单独的读写命令,启用和禁用自动预充电。除非在整页突发模式下,否则自动预充电不适用。读写突发完成后,将自动(不需要发送自动预充电)对寻找的bank或行进行预充电。

READ,读命令

读命令从BA0、BA1输入中选择bank,并启动突发读访问已激活的行。输入A0~A7提供起始列位置。

当A10高电平时,该命令作为自动预充电命令运行。当选择自动预充电时,正在访问的行将在读突发结束时预充电。未选择自动预充电时,该行将保持打开状态,以便后续访问。

DQ的读数据受两个时钟之前(与潜伏期无关)DQM输入上的逻辑电平的影响。

当给定的DQM信号被记录为高电平时,对应的DQ将在两个时钟后变成高阻态。当给定的DQM信号被记录为低电平时,对应的DQ将提供有效数据

发送读指令后,SDRAM会经过2~3个时钟周期才会给出数据(潜伏期,CAS延迟)。

WRITE,写命令

写命令启动对活动行的突发写访问。从BA0、BA1输入中选择bank,从A0~A7输入中提供起始列位置。

当A10高电平时,该命令作为自动预充电命令运行。当选择自动预充电时,正在访问的行将在写突发结束时预充电。未选择自动预充电时,该行将保持打开状态,以便后续访问。

写入内存阵列时,DQ和DQM输入逻辑级上的相应输入数据同时出现。

当给定的DQM信号被记录为高电平时,将忽略相应的数据输入,并且不会对其执行写操作。当给定的DQM信号被记录为低电平时,数据写入存储器。

LOAD MODE REGISTER,加载模式寄存器命令

在加载模式寄存器命令期间,模式寄存器从A0~A11加载。此命令只能在所有bank空闲时发出。

BURST TERMINATE,突发终止命令

突发终止命令通过截断固定长度或整页突发以及突发终止前最近注册的读写命令,强制终止突发读写操作,可以通过此命令结合全页读写模式实现任意长度的读写操作。

COMMAND INHIBIT,命令禁止

命令禁止表示禁止执行新的命令。除CLK信号是否启用外,正在进行的操作不受影响。与空指令(NO OPERATION)效果差不多,当CS低电平时NOP无操作,NOP命令可防止在空闲或等待状态下注册不需要的命令。

上电初始化

SDRAM必须按预定义的方式上电并初始化。

在VDD和VDDQ同时通电后,64Mb SDRAM初始化,DQM和CKE高电平时时钟稳定。

除了命令禁止和NOP外,在发出任何命令之前都需要100us的命令。可以在100us的时间段内应用命令禁止和NOP命令,并应至少持续到时间段结束。如果至少应用了一个命令禁止或NOP命令,则应在满足100us延迟后应用预充电命令。

所有bank都必须预充电,使得所有bank处于空闲状态,之后必须执行至少两个自动预充电周期,自动预充电周期完成后,SDRAM就可以进行模式寄存器编程了。

应该在应用任何操作命令之前加载模式寄存器,因为它将在未知状态下上电。在加载模式寄存器命令之后,必须在任何命令之前断言至少一个NOP命令。

模式寄存器

用于定义SDRAM的特定操作模式。通过LOAD MODE REGISTER命令应用,并将保留存储的信息,直到再次应用或设备失去电源。

模式寄存器必须在所有bank空闲时加载,控制器必须等待指定时间才能启动后续操作。违反任何一个要求都将导致未指定的操作。

CAS延迟

CAS延迟是以时钟周期为单位,从注册一个READ命令到第一个输出数据可用之间的延迟。可以设置为2~3个时钟周期。

实验环节

STM32CubeMX配置

 配置完成后导出工程,还需要额外的配置模式寄存器。创建drive_sdram.c和.h文件,存放于UserCode/Drivers子目录下。

drive_sdram.c

/* Includes ------------------------------------------------------------------*/
#include "drive_sdram.h"
#include "fmc.h"
#include "main.h"
#include "cmsis_os.h"

/* Private includes ----------------------------------------------------------*/

/* Private typedef -----------------------------------------------------------*/

/* Private define ------------------------------------------------------------*/
#define SDRAM_MODEREG_BURST_LENGTH_1             	((uint16_t)0x0000)
#define SDRAM_MODEREG_BURST_LENGTH_2             	((uint16_t)0x0001)
#define SDRAM_MODEREG_BURST_LENGTH_4             	((uint16_t)0x0002)
#define SDRAM_MODEREG_BURST_LENGTH_8             	((uint16_t)0x0003)
#define SDRAM_MODEREG_BURST_TYPE_SEQUENTIAL      	((uint16_t)0x0000)
#define SDRAM_MODEREG_BURST_TYPE_INTERLEAVED     	((uint16_t)0x0008)
#define SDRAM_MODEREG_CAS_LATENCY_2              	((uint16_t)0x0020)
#define SDRAM_MODEREG_CAS_LATENCY_3              	((uint16_t)0x0030)
#define SDRAM_MODEREG_OPERATING_MODE_STANDARD    	((uint16_t)0x0000)
#define SDRAM_MODEREG_WRITEBURST_MODE_PROGRAMMED 	((uint16_t)0x0000) 
#define SDRAM_MODEREG_WRITEBURST_MODE_SINGLE     	((uint16_t)0x0200) 

#define IS42S16400J_SIZE             				0x400000
#define SDRAM_BANK_ADDR     					 	((uint32_t)0xD0000000)

/* Private macro -------------------------------------------------------------*/

/* Private variables ---------------------------------------------------------*/
uint32_t pbuf[(1024*1024*64)/32] __attribute__((at(0xD0000000)));
/*******************************************
int x __attribute__((at(ADDR)));
可以将x存放在指定的地址ADDR中
********************************************/

/* Private function prototypes -----------------------------------------------*/
void SDRAM_InitSequence(void);

/* Private user code ---------------------------------------------------------*/

void SDRAM_Init(void)
{
    SDRAM_InitSequence();
    HAL_SDRAM_WriteProtection_Disable(&hsdram1);
}

void SDRAM_InitSequence(void)
{
  FMC_SDRAM_CommandTypeDef FMC_SDRAMCommandStructure;
  uint32_t tmpr = 0;
  
/* Step 3 --------------------------------------------------------------------*/
  /* Configure a clock configuration enable command */
  FMC_SDRAMCommandStructure.CommandMode = FMC_SDRAM_CMD_CLK_ENABLE;
  FMC_SDRAMCommandStructure.CommandTarget = FMC_SDRAM_CMD_TARGET_BANK2;
  FMC_SDRAMCommandStructure.AutoRefreshNumber = 1;
  FMC_SDRAMCommandStructure.ModeRegisterDefinition = 0;
  /* Wait until the SDRAM controller is ready */ 
  while(HAL_SDRAM_GetState(&hsdram1) == HAL_SDRAM_STATE_BUSY)
  {
  }
  /* Send the command */
  HAL_SDRAM_SendCommand(&hsdram1, &FMC_SDRAMCommandStructure, 0x1000);
    
  
/* Step 4 --------------------------------------------------------------------*/
  /* Insert 100 ms delay */
  HAL_Delay(100);
    
/* Step 5 --------------------------------------------------------------------*/
  /* Configure a PALL (precharge all) command */ 
  FMC_SDRAMCommandStructure.CommandMode = FMC_SDRAM_CMD_PALL;
  FMC_SDRAMCommandStructure.CommandTarget = FMC_SDRAM_CMD_TARGET_BANK2;
  FMC_SDRAMCommandStructure.AutoRefreshNumber = 1;
  FMC_SDRAMCommandStructure.ModeRegisterDefinition = 0;
  /* Wait until the SDRAM controller is ready */ 
  while(HAL_SDRAM_GetState(&hsdram1) == HAL_SDRAM_STATE_BUSY)
  {
  }
  /* Send the command */
  HAL_SDRAM_SendCommand(&hsdram1, &FMC_SDRAMCommandStructure, 0x1000);
  
/* Step 6 --------------------------------------------------------------------*/
  /* Configure a Auto-Refresh command */ 
  FMC_SDRAMCommandStructure.CommandMode = FMC_SDRAM_CMD_AUTOREFRESH_MODE;
  FMC_SDRAMCommandStructure.CommandTarget = FMC_SDRAM_CMD_TARGET_BANK2;
  FMC_SDRAMCommandStructure.AutoRefreshNumber = 2;
  FMC_SDRAMCommandStructure.ModeRegisterDefinition = 0;
  /* Wait until the SDRAM controller is ready */ 
  while(HAL_SDRAM_GetState(&hsdram1) == HAL_SDRAM_STATE_BUSY)
  {
  }
  /* Send the  first command */
  HAL_SDRAM_SendCommand(&hsdram1, &FMC_SDRAMCommandStructure, 0x1000);
  
/* Step 7 --------------------------------------------------------------------*/
  /* Program the external memory mode register */
  tmpr = (uint32_t)SDRAM_MODEREG_BURST_LENGTH_1          |
                   SDRAM_MODEREG_BURST_TYPE_SEQUENTIAL   |
                   SDRAM_MODEREG_CAS_LATENCY_3           |
                   SDRAM_MODEREG_OPERATING_MODE_STANDARD |
                   SDRAM_MODEREG_WRITEBURST_MODE_SINGLE;
  
  /* Configure a load Mode register command*/ 
  FMC_SDRAMCommandStructure.CommandMode = FMC_SDRAM_CMD_LOAD_MODE;
  FMC_SDRAMCommandStructure.CommandTarget = FMC_SDRAM_CMD_TARGET_BANK2;
  FMC_SDRAMCommandStructure.AutoRefreshNumber = 1;
  FMC_SDRAMCommandStructure.ModeRegisterDefinition = tmpr;
  /* Wait until the SDRAM controller is ready */ 
  while(HAL_SDRAM_GetState(&hsdram1) == HAL_SDRAM_STATE_BUSY)
  {
  }
  /* Send the command */
  HAL_SDRAM_SendCommand(&hsdram1, &FMC_SDRAMCommandStructure, 0x1000);
  
/* Step 8 --------------------------------------------------------------------*/

  /* Set the refresh rate counter */
  /* SDRAM Refresh Timer register(FMC_SDRTR)
	refresh rate = (sdram refresh rate * sdram clock frequency) - 20
	sdram refresh rate = sdram refresh period / Number of rows
  
	example:
		sdram refresh rate = 64ms / 4096 = 15.625us
		refresh rate = (15.625 * 90Mhz) - 20 = 1386.25
  */
  /* Set the device refresh counter */
  HAL_SDRAM_ProgramRefreshRate(&hsdram1, 1386);
  /* Wait until the SDRAM controller is ready */ 
  while(HAL_SDRAM_GetState(&hsdram1) == HAL_SDRAM_STATE_BUSY)
  {
  }
}

SDRAM的起始地址可以看STM32F429的数据手册。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/549884.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【CW32开发】00 开发环境搭建和示例代码运行

1.下载MDK 2.下载芯片相关的固件库 我用的是芯片是CW32F030系列,所以下载相应的固件库 下载地址:https://www.whxy.com/support/filelist/13 3.安装固件库 解压下载的文件,并在cw32f030-stdperiph-lib\IdeSupport\MDK路径下安装固件库 …

064:cesium设置点划线材质(material-8)

第064个 点击查看专栏目录 本示例的目的是介绍如何在vue+cesium中设置点划线材质,请参考源代码,了解PolylineDashMaterialProperty的应用。 直接复制下面的 vue+cesium源代码,操作2分钟即可运行实现效果. 文章目录 示例效果配置方式示例源代码(共89行)相关API参考:专栏…

CBFS Vault 2022 for .NET Crack

将多个文件打包到一个 Vault - 一个“文件中的文件系统”,完成每个文件的压缩、透明加密和随机读/写访问。 亮点包括新的日记选项、用于更好地控制和跟踪的新事件,以及一系列核心性能和可用性改进 [了解更多]。 CBFS保险库 在任何地方存储一个完整的文件…

chatgpt赋能Python-pythone怎么下载

Python下载:在SEO中的作用 介绍 Python是一种广泛应用于编程、自动化任务和数据处理的强大语言。在SEO中,Python也扮演了越来越重要的角色。Python有许多库和工具可供下载,这使得它成为了SEO领域中必不可少的一部分。在本文中,我…

人工智能之读懂CNN卷积神经网络

通过往期文章的分享,我们了解了神经网络的结构,一般分为输入层,隐藏层,输出层 TensorFlow神经网络 那什么是卷积神经网络那,这就要我们追溯一下人类识别图像的原理 人类的视觉原理如下:从原始信号摄入开始(瞳孔摄入像素 Pixels),接着做初步处理(大脑皮层某些细胞发现…

chatgpt赋能Python-pythondna匹配

Python DNA匹配:从概念到应用 概述 DNA匹配是生物学中的一个重要领域,它意味着找到两个DNA序列之间的相似性并确定它们之间的关系。Python是一种被广泛用于生物信息学中的编程语言,它提供了一系列强大的库和工具来处理DNA序列数据&#xff…

【C++从0到王者】第七站:内存管理(520没有对象?那就new一个)

文章目录 一、C/C内存分区二、C语言中动态内存管理方式三、C内存管理方式1.new/delete操作内置类型2.C为什么要搞new和delete3.C中new和delete操作自定义类型4.非要乱用delete和free会造成什么后果? 四、 operator new与operator delete函数五、operator new和opera…

在安卓中压缩GIF的几种方法(附实例代码)

前言 最近在划水摸鱼的时候,看到有位大佬发了一篇 GIF 压缩思路的文章。 让我突然想起来,很久以前我在我的项目 隐云图解制作 中就实现了一个动图工具箱,其中一个功能就是压缩GIF。 不过这位大佬只介绍了其中几种使用方法,还有…

Java动态类型语言支持

JDK7发布字节码首位新成员——invokedynamic指令。以实现动态类型语言支持。也是为JDK8里可以顺利实现Lambda表达式而做的技术储备。我们将在本文详细了解动态语言支持这项特性出现的背景和它的意义与价值。 1 动态类型语言 动态类型语言的关键特征是它的类型检查的主体过程是…

MJ discord 添加应用配置

discord 添加机器人 https://discord.com/developers/applications 刷新token后显示,即机器人Token,后续配置到 mj.discord.bot-token 如图勾选后,打开url进行授权 选择Midjourney Bot所在的服务器 勾上这两个选项,点击 Save Cha…

图片转Excel表格,识别准确率的重要性

摘要:随着科技的不断发展,图片转Excel表格的应用越来越广泛。通过OCR技术实现图片转Excel表格,OCR识别准确率的提高对于信息录入的精度以及后续数据分析的可靠性具有非常重要的作用。本文探讨了影响OCR识别准确率的因素,并提出了提…

2.MATLAB篇——基本操作与矩阵输入

>> cos(((12345)^5)^0.5)ans -0.3623>> help sinsin - 参数的正弦,以弧度为单位此 MATLAB 函数 返回 X 的元素的正弦。sin 函数按元素处理数组。该函数同时接受实数和复数输入。 对于 X 的实数值,sin(X) 返回区间 [-1, 1] 内的实数值。 对于…

数据结构-单调栈2

这里是解了一个新的题目,然后对于有重复值的单调栈做了一些改进(只适用于特殊题目):有重复值的单调栈不再使用ArrayList或者LinkedList,而是像无重复值的那样直接使用下标,这种方法能保证最终的正确性&…

Python图片转字符画,太好玩啦(46)

小朋友们好,大朋友们好! 我是猫妹,一名爱上Python编程的小学生。 和猫妹学Python,一起趣味学编程。 今日主题 什么是字符画呢? 如何用Python将图片转为字符画? 我们先找一幅原画,比如它吧&…

亚马逊云科技 一周回顾 – 2022 年 7 月 18 日

上周,Amazon 峰会(纽约)在贾维茨中心线下举办,有数千名与会者以及 100 多家赞助商和合作伙伴参加。在主题演讲中,Amazon 首席开发人员倡导者 Martin Beeby 谈到了云基础设施的创新如何帮助客户适应挑战并抓住新的机遇。…

知识积累(1)

(1) 当您在Git中看到消息 "HEAD is now at 1343ccb FAB-17419 Fix off_chain_data sample error (#146)" 时,这是Git告知您当前所在的分支和最新的提交哈希。 这条消息通常出现在使用Git命令后,如git pull或git check…

【Java校招面试】实战面经(四)

目录 前言一、Http协议状态码301和302的区别二、Time Wait状态的作用是什么?三、ConcurrentHashMap在JDK1.7和JDK1.8的区别四、MySQL的优化:怎么优化SQL、用过MySQL的性能分析工具吗?五、反转数组的算法六、JDBC怎么使用的,什么是…

Redis三种模式——主从复制、哨兵模式、集群

目录 一、Redis模式二、Redis主从复制2.1 主从复制概述2.2 主从复制2.3 Redis主从复制过程2.4 搭建Redis主从复制2.4-1 环境部署2.4-2 安装Redis2.4-3 修改 Redis 配置文件(Master节点操作)2.4-4 修改 Redis 配置文件(Slave节点操作&#xff…

『手撕 Mybatis 源码』03 - 解析映射配置文件

解析映射配置文件 SQL 映射文件只有很少的几个顶级元素&#xff08;按照定义顺序列出&#xff09; select 元素允许你配置很多属性来配置每条语句的行为细节 <selectid"select"parameterType"int"parameterMap"deprecated"resultType&quo…

Qt之界面 自定义标题栏、无边框、可移动、缩放

实现效果 注意&#xff1a;由于需要调用 Windows 上的头文件与库&#xff0c;所以不能跨平台&#xff0c;只支持 Windows 系统。如果想要跨平台&#xff0c;可以使用鼠标等事件实现&#xff0c;具体百度搜索参考下 自定义标题栏 titleBar.h #ifndef TITLEBAR_H #define TITL…