零死角玩转stm32中级篇3-SPI总线

这篇具有很好参考价值的文章主要介绍了零死角玩转stm32中级篇3-SPI总线。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

一.基础知识

1.什么是SPI

SPI(Serial Peripheral Interface,串行外设接口)是一种同步的串行通信协议,它被用于在微控制器、存储器芯片、传感器和其他外围设备之间传输数据。SPI通常由四个线组成:时钟线(SCK)、主设备输出/从设备输入(MOSI)、从设备输出/主设备输入(MISO)和片选线(SS)。SPI通信中,数据在时钟的边沿上进行传输,以实现高速、可靠的数据传输。SPI可以支持单主机和多从机的连接方式,并且具有简单、灵活和可扩展的特点。

(主从连接方式1:一个主机上有多个SS片选信号去连接从机,如下图)
spi菊花链,stm32(标准库),stm32,单片机,嵌入式硬件,SPI通信

(主从连接方式2:一个主机上只有一个片选信号可以通过daisy-chained(菊花链)进行连接,如下图)

spi菊花链,stm32(标准库),stm32,单片机,嵌入式硬件,SPI通信

备注:图片来源于https://www.circuitbasics.com/basics-of-the-spi-communication-protocol/。

2.SPI和IIC有什么不同

SPI和I2C(Inter-Integrated Circuit,即IIC)都是常见的串行通信协议,它们在一些方面有所不同:

  1. 总线结构:SPI是点对点的结构,每个设备占用一个片选线;而I2C是多主从结构,允许多个设备通过两根共享的线路进行通信。

  2. 传输速率:SPI的传输速率通常比I2C更快,SPI可以达到几百MHz的传输速率,而I2C通常只能达到几十kHz或几百kHz的传输速率。

  3. 电气特性:SPI时钟线和数据线的电平是由驱动器控制的,因此SPI的电气特性更容易控制和优化,而且SPI在长距离传输时噪声抗干扰能力更强;而I2C的时钟和数据线由开漏输出控制,需要加上外部上拉电阻,电气特性控制相对较难。

  4. 硬件资源:SPI传输需要占用多个GPIO,因此需要更多的硬件资源来实现;而I2C只需要两个GPIO,可以减少芯片上的硬件资源占用。

总之,SPI和I2C都有其适用的场景。SPI适用于高速、简单的点对点通信,而I2C适用于多设备的通信,因为I2C允许多个设备在同一个总线上进行通信。

3.SPI的优缺点

优点:

  • 没有启动和停止位,所以数据可以不间断地连续流
  • 没有像 I2C 这样复杂的从站寻址系统
  • 数据传输速率比 I2C 高(几乎是 I2C 的两倍)
  • 单独的MISO 和 MOSI 线路,这样数据可以同时发送和接收

缺点:

  • 使用四根线(I2C 和 UART 使用两根)
  • 没有确认数据已经成功接收(I2C 有这个)
  • 没有像 UART 中奇偶校验位那样的错误检查形式
  • 只允许一个master
4.SPI是怎么实现通信的

① 在SPI通信中,数据是通过一个主设备与一个或多个从设备进行的。通信的过程是主设备向从设备发送数据,并且同时接收从设备发回的数据。SPI总线由四个信号线构成,分别是:

  1. SCLK(Serial Clock):时钟线,用于同步主从设备之间的数据传输。

  2. MOSI(Master Out Slave In):主设备输出数据到从设备的信号线。

  3. MISO(Master In Slave Out):从设备输出数据到主设备的信号线。

  4. SS(Slave Select):从设备的选中信号线,用于让主设备控制从设备的选择。

② 简单来说,当主设备需要跟某个从设备通信时,它会先把该从设备的SS线拉低(低电平有效还是高电平有效要根据元件的数据手册来看),表示选中该从设备,然后主设备以时钟信号为基准,通过MOSI线发送数据,从设备则通过MISO线将响应数据发回主设备。通信结束后,主设备会将该从设备的SS线拉高,表示不再选中该从设备。
③ SPI通信的速度可以通过调整时钟频率来实现,而具体的通信协议和数据格式则需要根据具体的应用场景来确定。

5.SPI 数据传输的步骤
  • 主机(master)输出时钟信号

