玩转STM32-通信协议SPI(详细-慢工出细活)

news2024/11/16 22:49:27

文章目录

  • 一、SPI的基础知识
    • 1.1 接口定义
    • 1.2 单机和多机通信
  • 二、STM32的SPI工作过程
    • 2.1 从选择(NSS)脚管理
    • 2.2 时钟相位与极性
    • 2.3 SPI主模式
    • 2.4 SPI从模式
  • 三、应用实例

一、SPI的基础知识

1.1 接口定义

SPI系统可直接与各个厂家生产的多种标准外围器件接口,它只需4条线:串行时钟线(SCK)、主机输入/从机输出数据线(MISO)、主机输出/从机输入数据线(MOSI)和低电平有效的从机选择线(NSS)。
(1)MISO:该引脚在从模式下发送数据,在主模式下接收数据。
(2)MOSI:该引脚在主模式下发送数据,在从模式下接收数据。
(3)SCK:串口时钟,作为主设备的输出,从设备的输入。
(4)NSS:这个是一个可选的引脚,用来选择主/从设备。它的功能是用来作为片选引脚,让主设备可以单独地与特定从设备通信,避免数据线上的冲突。
SPI是一个环形总线结构,由NSS、SCK、MISO、MOSI构成,单主和单从设备连接图如下图所示,NSS引脚设置为输入,MOSI引脚相互连接,MISO引脚相互连接,数据在主和从之间串行地传输(MSB位在前)。通信总是由主设备发起,主设备通过MOSI脚把数据发送给从设备,从设备通过MISO引脚传回数据,属于全双工通信,数据输出和数据输入由同一个时钟信号同步,时钟信号由主设备通过SCK引脚提供。主机SPI时钟发生器在驱动移位寄存器移位的同时,产生时序由SCK引脚输出后控制从机移位寄存器。在SCK的控制下,主机移位寄存器的数据通过MOSI移位到从机移位寄存器中,而从机移位寄存器之前的数据通过MISO移位到主机移位寄存器中。
在这里插入图片描述

1.2 单机和多机通信

在多主机系统中,SPI还可以作为微处理器之间的通信。SPI子系统可以在软件控制下构成复杂或简单的系统,如一个主微控制器和几个从微控制器;几个微处理器互连,构成多主机系统,以及主微处理器和一个或多个从外围器件。

  1. 单主机通信
    多数应用场合用一个微处理器作为主机,它触发和控制向一个或多个外围器件传输数据或控制多个外围器件向主机传送数据,这些外围器件接收或提供传输的数据。
    这种主从SPI可用于微处理器与外围器件进行全双工、同步串行通信。SPI可以同时发出和接收串行数据。当SPI工作时,移位寄存器中的数据逐位从输出引脚输出(高位在前),同时从输入引脚接收的数据逐位移到移位寄存器(高位在前)。发出一个字节后,从另一个外围器件接收的字节数据进入移位寄存器。主SPI的是中国信号使传输过程同步。
    在这里插入图片描述
    许多简单的从外围器件只能接收主SPI的数据或只向主机发送数据。例如:串行-并行移位寄存器只能作为8位输出口。设置为主机的微处理器SPI控制向移位寄存器的发送过程。由于移位寄存器并不向SPI发出数据,因此SPI可以忽略接受的数据。
  2. 多主机通信
    SPI双主机多从机通信如下图所示。MOSI和MISO两个数据引脚用于接收和发送串行数据,高位MSB在前,地位LSB在后。当SPI设置为主机时,MISO是主机数据输入端,MOSI是主机数据输出端;当SPI设置为从机时,MISO是从机数据输出端,MOSI是从机数据输入端。
    在这里插入图片描述
    SCK是通过MISO和MOSI输入和输出数据的同步时钟。当SPI设置为主机时,SCK是主机时钟输出端;当SPI设置为从机时,SCK是从机时钟输入端。
    当SPI设置为主机时,SCK信号由内部微处理器总线时钟获得。当主机启动一次传输时,在SCK引脚自动产生8个时钟周期。对于主机或从机,都是从一个跳变沿进行采样,在另一个跳变沿移位输出或输入数据。
    NSS用于选择允许接收主机时钟和数据的从机。在数据传输之前NSS必须变为低电平,并在传输过程中保持为低电平。主机的NSS必须接到高电平。

