STM32标准库学习笔记(十)SPI

前言


学习永无止境!本篇是嵌入式开发之片上外设SPI,了解基本硬件原理以及通信协议。
注:本文章为学习笔记,部分图片与文字来源于网络/江协科技课程/手册,如侵权请联系!谢谢!


一、SPI通信概述


1.1 SPI基本概念

        SPI(Serial Peripheral Interface)是由摩托罗拉公司提出的通信协议,是一种高速全双工、同步串行通信总线,主要应用在Flash、ADC、编码器等要求速率较高的场合。

1.2 SPI物理层

①信号线:

  • SCLK(Serial Clock):时钟线,用于数据同步,由主机产生,决定了通信速率,不同设备的之间的通信速率受限与低速设备;
  • MOSI(Master Output,Slave Input):主出从进,主设备发送数据,从设备接收数据线,数据方向由主机向从机;
  • MISO(Master Input,Slave Output):主进从出,主设备接收数据,从设备发送数据线,数据方向由从机到主机;
  • CS(Chip Select):片选线,也叫SS(Slave Select),用来寻址,指定要通信设备,一般有几个从设备,就要有几根CS。

②电平:

        TTL电平,时钟极性(CPOL)与时钟相位(CPHA)的的选择,决定数据是在时钟上升沿采样还是下降沿采样,共四种模式,下面具体介绍。

1.3 SPI协议层

①基本时序:

  • 起始信号:CS信号由高变低,表示选中从机,相应从机检测到起始信号,知道被主机选中;
  • 数据发送与接收:数据高位先行(MSB),双方数据在时钟上升沿或下降沿发送(接收);
  • 停止信号:CS信号由低变高,表示通信结束。

②SPI模式决定因素:

  • CPOL:时钟极性,SCLK时钟线在空闲状态是高电平还是低电平由此决定,当CPOL=0时,SCLK空闲低电平。当CPOL=1时,SCLK空闲状态高电平;
  • CPHA:时钟相位,数据的采样时刻,当CPHA=0时,MOSI或MISO数据线上的信号将会在时钟线的奇数边沿被采样。当CPHA=1时,MOSI或MISO数据线上的信号将会在时钟线的偶数边沿被采样。

③SPI四种通信模式:

  • CPHA=0,CPOL=0:数据在时钟奇数边沿采样,时钟默认空闲电平;
  • CPHA=0,CPOL=1:数据在时钟奇数边沿采样,时钟默认空闲电平;

  • CPHA=1,CPOL=0:数据在时钟偶数边沿采样,时钟默认空闲电平;
  • CPHA=1,CPOL=1:数据在时钟偶数边沿采样,时钟默认空闲电平;


二、STM32的SPI外设


2.1 基本简介

  • 主从机选择:可作为SPI通讯主机,也可作为从机;
  • 时钟频率:最大SCLK的频率为fpclk/2(f103的fpclk1为36MHZ,fpclk2为76MHZ);
  • 数据格式:可选MSB先行,也可选LSB先行;
  • 模式:除了支持正常双线全双工的四种模式,还有双向单线(同时向一个方向传输,速度提高一倍)以及单线模式(减少硬件接线);

2.2 硬件基本结构框图

  • ①外部引脚:四个通信口与GPIO口的对应关系,需要查表;
  • ②时钟控制:根据控制寄存器CR1BR[0:2],对fpclk进行分频;
  • ③数据控制:数据移位寄存器以发送缓冲区作为数据源,把数据一位一位发送,当从外部接收数据时,数据移位寄存器把数据线数据采样一位位存储到接收缓冲区。数据帧长度设定GFF位可为8或16位,配置LSBFIRST,可选择MSB先行或LSB先行;
  • ④主控:通过改变CR1/2的参数,可以配置SPI模式、波特率、LSB/MSB、主从模式、单双向模式,状态寄存器SR实时反馈SPI工作状态,除此之外还有中断、DMA请求等控制。

