使用STM32的FLASH保存数据

使用STM32的FLASH保存数据

为了防止“掉电丢失数据”,我们最先想到的是EEPROM,但是若考虑到降低成本和PCB布线的空间,使用CPU内部的FLASH空间来保存数据,是最好的选择。尤其是在STM32芯片上,应用案例还是比较多的。

STM32的FLASH是以半字保存数据的,因此,我们可以将“掉电数据”以双字节为最小存储单位去进行访问。

STM32没有自带 的EEPROM,但是它具有 IAP功能,因此,我们可以把它的 FLASH 当成 EEPROM 来使用。

1、STM32_Internal_FLASH.h文件如下:

#ifndef _STM32_Internal_FLASH_H

#define _STM32_Internal_FLASH_H

#include "stm32f10x.h"

#include "sys.h"

#define STM32_FLASH_BASE_ADDRESS   ( (uint32_t)0x08000000 )

//STM32 FLASH的编程起始地址

#define STM32_FLASH_SIZE           512

//所选STM32FLASH容量大小(单位为K)

#if STM32_FLASH_SIZE<256                //CPUFLASH空间小于256K字节

#define STM_SECTOR_SIZE           1024  //每个扇区为1024个字节

#define STMFLASH_WriteLength_SIZE 512   //每个扇区为512个半字

#define STM32_FLASH_START_ADDRESS  ( (uint32_t)0x0800FC00 )

//数据保存到STM32F103C8T6最后一页,即第64,号码为63,页大小为1KB

#else

#define STM_SECTOR_SIZE              2048  //每个扇区为2048个字节

#define STMFLASH_WriteLength_SIZE 1024  //每个扇区为1024个半字

#define STM32_FLASH_START_ADDRESS  ( (uint32_t)0x0807F800 )

//数据保存到STM32F103ZET6最后一页,即第256,号码为255,页大小为2KB

#endif

union EEPROM_Uint16_Data_TYPE

{ unsigned char b[2];   //b[1]Data的高8位值相等;b[0]Data的低8位值相等;

  uint16_t Data;

};

union EEPROM_U32_Data_TYPE

{ u8 b[4]; //b[3]Data的高8位值相等;b[0]Data的低8位值相等;

  u32 Data;

};

union EEPROM_FLOAT_DATA_TYPE

{ u8 b[4];   //b[3]float_data的高8位值相等;b[0]float_data的低8位值相等;

  float float_data;

};

union EEPROM_DOUBLE_DATA_TYPE

{ u8 b[8];   //b[7]和float_data的高8位值相等;b[0]和float_data的低8位值相等;

  double float_data;

};

extern void STMFLASH_Read(u32 tReadAddr,u16 *pBuffer,u16 tSize);

extern void IAP_Write_APP_BIN(uint32_t gIAP_FlashAddress,u8 *pBuffer,u16 len);

extern void EEPROM_U16_Data_Write(u16 x,uint32_t addr);

extern u16 EEPROM_U16_Data_Read(uint32_t addr);

extern void EEPROM_U32_Data_Write(u32 x,uint32_t addr);

extern u32 EEPROM_U32_Data_Read(uint32_t addr);

extern void EEPROM_Float_Data_Write(float x,uint32_t addr);

extern float EEPROM_Float_Data_Read(uint32_t addr);

extern void EEPROM_Double_Data_Write(double x,uint32_t addr);

extern double EEPROM_Double_Data_Read(uint32_t addr);

extern void CPU_FLASH_Read_Write_Test(void);

#endif

2、STM32_Internal_FLASH.c文件如下:

#include "STM32_Internal_FLASH.h"

#include "stdio.h"

void STMFLASH_Read(u32 tReadAddr,u16 *pBuffer,u16 tSize);

void IAP_Write_APP_BIN(uint32_t gIAP_FlashAddress,u8 *pBuffer,u16 len);

void EEPROM_U16_Data_Write(u16 x,uint32_t addr);

u16 EEPROM_U16_Data_Read(uint32_t addr);

void EEPROM_U32_Data_Write(u32 x,uint32_t addr);

u32 EEPROM_U32_Data_Read(uint32_t addr);

void EEPROM_Float_Data_Write(float x,uint32_t addr);

float EEPROM_Float_Data_Read(uint32_t addr);

void EEPROM_Double_Data_Write(double x,uint32_t addr);

double EEPROM_Double_Data_Read(uint32_t addr);

void CPU_FLASH_Read_Write_Test(void);

//函数功能:CPUFLASH地址tReadAddr处读取半字(16位数据)

//tReadAddr:CPUFLASH地址(此地址必须为2的倍数!!)

//返回值:返回半字(16位数据)

u16 STMFLASH_ReadHalfWord(u32 tReadAddr)

{

         return *(vu16*)tReadAddr;

}

//CPUFLASH地址tReadAddr处读取tSize个半字(16位数据),保存到首地址为pBuffer的缓冲区中

//tReadAddr:起始地址

//pBuffer:数据指针

//tSize:半字(16)

