完善资料让更多小伙伴认识你,还能领取20积分哦, 立即完善>
电子发烧友论坛|
【实验目的】
1、通过实验了解SPI的通信模式及配置过程。 2、通过使用SPI与蓝牙模块NRF2401进行通信,送内部数据到蓝牙模块并读取从蓝牙主机上发送的控制信息,了解蓝牙模块的配置和通信过程。 【实验原理】 SPI 协议(Serial Peripheral Interface),即串行外围设备接口,是一种高速全双工的通信总线,它由摩托罗拉公司提出,当前最新的为 V04.01-2004 版。它被广泛地使用在 ADC、 LCD 等设备与 MCU 间通讯的场合。SPI通讯设备之间常用连接方式如图1所示。 图1 常见的SPI通讯系统 一、SPI原理 SPI 接口一般使用 4 条线: MISO 主设备数据输入,从设备数据输出。 MOSI 主设备数据输出,从设备数据输入SCLK 时钟信号,由主设备产生。CS 从设备片选信号,由主设备控制。根据 SPI 时钟极性(CPOL)和时钟相位(CPHA) 配置的不同,分为四种 SPI 模式。时钟极性是指 SPI 通讯设备处于空闲状态时SCK 信号线的电平信号。时钟相位是指数据的采样的时刻。 二、SPI特性 SPI 主要特点有:可以同时发出和接收串行数据;可以当作主机或从机工作; 提供频率可编程时钟;发送结束中断标志;写冲突保护;总线竞争保护等。 SPI 模块为了和外设进行数据交换,根据外设工作要求,其输出串行同步时钟极性和相位可以进行配置,时钟极性(CPOL)对传输协议没有重大的影响。如果CPOL=0,串行同步时钟的空闲状态为低电平;如果 CPOL=1,串行同步时钟的空闲状态为高电平。时钟相位(CPHA)能够配置用于选择两种不同的传输协议之一进行数据传输。如果CPHA=0,在串行同步时钟的第一个跳变沿(上升或下降)数据被采样;如果 CPHA=1,在串行同步时钟的第二个跳变沿(上升或下降)数据被采样。SPI 主模块和与之通信的外设备时钟相位和极性应该一致。 三、SPI库函数分析 跟其他外设一样,STM32标准库提供了SPI初始化结构体及初始化函数来配置SPI外设。初始化结构体及函数定义在库文件“stm32f4xx_spi.h”和“stm32f4xx_spi.c”中。 SPI初始化结构体为SPI_InitTypeDef,其中包含: 表1 SPI_InitTypeDef配置 配置完这些结构体成员后,我们要调用库函数: SPI_Init(SPI_TypeDef* SPIx, SPI_InitTypeDef* SPI_InitStruct) 把这些参数写入到寄存器中,实现 SPI 的初始化,然后调用库函数:SPI_Cmd(SPI_TypeDef* SPIx, FunctionalState NewState) 来使能 SPI 外设。 在进行SPI发送数据时我们需要用到库函数:SPI_I2S_GetFlagStatus(SPI_TypeDef* SPIx, uint16_t SPI_I2S_FLAG) 来判断指定的SPI的标志位,在本实验中,检查指定的SPI标志位设置与否:发送缓存空标志位。 SPI_I2S_SendData(SPI_TypeDef* SPIx, uint16_t Data)来发送指定的数据。 在进行SPI接受数据时我们需要用到库函数: SPI_I2S_GetFlagStatus(SPI_TypeDef* SPIx, uint16_t SPI_I2S_FLAG) 来判断指定的SPI的标志位,在本实验中,检查指定的SPI标志位设置与否:接受缓存非空标志位。 SPI_I2S_ReceiveData(SPI_TypeDef* SPIx)通过SPIx最近接受的数据。 四、蓝牙模块NRF2401 nRF24L01是一款工作在2.4~2.5GHz世界通用ISM频段的单片无线收发器芯片。无线收发器包括:频率发生器、增强型SchockBurst模式控制器、功率放大器、晶体振荡器、调制器、解调器、输出功率、频道选择和协议的设置可以通过SPI接口进行设置。极低的电流消耗:当工作在发射模式下发射功率为-6dBm时电流消耗为9.0mA,接收模式时为12.3mA,掉电模式和待机模式下电流消耗更低。 五、软件流程图 图2 程序流程图 【实验环境】 操作系统 Windows7/8/10,32bit/64bit 硬件设备 小车所搭载的电路板主控芯片留有蓝牙调试端口,可以通过SPI连接蓝牙设备发送数据进行调试。 软件 Keil 5,串口助手软件 【实验步骤】 一、配置工程环境 1.1 在操作之前需要把关于GPIO,SPI,USART等的库文件添加到工程模板之中。在添加这些库文件之前需要把与stm32f10x_xxx.c 文件对应的一个 stm32f10x_xxx.h 头文也包含进我们的工程中才能够使用这些外设库。如图3所示。 图3 所需的头文件 二、开启时钟,完成端口初始化 2.1 打开程序中的spi.c文件,对SPI1_init函数进行编写和修改。在这个函数中我们调用了库函数 RCC_APB2PeriphClockCmd()初始化SPI1 和 GPIOC 的时钟。 2.2 GPIO端口时钟初始化 /*对GPIOC端口进行初始化设置*/ GPIO_InitTypeDef GPIO_InitStructure; RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE ); //初始化时钟 2.3 GPIO 端口模式设置 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5 | GPIO_Pin_6 | GPIO_Pin_7; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; //复用推挽输出 GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; //GPIO速率 GPIO_Init(GPIOA, &GPIO_InitStructure); //GPIO初始化 GPIO_SetBits(GPIOA,GPIO_Pin_5|GPIO_Pin_6|GPIO_Pin_7); 2.4 对SPI进行初始化配置,使用SPI初始化结构体。 SPI_InitStructure.SPI_Direction = SPI_Direction_2Lines_FullDuplex; //设置SPI单向或者双向的数据模式:SPI设置为双线双向全双工 SPI_InitStructure.SPI_Mode = SPI_Mode_Master; //设置SPI工作模式:设置为主SPI SPI_InitStructure.SPI_DataSize = SPI_DataSize_8b; //设置SPI的数据大小:SPI发送接收8位帧结构 SPI_InitStructure.SPI_CPOL = SPI_CPOL_High; //选择了串行时钟的稳态:时钟悬空高 SPI_InitStructure.SPI_CPHA = SPI_CPHA_2Edge; //数据捕获于第二个时钟沿 SPI_InitStructure.SPI_NSS = SPI_NSS_Soft; //NSS信号由硬件(NSS管脚)还是软件(使用SSI位)管理:内部NSS信号有SSI位控制 SPI_InitStructure.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_256 //定义波特率预分频的值:波特率预分频值为256 SPI_InitStructure.SPI_FirstBit = SPI_FirstBit_MSB; //指定数据传输从MSB位还是LSB位开始:数据传输从MSB位开始 SPI_InitStructure.SPI_CRCPolynomial = 7; //CRC值计算的多项式 SPI_Init(SPI1, &SPI_InitStructure); //根据SPI_InitStruct中指定的参数初始化外设SPIx寄存器,这里我们初始化的是 SPI1 2.5 使能SPI。 SPI_Cmd(SPI1, ENABLE); //使能SPI外设SPI1 SPI1_ReadWriteByte(0xff); //启动传输 2.6 设置SPI的传输速率。 图4 SPI控制寄存器1结构图 通过SPI的控制寄存器1设置SPI的传输速率,如图4所示 void SPI1_SetSpeed(u8 SpeedSet) { SPI1->CR1&=0XFFC7; //将第3位,第4位,第5位清零 SPI1->CR1|=SpeedSet; //设置SPI1速度 SPI1->CR1|=1<<6; //SPI设备使能 } 2.7 编写SPI读写字节函数。 //SPIx 读写一个字节 //TxData:要写入的字节 //返回值:读取到的字节 u8 SPI1_ReadWriteByte(u8 TxData) { u8 retry=0; while (SPI_I2S_GetFlagStatus(SPI1, SPI_I2S_FLAG_TXE) == RESET) //检查指定的SPI标志位设置与否:发送缓存空标志位 { retry++; if(retry>200) return 0; //超时退出 } //判断数据寄存器中是否有数据,若没有,则写入我们的数据 SPI_I2S_SendData(SPI1, TxData); //通过外设SPIx发送一个数据 retry=0; while (SPI_I2S_GetFlagStatus(SPI1,SPI_I2S_FLAG_RXNE)== RESET) //检查指定的SPI标志位设置与否:接受缓存非空标志位 { retry++; if(retry>200) return 0; //超时退出 } //判断数据寄存器中是否有数据,若有,则将其读取出来 return SPI_I2S_ReceiveData(SPI1); //返回通过SPIx最近接收的数据 } 在24L01.c中编辑传输函数。 因为24L01程序较为复杂,建议使用参考例程。仅做初始化配置。 三、编写SPI模块 3.1 初始化24L01的IO口。 //初始化24L01的IO口 void NRF24L01_Init(void) { GPIO_InitTypeDef GPIO_InitStructure; RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB| RCC_APB2Periph_GPIOC, ENABLE ); //PC5端口设置 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5; GPIO_InitStructure.GPIO_Mode =GPIO_Mode_Out_PP ; //推挽输出 GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_Init(GPIOC, &GPIO_InitStructure); GPIO_SetBits(GPIOC,GPIO_Pin_5); //PB12端口设置 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_12; GPIO_InitStructure.GPIO_Mode =GPIO_Mode_Out_PP ; //推挽输出 GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_Init(GPIOB, &GPIO_InitStructure); GPIO_SetBits(GPIOB,GPIO_Pin_12); //PC4端口设置 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU ; //上拉输入 GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_Init(GPIOC, &GPIO_InitStructure); GPIO_SetBits(GPIOC,GPIO_Pin_4); } // NRF24L01_CE和NRF24L01_CSN的定义在24l01.h中 NRF24L01_CE=0; //使能24L01 NRF24L01_CSN=1; //SPI片选取消 3.2编写基于SPI的24L01读写函数。 //SPI写寄存器 //reg:指定寄存器地址 //value:写入的值 u8 NRF24L01_Write_Reg(u8 reg,u8 value) { u8 status; NRF24L01_CSN=0; //使能SPI传输 status =SPI1_ReadWriteByte(reg);//发送寄存器号 SPI1_ReadWriteByte(value); //写入寄存器的值 NRF24L01_CSN=1; //禁止SPI传输 return(status); //返回状态值 } //读取SPI寄存器值 //reg:要读的寄存器 u8 NRF24L01_Read_Reg(u8 reg) { u8 reg_val; NRF24L01_CSN = 0; //使能SPI传输 SPI1_ReadWriteByte(reg); //发送寄存器号 reg_val=SPI1_ReadWriteByte(0XFF);//读取寄存器内容 NRF24L01_CSN = 1; //禁止SPI传输 return(reg_val); //返回状态值 } 四、编写main函数和控制模块 通过蓝牙遥控器控制小车 4.1 控制语句编写在程序control.c文件中。因为在24L01和按键程序中已经对输入信号做了处理,因此只需判断蓝牙发送的数据是何种控制命令即可。 /************************************************************ 函数功能:采集遥控器的信号 入口参数:无 返回 值:无 ************************************************************/ void Get_MC6(void) { if(Flag_Left==0&&Flag_Right==0) //判断左转和右转标志位是否为零 { if((Remoter_Ch1>1650&&Remoter_Ch1<2100) ||(Remoter_Ch1>21650&&Remoter_Ch1<22100)) Flag_Qian=1,Flag_Hou=0,Flag_sudu=1; //判断遥控接收变量Remoter_Ch1 //前进 else if((Remoter_Ch1<1350&&Remoter_Ch1>900) ||(Remoter_Ch1<21350&&Remoter_Ch1>20900)) Flag_Qian=0,Flag_Hou=1,Flag_sudu=1; //后退 else if ((Remoter_Ch1>1350&&Remoter_Ch1<1650) ||(Remoter_Ch1>21350&&Remoter_Ch1<21650)) Flag_Qian=0,Flag_Hou=0; //停 } if(Flag_Qian==0&&Flag_Hou==0)//判断前进和后退标志位是否为零 { if((Remoter_Ch2>1650&&Remoter_Ch2<2100) ||(Remoter_Ch2>21650&&Remoter_Ch2<22100)) Flag_Left=1,Flag_Right=0,Flag_sudu=1; //判断遥控接收变Remoter_Ch2 //左转 else if((Remoter_Ch2<1350&&Remoter_Ch2>900) ||(Remoter_Ch2<21350&&Remoter_Ch2>20900)) Flag_Left=0,Flag_Right=1,Flag_sudu=1; //右转 else if ((Remoter_Ch2>1350&&Remoter_Ch2<1650) ||(Remoter_Ch2>21350&&Remoter_Ch2<21650)) Flag_Left=0,Flag_Right=0; //停 } } — 五、编译并下载程序到小车。 图5 Keil编译环境下的下载按键 【实验思考】 一、选择题 题目1:下面哪条语句是设置SPI通讯数据的大小(D) A:SPI_InitStructure.SPI_CPHA = SPI_CPHA_2Edge; B:SPI_InitStructure.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_256; C:SPI_InitStructure.SPI_Mode = SPI_Mode_Master; D:SPI_InitStructure.SPI_DataSize = SPI_DataSize_8b; 题目2:在对SPI进行初始化时,涉及到的端口应该设置成哪种模式?(A) A:复用推挽输出 B:复用开漏输出 二、简答题 题目1:SPI的片选线(CS)设置是必需的吗? CS线用于控制片选信号。当一个SPI从设备的CS线识别到了预先规定的片选电平,则表示该设备被选中,接下来的操作对其有效。显然,使用CS线可以完成“一主多从”的SPI网络架设,但是,在“一主一从”的SPI通信时,CS线不是必需的。 附录:SPI 库函数 |
|
|
|
|
只有小组成员才能发言,加入小组>>
调试STM32H750的FMC总线读写PSRAM遇到的问题求解?
4130 浏览 1 评论
X-NUCLEO-IHM08M1板文档中输出电流为15Arms,15Arms是怎么得出来的呢?
3221 浏览 1 评论
2747 浏览 2 评论
STM32F030F4 HSI时钟温度测试过不去是怎么回事?
2175 浏览 2 评论
ST25R3916能否对ISO15693的标签芯片进行分区域写密码?
14939 浏览 2 评论
STM32仿真器是选择ST-LINK还是选择J-LINK?各有什么优势啊?
3084浏览 4评论
stm32f4下spi+dma读取数据不对是什么原因导致的?
1892浏览 3评论
STM32F0_TIM2输出pwm2后OLED变暗或者系统重启是怎么回事?
2064浏览 3评论
1976浏览 3评论
stm32cubemx生成mdk-arm v4项目文件无法打开是什么原因导致的?
2165浏览 3评论
/9
小黑屋| 手机版| Archiver| 电子发烧友 ( 湘ICP备2023018690号 )
GMT+8, 2025-12-1 22:38 , Processed in 0.556468 second(s), Total 72, Slave 55 queries .
Powered by 电子发烧友网
© 2015 bbs.elecfans.com
关注我们的微信
下载发烧友APP
电子发烧友观察
版权所有 © 湖南华秋数字科技有限公司
电子发烧友 (电路图) 湘公网安备 43011202000918 号 电信与信息服务业务经营许可证:合字B2-20210191

淘帖
1866