2.3 主机发送与接收

  • ①CS片选:当要进行通信,首先选择从机地址,CS由高电平置为低电平;
  • ②SCLK:CS片选同时时钟进行工作;
  • ③数据发送:MOSI把发送缓冲区数据一位位传输出去,当发送完一帧数据,SR置TXE标志1(表示发送缓冲区已清空),若要再次发送数据,在TXE为1时,写入DR数据即可;
  • ④数据接收:MISO移位寄存器把数据线数据采样一位位存储到接收缓冲区,当接收完一帧数据,SR置RXNE标志1(表示接收缓冲区非空),即可读取DR数据,每次等待RXNE标志位1,即可读取接收数据。


三、应用


3.1 软件模拟SPI

①本次示例说明:使用软件模拟CS、SCLK、MOSI、MISO,利用I/O高低电平翻转实现SPI时序。

配置CS\SCLK\MOSI为推挽输出(PA4\5\7),MISO为上拉输入(PA6),选用模式1(SCLK初始为低电平,数据在奇数边沿采样)。

②配置步骤:

  • 开启RCC时钟:开启GPIOA外设时钟;
  • 配置GPIO:CS(GPIOA_Pin4)、SCLK(GPIOA_Pin5)、MOSI(GPIOA_Pin7)推挽输出,MISO(GPIOA_Pin6)上拉输出;
  • 起始信号;
  • 停止信号;
  • 发送字bit;
  • 接收bit;
  • 发送与接收字节:通过循环移位进行写入与读取,将数据从高位依次取出放在发送地址,每取一位,置SCLK位1,并读取MISO电平信号,读取完毕,置SCLK为0。

③代码实战:

MSPI.c:

#include "stm32f10x.h"                  // Device header/*设置CS电平*/
void MySPI_W_CS(uint8_t BitValue)
{GPIO_WriteBit(GPIOA, GPIO_Pin_4, (BitAction)BitValue);		
}/*写SCLK*/
void MySPI_W_SCK(uint8_t BitValue)
{GPIO_WriteBit(GPIOA, GPIO_Pin_5, (BitAction)BitValue);		
}/*发送MOSI电平*/
void MySPI_W_MOSI(uint8_t BitValue)
{GPIO_WriteBit(GPIOA, GPIO_Pin_7, (BitAction)BitValue);		
}/*接收MISO电平*/
uint8_t MySPI_R_MISO(void)
{return GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_6);			
}/*初始化引脚配置*/
void MySPI_Init(void)
{/*开启GPIOA外设时钟*/RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);	/*GPIO初始化*/GPIO_InitTypeDef GPIO_InitStructure;GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;//将PA4、PA5和PA7引脚初始化为推挽输出GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4 | GPIO_Pin_5 | GPIO_Pin_7;GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;GPIO_Init(GPIOA, &GPIO_InitStructure);					GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6;//将PA6引脚初始化为上拉输入GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;GPIO_Init(GPIOA, &GPIO_InitStructure);					/*设置默认电平*/MySPI_W_CS(1);	//CS默认高电平										MySPI_W_SCK(0);	//SCLK默认低电平										
}/*协议层*//*起始信号*/
void MySPI_Start(void)
{MySPI_W_CS(0);				
}/*结束信号*/
void MySPI_Stop(void)
{MySPI_W_CS(1);				
}/*发送接收数据字节*/
uint8_t MySPI_SwapByte(uint8_t ByteSend)
{uint8_t i, ByteReceive = 0x00;					//定义接收的数据,并赋初值0x00,此处必须赋初值0x00,后面会用到for (i = 0; i < 8; i ++)						//循环8次,依次交换每一位数据{MySPI_W_MOSI(ByteSend & (0x80 >> i));		//使用掩码的方式取出ByteSend的指定一位数据并写入到MOSI线MySPI_W_SCK(1);								//拉高SCK,上升沿移出数据if (MySPI_R_MISO() == 1){ByteReceive |= (0x80 >> i);}	//读取MISO数据,并存储到Byte变量//当MISO为1时,置变量指定位为1,当MISO为0时,不做处理,指定位为默认的初值0MySPI_W_SCK(0);								//拉低SCLK,下降沿移入数据}return ByteReceive;								//返回接收到的一个字节数据
}

3.2 硬件SPI

