Questions:AT32 部分型号有零等待闪存和非零等待闪存,程序在零等待闪存执行速度比在非零等待闪存执行速度快,如果有函数对执行速度有要求,可以将该函数加载到零等待区执行。当零等待闪存使用完后,如果还有函数对执行速度有要求,可以将该函数加载到 SRAM 执行,前提是SRAM 还有足够的空间存放该函数代码。
Answer: 有两种将某个函数加载到 Flash 或 SRAM 指定地址执行的方法,以 AT32F403A 为例:
1. 方法一:修改分散加载描述文件
在 Keil 中:修改分散加载描述文件*.sct(scatter file),在工程选项的 linker 页面中,选择*.sct 文件,编辑*.sct。
如果是要放在Flash零等待区指定地址,添加*.o(ZWROMCODE),自定义一个叫做ZWROMCODE的section。LR_IROM1 为零等待区,LR_IROM2 为非零等待区,设置如下:
; *************************************************************
; *** Scatter-Loading Description File generated by uVision ***
; *************************************************************
LR_IROM1 0x08000000 0x00020000 { ; load region size_region
ER_IROM1 0x08000000 0x00020000 { ; load address = execution address
*.o (RESET, +First)
*(InRoot$Sections)
.ANY (+RO)
*.o(ZWROMCODE)
}
RW_IRAM1 0x20000000 0x00038000 { ; RW data
.ANY (+RW +ZI)
}
}
LR_IROM2 0x08020000 0x000E0000 { ; load region size_region
ER_IROM2 0x08020000 0x000E0000 { ; load address = execution address
.ANY (+RO)
}
}
也可以根据需要单独分配一块零等待区 LR_IROM2,添加*.o(ZWROMCODE),自定义一个叫做ZWROMCODE 的 section。LR_IROM1 和 LR_IROM2 为零等待区,LR_IROM3 为非零等待区,设置如下:
; *************************************************************
; *** Scatter-Loading Description File generated by uVision ***
; *************************************************************
LR_IROM1 0x08000000 0x00010000 { ; load region size_region
ER_IROM1 0x08000000 0x00010000 { ; load address = execution address
*.o (RESET, +First)
*(InRoot$Sections)
.ANY (+RO)
}
RW_IRAM1 0x20000000 0x00038000 { ; RW data
.ANY (+RW +ZI)
}
}
LR_IROM2 0x08010000 0x00010000 { ; load region size_region
ER_IROM2 0x08010000 0x00010000 { ; load address = execution address ZW
*.o(ZWROMCODE)
}
}
LR_IROM3 0x08020000 0x000E0000 { ; load region size_region
ER_IROM3 0x08020000 0x000E0000 { ; load address = execution address NZW
.ANY (+RO)
}
}
如果是要放在 SRAM 指定地址,根据需要分配 RW_IRAM2 大小,添加*.o(RAMCODE),自定义一个叫做RAMCODE 的 section。设置如下,以 SRAM 扩大到 224KB 为例,RW_IRAM1 大小 196KB,RW_IRAM2大小 28KB。
复制; *************************************************************
; *** Scatter-Loading Description File generated by uVision ***
; *************************************************************
LR_IROM1 0x08000000 0x000C0000 { ; load region size_region
ER_IROM1 0x08000000 0x000C0000 { ; load address = execution address
*.o (RESET, +First)
*(InRoot$Sections)
.ANY (+RO)
}
RW_IRAM1 0x20000000 0x00031000 { ; RW data
.ANY (+RW +ZI)
}
RW_IRAM2 0x20031000 0x00007000 { ; RW data
*.o(RAMCODE)
}
}
在需要加载到 Flash 或 SRAM 指定地址中的函数前,用__attribute__((section("SECTION_NAME")))声明该函数放在该 section 中,SECTION_NAME 可以自行命名,如下面是使用 RAMCODE。如果有多个函数,那么每个函数前面都需要加__attribute__((section("RAMCODE")))。设置如下:
__attribute__((section("RAMCODE")))
void delay(unsigned char num)
{
while(num)
{
num--;
}
}
__attribute__((section("RAMCODE")))
void test(void)
{
for(;;)
{
led_gpio_port[led]->odt ^= led_gpio_pin[led];
delay(10);
}
}
也可以将所有需要加载的函数放在以#pragma arm section code = “RAMCODE” 开头,以#pragma arm section 结尾的中间,声明所有函数放在 RAMCODE section 中。设置如下:
#pragma arm section code = “RAMCODE"
void delay(unsigned char num)
{
while(num)
{
num--;
}
}
void test(void)
{
for(;;)
{
led_gpio_port[led]->odt ^= led_gpio_pin[led];
delay(10);
}
}
#pragma arm section