二、STM32的SPI工作过程

2.1 从选择(NSS)脚管理

通过SPI_CR1寄存器的SSM位可以设置NSS的两种模式:软件NSS模式和硬件NSS模式。硬件/软件的从选择管理如下图所示。
在这里插入图片描述

  1. 软件NSS模式
    可以通过设置SPI_CR1寄存器的SSM位来使能这种模式。在这种模式下NSS引脚可以用作它用,而内部NSS信号电平可以通过写SPI_CR1的SSI位来驱动。
  2. 硬件NSS模式
    (1)NSS输出被使能:当STM32F10xxx工作为主SPI,并且NSS输出已经通过SPI_CR2寄存器的SSOE位使能,这时NSS引脚被拉低,所有NSS引脚与这个主SPI的NSS引脚相连并配置为硬件NSS的SPI设备,将自动变成从SPI设备。
    当一个SPI设备需要发送广播数据,它必须拉低NSS信号,以通知所有其它的设备它是主设备;如果它不能拉低NSS,这意味着总线上有另外一个主设备在通信,这时将产生一个硬件失败错误(Hard Fault)。
    (2)NSS输出被关闭:允许操作于多主环境。

2.2 时钟相位与极性

SPI_CR寄存器的CPOL和CPHA位,能够组合成四种可能的时序关系。CPOL(时钟极性)位控制在没有数据传输时时钟的空闲状态电平,此位对主模式和从模式下的设备都有效。如果CPOL被清’0’,SCK引脚在空闲状态保持低电平;如果CPOL被置’1’,SCK引脚在空闲状态保持高电平。 如果CPHA(时钟相位)位被置’1’,SCK时钟的第二个边沿(CPOL位为0时就是下降沿,CPOL位为’1’时就是上升沿)进行数据位的采样,数据在第二个时钟边沿被锁存。如果CPHA位被清’0’,SCK时钟的第一边沿(CPOL位为’0’时就是上升沿,CPOL位为’1’时就是下降沿)进行数据位采样,数据在第一个时钟边沿被锁存。CPOL时钟极性和CPHA时钟相位的组合选择数据捕捉的时钟边沿。
下图显示了SPI传输的4种CPHA和CPOL位组合。此图可以解释为主设备和从设备的SCK脚、MISO脚、MOSI脚直接连接的主或从时序图。
来自数据手册

2.3 SPI主模式

  1. 主模式配置过程
    在主配置时,在SCK脚产生串行时钟。 配置步骤
    1) 通过SPI_CR1寄存器的BR[2:0]位定义串行时钟波特率。
    2)选择CPOL和CPHA位,定义数据传输和串行时钟间的相位关系(见图212)。
    3) 设置DFF位来定义8位或16位数据帧格式。
    4)配置SPI_CR1寄存器的LSBFIRST位定义帧格式。
    5)如果需要NSS引脚工作在输入模式,硬件模式下,在整个数据帧传输期间应把NSS脚连接到高电平;在软件模式下,需设置SPI_CR1寄存器的SSM位和SSI位。如果NSS引脚工作在输出模式,则只需设置SSOE位。
    6)必须设置MSTR位和SPE位(只当NSS脚被连到高电平,这些位才能保持置位)。
    在这个配置中,MOSI引脚是数据输出,而MISO引脚是数据输入。
  2. 数据发送过程
    当写入数据至发送缓冲器时,发送过程开始。 在发送第一个数据位时,数据字被并行地(通过内部总线)传入移位寄存器,而后串行地移出到MOSI脚上;MSB在先还是LSB在先,取决于SPI_CR1寄存器中的LSBFIRST位的设置。数据从发送缓冲器传输到移位寄存器时TXE标志将被置位,如果设置了SPI_CR1寄存器中的TXEIE位,将产生中断。
  3. 数据接收过程
    对于接收器来说,当数据传输完成时:
    1)传送移位寄存器里的数据到接收缓冲器,并且RXNE标志被置位。
    2) 如果设置了SPI_CR2寄存器中的RXNEIE位,则产生中断。
    在最后采样时钟沿,RXNE位被设置,在移位寄存器中接收到的数据字被传送到接收缓冲器。读SPI_DR寄存器时,SPI设备返回接收缓冲器中的数据。 读SPI_DR寄存器将清除RXNE位。 一旦传输开始,如果下一个将发送的数据被放进了发送缓冲器,就可以维持一个连续的传输流。在试图写发送缓冲器之前,需确认TXE标志应该为’1’。