①本次示例说明:使用STM32硬件SPI配置,选用SPI1外设,配置时钟极性为低电平,相位为上升沿采样。

②配置步骤:

  • 开启RCC时钟:开启GPIOA、SPI1外设时钟;
  • 配置GPIO:CS(GPIOA_Pin4)为推挽输出,SCLK(GPIOA_Pin5)、MOSI(GPIOA_Pin7)复用推挽输出,MISO(GPIOA_Pin6)上拉输出;
  • 起始信号;
  • 停止信号;
  • 发送与接收字节:等待TXE为空,写入字节,等待RXNE非空,读取字节。

③代码实战:

MSPI.c:

#include "stm32f10x.h"                  // Device header/*设置CS*/
void MySPI_W_CS(uint8_t BitValue)
{GPIO_WriteBit(GPIOA, GPIO_Pin_4, (BitAction)BitValue);		
}/*初始化SPI*/
void MySPI_Init(void)
{/*开启GPIOA\SPI1的外设时钟*/RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);RCC_APB2PeriphClockCmd(RCC_APB2Periph_SPI1, ENABLE);	/*GPIO初始化*/GPIO_InitTypeDef GPIO_InitStructure;GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4;//将PA4引脚初始化为推挽输出GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;GPIO_Init(GPIOA, &GPIO_InitStructure);					GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5 | GPIO_Pin_7;//将PA5和PA7引脚初始化为复用推挽输出GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;GPIO_Init(GPIOA, &GPIO_InitStructure);					GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6;//将PA6引脚初始化为上拉输入GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;GPIO_Init(GPIOA, &GPIO_InitStructure);					/*SPI初始化*/SPI_InitTypeDef SPI_InitStructure;						//定义结构体变量SPI_InitStructure.SPI_Mode = SPI_Mode_Master;			//模式,选择为SPI主模式SPI_InitStructure.SPI_Direction = SPI_Direction_2Lines_FullDuplex;	//方向,选择2线全双工SPI_InitStructure.SPI_DataSize = SPI_DataSize_8b;		//数据宽度,选择为8位SPI_InitStructure.SPI_FirstBit = SPI_FirstBit_MSB;		//先行位,选择高位先行SPI_InitStructure.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_128;	//波特率分频,选择128分频SPI_InitStructure.SPI_CPOL = SPI_CPOL_Low;				//SPI极性,选择低极性SPI_InitStructure.SPI_CPHA = SPI_CPHA_1Edge;			//SPI相位,选择第一个时钟边沿采样,极性和相位决定选择SPI模式0SPI_InitStructure.SPI_NSS = SPI_NSS_Soft;				//NSS,选择由软件控制SPI_InitStructure.SPI_CRCPolynomial = 7;				//CRC多项式,暂时用不到,给默认值7SPI_Init(SPI1, &SPI_InitStructure);						//将结构体变量交给SPI_Init,配置SPI1/*SPI使能*/SPI_Cmd(SPI1, ENABLE);									/*设置默认电平*/MySPI_W_CS(1);											
}/*起始信号*/
void MySPI_Start(void)
{MySPI_W_CS(0);				
}/*结束信号*/
void MySPI_Stop(void)
{MySPI_W_CS(1);				
}/*** 函    数:SPI交换传输一个字节,使用SPI模式0* 参    数:ByteSend 要发送的一个字节* 返 回 值:接收的一个字节*/
uint8_t MySPI_SwapByte(uint8_t ByteSend)
{while (SPI_I2S_GetFlagStatus(SPI1, SPI_I2S_FLAG_TXE) != SET);	//等待发送数据寄存器空SPI_I2S_SendData(SPI1, ByteSend);								//写入数据到发送数据寄存器,开始产生时序while (SPI_I2S_GetFlagStatus(SPI1, SPI_I2S_FLAG_RXNE) != SET);	//等待接收数据寄存器非空return SPI_I2S_ReceiveData(SPI1);								//读取接收到的数据并返回
}

待续...

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

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

相关文章

探索 C++ 与 LibUSB:开启 USB 设备交互的奇幻之旅