void STMFLASH_Read(u32 tReadAddr,u16 *pBuffer,u16 tSize)      

{

         u16 i;

         for(i=0;i<tSize;i++)

         {

                   pBuffer[i]=STMFLASH_ReadHalfWord(tReadAddr);

                   //CPUFLASH地址tReadAddr处读取半字(16位数据)

                   tReadAddr=tReadAddr+2;//偏移2个字节.   

         }

}

//函数功能:STMFLASH_BUF[]中前STMFLASH_WriteLength个半字,FLASH起始地址为STMFLASH_WriteAddress开始处写入,在写之前不检查能否写入

//STMFLASH_WriteAddress:CPUFLASH起始地址

//STMFLASH_BUF:待写数据缓冲区的起始指针

//STMFLASH_WriteLength:需要写入"半字(16)"的个数

//STMFLASH_WriteLength;要写入的双字节数,待写数据在STMFLASH_BUF[]

//STMFLASH_BUF[STMFLASH_WriteLength_SIZE];//最多是2K字节,1K个双字节

void STMFLASH_Write_NoCheck(u32 STMFLASH_WriteAddress,u16 *STMFLASH_BUF,u16 STMFLASH_WriteLength)

{

         u16 k;

         for(k=0;k<STMFLASH_WriteLength;k++)

    //循环写入STMFLASH_WriteLength"半字"

         {

                   FLASH_ProgramHalfWord(STMFLASH_WriteAddress,STMFLASH_BUF[k]);

                   STMFLASH_WriteAddress=STMFLASH_WriteAddress+2;

       //CPUFLASH地址增加2,指向下一个地址

         }

}

const char IAP_FlashAddress1_REG[]="\r\nIAP_FlashAddress1=0x";

const char IAP_FlashAddress2_REG[]="\r\nIAP_FlashAddress2=0x";

//函数功能:pBuffer[]中前tSize个半字,FLASH起始地址为tWriteAddr开始处写入,在写之前检查能否写入

//IAP_FlashAddress:起始地址(此地址必须为2的倍数!!)

//pBuffer:数据指针,数据在IAP_Buffer[]

//tSize:装载烧写数据的数量,单位为半字,IAP_Buffer_Length一样

void STMFLASH_Write(uint32_t gIAP_FlashAddress,u16 *pBuffer,u16 tSize)