2.4 SPI从模式

在从模式下,SCK引脚用于接收从主设备来的串行时钟。SPI_CR1寄存器中BR[2:0]的设置不影响数据传输速率。

  1. 从模式配置步骤
    1)设置DFF位以定义数据帧格式为8位或16位。
    2) 选择CPOL和CPHA位来定义数据传输和串行时钟之间的相位关系(见图212)。为保证正确的数据传输,从设备和主设备的CPOL和CPHA位必须配置成相同的方式。
    3)帧格式(SPI_CR1寄存器中的LSBFIRST位定义的”MSB在前”还是”LSB在前”)必须与主设备相同。
    4) 硬件模式下(参考从选择(NSS)脚管理部分),在完整的数据帧(8位或16位)传输过程中,NSS引脚必须为低电平。在NSS软件模式下,设置SPI_CR1寄存器中的SSM位并清除SSI位。
    5)清除MSTR位、设置SPE位(SPI_CR1寄存器),使相应引脚工作于SPI模式下。 在这个配置中,MOSI引脚是数据输入,MISO引脚是数据输出。
  2. 数据发送过程
    在写操作中,数据字被并行地写入发送缓冲器。 当从设备收到时钟信号,并且在MOSI引脚上出现第一个数据位时,发送过程开始(译注:此时第一个位被发送出去)。余下的位(对于8位数据帧格式,还有7位;对于16位数据帧格式,还有15位)被装进移位寄存器。当发送缓冲器中的数据传输到移位寄存器时,SPI_SP寄存器的TXE标志被设置,如果设置了SPI_CR2寄存器的TXEIE位,将会产生中断。
  3. 数据接收过程
    对于接收器,当数据接收完成时:
    1)移位寄存器中的数据传送到接收缓冲器,SPI_SR 寄存器中的RXNE标志被设置。
    2)如果设置了SPI_CR2寄存器中的RXNEIE位,则产生中断。 在最后一个采样时钟边沿后,RXNE位被置’1’,移位寄存器中接收到的数据字节被传送到接收缓冲器。当读SPI_DR寄存器时,SPI设备返回这个接收缓冲器的数值。 读SPI_DR寄存器时,RXNE位被清除。

三、应用实例

  1. 实例介绍
    STM32通过SPI接口读取Flash W25X16的ID,并通过USART发送到上位机,通过串口调试助手显示ID信息。
  2. 实例程序
    // 串口初始化
void UsartDriver_Init(void)
{
	GPIO_InitTypeDef GPIO_InitStureture;
	USART_InitTypeDef USART_InitStructure;
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1,ENABLE);
	
	// PA9  TX
	GPIO_InitStureture.GPIO_Mode = GPIO_Mode_AF_PP;
	GPIO_InitStureture.GPIO_Pin = USART_GPIO_TX_PIN;
	GPIO_InitStureture.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(USART_GPIO_PORT,&GPIO_InitStureture);
	
	// PA10 rx
	GPIO_InitStureture.GPIO_Mode = GPIO_Mode_IN_FLOATING;
	GPIO_InitStureture.GPIO_Pin = USART_GPIO_RX_PIN;
	GPIO_Init(USART_GPIO_PORT,&GPIO_InitStureture);
	
	USART_InitStructure.USART_BaudRate = 9600;
	USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;
	USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;
	USART_InitStructure.USART_Parity = USART_Parity_No;
	USART_InitStructure.USART_StopBits = USART_StopBits_1;
	USART_InitStructure.USART_WordLength = USART_WordLength_8b;
	USART_Init(USART_PORT,&USART_InitStructure);
	USART_Cmd(USART_PORT,ENABLE);
}

