__attribute__ 指定函数、变量到某个特定的section段

我们先来看一个业务逻辑需求:

假设项目需要把RAM分2个bin区域,    bin1负责OS,以及一些驱动,BIN2负责相对业务,我想只让BIN1可定义全局变量,BIN2可以使用,但是不可以定义全局变量。

该如何实现?

__attribute__ 和 section

__attribute__这个关键词是GNU编译器中的编译属性,ARM编译器也支持这个用法。__attribute__主要用于改变所声明或定义的函数或 数据的特性,它有很多子项,用于改变作用对象的特性。

__attribute__肯定有很多的用法,今天就用到了section部分,所以就只针对这个做一些记录。

提到section,就得说RO RI ZI了,在ARM编译器编译之后,代码被划分为不同的段,RO Section(ReadOnly)中存放代码段和常量,RW Section(ReadWrite)中存放可读写静态变量和全局变量,ZI Section(ZeroInit)是存放在RW段中初始化为0的变量。
 

链接脚本

众所周知,我们所有的全局变量都是在  RW ZI两个段中。(具体可以百度下为什么)

那为了完成我们的前面提到的问题,我们就可以重新规划我们的 .sct 链接脚本。

我们把 RAM 分为两个区域    BIN_OS 和 BIN_APP 

其中  BIN_OS 用来存放我们的全局变量 和 OS 的代码:

;BIN_OS为48K,只存放 OS的代码 和 +RW +ZI 也就是全局变量了。
 
RW_IRAM1 0x20000000 0x0000c000 {

*.o(OSRAMCODE)
.ANY (+RW +ZI)

}
 
 
 
;BIN_APP 为16K,用于存放APP的函数的内容
 
RW_IRAM2 0x2000c000 0x00004000 {

*.o(APPRAMCODE)

}
 

注意看:BIN_OS  和 BIN_APP

关于 BIN_OS:

*.o(OSRAMCODE) —— 存放OS的代码 
.ANY (+RW +ZI)——存放 RW 和 ZI ,其实也就是全局变量了

 

关于 BIN_APP :

*.o(APPRAMCODE) ——只有 APP 的代码部分,没有 全局变量

 

好了,那这里我们就很清楚了我们的内存架构大致如下:

__attribute__ 指定函数、变量到某个特定的section段_第1张图片

 

如何编写代码?

代码的编写部分主要分三部分,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)

}

 

你可能感兴趣的:(文章杂谈)