|
|
|
EFI Protocol VS C++ & y) ] p+ W; z8 }( F2 J B
1.Introduction k% M" o- Y/ d5 H' a# C
0 P5 q0 I: t6 F; E Protocol是EFI引出的新概念,翻翻EDK就会发现与protocol相关的code散落在EFI的各个角落,大到host bus driver,小到Hello Word App都和protocol 脱不了关系,因此毫不夸张的说protocol应该是EFI的精髓所在,DXE阶段模块之间的通信都经由protocol完成(我觉得PEI阶段的ppi应该和protocol一个概念,只不过是限于PEI的特殊性而稍有差异而已).EFI Driver基本上就是使用一堆protocol的实现的一个可执行的文件.一个protocol就是一堆函数指针和数据成员的集合,官方称呼为protocol interface structure;这个protocol必须要定义一个GUID作为它的唯一标识,DXE driver使用Boot Service table中的类似InstallProtocolInterface等函数将porotocol安装到device handle上这就叫做produce protocol, InstallProtocolInterface成功的话该protocol就是published,后续其它driver使用该protocol就只要用Boot Service table中的LocateProtocol等相关函数从handle database中索引出该protocol就可以了.protocol为DXE阶段模块之间的通信提供了非常有效的手段.
7 `& |- m. Z( N8 \
* O6 E% {, a8 i0 a& T S, d J2.Protocol Implement OO2 Z* N. L* U* s
- O; r! X f$ F+ Z7 _0 v- v6 M
EFI是一个使用C实现的OO的Framework,它的OO特性主要经由protocl实现.EFI可以说是一个优秀的
$ \* a, h) ~. u. Z& EFramework, 对它的扩展可以优雅的实现.使用过C++这类面向对象语言的朋友都会知道OO有三大基本特性:1.Encapsulation,2.Inheritance,3.Polymorphism. 那么既然EFI是使用protocol实现的OO,那么protocol是如何实现这三大特性的呢?
' \, U; f3 T4 u5 ?# a& `6 V: A- J# s& p8 m# u
1.Encapsulation的实现其实比较简单,它所表达的信息隐藏的概念,这个部分在protocol中可以方便的实现,protocol通常是函数指针和属性的集合,我们导出函数的接口,而将数据属性放在函数接口内部处理.如下述EDK中的code所示:
3 o9 W# x0 X T3 F. k2 qtypedef struct _EFI_CPU_ARCH_PROTOCOL {
* M3 \8 d4 d4 Z T# j
- [6 z5 i& ^& p) \% B, U* EEFI_CPU_FLUSH_DATA_CACHE& b' q4 t9 x. q- a. q
FlushDataCache;8 H/ I3 a; e8 [* G1 M$ Y0 Y
, i6 M' g U3 g$ i$ M. }; n
EFI_CPU_ENABLE_INTERRUPT2 X: P) Q9 }8 x" m1 M' W
EnableInterrupt;& {- J% [: B: Y2 T
/ j/ u* `) ]+ m9 y F e9 I
EFI_CPU_DISABLE_INTERRUPT; g3 C! X4 e: z1 S! N( D" x! p
DisableInterrupt;
5 X/ |9 F* F& b2 a. c0 P. C! m) h+ p9 J
EFI_CPU_GET_INTERRUPT_STATE1 ]- @' A- o) {8 i& I
GetInterruptState;- J; n, G& G6 ~/ `( Z7 n3 W3 Y0 {
4 a1 Z8 `7 A2 T* A5 a2 c
EFI_CPU_INIT
$ G7 }3 m9 Q+ i/ u* w' cInit;1 T8 r k6 u; A. v/ l3 O% h
4 p" @* b N o0 |3 p' z$ T& BEFI_CPU_REGISTER_INTERRUPT_HANDLER+ @3 Z$ X- g* ]( T: y
' E7 m& A4 W$ ?0 R
RegisterInterruptHandler;" f& T' E' x; p
: U8 l8 T& J4 I7 j5 S* o
EFI_CPU_GET_TIMER_VALUE
! H$ s* e. ^) _& p5 Q* \GetTimerValue;
" R! J2 L+ v* f/ Q& b# B d4 L2 [/ S4 ]2 u7 R( C
EFI_CPU_SET_MEMORY_ATTRIBUTES9 n8 n4 ?/ ~! |, ~/ i
SetMemoryAttributes;
* {; o/ l% a! M& ]- T$ v m2 I
& g4 _2 L5 g5 X3 Q$ RUINT32
7 G/ B7 p0 ?: ~( N' {) q2 S, ~' ?7 X. W* b: L9 \/ @0 Q7 `+ [( ^
NumberOfTimers;
3 b8 q5 |1 c: r$ a9 A1 u8 g |5 ?( e0 M8 G8 L1 I
UINT32' }5 [9 x' `8 P" H3 s
DmaBufferAlignment;1 N1 u1 |3 d* | q9 h
} EFI_CPU_ARCH_PROTOCOL;. A+ c4 a" C( A8 s
像NumberOfTimers, DmaBufferAlignment就是数据属性了.% [- Q3 P8 E4 f5 P% u- F: `! g
5 g; y3 ?& U3 A2.Inheritance子类继承了父类的所有的方法和属性,在C的级别上表示就是内
. i j# o$ X+ Z n) A6 U$ M存的叠加.同时它还代表的是IS-A的概念.如下面的sample所示B_PROTOCOL" Q! H' c# `/ b v( W
继承自A_PROTOCOL.0 G; ^, v; Y; Y' M8 O
typedef struct _ A_PROTOCOL {
6 Z* N* _% V; I5 g3 @. b& i7 tUINT32! `6 E0 z2 V# h; L1 y
# S: M/ K$ t8 v/ J
A;
3 Y! B7 ^, @( c: g
2 r t1 n0 v; Q2 \}A_PROTOCOL;6 F$ N8 d7 x3 G: L
typedef struct _ B_PROTOCOL {
- H* E( R' _ y8 b, eA_ PROTOCOL
3 X' g; r9 U) T1 j5 ~: ?+ U1 g/ Q+ j1 ^* B( h" g
AP;
# F$ ~2 h1 ^* ?0 s" ~4 DUINT322 s5 r2 J" v- _$ C
B;
- b) a( D+ P" }& x; m}B_PROTOCOL;
" m/ u- B1 T) v) R k它们的类图如下图1所示,内存布局如下图2所示:
: R; k& x! H5 w
! }0 k5 Z$ [' q: [+ \; T; Z+ D7 S& F' \* j# p
7 C7 n8 Y9 j9 L' B( @/ g6 z$ M9 p
8 S# U1 H0 v3 t3 B1 h! o2 x
3. Polymorphism是面向对象的精髓所在了,如果没有了多态,则只能称之为基于对象了.多态是指基于相同的接口实现的不同的class(EFI之中应该是指实现protocol的driver),具有不同的行为.如两个EFI Driver A&B 它们都实现了ComponentName protocol,A在它的GetDriverName interface回复的Driver Name是”A”而Driver B则回复的是”B”,所以同样的接口具有不同的行为.protocol相当于一个共同的接口,因此我们可以以统一的方式去管理和配置这些实现了该protocol的这些Driver.如shell app dirver.efi它就是通过枚举Handle database找到挂在这些device handle上的ComponentName protocol,然后call它们的GetDriverName interface获悉driver的名字信息.
/ q. i- @* e; F% ?! x2 m% H, D2 b { W6 X
3.Who Win(Protocol vs C++)?
2 p O6 n; _2 G$ Q& w* c$ w* Y- s& }5 @; p, T% P& Z5 L& P% G
看上去貌似用protocol实现OO特性还是比较费事的,那么为什么不直接使用C++实现呢?我不知道真正
* [! _) `( m! A原因L!呵呵,可是我猜有可能的几种原因如下:
: p* i/ s2 [! ~, a
! i) M& p: h. T8 q% R) i1. C++无法做到二进制级别上的复用,C++是一个非常复杂的语言,它有非常多的特性,C++标准委员会制定出了编译器厂商所需要实现的一堆特性,比如构造函数,析构函数,继承,重载,多态等等一堆的规则,但是标准委员会并没有规定编译器厂商如何实现这些特性于是问题就来了,我们使用A厂商编译器build driver da和B厂商编译器build同一只driver da它们的内存布局就不相同,于是二进制级别上的互操作就不可能了,设备厂商都要把source code拿出来放在一个编译器上build 才OK,不同的编译器实现不同的一个地方就体现在实现多态时,vptr存放的位置上,有一些编译器会把该指针放在对象内存布局的头部,有些厂商则喜欢放在尾部,这个差异非常大;如果存在多重继承那么就可能出现多个vptr那么vptr摆放的顺序又会有不同等等不胜其扰的问题。而protocol使用标准的c实现,它就没有二进制文件不一致的问题。7 Q) ~ Z: Y/ l9 { ~( ~
3 Y) Y: h& K; ]) `* l
2.C++实现多态,虚继承以及name mangling等会带来空间以及编译时间上的开销。C++中的多态是通过在基类的函数声明中加上virtual实现的,这一个关键字里面大有文章,我们使用一个class VI演示该过程:* E& Z# W! m# m2 E2 j
class VI- Y' I; G/ c+ ^2 D4 o' w
{
3 \, G0 i5 Y% r0 j/ K) Q* O
: N+ l$ n: L; T7 u% w3 L# k! Iint a;
& C0 ~/ X( Z# u' e5 ~3 A; L/ S; P
" P3 K! }/ f- ?* L0 T& j) s" \int b;
' F$ D- Y: Q' I! A; E: v- i' T
" y8 _3 {2 c; m# ivoid test0(void) virtual;
_5 ]) O% v/ h! e};
% F% k" k3 E& Q: n1 E没有virtual这个关键字的时候它的内存布局如下图3所示,加上之后就如图4所示:! ?' z; p$ L2 I5 H: r/ U `4 Z, \
2 M' I/ h: k0 J2 L( u( i3 |% f- S! o+ L+ B8 D! ? I7 d
; U0 Y/ T T! C, W" N/ a- |. \1 G( h l
, n3 S' T9 ^" ~1 C+ O编译器通过增加一个vptr虚函数指针,和vtbl虚函数表实现多态的功能,一旦子类改写了test0那么子类就会修改掉vtbl中的test0的地址。如此一来开销就来了J,一旦涉及到虚继承多重继承,那个开销就更大了。相比较来看protocol的实现就完全没有这个问题。& i; s' I- i! M# l1 Q
5 M+ x8 ?, o- C) l$ B( h8 `/ k3.C++使用字符命名对象会有命名冲突的可能而protocol使用64位GUID是不可能出现命名冲突的。当project大到一定程度就会感受到变量命名冲突的讨厌了,你定义的变量可能在别的你不知道的模块里已经被用过了,然后编译器报上一大堆另人恐慌的错误出来。当然C++也有解决命名冲突的方法,那就是namespace可是这样又会造成性能上的影响。
6 z9 l/ z& I( C+ H! k p( }! \0 C" e
所以综上所述,protocol应该较C++稍有优势一点哦J,它完整的实现了OO的所有特性,而无性能上的损耗只是感官上有些差异,如intel的spec所说的那样Look/fell very different.
& y" J% N7 T! u
5 S* b7 b$ l/ D5 [That’s all!
. n8 j. a9 d1 m# P ~- x4 J, x
7 v' F3 d3 u' q1 a/ TEnjoy it!3 t; w2 S- K; s8 j4 I
. c9 P( w4 l) N9 x8 A
Peter$ k( p/ q! ?( j; y2 f' L$ H6 T1 y: ?
3 m8 s h1 h7 v0 c, m; }& T" Q w- O! S0 L% c6 B
: h4 [7 @; B" L% E: C
[ 本帖最后由 peterhu 于 2009-7-13 17:05 编辑 ] |
本帖子中包含更多资源
您需要 登录 才可以下载或查看,没有账号?加入计匠网
×
|