__attribute__((section(”name“)))是gcc编译器支持的一个编译特性(arm编译器也支持此特性),实现在编译时把某个函数/数据放到name的数据段中。
__attribute__主要用于改变所声明或定义的函数或数据的特性,它有很多子项,用于改变作用对象的特性。
比如对函数,noline将禁止进行内联扩展、noreturn表示没有返回值、pure表明函数除 返回值外,不会通过其它(如全局变量、指针)对函数外部产生任何影响。
当然,__attribute__肯定有很多的用法,今天就用到了section部分,所以就只针对这个做一些记录。
提到section,就得说RO RI ZI了,在ARM编译器编译之后,代码被划分为不同的段,RO Section(ReadOnly)中存放代码段和常量,RW Section(ReadWrite)中存放可读写静态变量和全局变量,ZI Section(ZeroInit)是存放在RW段中初始化为0的变量。
于是本文的大体意思就清晰了,__attribute__((section("xxxx"))),其作用是将作用的函数或数据放入指定名为"XXXX"对应的段中。
代码的编写部分主要分三部分,OSRAMCODE (OS 代码部分)、 全局变量、APPRAMCODE(APP 代码部分)
最简单了,因为你不做修改,你定义的全局变量默认都是在 RW 和 ZI 段,而这两段正好被我们分配到 了 BIN_OS 中。故而我们不需要操作什么
剩下的难题,我们怎么才能将指定的函数放进BIN_OS 和 BIN_APP 这两个区域呢?
实现方法如下
//指定函数放在 OSRAMCODE 也就是 BIN_OS 这一段中
#define __AT_OS __attribute__((used, section("OSRAMCODE")))
//指定函数放在 APPRAMCODE 也就是 BIN_APP 这一段中
#define __AT_APP __attribute__((used, section("APPRAMCODE")))
所以我们编写代码的时候,只要在函数面前加入我们定义好的宏即可,例如:
//OS代码
__AT_OS int os_fun()
{
return 1;
}
//APP的代码
__AT_APP int app_f()
{
return 1;
}
上面的方法就要求我们写的代码必须为每个函数指定位置,非常繁琐。
当然,还有更好的实现,那就是修改 .sct 链接脚本
具体就不细说了,大致如下,直接把 os 相关的 .c 文件指定到链接脚本中去。编译如果有错误就自己改下吧
BIN_OS为48K,只存放 OS的代码 和 +RW +ZI 也就是全局变量了。
RW_IRAM1 0x20000000 0x0000c000 {
os1.o(OSRAMCODE)
os2.o(OSRAMCODE)
os3.o(OSRAMCODE)
.ANY (+RW +ZI)
}
;BIN_APP 为16K,用于存放APP的函数的内容
RW_IRAM2 0x2000c000 0x00004000 {
*.o(APPRAMCODE)
}