{

         u16 t;

         u16 i;

         u16 tmp;

         u8 flag;

         u16 STMFLASH_BUF[STMFLASH_WriteLength_SIZE];

    //最多是2K字节,1K个双字节

         u16 STMFLASH_WriteLength;

    //要写入的双字节数,待写数据在STMFLASH_BUF[]

         u32 STMFLASH_WriteAddress; //写地址

         u16 STMFLASH_BUF_Index;  //用来指示STMFLASH_BUF[]中待写数据的下标

         u16 STMFLASH_BUF_Load_Length;

    //用来指示STMFLASH_BUF[]中待装载的数据长度

         uint32_t gIAP_FlashSectorNumber; //扇区号从0开始

         uint32_t gIAP_FlashOffsetAddress;//其值为(IAP_FlashAddress-0X08000000)

(void)flag;

if( gIAP_FlashAddress<STM32_FLASH_BASE_ADDRESS||(gIAP_FlashAddress>=(STM32_FLASH_BASE_ADDRESS+1024*STM32_FLASH_SIZE)) )

                   return;//非法地址

         FLASH_Unlock();//解锁

         while(1)//循环写FLASH

         {

           gIAP_FlashOffsetAddress=gIAP_FlashAddress-STM32_FLASH_BASE_ADDRESS;

      //计算"待写数据"FLASH中的偏移地址

           gIAP_FlashSectorNumber=gIAP_FlashOffsetAddress/STM_SECTOR_SIZE;

      //计算"待写数据"FLASH中的扇区号码,0开始

           tmp=gIAP_FlashOffsetAddress%STM_SECTOR_SIZE;

      //计算"当前扇区"已经写了多少个字节

           STMFLASH_BUF_Index=tmp/2;//计算"当前扇区"已经写了多少个半字

           STMFLASH_BUF_Load_Length=STMFLASH_WriteLength_SIZE-STMFLASH_BUF_Index;

//计算"待写数据"的数量 

           if(tSize<=STMFLASH_BUF_Load_Length)

      //"待写数据的数量"没有越过当前扇区的范围

           {

                     flag=0;

                     STMFLASH_BUF_Load_Length=tSize;

           }

           else flag=1;

        

                   gIAP_FlashAddress=gIAP_FlashSectorNumber*STM_SECTOR_SIZE;

gIAP_FlashAddress=gIAP_FlashAddress+STM32_FLASH_BASE_ADDRESS;

//计算扇区的首地址

                   STMFLASH_Read(gIAP_FlashAddress,STMFLASH_BUF,STMFLASH_WriteLength_SIZE);//根据扇区首地址,读出整个扇区的内容

                   for(i=0;i<STMFLASH_BUF_Load_Length;i++)//检查"待写的区域"是否需要擦除

                   {

                            t=STMFLASH_BUF_Index+i;//计算"修改数据"的下标值

                            if(STMFLASH_BUF[t]!=0XFFFF)break;//需要擦除       

                   }

                   if(i<STMFLASH_BUF_Load_Length)//"待写的区域"需要擦除

                   {

                            FLASH_ErasePage(gIAP_FlashAddress);//根据扇区首地址,擦除这个扇区

                            for(i=0;i<STMFLASH_BUF_Load_Length;i++)//装载"待编程数据"

                            {

                                     t=STMFLASH_BUF_Index+i;

               //得到在STMFLASH_BUF[]中的下标偏移量

                                     STMFLASH_BUF[t]=pBuffer[i];

               //装载"待写数据"STMFLASH_BUF[]         

                            }

                            STMFLASH_WriteAddress=gIAP_FlashAddress;

            //"扇区首地址"作为"写起始地址"

                            STMFLASH_WriteLength=STMFLASH_BUF_Index+STMFLASH_BUF_Load_Length;

    //计算“写入半字的数量“

                            STMFLASH_Write_NoCheck(STMFLASH_WriteAddress,STMFLASH_BUF,STMFLASH_WriteLength);//写入整个扇区

                            tmp=STMFLASH_WriteLength;tmp=(u16)(tmp<<1);//计算已写字节数量

                            gIAP_FlashAddress=gIAP_FlashAddress+tmp;//记录写结束地址

             printf("%s",IAP_FlashAddress1_REG);

             printf("%x",gIAP_FlashAddress);                        

                   }

                   else//"待写的区域"不需要擦除

                   {

                            for(i=0;i<STMFLASH_BUF_Load_Length;i++)//装载"待编程数据"

                            {

                                     STMFLASH_BUF[i]=pBuffer[i];   

                            }

                            gIAP_FlashAddress=gIAP_FlashAddress+tmp;   //计算写起始地址

                            STMFLASH_WriteAddress=gIAP_FlashAddress;   //记录写地址

                            STMFLASH_WriteLength=STMFLASH_BUF_Load_Length;

            //计算“写入半字的数量“

                     STMFLASH_Write_NoCheck(STMFLASH_WriteAddress,STMFLASH_BUF,STMFLASH_WriteLength);

//写已经擦除了的,直接写入扇区剩余区间

                            tmp=STMFLASH_WriteLength;tmp=(u16)(tmp<<1);//计算已写字节数量

                            gIAP_FlashAddress=gIAP_FlashAddress+tmp;//记录写结束地址

                            printf("%s",IAP_FlashAddress2_REG);

             printf("%x",gIAP_FlashAddress);

                   }

                  

                   if(tSize==STMFLASH_BUF_Load_Length)

                   {

                            break;//写入结束了

                   }

                   else//写入未结束

                   {

                            gIAP_FlashSectorNumber++;//扇区编号增1

                            STMFLASH_BUF_Index=0;//偏移位置为0

                            pBuffer=pBuffer+STMFLASH_BUF_Load_Length;

//已经写了STMFLASH_BUF_Load_Length个半字,修改pBuffer指针,指向需要写的数据

                            tSize=tSize-STMFLASH_BUF_Load_Length;//计算"剩余的待写数据"的个数

                            if( tSize>(STMFLASH_WriteLength_SIZE) )//超过一个扇区

                                     STMFLASH_BUF_Load_Length=STMFLASH_WriteLength_SIZE;

                //下一个扇区还是写不完

                            else//不足一个扇区

                            {

                                     STMFLASH_BUF_Load_Length=tSize;//下一个扇区可以写完了

                            }

                   }      

         }

         FLASH_Lock();//上锁

}

//gIAP_FlashAddress:烧写数据的起始地址

//pBuffer[]:待烧写数据块的首地址

//len:烧写数据的数量(字节)

void IAP_Write_APP_BIN(uint32_t gIAP_FlashAddress,u8 *pBuffer,u16 len)

{

         u16 t;

         u16 i;

         u16 tmp1,tmp2;

         u16 gIAP_Buffer[1024];

         u16 gIAP_Buffer_Length;

if( gIAP_FlashAddress<STM32_FLASH_BASE_ADDRESS||(gIAP_FlashAddress>=(STM32_FLASH_BASE_ADDRESS+1024*STM32_FLASH_SIZE)) )return;//非法地址

         i=0;

         for(t=0;t<len;)//Len>2048则循环写入

         {

                   tmp1=*pBuffer;tmp1=(u16)(tmp1&0x00FF);pBuffer++;

                   tmp2=*pBuffer;tmp2=(u16)(tmp2<<8);tmp2=(u16)(tmp2&0xFF00);pBuffer++;

                   tmp2=(u16)(tmp2|tmp1);//合成半字(16),生成"待写数据"

                   gIAP_Buffer[i]=tmp2;i++;//"待写数据"保存到IAP_Buffer[]

                   if(i==STMFLASH_WriteLength_SIZE)//1K个半字,则执行写入CPUFLASH

                   {

                            i=0;

                            gIAP_Buffer_Length=STMFLASH_WriteLength_SIZE;

           //装载烧写数据的数量,单位为半字

                            STMFLASH_Write(gIAP_FlashAddress,gIAP_Buffer,gIAP_Buffer_Length);

                   }

                   t=t+2;//因为读取的是半字,所以要加2

         }

         if(i)//还有i个半字等待写入FLASH

         {

                   gIAP_Buffer_Length=i;//装载烧写数据的数量,单位为半字

                   STMFLASH_Write(gIAP_FlashAddress,gIAP_Buffer,gIAP_Buffer_Length);

        //将最后的一些内容字节写进去

         }

}

