STM32CUBEIDE配置STM32F072使用FLASH进行读写操作

1. 配置

似乎没有与FLASH相关的…

在这里插入图片描述

2. 时钟

大同小异

在这里插入图片描述

3. 中断

中断:一定要打开

在这里插入图片描述

4. 增加代码部分

在此增加了 “stm32flash.c” 和 “stm32flash.h” 文件

在这里插入图片描述

4.1 stm32flash.c 内部代码

写入部分:

/*带检查的写入*/
#define FLASH_WAITETIME 1000

#define STM32_FLASH_BASE   0x08000000 		//STM32 FLASH的起始地址

#define STM_SECTOR_SIZE	2048	//大容量STM32的扇区大小为2K

uint16_t STMFLASH_BUF[STM_SECTOR_SIZE/2];

void STMFLASH_Write(uint32_t WriteAddr,uint16_t *pBuffer,uint16_t NumToWrite)
{
	uint32_t secpos;	   //扇区地址
	uint16_t secoff;	   //扇区内偏移地址(16位字计算)
	uint16_t secremain; //扇区内剩余地址(16位字计算)
 	uint16_t i;
	uint32_t offaddr;   //去掉0X08000000后的地址

	if(WriteAddr<STM32_FLASH_BASE||(WriteAddr>=(STM32_FLASH_BASE+1024*512)))
	{
		return;//非法地址
	}
	HAL_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_PageErase(secpos*STM_SECTOR_SIZE+STM32_FLASH_BASE);	//擦除这个扇区
			FLASH_WaitForLastOperation(FLASH_WAITETIME);            	//等待上次操作完成
			CLEAR_BIT(FLASH->CR, FLASH_CR_PER);							//清除CR寄存器的PER位,此操作应该在FLASH_PageErase()中完成!
																		//但是HAL库里面并没有做,应该是HAL库bug!
			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
		{
			FLASH_WaitForLastOperation(FLASH_WAITETIME);       	//等待上次操作完成
			STMFLASH_Write_NoCheck(WriteAddr,pBuffer,secremain);//写已经擦除了的,直接写入扇区剩余区间.
		}

		if(NumToWrite==secremain)
		{
			break;//写入结束了
		}
		else//写入未结束
		{
			secpos++;				//扇区地址增1
			secoff=0;				//偏移位置为0
		   	pBuffer+=secremain;  	//指针偏移
			WriteAddr+=secremain*2;	//写地址偏移(16位数据地址,需要*2)
		   	NumToWrite-=secremain;	//字节(16位)数递减
			if(NumToWrite>(STM_SECTOR_SIZE/2))
			{
				secremain=STM_SECTOR_SIZE/2;//下一个扇区还是写不完
			}
			else
			{
				secremain=NumToWrite;//下一个扇区可以写完了
			}
		}
	};
	HAL_FLASH_Lock();		//上锁
}


/*
 *   无检查写入
 *   WriteAddr - 地址
 *   pBuffer - 数据
 *   NumToWrite - 长度
 *
 * */
void STMFLASH_Write_NoCheck(uint32_t WriteAddr,uint16_t *pBuffer,uint16_t NumToWrite)
{
	uint16_t i;
	for(i=0;i<NumToWrite;i++)
	{
		HAL_FLASH_Program(FLASH_TYPEPROGRAM_HALFWORD,WriteAddr,pBuffer[i]);
	  WriteAddr+=2;		//地址增加2.
	}
}

读取部分:

/*
 * 读取半字  16位
 * faddr - 地址
 * */
uint16_t STMFLASH_ReadHalfWord(uint32_t faddr)
{
	return *(__IO uint16_t*)faddr;
}

/*读取len个16位数据*/
void STMFLASH_Read(uint32_t ReadAddr,uint16_t *pBuffer,uint16_t NumToRead)
{
	uint16_t i;
	for(i=0;i<NumToRead;i++)
	{
		pBuffer[i]=STMFLASH_ReadHalfWord(ReadAddr);//读取2个字节.
		ReadAddr+=2;//偏移2个字节.
	}
}

4.2 stm32flash.h 内部代码

#ifndef __STMFLASH_H
#define __STMFLASH_H

#include "stm32f0xx_hal.h"
#include "main.h"


void STMFLASH_Write(uint32_t WriteAddr,uint16_t *pBuffer,uint16_t NumToWrite);
void STMFLASH_Write_NoCheck(uint32_t WriteAddr,uint16_t *pBuffer,uint16_t NumToWrite);

void STMFLASH_Read(uint32_t ReadAddr,uint16_t *pBuffer,uint16_t NumToRead);
uint16_t STMFLASH_ReadHalfWord(uint32_t faddr);

#endif

4.3 例子

#define  FLASH_0400_ADDR   0x0800F000   //地址宏  60K

uint16_t R_init_data = 16000;
uint16_t  R_0400_F=16000;

if(STMFLASH_ReadHalfWord(FLASH_0400_ADDR) > 50000) //读半字 2byte,16位数据,初始为0xFFFF(65535);
{
 STMFLASH_Write(FLASH_0400_ADDR, &R_init_data, 1); //写入16位数据,16000(0x3E80),长度1个半字,2byte(1个16位数据);
}
else
{
 R_0400_F = STMFLASH_ReadHalfWord(FLASH_0400_ADDR); //读取数据赋值
}
  • 2
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

LuDvei

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值