spi菊花链,stm32(标准库),stm32,单片机,嵌入式硬件,SPI通信

  • 主机(master)将对应从机(方式一)的片选信号切换到低电平,从而激活对应从机

spi菊花链,stm32(标准库),stm32,单片机,嵌入式硬件,SPI通信

  • 主机(master)通过MOSI线向从设备一次一位地向从机发送数据,从设备读取接收到的数据

spi菊花链,stm32(标准库),stm32,单片机,嵌入式硬件,SPI通信

  • 如果从设备有相应的回应,从设备通过MISO线向主设备一次一位地向主机发送数据,主设备读取接收到的数据

spi菊花链,stm32(标准库),stm32,单片机,嵌入式硬件,SPI通信

6.SPI菊花链

在SPI菊花链方式中,各个从机的MISO(Master In Slave Out)输入都连接到前一个从机的MOSI(Master Out Slave In)输出上,一直到链的最后一个从机(如下图所示)。主机通过片选信号来选择与其通讯的从机,只有被选中的从机的MISO输出的数据才会被主机的MOSI输入。

spi菊花链,stm32(标准库),stm32,单片机,嵌入式硬件,SPI通信
备注:图片来源于https://zhuanlan.zhihu.com/p/290620901

具体的连接图如下:

spi菊花链,stm32(标准库),stm32,单片机,嵌入式硬件,SPI通信

SPI菊花链方式通信的基本步骤:

  • 主机发送片选信号(CS)来选择要与之通讯的从机。

  • 在所选从机的MISO输入处放置数据,同时主机在MOSI输出口发送相应的数据。

  • 当所选从机选定数据并将其内容从MISO输出时,主机也会将其内容从它的MISO输入口中读取,完整的数据交换完成。

  • 当主机需要与另一个从机通讯时,它会将片选信号切换到下一个从机上,然后重复上述步骤。

  • 当通讯完成时,主机可以停止发送片选信号。

需要注意的是,在SPI菊花链方式中,所有从机的MISO都连接在同一条线上。在未选中的情况下,从机将忽略主机发出的数据。因此,在设计SPI系统时需要确保未选中的从机在通讯期间处于高阻状态,以避免因为信号冲突而产生干扰。

  • 上文说所的未选中

① SPI菊花链中,所有从机都与同一条MISO线相连,但是在不同时间内只会有一个从机处于被选中状态,其他从机都处于未选中状态。这是通过从机的片选信号(CS)来实现的。当主机选择与某个从机通信时,它会向该从机的CS引脚发送低电平信号,从而告诉该从机它正在被选中。其他未被选中的从机,它们未选中时CS引脚通常为高电平状态,并且未选中的从机的MISO输出需要设置为高阻状态。
② 因此,在SPI菊花链方式中,需要确保未被选中的从机在通讯期间处于高阻状态,以避免干扰。最好的做法是在主机与某个从机通信之前,先将所有其他未选中的从机的片选信号拉高,保证它们的MISO输出都处于高阻状态。这样可以减少通信期间出现干扰的可能性。

  • 什么是高阻态

① 高阻状态是指一个电路中的输入端或输出端等待输入或输出信号时,它处于一种电气状态,该状态被称作高阻态或三态,简称"Z"态。处于高阻态的信号线会表现出一种很高的电阻,阻止其他电路对其进行电流或电压的驱动,这样可以保证电路的安全和稳定性。
② 在数字电路中,高阻态被广泛应用于多路复用器、锁存器和开关等电路中。例如,在多路复用器中,当选择器控制线不代表选中任何一个输入端口时,所有的输入端口都处于高阻状态,以避免输入端口之间的干扰。在锁存器中,当时钟信号处于非稳定状态时,输入端口处于高阻态,以避免输出端脱离预期状态。在开关中,当输出端口未被激活时,它处于高阻状态,以防止从该端口流出意外的电流引起不必要的能量损耗。

spi菊花链,stm32(标准库),stm32,单片机,嵌入式硬件,SPI通信

7.通过SPI实现数据的读和写

通过SPI协议进行数据的读写操作和其他通信协议(UART,IIC)一样也会有相应的数据格式,下面给出是93C46存储器的SPI数据读写的格式(具体的读写的数据格式通过数据手册进行查询)。

spi菊花链,stm32(标准库),stm32,单片机,嵌入式硬件,SPI通信