/*串口1连续发送函数*/
void BdUsart1Trans(float *p, int16_t len)
{
	uint16_t i;
  for(i = 0;i < len; i++)
  {
   USART_SendData(USART1 , p[i]);
   while(USART_GetFlagStatus(USART1 , USART_FLAG_TC) == RESET) {}; // FLAG=0,未发完,等待
  }
}

/*串口1连续接收函数*/
void BdUsart1Recv(char *p, int16_t len)
{
	uint16_t i;
  for(i = 0;i < len; i++)
  {
    if(USART_GetFlagStatus(USART1 , USART_FLAG_RXNE) == SET)
		{
			p[i] = USART_ReceiveData(USART1);
		}; // FLAG=1, 收到数据
	}
}

// SPI初始化

// SPI
void SPIDriver_Init(void)
{
	GPIO_InitTypeDef GPIO_InitStureture;
	SPI_InitTypeDef SPI_InitStructure;
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA | RCC_APB2Periph_SPI1,ENABLE);
	
	GPIO_InitStureture.GPIO_Mode = GPIO_Mode_Out_PP;
	GPIO_InitStureture.GPIO_Pin = GPIO_Pin_5 | GPIO_Pin_6| GPIO_Pin_7;
	GPIO_InitStureture.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(GPIOA,&GPIO_InitStureture);
  // cs
	GPIO_InitStureture.GPIO_Mode = GPIO_Mode_Out_PP;
	GPIO_InitStureture.GPIO_Pin = GPIO_Pin_2;
	GPIO_Init(GPIOA,&GPIO_InitStureture);
	GPIO_SetBits(GPIOA,GPIO_Pin_5 | GPIO_Pin_6| GPIO_Pin_7);
	// 设置SPI单向或双向的数据模式
	SPI_InitStructure.SPI_Direction = SPI_Direction_2Lines_FullDuplex;
	// 设置主模式
	SPI_InitStructure.SPI_Mode = SPI_Mode_Master;
	// 8位帧结构
	SPI_InitStructure.SPI_DataSize = SPI_DataSize_8b;
	// 时钟悬空高
	SPI_InitStructure.SPI_CPOL = SPI_CPOL_High;
	// 数据捕获在第二个时钟沿
	SPI_InitStructure.SPI_CPHA = SPI_CPHA_2Edge;
	// 内部NSS信号由软件管理
	SPI_InitStructure.SPI_NSS = SPI_NSS_Soft;
	// 波特率预分频为256
	SPI_InitStructure.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_256;
	// 数据从MSB位开始传输
	SPI_InitStructure.SPI_FirstBit = SPI_FirstBit_MSB;
	// CRC值计算的多项式,大于7即可
	SPI_InitStructure.SPI_CRCPolynomial = 7;
	SPI_Init(SPI1,&SPI_InitStructure);
	SPI_Cmd(SPI1,ENABLE);
}

