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模块其特性如下:

2       SPI 初始化

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

l  引脚的配置

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

l  通信参数的设置

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通信时钟的极性和相位的,一共有四种模式

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

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

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

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

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

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

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

引用网友的一句话::

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

5.     SPI_BaudRatePrescaler  波特率的设置

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

(实际测试中发现:当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。

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

三、     SPI的读写函数

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

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

简而言之,

发送时,可以通过检测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 读写一个字节,读写一体

当能成功发送和接收一个字节时,发送数组数据就变的简单了,只需要一个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使能

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

五、     SPI的DMA操作

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

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

DMA

的初始化

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传输完成中断。

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

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

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

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

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

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

基于STM32的SPI基本介绍相关推荐

  1. 基于stm32、spi协议的Fatfs文件系统移植(附完整代码下载)

    开发环境:Window 7 32bit 开发工具:Keil uVision4 硬件:stm32f103vct6 目录 1.硬件设计: 2.软件设计 1.SPI收发数据 2.向SD卡发送的命令格式: 3 ...

  2. 基于STM32 + 超详细对新手全面解析讲解SPI协议(附源码)

    前言        本次我们学习一下STM32的一个基本外设 --- SPI,全程参考手册讲解,讲述SPI的工作模式和作用,让大家快速掌握和了解SPI通讯协议.本篇博客大部分是自己收集和整理,借鉴了很 ...

  3. STM32CubeMX | 基于STM32使用HAL库硬件SPI驱动WK2124一拖四SPI转四路串口芯片

    STM32CubeMX | 基于STM32使用HAL库硬件SPI驱动WK2124一拖四SPI转四路串口芯片 STM32基础工程生成 首先使用STM32CUBEMX生成STM32的基础工程,配置时钟到7 ...

  4. 基于STM32与NOR FLASH的SPI通信

      SPI的通信很容易实现,相比之下,驱动FLASH反而耗费了我学习SPI整个过程的大部分时间.下面是我学习过程的一些记录. 硬件平台:秉火ISO_V2开发板 实现功能:STM32使用SPI协议读写板 ...

  5. 基于STM32+0.96寸OLED - - 7脚SPI接线显示+代码解析

    前言  本次我们学习一下STM32关于SPI七脚OLED的 接线方法,OLED的代码解析,主要教会大家怎么使用OLED调试和看懂OLED驱动的代码,让大家对OLED有个简单的了解,本篇博客大部分是自己 ...

  6. SPI通讯协议详解 基于STM32

    SPI 协议简介 SPI 协议是由摩托罗拉公司提出的通讯协议 (Serial Peripheral Interface),即串行外围设备接口,是 一种高速全双工的通信总线.它被广泛地使用在 ADC.L ...

  7. stm32捕获占空比_基于STM32超声波避障小车

    不管是对于初学者还是对于一个玩过单片机的电子爱好者来说,或多或少都接触到过小车项目,今天给大家介绍的的一个项目基于STM32超声波避障小车.这也是我曾经的一个课设,在此开源分享给大家,全文5000多字 ...

  8. stm32串口接收不定长数据_基于STM32之UART串口通信协议--接收

    一.前言 1.简介 回顾上一篇UART发送当中,已经讲解了如何实现UART的发送操作了,接下来这一篇将会继续讲解如何实现UART的接收操作. 2.UART简介 嵌入式开发中,UART串口通信协议是我们 ...

  9. 2个相同的 stm32 can通讯不成功_一文读懂,基于 STM32 和 CAN 总线的温度监控系统的设计方法...

    1 系统总体方案概述 系统总体框图如图 1 所示,本系统采用主站+从站的结构,CAN 主站主要实现温度数据的存储以及 CAN 总线协议和串口协议之间的桥接,CAN 从站主要实现温度的采集.CAN 从站 ...

最新文章

  1. ​GEB:焦硕等发表生态位的系统发育保守性决定土壤古菌地理格局
  2. [Hadoop][笔记]4个节点搭建Hadoop2.x HA测试集群
  3. 【C++ 语言】面向对象 ( 成员函数 | 常量函数 | 友元函数 | 友元类 | 单例类 )
  4. main_loop()函数解析(1)
  5. 系统安装操作优化:chapter4 多系统的安装与管理
  6. 桌面电话的进化到统一通信
  7. asp.net mvc使用的心得和问题的解决方法(陆续更新ing)
  8. 零起步教你搭建Discuz!论坛
  9. 今日恐慌与贪婪指数为94 贪婪程度有所上升
  10. 封装条形码MaHelper
  11. 驱动兼容_「图」英特尔DCH驱动新版发布:重点修复Windows 10兼容性问题
  12. python-pygame小游戏之五子棋
  13. 【CSS】span标签设置宽度
  14. 手机无线如何共享给台式计算机,怎么把电脑网络共享给手机上网
  15. Bug、Defect(缺陷)、Error、Fault、Exception的区别
  16. shell脚本回车换行_终于搞懂了回车与换行的区别
  17. 360[极速模式]浏览器无法加载插件(mooc)的解决方案摸索
  18. 使用bins对数据进行分段并计算每个分段的数据量
  19. 人民大学 环境学院 雷洋(1987-2016)
  20. 英语老师唱歌软件测试,【出彩教育人】课上打电话,课下能K歌,这样的英语课给我来一打!...

热门文章

  1. 大数据学习笔记1000条
  2. Go语言入门——dep入门
  3. 数据中心行业在能源转型中将发挥重要作用
  4. IDC公司对未来5年的10项IT预测
  5. 给数据中心来个例行“体检”——动力系统
  6. mac全选文字的快捷键_在mac上写作必须知道的四组快捷键
  7. NLP:自然语言处理技术近十年发展技术更迭的简介、案例之详细攻略(持续更新)
  8. sklearn之pipeline:pipeline的简介、使用技巧、案例应用之详细攻略
  9. Paper:《Realtime Multi-Person 2D Pose Estimation using Part Affinity Fields ∗》翻译并解读
  10. Python语言学习之双下划线那些事:python和双下划线使用方法之详细攻略