和异步不同的是,数据的发送会受到时钟线的控制,如下,就是当SS(片选线)为高电平时,进行数据的读写操作,从机采样数据由极性和相位决定,极性决定时钟SCK空闲时是高电平还是低电平,相位决定在第一个跳边沿还是第二个跳边沿进行数据采集,下图就是极性为低电平,相位为第一个跳边沿(上升沿:低->高)进行数据采集,采集MOSI上的数据,从机采集到了1 01 0000001 0000 1111的数据,然后进行解析,数据解析的结果表示将向000 0001地址写入数据0000 1111数据,从机解析完毕,执行解析的动作。

spi菊花链,stm32(标准库),stm32,单片机,嵌入式硬件,SPI通信

① SPI数据采样是由SPI总线信号的时钟极性和相位来决定的。通常情况下,SPI信号是由主设备(如微控制器)发出的,从设备(如传感器或存储器)则根据时钟信号进行响应。使用SPI时,必须确保主设备与从设备使用相同的时钟极性和相位,以确保数据采样的正确性。
② SPI采样方式有四种:mode0、mode1、mode2和mode3。下面分别介绍各种采样方式的时钟极性和相位:

  1. mode0:时钟极性为0,时钟相位为0。时钟极性为0表示空闲时时钟处于低电平,采样时时钟沿上升。时钟相位为0表示数据采样在时钟的上升沿进行,数据产生在下降沿。mode0是最常用的采样方式。

  2. mode1:时钟极性为0,时钟相位为1。时钟相位为1表示数据采样在时钟的下降沿进行,数据产生在上升沿。

  3. mode2:时钟极性为1,时钟相位为0。时钟极性为1表示空闲时时钟处于高电平,采样时时钟沿下降。时钟相位为0表示数据采样在时钟的下降沿进行,数据产生在上升沿。

  4. mode3:时钟极性为1,时钟相位为1。时钟相位为1表示数据采样在时钟的上升沿进行,数据产生在下降沿。

上面的解析可能不容易进行理解,大致意思就是SPI由极性和相位决定,极性和相位有二种取值即0和1,这样一共就由4种采集方式(00 01 10 11),其中当极性为0时表示时钟空闲状态为低电平(第一个跳变为上升沿),为1表示时钟空闲状态为高电平(第一个跳变为下降沿);而相位为0表示在第一个跳边沿进行数据采集,为1时表示在第二个跳边沿进行数据采集。下图中的图1的CPOL表示极性;图2CPHA表示相位;这二张图片来源于https://www.cnblogs.com/gmpy/p/12461461.html。

(图1)
spi菊花链,stm32(标准库),stm32,单片机,嵌入式硬件,SPI通信

(图2)
spi菊花链,stm32(标准库),stm32,单片机,嵌入式硬件,SPI通信

具体的四种采样方式如下图所示:

spi菊花链,stm32(标准库),stm32,单片机,嵌入式硬件,SPI通信

至于数据的读取,主机先通过MOSI向从机发送读取的操作指令(1 10 地址),从机收到该命令后,就会将数据放在MISO线上,主机就可以通过MISO,从从机的MISO上采集到对应的数据。

spi菊花链,stm32(标准库),stm32,单片机,嵌入式硬件,SPI通信
spi菊花链,stm32(标准库),stm32,单片机,嵌入式硬件,SPI通信

备注上图的图片和表格来源于:https://www.bilibili.com/video/BV1F54y1M7e7/

二.STM32F103C8T6芯片SPI协议案例代码

以下是一个基于STM32F103C8T6单片机的SPI协议案例代码,仅供参考(来源于ChartGPT):




#include "stm32f10x.h"

#define SPI_SCK_PIN  GPIO_Pin_5                //SPI时钟引脚
#define SPI_MISO_PIN GPIO_Pin_6                //SPI数据接收引脚
#define SPI_MOSI_PIN GPIO_Pin_7                //SPI数据发送引脚
#define SPI_CS_PIN   GPIO_Pin_4                //SPI片选引脚

