NRF24L01(2.4G)模块的使用——SPI时序(软件)篇

2024-06-08 18:12

本文主要是介绍NRF24L01(2.4G)模块的使用——SPI时序(软件)篇,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一、SPI的简介:

        SPI 是英语Serial Peripheral interface的缩写,顾名思义就是串行外围设备接口Motorola首先在其MC68HCXX系列处理器上定义的

        SPI,是一种高速的,全双工,同步的通信总线,并且在芯片的管脚上占用四根线,SCK(Serial Clock)、MOSI(Master Output Slave Input)、MISI(Master Input Slave Output)、SS(Slave Select) ,其支持总线挂载多设备(一主多从)。主要应用在 EEPROM,FLASH,实时时钟,AD转换器,还有数字信号处理器和数字信号解码器之间

                                SPI接口一般使用4条线通信:

   MISO 主机输入从机输出。主机通过MOSI输入,从机通过MOSI输出。

   MOSI 主机输出从机输入。主机通过MOSI输出,从机通过MOSI输入。

   SCLK时钟线,完全由主机掌握,主要是产生时钟信号,由主设备产生。对于主机来说时钟线为输出,对于从机来说,所有时钟线为输入。

   CS从设备片选信号,由主设备控制,低电平有效。主机选择从机时,只需要将连接对应的ss线置0就可以选择此从机了。相较于IIC,这种方法更简单但会浪费更多引脚,但无需IIC一样先进行寻址(SS线置0相当于寻址了)。

二、SPI时序介绍

1. spi四种模式详解:

        在spi的模式配置中,有两个很关键的东西,即SPI_CR1的第0和1位:

模式0:CPOL = 0、CPHA = 0

        由于CPOL为0,也就是空闲状态下SCK为低电平;CPHA = 0,也就是从第一个边沿开始采样,也就是上升沿采样。下图是截取NRF24l01的读写时序图,为模式0;起始SCK为低电平,上升沿采样。下降沿移出数据为下一次采样做准备。   由于第一个上升沿就要采样数据,所以得在第一个上升沿就要把数据移出,也就是把CSN的下降沿当作时钟的一部分了。从图中可以看见,CSN下降沿时数据变化(主机输出数据最高位,将数据放于MOSI线上),然后第一个上升沿采样(主机和从机读取数据,主机读从机的最高位,从机读主机的数据最高位),第一个下降沿主机输出次高位(当然,从机也会动作,但从机不需要我们操作),再第二上升沿采样......即先有了下降沿才能有数据变换的条件

第一个时序代码如下:CSN 在选择从机时会拉低,所以这里没写,拉低SS后,主机移出数据,方便从机在下一个上升沿读。此时拉高SCK,主机读从机发来的数据,从机也会读主机发来的数据,注意,为高位先行。然后拉低SCK,主机和从机输出数据,方便下个上升沿读取数据,这样,第一个周期时序就完成了,接下来只需for循环8次就可以了,这样就完成了一次数据交换。和IIC不同的是IIC有读写函数,而SPI读写是同时进行的。

uint8_t MySPI_SW_Byte(uint8_t Byte)
{uint8_t receivebyte = 0x00;for(uint8_t i=0;i<8;i++){SPI_MOSI((BitAction)(Byte & 0x80));SPI_SCK(1);if(Read_MOSI == 1)receivebyte |= 0x80;SPI_SCK(0);}return receivebyte;
}

模式1:CPOL = 0、CPHA = 1

        由于CPOL为0,也就是空闲状态下SCK为低电平;CPHA = 1,也就是从第一个边沿开始移出数据。此时主机将数据最高位放到MOSI线上,从机将数据放到MISO上,也就是上升沿输出。第二个边沿(下降沿)移入数据,即读取数据(主机读MISO,从机读MOSI)到这里完成了一个时序周期。

