STM32第十八课:SPIFlash

目录

  • 需求
  • 一、SPI概要
  • 二、SPI配置
    • 1.开时钟
    • 2.配置IO
    • 3.配置&使能SPI
  • 三、FLash操作函数
    • 1.SPI发送数据
    • 2.FLASH写使能
    • 3.FLASH等待操作完成
    • 4.FLASH页写操作
    • 5.FLASH读操作
    • 6.FLASH扇区擦除
  • 四、需求实现


需求

通过SPI控制FLash进行数据的保存和删除。
在这里插入图片描述


一、SPI概要

在我们使用UART(通用串行异步通讯协议)时,因为UART没有时钟信号,速度不同,无法控制何时收发数据
非要解决这个问题的话,需要为UART传输的数据添加起始位停止位,双方波特率还需同步,很麻烦,比较垃圾。
于是由摩托摩拉公司牵头推出了一种新的通讯总线 SPI:串行外设接口
SPI是一个高速、全双工、同步的串行通信总线。应用场景:OLED屏幕、FLASH存储器、AD转换器
通信方式:串行同步全双工(人话就是数据在线上按照时间顺序一位一位的传输,发送和接收时要在通信时钟的同步下进行数据传输,且可以同时发送和接收)
该总线就是利用单独的数据线(MISO和MOSI)和单独的时钟信号线(SCK)完美解决了收发端的数据完美同步。
接线
MOSI: 主设备输出、从设备输入 TX
MISO: 主设备输入、从设备输出 RX
SCK: 时钟线
CS: 片选信号线 一主多从,拉低选择和哪个从机通信
GND: 地线
单从机时:
在这里插入图片描述
多从机时:
在这里插入图片描述
总结:除了SS片选线需要单独备一根线和从机进行一对一的链接,其他都可以一对多,多个从机接同一个主机接口。
SPI通信模式:环形传输,发多少收多少。
操作逻辑:
1、用时先拉低CS。
2、开始操作发数据,收数据。
3、用完后拉高CS。

STM32中SPI接口
在这里插入图片描述

二、SPI配置

1.开时钟

打开原理图,找到引脚
在这里插入图片描述

在这里插入图片描述
根据引脚的开SPI2和GPIOB的时钟

	//1,开时钟,GPIO SPI2 RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB,ENABLE);RCC_APB1PeriphClockCmd(RCC_APB1Periph_SPI2,ENABLE);

2.配置IO

MOSI,SCK:复用推挽输出
均需要输出高低电平,还是复用功能,所以配复用推挽输出
MISO:浮空输入
需要接收数据,没什么好说的
CS:通用推挽输出
由于是主机所以CS当做GPIO配就行(软件模式)
从机的话配硬件模式

	//2,配置IO模式GPIO_InitStruct.GPIO_Mode = GPIO_Mode_Out_PP;GPIO_InitStruct.GPIO_Pin =GPIO_Pin_12;//CSGPIO_InitStruct.GPIO_Speed = GPIO_Speed_50MHz;GPIO_Init(GPIOB,&GPIO_InitStruct);GPIO_InitStruct.GPIO_Mode = GPIO_Mode_AF_PP;GPIO_InitStruct.GPIO_Pin =GPIO_Pin_13|GPIO_Pin_15;//SCK和MOSIGPIO_InitStruct.GPIO_Speed = GPIO_Speed_50MHz;GPIO_Init(GPIOB,&GPIO_InitStruct);	GPIO_InitStruct.GPIO_Mode = GPIO_Mode_IN_FLOATING;GPIO_InitStruct.GPIO_Pin =GPIO_Pin_14;//MISOGPIO_Init(GPIOB,&GPIO_InitStruct);	GPIO_SetBits(GPIOB,GPIO_Pin_12);//拉高CS避免通信出错

最后拉高CS是因为当CS拉低时表示开始传输信号

3.配置&使能SPI


参考手册中的步骤如下:

在这里插入图片描述
CPHA:时钟相位, CPHA=0时,在时钟的第一个边沿进行采样,第二个边沿进行输出
              CPHA=1时,在时钟的第二个边沿进行采样,第三个边沿进行输出
CPOL:时钟极性, CPOL=0时,空闲时时钟为低电平
              CPOL=1时,空闲时时钟为高电平