void SPI_Init(void)
{
    GPIO_InitTypeDef GPIO_InitStructure;
    SPI_InitTypeDef  SPI_InitStructure;

    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA | RCC_APB2Periph_SPI1, ENABLE);  //使能SPI1外设时钟

    //配置SPI1对应的GPIO口
    GPIO_InitStructure.GPIO_Pin   = SPI_SCK_PIN | SPI_MOSI_PIN | SPI_CS_PIN;
    GPIO_InitStructure.GPIO_Mode  = GPIO_Mode_AF_PP;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_Init(GPIOA, &GPIO_InitStructure);

    GPIO_InitStructure.GPIO_Pin   = SPI_MISO_PIN;
    GPIO_InitStructure.GPIO_Mode  = GPIO_Mode_IPU;
    GPIO_Init(GPIOA, &GPIO_InitStructure);

    //配置SPI1参数
    SPI_InitStructure.SPI_Direction         = SPI_Direction_2Lines_FullDuplex;
    SPI_InitStructure.SPI_Mode              = SPI_Mode_Master;
    SPI_InitStructure.SPI_DataSize          = SPI_DataSize_8b;
    SPI_InitStructure.SPI_CPOL              = SPI_CPOL_Low;
    SPI_InitStructure.SPI_CPHA              = SPI_CPHA_1Edge;
    SPI_InitStructure.SPI_NSS               = SPI_NSS_Soft;
    SPI_InitStructure.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_8;
    SPI_InitStructure.SPI_FirstBit          = SPI_FirstBit_MSB;
    SPI_InitStructure.SPI_CRCPolynomial     = 7;
    SPI_Init(SPI1, &SPI_InitStructure);

    SPI_Cmd(SPI1, ENABLE);  //使能SPI1外设
}

void SPI_SendByte(SPI_TypeDef* SPIx, uint8_t byte)
{
    while (SPI_I2S_GetFlagStatus(SPIx, SPI_I2S_FLAG_TXE) == RESET) ;  //等待发送缓冲区为空
    SPI_I2S_SendData(SPIx, byte);                                     //将数据写入发送缓冲区
    while (SPI_I2S_GetFlagStatus(SPIx, SPI_I2S_FLAG_RXNE) == RESET) ; //等待接收缓冲区非空
    SPI_I2S_ReceiveData(SPIx);                                        //读取接收缓冲区数据,清除标志位
}

uint8_t SPI_ReceiveByte(SPI_TypeDef* SPIx)
{
    while (SPI_I2S_GetFlagStatus(SPIx, SPI_I2S_FLAG_TXE) == RESET) ;  //等待发送缓冲区为空
    SPI_I2S_SendData(SPIx, 0xFF);                                     //发送一个空数据,触发SPI通信
    while (SPI_I2S_GetFlagStatus(SPIx, SPI_I2S_FLAG_RXNE) == RESET) ; //等待接收缓冲区非空
    return SPI_I2S_ReceiveData(SPIx);                                 //读取接收缓冲区数据
}

int main(void)
{
    uint8_t tx_data = 0x55;
    uint8_t rx_data;

    SPI_Init();  //初始化SPI1

    GPIO_ResetBits(GPIOA, SPI_CS_PIN);  //拉低SPI片选引脚,开始SPI通信
    SPI_SendByte(SPI1, tx_data);        //发送数据
    rx_data = SPI_ReceiveByte(SPI1);    //接收数据
    GPIO_SetBits(GPIOA, SPI_CS_PIN);    //拉高SPI片选引脚,结束SPI通信

    while (1)
    {
        //此处可添加其他代码
    }
}

以上代码实现了STM32F103C8T6单片机的SPI通信,并利用SPI1对应的GPIO口进行了初始化配置。在主函数中,通过拉低片选引脚、发送数据、接收数据、拉高片选引脚的方式实现了SPI通信。用户可以根据实际需求,在SPI_SendByte和SPI_ReceiveByte函数中修改数据的长度和数据位数等参数。文章来源地址https://www.toymoban.com/news/detail-739421.html