模式2:CPOL = 1、CPHA = 0

        由于CPOL为0,也就是空闲状态下SCK为高电平;CPHA = 0,也就是从第一个边沿开始移入数据(注:由于空闲时为高电平,所以第一个边沿变成了下降沿)。此时读取数据(主机读MISO,从机读MOSI)。第二个边沿(上升沿)移出数据,主机将数据最高位放到MOSI线上,从机将数据放到MISO上,也就是上升沿输出。到这里完成了一个时序周期即。

模式3:CPOL = 1、CPHA = 1

          由于CPOL为1,也就是空闲状态下SCK为高电平;CPHA = 1,也就是从第一个边沿(下降沿)开始移出数据。此时主机将数据最高位放到MOSI线上,从机将数据放到MISO上,也就是下降沿输出。第二个边沿(上升沿)移入数据,即读取数据(主机读MISO,从机读MOSI)到这里完成了一个时序周期。

三、SPI的初始化

下面进行SPI的初始化:

        由于我们采用的是软件SPI,所以MOSI和SCK配置为推挽输出,而MISO是从机输出线,从机掌握主动权,所以配置为上拉输入。

void MySPI_Init(void)
{GPIO_InitTypeDef GPIO_InitStructure;//定义一个GPIO_InitTypeDef类型的结构体RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);//开启GPIO端口时钟GPIO_InitStructure.GPIO_Pin=GPIO_Pin_5 | GPIO_Pin_7; GPIO_InitStructure.GPIO_Mode=GPIO_Mode_Out_PP;//通用推挽输出GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;//引脚速率50MHZGPIO_Init(GPIOA, &GPIO_InitStructure);GPIO_InitStructure.GPIO_Pin=GPIO_Pin_6; GPIO_InitStructure.GPIO_Mode=GPIO_Mode_IPU;//上拉输入GPIO_Init(GPIOA, &GPIO_InitStructure);SPI_SCK(0);
}

即SPI代码如下:

spi.c

#include "spi.h"void MySPI_Init(void)
{GPIO_InitTypeDef GPIO_InitStructure;//定义一个GPIO_InitTypeDef类型的结构体RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);//开启GPIO端口时钟GPIO_InitStructure.GPIO_Pin=GPIO_Pin_5 | GPIO_Pin_7; GPIO_InitStructure.GPIO_Mode=GPIO_Mode_Out_PP;//通用推挽输出GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;//引脚速率50MHZGPIO_Init(GPIOA, &GPIO_InitStructure);GPIO_InitStructure.GPIO_Pin=GPIO_Pin_6; GPIO_InitStructure.GPIO_Mode=GPIO_Mode_IPU;//上拉输入GPIO_Init(GPIOA, &GPIO_InitStructure);SPI_SCK(0);
}uint8_t MySPI_SW_Byte(uint8_t Byte)
{uint8_t receivebyte = 0x00;for(uint8_t i=0;i<8;i++){SPI_MOSI((BitAction)(Byte & (0x80>>i)));SPI_SCK(1);if(Read_MOSI == 1)receivebyte |= (0x80>>i);SPI_SCK(0);}return receivebyte;
}

spi.h

#ifndef __SPI_H
#define __SPI_H#include "stm32f10x.h"                  // Device header
#include "sys.h"#define SPI_MOSI(x)  GPIOA->BSRR = GPIO_Pin_7<<(16*(!x))
#define SPI_SCK(x)   GPIOA->BSRR = GPIO_Pin_5<<(16*(!x))
#define Read_MOSI    PAin(7) 		//输入MOSIvoid MySPI_Init(void);
uint8_t MySPI_SW_Byte(uint8_t Byte);#endif

下一章将讲如何使用2.4G模块。 

这篇关于NRF24L01(2.4G)模块的使用——SPI时序(软件)篇的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1042933

相关文章

使用Python实现可恢复式多线程下载器

