正点原子:第67讲 FLASH闪存编程原理与步骤-M3第68讲 FLASH模拟EEPROM实验-M3
其实就是说STM32的闪存,主要是读/写/擦除,做出了一个风格和SPI FLASH一致的模块。
而所谓的模拟EEPROM其实是:前者是页为单位,写之前要擦除,一擦就是2k这样
(李工说:主要是器件只能1-->0而不能0-->1所以擦除其实是全部覆盖成1),
而后者EEPROM是任意字节的读写。
模拟就是先把Flah的东西读出来,在把要写的拼接进去,随后一起写入,假装你可以单字节控制。
说个C基础知识:
void St_Flash_Write(unsigned char* pBuffer,unsigned int WriteAddr,short NumByteTowrite)
{
short i=0;
unsigned int data;
for(i=0;i<NumByteTowrite;i+=4)
{
data=(pBuffer[i+3]<<24)|(pBuffer[i+2]<<16)|(pBuffer[i+1]<<8)|(pBuffer[i+0]);//3-2-1-0//0-1-2-3
St_Flash_Write_U32(WriteAddr+i,data);
}
}
传入一个u8的数组,在里面我自己操作的,每4个拼接成一个U32,真是愚蠢。
怎么改造?????
改进1 ==OK data = *(unsigned int *)&pBuffer[i];
这样就好了,别自己手动搞。
也就是:data = *(unsigned int *)&pBuffer[i]; 《== 》(pBuffer[i+3]<<24)|(pBuffer[i+2]<<16)|(pBuffer[i+1]<<8)|(pBuffer[i+0]);
改进2==不好
其实可以更加高级一点,现在别人传递参数u8数组,是挺方便的,我可以让别人在传递参数的强行转化成U32。
const u8 T_Buffer[]={"HELLO"};
#define SIZE sizeof(T_Buffer) //数组长度
#define FLASH_SAVE_ADDR 0X0807F800
St_Flash_Page_Clear(FLASH_SAVE_ADDR);
St_Flash_Write((u32*)T_Buffer,FLASH_SAVE_ADDR,SIZE);
这样也挺好的!自动转化了,效果呢?
自己的函数要优化一下
void St_Flash_Write(unsigned int* pBuffer,unsigned int WriteAddr,short NumByteTowrite)
{
short i=0;
short total=NumByteTowrite%4?(NumByteTowrite/4+1):(NumByteTowrite/4);
for(i=0;i<total;i++)
{
St_Flash_Write_U32(WriteAddr+i*4,pBuffer[i]);
}
// while(pBuffer[i])
// {St_Flash_Write_U32(WriteAddr+i*4,pBuffer[i]);
// i++;};
}
上面两种方法都可以。
第一种for是重新确定数组长度,比如u8个数是5的时候变成U32就是2啦!
第二种也好,不需要长度了,数组结束的地方是0
+++++++++++++++++20180712补充强风格的读写函数+++++++
#include<stdio.h>
#include<string.h>
int chip_flash_write( int addr , char* data , int len)
{
int ramsource = 0;
char buff[20];
memset(buff , 0x00 , 20);
memcpy(buff , data , len);
ramsource = (int)&buff;
FLASH_Write(addr , (uint32_t*) ramsource , 512);
printf("1%s\n",data);
printf("2%d\n",ramsource);
printf("3%s\n",ramsource);
return 1;
}
int main()
{
char * str="hello";
chip_flash_write(0,str,6);
}
1hello
26422248
3hello
[Finished in 2.4s]
这样转化也是可以的,就可以调用API了 u8--u32
++++++++++++++++++++++++20180720补充爱动脑的人+++++++++++++
#include<stdio.h>
unsigned int ARRY_U8_TO_U32(unsigned char* pBuffer)
{
unsigned int data;
data=(pBuffer[3]<<24)|(pBuffer[2]<<16)|(pBuffer[1]<<8)|(pBuffer[0]);
data=*(unsigned int *)( pBuffer );//04030201
data=*(unsigned int *)(&pBuffer[0]);//04030201
data=*(pBuffer);//1
return data;
}
int main()
{
unsigned char arr[4]={1,2,3,4};
printf("%x\n",ARRY_U8_TO_U32(arr) );
}
比如看到强哥的下面代码 就可以优化!!
不好: communityTemp.village_id = (communityID[0]<<24)|(communityID[1]<<16)|(communityID[2]<<8)|(communityID[3]);
好:
data=*(unsigned int *)( pBuffer );//04030201
data=*(unsigned int *)(&pBuffer[0]);//04030201
这里面涉及到一个字节序的问题