到了这里,关于零死角玩转stm32中级篇3-SPI总线的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 2023版 STM32实战11 SPI总线读写W25Q

    英文全称:Serial peripheral Interface 串行外设接口 -1- 串行(逐bit传输) -2- 同步(共用时钟线) -3- 全双工(收发可同时进行) -4- 通信只能由主机发起(一主,多从机) -1- CS片选一般配置为软件控制 -2- 片选低电平有效,从器件CS引脚可直接连接GND -3- 从机不能主动给主机发数据 -4- 主机想要

    2024年02月08日
    浏览(29)
  • STM32存储左右互搏 SPI总线读写SD/MicroSD/TF卡

    SD/MicroSD/TF卡是基于FLASH的一种常见非易失存储单元,由接口协议电路和FLASH构成。市面上由不同尺寸和不同容量的卡,手机领域用的TF卡实际就是MicroSD卡,尺寸比SD卡小,而电路和协议操作则是一样。这里介绍STM32CUBEIDE开发平台HAL库SPI总线操作SD/MicroSD/TF卡的例程。 除了在硬件

    2024年04月12日
    浏览(34)
  • arm学习stm32之spi总线数码管倒计时,裸机开发,soc

    由于时间没有用时间计时器操作,有些误差,后续有空会翻新计时器版本 main.c spi.c spi.h

    2024年02月16日
    浏览(41)
  • STM32存储左右互搏 SPI总线读写FRAM MB85RS2M

    在中低容量存储领域,除了FLASH的使用,,还有铁电存储器FRAM的使用,相对于FLASH,FRAM写操作时不需要预擦除,所以执行写操作时可以达到更高的速度,其主要优点为没有FLASH持续写操作跨页地址需要变换的要求。相比于SRAM则具有非易失性, 因此价格方面会高一些。MB85RS2M是

    2024年01月22日
    浏览(37)
  • STM32存储左右互搏 SPI总线FATS文件读写SD/MicroSD/TF卡

    SD/MicroSD/TF卡是基于FLASH的一种常见非易失存储单元,由接口协议电路和FLASH构成。市面上由不同尺寸和不同容量的卡,手机领域用的TF卡实际就是MicroSD卡,尺寸比SD卡小,而电路和协议操作则是一样。这里介绍STM32CUBEIDE开发平台HAL库SPI总线FATS文件操作读写SD/MicroSD/TF卡的例程。

    2024年04月25日
    浏览(29)
  • STM32存储左右互搏 SPI总线FATS读写FRAM MB85RS2M

    在中低容量存储领域,除了FLASH的使用,,还有铁电存储器FRAM的使用,相对于FLASH,FRAM写操作时不需要预擦除,所以执行写操作时可以达到更高的速度,其主要优点为没有FLASH持续写操作跨页地址需要变换的要求。相比于SRAM则具有非易失性, 因此价格方面会高一些。MB85RS2M是

    2024年01月18日
    浏览(35)
  • ARM开发,stm32mp157a-A7核SPI总线实验(实现数码管的显示)

            a.数码管显示相同的值 0000 1111 ......9999;         b.数码管显示不同的值 1234; ---spi.h头文件--- ---spi.c函数文件--- ---main.c测试文件--- a.数码管显示相同的值 0000 1111 ......9999; b.数码管显示不同的值 1234;

    2024年02月11日
    浏览(33)
  • STM32F103C8T6+2.4寸SPI TFT触摸屏代码+标准库 项目开发

    目录 模块清单: 模块介绍: 1:STM32F103C8T6 2:2.4寸SPI TFT触摸屏 项目结果展示 2.4寸 TFT SPI显示触摸屏 2.4寸 SPI TFT 显示触摸屏代码下载链接: https://download.csdn.net/download/weixin_49492286/88458377 清单 STM32F103C8T6 2.4寸SPI TFT触摸屏         STM32F103C8T6是意法半导体(STMicroelectronics)推

    2024年02月07日
    浏览(52)
  • 零死角玩转stm32初级篇2-STM32如何编译和下载程序

    一.程序的编译 Keil uVision5 工具中有四个编译如图 ,他们分别表示什么意思,下面进行介绍,解释来源于零死角玩转stm32。 第一个按钮: Translate 就是翻译当下修改过的文件,说明白点就是检查下有没有语法错误,并不会去链接库文件,也不会生成可执行文件。 第二个按钮: B

    2024年02月03日
    浏览(35)
  • 【SPI】STM32 SPI 双机通信,SPI从机模式使用

    最近要用到STM32的SPI从机模式,从其他板子读SPI数据过来,踩了2天坑,记录一下过程。 (因为hal库提供了三种函数,我调试的过程是阻塞、中断、DMA依次来调试学习的, 这份代码为使用DMA方式的代码。 ) 软件 :keil5、STM32CubeMX 硬件 :两块STM32F103C8T6最小系统 实现功能 :两

    2024年02月17日
    浏览(46)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包