STM32复习笔记(五):FSMC连接外部SRAM

目录

Preface:

(一)原理相关

(二)CUBEMX配置

(三)轮询方式读写

(四)DMA方式读写


Preface:

STM32F4有一个FSMC(Flexible Static Memory Controller,可变静态存储控制器),可以用来驱动8080接口的TFT LCD,我之前就写过一篇blog,是用FSMC来驱动4.3寸液晶屏;此外,还可以用FSMC来连接外部的各种存储器,比如说SRAM、NOR FLASH、PSRAM等等;但是每个区(Bank)的功能是不一样的;Bank1可以连接多达4个NOR FLASH或PSRAM/SRAM存储器件(通过片选);Bank2和Bank3只能用于访问NAND FLASH,且每个Bank只能连一个设备;Bank4只能用于连接PC Card设备。


(一)原理相关

STM32F4的FSMC控制器的存储区分为4个区,分别为Bank 1~Bank 4,每个Bank大小为2e28个字节,即256MB,因此总共管理的内存可达到1GB;而每个Bank又分成4个子区,每个子区64MB;Bank1的地址范围为0x6000 0000h~6FFF FFFFh,Bank2的地址范围为0x7000 0000h~7FFF FFFFh,Bank3的地址范围为0x8000 0000h~8FFF FFFFh,Bank4的地址范围为0x9000 0000h~9FFF FFFFh;如下图:

对于STM32F407ZGT6来说,其内部SRAM为192kB,一般的应用程序是足够用了,但是在使用GUI(特别是要做得很炫酷那种)等需要大量内存的功能时,192kB是不太够的,可能就需要扩展SRAM了。FSMC连接PSRAM/SRAM设备时,接口线的功能如下表所示: 

根据开发板的原理图(如下图)可知,FSMC的NE3线连接到了外部SRAM的片选,而由于只有Bank1才能连接SRAM,所以可知板子用的是FSMC的Bank1的子区3来连接外部SRAM;

该SRAM芯片为IS62WV51216,这是一个16位宽512K容量(512K×16位,即1024KB)的静态内存芯片。它与MCU的连接电路如下图所示。芯片几个主要管脚的功能,以及与MCU的连接原理如下:

  • A0至A18是19根地址线,连接FSMC的19根地址线,即FSMC_A0至FSMC_A18;
  • I/O0至I/O15是16位数据线,连接FSMC的FSMC_D0至FSMC_D15数据线;
  • CE是芯片的片选信号,连接MCU的FSMC_NE3(PG10引脚),也就是Bank1子区3的片选信号;
  • OE是输出使能信号,连接MCU的FSMC_NOE(PD4引脚),是读数据时的使能信号;
  • WE是写使能信号,连接MCU的FSMC_NWE (PD5引脚),是写数据使能信号;
  • UB是高字节使能信号,连接MCU的FSMC_NBL[1](PE1引脚);LB是低字节使能信号,连接MCU的FSMC_NBL[0](PE0引脚);通过UB和LB的控制可以只读取一个地址的高字节(I/O8~I/O15)或低字节(I/O0~I/O7),或读取16位数据;

IS62WV51216有19根地址线,能表示的地址范围是512K,而数据宽度是16位(2B),因此实际存储容量是1024KB,偏移地址范围是0x00000~0x7FFFF。又因为Bank1子区3的起始地址是0x68000000,所以IS62WV51216的全部1024KB的地址范围是 0x68000000~0x680FFFFF。FSMC_NBL[1]和FSMC_NBL[0]分别控制高位字节和低位字节访问,实现全部1024KB存储空间的按字节访问。


(二)CUBEMX配置

cubemx中的FSMC模式配置如下(选择子区3,片选为NE3;Mem类型SRAM;地址19位;数据16位;Wait是PSRAM芯片发送给FSMC的等待输入信号,IS62WV51216没有该线,所以disable掉;最后勾上Byte enable,允许字节访问):

开启之后再对照原理图看一遍发现引脚刚好与开发板上的一致,因此无需更改引脚重映射:

