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

news2025/1/8 19:53:56

目录

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.coloradmin.cn/o/1059123.html

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

相关文章

【C++11新算法】all_of、any_of、none_of算法

文章目录 前言一、概念1.1all_of1.2any_of1.3none_of 二、使用方式三、示例代码3.1all_of3.2any_of3.3none_of3.4检查一个字符串中的所有字符是否为小写字母3.5查一个容器中是否至少存在一个字符串长度超过5的元素 总结 前言 在C11标准中&#xff0c;引入了许多重要的新特性和…

Elasticsearch安装并使用Postman访问

Elasticsearch&#xff0c;一个强大的开源搜索和分析引擎&#xff0c;已经在全球范围内被广泛应用于各种场景&#xff0c;包括网站搜索、日志分析、实时应用等。由于其强大的功能和灵活性&#xff0c;Elasticsearch 已经成为大数据处理的重要工具。然而&#xff0c;对于许多初次…

C#,数值计算——Ranq2的计算方法与源程序

1 文本格式 using System; namespace Legalsoft.Truffer { /// <summary> /// Backup generator if Ranq1 has too short a period and Ran is too slow.The /// period is 8.5E37. Calling conventions same as Ran, above. /// </summary> …

算法题系列10·最长公共前缀

目录 题目描述 思路 实现 题目描述 编写一个函数来查找字符串数组中的最长公共前缀。如果不存在公共前缀&#xff0c;返回空字符串 ""。示例 1&#xff1a; 输入&#xff1a;strs ["flower","flow","flight"] 输出&#xff1a;&qu…

大数据Flink(九十六):DML:Deduplication

文章目录 DML:Deduplication DML:Deduplication Deduplication 定义(支持 Batch\Streaming):Deduplication 其实就是去重,也即上文介绍到的 TopN 中 row_number = 1 的场景,但是这里有一点不一样在于其排序字段一定是时间属性列,不能是其他非时间属性的普通列。在 ro…

python中实现定时任务的几种方案

目录 while True: sleep()Timeloop库threading.Timersched模块schedule模块APScheduler框架Celery框架数据流工具Apache Airflow概述Airflow 核心概念Airflow 的架构 总结以下几种方案实现定时任务&#xff0c;可根据不同需求去使用不同方案。 while True: sleep() 利用whil…

【2023年11月第四版教材】第17章《干系人管理》(合集篇)

第17章《干系人管理》&#xff08;合集篇&#xff09; 1 章节内容2 管理基础3 管理过程3.1 管理的过程★★★ &#xff08;22上44&#xff09;3.2 管理ITTO汇总★★★ 4 过程1-识别干系人4.1 数据收集★★★4.3数据分析4.4 权力利益方格4.5 数据表现&#xff1a;干系人映射分析…

【数据科学】Scikit-learn[Scikit-learn、加载数据、训练集与测试集数据、创建模型、模型拟合、拟合数据与模型、评估模型性能、模型调整]

这里写目录标题 一、Scikit-learn二、加载数据三、训练集与测试集数据四、创建模型4.1 有监督学习评估器4.1.1 线性回归4.1.2 支持向量机(SVM)4.1.3 朴素贝叶斯4.1.4 KNN 4.2 无监督学习评估器4.2.1 主成分分析(PCA)4.2.2 K Means 五、模型拟合5.1 有监督学习5.2 无监督学习 六…

全新UI彩虹外链网盘系统源码(前后端美化模板)

全新UI彩虹外链网盘系统源码前后端美化模板&#xff0c;支持所有格式文件的上传、生成文件外链、图片外链、音乐视频外链等功能&#xff0c;同时还可以自动生成相应的 UBB 代码和 HTML 代码&#xff0c;支持文本、图片、音乐、视频在线预览。这不仅仅是一个网盘&#xff0c;更是…

竞赛 机器视觉 opencv 深度学习 驾驶人脸疲劳检测系统 -python

文章目录 0 前言1 课题背景2 Dlib人脸识别2.1 简介2.2 Dlib优点2.3 相关代码2.4 人脸数据库2.5 人脸录入加识别效果 3 疲劳检测算法3.1 眼睛检测算法3.2 打哈欠检测算法3.3 点头检测算法 4 PyQt54.1 简介4.2相关界面代码 5 最后 0 前言 &#x1f525; 优质竞赛项目系列&#x…

算法题系列9·最后一个单词的长度

目录 题目描述 实现 题目描述 给你一个字符串 s&#xff0c;由若干单词组成&#xff0c;单词前后用一些空格字符隔开。返回字符串中最后一个单词的长度。 单词 是指仅由字母组成、不包含任何空格字符的最大子字符串。示例 1&#xff1a; 输入&#xff1a;s "Hello Worl…

WSL安装异常:WslRegisterDistribution failed with error: 0xc03a001a

简介&#xff1a;如果文件夹右上角是否都有两个相对的蓝色箭头&#xff0c;在进行安装wsl时&#xff0c;设置就会抛出 Installing WslRegisterDistribution failed with error: 0xc03a001a的异常 历史攻略&#xff1a; 卸载WSL WSL&#xff1a;运行Linux文件 WSL&#xff1…

关于MAC电脑无法正常登陆H3C iNodes登陆的解决办法

背景 前段时间&#xff0c;单位的网络在做升级改造&#xff0c;网络出口也进行彻底调整同时单位的网络出口设备做了机房物理迁移&#xff0c;迁移后网络正常使用&#xff0c;但是出现自己的MAC电脑无法登陆iNodes问题&#xff0c;总是出现“正在查询SSL 网关参数..查询SSL 网关…

【记录】IDA|IDA怎么查看当前二进制文件自动分析出来的内存分布情况(内存范围和读写性)

IDA版本&#xff1a;7.6 背景&#xff1a;我之前一直是直接看Text View里面的地址的首尾地址来判断内存分布情况的&#xff0c;似乎是有点不准确&#xff0c;然后才想到IDA肯定自带查看内存分布情况的功能&#xff0c;而且很简单。 可以通过View-Toolbars-Segments&#xff0c…

STM32复习笔记(四):独立看门狗IWDG

目录 &#xff08;一&#xff09;简介 &#xff08;二&#xff09;CUBEMX工程配置 &#xff08;三&#xff09;相关函数 总结&#xff1a; &#xff08;一&#xff09;简介 独立看门狗本质是一种定时器&#xff0c;其作用是监视系统的运行&#xff0c;当系统发生错误&…

linux命令行配置音频设备

linux命令行配置音频设备 TLTR在linux命令行播放音乐cmus需要开始声音条件功能才能调节播放的音量&#xff0c;看这个链接&#xff0c;继续折腾&#xff0c;have fun! TLTR 以archLinux为例&#xff0c;把下面软件都装一遍。 sudo pacman -S alsa-utils sudo pacman -S alsa-…

怎么才能实现一个链接自动识别安卓.apk苹果.ipa手机和win电脑wac电脑

您想要实现的功能是通过检测用户代理&#xff08;User Agent&#xff09;来识别访问设备类型并根据设备类型展示相应的页面。您可以根据以下步骤进行实现&#xff1a; 选择后端语言和框架&#xff0c;例如&#xff1a;Node.js、Express。 创建一个新的Express项目。 编写一个…

【单片机】16-LCD1602和12864显示器

1.LCD显示器相关背景 1.LCD简介 &#xff08;1&#xff09;显示器&#xff0c;常见显示器&#xff1a;电视&#xff0c;电脑 &#xff08;2&#xff09;LCD&#xff08;Liquid Crystal Display&#xff09;&#xff0c;液晶显示器&#xff0c;原理介绍 &#xff08;3&#xff…

Django基础讲解-路由控制器和视图(Django-02)

一 路由控制器 参考链接&#xff1a; Django源码阅读&#xff1a;路由&#xff08;二&#xff09; - 知乎 Route路由, 是一种映射关系&#xff01;路由是把客户端请求的 url路径与视图进行绑定 映射的一种关系。 这个/timer通过路由控制器最终匹配到myapp.views中的视图函数 …

[spring] spring core - 配置注入及其他内容补充

[spring] spring core - 配置注入及其他内容补充 上篇 [sping] spring core - 依赖注入 这里主要补一些 core 相关内容补充&#xff0c;同时添加了 java config bean 的方法 java config bean 是除了 XML、java 注解之外另一给实现 DI 的方法 java config bean 这个方法不…