MSP430G2553 flash操作例子

2024-03-26 20:08

本文主要是介绍MSP430G2553 flash操作例子,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

/* 实际适用于MSP430x2xx系列单片机,包含以下10个常用功能函数: 
(1)初始化。Flash_Init(unsigned char Div,unsigned char Seg ):依据SMCLK频率计算设定Flash的时
钟的分频系数,靠Seg段号码确定计划操作的段起始地址。 
(2)整段擦除。Flash_Erase():段擦除函数。 
(3)读字节。Flash_ReadChar(unsigned int Addr):读取偏移地址Addr位置1个字节的数据。 
(4)读字。Flash_ReadWord(unsigned int Addr):读取偏移地址Addr位置1个字的数据。 
(5)读一串字节到RAM数组。Flash_ReadSeg(unsigned int Addr, unsigned int SegSize,char * 
Array):读取起始偏移地址为Addr,长度SegSize个字节数据到RAM的Array数组。 
(6)直接写1个字节。Flash_Direct_WriteChar(unsigned int Addr):直接写偏移地址Addr位置1个字节
的数据。 
(7)直接写1个字。Flash_Direct_WriteWord(unsigned int Addr):直接写偏移地址Addr位置1个字的数
据。 
(8)备份后写1字节。Flash_Bak_WriteChar(unsigned int Addr):先备份段内其他数据,擦写后,在偏
移地址Addr位置写1个字节的数据,再还原段内其他数据。(仅限信息flash段,使用RAM备份) 
(9)备份后写1个字。Flash_Bak_WriteWord(unsigned int Addr):先备份段内其他数据,擦写后,在偏
移地址Addr位置写1个字的数据,再还原段内其他数据。(仅限信息flash段,使用RAM备份)。 
(10)读SegA专用函数。Flash_SegA_ReadChar(unsigned int Addr):读取SegA段偏移地址Addr位置1
个字节的数据。 
说明:  1、块写函数需要在RAM中调用函数指针来使用,本库函数未涉及 2、其他长字节的数据类型读写需使用结构体,本库函数未涉及 3、所有函数均针对无符号整型数据,如需使用有符号整型,需修改函数 4、对InfoA段单独处理,只有读字节函数Flash_SegA_ReadChar(),不提供擦写函数。 5、其他函数的段操作首地址SegAddr被Flash_Init()函数“限定”,不易发生误写 */ #include   "MSP430G2553.h" 
unsigned  int SegAddr=0;                        //全局变量 
unsigned  int SegPre=0;                         //全局变量  当前信息段 
/******************************************************************************************************  
* 名     称:Flash_Init() 
* 功       能:对Flash时钟进行初始化设置 
* 入口参数:Div:根据SMCLK频率计算的所需分频值,可设定为1-64 
*           选择原则: SMCLK分频后,落在257kHz~476kHz
*        Seg:段号,可设为"0"-"31"或”"A"、"B"、"C"、"D"。 
* 出口参数:1:配置成功 
*           0:配置失败 
* 说     明:操作Flash其他函数前,需要调用该初始化函数设置时钟分频和待操作段首地址。 
*            其他函数中均不出现绝对地址,防止误操作。 
* 范     例:  Flash_Init(3,'B' ) 3分频、对Info B段操作 
******************************************************************************************************/ 
unsigned char Flash_Init(unsigned  char Div,unsigned  char Seg ) 
{ //-----设置Flash的时钟和分频,分频为恰好为最低位,直接用Div-1即可----- if(Div<1) Div=1; if(Div>64) Div=64; FCTL2 = FWKEY + FSSEL_2 + Div-1;    //  默认使用SMCLK,分频系数参数传入 //-----操作对象为主Flash段的情况,可通过512的倍数设置段起始地址----- SegPre = Seg;            //获取当前段 if (Seg <= 31)             //判断是否处于主Flash段 { SegAddr=0xFFFF-(Seg+1)*512+1;         //计算段起始地址 return(1);                   //赋值成功后即可退出并返回成功标志”1“ } //-----操作对象为信息Flash段的情况,穷举即可----- switch(Seg)                //判断是否处于信息Flash段 { case  'A':  case'a':    SegAddr=0x10C0; break; case  'B':  case'b':    SegAddr=0x1080; break; case  'C':  case'c':    SegAddr=0x1040; break; case  'D':  case'd':    SegAddr=0x1000; break; default:    SegAddr=0x20FF;  return(0);       //0x20FF地址为空白区,保护Flash } return(1); 
} 
/******************************************************************************************************  
* 名    称:Flash_Erase() 
* 功    能:擦除Flash的一个数据块,擦写段由初始化函数 Flash_Init()的SegAddr变量决定 
* 入口参数:无 
* 出口参数:无 
* 说    明:函数中给出了擦除InfoFlashA段的操作代码(已注释掉了),但不建议初学者使用。 
* 范    例:无 
******************************************************************************************************/ 
void Flash_Erase() 
{ unsigned char  *Ptr_SegAddr;                     //Segment  pointer Ptr_SegAddr = (unsigned char *)SegAddr;        //Initialize Flash  pointer FCTL1 = FWKEY + ERASE;                       //段擦除模式 FCTL3 = FWKEY;                                  //解锁 //FCTL3 = FWKEY+LOCKA;                     //对InfoFlashA也解锁 _DINT(); *Ptr_SegAddr = 0;                              //擦除待操作段 while(FCTL3&BUSY);                           //Busy _EINT(); FCTL1 = FWKEY;                                //取消擦模式 FCTL3 = FWKEY+LOCK;                          //上锁 
// FCTL3 = FWKEY+LOCK+LOCKA;              //对InfoFlashA也上锁 
} 
/******************************************************************************************************  
* 名    称:Flash_ReadChar() 
* 功    能:从Flash中读取一个字节 
* 入口参数:Addr:存放数据的偏移地址 
* 出口参数:Data:读回的数据;当偏移溢出时返回0 
* 说    明:无 
* 范    例:无 
******************************************************************************************************/ 
unsigned char Flash_ReadChar  (unsigned  int Addr) 
{ unsigned char Data=0; unsigned  int *Ptr_SegAddr,temp=0;                       //Segment  pointer //-----  段范围限定。为了内存管理安全,只允许本段操作----- if((SegPre<=31&&Addr>=512) ||(SegPre>31&&Addr>=64) ) return 0; temp =SegAddr+Addr; Ptr_SegAddr =(void*)temp;                     //initialize Flash  pointer Data=*(Ptr_SegAddr); return(Data); 
} 
/******************************************************************************************************  
* 名    称:Flash_ReadWord() 
* 功    能:从FlashROM读回一个整型变量,地址应为偶数 
* 入口参数:Addr:存放数据的偏移地址,仍按字节计算,需为偶数 
* 出口参数:Data:读回的整型变量值  ;当偏移溢出时返回0 
* 说    明:无 
* 范    例:无 
******************************************************************************************************/ 
unsigned  int Flash_ReadWord (unsigned  int Addr) 
{ unsigned  int *Ptr_SegAddr; unsigned  int temp=0,Data=0;                             //Segment  pointer //-----  段范围限定。为了内存管理安全,只允许本段操作----- if((SegPre<=31&&Addr>=512) ||(SegPre>31&&Addr>=64) ) return 0; temp = SegAddr+Addr; Ptr_SegAddr = (void  *)temp;                  //Initialize Flash pointer Data=*(Ptr_SegAddr); return(Data); 
} 
/******************************************************************************************************  
* 名    称:Flash_ReadSeg() 
* 功    能:将Flash段内一串数据拷贝到RAM的Array数组 
* 入口参数:Addr:起始偏移地址 
*          SegSize:数据个数 
*          *Array:RAM中数组的头指针 
* 出口参数:返回出错信息  0:偏移溢出 ;1:正常工作 
* 说    明:无 
* 范    例:无 
******************************************************************************************************/ 
char Flash_ReadSeg(unsigned  int Addr, unsigned  int SegSize,unsigned char * Array) 
{ unsigned  int i=0,temp=0; unsigned char  *Ptr_SegAddr;                        //Segment  pointer //-----  段范围限定。为了内存管理安全,只允许本段操作----- if((SegPre<=31&&(Addr+SegSize)>512) ||(SegPre>31&&(Addr+SegSize)>64) ) return 0; for(i=0;i<SegSize;i++) { temp=SegAddr+Addr+i;             //防止编译器处理指针偏移出错 Ptr_SegAddr = (unsigned char *)temp;             //Initialize Flash  pointer Array[i]=*Ptr_SegAddr;            //指针移位方法赋值 } return 1; 
} 
/******************************************************************************************************  
* 名    称:Flash_Direct_WriteChar() 
* 功    能:强行向Flash中写入一个字节(Char型变量),而不管是否为空 
* 入口参数:Addr:存放数据的偏移地址 Data:待写入的数据 
* 出口参数:返回出错信息  0:偏移溢出 ;1:正常工作 
* 范    例:Flash_Direct_WriteChar(0,123);将常数123写入0单元 Flash_Direct_WriteChar(1,a);将整型变量a写入1单元 
******************************************************************************************************/ 
char Flash_Direct_WriteChar (unsigned  int Addr,unsigned char Data) 
{ unsigned  int temp=0; unsigned char  *Ptr_SegAddr;                    //Segment  pointer //-----  段范围限定。为了内存管理安全,只允许本段操作----- if((SegPre<=31&&Addr>=512) ||(SegPre>31&&Addr>=64) ) return 0; temp = SegAddr+Addr; Ptr_SegAddr = (unsigned char *)temp;                //Initialize Flash  pointer FCTL1=FWKEY+WRT;            //正常写状态 FCTL3=FWKEY;              //解除锁定 
//  FCTL3=FWKEY+LOCKA;            //解除锁定(包括A段) _DINT();            //关总中断 *Ptr_SegAddr=Data;              //指定地址,写1字节 while(FCTL3&BUSY);            //等待操作完成 _EINT();            //开总中断 FCTL1=FWKEY;              //退出写状态 FCTL3=FWKEY+LOCK;            //恢复锁定,保护数据 
//  FCTL3=FWKEY+LOCK+LOCKA;        //恢复锁定,保护数据(包括A段) return 1; 
} 
/****************************************************************************************************** 
* 名    称:Flash_Direct_WriteWord() 
* 功    能:强行向Flash中写入一个字型变量,而不管存储位置是否事先擦除 
* 入口参数:Addr:存放数据的偏移地址,仍按字节计算,需为偶数 Data:待写入的数据 
* 出口参数:返回出错信息  0:偏移溢出 ;1:正常工作 
* 范    例:Flash_Direct_WriteWord(0,123);将常数123写入0单元 Flash_Direct_WriteWord(2,a);将整型变量a写入2单元 
******************************************************************************************************/ 
char Flash_Direct_WriteWord  (unsigned  int Addr,unsigned  int Data) 
{ unsigned  int temp=0; unsigned  int *Ptr_SegAddr;                       //Segment  pointer //-----  段范围限定。为了内存管理安全,只允许本段操作----- if((SegPre<=31&&Addr>=512) ||(SegPre>31&&Addr>=64) ) return 0; temp=SegAddr+Addr; Ptr_SegAddr = (unsigned  int *)temp;                 //Initialize Flash  pointer FCTL1=FWKEY+WRT;            //正常写状态 FCTL3=FWKEY;              //解除锁定 
//  FCTL3=FWKEY+LOCKA;            //解除锁定(包括A段) _DINT();            //关总中断 *Ptr_SegAddr=Data;              //写16位字 while(FCTL3&BUSY);            //等待操作完成 _EINT();            //开总中断 FCTL1=FWKEY;              //退出写状态 FCTL3=FWKEY+LOCK;            //恢复锁定,保护数据 
//  FCTL3=FWKEY+LOCK+LOCKA;        //恢复锁定,保护数据(包括A段) return 1; 
} 
/********************************************************************************************* ********* 
* 名    称:Flash_Bak_WriteChar() 
* 功    能:不破坏段内其他数据,向Flash中写入一个字节(Char型变量) 
* 入口参数:Addr:存放数据的地址 Data:待写入的数据 
* 出口参数:返回出错信息  0:偏移溢出 ;1:正常工作 
* 范    例:Flash_Bak_WriteChar(0,123);将常数123写入0单元 Flash_Bak_WriteChar(1,a);将变量a写入1单元 
******************************************************************************************************/  
char Flash_Bak_WriteChar (unsigned char Addr,unsigned  char Data) 
{ unsigned  int temp=0; unsigned char  *Ptr_SegAddr;                      //Segment  pointer unsigned char BackupArray[64];        //开辟64字节的临时RAM备份Seg unsigned char  i = 0; //-----  段范围限定。为了内存管理安全,只允许本段操作----- if((SegPre<=31&&Addr>=512) || (SegPre>31&&Addr>64) ) return 0; for(i=0;i<64;i++) { temp=SegAddr+i; Ptr_SegAddr = (unsigned char *)temp;           //Initialize Flash  pointer BackupArray[i]=*Ptr_SegAddr;        //指针移位方法赋值 } Flash_Erase();                 //擦除待操作段 FCTL1 = FWKEY + WRT;                           //正常写入(非块写) FCTL3 = FWKEY ;                                 //解锁 
//  FCTL3 = FWKEY ;                                  //解锁(含A段) for (i=0; i<64; i++) { _DINT();              //关总中断 if(i==Addr) { temp=SegAddr+Addr; Ptr_SegAddr = (unsigned char *)temp;            //Initialize Flash  pointer *Ptr_SegAddr =Data;                               //写数据 while(FCTL3&BUSY);                           //等待写操作完成 } else { temp=SegAddr+i; Ptr_SegAddr = (unsigned char *)temp;               //Initialize Flash  pointer *Ptr_SegAddr = BackupArray[i];                   //恢复Flash内的其他数据 while(FCTL3&BUSY);                               //等待写操作完成 } _EINT();                  //开总中断 } FCTL1 = FWKEY;                                  //清除写 FCTL3 = FWKEY + LOCK;                            //上锁 
//     FCTL3 = FWKEY + LOCK;                            //上锁(含A段) return 1; 
} 
/******************************************************************************************************  
* 名    称:Flash_Bak_WriteWord() 
* 功    能:不破坏段内其他数据,向Flash中写入一个字(int型变量) 
* 入口参数:Addr:存放数据的地址,仍然是以字节为单位的偏移地址,需为偶数 Data:待写入的数据 
* 出口参数:返回出错信息  0:偏移溢出 ;1:正常工作 
* 说    明:MSP430单片机可以对16位数据直接操作,所以为了加快速度 
*              函数中均直接对word进行操作。 
* 范    例:Flash_Bak_WriteWord(0,123);将常数123写入0单元 Flash_Bak_WriteWord(1,a);将变量a写入1单元 
******************************************************************************************************/ 
char Flash_Bak_WriteWord(unsigned int Addr,unsigned  int Data) 
{ unsigned  int *Ptr_SegAddr;                          //Segment  pointer Ptr_SegAddr = (unsigned  int *)SegAddr;               //Initialize Flash  pointer //-----注意:以下操作数全部为word16位数据类型----- unsigned  int BackupArray[32];          //开辟32字(64字节)的临时RAM备份Seg unsigned  int i = 0; //-----  段范围限定。为了内存管理安全,只允许本段操作----- if((SegPre<=31&&Addr>=512) ||(SegPre>31&&Addr>64) ) return 0; for(i=0;i<32;i++)                 //word型占两个字节 { BackupArray[i]= *(Ptr_SegAddr+i);           //指针移位方法对字赋值 } Flash_Erase();                    //擦除待操作段 FCTL1 = FWKEY + WRT;                            //正常写入(非块写) FCTL3 = FWKEY;                                  //解锁 
//     FCTL3 = FWKEY+LOCKA;                           //解锁  (含A段) for (i=0; i<32; i++)                      //word型占两个字节,需跳过奇数地址 { _DINT();               //关总中断 if(i==Addr) { *(Ptr_SegAddr+Addr) =Data;                 //写字型数据 while(FCTL3&BUSY);                              //等待写操作完成 } else { *(Ptr_SegAddr+i)= BackupArray[i];                    //恢复Flash内的其他数据,按字恢复 while(FCTL3&BUSY);                               //等待写操作完成 } _EINT();                //开总中断 } FCTL1 = FWKEY;                                  //清除写 FCTL3 = FWKEY + LOCK;                            //上锁 
//  FCTL3 = FWKEY + LOCK+LOCKA;                     //上锁(含LOCKA) return 1; 
} 
/******************************************************************************************************  
* 名    称:Flash_SegA_ReadChar() 
* 功    能:从InfoA中读取一个字节 
* 入口参数:Addr:存放数据的偏移地址 
* 出口参数:Data:读回的数据;偏移溢出时,返回 0 
* 说    明:无 
* 范    例:无 
******************************************************************************************************/ 
unsigned char Flash_SegA_ReadChar  (unsigned  int Addr) 
{ unsigned  int temp=0; unsigned char Data=0; //-----  段范围限定。为了内存管理安全,只允许本段操作----- if(Addr>=64) return 0; unsigned char  *Ptr_SegAddr;            //Segment  pointer temp = 0x10c0+Addr; Ptr_SegAddr = (unsigned char *)temp;     //Initialize Flash  pointer Data=*Ptr_SegAddr;                  //直接为InfoA首地址,未使用全局变量SegAddr return(Data); 
} unsigned char gTemp=0;
unsigned char gA;
void main()
{WDTCTL = WDTPW + WDTHOLD; //关狗   DCOCTL = CALDCO_8MHZ;                  // 先设DCO为8MHz BCSCTL1 = CALBC1_8MHZ;                        BCSCTL2 |= DIVM_1+DIVS_2;              // 2分频后到MCLK为4MHz ,4分频后到SMCLK为2MHzFlash_Init(6,'B' );                   //6分频后为333.3kHz,   操作B信息段gTemp=Flash_ReadChar(0);              //读取Flash_Bak_WriteChar (0,0x23);         //写入gTemp=Flash_ReadChar(0);              //再读取gA=gTemp;}

这篇关于MSP430G2553 flash操作例子的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/849707

相关文章

什么是 Flash Attention

Flash Attention 是 由 Tri Dao 和 Dan Fu 等人在2022年的论文 FlashAttention: Fast and Memory-Efficient Exact Attention with IO-Awareness 中 提出的, 论文可以从 https://arxiv.org/abs/2205.14135 页面下载,点击 View PDF 就可以下载。 下面我

STM32内部闪存FLASH(内部ROM)、IAP

1 FLASH简介  1 利用程序存储器的剩余空间来保存掉电不丢失的用户数据 2 通过在程序中编程(IAP)实现程序的自我更新 (OTA) 3在线编程(ICP把整个程序都更新掉) 1 系统的Bootloader写死了,只能用串口下载到指定的位置,启动方式也不方便需要配置BOOT引脚触发启动  4 IAP(自己写的Bootloader,实现程序升级) 1 比如蓝牙转串口,

动手学深度学习【数据操作+数据预处理】

import osos.makedirs(os.path.join('.', 'data'), exist_ok=True)data_file = os.path.join('.', 'data', 'house_tiny.csv')with open(data_file, 'w') as f:f.write('NumRooms,Alley,Price\n') # 列名f.write('NA

线程的四种操作

所属专栏:Java学习        1. 线程的开启 start和run的区别: run:描述了线程要执行的任务,也可以称为线程的入口 start:调用系统函数,真正的在系统内核中创建线程(创建PCB,加入到链表中),此处的start会根据不同的系统,分别调用不同的api,创建好之后的线程,再单独去执行run(所以说,start的本质是调用系统api,系统的api

Java IO 操作——个人理解

之前一直Java的IO操作一知半解。今天看到一个便文章觉得很有道理( 原文章),记录一下。 首先,理解Java的IO操作到底操作的什么内容,过程又是怎么样子。          数据来源的操作: 来源有文件,网络数据。使用File类和Sockets等。这里操作的是数据本身,1,0结构。    File file = new File("path");   字

MySQL——表操作

目录 一、创建表 二、查看表 2.1 查看表中某成员的数据 2.2 查看整个表中的表成员 2.3 查看创建表时的句柄 三、修改表 alter 3.1 重命名 rename 3.2 新增一列 add 3.3 更改列属性 modify 3.4 更改列名称 change 3.5 删除某列 上一篇博客介绍了库的操作,接下来来看一下表的相关操作。 一、创建表 create

STM32 ADC+DMA导致写FLASH失败

最近用STM32G070系列的ADC+DMA采样时,遇到了一些小坑记录一下; 一、ADC+DMA采样时进入死循环; 解决方法:ADC-dma死循环问题_stm32 adc dma死机-CSDN博客 将ADC的DMA中断调整为最高,且增大ADCHAL_ADC_Start_DMA(&hadc1, (uint32_t*)adc_buffer, ADC_Buffer_Size); 的ADC_Bu

封装MySQL操作时Where条件语句的组织

在对数据库进行封装的过程中,条件语句应该是相对难以处理的,毕竟条件语句太过于多样性。 条件语句大致分为以下几种: 1、单一条件,比如:where id = 1; 2、多个条件,相互间关系统一。比如:where id > 10 and age > 20 and score < 60; 3、多个条件,相互间关系不统一。比如:where (id > 10 OR age > 20) AND sco

PHP7扩展开发之流操作

前言 啥是流操作?简单来讲就是对一些文件,网络的IO操作。PHP已经把这些IO操作,封装成流操作。这节,我们将使用PHP扩展实现一个目录遍历的功能。PHP示例代码如下: <?phpfunction list_dir($dir) {if (is_dir($dir) === false) {return;} $dh = opendir($dir);if ($dh == false) {ret

浙大数据结构:树的定义与操作

四种遍历 #include<iostream>#include<queue>using namespace std;typedef struct treenode *BinTree;typedef BinTree position;typedef int ElementType;struct treenode{ElementType data;BinTree left;BinTre