接下来进行参数配置;首先是控制参数:Memory type只能选SRAM;Bank只能选Bank1子区3,这两项与模式设置部分是一一对应的;Write operation设置为Enabled,表示使能写操作;Extended mode设置为Disabled,FSMC自动使用模式A对SRAM进行操作,SRAM的读写操作速度基本相同,所以读写操作可以使用相同的时序参数,无需使用扩展模式单独设置读时序和写时序;接下来是时序参数:地址建立时间ADDSET,设置范围为0~15,设置为0即可;数据建立时间DATASET,设置范围为1~255,设置为8;总线翻转时间,设置范围为0~15,设置为0即可;

另外,因为FSMC参数设置部分没有DMA设置页面,如果要用DMA的话需要去System Core的DMA里面手动创建,并且在代码里要手动LINK DMA;

如下图所示:

然后,因为代码里会使用随机数生成器,所以打开Security分组下的RNG模块,启用RNG;RNG需要用到48MHz时钟,时钟树上可能会提示错误;单击时钟树界面上的Resolve Clock Issues,让cubemx自动解决即可:

 

配置好后直接生成代码即可


(三)轮询方式读写

首先加入3个宏,分别表示Bank1子区3的SRAM起始地址、中间地址、结束地址,如下所示:

#define SRAM_ADDR_BEGIN     0x68000000UL    //Bank1子区3的SRAM起始地址
#define SRAM_ADDR_HALF      0x68080000UL    //SRAM中间地址,一共512KB
#define SRAM_ADDR_END       0x680FFFFFUL    //SRAM结束地址,一共1024KB

然后封装一下读取、写入数据;如下:

#include "fsmc_func.h"
#include "fsmc.h"
#include "rng.h"/** 用HAL函数写入数据* */
HAL_StatusTypeDef SRAM_WriteByFunc() {HAL_StatusTypeDef status = HAL_OK;uint8_t str[] = "Input Data";   //待写入字符uint16_t length = sizeof (str); //数据长度(注意是字节数),包括'\0'auto *paddr = (uint32_t*)(SRAM_ADDR_BEGIN); //目标地址//写入字符串if (HAL_OK == HAL_SRAM_Write_8b(&hsram3, paddr, str, length)) {HAL_Delay(1);} else {status = HAL_ERROR;}//写入数字uint32_t num = 0;paddr = (uint32_t*)(SRAM_ADDR_HALF); //修改目标地址HAL_RNG_GenerateRandomNumber(&hrng, &num);  //生成随机数if (HAL_OK == HAL_SRAM_Write_32b(&hsram3, paddr, &num, 1)) {HAL_Delay(1);} else {status = HAL_ERROR;}return status;
}
/** 用HAL函数读取数据* */
HAL_StatusTypeDef SRAM_ReadByFunc() {HAL_StatusTypeDef status = HAL_OK;auto *paddr = (uint32_t*)(SRAM_ADDR_BEGIN);uint8_t str[30];uint16_t length = 30;   //读取字节数//读取字符if (HAL_OK == HAL_SRAM_Read_8b(&hsram3, paddr, str, length)) {HAL_Delay(1);} else {status = HAL_ERROR;}//读取数字uint32_t num = 0;paddr = (uint32_t*)(SRAM_ADDR_HALF);if (HAL_OK == HAL_SRAM_Read_32b(&hsram3, paddr, &num, 1)) {HAL_Delay(1);} else {status = HAL_ERROR;}return status;
}

 然后在主函数中输入测试代码,测试是否正确写入、读出:

进入调试,跑了上半部分,status为HAL_OK,说明成功写入字符串:

 跑完下半部分,status为HAL_OK,说明成功写入随机数num:

接下来是读出调试;重新读出SRAM开始出的字符,发现前面部分与刚刚写入的字符串一模一样,且status仍为HAL_OK,表示成功写入,读出字符串:

接着重新读出SRAM中间部分的一个32位数字,发现status为HAL_OK,说明读取成功,并且能看到num中的数字与刚刚写入的随机数一模一样,表示成功写入、读出数字:

除此之外,因为这个扩展RAM本质上还是存储器,所以还可以不使用HAL库函数,直接使用指针读取指定地址的内容;STM32是32位机器,最大能够管理的地址空间为2e32 = 4GB,只要在0x0000 0000h~0xFFFF FFFFh中实际存在的地址,STM32都能访问;下面代码是通过指针直接访问对应地址中的内容:

