基于STM32的SPI基本介绍

STM32---SPI(DMA)通信的总结(库函数操作)

本文主要由7项内容介绍SPI并会在最后附上测试源码供参考:

1.     SPI的通信协议

2.     SPI通信初始化(以STM32为从机,LPC1114为主机介绍)

3.     SPI的读写函数

4.     SPI的中断配置

5.     SPI的SMA操作

6.     测试源码

7.     易出现的问题及原因和解决方法

一、     SPI的通信协议

SPI(Serial Peripheral Interface)是一种串行同步通讯协议,由一个主设备和一个或多个从设备组成,主设备启动一个与从设备的同步通讯,从而完成数据的交换。SPI 接口一般由4根线组成,CS片选信号(有的单片机上也称为NSS),SCLK时钟信号线,MISO数据线(主机输入从机输出),MOSI数据线(主机输出从机输入),CS 决定了唯一的与主设备通信的从设备,如没有CS 信号,则只能存在一个从设备,主设备通过产生移位时钟信号来发起通讯。通讯时主机的数据由MISO输入,由MOSI 输出,输入的数据在时钟的上升或下降沿被采样,输出数据在紧接着的下降或上升沿被发出(具体由SPI的时钟相位和极性的设置而决定)。

 

 

 

二、     以STM32为例介绍SPI通信

1.     STM32f103 带有3个SPI模块其特性如下:

基于STM32的SPI基本介绍
 

2       SPI 初始化

初始化SPI 主要是对SPI要使用到的引脚以及SPI通信协议中时钟相位和极性进行设置,其实STM32的工程师已经帮我们做好了这些工作,调用库函数,根据自己的需要来修改其中的参量来完成自己的配置即可,主要的配置是如下几项:

l  引脚的配置

 SPI1的SCLK, MISO ,MOSI分别是PA5,PA6,PA7引脚,这几个引脚的模式都配置成GPIO_Mode_AF_PP 复用推挽输出(关于GPIO的8种工作模式如不清楚请自己百度,在此不解释),如果是单主单从,CS引脚可以不配置,都设置成软件模式即可。

基于STM32的SPI基本介绍
 

l  通信参数的设置

基于STM32的SPI基本介绍
 

 

1.     SPI_Direction_2Lines_FullDuplex把SPI设置成全双工通信;

2.     在SPI_Mode 里设置你的模式(主机或者从机),

3.     SPI_DataSize是来设置数据传输的帧格式的SPI_DataSize_8b是指8位数据帧格式,也可以设置为SPI_DataSize_16b,即16位帧格式

4.     SPI_CPOL和SPI_CPHA是两个很重要的参数,是设置SPI通信时钟的极性和相位的,一共有四种模式

基于STM32的SPI基本介绍
 

 

在库函数中 CPOL有两个值SPI_CPOL_High(=1)和SPI_CPOL_Low ( =0).

CPHA有两个值SPI_CPHA_1Edge (=0) 和SPI_CPHA_2Edge(=1)

CPOL表示时钟在空闲状态的极性是高电平还是低电平,而CPHA则表示数据是在什么时刻被采样的,手册中如下:

基于STM32的SPI基本介绍
 

 

我的程序中主、从机的这两位设置的相同都是设置成1,即空闲时时钟是高电平,数据在第二个时钟沿被采样,实验显示数据收发都正常。

要特别注意极性和相位的设置否则,数据传输会出现错位的现象

一般主从机的这两个位要设置的一样,但是网上也有人说不能设置成一样的,在后文中我对主从机极性和相位的配置的16种情况都做了测试,结果见下文。

下图很好的描述了4种模式下的时序状况

基于STM32的SPI基本介绍
 

 

 

引用网友的一句话:

 

 SPI主模块和与之通信的外设备时钟相位和极性应该一致。个人理解这句话有 2层意思:其一,主设备SPI时钟和极性的配置应该由外设的从设备来决定;其二,二者的配置应该保持一致,即主设备的SDO同从设备的SDO配置一致,主设备的 SDI同从设备的SDI配置一致。因为主从设备是在SCLK的控制下,同时发送和接收数据,并通过2个双向移位寄存器来交换数据 

5.     SPI_BaudRatePrescaler  波特率的设置