一、引言 在当今数字化时代&#xff0c;USB&#xff08;通用串行总线&#xff09;设备无处不在&#xff0c;从常见的 U 盘、鼠标、键盘&#xff0c;到复杂的工业数据采集设备、医疗监测仪器等&#xff0c;它们以方便快捷的插拔式连接&#xff0c;为人们的生活和工作带来了极大…

LeetCode 209 题:长度最小的子数组

LeetCode 209 题&#xff1a;长度最小的子数组 (Minimum Size Subarray Sum) LeetCode 第209题要求给定一个正整数数组 nums 和一个正整数 target&#xff0c;找出具有最小长度的连续子数组&#xff0c;使得子数组的和大于或等于 target。如果不存在这样的子数组&#xff0c;返…

从github上,下载的android项目,从0-1进行编译运行-踩坑精力,如何进行部署

因为国内的网络原因&#xff0c;一直在anroidstudio开发的问题上&#xff0c;是个每个开发者都会踩坑 一直以为是自己的原因&#xff0c;其实很多都是国内网络的原因&#xff0c;今天就从一个开发者的视角 把从github上一个陌生的项目&#xff0c;如何通过本地就行运行的 首先…

计算机网络 (40)域名系统DNS

前言 计算机网络域名系统DNS&#xff08;Domain Name System&#xff09;是互联网的基础技术之一&#xff0c;它负责将人类可读的域名转换为计算机用来通信的数字IP地址。 一、基本概念 DNS的主要目的是将域名解析或翻译为IP地址&#xff0c;使得用户可以通过简单易记的域名来访…

使用Dify创建个问卷调查的工作流

为啥要使用Dify创建工作流呢&#xff1f;一个基于流程的智能体的实现&#xff0c;特别是基于业务的实现&#xff0c;使用Dify去实现时&#xff0c;通常都是一个对话工作流&#xff0c;当设计到相对复杂一些的流程时&#xff0c;如果将所有逻辑都放在对话工作流中去实现&#xf…

toRef 和 toRefs 详解及应用

在 Vue 3 中&#xff0c;toRef 和 toRefs 是两个用于创建响应式引用的工具&#xff0c;主要用于组合式 API&#xff08;Composition API&#xff09;的场景中 1. toRef 定义 toRef 将某个对象的某个属性包装成一个响应式引用。这样可以直接对该引用进行操作&#xff0c;而不需…

八 rk3568 android11 AP6256 蓝牙调试

一 经典蓝牙 经典蓝牙默认可以工作, 验证可以连接 蓝牙鼠标,键盘, 连接手机等等, 在 系统设置里打开蓝牙 ,扫描设备,配对连接即可。 注: 连接 ANDROID 手机的坑 1 手机连接之后空闲状态会断开 ,变成 配对的设备不是已连接,是正常,使用时又会自动 连接 2 手机传…

解读若依微服务架构图:架构总览、核心模块解析、消息与任务处理、数据存储与缓存、监控与日志

文章目录 1. 引言2. 架构总览3. 核心模块解析3.1 服务注册与配置中心Nacos&#xff1a;微服务的中枢 3.2 网关层ruoyi-gateway&#xff1a;服务的统一入口 3.3 核心业务服务3.4 认证服务ruoyi-auth&#xff1a;认证与授权的守护者 3.5 异构服务整合Sidecar&#xff1a;连接异构…

【MySQL】基础架构分析

考察频率难度40%⭐⭐⭐⭐ 这道题在面试时的出现频率其实并不高&#xff0c;最起码对于笔者来说是没有遇到过。那为什么还是选择把这个问题作为 MySQL 八股文系列的第一个呢&#xff1f;其实原因也挺简单的&#xff0c;还是老规矩&#xff0c;先通过一个问题把整个知识框架来一…

【已解决】【记录】2AI大模型web UI使用tips 本地

docker desktop使用 互动 如果需要发送网页链接&#xff0c;就在链接上加上【#】号 如果要上传文件就点击这个➕号 中文回复 命令它只用中文回复&#xff0c;在右上角打开【对话高级设置】 输入提示词&#xff08;提示词使用英文会更好&#xff09; Must reply to the us…