//函数功能:u16型数据x保存到CPUFLASH,保存数据的地址为addr;

void EEPROM_U16_Data_Write(u16 x,uint32_t addr)

{

         union EEPROM_Uint16_Data_TYPE  temp;

         temp.Data=x;

         IAP_Write_APP_BIN(addr,temp.b,2);

}

//函数功能:CPUFLASH中读取u16型数据

u16 EEPROM_U16_Data_Read(uint32_t addr)

{

         u16 rerurn_value;

         STMFLASH_Read(addr,&rerurn_value,1);

         //CPUFLASH地址addr处读取1个半字(16位数据),保存到首地址为"&rerurn_value"的缓冲区中

         //返回数据保存在rerurn_value

         return(rerurn_value);

}

//函数功能:u32型数据x保存到CPUFLASH,保存数据的地址为addr;

void EEPROM_U32_Data_Write(u32 x,uint32_t addr)

{

         union EEPROM_U32_Data_TYPE  temp;

         temp.Data=x;

         IAP_Write_APP_BIN(addr,temp.b,4);

}

//函数功能:CPUFLASH中读取u32型数据

u32 EEPROM_U32_Data_Read(uint32_t addr)

{

         union EEPROM_U32_Data_TYPE  temp;

         u16 rerurn_value;

         temp.Data=0;

         STMFLASH_Read(addr,&rerurn_value,1);

         //CPUFLASH地址addr处读取1个半字(16位数据),保存到首地址为"&rerurn_value"的缓冲区中

         //返回数据保存在rerurn_value

         temp.b[0]=(u8)( rerurn_value&0x00FF );

         temp.b[1]=(u8)( (u16)(rerurn_value>>8)&0x00FF );

         addr=addr+2;//因为数据是以半字保存的,因此地址addr需要加2

         STMFLASH_Read(addr,&rerurn_value,1);

         //CPUFLASH地址addr处读取1个半字(16位数据),保存到首地址为"&rerurn_value"的缓冲区中

         //返回数据保存在rerurn_value

         temp.b[2]=(u8)( rerurn_value&0x00FF );

         temp.b[3]=(u8)( (u16)(rerurn_value>>8)&0x00FF );

         return(temp.Data);

}

//函数功能:float型数据x保存到CPUFLASH,保存数据的地址为addr;

void EEPROM_Float_Data_Write(float x,uint32_t addr)

{

         union EEPROM_FLOAT_DATA_TYPE  temp;

         temp.float_data=x;

         IAP_Write_APP_BIN(addr,temp.b,4);

}

//函数功能:CPUFLASH中读取float型数据

float EEPROM_Float_Data_Read(uint32_t addr)

{

         union EEPROM_FLOAT_DATA_TYPE  temp;

         u16 rerurn_value;

         temp.float_data=0;

         STMFLASH_Read(addr,&rerurn_value,1);

         //CPUFLASH地址addr处读取1个半字(16位数据),保存到首地址为"&rerurn_value"的缓冲区中

         //返回数据保存在rerurn_value

         temp.b[0]=(u8)( rerurn_value&0x00FF );

         temp.b[1]=(u8)( (u16)(rerurn_value>>8)&0x00FF );

         addr=addr+2;//因为数据是以半字保存的,因此地址addr需要加2

         STMFLASH_Read(addr,&rerurn_value,1);

         //CPUFLASH地址addr处读取1个半字(16位数据),保存到首地址为"&rerurn_value"的缓冲区中

         //返回数据保存在rerurn_value

         temp.b[2]=(u8)( rerurn_value&0x00FF );

         temp.b[3]=(u8)( (u16)(rerurn_value>>8)&0x00FF );

         return(temp.float_data);

}

//函数功能:double型数据x保存到CPUFLASH,保存数据的地址为addr;

void EEPROM_Double_Data_Write(double x,uint32_t addr)

{

         union EEPROM_DOUBLE_DATA_TYPE  temp;

         temp.float_data=x;

         IAP_Write_APP_BIN(addr,temp.b,8);

}

//函数功能:CPUFLASH中读取double型数据

double EEPROM_Double_Data_Read(uint32_t addr)