/** 用指针写入数据* */
void SRAM_WriteByPointer() {uint16_t num = 100;uint16_t *paddr_16b = (uint16_t*)(SRAM_ADDR_BEGIN); //uint16_t类型的指针for (int i = 0; i < 5; ++i) {num += 10;*paddr_16b = num;   //指定地址写入数据paddr_16b++;    //每次自增2B}
}/** 用指针读出数据* */
void SRAM_ReadByPointer() {uint16_t num[5] = {0};uint16_t *paddr_16b = (uint16_t*)(SRAM_ADDR_BEGIN); //uint16_t类型的指针for (int i = 0; i < 5; ++i) {num[i] = *paddr_16b;paddr_16b++;}
}

通过调试可以发现,用指针来读写数据也无误:

PS.注意,在使用HAL函数读写外部SRAM数据时,传递的目的地址必须是uint32_t类型的指针;而在使用指针直接访问SRAM时,指针的类型需要与实际访问的数据类型一致,比如说要读一个16位的数据,就要指定读取地址为一个uint16_t的指针(因为指针只是一个数,指针的类型就是表示该指针所指向的地址中的数据的类型)


(四)DMA方式读写

前面说了,要用FSMC的DMA需要去System Core的DMA里面手动创建,并且在代码里要手动LINK DMA;为了使用DMA,重新打开项目中的cubemx,如下:

然后进入DMA设置页面,在MenToMem栏新建一个DMA流,发现DMA2中出现了一个同样的DMA流,这是因为只有DMA2控制器支持mem到mem的传输,DMA1不支持;设置属性如下:

  • DMA的工作模式只能设置为Normal模式,没有Circular模式;
  • DMA流自动使用FIFO(DMA流队列),且不能关闭,Burst Size保持默认Single即可;
  • 源存储器和目标存储器的数据宽度设置为Word,这是因为HAL_SRAM_Write_DMA()和HAL_SRAM_Read_DMA()函数只支持uint32_t类型的数据buffer;
  • 源存储器和目标存储器都应开启地址自增;

配置如下图所示: 

此外,还要在NVIC中开启该DMA流的中断,否则系统不会调用中断回调函数;然后生成代码即可;

首先添加几个定义,主要是定义需要用得的宏、变量;如下:

#define     COUNT   5           //缓冲区数据个数
uint32_t    txbuf[COUNT];       //DMA发送缓冲区
uint32_t    rxbuf[COUNT];       //DMA接收缓冲区
bool        direction = true;   //DMA传输方向:ture表示MCU向外部SRAM传,false则相反
bool        is_busy = false;    //DMA状态:true表示正忙,false表示idle

还有,在主函数初始化FSMC后,需要加上LINK,将DMA流对象连接到SRAM对象:

接下来,封装一下DMA数据读、写函数;如下:

