OpenEdv-开源电子网

 找回密码
 立即注册
正点原子全套STM32/Linux/FPGA开发资料,上千讲STM32视频教程免费下载...
查看: 2868|回复: 3

求助!STM32大容量flash单页写入2048字节数失败

[复制链接]

1

主题

3

帖子

0

精华

新手入门

积分
18
金钱
18
注册时间
2019-8-5
在线时间
3 小时
发表于 2020-9-21 14:23:16 | 显示全部楼层 |阅读模式
5金钱
我需要用到片内flash存储参数,找了一个例程可以用,但是单页写入字数超过1024就不行了,不知道为什么。stm32f103rc 单页2k,按逻辑可以写入2048字节数,下面是驱动代码,大伙来探讨一下。
//从指定地址开始写入指定长度的数据
//WriteAddr:起始地址(此地址必须为2的倍数!!)
//pBuffer:数据指针
//NumToWrite:半字(16位)数(就是要写入的16位数据的个数.)
#if STM32_FLASH_SIZE<256
#define STM_SECTOR_SIZE 1024 //字节
#else
#define STM_SECTOR_SIZE        2048
#endif                 
u16 STMFLASH_BUF[STM_SECTOR_SIZE/2];//最多是2K字节
void STMFLASH_Write(u32 WriteAddr,u16 *pBuffer,u16 NumToWrite)       
{
        u32 secpos;           //扇区地址
        u16 secoff;           //扇区内偏移地址(16位字计算)
        u16 secremain; //扇区内剩余地址(16位字计算)          
        u16 i;   
        u32 offaddr;   //去掉0X08000000后的地址
        if(WriteAddr<STM32_FLASH_BASE||(WriteAddr>=(STM32_FLASH_BASE+1024*STM32_FLASH_SIZE)))return;//非法地址
        FLASH_Unlock();                                                //解锁
        offaddr=WriteAddr-STM32_FLASH_BASE;                //实际偏移地址.
        secpos=offaddr/STM_SECTOR_SIZE;                        //扇区地址  0~127 for STM32F103RBT6
        secoff=(offaddr%STM_SECTOR_SIZE)/2;                //在扇区内的偏移(2个字节为基本单位.)
        secremain=STM_SECTOR_SIZE/2-secoff;                //扇区剩余空间大小   
        if(NumToWrite<=secremain)secremain=NumToWrite;//不大于该扇区范围
        while(1)
        {       
                STMFLASH_Read(secpos*STM_SECTOR_SIZE+STM32_FLASH_BASE,STMFLASH_BUF,STM_SECTOR_SIZE/2);//读出整个扇区的内容
                for(i=0;i<secremain;i++)//校验数据
                {
                        if(STMFLASH_BUF[secoff+i]!=0XFFFF)break;//需要擦除            
                }
                if(i<secremain)//需要擦除
                {
                        FLASH_ErasePage(secpos*STM_SECTOR_SIZE+STM32_FLASH_BASE);//擦除这个扇区
                        for(i=0;i<secremain;i++)//复制
                        {
                                STMFLASH_BUF[i+secoff]=pBuffer[i];          
                        }
                        STMFLASH_Write_NoCheck(secpos*STM_SECTOR_SIZE+STM32_FLASH_BASE,STMFLASH_BUF,STM_SECTOR_SIZE/2);//写入整个扇区  
                }else STMFLASH_Write_NoCheck(WriteAddr,pBuffer,secremain);//写已经擦除了的,直接写入扇区剩余区间.                                   
                if(NumToWrite==secremain)break;//写入结束了
                else//写入未结束
                {
                        secpos++;                                //扇区地址增1
                        secoff=0;                                //偏移位置为0          
                           pBuffer+=secremain;          //指针偏移
                        WriteAddr+=secremain;        //写地址偏移          
                           NumToWrite-=secremain;        //字节(16位)数递减
                        if(NumToWrite>(STM_SECTOR_SIZE/2))secremain=STM_SECTOR_SIZE/2;//下一个扇区还是写不完
                        else secremain=NumToWrite;//下一个扇区可以写完了
                }         
        };       
        FLASH_Lock();//上锁
}
#endif