SPI的模式0CPHA=0,CPOL=0 上升沿采样、下降沿接受
SPI的模式1CPHA=0,CPOL=1 下降沿采样、上升沿接受
SPI的模式2CPHA=1,CPOL=0 下降沿采样、上升沿接受
SPI的模式3CPHA=1,CPOL=1 上升沿采样、下降沿接受

  一般能够支持SPI模式0的设备也支持SPI模式3,支持模式1的设备也支持模式2。SPI通信没有具体的协议格式,格式根据通信对象的要求来定。一般的传输数据时候采用的就是8位进行,高位先传还是低位先传,传输的速度。这些参数都需要根据通信对象来进行配置

	//2,配置SPISPI_InitStruct.SPI_BaudRatePrescaler=SPI_BaudRatePrescaler_2;//时钟分频2SPI_InitStruct.SPI_CPHA=SPI_CPHA_1Edge;//时钟相位,第一个边沿采样SPI_InitStruct.SPI_CPOL=SPI_CPOL_Low;//时钟极性 ,低电平SPI_InitStruct.SPI_CRCPolynomial=0x12;//不使用CRC校验,参数无意义SPI_InitStruct.SPI_DataSize=SPI_DataSize_8b;//数据宽度8未SPI_InitStruct.SPI_Direction=SPI_Direction_2Lines_FullDuplex;//双线全双工模式SPI_InitStruct.SPI_FirstBit=SPI_FirstBit_MSB;//先发高位SPI_InitStruct.SPI_Mode=SPI_Mode_Master;//主机模式SPI_InitStruct.SPI_NSS=SPI_NSS_Soft;//软件NSS模式SPI_Init(SPI2,&SPI_InitStruct);//3,使能SPISPI_Cmd(SPI2,ENABLE);

本此SPI配置的对象是FLash,需求如下:
在这里插入图片描述

三、FLash操作函数

  本次用到的FLash为W25Q64,是华邦存储推出的一款串行FLASH存储芯片。
空间规格:
64Mbit (兆位)== 8Mbyte(兆字节)
256字节为一页、16页为一扇区、16扇区为一块
在这里插入图片描述

地址表示:
在这里插入图片描述

1.SPI发送数据

由于SPI是环形传输,所以发多少,我们也要接收多少。
在这里插入图片描述
发送:判断发送缓冲器空闲标志(TXE)是否为0,为0表示正在发,此时持续等待数据发完。标志为1表示之前的发送完了,空闲下来了。此时就可以开始往DR发数据。
接收:判断接收缓冲器非空(RXNE)是否为0,为0表示接受缓冲区为空,没数据,此时持续等待数据到来。标志为1表示数据来了,此时就可以开始读DR的数据了。

/******************************************************************函 数 名 称:SPI2_SendData*函 数 功 能:SPI2发送数据*函 数 形 参:SPI2_SendData 发送内容*函 数 返 回:SPI2_RecvData 返回收到的数据*作       者:ZHT*修 改 日 期:xx\xx\xx*******************************************************************/
uint8_t SPI2_SendData(uint8_t SPI2_SendData)
{uint8_t SPI2_RecvData = 0;while((SPI2->SR & (0x1<<1)) == 0);SPI2->DR = SPI2_SendData;while((SPI2->SR & (0x1<<0)) == 0);SPI2_RecvData = SPI2->DR;return SPI2_RecvData;
}

2.FLASH写使能

看手册上的时序:
在这里插入图片描述
得知逻辑:
1.先拉低CS。
2.SPI发出06指令。
3.最后拉高CS。

/******************************************************************函 数 名 称:Write_Enable*函 数 功 能:FLASH写使能操作*函 数 形 参:无*函 数 返 回:无*作       者:ZHT*修 改 日 期:xx/xx/xx
*******************************************************************/
void Write_Enable(void)
{//拉低CSGPIO_ResetBits(GPIOB,GPIO_Pin_12);//发送0x06指令SPI2_SendData(0x06);//拉高CSGPIO_SetBits(GPIOB,GPIO_Pin_12);	
}

3.FLASH等待操作完成

看手册上的时序:
在这里插入图片描述

在这里插入图片描述
得知逻辑:
1.先拉低CS。
2.发送05或35指令。
3.循环往FLash发送数据(无所谓什么数据,只是为了置换出FLash状态寄存器的值),判断寄存器的第0位BUSY是否为0,为0时就代表Flash为空闲状态,可以执行其他操作。
4.判断结束就拉高CS。

/******************************************************************函 数 名 称:Flash_WaitForWriteEnd*函 数 功 能:FLASH等待操作完成*函 数 形 参:无*函 数 返 回:无*作       者:ZHT*修 改 日 期:xx/xx/xx
*******************************************************************/
void Flash_WaitForWriteEnd(void)
{uint8_t recv = 0;//拉低CSGPIO_ResetBits(GPIOB,GPIO_Pin_12);SPI2_SendData(0x05);//发送读取指令do{recv = SPI2_SendData(0x55);//循环读取寄存器数据,0x55随便改}while((recv & 0x01) == SET);//判断寄存器的第0位是否是1//拉高CSGPIO_SetBits(GPIOB,GPIO_Pin_12);	
}

4.FLASH页写操作