《使用Python实现可恢复式多线程下载器》在数字时代,大文件下载已成为日常操作,本文将手把手教你用Python打造专业级下载器,实现断点续传,多线程加速,速度限制等功能,感兴趣的小伙伴可以了解下... 目录一、智能续传:从崩溃边缘抢救进度二、多线程加速:榨干网络带宽三、速度控制:做网络的好邻居四、终端交互

Python中注释使用方法举例详解

《Python中注释使用方法举例详解》在Python编程语言中注释是必不可少的一部分,它有助于提高代码的可读性和维护性,:本文主要介绍Python中注释使用方法的相关资料,需要的朋友可以参考下... 目录一、前言二、什么是注释?示例:三、单行注释语法:以 China编程# 开头,后面的内容为注释内容示例:示例:四

Python中re模块结合正则表达式的实际应用案例

《Python中re模块结合正则表达式的实际应用案例》Python中的re模块是用于处理正则表达式的强大工具,正则表达式是一种用来匹配字符串的模式,它可以在文本中搜索和匹配特定的字符串模式,这篇文章主... 目录前言re模块常用函数一、查看文本中是否包含 A 或 B 字符串二、替换多个关键词为统一格式三、提

Go语言数据库编程GORM 的基本使用详解

《Go语言数据库编程GORM的基本使用详解》GORM是Go语言流行的ORM框架,封装database/sql,支持自动迁移、关联、事务等,提供CRUD、条件查询、钩子函数、日志等功能,简化数据库操作... 目录一、安装与初始化1. 安装 GORM 及数据库驱动2. 建立数据库连接二、定义模型结构体三、自动迁

ModelMapper基本使用和常见场景示例详解

《ModelMapper基本使用和常见场景示例详解》ModelMapper是Java对象映射库,支持自动映射、自定义规则、集合转换及高级配置(如匹配策略、转换器),可集成SpringBoot,减少样板... 目录1. 添加依赖2. 基本用法示例:简单对象映射3. 自定义映射规则4. 集合映射5. 高级配置匹

Spring 框架之Springfox使用详解

《Spring框架之Springfox使用详解》Springfox是Spring框架的API文档工具,集成Swagger规范,自动生成文档并支持多语言/版本,模块化设计便于扩展,但存在版本兼容性、性... 目录核心功能工作原理模块化设计使用示例注意事项优缺点优点缺点总结适用场景建议总结Springfox 是

嵌入式数据库SQLite 3配置使用讲解

《嵌入式数据库SQLite3配置使用讲解》本文强调嵌入式项目中SQLite3数据库的重要性,因其零配置、轻量级、跨平台及事务处理特性,可保障数据溯源与责任明确,详细讲解安装配置、基础语法及SQLit... 目录0、惨痛教训1、SQLite3环境配置(1)、下载安装SQLite库(2)、解压下载的文件(3)、

使用Python绘制3D堆叠条形图全解析

《使用Python绘制3D堆叠条形图全解析》在数据可视化的工具箱里,3D图表总能带来眼前一亮的效果,本文就来和大家聊聊如何使用Python实现绘制3D堆叠条形图,感兴趣的小伙伴可以了解下... 目录为什么选择 3D 堆叠条形图代码实现:从数据到 3D 世界的搭建核心代码逐行解析细节优化应用场景:3D 堆叠图

Springboot如何正确使用AOP问题

《Springboot如何正确使用AOP问题》:本文主要介绍Springboot如何正确使用AOP问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录​一、AOP概念二、切点表达式​execution表达式案例三、AOP通知四、springboot中使用AOP导出

Navicat数据表的数据添加,删除及使用sql完成数据的添加过程

《Navicat数据表的数据添加,删除及使用sql完成数据的添加过程》:本文主要介绍Navicat数据表的数据添加,删除及使用sql完成数据的添加过程,具有很好的参考价值,希望对大家有所帮助,如有... 目录Navicat数据表数据添加,删除及使用sql完成数据添加选中操作的表则出现如下界面,查看左下角从左