• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            為生存而奔跑

               :: 首頁 :: 聯系 :: 聚合  :: 管理
              271 Posts :: 0 Stories :: 58 Comments :: 0 Trackbacks

            留言簿(5)

            我參與的團隊

            搜索

            •  

            積分與排名

            • 積分 - 330192
            • 排名 - 74

            最新評論

            閱讀排行榜

            評論排行榜

            list_entry的宏定義:
            #define list_entry(ptr, type, member) \ 
            ((type *)((char *)(ptr)-(unsigned long)(&((type *)0)->member)))
            這個倒是不難理解:從一個結構的成員指針找到其容器的指針。
            但是正因為如此,我的第一感覺是,這個宏的名字應該更加抽象,名字似乎應該改稱叫“尋找容器”一類的,查看list.h源代碼,發現現在的定義是這樣的:
            #define list_entry(ptr, type, member) \
                container_of(ptr, type, member)

            #define container_of(ptr, type, member)                 \
            ({                                                        \
                const typeof( ((type *)0)->member ) *__mptr = (ptr);\
                (type *)( (char *)__mptr - offsetof(type,member) ); \
            })

            #define offsetof(TYPE, MEMBER) ((size_t) &((TYPE *)0)->MEMBER)
            linux不想用C++,但又想利用C++的優點,如是出現了很多奇怪的宏,他們叫做trick。
            ptr是找容器的那個變量的指針,把它減去自己在容器中的偏移量的值就應該 得到容器的指針。(容器就是包含自己的那個結構)。指針的加減要注意類型,用(char*)ptr是為了計算字節偏移。((type *)0)->member是一個小技巧。自己理解吧。前面的(type *)再轉回容器的類型。
            =====================
            #define list_entry(ptr, type, member) \
                    ((type *)((char *)(ptr)-(unsigned long)(&((type *)0)->member)))

                ptr是指向list_head類型鏈表的指針,type為一個結構,而member為結構type中的一個域,類型為list_head,這個宏返回指向type結構的指針。在內核代碼中大量引用了這個宏,因此,搞清楚這個宏的含義和用法非常重要。

              
            設有如下結構體定義:
            typedef struct xxx
            {
                 ……(結構體中其他域,令其總大小為size1)
                 type1 member;
                 ……(結構體中其他域)
            }type;


            定義變量:
               type a;
               type * b;
               type1 * ptr;
            執行:
               ptr=&(a.member);
               b=list_entry(ptr,type,member);
            則可使b指向a,得到了a的地址

            如何做到的呢?

            先看&((type *)0)->member:
            把“0”強制轉化為指針類型,則該指針一定指向“0”(數據段基址)。因為指針是“type *”型的,所以可取到以“0”為基地址的一個type型變量member域的地址。那么這個地址也就等于member域到結構體基地址的偏移字節數。

             


            再來看 ((type *)((char *)(ptr)-(unsigned long)(&((type *)0)->member))):
            (char *)(ptr)使得指針的加減操作步長為一字節,(unsigned long)(&((type *)0)->member)等于ptr指向的member到該member所在結構體基地址的偏移字節數。二者一減便得出該結構體的地址。轉換為 (type *)型的指針,大功告成。

            ==============

             list_entry定義如下:

            /**
            * list_entry - get the struct for this entry
            * @ptr:        the &struct list_head pointer.
            * @type:        the type of the struct this is embedded in.
            * @member:        the name of the list_struct within the struct.
            */
            #define list_entry(ptr, type, member) \
                    ((type *)((char *)(ptr)-(unsigned long)(&((type *)0)->member)))
            ===================
            posted on 2011-01-27 16:45 baby-fly 閱讀(1172) 評論(0)  編輯 收藏 引用 所屬分類: Ubuntu&Linux
            人妻精品久久久久中文字幕| 久久99热这里只有精品国产| 中文字幕久久久久人妻| 欧美日韩中文字幕久久伊人| 久久91精品国产91久久小草| 久久精品国产一区二区电影| 伊人久久大香线焦AV综合影院| 老司机国内精品久久久久| 狠狠色丁香久久婷婷综合蜜芽五月| 亚洲国产美女精品久久久久∴ | 影音先锋女人AV鲁色资源网久久 | 久久亚洲国产精品五月天婷| 久久精品国产AV一区二区三区 | 99久久精品费精品国产| 久久精品中文字幕大胸| 久久福利青草精品资源站免费| 亚洲va久久久久| 久久久久亚洲精品中文字幕| 久久精品成人免费网站| 久久亚洲中文字幕精品一区| 久久久久人妻一区精品| 潮喷大喷水系列无码久久精品| 久久成人小视频| 狠狠色伊人久久精品综合网 | 精品国产91久久久久久久a| 人妻少妇久久中文字幕| 国产69精品久久久久久人妻精品| 久久精品无码一区二区三区免费| 久久无码专区国产精品发布| 久久久久99精品成人片三人毛片 | 久久99亚洲综合精品首页| 久久久久久a亚洲欧洲aⅴ| 狠狠色婷婷久久一区二区三区| 亚洲人成无码网站久久99热国产 | 亚洲伊人久久成综合人影院| 国产精品99久久久久久猫咪| 91久久九九无码成人网站| 国产精品久久久久久久久| 久久国产精品久久国产精品| 久久久久久综合一区中文字幕| 久久精品无码一区二区无码|