顾名思义,就是往FLash中写入一页(256Byte)的操作。
手册上的时序:
在这里插入图片描述
得出逻辑:
1.由于要进行写操作,所以要先进行写使能。
2.拉低CS,发送02指令。
3.发送三个字节,即24位的地址,每次发8位分3次发送。为了告知写入的位置。
4.发送数据,8位8位发,最多256。
5.调用FLash等待函数,等待写入完成。
6.写入完成后,拉高CS。

/******************************************************************函 数 名 称:Flash_WritePage*函 数 功 能:FLASH页写操作*函 数 形 参:Write_Local:写入地址 Write_data:写入数据 Write_len:写入数据长度*函 数 返 回:无*作       者:ZHT*修 改 日 期:xx/xx/xx
*******************************************************************/
void Flash_WritePage(uint8_t *Write_data,uint32_t Write_Local,uint16_t Write_len)
{//1,先写使能Write_Enable();//2,拉低CSGPIO_ResetBits(GPIOB,GPIO_Pin_12);//3,发送命令SPI2_SendData(0x02);//4,发送三个字节地址 add = 0x123456SPI2_SendData((Write_Local&0xFF0000)>> 16);SPI2_SendData((Write_Local&0x00FF00) >> 8);SPI2_SendData((Write_Local&0x0000FF));while(Write_len--){SPI2_SendData( *Write_data);Write_data++;}//5,拉高CSGPIO_SetBits(GPIOB,GPIO_Pin_12);//6,等待操作完成Flash_WaitForWriteEnd();
}

5.FLASH读操作

手册上的时序:
在这里插入图片描述
得出逻辑:
1.拉低CS,发送03指令。
2.发送3字节读取的地址。
3.随便发送1个字节的数据,返回值就是要获取的数据。
4.拉高CS。(千万不要等待操作完成,因为手册里没写)

/******************************************************************函 数 名 称:Flash_ReadData*函 数 功 能:FLASH读操作*函 数 形 参:add:读取地址 data:保存读取数据 len:读取数据长度*函 数 返 回:无*作       者:ZHT*修 改 日 期:xx/xx/xx
*******************************************************************/
void Flash_ReadData(uint32_t add, uint8_t *data,uint16_t len)
{//1,拉低CSGPIO_ResetBits(GPIOB,GPIO_Pin_12);	//2,发送命令SPI2_SendData(0x03);	//3,发送三个字节地址 add = 0x123456SPI2_SendData((add&0xFF0000)>> 16);SPI2_SendData((add&0x00FF00) >> 8);SPI2_SendData((add&0x0000FF));while(len--){*data = SPI2_SendData(0x55);data++;}//4,拉高CSGPIO_SetBits(GPIOB,GPIO_Pin_12);	
}

6.FLASH扇区擦除

看时序图
在这里插入图片描述
逻辑:
1.擦除操作实际上也是写入操作,是往Flash中写入0xFF。所以此时也要先开写使能。
2.拉低CS,发送指令20。
3.发送三个字节。
4.拉高电平,等待操作完成。

/******************************************************************函 数 名 称:Flash_SectorErase*函 数 功 能:FLASH扇区擦除*函 数 形 参:add:读取地址*函 数 返 回:无*作       者:CYM*修 改 日 期:xx/xx/xx
*******************************************************************/
void Flash_SectorErase(uint32_t add)
{//先写使能Write_Enable();//1,拉低CSGPIO_ResetBits(GPIOB,GPIO_Pin_12);	//2,发送命令SPI2_SendData(0x20);	//3,发送三个字节地址 add = 0x123456SPI2_SendData(add >> 16);SPI2_SendData((add&0x00FF00) >> 8);SPI2_SendData((add&0x0000FF));	//4,拉高CSGPIO_SetBits(GPIOB,GPIO_Pin_12);		//5,等待忙标志变0Flash_WaitForWriteEnd();
}

四、需求实现

main.c

#include "stm32f10x.h"
#include "usart.h"
#include "stdio.h"
#include "delay.h"
#include "string.h"
#include "pwm.h"
#include "adc.h"
#include "su03t.h"
#include "dht11.h"
#include "kqm.h"
#include "key.h"
#include "RTC.h"
#include "bsp_lcd.h"
#include "wifi.h"
#include "aliot.h"
#include "time.h"
#include "spi.h"
#define TestAddr 0x000700
char buff1[20]; 
int main()
{NVIC_SetPriorityGrouping(5);//两位抢占两位次级Usart1_Config(); SysTick_Config(72000);SPI2_Config();Flash_ReadID();Flash_SectorErase(TestAddr);FLASH_WriteBuffer((u8*)"好饿好饿好饿~",TestAddr,300);printf("写数据后\r\n");Flash_ReadData(TestAddr,buff1,300);printf("FLASH:%s\r\n",buff1);Flash_SectorErase(TestAddr);printf("清数据后\r\n");Flash_ReadData(TestAddr,buff1,300);printf("FLASH:%s\r\n",buff1);FLASH_WriteBuffer((u8*)"好饿好饿好饿~",TestAddr,300);printf("重写数据后\r\n");Flash_ReadData(TestAddr,buff1,300);printf("FLASH:%s\r\n",buff1);while(1){	if(ledcnt[0]>=ledcnt[1])//2S一次{ledcnt[0]=0;printf("FLASH:%s\r\n",buff1);}}return 0;
}