{

         union EEPROM_DOUBLE_DATA_TYPE  temp;

         u16 rerurn_value;

         temp.float_data=0;

         STMFLASH_Read(addr,&rerurn_value,1);

         //CPUFLASH地址addr处读取1个半字(16位数据),保存到首地址为"&rerurn_value"的缓冲区中

         //返回数据保存在rerurn_value

         temp.b[0]=(u8)( rerurn_value&0x00FF );

         temp.b[1]=(u8)( (u16)(rerurn_value>>8)&0x00FF );

         addr=addr+2;//因为数据是以半字保存的,因此地址addr需要加2

         STMFLASH_Read(addr,&rerurn_value,1);

         //CPUFLASH地址addr处读取1个半字(16位数据),保存到首地址为"&rerurn_value"的缓冲区中

         //返回数据保存在rerurn_value

         temp.b[2]=(u8)( rerurn_value&0x00FF );

         temp.b[3]=(u8)( (u16)(rerurn_value>>8)&0x00FF );

         addr=addr+2;//因为数据是以半字保存的,因此地址addr需要加2

         STMFLASH_Read(addr,&rerurn_value,1);

         //CPUFLASH地址addr处读取1个半字(16位数据),保存到首地址为"&rerurn_value"的缓冲区中

         //返回数据保存在rerurn_value

         temp.b[4]=(u8)( rerurn_value&0x00FF );

         temp.b[5]=(u8)( (u16)(rerurn_value>>8)&0x00FF );

         addr=addr+2;//因为数据是以半字保存的,因此地址addr需要加2

         STMFLASH_Read(addr,&rerurn_value,1);

         //CPUFLASH地址addr处读取1个半字(16位数据),保存到首地址为"&rerurn_value"的缓冲区中

         //返回数据保存在rerurn_value

         temp.b[6]=(u8)( rerurn_value&0x00FF );

         temp.b[7]=(u8)( (u16)(rerurn_value>>8)&0x00FF );

         return(temp.float_data);

}

#define STM32_FLASH_U16_DATA_ADDRESS     (STM32_FLASH_START_ADDRESS)          //2个字节,如保存0xABCD

#define STM32_FLASH_U32_DATA_ADDRESS     (STM32_FLASH_U16_DATA_ADDRESS+2)     //4个字节,如保存0x12345678

#define STM32_FLASH_FLOAT_DATA_ADDRESS   (STM32_FLASH_U32_DATA_ADDRESS+4)     //4个字节,如保存123.456

#define STM32_FLASH_DOUBLE_DATA_ADDRESS  (STM32_FLASH_FLOAT_DATA_ADDRESS+4)   //8个字节,如保存1234567.89

#define STM32_FLASH_U16_DATA_ADDRESS1     (STM32_FLASH_DOUBLE_DATA_ADDRESS+8) //2个字节,如保存0xDCBA

#define STM32_FLASH_U32_DATA_ADDRESS1     (STM32_FLASH_U16_DATA_ADDRESS1+2)   //4个字节,如保存0x87654321

#define STM32_FLASH_FLOAT_DATA_ADDRESS1   (STM32_FLASH_U32_DATA_ADDRESS1+4)   //4个字节,如保存456.123

#define STM32_FLASH_DOUBLE_DATA_ADDRESS1  (STM32_FLASH_FLOAT_DATA_ADDRESS1+4) //8个字节,如保存89.1234567

const char Address_REG[]="\r\nAddress: ";

const char Data_REG[]="     Data: ";

/*

Address: 0807F800     Data: ABCD

Address: 0807F812     Data: DCBA

Address: 0807F802     Data: 12345678

Address: 0807F814     Data: 87654321

Address: 0807F806     Data: 123.456001

Address: 0807F818     Data: 456.122986

Address: 0807F80A     Data: 1234567.890000

Address: 0807F81C     Data: 89.123457

Modify Data Test

Address: 0807F800     Data: AAAA

Address: 0807F812     Data: BBBB

Address: 0807F802     Data: CCCCCCCC

Address: 0807F814     Data: DDDDDDDD

Address: 0807F806     Data: 111.222000

Address: 0807F818     Data: 333.444000

Address: 0807F80A     Data: 555.666000

Address: 0807F81C     Data: 777.888000

*/

void CPU_FLASH_Read_Write_Test(void)

