【UEFI基础】EDK网络框架(DNS4)

2024-01-22 10:12
文章标签 基础 框架 网络 uefi edk dns4

本文主要是介绍【UEFI基础】EDK网络框架(DNS4),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

DNS4

DNS4协议说明

IP地址是一串数据,不便记忆。一般用户在使用TCP/IP协议进行通信时也不使用IP地址,而是使用英文和点号组成的字符串,两者的转换通过DNS(Domain Name System)来完成。

DNS也有v4和v6版本,这里只介绍v4版本。其报文格式如下:

在这里插入图片描述

各个参数的说明如下:

字段长度(bit)描述
Header96头部字段,是必须存在的,它定义了报文是请求还是应答,也定义了其他段是否需要存在,以及是标准查询还是其他。
Question变长大多数查询中,Question段包含着问题(question),比如,指定问什么。
这个段包含QDCOUNT(头部字段中的QDCOUNT字段,通常值是1)个问题。
Anser变长分别指应答,授权,附加字段。
都共用相同的格式:多个资源记录,资源记录的个数由报文头段中对应的几个数值确定。
Authority变长
Additianal变长

其中的头部有12个字节,内容如下:

在这里插入图片描述

各个参数的说明如下:

字段长度(bit)描述
ID16标识字段,客户通过标识字段来确定DNS响应是否与查询请求匹配。
QR1操作类型:
0:查询报文
1:响应报文
OPCODE4查询类型:
0:标准查询
1:反向查询
2:服务器状态查询
3~15:保留未用反向查询是客户端请求服务器根据回答生成导致此回答的问题,这个查询类型的使用并不多。
AA1若置位,则表示该域名解析服务器是授权回答该域的。
TC1若置位,则表示报文被截断。使用UDP传输时,应答的总长度超过512字节时,只返回报文的前512个字节内容。
RD1客户端希望域名解析服务器采取的解析方式:
0:表示希望域名解析服务器采取迭代解析
1:表示希望域名解析服务器采取递归解析
RA1域名解析服务器采取的解析方式:
0:表示域名解析服务器采取迭代解析
1:表示域名解析服务器采取递归解析
Z3全部置0,保留未用。
RCODE4响应类型:
0:无差错
1:查询格式错
2:服务器失效
3:域名不存在
4:查询没有被执行
5:查询被拒绝
6-15: 保留未用
QDCOUNT16无符号16位整数表示报文请求段中的问题记录数。
ANCOUNT16无符号16位整数表示报文回答段中的回答记录数。
NSCOUNT16无符号16位整数表示报文授权段中的授权记录数。
ARCOUNT16无符号16位整数表示报文附加段中的附加记录数。

对应代码中的结构体:

union _DNS_FLAGS {struct {UINT16    RCode  : 4;UINT16    Zero   : 3;UINT16    RA     : 1;UINT16    RD     : 1;UINT16    TC     : 1;UINT16    AA     : 1;UINT16    OpCode : 4;UINT16    QR     : 1;} Bits;UINT16    Uint16;
};typedef struct {UINT16       Identification;DNS_FLAGS    Flags;UINT16       QuestionsNum;UINT16       AnswersNum;UINT16       AuthorityNum;UINT16       AditionalNum;
} DNS_HEADER;

后面变长部分的内容,后面如何使用到再介绍。

DNS4代码综述

UDP4也是一个通用的网络协议,其实现在NetworkPkg\DnsDxe\DnsDxe.inf,这里首先需要看下它的入口:

EFI_STATUS
EFIAPI
DnsDriverEntryPoint (IN EFI_HANDLE        ImageHandle,IN EFI_SYSTEM_TABLE  *SystemTable)
{//// Install the Dns4 Driver Binding Protocol.//Status = EfiLibInstallDriverBindingComponentName2 (ImageHandle,SystemTable,&gDns4DriverBinding,ImageHandle,&gDnsComponentName,&gDnsComponentName2);//// Install the Dns6 Driver Binding Protocol.//Status = EfiLibInstallDriverBindingComponentName2 (ImageHandle,SystemTable,&gDns6DriverBinding,NULL,&gDnsComponentName,&gDnsComponentName2);//// Create the driver data structures.//mDriverData = AllocateZeroPool (sizeof (DNS_DRIVER_DATA));//// Create the timer event to update DNS cache list.//Status = gBS->CreateEvent (EVT_NOTIFY_SIGNAL | EVT_TIMER,TPL_CALLBACK,DnsOnTimerUpdate,NULL,&mDriverData->Timer);Status = gBS->SetTimer (mDriverData->Timer, TimerPeriodic, TICKS_PER_SECOND);InitializeListHead (&mDriverData->Dns4CacheList);InitializeListHead (&mDriverData->Dns4ServerList);InitializeListHead (&mDriverData->Dns6CacheList);InitializeListHead (&mDriverData->Dns6ServerList);
}

