STM32——SPI通信实验
程序配置过程:
1、使能SPIx和IO口时钟:
RCC_AHBxPeriphClockCmd()/RCC_APBxPeriphClockCmd();
2、初始化IO口为复用功能:
void GPIO_Init(GPIO_TypeDef* GPIOx,GPIO_InitTypeDef* GPIO_InitStruct);
3、设置引脚复用映射:
GPIO_PinAFConfig();
4、初始化SPIx,设置SPIx工作模式:
void SPI_Init(SPI_TypeDef* SPIx,SPI_InitTypeDef* SPI_InitStruct);
5、使能SPIx:
void SPI_Cmd(SPI_TypeDef* SPIx,FunctionalState NewState);
6、SPI传输数据:
void SPI_I2C_SendData(SPI_TypeDef* SPIx,uint16_t Data);
uint16_t SPI_I2S_ReceiveData(SPI_TypeDef* SPIx);
7、查看SPI传输数据:
SPI_I2C_GetFlagStatus(SPI2,SPI_I2S_FLAG_RXNE);
使能SPIx和IO口时钟
void SPI_Init(void)
{RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA,ENABLE);RCC_APB2PeriphClockCmd(RCC_APB2Periph_SPI1,ENABLE);
}
初始化IO口为复用功能
void SPI_Init(void)
{RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA,ENABLE);RCC_APB2PeriphClockCmd(RCC_APB2Periph_SPI1,ENABLE);GPIO_InitTypeDef GPIO_InitABC;GPIO_InitABC.GPIO_Pin=GPIO_Pin_3|GPIO_Pin_4|GPIO_Pin_5;//PB3~5复用功能输出GPIO_InitABC.GPIO_Mode=GPIO_Mode_AF;//复用功能GPIO_InitABC.GPIO_OType=GPIO_OType_PP;//推挽输出GPIO_InitABC.GPIO_Speed=GPIO_Speed_100MHz;//100MHzGPIO_InitABC.GPIO_PuPd=GPIO_PuPd_UP;//上拉GPIO_Init(GPIOB,&GPIO_InitABC);
}
设置引脚复用映射
void SPI_Init(void)
{GPIO_PinAFConfig(GPIOB,GPIO_PinSource3,GPIO_AF_SPI1);//PB3复用为SPI1GPIO_PinAFConfig(GPIOB,GPIO_PinSource4,GPIO_AF_SPI1);//PB4复用为SPI1GPIO_PinAFConfig(GPIOB,GPIO_PinSource5,GPIO_AF_SPI1);//PB5复用为SPI1RCC_APB2PeriphResetCmd(RCC_APB2Periph_SPI1,ENABLE);//复位SPI1RCC_APB2PeriphResetCmd(RCC_APB2Periph_SPI1,DISABLE);//停止复位SPI1
}
初始化SPIx,设置SPIx工作模式
void SPI_Init(void)
{SPI_InitTypeDef SPI_InitABC;SPI_InitABC.SPI_Direction=SPI_Direction_2Lines_FullDuplex;//设置SPI单向或者双向的数据模式SPI_InitABC.SPI_Mode=SPI_Mode_Master;//设置SPI工作模式:设置为主SPISPI_InitABC.SPI_DataSize=SPI_DataSize_8b;//设置SPI的数据大小:SPI发送接收8为帧结构SPI_InitABC.SPI_CPOL=SPI_CPOL_High;//串行同步时钟的空闲状态为高电平SPI_InitABC.SPI_CPHA=SPI_CPHA_2Edge;//串行同步时钟的第二个跳变沿(上升或下降)数据被采样SPI_InitABC.SPI_NSS=SPI_NSS_Soft;//NSS信号由硬件(NSS管脚)还是软件(使用SSI位)管理SPI_InitABC.SPI_BaudRatePrescaler=SPI_BaudRatePrescaler_256;//定义波特率预分频的值SPI_InitABC.SPI_FirstBit=SPI_FirstBit_MSB;//指定数据传输从MSB位开始SPI_InitABC.SPI_CRCPolynomial=7;//CRC值计算的多项式SPI_Init(SPI1,&SPI_InitABC);
}
使能SPIx
void SPI_Init(void)
{
SPI_Cmd(SPI1,ENABLE);
SPI1_ReadWriteByte(0xff);
}
//设置SPI1速度
void SPI1_SetSpeed(u8 SPI_BaudRatePrescaler)
{assert_param(IS_SPI_BAUDRATEPRESCALER(SPI_BauudRatePrescaler));//判断有效性SPI1->CR1&=0XFFC7;//位3~5清零,用来设置波特率SPI1->CR1|=SPI_BaudRatePrescaler;//设置SPI1速度SPI_Cmd(SPI1,ENABLE);//使能SPI1
}
SPI传输数据
u8 SPI1_ReadEriteByte(u8 TxData)
{while(SPI_I2S_GetFlagStatus(SPI1,SPI_I2S_FLAG_TXE)==RESET){}//等待发送区空SPI_I2S_SendData(SPI1,TxData);//通过外设SPIx发送一个byte数据while(SPI_I2S_GetFlagStatus(SPI1_I2S_FLAG_RXNE)==RESET){}//等待接收完一个bytereturn SPI_I2S_ReceiveData(SPI1);//返回通过SPIx最近接收的数据
}
程序思路:
写一个地址之前需要先判断是不是被擦除了(比如擦除之后的值为100,则需要将内容与100相比较从而判断是否擦除了),若未擦除则需要擦除,FLASH的最小单位是4K。
W25QXX_Write函数思路:每个sector是4K(等于4096个地址),在写任何一个地址前,如果该地址的值不是0xFF,必须先擦除对应的sector然后再写。
擦除的原理:将整个sector读出来放到一个4k的Buffer中,然后将sector擦除,在Buffer中更新数据,然后将Buffer中的数据写到sector中。(写的数据未超越一个sector)如果写的数据超过一个sector,处理方式相同。
相关代码:
void W25QXX_Write(u8* pBuffer,u32 WriteAddr,u16 NumByteToWrite)
{u32 secpos;u16 secoff;u16 i;u16 secremain;u8* W25QXX_BUF;W25QXX_BUF=W25QXX_BUFFER;secpos=WriteAddr/4096;//确定扇区地址(也就是sector的地址)secoff=WriteAddr%4096;//判断在扇区的偏移(判断sector中数据的位置,是否跨越一个sector)secremain=4096-secoff;//扇区剩余空间(sector剩余空间)if(NmByteToWrite<=secremain) secremain=NumByteToWrite;//如果写入的数据小于剩余的空间,就把secremain设置为写入的数据大小,这样只要写到了secremain就会终止。while(1)//先把扇区的数据读取出来{W25QXX_Read(W25QXX_BUF,secpos*4096,4096);//读出整个扇区的内容for(i=1;i<secremain;i++)//校验数据,是否擦除{if(W25QXX_BUF[secoff+i]!=oXFF) break;//需要擦除} if(i<Secremain)//{W25QXX_Erase_Sector(secpos);//擦除扇区for(i=0;i<secremain;i++){W25QXX_BUF[i+secoff]=pBuffer[i];}W25QXX_Write_NoCheck(W25QXX_BUF,secpos*4096,4096);//写入整个扇区}else W25QXX_Write_NoCheck(pBuffer,WriteAddr,secremain);//写已经擦除了的,直接写入扇区剩余区间。if(NumByteToWrite==secremain) break;//写入结束了else{secpos++;//扇区地址递增1secoff=0;//偏移位置为0pBuffer+=secremain;//指针偏移WriteAddr+=secremain;//写地址偏移NumByteToWrite-=secremain;//字节数递减if(NumByteToWrite>4096) secremain=4096;//下一个扇区还是没写完else secremain=NumByteToWrite;//下一个扇区可以写完了}}
}
STM32——SPI通信实验相关推荐
- SPI通信实验---verilog(FPGA作为从机,使用可读可写)
文章原地址: http://www.cnblogs.com/fhyfhy/p/4429302.html SPI通信实验---verilog(FPGA作为从机,使用可读可写) 本实验讲究实用性,故设计思 ...
- 通俗理解STM32 SPI通信(主从双机SPI通信)
STM32 SPI通信 高速全双工的通信总线 SPI 通讯使用 3 条总线及片选线,3 条总线分别为 SCK.MOSI.MISO,片选线为NSS(CS) NSS 信号线由高变低 ,是 SPI 通讯的起 ...
- STM32——串口通信实验
STM32串口通信实验:字符.字符串.数组 准备工作 编程逻辑顺序 初始化结构体 1.USART初始化结构体: 2.同步时钟初始化结构体: 源码 准备工作 1.找一个空白文件,新建usart.c 和u ...
- Zigbee(cc2530)和STM32 SPI通信,温度采集,组网
实验板子: 1. zigbee底板3块(1终端.1路由.1协调器,均基于CC2530) 2. stm32核心板(采集温度) 说明:下面的程序都是在TI官方例程SampleApp工程下面进行的更改.Z- ...
- STM32 SPI通信(读写flash)
SPI通信 四线制 MISO:主设备数据输入,从设备数据输出 MOSI:主设备数据输出,从设备数据输入 SCLK:时钟信号 CS:片选引脚(低电平有效) 时序图 读写程序 flash读写函数: u8 ...
- STM32 RS232通信实验
stm32F103 RS232通信实验 什么是RS232? 软件设计 完整工程下载 什么是RS232? 先来看看UART传输所存在的问题 于是就有了RS232协议 这里注意使用的是负逻辑电平信号,在规 ...
- STM32——SPI通信
文章目录 一.SPI通信 二.硬件电路 三.移位示意图 四.SPI时序基本单元 交换一个字节(模式0)[用的多] 交换一个字节(模式1) 交换一个字节(模式2) 交换一个字节(模式3) 五.SPI时序 ...
- STM32——485通信实验
一.串口 串口是一种接口标准,它规定了接口的电气标准,简单说只是物理层的一个标准.没有规定接口插件电缆以及使用的协议,所以只要我们使用的接口插件电缆符合串口标准就可以在实际中灵活使用,在串口接口标准上 ...
- Spi通信,写操作时,发送缓冲区非空停留在while里面。arm a5与w5500芯片udp网络通信。网络调试助手界面及sscom无法点击
问题描述 w5500 与SAMA5D3x处理器通过硬件spi实现网络通信.采用w5500自带的函数进行udp通信,接口函数sendto,发送100Hz的数据一段时间后,程序停留在函数SPI_write ...
最新文章
- 通俗易懂的ReentrantLock,不懂你来砍我
- Linux_IPtables防火墙详解
- [原] XAF How to Edit multiple objects in a ListViewAndDetailView
- Python编程基础:第四十节 类变量Class Variables
- Oracle学习(十四)分表分区
- (软件工程)用例说明模板
- 弹性伸缩Auto Scaling产品全面升级,轻松应对业务负载变化,张弛有度,收放自如!...
- 1128. 等价多米诺骨牌对的数量
- UUID生成唯一激活码
- java里的sqlite在哪_设备上sqlite数据库的位置
- ubuntu14.04中卸载cuda、cudnn
- 36. BOM (2)
- JS 字符串变量全局替换
- word去掉标题前面的黑点
- ADS仿真功率放大器模型导入报错问题解决
- linux 批量convert,使用convert来批量处理图片
- 转让天嵌2440开发板arm9
- Twitter开发者账号【推特开发者文档系列10】——API参考索引
- 《英语修辞与写作(修订版)》黄任(编著)epub+mobi+azw3格式下载
- Linux的tty架构及UART驱动详解
热门文章
- session 是什么
- 防火墙区域及工作模式
- 给定平面上任意三个点的坐标(x​1​​,y​1​​)、(x​2​​,y​2​​)、(x​3​​,y​3​​),检验它们能否构成三角形
- 解决不同版本jar包引入问题,修改jar包包名
- 孩子用什么样的灯对眼睛没有伤害呢?2023这样的灯光更护眼
- cmd黑客入侵命令大全
- Numpy删除指定行
- 小米路由器3G R3G 刷入Breed和OpenWrt 20.02.2 的记录
- 输入一无符号整数,求所有数位上数字之和
- Wake-on-LAN(远端唤醒) 原理及实现