热烈祝贺“钛然科技”选择使用订单日记

感谢珠海钛然科技有限公司选择使用订单日记&#xff01; 珠海钛然科技有限公司&#xff0c;成立于2020年&#xff0c;位于广东省珠海市高新区&#xff0c;是一家以从事研发和生产功能型纳米高分子涂层为主的企业。 在业务不断壮大的过程中&#xff0c;想使用一种既能提升运营…

关于H5复制ios没有效果

问题场景&#xff1a;今天遇到这样一个问题&#xff0c;需要从后端接口获取到的值进行复制&#xff0c;且不能提现调用获取值&#xff0c;因为是一个数据列表&#xff0c;每个列表元素需要当场点击调用接口获取值进行复制&#xff0c;本来以为很简单的一个需求&#xff0c;当做…

使用 versions-maven-plugin 和 flatten-maven-plugin 插件惯例 maven 项目版本

在 Maven 项目中&#xff0c;依赖版本管理和 POM 文件的规范化是确保项目可维护性和一致性的关键。今天&#xff0c;我们将介绍两个强大的 Maven 插件&#xff1a;versions-maven-plugin 和 flatten-maven-plugin&#xff0c;它们可以帮助我们更高效地管理项目版本和 POM 文件。…

【Vue】Vue 拖拽指令 禁选文字 解决子元素 input 不能输入 、拖动粘连鼠标

directives: { drag(el) { // 解决拖动会选中文字的问题 document.onselectstart () > { return false } el.onmousedown function (event) { // 解决子元素 input 框无法输入等情况 if(event.target?.nodeName INPUT || event.target?.nodeName IMG) { return } cons…

freertos的基础(二)内存管理:堆和栈

1. 堆&#xff08;Heap&#xff09; 定义 堆是 FreeRTOS 中用于动态内存分配的内存区域。FreeRTOS 提供了多种堆管理方案&#xff08;如 heap_1、heap_2、heap_4 等&#xff09;&#xff0c;开发者可以根据需求选择合适的内存管理策略。 作用 用于动态分配内存&#xff0c;例…

Linux-----进程通讯(消息队列)

目录 相关API 1.相关数据类型 mqd_t struct mq_attr struct timespec 2.相关系统调用接口 mq_open() mq_timedsend() && mq_send() mq_timedreceive() && mq_receive() mq_unlink() clock_gettime() 父子进程使用消息队列通讯 平行进程使用消息队列…

【微服务】面试题 5、分布式系统理论:CAP 与 BASE 详解

分布式系统理论&#xff1a;CAP 与 BASE 详解 一、CAP 定理 背景与定义&#xff1a;1998 年由加州大学科学家埃里克布鲁尔提出&#xff0c;分布式系统存在一致性&#xff08;Consistency&#xff09;、可用性&#xff08;Availability&#xff09;、分区容错性&#xff08;Part…

数据结构与算法之二叉树: LeetCode 572. 另一棵树的子树 (Ts版)

另一棵树的子树 https://leetcode.cn/problems/subtree-of-another-tree/description/ 描述 给你两棵二叉树 root 和 subRoot检验 root 中是否包含和 subRoot 具有相同结构和节点值的子树如果存在&#xff0c;返回 true &#xff1b;否则&#xff0c;返回 false二叉树 tree …

NineData云原生智能数据管理平台新功能发布|2024年12月版

本月发布 7 项更新&#xff0c;其中重点发布 2 项、功能优化 5 项。 重点发布 数据库 Devops - Oracle 非表对象支持可视化创建与管理 Oracle 非表对象&#xff0c;包括视图&#xff08;View&#xff09;、包&#xff08;Package&#xff09;、存储过程&#xff08;Procedur…

[Unity]MacOS下开发Unity

需要的插件 我使用的是vscode&#xff0c;经过长时间的使用我发现一个问题就是很多插件都是动态的在变化的&#xff0c;不是一成不变的&#xff0c;可能是重构&#xff0c;可能直接换了其他的工具。 所以这个插件也会是更新的状态。 2025年01月08日更新 .NET Install Tool (…