本文旨在简单介绍一下UEFI中驱动程序的加载方式(这里涉及的模块指的是符合UEFI Driver Model的模块):

在UEFI中,当一个驱动模块被加载时,在模块入口点只会安装EFI_DRIVER_BINDING_PROTOCOL等,而不会去执行驱动程序的初始化(这一点与Linux中不同,在Linux中,当我们在驱动模块的入口点调用driver_register()来注册驱动的时候,会在driver_register()的里面调用总线的match(),驱动与设备匹配成功之后紧接着就会调用驱动程序的probe()函数来执行驱动的初始化),而在UEFI中,只有当我们在Boot Manager中调用gBS->ConnectController()时才会去匹配并初始化驱动程序。

下面我们设想一个硬件结构,以此为例来描述一下UEFI的驱动加载过程:

硬件结构:CPU内部有一个PCI Host Bridge,PCI Host Bridge下面有一个Root Bridge,USB Host Controller(EHCI)挂在Root Bridge管理的PCI总线上,再外接一个USB KeyBoard(USB键盘)。

目标:现在我们要通过USB键盘输入字符。

涉及的设备驱动程序:pci host bridge driver、pci bus driver、usb host driver(EHCI)、usb bus driver、usb keyboard driver。

UEFI相关背景知识:

1. POST过程中,DXE内核会Load所有的模块,模块会执行他们的入口函数;

2. 大部分设备驱动程序(这里指pci bus driver、usb host driver、usb bus driver、usb keyboard driver)在入口函数只做一件事(安装EFI_DRIVER_BINDING_PROTOCOL);

3. pci host bridge driver有点不一样,它在入口函数会:

-> 创建host bridge的handle并在handle上安装EFI_PCI_HOST_BRIDGE_RESOURCE_ALLOCATION_PROTOCOL;
-> 创建root bridge的handle并在handle上安装EFI_PCI_ROOT_BRIDGE_IO_PROTOCOL;

4. EFI_BOOT_SERVICES.LocateHandleBuffer()能够获取系统中所有安装某种Protocol的handle;

5. EFI_BOOT_SERVICES.ConnectController()会把系统中所有的EFI_DRIVER_BINDING_PROTOCOL都找出来,并执行EFI_DRIVER_BINDING_PROTOCOL->Support()来匹配Device和Driver;匹配成功之后,ConnectController()会接着调用EFI_DRIVER_BINDING_PROTOCOL->Start()来执行驱动程序的初始化。

6. 关于Handle与Protocol:一个Handle上可以安装多个不同的Protocol,不同的Handle上可以安装同一个Protocol的不同实例(类似一个二维链表)

下面的代码模拟执行所有硬件相关的初始化:

//第一步:查找系统中的PCI Root Bridge,并加载PCI总线驱动

 // 获取系统中所有安装EFI_PCI_ROOT_BRIDGE_IO_PROTOCOL的handle;即找出系统中所有的pci root bridge
gBS->LocateHandleBuffer (
ByProtocol,
&gEfiPciRootBridgeIoProtocolGuid,
NULL,
&NumHandles,
&HandleBuffer); // 查找并加载pci bus driver
// PCI bus driver的Support()通过判断device handle上是否安装有EFI_PCI_ROOT_BRIDGE_IO_PROTOCOL来匹配;
for (Index = ; Index < NumHandles; Index++) { //循环root bridge
Status = gBS->ConnectController (
HandleBuffer [Index],
NULL,
NULL,
FALSE);
}
// pci bus driver会枚举出所有的pci device,为每个pci device创建handle,并在handle上安装EFI_PCI_IO_PROTOCOL; //第二步:加载EHCI driver,初始化USB Host Controller // 获取系统中所有安装EFI_PCI_IO_PROTOCOL的handle(即找出系统中所有的pci device);
Status = gBS->LocateHandleBuffer (
ByProtocol,
&gEfiPciIoProtocolGuid,
NULL,
&HandleCount,
&Handles
); // 因为我们只关心EHCI,所以通过判断pci device的class code找到EHCI
for (Index = ; Index < HandleCount; Index++) {
Status = gBS->HandleProtocol (
Handles[Index],
&gEfiPciIoProtocolGuid,
(VOID **) &PciIo
);
if (!EFI_ERROR (Status)) {
// 读取pci配置空间
Status = PciIo->Pci.Read (PciIo, EfiPciIoWidthUint8, 0x09, , &Class);
if (!EFI_ERROR (Status) &&((PCI_CLASS_SERIAL == Class[]) && (PCI_CLASS_SERIAL_USB == Class[]))) { // pci device 是 EHCI
// 查找并加载驱动
// EHCI driver的support()通过判断device handle是否安装了EFI_PCI_IO_PROTOCOL以及class code是否正确来匹配
Status = gBS->ConnectController (
Handles[Index],
NULL,
DevicePath,
FALSE
);
}
}
}
// EHCI driver的start()会为handle安装EFI_USB_HC2_PROTOCOL;表明这是一个USB Host Controller //第三步:加载USB总线驱动