spi.c

#include "spi.h"
#define sFLASH_SPI_PAGESIZE       0x100
/******************************************************************函 数 名 称:SPI2_Config*函 数 功 能:初始化SPI2*函 数 形 参:无*函 数 返 回:无*作       者:ZHT*修 改 日 期:xx\xx\xx*******************************************************************/
void SPI2_Config()
{
#if 0//1,开时钟,GPIO SPI2  代码简短,但不直观RCC->APB2ENR |= 1<<3;RCC->APB1ENR |= 1<<14;//2,配置IO模式GPIOB->CRH &= ~(0xFFFF<<16);GPIOB->CRH |= 0xB4B3<<16;//配置4个IO口的模式 1011GPIOB->ODR |= 0x1<<12;//拉高PB12,避免对通信产生影响//3,配置SPISPI2->CR1 |= 0x0244;//配置SPI工作模式#elseGPIO_InitTypeDef GPIO_InitStruct={0};SPI_InitTypeDef SPI_InitStruct={0};//1,开时钟,GPIO SPI2 RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB,ENABLE);RCC_APB1PeriphClockCmd(RCC_APB1Periph_SPI2,ENABLE);//2,配置IO模式GPIO_InitStruct.GPIO_Mode = GPIO_Mode_Out_PP;GPIO_InitStruct.GPIO_Pin =GPIO_Pin_12;//CSGPIO_InitStruct.GPIO_Speed = GPIO_Speed_50MHz;GPIO_Init(GPIOB,&GPIO_InitStruct);GPIO_InitStruct.GPIO_Mode = GPIO_Mode_AF_PP;GPIO_InitStruct.GPIO_Pin =GPIO_Pin_13|GPIO_Pin_15;//SCK和MOSIGPIO_InitStruct.GPIO_Speed = GPIO_Speed_50MHz;GPIO_Init(GPIOB,&GPIO_InitStruct);	GPIO_InitStruct.GPIO_Mode = GPIO_Mode_IN_FLOATING;GPIO_InitStruct.GPIO_Pin =GPIO_Pin_14;//MISOGPIO_Init(GPIOB,&GPIO_InitStruct);	GPIO_SetBits(GPIOB,GPIO_Pin_12);//拉高CS避免通信出错//2,配置SPISPI_InitStruct.SPI_BaudRatePrescaler=SPI_BaudRatePrescaler_2;//时钟分频2SPI_InitStruct.SPI_CPHA=SPI_CPHA_1Edge;//时钟相位,第一个边沿采样SPI_InitStruct.SPI_CPOL=SPI_CPOL_Low;//时钟极性 ,低电平SPI_InitStruct.SPI_CRCPolynomial=0x12;//不使用CRC校验,参数无意义SPI_InitStruct.SPI_DataSize=SPI_DataSize_8b;//数据宽度8未SPI_InitStruct.SPI_Direction=SPI_Direction_2Lines_FullDuplex;//双线全双工模式SPI_InitStruct.SPI_FirstBit=SPI_FirstBit_MSB;//先发高位SPI_InitStruct.SPI_Mode=SPI_Mode_Master;//主机模式SPI_InitStruct.SPI_NSS=SPI_NSS_Soft;//软件NSS模式SPI_Init(SPI2,&SPI_InitStruct);//3,使能SPISPI_Cmd(SPI2,ENABLE);#endif
}/******************************************************************函 数 名 称:SPI2_SendData*函 数 功 能:SPI2发送数据*函 数 形 参:SPI2_SendData 发送内容*函 数 返 回:SPI2_RecvData 返回收到的数据*作       者:ZHT*修 改 日 期:xx\xx\xx*******************************************************************/
uint8_t SPI2_SendData(uint8_t SPI2_SendData)
{uint8_t SPI2_RecvData = 0;while((SPI2->SR & (0x1<<1)) == 0);SPI2->DR = SPI2_SendData;while((SPI2->SR & (0x1<<0)) == 0);SPI2_RecvData = SPI2->DR;return SPI2_RecvData;
}/************************SPI-FLASH********************************/
/******************************************************************函 数 名 称:Write_Enable*函 数 功 能:FLASH写使能操作*函 数 形 参:无*函 数 返 回:无*作       者:ZHT*修 改 日 期:xx/xx/xx
*******************************************************************/
void Write_Enable(void)
{//拉低CSGPIO_ResetBits(GPIOB,GPIO_Pin_12);//发送0x06指令SPI2_SendData(0x06);//拉高CSGPIO_SetBits(GPIOB,GPIO_Pin_12);	
}/******************************************************************函 数 名 称:Flash_WaitForWriteEnd*函 数 功 能:FLASH等待操作完成*函 数 形 参:无*函 数 返 回:无*作       者:CYM*修 改 日 期:xx/xx/xx
*******************************************************************/
void Flash_WaitForWriteEnd(void)
{uint8_t recv = 0;//拉低CSGPIO_ResetBits(GPIOB,GPIO_Pin_12);SPI2_SendData(0x05);//发送读取指令do{recv = SPI2_SendData(0x55);//循环读取寄存器数据,0x55随便改}while((recv & 0x01) == SET);//判断寄存器的第0位是否是1//拉高CSGPIO_SetBits(GPIOB,GPIO_Pin_12);	
}/******************************************************************函 数 名 称:Flash_WritePage*函 数 功 能:FLASH页写操作*函 数 形 参:Write_Local:写入地址 Write_data:写入数据 Write_len:写入数据长度*函 数 返 回:无*作       者:CYM*修 改 日 期:xx/xx/xx
*******************************************************************/
void Flash_WritePage(uint8_t *Write_data,uint32_t Write_Local,uint16_t Write_len)
{//1,先写使能Write_Enable();//2,拉低CSGPIO_ResetBits(GPIOB,GPIO_Pin_12);//3,发送命令SPI2_SendData(0x02);//4,发送三个字节地址 add = 0x123456SPI2_SendData((Write_Local&0xFF0000)>> 16);SPI2_SendData((Write_Local&0x00FF00) >> 8);SPI2_SendData((Write_Local&0x0000FF));while(Write_len--){SPI2_SendData( *Write_data);Write_data++;}//5,拉高CSGPIO_SetBits(GPIOB,GPIO_Pin_12);//6,等待操作完成Flash_WaitForWriteEnd();
}/******************************************************************函 数 名 称:Flash_ReadData*函 数 功 能:FLASH读操作*函 数 形 参:add:读取地址 data:保存读取数据 len:读取数据长度*函 数 返 回:无*作       者:CYM*修 改 日 期:xx/xx/xx
*******************************************************************/
void Flash_ReadData(uint32_t add, uint8_t *data,uint16_t len)
{//1,拉低CSGPIO_ResetBits(GPIOB,GPIO_Pin_12);	//2,发送命令SPI2_SendData(0x03);	//3,发送三个字节地址 add = 0x123456SPI2_SendData((add&0xFF0000)>> 16);SPI2_SendData((add&0x00FF00) >> 8);SPI2_SendData((add&0x0000FF));while(len--){*data = SPI2_SendData(0x55);data++;}//4,拉高CSGPIO_SetBits(GPIOB,GPIO_Pin_12);	
}
/******************************************************************函 数 名 称:Flash_SectorErase*函 数 功 能:FLASH扇区擦除*函 数 形 参:add:读取地址*函 数 返 回:无*作       者:CYM*修 改 日 期:xx/xx/xx
*******************************************************************/
void Flash_SectorErase(uint32_t add)
{//先写使能Write_Enable();//1,拉低CSGPIO_ResetBits(GPIOB,GPIO_Pin_12);	//2,发送命令SPI2_SendData(0x20);	//3,发送三个字节地址 add = 0x123456SPI2_SendData(add >> 16);SPI2_SendData((add&0x00FF00) >> 8);SPI2_SendData((add&0x0000FF));	//4,拉高CSGPIO_SetBits(GPIOB,GPIO_Pin_12);		//5,等待忙标志变0Flash_WaitForWriteEnd();
}
/******************************************************************函 数 名 称:Flash_ReadID*函 数 功 能:FLASH读取ID*函 数 形 参:无*函 数 返 回:无*作       者:ZHT*修 改 日 期:xx/xx/xx
*******************************************************************/
void Flash_ReadID(void)
{uint32_t Temp = 0, Temp0 = 0, Temp1 = 0, Temp2 = 0;//1,拉低CSGPIO_ResetBits(GPIOB,GPIO_Pin_12);	SPI2_SendData(0x9F);Temp0 = SPI2_SendData(0x55);Temp1 = SPI2_SendData(0x55);Temp2 = SPI2_SendData(0x55);GPIO_SetBits(GPIOB,GPIO_Pin_12);	Temp = (Temp0 << 16) | (Temp1 << 8) | Temp2;printf("FLASH_ID:%x\r\n",Temp);
}/******************************************************************函 数 名 称:Flash_ChipClean*函 数 功 能:全片擦除*函 数 形 参:无*函 数 返 回:无*作       者:ZHT*修 改 日 期:xx\xx\xx*******************************************************************/
void Flash_ChipClean()
{//先写使能Write_Enable();//1,拉低CSGPIO_ResetBits(GPIOB,GPIO_Pin_12);	//2,发送命令SPI2_SendData(0x60);	//4,拉高CSGPIO_SetBits(GPIOB,GPIO_Pin_12);	Flash_WaitForWriteEnd();
}/******************************************************************函 数 名 称:FLASH_WriteBuffer*函 数 功 能:随意写入*函 数 形 参:uint8_t* pBuffer:指向要写入数据的缓冲区的指针。uint32_t WriteAddr:写入数据的起始地址。uint16_t NumByteToWrite:要写入的字节数。*函 数 返 回:无*作       者:ZHT*修 改 日 期:xx\xx\xx*******************************************************************/void FLASH_WriteBuffer(uint8_t* pBuffer, uint32_t WriteAddr, uint16_t NumByteToWrite)
{uint8_t NumOfPage = 0,NumOfSingle = 0,Addr = 0,count = 0,temp = 0;
//       整页的数量      不足一页的剩余字节数  起始地址偏移		 当前页剩余可写入的字节数	 临时变量,用于存储超出当前页的字节数Addr = WriteAddr % 256;count = 0x100 - Addr;NumOfPage =  NumByteToWrite / 256;NumOfSingle = NumByteToWrite % 256;if (Addr == 0) /*!< WriteAddr is sFLASH_PAGESIZE aligned  */{if (NumOfPage == 0) /*!< NumByteToWrite < sFLASH_PAGESIZE */{Flash_WritePage(pBuffer, WriteAddr, NumByteToWrite);}else /*!< NumByteToWrite > sFLASH_PAGESIZE */{while (NumOfPage--){Flash_WritePage(pBuffer, WriteAddr, 256);WriteAddr +=  256;pBuffer += 256;}Flash_WritePage(pBuffer, WriteAddr, NumOfSingle);}}else /*!< WriteAddr is not sFLASH_PAGESIZE aligned  */{if (NumOfPage == 0) /*!< NumByteToWrite < sFLASH_PAGESIZE */{if (NumOfSingle > count) /*!< (NumByteToWrite + WriteAddr) > sFLASH_PAGESIZE */{temp = NumOfSingle - count;Flash_WritePage(pBuffer, WriteAddr, count);WriteAddr +=  count;pBuffer += count;Flash_WritePage(pBuffer, WriteAddr, temp);}else{Flash_WritePage(pBuffer, WriteAddr, NumByteToWrite);}}else /*!< NumByteToWrite > sFLASH_PAGESIZE */{NumByteToWrite -= count;NumOfPage =  NumByteToWrite / 256;NumOfSingle = NumByteToWrite % 256;Flash_WritePage(pBuffer, WriteAddr, count);WriteAddr +=  count;pBuffer += count;while (NumOfPage--){Flash_WritePage(pBuffer, WriteAddr, 256);WriteAddr +=  256;pBuffer += 256;}if (NumOfSingle != 0){Flash_WritePage(pBuffer, WriteAddr, NumOfSingle);}}}
}void sFLASH_WriteBuffer(uint8_t* pBuffer, uint32_t WriteAddr, uint16_t NumByteToWrite)
{uint8_t NumOfPage = 0, NumOfSingle = 0, Addr = 0, count = 0, temp = 0;Addr = WriteAddr % sFLASH_SPI_PAGESIZE;count = sFLASH_SPI_PAGESIZE - Addr;NumOfPage =  NumByteToWrite / sFLASH_SPI_PAGESIZE;NumOfSingle = NumByteToWrite % sFLASH_SPI_PAGESIZE;if (Addr == 0) /*!< WriteAddr is sFLASH_PAGESIZE aligned  */{if (NumOfPage == 0) /*!< NumByteToWrite < sFLASH_PAGESIZE */{Flash_WritePage(pBuffer, WriteAddr, NumByteToWrite);}else /*!< NumByteToWrite > sFLASH_PAGESIZE */{while (NumOfPage--){Flash_WritePage(pBuffer, WriteAddr, sFLASH_SPI_PAGESIZE);WriteAddr +=  sFLASH_SPI_PAGESIZE;pBuffer += sFLASH_SPI_PAGESIZE;}Flash_WritePage(pBuffer, WriteAddr, NumOfSingle);}}else /*!< WriteAddr is not sFLASH_PAGESIZE aligned  */{if (NumOfPage == 0) /*!< NumByteToWrite < sFLASH_PAGESIZE */{if (NumOfSingle > count) /*!< (NumByteToWrite + WriteAddr) > sFLASH_PAGESIZE */{temp = NumOfSingle - count;Flash_WritePage(pBuffer, WriteAddr, count);WriteAddr +=  count;pBuffer += count;Flash_WritePage(pBuffer, WriteAddr, temp);}else{Flash_WritePage(pBuffer, WriteAddr, NumByteToWrite);}}else /*!< NumByteToWrite > sFLASH_PAGESIZE */{NumByteToWrite -= count;NumOfPage =  NumByteToWrite / sFLASH_SPI_PAGESIZE;NumOfSingle = NumByteToWrite % sFLASH_SPI_PAGESIZE;Flash_WritePage(pBuffer, WriteAddr, count);WriteAddr +=  count;pBuffer += count;while (NumOfPage--){Flash_WritePage(pBuffer, WriteAddr, sFLASH_SPI_PAGESIZE);WriteAddr +=  sFLASH_SPI_PAGESIZE;pBuffer += sFLASH_SPI_PAGESIZE;}if (NumOfSingle != 0){Flash_WritePage(pBuffer, WriteAddr, NumOfSingle);}}}
}

