• <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精品国产麻豆| 久久久久国产精品麻豆AR影院 | 午夜天堂av天堂久久久| 国产精品久久久久…| 久久国产三级无码一区二区| 思思久久99热只有频精品66| 国产精品久久久久久| 久久久久亚洲国产| 久久综合九色综合欧美狠狠| 久久久久亚洲国产| 久久99精品久久久久久野外| 亚洲伊人久久精品影院| 91精品观看91久久久久久| 亚洲精品tv久久久久久久久 | 久久国产午夜精品一区二区三区| 2019久久久高清456| 久久亚洲高清综合| 麻豆精品久久久一区二区| 狠狠色婷婷久久综合频道日韩| 99久久精品国产一区二区三区| 精品国产乱码久久久久久1区2区| 久久无码国产| 亚洲精品视频久久久| 久久国产成人午夜AV影院| 亚洲国产二区三区久久| 俺来也俺去啦久久综合网| 欧美va久久久噜噜噜久久| 久久精品亚洲AV久久久无码| 亚洲国产成人乱码精品女人久久久不卡| 国产一区二区三区久久| 九九精品99久久久香蕉| 国产精品一区二区久久不卡| 老色鬼久久亚洲AV综合| 亚洲AV日韩AV天堂久久| 日本强好片久久久久久AAA| 亚洲精品白浆高清久久久久久| 久久无码AV中文出轨人妻| 久久婷婷五月综合国产尤物app | 久久精品国产亚洲av影院| 久久久久亚洲AV无码永不|