STM32新手入门-什么是寄存器
单片机的本质其实就是在操作寄存器,让单片机完成我们想要的动作例如点亮一个LED灯,stm32的库函数开发也不例外它只不过是将操作寄存器封装成一个个函数,我们只要配置指定函数的参数,再调用该函数自动把对应的寄存器配置好,其实本质还是操作寄存器,更加方便快捷,如果你只学库函数的话后期就有种空中阁楼的感觉,知其然不知其所以然。
二.STM32的系统架构
STM32 芯片是已经封装好的成品,主要由内核和片上外设组成。若与电脑类比,内核与外设就如同电脑上的 CPU 与主板、内存、显卡、硬盘的关系。
下面这张stm32系统结构的图非常重要,我们要理解stm32如何运作,以及各个外设挂载在哪条总线上,而我们主要学习的是挂载在 AHB系统总线上的外设
1.四个驱动单元(CUP)
Cortex™-M3内核DCode总线
Cortex™-M3内核系统总线System
通用DMA1
通用DMA2
2.四个被动单元(外设)
内部SRAM
内部闪存存储器FLASH
FSMC
AHB到APB的桥,它连接所有的APB外设
3.驱动单元
ICode 总线
ICode 中的 I 表示 Instruction,即指令。内核通过ICode 总线读取内部FLASH代码指令来执行程序.。
DCode 总线
DCode 中的 D 表示 Data,即数据,那说明这条总线是用来取数的。因为数据可以被 Dcode 总线和 DMA 总线访问(向flash,SRAM,或外设数据寄存器里面取数据),所以为了避免访问冲突,在取数的时候需要经过一个总线矩阵来仲裁,决定哪个总线在取数,取到的数据可以暂存在Cortex™-M3内核里面的寄存器在进行处理。
系统总线System
系统总线主要是访问外设的寄存器,我们通常说的寄存器编程,即读写寄存器都是通过这根系统总线来完成的。
DMA 总线
DMA 总线与DCode总线一样主要是用来传输数据,但Dcode总线传输数据要占用内核(cpu)的资源,而DMA总线相当于独立于内核cpu但帮助内核cpu传输数据而不用占用内核(cpu)的资源,就是在DMA传输数据的同时内核cpu可以干别的事情比如点亮一个LED灯
总线矩阵
总线矩阵协调内核系统总线和DMA主控总线之间的访问仲裁,仲裁利用轮换算法。因为数据可以被 Dcode 总线和 DMA 总线访问,数据可以是在某个外设的数据寄存器,可以在SRAM,可以在内部的 FLASH。所以为了避免访问冲突,在取数的时候需要经过一个总线矩阵来仲裁,决定哪个总线在取数
4. 被动单元
内部FLASH
简单介绍在flash存储内容:我们写好的程序编译之后都是一条条指令(二进制代码),存放在 FLASH 中,我们常量或常变量C 语言中的 const 关键字修饰也存放在FLASH
内部SRAM
就是我们常说的电脑内存条,程序函数内部的局部变量和全局变量,堆(malloc分配)栈(局部变量)等的开销都是基于内部的SRAM。内核通过 DCode 总线来访问它
FSMC
FSMC 的英文全称是 Flexible static memory controller,叫灵活的静的存储器控制器,是 STM32F10xx 中一个很有特色的外设通过FSMC我们可以扩展内存,如外部的SRAM,NANDFLASH 和 NORFLASH。但有一点我们要注意的是,FSMC 只能扩展静态的内存,即名称里面的 S:static,不能是动态的内存,比如 SDRAM 就不能扩展。
AHB 到 APB 的桥
两个AHB/APB桥在AHB和2个APB总线间提供同步连接。APB1操作速度限于36MHz,APB2操作于全速(最高72MHz),上面挂载着 STM32 各种各样的特色外设。我们经常说的 GPIO、串口、I2C、SPI 这些外设就挂载在这两条总线上,这个是我们学习 STM32 的重点,就是要学会编程这些外设去驱动外部的各种设备。
三.存储器映射
存储器本身不具有地址信息,它的地址是由芯片厂商或用户分配,给存储器分配地址的过程就称为存储器映射,如果给存储器再分配一个地址就叫存储器重映射。
程序存储器、数据存储器、寄存器和输入输出端口被组织在同一个4GB的线性地址空间内。数据字节以小端格式存放在存储器中。一个字里的最低地址字节被认为是该字的最低有效字节,而最高地址字节是最高有效字节。
1. STM32的存储空间
上面说的4GB但为什么是4GB,存储空间的大小是由芯片内CPU内的地址总线的数量来决定,而stm32芯片内部的总线为32根
内存被划分为一个个的内存单元,每个内存单元的大小是一个字节,为了能有效的访问到内存的每个单元就给内存单元进行编号,编号就被称为该内存单元的地址
怎样产生地址
32根地址线每根线 可以输出正电和负电(1 或 0)
分配好地址后,被控单元的 FLASH,RAM,FSMC 和 AHB 到 APB 的桥(即片上外设),这些功能部件共同排列在一个 4GB 的地址空间内。我们在编程的时候,可以通过他们的地址找到他们,然后来操作他们( C语言里的将地址解引用操作 * 取出内容对他们进行数据的读和写)。
2. 存储器区域功能划分
在这 4GB 的地址空间中,ARM 已经粗线条的平均分成了 8 个块,每块 512MB,每个块也都规定了用途每个块的大小都有512MB,显然这是非常大的,
在这 8 个 Block 里面,有 3 个块非常重要,也是我们最关心的三个块。Block0 用来设计成内部 FLASH,Block1 用来设计成内部 RAM,Block2 用来设计成片上的外设,下面我们简单的介绍下这三个 Block 里面的具体区域的功能划分。
存储器 Block0 内部区域功能划分
储存器 Block1 内部区域功能划分
储存器 Block2 内部区域功能划分
Block2 用于设计片内的外设,根据外设的总线速度不同,Block 被分成了 APB 和 AHB两部分,其中 APB 又被分为 APB1 和 APB2
解释一下预留地址,这些地址没有给他分配存储单元,理论是4GB但实际并没有这么多,只是给了你指标而并没有全用
三.什么是寄存器
1. 寄存器映射
在存储器 Block2 这块区域,设计的是片上外设,它们以四个字节为一个单元,共32bit,每一个单元对应不同的功能,当我们控制这些单元时就可以驱动外设工作。我们可以找到每个单元的起始地址,然后通过 C 语言指针的操作方式 (既然一个单元是四个字节那我们就用一次取四个字节的指针(int * )来操作这些功能单元) 来访问这些单元,如果每次都是通过这种地址的方式来访问,不仅不好记忆还容易出错,这时我们可以根据每个单元功能的不同,以功能为名给这个内存单元取一个别名,这个给已经分配好地址的有特定功能的内存单元取别名的过程就叫寄存器映射。
2. 什么是寄存器
寄存器:给特定功能的的单元取的别名这个别名就叫做寄存器,所以寄存器只是特定功能的的单元的名字而已
比如,我们找到 GPIOB 端口的输出数据寄存器 ODR 的地址是 0x4001 0C0C(至于这个地址如何找到可以先跳过,后面我们会有详细的讲解),ODR 寄存器(就是4字节的功能单元)是 32bit,低 16bit有效,对应着 16 个外部 IO,写 0/1 对应的的 IO 则输出低/高电平。现在我们通过 C 语言指针的操作方式,让 GPIOB 的 16 个 IO 都输出高电平.
3. 如何给功能单元取个别名(寄存器)
我们利用C语言的#define 定义一个寄存器标识符
这样再操作一个功能单元就简单多了,而且更容易理解
接下来就是层层套娃找到GPIOB端口的所有功能单元地址,将他们分不同的功能改个别名变成寄存器,以后直接操作对应的寄存器就OK啦。
4. STM32的外设地址映射
片上外设区分为三条总线,根据外设速度的不同,不同总线挂载着不同的外设,APB1挂载低速外设,APB2和AHB挂载高速外设。相应总线的最低地址我们称为该总线的基地址,总线基地址也是挂载在该总线上的首个外设的地址。其中 APB1 总线的地址最低,片上外设从这里开始,也叫外设基地址。
列出了所用STM32F10xxx中内置外设的起始地址。
这里我截取一段教你们这么找总线基地址,外设基地址。
看上图就能一 一找到下面各个总线或各个外设的基地址
总线基地址
相对外设基地址偏移”即该总线地址与“片上外设”基地址 0x4000 0000的差值。
外设基地址
总线上挂载着各种外设,这些外设也有自己的地址范围,特定外设的首个地址称为“XX 外设基地址”。
这里以GPIO为例其他外设是一样的操作
这里相对APB2总线的地址偏移,外设基地址减去APB2总线基地址0X4001 0000 就是相对APB2总线的地址偏移,因为GPIO端口全是挂载在APB2总线。
外设寄存器
在 XX 外设的地址范围内,分布着的就是该外设的寄存器。以 GPIO 外设为例, - GPIO(general purpose input output)是通用输入输出端口的简称,简单来说就是 STM32 可控制的引脚 ,基本功能是控制引脚输出高电平或者低电平。最简单的应用就是把 GPIO 的引脚连接到 LED 灯的阴极,LED 灯的阳极接电源,然后通过 STM32 控制该引脚的电平,从而实现控制 LED 灯的亮灭。
这里我们以 GPIOB 端口为例,来说明 GPIO 都有哪些寄存器.
GPIO 有很多个寄存器,每一个都有特定的功能。每个寄存器32bit,占四个字节,在该外设的基地址上按照顺序排列,寄存器的位置都以相对该外设基地址的偏移地址来描述。
各个寄存器的地址=外设基地址+寄存器相对于外设基地址的偏移
这里我们按照地址偏移顺序看几个寄存器,理解寄存器的说明,具体见图
这里我就不一 一列举了,所有的外设基本都有自己的寄存器只不过功能不同名字不同都是按照此种模式来排列
四.C语言对寄存器的封装
上面方式还不够方便,接下来经过层层套娃,找到各个外设寄存器的地址再用C语言的结构体进行封装
1. 封装总线和外设基地址
在编程上为了方便理解和记忆,我们把总线基地址和外设基地址都以相应的宏定义起,总线或者外设都以他们的名字作为宏名
2. 封装寄存器列表
各个寄存器的地址=外设基地址+寄存器相对于外设基地址的偏移
到这里大家有没有发现一个特点寄存器的地址每次偏移4,如果定义一个寄存器的类型为(unsigned int)是不是正好在内存中占4个字节,而一个内存单元就是1个字节,分配一个地址,那4个字节不正好每次偏移4个地址,而且有符合结构体的内存对齐,关于结构体的内存对齐这里不细讲,以后会出有关结构体的文章详细阐述。
把寄存器封装成结构体后,接下来就是取出寄存器对寄存器进行操作以GPIOA为例,我们将GPIOA外设的基地址强制类型转化为该结构体的首地址不就完美解决了嘛
我们定义的这个 GPIO_TypeDef ,这个结构体的首地址就为 0x4001 0800(这也是第一个成员变量 CRL 的地址), 那么结构体中第二个成员变量 CRH 的地址即为 0x4001 0800 +0x04 ,加上的这个 0x04 ,正是代表 CRL 所占用的 4 个字节地址的偏移量,其它成员变量相对于结构体首地址的偏移
3. 操作寄存器
最后我们就可以直接使用宏定义好 GPIO_TypeDef 类型的指针,而且指针指向各个 GPIO 端口的首地址,使用时我们直接用GPIOA这个指针对结构体成员寄存器进行访问
单片机-XIP-外部闪存就地执行代码
1. 常说的 "单片机的norflash上可以执行代码 “ 这句话该如何理解?CPU做取值、译码、执行。
常说的哪些介质可以执行程序,实际上指的是CPU可以从那里取指,以供后续译码和执行。
2. 单片机一般在内部norfalsh上执行代码
Soc读取内部NorFlash时可以总线式访问,这对SoC来说更方便,由SoC直接以地址方式来访问,而且,NorFlash不需要初始化,时序简单。
PS:nandflash接口管理复杂,需要专门的控制器。
3. 单片机内部norflash上执行代码面临的困局
用ST做GUI界面,面临最大的问题就是芯片内部flash最大才2M。在现在这个时代,2M其实也放不了几张图片。
如果嫌弃你单片机的norflash不够大呢?你换再大的单片机norflash也大不了多少了,所以:
方案1. 使用ST在M7内核芯片上增加的QSPI控制器,让用户把代码放在外部存储就地执行。
方案2. 使用sdram,将代码搬运到sdram内执行。
方案1 对应的方式称为XIP 。
方案2 对应的方式称为 BootROM,顾名思义,从ROM搬运代码到RAM去执行,所以叫BootROM。
4. norflash不够的解决方案1 -- XIP
(1). 在外部闪存(外部QSPI或FMC-NOR闪存 )“就地执行”。
(2). 用户应用程序代码应链接到目标执行存储器地址(外部QSPI或FMC-NOR闪存)。
(3). 需要内存映射支持
上面 (1)意思是:XiP是可以在外部闪存直接执行代码的,就像芯片在内部flash 的地址0x0800 0000直接执行一样,称为“就地执行”。
上面(2) 意思是:用户应用程序代码编译的时候链接地址要改成外部闪存的地址,如STM32 H7系统给QSPI Flash在系统总线分配的地址是0x9000 0000,那么代码的地址就要改成0x9000 0000。
XIP模型操作流程
XiP启动方式实际是借助一个bootloader,这个bootloader做了一件很与众不同的事情,就是把QSPI FLASH映射到了系统总线0x9000 0000 这个地址上。
映射上去以后,只要我们访问0x9000 0000这个地址,系统总线就会自动去读QSPI FLASH 0地址的数据。此时是自动去读!
比如:uint32_t temp = *((uint32_t *)0x90000000)就能直接拿到QSPI FLASH第0~3地址数据。
当映射完成后,先关闭全部中断和cache,然后跳转到用户程序执行。
借助STM32H7的QSPI、XIP的方式,我们就可以实现让程序在W25QXX系列芯片内跑起来了。
5 . norflash不够的解决方案2 -- BootROM
(1). 从闪存启动,配置外部RAM存储器(SDRAM或SRAM),
先从闪存复制用户应用程序二进制文件(SDCARD或SPI-Flash存储器)到外部SDRAM或外部SRAM,然后跳转执行用户应用程序。
(2). 用户应用程序代码应链接到目标执行存储器地址(外部SDRAM或SRAM)。
这种模型就很通用了,适用于凡是有FMC能驱动外部RAM的任意一款ST芯片。当然链接地址肯定是外部RAM存储器的地址了。至于闪存,随意都可以,什么SPI FLASH、TF卡、I2C存储器都可以。
比如:STM32F407ZG + SRAM(32MB) + TF卡,把程序编译成二进制放到TF卡里面,每当我把新的程序更新到TF卡,运行的程序就更新了,就有点像玩Linux TF卡启动的感觉了。
相关问答
nandflash 最快读写速度?NAND闪存的最快读写速度取决于具体的技术和规格。目前,高性能NAND闪存的读取速度可以达到几百兆字节每秒(MB/s),而写入速度通常在几十到一百多兆字节每秒之间...
单片机flash 读取速度是多少?速度是100MHz的工作频率,因为C8051F120就达到了100MHz的工作频率,没有加延迟就以这么高的速度运行。但像STM32,在48MHz有1周期的延迟,72MHz有2周期延迟,所...
单片机 节日彩灯课程设计,从两边向中间点亮依次循环不止。亮灭时间为0.25S,用一个开关控制彩灯的工作?ORG0000HLJMPSTARTORG0100HSTART:MOVDPTR,#TABLE;取DPTR的地址,即表格的起始地址LOOP:CLRA;...
单片机 中24c02n芯片是做什么用的啊?这个问题我来回答一下。单片机中用到的at24c02n芯片是用来存储数据的,相当于电脑的硬盘。单片机的存储系统是由flash存储器和sram存储器构成的。flash存储器用...
什么叫 单片机 ?单片机(MCU)是由运算器(ALU)、控制器、存储器(ROM、RAM、EEPROM)、输入输出端口(I/O)等组成的一种集成芯片。目前常见的两种单片机架构-冯.诺依曼架构(51单片...
一次性烧录 单片机 叫什么?pic16C54c即是OTPROM,这款单片机有FLASHROM形式的。pic16C54c单片机简介:PIC16C54C为Microchip公司PIC16系列的产品,采用了0.7微米工艺技术制造...
单片机 的按键怎么外接-ZOL问答wanlingjiang一般单片机的外接按键有三种接法:1、每个I/O端口接一个按键:按键接另一端通常是接地,I/O口通常还应该接一个上拉电阻,有些单片机的I/O口可以配置...
32 单片机 有多少个 flash ?32的面积有16个fries。32的面积有16个fries。
xdata是什么?xdata指单片机里的Flash存取器,Flash相对单片机里的RAM属于外部存取器,虽其结构位置装在单片机中,所以搂主误解为xdata一定是放在单片机外面,其是xdata是放...
一般 单片机 内集成的 flash 是什么类型?FLASH闪存闪存的英文名称是"FlashMemory",一般简称为"Flash",它属于内存器件的一种,是一种不挥发性(Non-Volatile)内存...