// 获取系统中所有安装EFI_USB_HC2_PROTOCOL的handle(即找出系统中所有的USB Host Controller);
Status = gBS->LocateHandleBuffer (
ByProtocol,
&gEfiUsb2HcProtocolGuid,
NULL,
&UsbHcHandlesCount,
&UsbHcHandles); // 查找并加载usb bus driver
// usb bus driver的support通过判断device handle是否安装了EFI_USB_HC2_PROTOCOL来匹配
if (!EFI_ERROR (Status)) {
for (i = ; i < UsbHcHandlesCount; i++) {
gBS->ConnectController (UsbHcHandles [i], NULL, NULL, TRUE);
}
}
// usb bus driver的start()会枚举所有的USB device,为每个device创建device handle,并安装EFI_USB_IO_PROTOCOL(用来表明这是一个USB设备);
// 当usb keyboard device被枚举之后,usb bus driver会调用EFI_BOOT_SERVICES.ConnectController()查找他的驱动;
// usb keyboard driver的support()会判断device handle是否安装了EFI_USB_IO_PROTOCOL以及Interface描述符的class、subclass、protocol来匹配;
// usb keyboard driver的start()会在keyboard的device handle上安装EFI_SIMPLE_TEXT_INPUT_PROTOCOL。 //第四步:调用EFI_SIMPLE_TEXT_INPUT_PROTOCOL接收键盘的输入 while (TRUE) {
// 调用EFI_SIMPLE_TEXT_INPUT_PROTOCOL
gST->ConIn->ReadKeyStroke (gST->ConIn, &Key);
if (Key.ScanCode == CONFIG_SYSTEM_ERROR_MANAGER_MENU_RESUME_KEY) { // 用户输入正确的按键
return;
}
}

总结:UEFI中通过Protocol来标识device handle的类型(当然底层驱动也通过Protocol向上层驱动提供接口)。UEFI中的驱动程序的分层很清晰,由底向上依次依赖。驱动程序的初始化由POST过程控制,便于控制和理解(Linux中则由各个子系统控制,以USB系统为例:UEFI必须先初始化Host Controller Driver,然后初始化USB Bus Driver,最后初始化USB Device Driver;而在Linux中,不存在这种依赖关系)