{

         u16 u16_Data;

         u32 u32_Data;

         float float_Data;

         double double_Data;

         EEPROM_U16_Data_Write(0xABCD,STM32_FLASH_U16_DATA_ADDRESS);

         EEPROM_U16_Data_Write(0xDCBA,STM32_FLASH_U16_DATA_ADDRESS1);

         EEPROM_U32_Data_Write(0x12345678,STM32_FLASH_U32_DATA_ADDRESS);

         EEPROM_U32_Data_Write(0x87654321,STM32_FLASH_U32_DATA_ADDRESS1);

         EEPROM_Float_Data_Write(123.456,STM32_FLASH_FLOAT_DATA_ADDRESS);

         EEPROM_Float_Data_Write(456.123,STM32_FLASH_FLOAT_DATA_ADDRESS1);

         EEPROM_Double_Data_Write(1234567.89,STM32_FLASH_DOUBLE_DATA_ADDRESS);

         EEPROM_Double_Data_Write(89.1234567,STM32_FLASH_DOUBLE_DATA_ADDRESS1);

         u16_Data=EEPROM_U16_Data_Read(STM32_FLASH_U16_DATA_ADDRESS);

         printf("%s",Address_REG);printf("%08X",STM32_FLASH_U16_DATA_ADDRESS);

         printf("%s",Data_REG);printf("%02X",u16_Data);

         u16_Data=EEPROM_U16_Data_Read(STM32_FLASH_U16_DATA_ADDRESS1);

         printf("%s",Address_REG);printf("%08X",STM32_FLASH_U16_DATA_ADDRESS1);

         printf("%s",Data_REG);printf("%02X",u16_Data);

         u32_Data=EEPROM_U32_Data_Read(STM32_FLASH_U32_DATA_ADDRESS);

         printf("%s",Address_REG);printf("%08X",STM32_FLASH_U32_DATA_ADDRESS);

         printf("%s",Data_REG);printf("%04X",u32_Data);

         u32_Data=EEPROM_U32_Data_Read(STM32_FLASH_U32_DATA_ADDRESS1);

         printf("%s",Address_REG);printf("%08X",STM32_FLASH_U32_DATA_ADDRESS1);

         printf("%s",Data_REG);printf("%04X",u32_Data);

         float_Data=EEPROM_Float_Data_Read(STM32_FLASH_FLOAT_DATA_ADDRESS);

         printf("%s",Address_REG);printf("%08X",STM32_FLASH_FLOAT_DATA_ADDRESS);

         printf("%s",Data_REG);printf("%f",float_Data);

         float_Data=EEPROM_Float_Data_Read(STM32_FLASH_FLOAT_DATA_ADDRESS1);

         printf("%s",Address_REG);printf("%08X",STM32_FLASH_FLOAT_DATA_ADDRESS1);

         printf("%s",Data_REG);printf("%f",float_Data);

         double_Data=EEPROM_Double_Data_Read(STM32_FLASH_DOUBLE_DATA_ADDRESS);

         printf("%s",Address_REG);printf("%08X",STM32_FLASH_DOUBLE_DATA_ADDRESS);

         printf("%s",Data_REG);printf("%lf",double_Data);

         double_Data=EEPROM_Double_Data_Read(STM32_FLASH_DOUBLE_DATA_ADDRESS1);

         printf("%s",Address_REG);printf("%08X",STM32_FLASH_DOUBLE_DATA_ADDRESS1);

         printf("%s",Data_REG);printf("%lf",double_Data);

//修改数据测试

  printf("\r\nModify Data Test");

         EEPROM_U16_Data_Write(0xAAAA,STM32_FLASH_U16_DATA_ADDRESS);

         EEPROM_U16_Data_Write(0xBBBB,STM32_FLASH_U16_DATA_ADDRESS1);

         EEPROM_U32_Data_Write(0xCCCCCCCC,STM32_FLASH_U32_DATA_ADDRESS);

         EEPROM_U32_Data_Write(0xDDDDDDDD,STM32_FLASH_U32_DATA_ADDRESS1);

         EEPROM_Float_Data_Write(111.222,STM32_FLASH_FLOAT_DATA_ADDRESS);

         EEPROM_Float_Data_Write(333.444,STM32_FLASH_FLOAT_DATA_ADDRESS1);

         EEPROM_Double_Data_Write(555.666,STM32_FLASH_DOUBLE_DATA_ADDRESS);

         EEPROM_Double_Data_Write(777.888,STM32_FLASH_DOUBLE_DATA_ADDRESS1);

         u16_Data=EEPROM_U16_Data_Read(STM32_FLASH_U16_DATA_ADDRESS);

         printf("%s",Address_REG);printf("%08X",STM32_FLASH_U16_DATA_ADDRESS);

         printf("%s",Data_REG);printf("%02X",u16_Data);

         u16_Data=EEPROM_U16_Data_Read(STM32_FLASH_U16_DATA_ADDRESS1);

         printf("%s",Address_REG);printf("%08X",STM32_FLASH_U16_DATA_ADDRESS1);

         printf("%s",Data_REG);printf("%02X",u16_Data);

         u32_Data=EEPROM_U32_Data_Read(STM32_FLASH_U32_DATA_ADDRESS);

         printf("%s",Address_REG);printf("%08X",STM32_FLASH_U32_DATA_ADDRESS);

         printf("%s",Data_REG);printf("%04X",u32_Data);

         u32_Data=EEPROM_U32_Data_Read(STM32_FLASH_U32_DATA_ADDRESS1);

         printf("%s",Address_REG);printf("%08X",STM32_FLASH_U32_DATA_ADDRESS1);

         printf("%s",Data_REG);printf("%04X",u32_Data);

         float_Data=EEPROM_Float_Data_Read(STM32_FLASH_FLOAT_DATA_ADDRESS);

         printf("%s",Address_REG);printf("%08X",STM32_FLASH_FLOAT_DATA_ADDRESS);

         printf("%s",Data_REG);printf("%f",float_Data);

         float_Data=EEPROM_Float_Data_Read(STM32_FLASH_FLOAT_DATA_ADDRESS1);

         printf("%s",Address_REG);printf("%08X",STM32_FLASH_FLOAT_DATA_ADDRESS1);

         printf("%s",Data_REG);printf("%f",float_Data);

         double_Data=EEPROM_Double_Data_Read(STM32_FLASH_DOUBLE_DATA_ADDRESS);

         printf("%s",Address_REG);printf("%08X",STM32_FLASH_DOUBLE_DATA_ADDRESS);

         printf("%s",Data_REG);printf("%lf",double_Data);

         double_Data=EEPROM_Double_Data_Read(STM32_FLASH_DOUBLE_DATA_ADDRESS1);

         printf("%s",Address_REG);printf("%08X",STM32_FLASH_DOUBLE_DATA_ADDRESS1);

         printf("%s",Data_REG);printf("%lf",double_Data);

}