这在主机模式中,这一位的设置直接决定了通信的传输速率,而从机的设置不会影响数据传输的速率,手册中有这样一句话:

基于STM32的SPI基本介绍
 

 

(实际测试中发现:当STM32作为从机时,它对波特率的设置会影响数据的通信,特别是在大数据两传输时,当主机SPI时钟设置为15M时,STM32从机如果是2分频即18M则会在多次传输时出现错误,我记得在资料中看到过有前辈的经验贴说SPI从机的时钟设置不能高于SPI主机的时钟设置,虽然理论上从机的时钟设置不影响SPI通信,但是在试验中我也验证,当STM32从机时钟设为4分频 9M,低于15M时,通信就不会出现问题。所以SPI从机波特率的设置最好低于SPI主机波特率的设置。)

6.     SPI_FirstBit 这一位是设置首先传输的高字节还是低字节

SPI_FirstBit_MSB 是先传输高字节,SPI_FirstBit_LSB 是先传输低字节

注意在初始化函数里还有两项重要的内容就是在初始化之前先使能SPI的时钟和在初始化配置完成后使能SPI。

基于STM32的SPI基本介绍
 

 

(………..初始化配置……………)

 

 基于STM32的SPI基本介绍
 

三、     SPI的读写函数

SPI有一个16位的数据寄存器SPI_DR,它对应两个缓冲区,1个发送缓冲区,1个接收缓冲区,当在控制寄存器里SPI_CR1里对DFF位设置数据帧格式为8位时,发送和接收只用到SPI_DR[7:0]这8位,15-8位被强制为0,帧格式设置成16位时全用。

 

读写过程在手册中是这样描述的:

基于STM32的SPI基本介绍
 

 

简而言之,

发送时,可以通过检测SPI_SR中的TXE位,当数据寄存器里有数据时,TXE位是0,当数据全部从数据寄存器的发送缓冲区传输到移位寄存器时TXE位被置1,这时候可以再往数据寄存器里写入数据。可以通过

while (SPI_I2S_GetFlagStatus(SPI1, SPI_I2S_FLAG_TXE) == RESET) 来检测。

 SPI_I2S_GetFlagStatus(SPI1, SPI_I2S_FLAG_TXE)  是库函数,可以检测SPI的一些状态位。

接收时,可以通过检测SPI_SR中的RXNE位,当数据寄存器里有数据时,RXNE位是0,当数据全部从数据寄存器的接收缓冲区传输到移位寄存器时RXNE位被置1,这时候可以从数据寄存器里读出数据。可以通过

while (SPI_I2S_GetFlagStatus(SPI1, SPI_I2S_FLAG_RXNE) == RESET);   来检测。源程序如下,

SPI 读写一个字节,读写一体

基于STM32的SPI基本介绍
 

 

 

当能成功发送和接收一个字节时,发送数组数据就变的简单了,只需要一个for循环,和指针变量的递增即可。以下仅为参考:

有一点特别注意,从机数据传输时要依赖主机的时钟,所以主机在接收从机发送的数据时要往从机发送哑巴字节,这个字节可以自己定义 0xff,0xfe等什么字节都可以

 

读写分开的函数

 

void SPI_Ecah_Buffer_Send(u8* pBuffer, u16 NumByteToRead)

{

       for(int i = 0; i < NumByteToRead; i++)

       {

              SPI_Conmunication_SendByte(*pBuffer);

              pBuffer++;

       }

}

 

void SPI_Buffer_Receive(u8* pBuffer, u16 NumByteToRead)

{

       while (NumByteToRead--)

       {     

              *pBuffer = SPI_Conmunication_SendByte (Dummy_Byte);

              pBuffer++;

       }

}

读写一体的函数

 

void SPI_Ecah_Buffer_Send(u8* str , u8* pBuffer, u16 NumByteToRead)

{

       for(int i = 0; i < NumByteToRead; i++)

       {

              *str = SPI_Conmunication_SendByte(*pBuffer);

              pBuffer++;

              str++;

       }

}

 

 

 

四、     SPI的中断配置

在SPI的SPI_CR2 中可以配置,STM32的SPI的通信一共有8个中断其中最常用的是如下4个。

TXEIE:发送缓冲区空中断使能

