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

            我參與的團隊

            搜索

            •  

            積分與排名

            • 積分 - 326969
            • 排名 - 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
            亚洲欧美日韩中文久久| 国产免费福利体检区久久| 伊人久久精品无码二区麻豆| 国产精品久久久久jk制服| 精品久久久久久综合日本| 久久精品二区| 久久精品国产亚洲av麻豆小说| 91精品国产91久久久久久| 久久这里只精品99re66| 伊人久久大香线焦综合四虎| 久久久久亚洲精品日久生情| 女人香蕉久久**毛片精品| 亚洲精品乱码久久久久久蜜桃不卡| 伊人热人久久中文字幕| 亚洲欧美日韩中文久久| 综合久久给合久久狠狠狠97色| 国内精品久久人妻互换| 久久精品国产亚洲AV影院| 国产成人精品综合久久久| 久久99精品久久只有精品| 免费精品国产日韩热久久| 精品久久久久一区二区三区| 丁香五月网久久综合| 亚洲欧洲精品成人久久奇米网| 一本久道久久综合狠狠躁AV| 国产午夜精品久久久久九九| 99久久99这里只有免费的精品| 99精品久久精品一区二区| 一本色道久久88综合日韩精品 | 亚洲中文字幕久久精品无码喷水| 国产精品va久久久久久久| 久久精品免费一区二区三区| 久久久久亚洲AV无码专区体验| 亚洲色欲久久久综合网| 亚洲国产精品无码久久一区二区| 久久久久久久波多野结衣高潮 | 久久综合九色综合97_久久久 | 久久精品亚洲男人的天堂| 99热精品久久只有精品| 国产成人综合久久久久久| 国产一区二区精品久久凹凸|