#include "string.h"

//CPU的最后一个扇区

void Write_CPU_Last_SECTOR (void)

{

         u8 buf[STM_SECTOR_SIZE];

         memset(buf,0,STM_SECTOR_SIZE);

         buf[0]=0x11;

         buf[1]=0x22;

         buf[2]=0x33;

         buf[3]=0x44;

         buf[4]=0x55;

         buf[5]=0x66;

         buf[6]=0x77;

         buf[7]=0x88;

         buf[STM_SECTOR_SIZE-9]=0x99;

         buf[STM_SECTOR_SIZE-8]=0x88;

         buf[STM_SECTOR_SIZE-7]=0x77;

         buf[STM_SECTOR_SIZE-6]=0x66;

         buf[STM_SECTOR_SIZE-5]=0x55;

         buf[STM_SECTOR_SIZE-4]=0x44;

         buf[STM_SECTOR_SIZE-3]=0x33;

         buf[STM_SECTOR_SIZE-2]=0x22;

         buf[STM_SECTOR_SIZE-1]=0x11;

         IAP_Write_APP_BIN(STM32_FLASH_START_ADDRESS,buf,STM_SECTOR_SIZE);

}

3、测试结果:

编译配置:

使用J-Flash烧写程序,启动CPU,然后再读回程序,发现以前保存的数据仍旧存在,便于反复烧写,方便生产。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

GitHub搭建免费博客

一、GitHub仓库准备 ​ 搭建博客需要准备两个仓库。一个存放博客图床的仓库&#xff0c;另一个存放博客网站的仓库。 1.1、图床创建 新建仓库 第一步&#xff1a; ​ 第二步&#xff1a; 生成Token令牌 点击右上角头像->Settings->下拉&#xff0c;直到左侧到底&#…

【Word】写论文,参考文献涉及的上标、尾注、脚注 怎么用

一、功能位置 二、脚注和尾注区别 1.首先脚注是一个汉语词汇&#xff0c;论文脚注就是附在论文页面的最底端&#xff0c;对某些内容加以说明&#xff0c;印在书页下端的注文。脚注和尾注是对文本的补充说明。 2.其次脚注一般位于页面的底部&#xff0c;可以作为文档某处内容的…

原型模式类图与代码

现要求实现一个能够自动生成求职简历的程序&#xff0c;简历的基本内容包括求职者的姓名、性别、年龄及工作经历。希望每份简历中的工作经历有所不同&#xff0c;并尽量减少程序中的重复代码。 采用原型模式(Prototype)来实现上述要求&#xff0c;得到如图 7.25 所示的类图。 原…

QT+多线程编程

QT的多线程编程有两种 1、自定义类继承QThread 第一种是自定义一个类继承于QThread&#xff0c;重写run()方法来实现。然后当需要使用线程的时候你就新建一个自定义对象&#xff0c;然后调用start方法开始运行。 下面的例子是widget里面创建一个线程&#xff0c;然后调用sta…

MySQL数据库练习——视图

schooldb库——utf8字符集——utf8_general_ci排序规则 先创建库&#xff0c;再去使用下列的DDL语句。 DDL CREATE TABLE student (id int(11) NOT NULL AUTO_INCREMENT COMMENT 学号,createDate datetime DEFAULT NULL COMMENT 创建时间,modifyDate datetime DEFAULT NULL …

电度表抄表是什么?什么叫电度表抄表?

一、电度表抄表的概念和作用 电度表抄表是电力系统中一个基本但非常重要的阶段。它指的是对安装在用户处电度表开展载入&#xff0c;记录下来电力消耗的值&#xff0c;便于测算电费的一个过程。此项工作不仅有利于供电公司精确扣除电费&#xff0c;都是监控和管理电力工程应用…

【Pytorch】4.torchvision.datasets的使用

什么是torchvision.datasets、 是pytorch官方给出的关于cv领域的训练数据集&#xff0c;我们可以用官方提供的数据集进行学习与训练 如何查看 我们可以进入Pytorch官网 切换一下版本到v0.9.0&#xff0c;就可以看到官方给出的数据集了 同时也有官方训练好的cv模型可以供我们…