在发送过程中,数据全部从数据寄存器的发送缓冲区传输到移位寄存器时TXE位被置1这时如果使能了TXEIE 就会触发发送完成的中断请求。在中断服务函数里可以做你想做的事情,也可以用一个标志位,在外面完成相应的操作。

(使用中断时要特别注意,及时的清除中断标志,为下一次能够触发中断做准备。而清除中断的操作可以放在中断服务函数中,或者其他你认为何时的地方。)

RXNEIE:接收缓冲区非空中断使能

接收同发送。

TXDMAEN:发送缓冲区DMA使能

RXDMAEN:接收缓冲区DMA使能

基于STM32的SPI基本介绍

 

手册中有这样一句话,“不能同时设置TXEIETXDMAEN”这一点要特别注意。也就是说如果你在SPI的通信中不用DMA则使能TXEIE的中断,禁能TXDMAEN的中断,如果在SPI中使用DMA传输,则禁能TXEIE 的中断,只使能TXDMAEN 的中断。

 

五、     SPIDMA操作

DMA(Direct Memory Access)直接内存存取,直接存储器存取用来提供在外设和存储器之间或者存储器和存储器之间的高速数据传输。无须CPU任何干预,通过DMA数据可以快速地移动。使用DMA最大的特点就是数据传输不经过CPU这就节省了CPU的资源,让CPU能有更多的时间来做其他的事情。

SPI的DMA操作,就是在SPI->TXE为1时,会向对应的DMA通道发出请求,DMA通道会发出应答信号,SPI收到应答信号后撤销请求信号,DMA撤销应答信号,并把内存值装入SPI_DR的发送缓区,SPI的传送开始。

DMA基于STM32的SPI基本介绍

的初始化

 

 

DMA_PeripheralBaseAddr是值外设数据的地址,用SPI1故DMA外设地址对应的是SPI1_DR_Addr,

DMA_MemoryBaseAddr是内存地址,它的值可以使,你要发送的数据所存放的数组的名,因为数组名代表的是数组数据存放的首地址,在SPI-DMA的发送中可以理解为把DMATX[]数组里的数据传送到SPI1_DR_Addr

DMA_DIR 是指数据传输的方向,其值发送时其值为DMA_DIR_PeripheralDST 即外设是目的地,方向是DMATX—> SPI1_DR_Addr,

在接受收时其值为DMA_DIR_PeripheralSRC,即外设是数据的来源,传输方向是 SPI1_DR_Addr—>用户指定的数据存储数组。

DMA_BufferSize 用来设置传输数据的个数,在STM32的DMA中其值的范围是0—65536.

DMA_Mode 指 DMA的传输模式 DMA_Mode_Normal为正常工作模式

DMA_Mode_Circular 是循环工作模式,这里对循环模式的解释我认为有位网友解释的很不错如下:

“循环的意思是指DMA的传输数量计数器会重置初值,由于DMA每传一个数据,传输数量计数器减一,只有在传输数量计数器的值不为零时,才会响应请求。在循环模式下,当传输计数器的值减为0后,会重新装载;而内存(缓存)地址则不管循环非循环模式,都会在每次传输完成后重置为基地址。所以,如果我们把DMA设置会正常模式,那么在下次传输前,只需对DMA的传输数量计数器重新写入就行。循环模式一般用于数据更新,比如ADC采用需要不停更新数据。”

在初始化完成之后要开启DMA的中断,在我的程序中开启的是DMA传输完成中断。

基于STM32的SPI基本介绍
 

 

DMA传输有3个中断标志位,常用的是传输完成的中断。如下:

基于STM32的SPI基本介绍
 

 

这样在传输完设定的数据个数之后就会触发传输完成的中断,用户可以再中断服务函数中,进行相应的操作,有一点特别注意,就是要及时清除中断标志位,为下次能够正常触发中断做准备

基于STM32的SPI基本介绍
 

 

在我的中断服务函数中有一个标志位SpiCommon,被置1后再中断之外进行其他的处理,同时调用DMA_ClearITPendingBit(DMA1_IT_TC2)来及时清除中断标志。

在进行DMA的数据传输时要先禁能DMA的通道,重置传输数据个数的值,数据的存储位置等,再使能DMA的通道,等待DMA的传输完成。