spi.h

#ifndef __SPI_H
#define __SPI_H#include "stm32f10x.h"
#include "stdio.h"void SPI2_Config();
void Flash_WritePage(uint8_t *Write_data,uint32_t Write_Local,uint16_t Write_len);
void Flash_ReadData(uint32_t add, uint8_t *data,uint16_t len);
void Flash_SectorErase(uint32_t add);
void Flash_ReadID(void);
void Flash_WaitForWriteEnd(void);
void Flash_ChipClean();
void FLASH_WriteBuffer(uint8_t* pBuffer, uint32_t WriteAddr, uint16_t NumByteToWrite);
void sFLASH_WriteBuffer(uint8_t* pBuffer, uint32_t WriteAddr, uint16_t NumByteToWrite);
#endif

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

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

相关文章

【python】OpenCV—European Article Number

参考学习来自&#xff1a;OpenCV基础&#xff08;25&#xff09;条码和二维码扫的生成与识别 1 条形码介绍 EAN-13是欧洲物品编码&#xff08;European Article Number&#xff09;的缩写&#xff0c;是一种广泛使用的条形码标准&#xff0c;特别是在超级市场和其它零售业中。…

OpenCV解决验证码(数字和字母)识别(Python)

文章目录 前言一、准备验证码图片 前言 OpenCV是一个基于Apache2.0许可&#xff08;开源&#xff09;发行的跨平台计算机视觉和机器学习软件库。它支持Windows、Linux、Mac OS、Android和iOS等多个操作系统&#xff0c;提供了丰富的图像处理和计算机视觉功能&#xff0c;包括但…

