请教:擦除了一个函数,重启就不再工作了

2020-01-01 17:41发布

随便写了一个很简单的函数,将其编译在一个特定的页
1.此页只有它,无其它内容;
2.板子重启后,也不会被调用(特定条件才会被执行)

用烧写软件,将此页删除,程序立即死掉(可以理解)

断电重启,发现板子还是死掉的,这一点很难理解,大家有什么见解?谢谢
友情提示: 此问题已得到解决,问题已经关闭,关闭后问题禁止继续编辑,回答。
该问题目前已经被作者或者管理员关闭, 无法添加新回复
40条回答
zchong
1楼-- · 2020-01-04 22:46
要学会调试,找到hardfault的原因,结果就很明朗
guzhongqi
2楼-- · 2020-01-05 02:43
是不是放在最后一页了?
MDK编译的时候会在固件结尾处添加一些有用的信息,这些信息丢失的话程序就不能执行了,所以如果你要在指定位置通过编译的方式存储数据的话,必须是完整一页的并且不能是最后一页,这样即使擦除这页也不会把编译器添加的有用信息擦除掉。
popo_new
3楼-- · 2020-01-05 07:00
 精彩回答 2  元偷偷看……
落叶随风
4楼-- · 2020-01-05 12:12
楼主,我不知道你擦除的那个程序有多复杂,我自己做了个简单的测试,擦除后重启也没问题的,不知道是不是太简单的关系。

下面是简单的测试程序

  1. #include <stm32f10x.h>

  2. void func_test(void) __attribute__((section("test")));

  3. void delay(void)
  4. {
  5.     int time = 0xfffff;
  6.     while(time-- > 0);
  7. }

  8. int main(void)
  9. {
  10.     RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC | RCC_APB2Periph_GPIOD, ENABLE);
  11.    
  12.     GPIO_InitTypeDef GPIO_InitStructure;
  13.     GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
  14.     GPIO_InitStructure.GPIO_Pin = GPIO_Pin_12;
  15.     GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  16.     GPIO_Init(GPIOC, &GPIO_InitStructure);
  17.    
  18.     GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2;
  19.     GPIO_Init(GPIOD, &GPIO_InitStructure);
  20.     while(1)
  21.     {
  22.         GPIO_SetBits(GPIOC, GPIO_Pin_12);
  23.         delay();
  24.         GPIO_ResetBits(GPIOC, GPIO_Pin_12);
  25.         delay();
  26.         if((*(uint32_t *)func_test) != 0xffffffff)
  27.         {
  28.             func_test();
  29.             RCC_HSICmd(ENABLE);
  30.             FLASH_Unlock();
  31.             FLASH_ClearFlag(FLASH_FLAG_EOP|FLASH_FLAG_PGERR|FLASH_FLAG_WRPRTERR);
  32.             FLASH_ErasePage((uint32_t)func_test);                                                                        
  33.             FLASH_Lock();
  34.         }
  35.     }
  36. }

  37. void func_test(void)
  38. {
  39.     GPIO_SetBits(GPIOD, GPIO_Pin_2);
  40.     delay();
  41.     GPIO_ResetBits(GPIOD, GPIO_Pin_2);
  42.     delay();
  43. }
复制代码

scatter文件

  1. ; *************************************************************
  2. ; *** Scatter-Loading Description File generated by uVision ***
  3. ; *************************************************************

  4. LR_IROM1 0x08000000   {    ; load region size_region
  5.   ER_IROM1 AlignExpr(+0, 0x0800)  {  ; load address = execution address
  6.    *.o (RESET, +First)
  7.    *(InRoot$Sections)
  8.    .ANY (+RO)
  9.   }
  10.   RW_IRAM1 0x20000000 0x00010000  {  ; RW data
  11.    .ANY (+RW +ZI)
  12.   }
  13. }

  14. LR_IROM2 AlignExpr(+0, 0x0800) 0x0800
  15. {
  16.         ER_IROM2 +0
  17.         {
  18.                 *(test)
  19.         }
  20. }
复制代码

看具体的FLASH数据

memory1_擦除后.jpg (160.98 KB, 下载次数: 0)

下载附件

擦除后

2017-1-7 20:43 上传


popo_new
5楼-- · 2020-01-05 17:20
落叶随风 发表于 2017-1-7 20:44
楼主,我不知道你擦除的那个程序有多复杂,我自己做了个简单的测试,擦除后重启也没问题的,不知道是不是太 ...

.sct文件的语法,一直不太懂,假如以你的文件为例,希望将func_test编译到0x0803E000处,应该如何修改;

