在linux 內核編程中,會(huì )經(jīng)常見(jiàn)到一個(gè)宏函數container_of(ptr,type,member), 但是當你通過(guò)追蹤源碼時(shí),像我們這樣的一般人就會(huì )絕望了(這一堆都是什么呀? 函數還可以這樣定義??? 怎么還有0呢??? 哎,算了,還是放棄吧。。。)。 這就是內核大佬們厲害的地方,隨便兩行代碼就讓我們懷疑人生,凡是都需要一個(gè)過(guò)程,慢慢來(lái)吧。
其實(shí),原理很簡(jiǎn)單: 已知結構體type的成員member的地址ptr,求解結構體type的起始地址。
type的起始地址 = ptr - size (這里需要都轉換為char *,因為它為單位字節)。
到此,該函數已經(jīng)講完,是不是很簡(jiǎn)單??? 其實(shí)也不是,這里并沒(méi)有提到size如何計算,而令我們頭暈的正是這里。
好吧,先上container of函數原型:
#define container_of(ptr, type, member) ({ \ const typeof( ((type *)0)->member ) *__mptr = (ptr); \ (type *)( (char *)__mptr - offsetof(type,member) );})
其次為 offserof 函數原型:
#define offsetof(TYPE, MEMBER) ((size_t) &((TYPE *)0)->MEMBER)
怎么樣,是不是很炫? 好吧,下面開(kāi)始揭開(kāi)面紗:
讓事實(shí)說(shuō)話(huà):
#include<stdio.h> struct test { char i ; int j; char k; }; int main() { struct test temp; printf("&temp = %p\n",&temp); printf("&temp.k = %p\n",&temp.k); printf("&((struct test *)0)->k = %d\n",((int)&((struct test *)0)->k)); }
編譯運行,可以得到如下結果:
&temp = 0xbf9815b4
&temp.k = 0xbf9815bc
&((struct test *)0)->k = 8
什么意思看到了吧,自定義的結構體有三個(gè)變量:i,j,k。 因為有字節對齊要求,所以該結構體大小為4bytes * 3 =12 bytes. 而&((struct test *)0)->k 的作用就是求 k到結構體temp起始地址的字節數大?。ň褪俏覀兊膕ize)。在這里0被強制轉化為struct test *型, 它的作用就是作為指向該結構體起始地址的指針,就是作為指向該結構體起始地址的指針,就是作為指向該結構體起始地址的指針, 而&((struct test *)0)->k 的作用便是求k到該起始指針的字節數。。。其實(shí)是求相對地址,起始地址為0,則&k的值便是size大?。ㄗⅲ捍蛴r(shí)因為需要整型,所以有個(gè)int強轉)所以我們便可以求我們需要的 size 了 。 好吧,一不小心把 offsetof() 函數的功能給講完了:::
#define offsetof(TYPE, MEMBER) ((size_t) &((TYPE *)0)->MEMBER)
這次再看就順眼了吧(底層為什么是這樣我還是不懂。。。只知道這樣確實(shí)可以) , 所以offsetof()的作用就是求我們夢(mèng)寐以求的size, 并以size_t形式返回(size_t: 無(wú)符號整型)。
#define container_of(ptr, type, member) ({ \ const typeof( ((type *)0)->member ) *__mptr = (ptr); \ (type *)( (char *)__mptr - offsetof(type,member) );})
這里我們只看第二行:
const typeof( ((type *)0)->member ) *__mptr = (ptr);
它的作用是什么呢? 其實(shí)沒(méi)什么作用(勿噴勿噴,讓我把話(huà)說(shuō)完),但就形式而言 _mptr = ptr, 那為什么要要定義一個(gè)一樣的變量呢??? 其實(shí)這正是內核人員的牛逼之處:如果開(kāi)發(fā)者使用時(shí)輸入的參數有問(wèn)題:ptr與member類(lèi)型不匹配,編譯時(shí)便會(huì )有warnning, 但是如果去掉改行,那個(gè)就沒(méi)有了,而這個(gè)警告恰恰是必須的(防止出錯有不知道錯誤在哪里)。。。這嚴謹性可以吧
typeof( ((type *)0)->member )
它的作用是獲取member的類(lèi)型僅此而已。至此基本結束
container_of(ptr, type,member)函數的實(shí)現包括兩部分:
1. 判斷ptr 與 member 是否為同意類(lèi)型
2. 計算size大小,結構體的起始地址 = (type *)((char *)ptr - size) (注:強轉為該結構體指針)
現在我們知道container_of()的作用就是通過(guò)一個(gè)結構變量中一個(gè)成員的地址找到這個(gè)結構體變量的首地址。
container_of(ptr,type,member),這里面有ptr,type,member分別代表指針、類(lèi)型、成員。
到此這篇關(guān)于linux內核編程container of()函數的文章就介紹到這了,更多相關(guān)linux container of()函數 內容請搜索腳本之家以前的文章或繼續瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
免責聲明:本站發(fā)布的內容(圖片、視頻和文字)以原創(chuàng )、來(lái)自本網(wǎng)站內容采集于網(wǎng)絡(luò )互聯(lián)網(wǎng)轉載等其它媒體和分享為主,內容觀(guān)點(diǎn)不代表本網(wǎng)站立場(chǎng),如侵犯了原作者的版權,請告知一經(jīng)查實(shí),將立刻刪除涉嫌侵權內容,聯(lián)系我們QQ:712375056,同時(shí)歡迎投稿傳遞力量。
Copyright ? 2009-2022 56dr.com. All Rights Reserved. 特網(wǎng)科技 特網(wǎng)云 版權所有 特網(wǎng)科技 粵ICP備16109289號
域名注冊服務(wù)機構:阿里云計算有限公司(萬(wàn)網(wǎng)) 域名服務(wù)機構:煙臺帝思普網(wǎng)絡(luò )科技有限公司(DNSPod) CDN服務(wù):阿里云計算有限公司 百度云 中國互聯(lián)網(wǎng)舉報中心 增值電信業(yè)務(wù)經(jīng)營(yíng)許可證B2
建議您使用Chrome、Firefox、Edge、IE10及以上版本和360等主流瀏覽器瀏覽本網(wǎng)站