RPC与服务的注册发现

文章目录 1. 什么是远程过程调用(RPC)?2. RPC的流程3. RPC实践4. RPC与REST的区别4.1 RPC与REST的相似之处4.2 RPC与REST的架构原则4.3 RPC与REST的主要区别 5. RPC与服务发现5.1 以zookeeper为服务注册中心5.2 以etcd为服务注册中心 6. 小结参考 1. 什么是远程过程调用(RPC)?…

(自用)网络编程

OSI七层协议模型 (open system interconnection) 应用层————为应用数据提供服务 表示层————数据格式转化&#xff0c;数据加密 会话层————建立、维护和管理会话 传输层————建立、维护和管理端到端的链接&#xff0c;控制数据传输的方式 网络层————数据…

手机删除的文件能恢复吗?删除不等于永别,3个技巧助你找回

安卓手机中的文件&#xff0c;就像是数字世界里的繁星&#xff0c;记录着我们的点点滴滴。然而&#xff0c;有时我们可能会不小心删除了某些重要的文件&#xff0c;让我们感到惋惜和困惑。删除的文件能恢复吗&#xff1f;别担心&#xff0c;删除并不等于永别&#xff0c;我们也…

CentOS 停服后,服务器 OS 路在何方?

2024 年 6 月 30 日&#xff0c;CentOS Linux 7 终止其生命周期&#xff08;EOL&#xff09;&#xff0c;至此 CentOS 全系列版本也已停止维护&#xff0c;属于 CentOS 的时代彻底终结。CentOS 停止维护后&#xff0c;用户将无法获得包括问题修复和功能更新在内的任何软件维护和…