我的操作时这样的,先往DMATX[]里写入相应的数据,然后如下

基于STM32的SPI基本介绍

 

这样可能有一点不好的地方,因为只改变了SpiTXSize的值,却又重新执行了DMATXInit() 函数,可能此处能够再改善一下。

标题“51单片机通过MPU6050-DMP获取姿态角例程”解析 “51单片机通过MPU6050-DMP获取姿态角例程”是一个基于51系列单片机(一种常见的8位微控制器)的程序示例,用于读取MPU6050传感器的数据,并通过其内置的数字运动处理器(DMP)计算设备的姿态角(如倾斜角度、旋转角度等)。MPU6050是一款集成三轴加速度计和三轴陀螺仪的六自由度传感器,广泛应用于运动控制和姿态检测领域。该例程利用MPU6050的DMP功能,由DMP处理复杂的运动学算法,例如姿态融合,将加速度计和陀螺仪的数据进行整合,从而提供稳定且实时的姿态估计,减轻主控MCU的计算负担。最终,姿态角数据通过LCD1602显示屏以字符形式可视化展示,为用户提供直观的反馈。 从标签“51单片机 6050”可知,该项目主要涉及51单片机和MPU6050传感器这两个关键硬件组件。51单片机基于8051内核,因编程简单、成本低而被广泛应用;MPU6050作为惯性测量单元(IMU),可测量设备的线性和角速度。文件名“51-DMP-NET”可能表示这是一个与51单片机及DMP相关的网络资源或代码库,其中可能包含C语言等适合51单片机的编程语言的源代码、配置文件、用户手册、示例程序,以及可能的调试工具或IDE项目文件。 实现该项目需以下步骤:首先是硬件连接,将51单片机与MPU6050通过I2C接口正确连接,同时将LCD1602连接到51单片机的串行数据线和控制线上;接着是初始化设置,配置51单片机的I/O端口,初始化I2C通信协议,设置MPU6050的工作模式和数据输出速率;然后是DMP配置,启用MPU6050的DMP功能,加载预编译的DMP固件,并设置DMP输出数据的中断;之后是数据读取,通过中断服务程序从DMP接收姿态角数据,数据通常以四元数或欧拉角形式呈现;再接着是数据显示,将姿态角数据转换为可读的度数格
MathorCup高校数学建模挑战赛是一项旨在提升学生数学应用、创新和团队协作能力的年度竞赛。参赛团队需在规定时间内解决实际问题,运用数学建模方法进行分析并提出解决方案。2021年第十一届比赛的D题就是一个典型例子。 MATLAB是解决这类问题的常用工具。它是一款强大的数值计算和编程软件,广泛应用于数学建模、数据分析和科学计算。MATLAB拥有丰富的函数库,涵盖线性代数、统计分析、优化算法、信号处理等多种数学操作,方便参赛者构建模型和实现算法。 在提供的文件列表中,有几个关键文件: d题论文(1).docx:这可能是参赛队伍对D题的解答报告,详细记录了他们对问题的理解、建模过程、求解方法和结果分析。 D_1.m、ratio.m、importfile.m、Untitled.m、changf.m、pailiezuhe.m、huitu.m:这些是MATLAB源代码文件,每个文件可能对应一个特定的计算步骤或功能。例如: D_1.m 可能是主要的建模代码; ratio.m 可能用于计算某种比例或比率; importfile.m 可能用于导入数据; Untitled.m 可能是未命名的脚本,包含临时或测试代码; changf.m 可能涉及函数变换; pailiezuhe.m 可能与矩阵的排列组合相关; huitu.m 可能用于绘制回路图或流程图。 matlab111.mat:这是一个MATLAB数据文件,存储了变量或矩阵等数据,可能用于后续计算或分析。 D-date.mat:这个文件可能包含与D题相关的特定日期数据,或是模拟过程中用到的时间序列数据。 从这些文件可以推测,参赛队伍可能利用MATLAB完成了数据预处理、模型构建、数值模拟和结果可视化等一系列工作。然而,具体的建模细节和解决方案需要查看解压后的文件内容才能深入了解。 在数学建模过程中,团队需深入理解问题本质,选择合适的数学模
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值