从这里可以看到会安装v4和v6版本的EFI_DRIVER_BINDING_PROTOCOL接口,不过这里只关注v4版本的:

EFI_DRIVER_BINDING_PROTOCOL  gDns4DriverBinding = {Dns4DriverBindingSupported,Dns4DriverBindingStart,Dns4DriverBindingStop,DNS_VERSION,NULL,NULL
};

除此之外,这里还为DNS_DRIVER_DATA分配了内存,并初始化了部分内容,最重要的是有一个定时器,是用来更新DNS缓存列表的,其结构体如下:

struct _DNS_DRIVER_DATA {EFI_EVENT     Timer;                 /// Ticking timer for DNS cache update.LIST_ENTRY    Dns4CacheList;LIST_ENTRY    Dns4ServerList;LIST_ENTRY    Dns6CacheList;LIST_ENTRY    Dns6ServerList;
};

DNS4在UEFI网络协议栈中的关系图:

支持
提供
支持
支持
提供
支持
提供
提供
提供
支持
提供
提供
支持
支持
提供
提供
提供
支持
提供
提供
gEfiPciIoProtocolGuid
UNDI
gEfiNetworkInterfaceIdentifierProtocolGuid_31
gEfiDevicePathProtocolGuid
SNP
gEfiSimpleNetworkProtocolGuid
MNP
gEfiVlanConfigProtocolGuid
gEfiManagedNetworkServiceBindingProtocolGuid
gEfiManagedNetworkProtocolGuid
ARP
gEfiArpServiceBindingProtocolGuid
gEfiArpProtocolGuid
IP4
gEfiIp4ServiceBindingProtocolGuid
gEfiIp4Config2ProtocolGuid
gEfiIp4ProtocolGuid
DHCP4
gEfiDns4ServiceBindingProtocolGuid
gEfiDns4ProtocolGuid

Dns4DriverBindingSupported

DNS4依赖于UDP4:

EFI_STATUS
EFIAPI
Dns4DriverBindingSupported (IN EFI_DRIVER_BINDING_PROTOCOL  *This,IN EFI_HANDLE                   ControllerHandle,IN EFI_DEVICE_PATH_PROTOCOL     *RemainingDevicePath OPTIONAL)
{//// Test for the Udp4ServiceBinding Protocol.//Status = gBS->OpenProtocol (ControllerHandle,&gEfiUdp4ServiceBindingProtocolGuid,NULL,This->DriverBindingHandle,ControllerHandle,EFI_OPEN_PROTOCOL_TEST_PROTOCOL
}

Dns4DriverBindingStart

Start函数的流程大致如下:

  1. 初始化DNS_SERVICE
  2. 安装gEfiDns4ServiceBindingProtocolGuid

同其它驱动一样,重点也是结构体,这里就是DNS_SERVICE

DNS_SERVICE

DNS_SERVICE在Start函数中创建:

