SPI 详解

介绍

SPI,是英语Serial Peripheral interface的缩写,顾名思义就是串行外围设备接口。SPI,是一种高速的,全双工,同步的通信总线。一主多从。

主-从模式(Master-Slave) 的控制方式

SPI 规定了两个 SPI 设备之间通信必须由主设备 (Master) 来控制次设备 (Slave).。SPI 协议规定 Slave 设备的 Clock 由 Master 设备通过 SCK 管脚提供给 Slave 设备, Slave 设备本身不能产生或控制 Clock, 没有 Clock 则 Slave 设备不能正常工作。在这里插入图片描述
命名从主设备角度看。(1)SDO/MOSI (Master Output Slave Input)– 主设备数据输出,即从设备数据输入;(2)SDI/MISO – 数据输入;(3)SCLK – 时钟信号,由主设备产生,没有数据就可以没有时钟;(4)CS/SS – 从设备片选,每个从设备都需要一个,由主设备控制。

SPI有四种传输模式

Master 设备会根据将要交换的数据来产生相应的时钟脉冲(Clock Pulse), 时钟脉冲组成了时钟信号(Clock Signal) , 时钟信号通过时钟极性 (CPOL) 和 时钟相位 (CPHA) 控制着两个 SPI 设备间何时数据交换以及何时对接收到的数据进行采样。
SPI总线的极性,会直接影响SPI总线空闲时的时钟信号是高电平还是低电平。
SPI总线的相位,一个时钟周期会有2个跳变沿。而相位,直接决定SPI总线从那个跳变沿开始采样数据。
图中第一行:总线空闲为高,第二个跳变采集数据。
在这里插入图片描述
从设备模式可能是出场就制定好的。

SPI控制器

在这里插入图片描述
SSPSR 是 SPI 设备内部的移位寄存器(Shift Register). 它的主要作用是根据 SPI 时钟信号状态, 往 SSPBUF 里移入或者移出数据, 每次移动的数据大小由 Bus-Width 以及 Channel-Width 所决定。
Bus-Width 的作用是指定地址总线到 Master 设备之间数据传输的单位.
例如, 我们想要往 Master 设备里面的 SSPBUF 写入 16 Byte 大小的数据: 首先, 给 Master 设备的配置寄存器设置 Bus-Width 为 Byte; 然后往 Master 设备的 Tx-Data 移位寄存器在地址总线的入口写入数据, 每次写入 1 Byte 大小的数据(使用 writeb 函数); 写完 1 Byte 数据之后, Master 设备里面的 Tx-Data 移位寄存器会自动把从地址总线传来的1 Byte 数据移入 SSPBUF 里; 上述动作一共需要重复执行 16次。
Channel-Width 的作用是指定 Master 设备与 Slave 设备之间数据传输的单位. 与 Bus-Width 相似, Master 设备内部的移位寄存器会依据 Channel-Width 自动地把数据从 Master-SSPBUF 里通过 Master-SDO 管脚搬运到 Slave 设备里的 Slave-SDI 引脚, Slave-SSPSR 再把每次接收的数据移入 Slave-SSPBUF里.
通常情况下, Bus-Width 总是会大于或等于 Channel-Width, 这样能保证不会出现因Master 与 Slave 之间数据交换的频率比地址总线与 Master 之间的数据交换频率要快, 导致 SSPBUF 里面存放的数据为无效数据这样的情况。
在这里插入图片描述
我们知道, 在每个时钟周期内, Master 与 Slave 之间交换的数据其实都是 SPI 内部移位寄存器从 SSPBUF 里面拷贝的. 我们可以通过往 SSPBUF 对应的寄存器 (Tx-Data / Rx-Data register) 里读写数据, 间接地操控 SPI 设备内部的 SSPBUF.
例如, 在发送数据之前, 我们应该先往 Master 的 Tx-Data 寄存器写入将要发送出去的数据, 这些数据会被 Master-SSPSR 移位寄存器根据 Bus-Width 自动移入 Master-SSPBUF 里, 然后这些数据又会被 Master-SSPSR 根据 Channel-Width 从 Master-SSPBUF 中移出, 通过 Master-SDO 管脚传给 Slave-SDI 管脚, Slave-SSPSR 则把从 Slave-SDI 接收到的数据移入 Slave-SSPBUF 里.
与此同时, Slave-SSPBUF 里面的数据根据每次接收数据的大小(Channel-Width), 通过 Slave-SDO 发往 Master-SDI, Master-SSPSR 再把从 Master-SDI 接收的数据移入 Master-SSPBUF.在单次数据传输完成之后, 用户程序可以通过从 Master 设备的 Rx-Data 寄存器读取 Master 设备数据交换得到的数据.
在这里插入图片描述
Master 设备里面的 Controller 主要通过时钟信号(Clock Signal)以及片选信号(Slave Select Signal)来控制 Slave 设备. Slave 设备会一直等待, 直到接收到 Master 设备发过来的片选信号, 然后根据时钟信号来工作.
Master 设备的片选操作必须由程序所实现. 例如: 由程序把 SS/CS 管脚的时钟信号拉低电平, 完成 SPI 设备数据通信的前期工作; 当程序想让 SPI 设备结束数据通信时, 再把 SS/CS 管脚上的时钟信号拉高电平.

