GitHub:https://github.com/RT-Thread/rt-thread
Gitee:https://gitee.com/rtthread/rt-thread?_from=gitee_search
歡迎star!
1、綜述
從4.1.0版本開始,RT-Thread在保證向前兼容的基礎上對原有的HOOK方式進行了改進,實現了如下的效果:
與原有使用函數指針進行“運行時刻”HOOK的方式兼容,依賴原有方式進行代碼插入或HOOK的模塊無需修改;
使用RT_USING_HOOK作為總開關
當未定義該宏時,將關閉所有HOOK功能,也不再生成任何相關代碼
當定義該宏時,在未開啟傳統“函數指針HOOK”時,默認情況下也不再生成額外代碼
允許用戶通過“插入宏的方式”在編譯時刻精細控制具體HOOK哪個位置
可以插入任意代碼塊——其中就包括函數指針、對普通函數的調用等等
2、使用方法
2.1 總開關
在rtconfig.h定義宏開關RT_USING_HOOK將開啟HOOK功能。默認情況下,在該宏未定義時,所有HOOK功能都將被強制關閉,不會生成任何代碼。
2.2 向前兼容
在4.1.0之前,RT-Thread提供了一套基于函數指針運行時注冊的HOOK機制。新版本中,在開啟HOOK功能的前提下,定義宏開關RT_HOOK_USING_FUNC_PTR將開啟與過去相同的基于函數指針運行時注冊的HOOK機制,實現向前兼容。
需要特別說明的是:
該機制的優點:在運行時刻允許用戶動態修改HOOK函數;
其缺點是:即便用戶并未使用對應的HOOK,該機制也將生成對應的代碼,并在運行時刻略微占用一些幾乎可以忽略不計的CPU周期數。
2.3 使用插入宏實現定點精細HOOK
無論用戶是否定義了宏RT_HOOK_USING_FUNC_PTR,新的HOOK機制都允許用戶“定點”、“精細”的對RT-Thread的任意錨點進行HOOK。具體方法為:在rtconfig.h(或其直接、間接包含的頭文件)中按照如下的格式定義插入宏:
#define __on_<錨點名稱>(<形參列表>) do { /* 你要插入的任意代碼 */ } while(0)
以內核的調度錨點rt_scheduler_hook為例,它的"等效"函數原型如下:
void __on_rt_scheduler_hook(struct rt_thread *from, struct rt_thread *to);
因此,我們只需要在rtconfig.h(或其直接、間接包含的頭文件)中定義如下的宏:
#define __on_rt_scheduler_hook(__from, __to) do { /* 你要插入的任意代碼 */ } while(0)
就可以將任意代碼插入到錨點rt_scheduler_hook在scheduler.c中的對應位置上。實際上,所有錨點的名稱都與過去函數指針的名稱相同。
除了插入代碼塊以外,一般插入宏會被用來將目標錨點直接替換為“對用戶指定函數的調用”,比如,在上述例子中,假設用戶想在調度器切換線程時調用一個指定的函數my_scheduler_notifier(),則可以修改插入宏為:
#define __on_rt_scheduler_hook(__from, __to) my_scheduler_notifier((__from), (__to))
這里存在一個明顯的問題,即,一般來說錨點所在的c源文件并不知道用戶目標函數的原型(function prototype),因此編譯器會報告"implicit function declaration" 一類的警告,甚至會將其視作編譯錯誤。為了避免這類情況的發生,我們可以在定義插入宏時順便引入對應的函數原型:
extern void my_scheduler_notifier(struct rt_thread *from, struct rt_thread *to); #define __on_rt_scheduler_hook(__from, __to) my_scheduler_notifier((__from), (__to))
2.4 注意事項
插入宏的方式優先級高于函數指針方式,即:當你同時使用兩種方法對同一個錨點進行HOOK時,插入宏的方式將覆蓋(Override)掉對應的函數指針HOOK。
RT-Thread一般不推薦用戶直接修改rtconfig.h也不推薦在其中添加額外的代碼:比如用戶函數的原型等等。為了解決這一問題,通常有兩個方案:
在rtconfig.h中包含一個專門存放用戶HOOK的頭文件,比如:
//! rtconfg.h ... #include “user_hook.h”
在編譯選項中以全局頭文件包含的形式將專門存放用戶HOOK的頭文件引用進來,比如:
在gcc、clang和arm compiler 6中使用-include "<頭文件>"的方式:
-include "user_hook.h"
在arm compiler 5中使用--preinclude=頭文件的方式:
--preinclude="user_hook.h"
-
內核
+關注
關注
3文章
1363瀏覽量
40228 -
RT-Thread
+關注
關注
31文章
1272瀏覽量
39922 -
函數指針
+關注
關注
2文章
56瀏覽量
3775
原文標題:聊聊新版RT-Thread內核中的鉤子
文章出處:【微信號:RTThread,微信公眾號:RTThread物聯網操作系統】歡迎添加關注!文章轉載請注明出處。
發布評論請先 登錄
相關推薦
評論