// 读取SPI
uint8_t SPI_ReadWriteByte(uint8_t TxData)
{
	uint8_t retry = 0;
	//检查指定的SPI标志位设置与否:发送缓存标志位
	while(SPI_I2S_GetFlagStatus(SPI1,SPI_I2S_FLAG_TXE) == RESET)
	{
		retry++;
		if(retry > 200)return 0;
	}
	//通过外设SPI发送一个数据
	SPI_I2S_SendData(SPI1,TxData);
	retry = 0;
	// 检查指定的SPI标志位设置与否
	while(SPI_I2S_GetFlagStatus(SPI1,SPI_I2S_FLAG_RXNE) == RESET)
	{
		retry++;
		if(retry > 200)return 0;
	}
	//接收数据
	return SPI_I2S_ReceiveData(SPI1);
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1800139.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

通俗易懂的解释保护性看跌期权和抛补看涨期权!

今天带你了解通俗易懂的解释保护性看跌期权和抛补看涨期权&#xff01;当涉及期权交易时&#xff0c;保护性看跌期权和抛补看涨期权是两种常见的策略&#xff0c;它们的目的都是为了在特定市场情况下对投资进行保护或增强收益。 保护性看跌期权 保护性看跌期权是一种风险管理策…

如何以非交互方式将参数传递给交互式脚本

文章目录 问题回答1. 使用 Here Document2. 使用 echo 管道传递3. 使用文件描述符4. 使用 expect 工具 参考 问题 我有一个 Bash 脚本&#xff0c;它使用 read 命令以交互方式读取命令参数&#xff0c;例如 yes/no 选项。是否有一种方法可以在非交互式脚本中调用这个脚本&…

粘性代理 vs 轮换代理: 特点、优势与选择指南

在网络领域&#xff0c;代理服务器是一种常见的工具&#xff0c;用于隐藏真实IP地址并提供更安全和匿名的网络体验。 粘性代理和轮换代理是两种常见的代理类型&#xff0c;它们在IP持久性和变更频率等方面有所不同。 本文将介绍粘性代理和轮换代理的区别&#xff0c;并分析在…

1.Linux入门

文章目录 一、介绍1.1 操作系统1.2 Linux1.3 虚拟机1.4 安装 CentOS7 二、远程连接 Linux2.1 FinalShell2.2 远程连接Linux 三、扩展3.1 WSL3.2 虚拟机快照 一、介绍 1.1 操作系统 我们平常所用的电脑是个人桌面操作系统&#xff0c;也就是Windows或者是macOS 目前我们要学的…

npm彻底清理缓存

在使用npm过程中&#xff0c;肯定会遇到清缓存的情况&#xff0c;网上的命令一般为 npm cache clear --force有时笔者在清理缓存之后npm install依然失败&#xff0c;仔细发现&#xff0c;执行该命令之后npm报了一个警告 npm WARN using --force Recommended protections dis…

使用AppJail配置网络并创建tiny jail(未成功)

创建tiny jail成功了&#xff0c;但是网络配置这块&#xff0c;jail里只能ping通外面&#xff0c;而无法pkg更新软件。本文章是这篇文章Jail管理器AppJail的使用FreeBSD-CSDN博客的网络篇。 首先host主机配置pf防火墙 参考这里&#xff1a;Packet Filter - AppJail Handbook …

[经验] 羊肺怎么清洗才干净视频 #经验分享#学习方法#其他

羊肺怎么清洗才干净视频 1、羊肺怎么清洗才干净 羊肺是一种营养丰富的食材&#xff0c;含有丰富的蛋白质和维生素&#xff0c;是众多美食菜谱的重要原料之一。但是&#xff0c;由于羊肺的内部结构复杂&#xff0c;清洗起来比较麻烦。那么&#xff0c;如何清洗羊肺才能让它干净…

基于Python的Selenium详细教程

一、PyCharm安装配置Selenium 本文使用环境&#xff1a;windows11、Python 3.10.5、PyCharm 2022.1.3、Selenium 4.3.0 需要你懂的技术&#xff1a;Python、HTML、CSS、JavaScript 1.Seleium安装&#xff1a; 在PyCharm终端或window命令窗口输入以下命令 #查看已安装的Pytho…

在 Qt Creator 上创建 ROS 项目并新建/导入 ROS 包

0、引言 ⚠️ 在开始之前&#xff0c;您需要确保您已经为 Qt 配置好了 ROS 开发环境了。如果您还没有配置好&#xff0c;可以参考这篇文章 本文将着手探讨如何在 Qt Creator 上编辑 ROS 项目&#xff08;工作空间&#xff09;。 1、本教程使用到的相关软件或产品 Ubuntu 20.0…

【Neo4j】Windows11使用Neo4j导入CSV数据可视化知识图谱

Windows11使用Neo4j导入CSV数据可视化知识图谱 序1. 安装JDK21&#xff08;1&#xff09;下载&#xff08;2&#xff09;安装&#xff08;3&#xff09;环境配置 2. 安装Neo4j&#xff08;1&#xff09;下载&#xff08;2&#xff09;解压安装&#xff08;3&#xff09;环境配置…

java 原生http服务器 测试JS前端ajax访问实现跨域传post数据

后端 java eclipse 字节流转字符 package Httpv3;import com.sun.net.httpserver.Headers; import com.sun.net.httpserver.HttpExchange; import com.sun.net.httpserver.HttpHandler; import com.sun.net.httpserver.HttpServer;import java.io.IOException; import java.i…

【Python报错】已解决Attributeerror: ‘list‘ object has no attribute ‘join‘( Solved)

解决Python报错&#xff1a;AttributeError: ‘list’ object has no attribute ‘join’ (Solved) 在Python中&#xff0c;字符串&#xff08;str&#xff09;对象有一个非常有用的join()方法&#xff0c;它允许你将序列中的元素连接&#xff08;join&#xff09;成一个字符串…

动态规划学习(混合背包,有依赖的背包,以及背包思想)

混合背包的定义&#xff1a; 混合背包问题就是混合01背包、完全背包和多重背包&#xff0c;可供选择的物体i可能有一个、或者无数个、或者有限个。 所以&#xff0c;就不要考虑这么多了&#xff0c;直接分这三种情况考虑就行&#xff01;&#xff01; 样例&#xff1a; for(…

优化财务管理制度提升企业经营效益—以审计代理记账为例

随着社会经济的快速发展&#xff0c;企业经营规模不断扩大&#xff0c;面临的财务管理问题也日益复杂&#xff0c;而作为其中的重要一环&#xff0c;审计代理记账已经成为了企业的必要组成部分&#xff0c;本文将重点探讨审计代理记账对于优化企业财务管理&#xff0c;提高经营…

【数据结构】图论入门

引入 数据的逻辑结构&#xff1a; 集合&#xff1a;数据元素间除“同属于一个集合”外&#xff0c;无其他关系线性结构&#xff1a;一个对多个&#xff0c;例如&#xff1a;线性表、栈、队列树形结构&#xff1a;一个对多个&#xff0c;例如&#xff1a;树图形结构&#xff1…

Linux 中常用的设置、工具和操作

1.设置固定的ip地址步骤 1.1 添加IPADDR“所设置的固定ip地址” TYPE"Ethernet" PROXY_METHOD"none" BROWSER_ONLY"no" BOOTPROTO"static" DEFROUTE"yes" IPV4_FAILURE_FATAL"no" IPV6INIT"yes" IPV6…

Vue——模板引用(不建议使用,了解)

文章目录 前言测试案例 前言 模板引用&#xff0c;在官方文档中也有很详细的描述。 虽然 Vue 的声明性渲染模型为你抽象了大部分对 DOM 的直接操作&#xff0c;但在某些情况下&#xff0c;我们仍然需要直接访问底层 DOM 元素。 个人理解为&#xff1a; 在vue中&#xff0c;依据…

gcc与g++的原理

程序的翻译 1 快速认识gcc/g2 程序翻译的过程2.1 预处理阶段2.2 编译阶段2.3 汇编阶段2.4 链接阶段 3 动静态库 1 快速认识gcc/g 首先我们需要在linux系统中安装对应的编译器gcc/g&#xff0c;安装命令如下&#xff08;centos7环境下&#xff09;&#xff1a; gcc: yum instal…

【HarmonyOS4学习笔记】《HarmonyOS4+NEXT星河版入门到企业级实战教程》课程学习笔记(十三)

课程地址&#xff1a; 黑马程序员HarmonyOS4NEXT星河版入门到企业级实战教程&#xff0c;一套精通鸿蒙应用开发 &#xff08;本篇笔记对应课程第 20 - 21节&#xff09; P20《19.ArkUI-属性动画和显式动画》 本节先来学习属性动画和显式动画&#xff1a; 在代码中定义动画&am…

Qt开发技术:Q3D图表开发笔记(四):Q3DSurface三维曲面图颜色样式详解、Demo以及代码详解

若该文为原创文章&#xff0c;转载请注明原文出处 本文章博客地址&#xff1a;https://hpzwl.blog.csdn.net/article/details/139424086 各位读者&#xff0c;知识无穷而人力有穷&#xff0c;要么改需求&#xff0c;要么找专业人士&#xff0c;要么自己研究 红胖子网络科技博…