/* 私有宏定义 ----------------------------------------------------------------*/
/* 获取缓冲区的长度 */
#define  FLASH_WriteAddress     0x8032000            // 写在靠后位置,防止破坏程序 page100 2048*100
#define  FLASH_ReadAddress      FLASH_WriteAddress
#define  FLASH_TESTSIZE       1024 //实际是512*2=1024字节 <=1024
/* 私有变量 ------------------------------------------------------------------*/
/* 发送缓冲区初始化 */
uint16_t Tx_Buffer[FLASH_TESTSIZE]={0};
uint16_t Rx_Buffer[FLASH_TESTSIZE]={0};

__IO TestStatus TransferStatus1 = FAILED;

/* 扩展变量 ------------------------------------------------------------------*/
/* 私有函数原形 --------------------------------------------------------------*/
static void Delay(uint32_t time);
static TestStatus Buffercmp(uint16_t* pBuffer1, uint16_t* pBuffer2, uint16_t BufferLength);

/* 函数体 --------------------------------------------------------------------*/

/**
  * 函数功能: 主函数.
  * 输入参数: 无
  * 返 回 值: 无
  * 说    明: 无
  */
int main(void)
{   
  uint16_t i;
  /* 调试串口初始化配置,115200-N-8-1.使能串口发送和接受 */
  DEBUG_USART_Init();  
        /*初始化LED*/
  LED_GPIO_Init();  

  for(i=0;i<FLASH_TESTSIZE;++i)
  {
     Tx_Buffer[i]=i;
  }

  /* 调用格式化输出函数打印输出数据 */
  printf("这是一个内部flash读写测试实验\n");  

  /* 向内部Flash写入数据 */
  STMFLASH_Write(FLASH_WriteAddress,Tx_Buffer,FLASH_TESTSIZE);
  /* 小延时 */
  Delay(100);

  /* 从内部Flash读取数据 */
        STMFLASH_Read(FLASH_WriteAddress,Rx_Buffer,FLASH_TESTSIZE);
  /* 小延时 */
        Delay(100);
               
  /* 检查写入的数据与读出的数据是否相等 */
  TransferStatus1 = Buffercmp(Tx_Buffer, Rx_Buffer, FLASH_TESTSIZE);

  if( PASSED == TransferStatus1 )
  {   
    printf("内部Flash测试成功!\r");
               
    LED1_ON;
  }
  else
  {        
    printf("内部Flash测试失败!\r");
    LED2_ON;
  }

  /* 无限循环 */
  while (1)
  {   
  }
}

正点原子逻辑分析仪DL16劲爆上市
回复

使用道具 举报

1

主题

3

帖子

0

精华

新手入门

积分
18
金钱
18
注册时间
2019-8-5
在线时间
3 小时
 楼主| 发表于 2020-9-21 14:25:27 | 显示全部楼层
#define  FLASH_TESTSIZE       1024  这里超过1024就出错了,有谁可以说一下为什么吗?
回复

使用道具 举报

33

主题

1628

帖子

0

精华

论坛元老

Rank: 8Rank: 8

积分
6657
金钱
6657
注册时间
2015-8-25
在线时间
1034 小时
发表于 2020-9-21 15:30:28 | 显示全部楼层
uint16是两个字节,一个扇区大小2048字节,你超过1024个uint16肯定超过一个扇区大小了
回复

使用道具 举报

1

主题

3

帖子

0

精华

新手入门

积分
18
金钱
18
注册时间
2019-8-5
在线时间
3 小时
 楼主| 发表于 2020-9-22 09:17:26 | 显示全部楼层
szczyb1314 发表于 2020-9-21 15:30
uint16是两个字节,一个扇区大小2048字节,你超过1024个uint16肯定超过一个扇区大小了

哦,明白了。每次写入半个字,半个字是2字节,所以1024个半字就是2048字节。
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则



关闭

原子哥极力推荐上一条 /2 下一条

正点原子公众号

QQ|手机版|OpenEdv-开源电子网 ( 粤ICP备12000418号-1 )

GMT+8, 2025-5-17 01:28

Powered by OpenEdv-开源电子网

© 2001-2030 OpenEdv-开源电子网

快速回复 返回顶部 返回列表