附我的sct(绝对定位成功,就是删除这段flash后,会进入hardfalut中断)

绝对定位的函数:void sec01(void)__attribute__((section("sec01")));

; *************************************************************
; *** Scatter-Loading Description File generated by uVision ***
; *************************************************************
LR_IROM1 0x08000000 0x00040000  {    ; load region size_region
  ER_IROM1 0x08000000 0x00040000  {  ; load address = execution address
   *.o (RESET, +First)
   *(InRoot$$Sections)
   .ANY (+RO)
  }
     sec01 0x0803E000 FIXED 0xffff   
   {
      sec01.o (sec01)     
    }
  RW_IRAM1 0x20000000 0x0000C000  {  ; RW data
   .ANY (+RW +ZI)
  }
}
落叶随风
6楼-- · 2020-01-05 21:45
这是因为你sct文件中分区分错了

你的sct文件中这一段

  1. sec01 0x0803E000 FIXED 0xffff   
  2.    {
  3.       sec01.o (sec01)     
  4.     }
  5.   RW_IRAM1 0x20000000 0x0000C000  {  ; RW data
  6.    .ANY (+RW +ZI)
  7.   }
复制代码

sec01确实在0x0803E000处了,单紧接着就是RW_IRAM1的数据了,

虽然这段数据运行时是在0x20000000处的,但存储地址是紧接着你

自己写的sec01后面,第一次上电后擦除了一页,同时也把RW_RAM1段

也擦除了,所以就进HardFault了。

sct文件应该这么写:

  1. ; *************************************************************
  2. ; *** Scatter-Loading Description File generated by uVision ***
  3. ; *************************************************************

  4. LR_IROM1 0x08000000  {    ; load region size_region
  5.   ER_IROM1 AlignExpr(+0, 0x0800)  {  ; load address = execution address
  6.    *.o (RESET, +First)
  7.    *(InRoot$Sections)
  8.    .ANY (+RO)
  9.   }
  10.   RW_IRAM1 0x20000000 0x00010000  {  ; RW data
  11.    .ANY (+RW +ZI)
  12.   }
  13. }

  14. LR_IROM2 AlignExpr(0x0803E000, 0x0800) 0x0800
  15. {
  16. ER_IROM2 AlignExpr(+0, 0x0800)
  17. {
  18.   *(test)
  19. }
  20. }
复制代码

LR_IROM1就不去动它了,另起一段LR_IROM2,当然这个名字可以自己定。

AlignExpr(0x0803E000, 0x0800)  表示:起始地址0x0803E000,按0x0800字节对齐

紧接着的0x0800 表示: 该区最大空间0x0800字节

里面的ER_IROM2 AlignExpr(+0, 0x0800)  这样写表示 execution address = load address,

就是那个+0,指偏移,相对与LR_IROM1指定的 load address,我们指定了0x0803E000。

*(test):这个section名字叫 test,* 表示任意的内容,也可以指定xxx.o什么的,不详述。

然后函数指定到这个section

  1. void func_test(void) __attribute__((section("test")));
复制代码

编译好后看map文件

  1. ==============================================================================

  2. Memory Map of the image

  3.   Image Entry point : 0x08000131

  4.   Load Region LR_IROM1 (Base: 0x08000000, Size: 0x0000067c, Max: 0xffffffff, ABSOLUTE)

  5.     Execution Region ER_IROM1 (Base: 0x08000000, Size: 0x0000067c, Max: 0xffffffff, ABSOLUTE)

  6.     Base Addr    Size         Type   Attr      Idx    E Section Name        Object

  7.     (略...)


  8.     Execution Region RW_IRAM1 (Base: 0x20000000, Size: 0x00000660, Max: 0x00010000, ABSOLUTE)

  9.     Base Addr    Size         Type   Attr      Idx    E Section Name        Object

  10.     (略...)


  11.   Load Region LR_IROM2 (Base: 0x0803e000, Size: 0x00000020, Max: 0x00000800, ABSOLUTE)

  12.     Execution Region ER_IROM2 (Base: 0x0803e000, Size: 0x00000020, Max: 0xffffffff, ABSOLUTE)

  13.     Base Addr    Size         Type   Attr      Idx    E Section Name        Object

  14.     0x0803e000   0x00000020   Code   RO            3    test                main.o
复制代码

可以看到 LR_IROM2 位置在Base: 0x0803e000,大小Size: 0x00000020,最大 Max: 0x00000800


我自己理解也不是很充分,可能还有不少疏漏的地方,还烦请指出。



一周热门 更多>