/** DMA发送函数* */
HAL_StatusTypeDef SRAM_WriteDMA() {HAL_StatusTypeDef status = HAL_OK;uint32_t val = 1000;//准备数据for (int i = 0; i < COUNT; ++i) {txbuf[i] = val;val += 100;}direction = true;dma_is_busy = true; //指示传输方向以及状态uint32_t *paddr = (uint32_t*)(SRAM_ADDR_BEGIN);if (HAL_OK == HAL_SRAM_Write_DMA(&hsram3, paddr, txbuf, COUNT)) {HAL_Delay(1);} else {status = HAL_ERROR;}return status;
}/** DMA读取函数* */
HAL_StatusTypeDef SRAM_ReadDMA() {HAL_StatusTypeDef status = HAL_OK;uint32_t *paddr = (uint32_t*)(SRAM_ADDR_BEGIN);direction = false;dma_is_busy = true; //指示传输方向以及状态if (HAL_OK == HAL_SRAM_Read_DMA(&hsram3, paddr, rxbuf, COUNT)) {HAL_Delay(1);} else {status = HAL_ERROR;}return status;
}/** DMA传输结束中断回调函数* */
volatile uint8_t test = 0;
/*
* 测试变量 test
* 当MCU向外部SRAM写入成功时,该变量赋值为1
* 当MCU从外部SRAM读取成功时,该变量赋值为2
* */
void HAL_SRAM_DMA_XferCpltCallback(DMA_HandleTypeDef *hdma)
{if (direction) {    //方向为test = 1;} else {test = 2;}dma_is_busy = false;    //表示dma传输结束
}

在主函数中调用这两个函数,打个断点,然后进入快乐的debug环节;一开始发现几个全局变量不在watch窗口中,首先加入窗口:

接着检查一下txbuf和rxbuf中的值,看是否正确:

然后在中断回调函数中打一个断点,看发送完成是否会进入回调;并注意发送数据前两个标志以及test变量的值:

接着走一步,发现发送成功了,进入了HAL_Delay()函数,然后再走一步,果然进入了回调函数;如下:

说明理论与实际情况一致,DMA发送成功。

接下来进入接收环节;一样的调试方法,最后发现依然进入回调函数,test被赋为2,此时查看rxbuf的值,可以看到与刚刚发送的5个数据一模一样,说明DMA接收也成功;如下:

大功告成!

工程链接:https://pan.baidu.com/s/18AJoG1epClGWzjHQkf6SRQ 
提取码:0xFF

完~


以上均为个人学习心得,如有错误,请不吝赐教~

THE END

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

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

相关文章

C++ YAML使用

C++工程如何使用YAML-cpp 一、前期准备工作 1、已安装minGW、cmake、make等本地工具。 2、下载YAML-cpp第三方开源代码(一定要下载最新的release版本,不然坑很多)。 3、生成YAML-cpp静态库 (1)在yaml-cpp-master下建立build文件夹; (2)在该文件夹下生成MakaFile文…

Ubuntu22.04 交叉编译gcc9.5 for arm

一、准备 环境&#xff1a;ubuntu22.04为刚刚安装&#xff0c;未安装gcc等包 vi ~/.bashrc输入 export PATH$PATH:/opt/gcc-arm-8.3-2019.03-x86_64-arm-linux-gnueabihf/bin 保存,reboot 安装&#xff1a; sudo apt install cmake sudo apt install gawk sudo apt instal…

C++ 程序员入门之路——旅程的起点与挑战

&#x1f337;&#x1f341; 博主猫头虎 带您 Go to New World.✨&#x1f341; &#x1f984; 博客首页——猫头虎的博客&#x1f390; &#x1f433;《面试题大全专栏》 文章图文并茂&#x1f995;生动形象&#x1f996;简单易学&#xff01;欢迎大家来踩踩~&#x1f33a; &a…

国庆假期day5

作业&#xff1a;请写出七层模型及每一层的功能&#xff0c;请绘制三次握手四次挥手的流程图 1.OSI七层模型&#xff1a; 应用层--------提供函 表示层--------表密缩 会话层--------会话 传输层--------进程的接收和发送 网络层--------寻主机 数据链路层----相邻节点的可靠传…

国庆10.4

QT实现TCP服务器客户端 服务器 头文件 #ifndef WIDGET_H #define WIDGET_H#include <QWidget> #include <QTcpServer> //服务器头文件 #include <QTcpSocket> //客户端头文件 #include <QList> //链表容器 #include <QMe…

mysql面试题14:讲一讲MySQL中什么是全同步复制?底层实现?

该文章专注于面试,面试只要回答关键点即可,不需要对框架有非常深入的回答,如果你想应付面试,是足够了,抓住关键点 面试官:讲一讲mysql中什么是全同步复制?底层实现? MySQL中的全同步复制(Synchronous Replication)是一种复制模式,主服务器在写操作完成后,必须等待…

Apacha Flume

0目录 1.Flume概述 2.Flume安装部署 3.案例1 4.案例2 5.案例3 1.Flume概述 1.1 Flume定义 Flume是Cloudera提供的一个高可用的&#xff0c;高可靠的&#xff0c;分布式的海量日志采集、聚合和传输的系统。Flume基于流式架构&#xff0c;灵活简单。 1.2 Flume基础架构 Flume组…

ES6中对象的扩展

1. 属性的简洁表示法 可以直接写入变量和函数作为对象的属性和方法。在对象中只写属性名&#xff0c;不写属性值&#xff0c;代表属性值等于和属性名相同的的变量的值。 属性的简写 let foo bar; let baz {foo}; // { foo: bar } // 等同于 let baz { foo: foo}方法的简写…

Nginx高级 第一部分:扩容

Nginx高级 第一部分&#xff1a;扩容 通过扩容提升整体吞吐量 1.单机垂直扩容&#xff1a;硬件资源增加 云服务资源增加 整机&#xff1a;IBM、浪潮、DELL、HP等 CPU/主板&#xff1a;更新到主流 网卡&#xff1a;10G/40G网卡 磁盘&#xff1a;SAS(SCSI) HDD&#xff08;机械…

十天学完基础数据结构-第二天(数据结构简介)

什么是数据结构&#xff1f; 在计算机科学中&#xff0c;数据结构是一种组织和存储数据的方式。它定义了数据的布局&#xff0c;以及对这些数据执行的操作。你可以把数据结构看作是计算机内存中的特定组织方式&#xff0c;就像图书馆中书籍的排列一样。 数据结构可以是各种形…

python获取时间戳

使用 datetime 库获取时间。 获取当前时间&#xff1a; import datetime print(datetime.datetime.now()) . 后面的是微秒&#xff0c;也是一个时间单位&#xff0c;1秒1000000微秒。 转为时间戳&#xff1a; import datetimedate datetime.datetime.now() timestamp date…

【数据结构】堆的应用-----TopK问题

目录 一、前言 二、Top-k问题 &#x1f4a6;解法一&#xff1a;暴力排序 &#x1f4a6;解法二&#xff1a;建立N个数的堆 &#x1f4a6;解法三&#xff1a;建立K个数的堆&#xff08;最优解&#xff09; 三、完整代码和视图 四、共勉 一、前言 在之前的文章中&#xff…

Springboot场景开发多面手

LinkedBear &#xff0c;资深 Java 高级工程师&#xff0c;底层技术研究者与分享者&#xff0c;倾心研究 Spring 技术体系多年&#xff0c;对 Spring、Spring Boot 、SpringCloud 等框架有独到的见解&#xff0c;拥有丰富的框架体系实践经验和架构封装经验。善于总结、输出&…

CharacterEncodingFilter的用法

CharacterEncoding是SpringMVC提供的一个一个过滤器,用于设置请求和响应的字符编码,解决乱码问题,他本身是一个过滤器 那么在SpringBoot中,CharacterEncoding就有一个很好的秒用 setEncoding("UTF-8")设置编码 setForceEncoding(true) 设置请求和响应编码 还需要在配…

leetcode做题笔记160. 相交链表

给你两个单链表的头节点 headA 和 headB &#xff0c;请你找出并返回两个单链表相交的起始节点。如果两个链表不存在相交节点&#xff0c;返回 null 。 图示两个链表在节点 c1 开始相交&#xff1a; 题目数据 保证 整个链式结构中不存在环。 注意&#xff0c;函数返回结果后&…

计算机视觉——飞桨深度学习实战-深度学习网络模型

深度学习网络模型的整体架构主要数据集、模型组网以及学习优化过程三部分&#xff0c;本章主要围绕着深度学习网络模型的算法架构、常见模型展开了详细介绍&#xff0c;从经典的深度学习网络模型以CNN、RNN为代表&#xff0c;到为了解决显存不足、实时性不够等问题的轻量化网络…

Day-05 CentOS7.5 安装docker

参考 &#xff1a; Install Docker Engine on CentOS | Docker DocsLearn how to install Docker Engine on CentOS. These instructions cover the different installation methods, how to uninstall, and next steps.https://docs.docker.com/engine/install/centos/ Doc…

计算机网络 第二章物理层

计算机网络第二章知识点速刷 其中重要的是信源和信宿&#xff0c;以及调制解调器在通信模型当中起到的作用。 单工、半双工和全双工。

【Java 进阶篇】JDBC 数据库连接池详解

数据库连接池是数据库连接的管理和复用工具&#xff0c;它可以有效地降低数据库连接和断开连接的开销&#xff0c;提高了数据库访问的性能和效率。在 Java 中&#xff0c;JDBC 数据库连接池是一个常见的实现方式&#xff0c;本文将详细介绍 JDBC 数据库连接池的使用和原理。 1…

计组——I/O方式

一、程序查询方式 CPU不断轮询检查I/O控制器中“状态寄存器”&#xff0c;检测到状态为“已完成”之后&#xff0c;再从数据寄存器取出输入数据。 过程&#xff1a; 1.CPU执行初始化程序&#xff0c;并预置传送参数&#xff1b;设置计数器、设置数据首地址。 2. 向I/O接口发…