小程序里面使用vant ui中的vant-field组件,如何使得输入框自动获取焦点

//.wxml <van-fieldmodel:value"{{ userName }}"placeholder"请输入学号"focus"{{focusUserName}}"/>// .js this.setData({focusUserName: true});vant-field

MSVC2017+Qt 打包

在环境变量下配置好 QT 和 MSVC 的路径 相关搜索&#xff1a; 找不到msvcp140.dll 1.搜索 Qt 选择在编译器路径下打开 2. Windeployqt 生成打包&#xff0c;正常情况下生成 VC 相关package&#xff0c; 即 msvcp140.dll 等MSVC 相关 但是lz尝试没有生成 解决办法 先将生成…

数据库作业d8

要求&#xff1a; 一备份 1 mysqldump -u root -p booksDB > booksDB_all_tables.sql 2 mysqldump -u root -p booksDB books > booksDB_books_table.sql 3 mysqldump -u root -p --databases booksDB test > booksDB_and_test_databases.sql 4 mysql -u roo…

MySQL 中的几种锁

MySQL 中的锁 #按锁粒度如何划分? 按锁粒度划分的话&#xff0c;MySQL 的锁有&#xff1a; 表锁&#xff1a;开销小&#xff0c;加锁快&#xff1b;锁定力度大&#xff0c;发生锁冲突概率高&#xff0c;并发度最低;不会出现死锁。行锁&#xff1a;开销大&#xff0c;加锁慢…