EDK II之驱动程序与硬件平台的初始化简介的更多相关文章

  1. EDK II之Secure Boot简述

    密钥对:公钥分发,私钥自留.常见的公钥格式:cer/der,常见的私钥格式:pfx. BIOS中Secure Boot的原理:把公钥包在code里面,当使用gBS->LoadImage()去加载 ...

  2. Setting up a EDK II build environment on Windows and Linux:搭建Windows和Linux开发环境[2.2]

    Setting up a EDK II build environment on Windows and Linux:搭建Windows和Linux开发环境[2.2] 2015-07   北京海淀区  ...

  3. Tools:downloading and Building EDK II工具篇:安装/使用EDKII源代码获取/编译工具[2.3]

    Tools:Installing and using the Required Tools for downloading and Building EDK II工具篇:安装/使用EDKII源代码获取 ...

  4. EDK II之USB设备驱动程序的加载与运行

    本文简单介绍一下USB设备的驱动程序是如何匹配设备以及被加载的: 上文(UDK中USB总线驱动的实现框架)提到USB总线枚举设备的最后一步是调用gBS->ConnectController()去 ...

  5. EDK II之USB主控制器(EHCI)驱动的实现框架

    本文简要介绍一下UEFI中EHCI驱动的代码实现框架: 下图是HCDI: 上图是Host驱动程序向上层驱动提供的接口图: 1.大部分接口的最后动作都是去操作主控制器寄存器,ECHI的spec:< ...

  6. EDK II之USB协议栈的实现简介

    本文旨在简单介绍一下 UEFI中USB协议栈的代码框架: 主要包括: USB主控制器驱动(HCDI:EFI_USB2_HC_PROTOCOL) USB总线驱动(USBDI:EFI_USB_IO_PRO ...

  7. EDK II之SMM/SMI

    SMM:System Managerment Mode SMM有自己的smm core以及dispatcher(可以简单的把smm core跟dxe core看成是平行的存在),smm有自己的运行空间 ...

  8. EDK II之DXE Core的事件管理

    本文简单介绍一下UEFI中的事件管理: UEFI是不支持多进程的,但是UEFI支持多事件分发机制.UEFI只支持时钟中断,并基于时钟中断实现事件分发.类似于OS中基于时钟中断来实现基于时间片的多任务调 ...

  9. EDK II之USB总线驱动的实现框架

    本文简单介绍一下UEFI中USB驱动的实现框架: 下图是USBD向上层驱动提供的接口: 1.从图中我们可以看出,USBDI的实现主要通过调用HCDI实现 和 访问USB_INTERFACE结构体(该结 ...

随机推荐

  1. 【Redis】持久化

    Redis提供了为持久化提供了两种方法:第一种是快照:他可以将存在某一时刻的所有数据都写入硬盘里面.第二种是只追加文件(AOF):它会在执行命令时,将被执行的写命令复制到硬盘里面. Redis支持持久 ...

  2. 如何比sketch和axure更方便地给原型做交互?

    在快速的工作环境中,我们现在都希望在工作的各个环节中提高效率.有些产品设计师们做产品原型时,会感觉sketch或者axure添加交互的方式不够快捷.下面就提供一种解决方案. 使用工具:墨刀. 交互链接 ...

  3. 数据分析与挖掘 - R语言:贝叶斯分类算法(案例一)

    一个简单的例子!环境:CentOS6.5Hadoop集群.Hive.R.RHive,具体安装及调试方法见博客内文档. 名词解释: 先验概率:由以往的数据分析得到的概率, 叫做先验概率. 后验概率:而在 ...

  4. Elasticsearch集群管理工具head插件安装

    Elasticsearch-head是一个elasticsearch的集群管理工具,它是完全由html5编写的独立网页程序,你可以通过插件把它集成到es.或直接下载源码,在本地打开index.html ...

  5. C# sapnco支持.net 4.5了,真是个意外的发现

    意外篇: 需要用C#写一个RFC直连的类库,需要引用sapnco.dll   sapnco_utils.dll两个文件 之前都是从网上下载的sapnco3.0,引用开发,在win10机器上使用没有问题 ...

  6. spring注解式开发之视图解析器

    http://localhost:8089/springmvc-04-viewResovler/springmvc/hello

  7. java RE Validation常用

    import java.util.regex.Matcher; import java.ulil.regex.Pattern; public class RegExpUtil { public sta ...

  8. 大数据-05-Spark之读写HBase数据

    本文主要来自于 http://dblab.xmu.edu.cn/blog/1316-2/ 谢谢原作者 准备工作一:创建一个HBase表 这里依然是以student表为例进行演示.这里假设你已经成功安装 ...

  9. CentOS中利用Docker安装Redis

    CentOS中利用Docker安装Redis 1.拉取镜像 #docker pull redis:4.0.10 2.加载镜像 #docker run -p 6379:6379 --name test- ...

  10. C# Control.Invoke匿名委托

    if (txbValue.InvokeRequired) txbValue.Invoke(new MethodInvoker(delegate() {                          ...