Linux(openEuler、CentOS8)企业内网主备DNS服务器搭建

本实验环境为openEuler系统<以server方式安装>&#xff08;CentOS类似&#xff0c;可参考本文&#xff09; 知识点 什么是DNS&#xff1a;DNS服务器&#xff0c;即域名服务器&#xff08;Domain Name Server&#xff09;&#xff0c;是进行域名和与之相对应的IP地址转换…

硬件工程师必读:10条职业发展黄金法则!

在快速发展的科技时代&#xff0c;硬件工程师作为推动技术创新和产业升级的重要力量&#xff0c;其职业发展之路既充满挑战也蕴含无限机遇。为了在这条道路上稳步前行&#xff0c;我们首先需要了解硬件产品的研发流程。 在这个过程中&#xff0c;公司内的每个岗位都发挥着不可或…

【机器学习】视觉基础模型的三维意识:前沿探索与局限

视觉基础模型的三维意识&#xff1a;前沿探索与局限 一、引言二、视觉基础模型的三维意识三、当前模型的局限性四、实验与结果五、总结与展望 大规模预训练的进展已经产生了具有强大能力的视觉基础模型。最近的模型不仅可以推广到任意图像的训练任务&#xff0c;而且它们的中间…

跨境电商日报:虾皮开设新物流中心;Tk Shop菲律宾调整佣金费率

2024年5月7日跨境电商日报目录&#xff1a; 1.Shopee开设新物流中心&#xff1b; 2.TikTok Shop菲律宾站调整佣金费率&#xff1b; 3.野莓海外仓卖家可以销售9公斤以上商品&#xff1b; 4.亚马逊一季度净销售额同比增长13%&#xff1b; 5.巴西对华二氧化钛启动反倾销调查 …

AI原生实践:测试用例创作探索

测试用例作为质量保障的核心&#xff0c;影响着研发-测试-发布-上线的全过程&#xff0c;如单元测试用例、手工测试用例、接口自动化用例、UI 自动化用例等&#xff0c;但用例撰写的高成本尤其是自动化用例&#xff0c;导致了用例的可持续积累、更新和迭代受到非常大制约。长久…

如何利用工作流自定义一个AI智能体

选择平台 目前已经有不少大模型平台都提供自定义智能体的功能&#xff0c;比如 百度的文心 https://agents.baidu.com/ 阿里的百炼平台 https://bailian.console.aliyun.com/。 今天再来介绍一个平台扣子&#xff08;https://www.coze.cn/&#xff09;&#xff0c;扣子是…

被忽略的C语言堆栈内存编程细节

没想到&#xff0c;评论区的一个提问&#xff0c;成了我知识的裂缝。&#x1f923; 前段时间解决了自己代码中的一个关于栈内存问题&#xff0c;便班门弄斧的写了一篇文章栈内存文章&#xff0c;没想到发出来很快便被大佬发现了其中一个问题&#x1f923;&#xff0c;在评论区里…

程序员的神器指南!揭秘软件开发必备工具

在软件开发的广袤海洋中&#xff0c;程序员们像是驾驶着帆船探索未知的航海者。他们面对的不仅仅是代码的挑战&#xff0c;还有项目管理、协作沟通和时间限制的压力。为了应对这些挑战&#xff0c;程序员们需要一系列强大的工具&#xff0c;就像是海中的指南针&#xff0c;帮助…

项目又延期了,怎么办?

Ada问&#xff1a;负责的几个版本上线都延期了&#xff0c;在测试环境出了问题&#xff0c;改过再测上线后&#xff0c;还是出问题。身为产品经理&#xff0c;很困惑&#xff0c;项目不断延期&#xff0c;又被打脸。怎么办&#xff1f; 学姐回答&#xff1a; 经理每天都是忙着…

mac安装linux的centos strem9 虚拟机解压rar文件报错

背景&#xff1a;解压rar文件需要再linux上安装unrar工具 yum install unrar直接安装的后解压报错,如图 解决办法&#xff1a; 下载&#xff1a;wget https://www.rarlab.com/rar/rarlinux-x64-6.0.2.tar.gz 安装&#xff1a; tar -zxvf rarlinux-x64-6.0.2.tar.gz cd rar …

C语言例题35、反向输出字符串(指针方式),例如:输入abcde,输出edcba

#include <stdio.h>void reverse(char *p) {int len 0;while (*p ! \0) { //取得字符串长度p;len;}while (len > 0) { //反向打印到终端printf("%c", *--p);len--;} }int main() {char s[255];printf("请输入一个字符串&#xff1a;");gets(s)…

redis持久化存储

Redis的持久化机制 Redis是一个基于内存的数据库&#xff0c;它的数据是存放在内存中&#xff0c;内存有个问题就是关闭服务或者断电会丢失。Redis的数据也支持写到硬盘中&#xff0c;这个过程就叫做持久化。 Redis提供如下两种持久化方式 RDB&#xff08;Redis DataBase&am…