• <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)

            我參與的團隊

            搜索

            •  

            積分與排名

            • 積分 - 326966
            • 排名 - 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 閱讀(1158) 評論(0)  編輯 收藏 引用 所屬分類: Ubuntu&Linux
            久久线看观看精品香蕉国产| 看全色黄大色大片免费久久久| 国产综合精品久久亚洲| 国产精品美女久久久m| 99久久做夜夜爱天天做精品| 久久性精品| 久久露脸国产精品| 一本色综合久久| 伊人久久大香线蕉综合影院首页 | 国产一区二区精品久久凹凸| 久久97精品久久久久久久不卡| 久久99国产精品久久99果冻传媒 | 久久天堂电影网| 国产精品伦理久久久久久| 狠狠人妻久久久久久综合| 久久久久黑人强伦姧人妻| 欧美午夜A∨大片久久| 99久久精品免费看国产一区二区三区| 2020国产成人久久精品| 久久亚洲私人国产精品vA| 久久国产亚洲精品麻豆| 久久免费99精品国产自在现线 | 99久久精品影院老鸭窝| 国产伊人久久| 午夜天堂精品久久久久| 久久精品一区二区| 久久午夜无码鲁丝片午夜精品| 久久人人爽人人爽人人片AV不| 久久精品国产亚洲AV无码偷窥| 久久91亚洲人成电影网站| 久久这里都是精品| 69国产成人综合久久精品| 欧洲性大片xxxxx久久久| 日本人妻丰满熟妇久久久久久| 久久综合九色综合97_久久久| 亚洲综合久久久| 亚洲精品国产成人99久久| 漂亮人妻被中出中文字幕久久 | 精品久久久久国产免费| 亚洲国产精品无码久久久秋霞2 | 久久精品九九亚洲精品天堂|