找回密码
 加入计匠网
搜索
热搜: BIOS ACPI CPU Windows
查看: 9395|回复: 3

[原创]CONTAINING_RECORD IN EFI

[复制链接]
发表于 2009-10-15 18:06:17 | 显示全部楼层 |阅读模式
CONTAINING_RECORD IN EFI

8 ]# e+ z: N7 Y6 }% `7 a# l& R: j. J& H, ^3 M9 l
EFI BIOS几乎全部用C完成,它几乎将C语言的各种技巧发挥到了极致。C的精髓泰半是指针,另外宏也是非常值得称道的。程序员对于宏的评价可谓褒贬不一,有人说它是万恶之源,有人则赞其为一把利刃。我个人觉得运用之妙,存乎一心,宏不是万能的,但是有一些场合使用宏确实可以大大的提高程序的可读性,有些跨平台的特性离开了宏还真是不行。_CREFI之中经常会被用的宏,我们来看看它的庐山真面目吧:' g/ _( S3 N: N& q- a
! E3 C, i4 A- `6 \/ n7 M$ m7 z! d
//

" b$ }; Z5 n4 ?8 I( i+ d
//- g5 I7 I- Y# w5 ], @7 ^) ?
CONTAINING_RECORD - returns a pointer to the structure

: l: g1 L6 I+ @/ _7 \6 o/ e  F  w
//
% S, b1 F% m# p0 @4 @; o* P( b3 Vfrom one of it's elements.

+ n( P6 J2 v8 `+ o) z
//

& e6 F! S& y9 e; {! H#define _CR(Record, TYPE, Field)
: _! k2 M0 i& n1 d6 P((TYPE *) ((CHAR8 *) (Record) - (CHAR8 *) &(((TYPE *) 0)->Field)))

" J+ _7 I" H* V: ?- N+ C! r这个宏的作用是根据一个结构体成员变量的的地址获得该结构体基地址。举例如下:' b* T' x( C5 z% K
struct _Test
# N; i' I: ]/ [9 f* H+ Q{$ ~$ X; R# X3 C
CHAR8 t0;

1 N* G# k% \/ S/ H! E. q2 P, M1 g- B# z% y3 r/ I, N2 Z+ t& u$ i
UINT16 t1;

( K. @7 R3 c3 r: e" r! l. a7 g: f, S- ]2 x+ J* U! @0 Q3 X
UINT8   t2;

' V$ d. u2 d" i! j# O: O  @0 E( W9 C  s
UINT32  t3;
, Q4 O# x( I/ `1 ?
};; b7 P; @, f9 o
7 v8 ^+ C* d/ }3 a
我们在某一个地方获得了t2的地址t2Ptr,这时如果要得到t2所在结构体的基地址就可以这样做:struct _Test* bPtr = _CR(t2Ptr,struct _Test,t2);预处理展开之后就是这副样子 struct _Test* bPtr = ((struct _Test*)((CHAR8*)(t2Ptr)-(CHAR8*)&((struct _Test *)0)->t2),其实我觉得关键的部分在这句“(TYPE *) 0)->Field”,它其实就是获得该成员在结构体中的偏移位置(offset),该成员变量的地址减去它在结构体中的偏移也就获得了基地址了。& @/ V$ ~) g6 ^" z; ]

. v1 M2 o- \5 l5 U: ^" ]* X% I5 ]. S! d: q: ~# W
一图胜千言,如上图所示,假设t2的地址是0x1005,那么bPtr应该是多少呢?我们只要用t2-t2offset0x03)即可,也就是0x1002。是不是很简单?$ g% p1 H! K7 N0 i+ |
! T3 C# w2 e0 G
大概是英雄所见略同吧,这种形式的宏在Linux kernel Windows Kernel中也都有存在,只不过名字叫的不一样罢了。先看看Linux下长的什么样子吧:% t2 d: |  E4 P, S$ _
/**

* F1 s3 c2 d  V$ N: N# l0 c1 L
7 G1 y- \4 X. ^0 C+ @, i
* container_of - cast a member of a structure out to the containing structure
; z4 X3 a' r! p% E! }
: U7 K4 J1 M1 Q8 L6 C
*
7 D% f5 h: B' k) v/ F& q! m) [

; T. I$ k$ Q# I! C( H; [* @ptr:$ G8 m8 I' y# M, s8 ?) x
the pointer to the member.

) S  N6 b0 O+ G9 h
9 j9 |  S  m5 z
* @type:5 g) W! w0 i3 G% ~3 M
the type of the container struct this is embedded in.

% S( v0 [" s$ {) R( _" k

; c7 Z2 C! Z7 E4 V! i0 q- X* Q* @member:
  z4 t0 V+ M$ V/ P& k6 lthe name of the member within the struct.
. l2 P2 k! Y" G( y" P! ~5 o4 W

! y+ z$ U$ O3 X, `( N*
. z9 }+ g" {4 l3 K, ]9 U
$ m# m5 ?, J. s
*/

. u6 C* ?0 g! r" D. F% k#define
% u& _; z4 z1 S( I7 \container_of(ptr, type, member) ({
3 x+ Y; A6 V7 ]" j% u9 P\

2 Y* A0 u' C- u" G3 x  H+ f
# }1 |( c9 ?! b8 @# q  gconst typeof( ((type *)0)->member ) *__mptr = (ptr);
8 A0 X; D: F9 Q& E- ?' l\
; g1 q9 }) y. h( O' z1 y6 f

* O6 a- K! W! L(type *)( (char *)__mptr - offsetof(type,member) );})
2 U& B3 m6 g6 ~' @7 F2 _$ {
% ]: I# O2 w) P9 {; w& M
#define offsetof(TYPE, MEMBER) ((size_t) &((TYPE *)0)->MEMBER); |) b& k$ N" L- w
, f/ Y( ]) e/ f4 M. _; ~) U
再来看看它在Windows Kernel中的小模样J:) d) K2 f" t+ S- ]/ e# x
#define, W$ B. D* k; [5 J3 {0 l- V
CONTAININT_RECORD(address, type, field) \3 Y) u% l3 z: r1 t
             ((type*)((PCHAR)(address) - (PCHAR)(&((type*)0)->field)))

1 k, T4 {, b" @7 M他们的作用都是一样的,是不是有点天下文章一大抄的感觉啊,呵呵
0 q$ s0 I& y' |4 H+ o8 q+ z3 ]) w1 H1 q" e) p+ O

4 Y, X% d1 I4 x, J( B4 Y4 _* _: i) |" f0 U, n; l; Z, a
Peter
( O3 x& d! {! R% s$ h# v9 P5 W2 t/ A( V0 M  f: ?# t7 y. U
2009-10-6

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?加入计匠网

×
发表于 2009-10-23 15:42:14 | 显示全部楼层
顺楼主者昌,逆楼主者亡.
回复

使用道具 举报

发表于 2009-10-26 09:49:54 | 显示全部楼层
象这样的宏挺绕人的,但又经常会碰到。2 X- p5 _* }8 P- G) P. u
楼主将其重点突出,从而问题变得很直接很明了了。。顶一个哈
回复

使用道具 举报

发表于 2009-10-30 22:51:12 | 显示全部楼层
崇拜中。。。。。。。
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 加入计匠网

本版积分规则

Archiver|手机版|小黑屋|计匠网

GMT+8, 2024-5-20 20:22 , Processed in 0.020993 second(s), 18 queries .

Powered by Discuz! X3.5

© 2001-2023 Discuz! Team.

快速回复 返回顶部 返回列表