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这个指针对结构体成员寄存器进行访问
单片机外围模块漫谈之二,如何提高ADC转换精度
在此我们简要总结一下ADC的各种指标如何理解,以及从硬件到软件都有哪些可以采用的手段来提高ADC的转换精度。
1.
ADC指标除了分辨率,速度,输入范围这些基本指标外,衡量一个ADC好坏通常会用到以下这些指标:失调误差,增益误差,微分非线性,积分非线性,信噪比,信纳比,有效位数,总谐波失真。让我们以下图为例来看一下这些指标的意义。
LSB
参数中经常用LSB作为单位,比如说差分非线性为2 LSB。这究竟是多大一个值呢?为了简单起见,我们以一个3bit分辨率,满量程为5V的ADC为例。1个LSB对应的电压大小为5V/7=714mV。如果是8bit分辨率,那么1个LSB对应5/255=19mV。
图中横轴为输入电压Vin增长方向,纵轴为数字输出。理想情况下输入电压每增长1LSB(714mV),那么输出会向上跳变一次,对应虚线L1所在转换曲线。但实际电路往往会引入偏差,转换曲线往往如 L2 所对应曲线。
失调误差(Offset Error)
电压从0开始增大时,引起输出第一次跳变的电压值,与理论上应该引起第一次跳变的电压值(0.5 LSB)的差值。衡量小电压时的转换精度。如图中,理论上应该在0.5 LSB处跳变,实际电压增大到1 LSB时才跳变,所以Offset Error是 1–0.5 = 0.5 LSB。
增益误差(Offset Error)
可以理解为实际转换曲线偏离理想曲线的程度。用最接近满量程时跳变点电压值和理论跳变点电压值的差表示。
差分非线性 DNL(Differential Non-Linearity)
理论上每增加或减少1 LSB 的电压,都会引起输出对应的一次跳变。但实际情况可能如图中a,b处所示,电压的步距大于或小于1个LSB的理论步距。
a 处 DNL = 1.5 – 1 = 0.5 LSB;
b 处 DNL = 0.5 – 1 = -0.5 LSB;
积分非线性 INL(Integral Non Linearity)
差分非线性累积起来造成的对实际转换曲线的最大偏离就是INL。如图中所示,需要注意的是INL不能表征对理想转换曲线的偏离程度。
总不可调整误差 TUE(Total Unadjusted Error)
实际转换曲线与理想转换曲线之间最大的偏离。在最糟糕的一点,我们通过ADC得到的电压,与实际电压的差值。通俗讲就是最不准的一点差多少。
思考一下,如果 DNL 和 INL 都非常好,那么是不是说明 TUE 就非常好?
对,还真不一定。即使线性度非常好,如果增益误差大,还是会导致最终结果大的偏差。
信噪比 SNR(Signal-to-Noise Ratio)
有用信号与噪声的能量比。我们总是期望信噪比越大越好。对于一个 N-Bit 分辨率的ADC来说,如果输入是一个满量程的正弦信号,在只考虑量化噪声的情况下,可以推导出一个有用的公式:
SNR = 6.02N + 1.76dB
推导过程见参考文档 ADI: MT-001
此公式直观的表明了ADC分辨率和信噪比之间的量化关系。
信纳比 SINAD (Signal-to-Noise-and-Distortion Ratio)
实际ADC是无法达到理想状态的,它的输出除了会引入噪声,还会引入输入信号的谐波。SINAD是有用信号能量,与谐波(Distortion)加噪声(Noise)能量的比,它更能体现现实世界中的ADC性能。
SINAD = 20log(S/(N+D))
*而SNR = 20log(S/N)
有效位数 ENOB(Effective Number of Bits)
体现ADC实际性能相当于多少位。可以从SINAD推出:
ENOB = (SINAD–1.76)/6.02
*和理想情况下的位数对应:NOB =(SNR - 1.76)/6.02。
总谐波失真 THD(Total Harmonic Distortion)
有用信号能量与谐波能量的比。
THD = 20log(S/D)
2. 如何提高转换精度
模拟电源(VDDA)和电压参考(VREF)
有的单片机ADC模块会引出单独的电源引脚和电压参考引脚,最好用LDO给这些引脚供电,或者用磁珠和滤波电容把这部分电源从数字部分隔离出来。
输入信号的输出阻抗
下图是ADC采样简化等效电路。ADC采样和保持电路的等效输入电阻电容Radc,和Cadc,手册中都会给出。在采样期间,开关SW会接通外部的信号输入电路,给采样电容Cadc充电至和输入信号相等(接近),之后SW断开,ADC对采样电容上的电压进行转换。如果采样时间过短,或者输入信号的输出阻抗过大,将导致采样电压不准。在信号源输出阻抗过高时,可以考虑增加一级运放。
高频串扰
如果与模拟输入引脚靠近的IO上有高频翻转的信号,或者PCB上有与输入信号长距离的平行走线,串扰将干扰输入信号。应避免ADC引脚临近信号高频翻转。在ADC输入布线和临近的走线之间用地线隔离开也可以避免ADC精度下降。
Wait,Stop模式
如果ADC在Wait和Stop模式下还可以工作,在此种模式下可以最大限度的降低MCU电源的波动,提高ADC的转换精度。
过采样(Oversampling)
如果采样频率为fs,那么对于fs/2以内的信号频率既可以获得完整信息。过采样是用远高于所需的采样频率去采样,这样噪声就会均摊在整个采样频带内。我们用数字滤波器可以滤除有用信号频带之外的噪声,从而使频带内的信噪比提高,获得更高的分辨率。对多次转换结果进行平均也可以提高结果的精度。
芯片内部校正
很多单片机内部都有校正机制(Calibration),每次上电后执行一次Calibration,可以以提高ADC的精度。
温度的影响
ADC受温度影响比较大,特别是失调误差和增益误差。如果芯片工作温度范围很宽,可以在不同的温度下预先测量,按温度做出查找表以在实际工作时做校正。
混入白噪声
这种方法对提高直流信号的分辨率很有用。如果输入信号接近直流而且很稳定,那么输出就不会产生跳变,比如输出一直是0x15A,我们无法确认输入电平是更接近0x159,还是更接近0x15B。那么我们可以人为地把白噪声混入信号,使信号产生小的波动从而输出产生跳变,然后再通过数学平均得出一个精度更高的数值。实际操作中可以用GPIO产生一个方波然后通过阻容耦合进输入信号引脚。
参考资料:
ADI: Data Converter Introduction
ADI: MT-001 TUTORIAL Taking the Mystery out of the Infamous Formula,
"SNR = 6.02N + 1.76dB," and Why You Should Care
ST: AN1636 UNDERSTANDING AND MINIMISING ADC CONVERSION ERRORS
ST: AN2834 Application note How to get the best ADC accuracy in STM32 microcontrollers
NXP: How to Increase the Analog-to-Digital Converter Accuracy in an Application
NXP: Cookbook for SAR ADC Measurements
相关问答
单片机 到底属不属于嵌入式系统? - 175****3103 的回答 - 懂得式系统是一个比较宽泛的概念,许多控制芯片,甚至微机的架构都能构建嵌入式系统;至于是用...个人是这么理解的:嵌入式系统是一个比较宽泛的概念,许多...
单片机 到底属不属于嵌入式系统? - 72XR1XAcm 的回答 - 懂得单片机只是嵌入式的核心部分,通过单片机和必要的外部元件组合起来,完成一定的功能,那才称为嵌入式,比如简单的MP3就是个嵌入式系统,里面是单片机、n...
单片机 anl是什么意思?单片机ANL是"AND逻辑运算指令"的缩写,是一种用于单片机的指令,用于进行按位AND逻辑运算。在计算机系统中,逻辑运算是对数据进行逻辑比较和操作的一种方法,AND...
1ORT1S2=1)ANDT1C3=0ANDT1C4=0ANDT1C1XW=0THEN定_作业帮[回答]一个IF必须对应一个ENDIF,你前面有3个IF,所以最后有三个ENDIF!这个程序好像是单片机的,我不太懂!
.pic 单片机 烧写问题Theovercurrentlogichasdetectedafaultco...[最佳回答]供电电流不够,烧写器的带载能力不强,你看看烧写器供电有没有选择上,然后是不是烧写器对电路板的其他外设进行了供电。我一般会在单片机的5V电源和外...
cpu和 单片机 的关系?CPU即中央处理器。中央处理器(CPU,CentralProcessingUnit)是一块超大规模的集成电路,是一台计算机的运算核心和控制核心。主要包括运算器(ALU,Arithmeti...
TⅤOC是什么?TⅤOC是什么?OC,outputcompare单片机相关输出比较的作用是用程序的方法在特定的时刻输出需要的电平,实现对外部电路的控制。详细资料可以见单片机里面的ETC模...
单片机 程序怎么在电脑上调试?用什么软件好?单片机可以实现在线调试程序,通过软件设置断点,进行单步调试,这样可以看到单片机是如何一步步执行程序的,从而快速找到问题,解决问题。以Keil和IAR为例介绍...单...
界面友好,功耗低.关键词: 单片机 数据采集温度控制[最佳回答]AbstractAtpresentthenumberofhouseholdelectronicbell,themajoritycanonlydisplayhours,minut...
STM32是什么啊,是32位的 单片机 吗 - 185****1270 的回答 - 懂得stm32是一种32位的单片机。单片机是嵌入式系统中最常用的核心部件,stm32本质上也是一种单片机。从事嵌入式方面工作,如果有一定的基础,可以从STM32...