电脑压缩软件哪个好?WinRAR、7-Zip、Bandizip 还是360压缩

文件压缩软件已成为我们日常工作中不可或缺的一部分&#xff0c;它不仅能够帮助我们节省存储空间&#xff0c;还能提高文件传输效率。本文简鹿办公小编将对四款主流的电脑压缩软件进行对比&#xff0c;它们分别是 WinRAR、7-Zip、Bandizip 和 360 压缩。 一、WinRAR WinRAR 是…

Qt中文个数奇数时出现问号解决

Qt中文个数奇数时出现问号解决 目录 Qt中文个数奇数时出现问号解决问题背景问题场景解决方案 问题背景 最近在开发一个小工具&#xff0c;涉及到一些中文注释自动打印&#xff0c;于是摸索如何把代码里面的中文输出到csv文件中&#xff0c;出现了乱码&#xff0c;按照网上的攻…

vue2-Django3-iframe解决方案,处理安全策略,事件拦截,处理iframe重载等

目录 简介 实现iframe 后端安全策略 通过Ngnix代理实现SAMEORIGIN iframe的事件拦截&#xff0c;自定义处理 iframe的状态保持&#xff08;解决vue中iframe重载&#xff09; 解决方法 简介 Iframe&#xff08;内联框架&#xff09;是一种HTML元素&#xff0c;用于在网页…

周鸿祎为什么建议Java、前端、大数据、PHP开发都要学一下大模型?_ai大模型全栈工程师跟java有关吗

ChatGPT的出现在全球掀起了AI大模型的浪潮&#xff0c;2023年可以被称为AI元年&#xff0c;AI大模型以一种野蛮的方式&#xff0c;闯入你我的生活之中。 从问答对话到辅助编程&#xff0c;从图画解析到自主创作&#xff0c;AI所展现出来的能力&#xff0c;超出了多数人的预料&…

智慧公厕系统实现人性化与节能化的完美结合

在当今社会&#xff0c;科技的飞速发展正不断改变着我们的生活方式&#xff0c;公厕也不例外。智慧公厕系统的出现&#xff0c;不仅提升了人们的使用体验&#xff0c;更实现了人性化与节能化的完美结合&#xff0c;为城市公共服务带来了全新的变革。 一、人性化&#xff0c;是智…

echarts 实现水利计算模型-雨量,流量,时间分割线

需求背景解决效果ISQQW代码地址index.vue 需求背景 实现水利计算模型-雨量&#xff0c;流量&#xff0c;时间分割线 解决效果 ISQQW代码地址 链接 index.vue <!--/** * author: liuk * date: 2024/06/13 * describe: 洪水预报结果图表 */--> <template><di…

算法篇 滑动窗口 leetcode 长度最小的子数组

长度最小的子数组 1. 题目描述2. 算法图分析2.1 暴力图解2.2 滑动窗口图解 3. 代码演示 1. 题目描述 2. 算法图分析 2.1 暴力图解 2.2 滑动窗口图解 3. 代码演示

数据结构进阶——使用数组实现栈和队列详解与示例(C,C#,C++)

文章目录 1、数组实现栈栈的基本操作C语言实现C#语言实现 2、 数组实现队列队列的基本操作C语言实现C# 语言实现C语言实现 总结 在编程世界中&#xff0c;数据结构是构建高效算法的基石。栈和队列作为两种基本的数据结构&#xff0c;它们的应用非常广泛。本文将带领大家使用C&a…

股票质押约定购回:机制、风险与策略!

​股票质押约定购回&#xff1a;机制、风险与策略 在复杂的金融市场中&#xff0c;股票质押约定购回作为一种常见的融资手段&#xff0c;受到了众多投资者和企业的关注。本文将深入探讨股票质押约定购回的定义、运作机制、潜在风险以及投资者和企业在操作时应采取的策略。 一、…

HackChat匿名聊天室

匿名聊天 聊天室地址 这是一款极简、无干扰的聊天应用程序&#xff0c;可以让你专注于交流而不必担心干扰. 频道通过 url 创建、加入和共享&#xff0c;通过更改问号后的文本来创建自己的频道. hack.chat 服务器上不会保留任何消息历史记录&#xff0c;链接断开消息就会删除. …