速度

SPI是一种事实标准,由Motorola开发,并没有一个官方标准。已知的有的器件SPI已达到50Mbps。具体到产品中SPI的速率主要看主从器件SPI控制器的性能限制。
STM32f103 中APB1的最高频率是36MHz,APB2的最高频率是72MHz,而PCLK1和PCLK2一般也默认配置为其最高工作频率36M和72M。
SPI1时钟由APB2时钟分频而来,可以选择2、4、8、16、32、64、128、256这几个分频系数。而手册规定STM32的SPI时钟最快是18MHz。对于STM32F103的SPI1接口时钟,由72M的PCLK2分频得到,所以分配系数大于等于4(72M/4 = 18M)。对于STM32F103的SPI2/SPI3接口时钟,由36M的PCLK1分频得到,所以分配系数大于等于2(36M/2 = 18M)。
若在配置SPI1时,选择了APB2的二分频选项,依据理论SPI1的时钟速率将为36MHz(有网友测试过,确实可以达到36hz)。这个频率大于手册中要求的最快的18Mhz。如果为了追求高速率使用36MHz,建议一定要加强测试环节,同时不建议在工控产品等高可靠性的场合使用。

<think>首先,用户要求使用中文回答问题。主题是“spi详解”,所以我需要用中文详细解释SPI(Serial Peripheral Interface)。 SPI是一种串行通信协议,常用于微控制器和外设之间的通信,如传感器、存储器等。我需要涵盖其基本概念、工作原理、优缺点等。 关键点: - **全称**:Serial Peripheral Interface(串行外设接口) - **特点**:全双工、同步通信、主从模式 - **信号线**: - SCLK:串行时钟(由主机提供) - MOSI:主机输出,从机输入 - MISO:主机输入,从机输出 - SS/CS:片选信号(选择从机) - **工作方式**:主机控制时钟,数据在时钟边沿传输。 - **模式**:有四种模式,取决于时钟极性和相位(CPOL和CPHA)。 - **优势**:高速、简单、无起始/停止位。 - **劣势**:需要更多引脚、无错误检测。 回答结构: 1. 介绍SPI是什么。 2. 解释核心组件和信号。 3. 描述数据传输过程。 4. 讨论模式(CPOL和CPHA)。 5. 优缺点。 6. 应用场景。 在回答末尾,我必须添加“§§相关问题§§”,然后列出3-5个相关问题。这些问题要基于用户的问题和我的回答。 相关问题示例: 1. SPI与其他协议(如I2C、UART)的区别是什么? 2. 如何在实际项目中使用SPI? 3. SPI的四种模式如何影响通信? 4. SPI嵌入式系统中的应用案例? 5. 如何解决SPI通信中的常见问题? 确保输出格式: - 回答内容用中文。 - 结束时输出“§§相关问题§§”,然后直接列出问题(1. ... 2. ... 3. ...)。 - 不要换行分散;问题列表要紧凑。 完整输出: - 先写回答。 - 然后“§§相关问题§§” - 接着问题列表。 现在,构建回答。</think>### SPI详解 SPI(Serial Peripheral Interface,串行外设接口)是一种同步串行通信协议,广泛用于嵌入式系统和微控制器中,以实现主设备(如微处理器)与从设备(如传感器、存储器芯片)之间的高速数据交换。它由Motorola公司开发,具有全双工通信能力,支持主从模式。以下是SPI的核心内容: #### 1. **基本组成** - **信号线**: - **SCLK**(Serial Clock):时钟信号,由主机提供,控制数据传输速率。 - **MOSI**(Master Out Slave In):主机输出、从机输入数据线。 - **MISO**(Master In Slave Out):主机输入、从机输出数据线。 - **SS/CS**(Slave Select/Chip Select):片选信号,由主机控制,用于选择特定从机(允许多个从机共享同一总线)。 - 典型连接:一个主机可连接多个从机,但每个从机需要独立的SS线。 #### 2. **工作原理** - **同步通信**:数据传输基于SCLK时钟信号,主机在时钟边沿(上升沿或下降沿)触发数据采样。 - **全双工传输**:数据同时在MOSI和MISO线上传输,主机发送数据给从机的同时接收从机数据。 - **传输过程**: 1. 主机拉低目标从机的SS线,激活通信。 2. 主机生成SCLK信号,并在每个时钟周期通过MOSI发送一位数据。 3. 从机在同步的时钟边沿通过MISO返回数据。 4. 传输完成后,主机拉高SS线释放从机。 - **数据帧**:通常以8位或16位为单位传输,无起始/停止位,效率高。 #### 3. **模式配置** SPI有四种工作模式,由时钟极性(CPOL)和时钟相位(CPHA)决定: - **CPOL**:时钟空闲状态(0为低电平,1为高电平)。 - **CPHA**:数据采样时机(0为第一个时钟边沿采样,1为第二个时钟边沿采样)。 - **模式表**: | 模式 | CPOL | CPHA | 描述 | |------|------|------|------| | 0 | 0 | 0 | 时钟空闲低电平,数据在上升沿采样 | | 1 | 0 | 1 | 时钟空闲低电平,数据在下降沿采样 | | 2 | 1 | 0 | 时钟空闲高电平,数据在下降沿采样 | | 3 | 1 | 1 | 时钟空闲高电平,数据在上升沿采样 | - 设备必须配置相同模式才能通信。 #### 4. **优缺点** - **优点**: - 高速传输:速率可达MHz级别(如10-100MHz),远高于I2C。 - 简单高效:无复杂协议开销,支持全双工。 - 灵活性:可自定义数据帧长度和时钟频率。 - **缺点**: - 引脚占用多:每个从机需独立SS线,限制多设备扩展。 - 无错误检测:缺乏内置机制(如CRC),易受干扰。 - 主从依赖:从机无法主动发起通信。 #### 5. **应用场景** - 常见于SD卡、OLED显示屏、温度传感器(如DS18B20)、Flash存储器(如W25Q系列)等。 - 示例代码(Python伪代码,模拟SPI通信逻辑): ```python def spi_transfer(host_data, mode=0): # 初始化SPI设置(简化版) cpol = mode >> 1 # 提取CPOL cpha = mode & 1 # 提取CPHA # 模拟数据传输 miso_data = 0 for bit in range(8): # 根据模式生成时钟边沿 if cpha == 0: # 第一个边沿采样 miso_bit = read_miso() # 读取从机数据 # 主机发送数据位 set_mosi((host_data >> bit) & 1) toggle_sclk() # 切换时钟 if cpha == 1: # 第二个边沿采样 miso_bit = read_miso() miso_data |= (miso_bit << bit) return miso_data # 返回从机数据 ``` SPI嵌入式开发的基础协议,理解其机制有助于高效设计硬件接口。
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值