EFI_STATUS
EFIAPI
Dns4DriverBindingStart (IN EFI_DRIVER_BINDING_PROTOCOL  *This,IN EFI_HANDLE                   ControllerHandle,IN EFI_DEVICE_PATH_PROTOCOL     *RemainingDevicePath OPTIONAL)
{Status = DnsCreateService (ControllerHandle, This->DriverBindingHandle, IP_VERSION_4, &DnsSb);Status = gBS->SetTimer (DnsSb->Timer, TimerPeriodic, TICKS_PER_SECOND);

其结构体定义如下:

struct _DNS_SERVICE {UINT32                          Signature;EFI_SERVICE_BINDING_PROTOCOL    ServiceBinding;UINT16                          Dns4ChildrenNum;LIST_ENTRY                      Dns4ChildrenList;UINT16                          Dns6ChildrenNum;LIST_ENTRY                      Dns6ChildrenList;EFI_HANDLE                      ControllerHandle;EFI_HANDLE                      ImageHandle;EFI_EVENT                       TimerToGetMap;EFI_EVENT                       Timer; /// Ticking timer for packet retransmission.UINT8                           IpVersion;UDP_IO                          *ConnectUdp;
};

其中比较重要的成员有:

  • ServiceBinding:对应mDns4ServiceBinding
EFI_SERVICE_BINDING_PROTOCOL  mDns4ServiceBinding = {Dns4ServiceBindingCreateChild,Dns4ServiceBindingDestroyChild
};
  • Dns4ChildrenNumDns4ChildrenList:对应DNS4子项,在Dns4ServiceBindingCreateChild()中创建。
  • TimerToGetMap:其说明如下:
  //// Create the timer used to time out the procedure which is used to// get the default IP address.//Status = gBS->CreateEvent (EVT_TIMER,TPL_CALLBACK,NULL,NULL,&DnsSb->TimerToGetMap);
  • Timer:其说明如下:
  //// Create the timer to retransmit packets.//Status = gBS->CreateEvent (EVT_NOTIFY_SIGNAL | EVT_TIMER,TPL_CALLBACK,DnsOnTimerRetransmit,DnsSb,&DnsSb->Timer);
  • IpVersion:对应DNS4来说就是IP_VERSION_4
  • ConnectUdp:对应结构体UDP_IO
struct _UDP_IO {UINT32                     Signature;LIST_ENTRY                 Link;INTN                       RefCnt;UINT8                      UdpVersion;//// Handle used to create/destroy UDP child//EFI_HANDLE                 Controller;EFI_HANDLE                 Image;EFI_HANDLE                 UdpHandle;EFI_SIMPLE_NETWORK_MODE    SnpMode;LIST_ENTRY                 SentDatagram;  ///< A list of UDP_TX_TOKEN.UDP_RX_TOKEN               *RecvRequest;union {EFI_UDP4_PROTOCOL    *Udp4;EFI_UDP6_PROTOCOL    *Udp6;} Protocol;union {EFI_UDP4_CONFIG_DATA    Udp4;EFI_UDP6_CONFIG_DATA    Udp6;} Config;
};

基本上都是与UDP(这里只关注UDP4版本),其中的Token在DoDnsQuery()中创建:

EFI_STATUS
DoDnsQuery (IN  DNS_INSTANCE  *Instance,IN  NET_BUF       *Packet)
{if (Instance->UdpIo->RecvRequest == NULL) {Status = UdpIoRecvDatagram (Instance->UdpIo, DnsOnPacketReceived, Instance, 0);if (EFI_ERROR (Status)) {return Status;}}

它在发送请求时使用。

EFI_DNS4_PROTOCOL

该Protocol的结构体如下:

///
/// The EFI_DNS4_Protocol provides the function to get the host name and address
/// mapping, also provides pass through interface to retrieve arbitrary information
/// from DNS.
///
struct _EFI_DNS4_PROTOCOL {EFI_DNS4_GET_MODE_DATA       GetModeData;EFI_DNS4_CONFIGURE           Configure;EFI_DNS4_HOST_NAME_TO_IP     HostNameToIp;EFI_DNS4_IP_TO_HOST_NAME     IpToHostName;EFI_DNS4_GENERAL_LOOKUP      GeneralLookUp;EFI_DNS4_UPDATE_DNS_CACHE    UpdateDnsCache;EFI_DNS4_POLL                Poll;EFI_DNS4_CANCEL              Cancel;
};

DoDnsQuery

本函数是DNS4模块中最重要的部分,其实现也比较简单:

EFI_STATUS
DoDnsQuery (IN  DNS_INSTANCE  *Instance,IN  NET_BUF       *Packet)
{//// Ready to receive the DNS response.//if (Instance->UdpIo->RecvRequest == NULL) {Status = UdpIoRecvDatagram (Instance->UdpIo, DnsOnPacketReceived, Instance, 0);if (EFI_ERROR (Status)) {return Status;}}//// Transmit the DNS packet.//NET_GET_REF (Packet);Status = UdpIoSendDatagram (Instance->UdpIo, Packet, NULL, NULL, DnsOnPacketSent, Instance);
}

它最终调用的就是UDP4接口:

EFI_STATUS
EFIAPI
UdpIoSendDatagram (IN  UDP_IO           *UdpIo,IN  NET_BUF          *Packet,IN  UDP_END_POINT    *EndPoint OPTIONAL,IN  EFI_IP_ADDRESS   *Gateway  OPTIONAL,IN  UDP_IO_CALLBACK  CallBack,IN  VOID             *Context)
{TxToken = UdpIoCreateTxToken (UdpIo, Packet, EndPoint, Gateway, CallBack, Context);//// Insert the tx token into SendDatagram list before transmitting it. Remove// it from the list if the returned status is not EFI_SUCCESS.//InsertHeadList (&UdpIo->SentDatagram, &TxToken->Link);if (UdpIo->UdpVersion == UDP_IO_UDP4_VERSION) {Status = UdpIo->Protocol.Udp4->Transmit (UdpIo->Protocol.Udp4, &TxToken->Token.Udp4);}

最终的调用流程:

EFI_DNS4_PROTOCOL.HostNameToIp
Dns4HostNameToIp
DoDnsQuery
EFI_DNS4_PROTOCOL.GeneralLookUp
Dns4GeneralLookUp

这就与EFI_DNS4_PROTOCOL联系起来了。

这篇关于【UEFI基础】EDK网络框架(DNS4)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/632666

相关文章

Debian 13升级后网络转发等功能异常怎么办? 并非错误而是管理机制变更

《Debian13升级后网络转发等功能异常怎么办?并非错误而是管理机制变更》很多朋友反馈,更新到Debian13后网络转发等功能异常,这并非BUG而是Debian13Trixie调整... 日前 Debian 13 Trixie 发布后已经有众多网友升级到新版本,只不过升级后发现某些功能存在异常,例如网络转

GSON框架下将百度天气JSON数据转JavaBean

《GSON框架下将百度天气JSON数据转JavaBean》这篇文章主要为大家详细介绍了如何在GSON框架下实现将百度天气JSON数据转JavaBean,文中的示例代码讲解详细,感兴趣的小伙伴可以了解下... 目录前言一、百度天气jsON1、请求参数2、返回参数3、属性映射二、GSON属性映射实战1、类对象映

从基础到高级详解Python数值格式化输出的完全指南

《从基础到高级详解Python数值格式化输出的完全指南》在数据分析、金融计算和科学报告领域,数值格式化是提升可读性和专业性的关键技术,本文将深入解析Python中数值格式化输出的相关方法,感兴趣的小伙... 目录引言:数值格式化的核心价值一、基础格式化方法1.1 三种核心格式化方式对比1.2 基础格式化示例

redis-sentinel基础概念及部署流程

《redis-sentinel基础概念及部署流程》RedisSentinel是Redis的高可用解决方案,通过监控主从节点、自动故障转移、通知机制及配置提供,实现集群故障恢复与服务持续可用,核心组件包... 目录一. 引言二. 核心功能三. 核心组件四. 故障转移流程五. 服务部署六. sentinel部署

从基础到进阶详解Python条件判断的实用指南

《从基础到进阶详解Python条件判断的实用指南》本文将通过15个实战案例,带你大家掌握条件判断的核心技巧,并从基础语法到高级应用一网打尽,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一... 目录​引言:条件判断为何如此重要一、基础语法:三行代码构建决策系统二、多条件分支:elif的魔法三、

Python WebSockets 库从基础到实战使用举例

《PythonWebSockets库从基础到实战使用举例》WebSocket是一种全双工、持久化的网络通信协议,适用于需要低延迟的应用,如实时聊天、股票行情推送、在线协作、多人游戏等,本文给大家介... 目录1. 引言2. 为什么使用 WebSocket?3. 安装 WebSockets 库4. 使用 We

从基础到高阶详解Python多态实战应用指南

《从基础到高阶详解Python多态实战应用指南》这篇文章主要从基础到高阶为大家详细介绍Python中多态的相关应用与技巧,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录一、多态的本质:python的“鸭子类型”哲学二、多态的三大实战场景场景1:数据处理管道——统一处理不同数据格式

Python开发简易网络服务器的示例详解(新手入门)

《Python开发简易网络服务器的示例详解(新手入门)》网络服务器是互联网基础设施的核心组件,它本质上是一个持续运行的程序,负责监听特定端口,本文将使用Python开发一个简单的网络服务器,感兴趣的小... 目录网络服务器基础概念python内置服务器模块1. HTTP服务器模块2. Socket服务器模块

解决若依微服务框架启动报错的问题

《解决若依微服务框架启动报错的问题》Invalidboundstatement错误通常由MyBatis映射文件未正确加载或Nacos配置未读取导致,需检查XML的namespace与方法ID是否匹配,... 目录ruoyi-system模块报错报错详情nacos文件目录总结ruoyi-systnGLNYpe

MySQL数据类型与表操作全指南( 从基础到高级实践)

《MySQL数据类型与表操作全指南(从基础到高级实践)》本文详解MySQL数据类型分类(数值、日期/时间、字符串)及表操作(创建、修改、维护),涵盖优化技巧如数据类型选择、备份、分区,强调规范设计与... 目录mysql数据类型详解数值类型日期时间类型字符串类型表操作全解析创建表修改表结构添加列修改列删除列