1、section的作用
section主要作用是將函數或者變量放在指定段中,這樣就可在指定的位置取出。
//section demo with gcc #include"stdio.h" int__attribute__((section("my_fun")))test1(inta,intb) { return(a+b); } inttest(intb) { return2*b; } int__attribute__((section("my_fun")))test0(inta,intb) { return(a*b); } int__attribute__((section("my_val")))chengi; int__attribute__((section("my_val")))chengj; intmain(void) { intsum,c,j; chengi=1,chengj=2; sum=test1(chengi,chengj); c=test(100); j=test0(chengi,chengj); printf("sum=%d,c=%d,j=%d ",sum,c,j); return0; }
編譯生成map文件:
gcc-o main.exe main.c-Wl,-Map,my_test.map
my_test.map 文件片段如下:
.text0x004014600xa0C:\Users\think\ccmGLaeH.o 0x00401460test 0x0040146amain .text0x004015000x0c:/mingw/bin/../libmingw32.a(CRTglob.o) ...... my_fun0x004040000x200 [!provide]PROVIDE(___start_my_fun,.) my_fun0x004040000x1cC:\Users\think\ccmGLaeH.o 0x00404000test1 0x0040400dtest0 [!provide]PROVIDE(___stop_my_fun,.) .data0x004050000x200 0x00405000\__data_start_\_=. ...... *(.data_cygwin_nocopy) my_val0x004060000x200 [!provide]PROVIDE(___start_my_val,.) my_val0x004060000x8C:\Users\think\ccdMcTrl.o 0x00406000chengi 0x00406004chengj [!provide]PROVIDE(___stop_my_val,.) .rdata0x004070000x400
分析可見,使用section修飾的函數和變量在自定義的片段,而且是連續存放在___start_xx到___stop_xx之間,這樣可根據變量的地址得出與其同段變量的地址,為后續自動初始化等功能提供了基礎。
2、 自動初始化
基于前面section的作用,可以將同類函數指針全部使用同一個段名修飾,然后開機后系統自動檢索段內函數指針,逐個執行,對上層應用就是無需主動調用,系統自動初始化。
考慮到硬件初始化與應用功能初始化的先后順序,可以對段名進行分配,map文件按段名排序。自動初始化主體是OS_INIT_EXPORT宏。
范例代碼出自中國移動的oneos開源版本,使用gcc,方案和國產RT-Thread類似。
typedefos_err_t(*os_init_fn_t)(void); #defineOS_INIT_EXPORT(fn,level) const os_init_fn_t __os_call_##fn OS_SECTION(".init_call."level)=fn #defineOS_BOARD_INIT(fn)OS_INIT_EXPORT(fn,"1") #defineOS_PREV_INIT(fn)OS_INIT_EXPORT(fn,"2") #defineOS_DEVICE_INIT(fn)OS_INIT_EXPORT(fn,"3") #defineOS_CMPOENT_INIT(fn)OS_INIT_EXPORT(fn,"4") #defineOS_ENV_INIT(fn)OS_INIT_EXPORT(fn,"5") #defineOS_APP_INIT(fn)OS_INIT_EXPORT(fn,"6")
例如shell初始化函數,定義如下:
OS_APP_INIT(sh_system_init);
將宏定義展開:
/*含義是函數指針__os_call_sh_system_init *其指向sh_system_init函數,且該指針編譯后放在".init_call.6"段 */ constos_init_fn_t__os_call_sh_system_init __attribute__((section((".init_call.6"))))=sh_system_init
系統自身也有自定義函數,用來標記起止點函數。
OS_INIT_EXPORT(os_init_start,"0");//段起點__start OS_INIT_EXPORT(os_board_init_start,"0.end"); OS_INIT_EXPORT(os_board_init_end,"1.end"); OS_INIT_EXPORT(os_init_end,"6.end");//段終點__stop
最終生成的map文件,如下圖所示:
//系統底層在合適的時機調用如下兩函數,將指定段區間內的所有函數自動執行 voidos_board_auto_init(void) { constos_init_fn_t*fn_ptr_board_init_start; constos_init_fn_t*fn_ptr_board_init_end; constos_init_fn_t*fn_ptr; fn_ptr_board_init_start=&__os_call_os_board_init_start+1; fn_ptr_board_init_end=&__os_call_os_board_init_end-1; //將段首尾區間內的函數全部遍歷執行 for(fn_ptr=fn_ptr_board_init_start;fn_ptr<=?fn_ptr_board_init_end;?fn_ptr++) ????{ ????????(void)(*fn_ptr)(); ????} ????return; } static?void?os_other_auto_init(void) { ????const?os_init_fn_t?*fn_ptr_other_init_start; ????const?os_init_fn_t?*fn_ptr_other_init_end; ????const?os_init_fn_t?*fn_ptr; ????fn_ptr_other_init_start?=?&__os_call_os_board_init_end?+?1; ????fn_ptr_other_init_end???=?&__os_call_os_init_end?-?1; ????for?(fn_ptr?=?fn_ptr_other_init_start;?fn_ptr?<=?fn_ptr_other_init_end;?fn_ptr++) ????{ ????????(void)(*fn_ptr)(); ????} ????return; }
系統執行os_other_auto_init時實現了sh_system_init的自動執行,即使應用層沒有顯示的去調用它。使用符號段的方式實現初始化函數自動執行,應用層修改軟件,增加功能啟動或者裁剪,對底層代碼無需任何改動。
注意:段中函數類型都是一樣的,范例是同一類函數指針,也可以是結構體,需要確保每個成員占用空間大小相同,這樣才能逐個遍歷。
3、總結
不同編譯器對section屬性的定義略有差異,但效果相同。
/*Compiler Related Definitions*/ #ifdefined(__CC_ARM)||defined(__CLANG_ARM) /*ARM Compiler*/ #defineSECTION(x)__attribute__((section(x))) #elifdefined(__IAR_SYSTEMS_ICC__)/*for IAR Compiler*/ #defineSECTION(x)@x #elifdefined(__GNUC__)/*GNU GCC Compiler*/ #defineSECTION(x)__attribute__((section(x))) #elifdefined(__ADSPBLACKFIN__)/*for VisualDSP++Compiler*/ #defineSECTION(x)__attribute__((section(x))) #elifdefined(_MSC_VER) #defineSECTION(x) #elifdefined(__TI_COMPILER_VERSION__) /* *The way that TI compiler set section is different from other(at least *GCC and MDK)compilers.See ARM Optimizing C/C++Compiler 5.9.3 for more *details. */ #defineSECTION(x) #else #errornot supported tool chain #endif
上面的#error也是個應用技巧,配搭#if / #else / #endif在編譯階段即可發現代碼問題,一般用于判斷宏定義的配置是否在預期之外,編譯報錯必須修改。
配合C關鍵字,對代碼的安全校驗、擴展移植都會有很好的效果。對小型項目、個人獨立開發看不出效果,但對復雜的多人合作的項目,合適的關鍵字對代碼的穩定性和架構是錦上添花。
審核編輯:劉清
-
C語言
+關注
關注
180文章
7561瀏覽量
133239 -
RT-Thread
+關注
關注
31文章
1225瀏覽量
39220 -
gcc編譯器
+關注
關注
0文章
78瀏覽量
3288
原文標題:C語言中section關鍵字的實際作用
文章出處:【微信號:玩點嵌入式,微信公眾號:玩點嵌入式】歡迎添加關注!文章轉載請注明出處。
發布評論請先 登錄
相關推薦
評論