首页 > 技术知识 > 正文

今天做的是读写Flash,因为最后从红外接收学习到的编码需要存入Flash中,所以读写Flash是一个很重要的模块! 然而新唐的例程中没有对数据Flash的读写,只有对LDROM 和 APROM的读写,其实差不了太多,为了不给以后的同仁多走弯路,这里特地写下笔者的经验,供大家参考! 这个板子的数据Flash大小是4KB,地址从0x0001f000 到 0x0001ffff。而笔者一开始是这样写的:

define unsigned int uint32_t uint32_t *flash; flash = (uint32_t *)0x0001f000; *flash = 0x55555555;

本打算往起始地址写0x55555555的值,结果执行的这里的时候,发生了硬件异常,进入了硬件异常处理函数里面去了! 这里要注意要往Flash里面写东西,是要讲究很高的安全性的,如果没有对相应的寄存器进行设置,那么一定会发生硬件异常! 那么怎么样才能正常的写入呢?记住一点,一定要对扇区进行擦除然后再写入,笔者一开始没有擦除直接写入,写不成功! 如下是笔者写的函数,作用就是往特定的地址里面写入值:

void writetoFlash(uint32_t offset,uint32_t date) {/*要先擦除才能写*/ int i; Un_Lock_Reg(); //往FLash写入的时候,是通过系统ISP写入,所以要先写入三个值,有例程的朋友们可以明白 ISPCON |= ISPEN; //使能ISP功能 ISPCMD = PAGE_ERASE; //选择为擦除模式 for(i = 0; i < 4096/512; i++) //4KB= 4096B,擦除的时候,系统规定页大小为512字节 { ISPADR = 0x0001f000 + i * 512; ISPTRG |= ISPGO; //触发执行ISP while((ISPTRG&ISPGO) == ISPGO);//等待ISP 执行结束 } ISPCMD = PROGRAM; //设置为编程模式 ISPADR = DFBADR + (offset << 2);//得到要写的地址,等同于0x0001f000+ 4*offset,offset 就是第几个值的意思 ISPDAT = date; //往数据寄存器中写入值 ISPTRG |= ISPGO; //即将执行 while((ISPTRG&ISPGO) == ISPGO); Un_Lock_Reg(); ISPCON &= ~ISPEN;//禁止ISP功能 } 然后是读出函数: uint32_t readfromFlash(uint32_t offset) { uint32_t data; Un_Lock_Reg(); //功能如上 ISPCON |= ISPEN; //使能ISP功能 ISPCMD = READ; //设定为读模式 ISPADR = DFBADR + (offset << 2);//得到要读的地址 data = ISPDAT; //得到数据 Un_Lock_Reg(); ISPCON &= ~ISPEN;//禁止ISP功能 return data; }
<

总结一下,就是首先要使能ISP功能,然后设置模式,然后给出要操作的地址,然后操作数据,最后在禁止ISP功能,但是一定要先擦除才能写!

原文链接:https://blog.csdn.net/iteye_2636/article/details/82090762?ops_request_misc=%257B%2522request%255Fid%2522%253A%2522159729012519195265905329%2522%252C%2522scm%2522%253A%252220140713.130102334.pc%255Fblog.%2522%257D&request_id=159729012519195265905329&biz_id=0&utm_medium=distribute.pc_search_result.none-task-blog-2~blog~first_rank_v1~rank_blog_v1-11-82090762.pc_v1_rank_blog_v1&utm_term=%E6%96%B0%E5%94%90&spm=1018.2118.3001.4187

猜你喜欢