NDIS协议驱动开发指南

这篇具有很好参考价值的文章主要介绍了NDIS协议驱动开发指南。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

NDIS协议驱动开发指南

我们知道,在以太网中所有的数据包都是通过以太网帧来发送的;但是在网络上面的应用程序如果需要通过网络数据包交互,就需要依赖网络协议来保障通信。平时我们用的最多的协议就是TCPIP协议。

其实在Windows中,我们可以注册自己的协议,开发自己的协议解析和封装驱动,实现以太网帧的通信,这就是本文的NDIS协议驱动。

以太网的以太包格式都是固定的,格式如下:

6字节 6字节 2字节 其他长度
源MAC 目的MAC 类型 数据部分

NDIS协议驱动就是针对以太包的协议封装和解析过程,对上层提供一个稳定的数据包通信的协议,对下层提供一个可以供以太网发送的以太数据包。本文我们来看一下NDIS协议驱动的开发原理。

1. 技术概览

在Windows下面,网络栈的基本架构如下:
基于ndis protocol driver的底层数据包发送,Windows驱动开发,驱动开发,c++,c语言,网络,windows

对于NDSI提供了三种功能(能力)的驱动:

  1. 上层的协议驱动。
  2. 中层的过滤驱动。
  3. 下层的小端口驱动。

由于NDIS早期并没有直接提供过滤层的基本框架,因此对于早期(XP系统下面)版本如果需要对NDIS层的数据包进行过滤,需要在中间层实现协议驱动和小端口驱动:

  • 对上层,创建小端口驱动来和上层协议层通信(主要过滤数据包的发送)。
  • 对下层,创建协议驱动来和小端口驱动通信(主要过滤数据包的接收)。

对于协议层驱动,实现比较简单,只需要设置和处理好NDIS相关的协议层回调函数即可,下面我们看一下协议层驱动的具体实现。

2. NDIS协议驱动

上面我们知道NDIS协议驱动主要处理NDIS的相关协议回调例程,向NDIS注册回调例程的函数为NdisRegisterProtocolDriver,该例程声明如下:

NDIS_STATUS NdisRegisterProtocolDriver(
  NDIS_HANDLE                           ProtocolDriverContext,
  PNDIS_PROTOCOL_DRIVER_CHARACTERISTICS ProtocolCharacteristics,
  PNDIS_HANDLE                          NdisProtocolHandle
);

其中NDIS_PROTOCOL_DRIVER_CHARACTERISTICS就是协议驱动的回调函数结构体,该结构体如下:

typedef struct _NDIS_PROTOCOL_DRIVER_CHARACTERISTICS {
  NDIS_OBJECT_HEADER                     Header;
  UCHAR                                  MajorNdisVersion;
  UCHAR                                  MinorNdisVersion;
  UCHAR                                  MajorDriverVersion;
  UCHAR                                  MinorDriverVersion;
  ULONG                                  Flags;
  NDIS_STRING                            Name;
  SET_OPTIONS_HANDLER                    SetOptionsHandler;
  BIND_HANDLER_EX                        BindAdapterHandlerEx;
  UNBIND_HANDLER_EX                      UnbindAdapterHandlerEx;
  OPEN_ADAPTER_COMPLETE_HANDLER_EX       OpenAdapterCompleteHandlerEx;
  CLOSE_ADAPTER_COMPLETE_HANDLER_EX      CloseAdapterCompleteHandlerEx;
  NET_PNP_EVENT_HANDLER                  NetPnPEventHandler;
  UNINSTALL_PROTOCOL_HANDLER             UninstallHandler;
  OID_REQUEST_COMPLETE_HANDLER           OidRequestCompleteHandler;
  STATUS_HANDLER_EX                      StatusHandlerEx;
  RECEIVE_NET_BUFFER_LISTS_HANDLER       ReceiveNetBufferListsHandler;
  SEND_NET_BUFFER_LISTS_COMPLETE_HANDLER SendNetBufferListsCompleteHandler;
  DIRECT_OID_REQUEST_COMPLETE_HANDLER    DirectOidRequestCompleteHandler;
} NDIS_PROTOCOL_DRIVER_CHARACTERISTICS, *PNDIS_PROTOCOL_DRIVER_CHARACTERISTICS;

在上述结构体中间,对于一个简要的协议驱动,只需要实现部分主要的回调函数即可,包括:

  • BindAdapterHandlerEx:绑定回调函数,当小端口驱动和协议驱动进行绑定的时候调用该函数通知协议驱动。
  • UnbindAdapterHandlerEx:解除绑定的回调函数,和BindAdapterHandlerEx相反。
  • OpenAdapterCompleteHandlerEx:当使用NdisOpenAdapterEx绑定小端口驱动完成的时候被调用(相当IRP的完成例程)。
  • CloseAdapterCompleteHandlerEx:当使用NdisCloseAdapterEx解除协议驱动和小端口驱动完成的时候被调用。
  • OidRequestCompleteHandlerNdisOidRequest请求完成的时候被调用的函数。
  • SendNetBufferListsCompleteHandler:表示使用NdisSendNetBufferLists发送完成数据包之后被调用的回调函数。
  • ReceiveNetBufferListsHandler:当小端口驱动接收到数据的时候就会通过该回调函数通知协议驱动数据包的到来。

2.1 BindAdapterHandlerEx

协议驱动是对网络数据包的封装,当将网络数据包按照协议封装为以太网数据包之后,就需要通过网卡发送出去,那么协议驱动就需要和网卡驱动进行关联(协议驱动的数据包知道如何发送给网卡驱动)。

有两种情况需要进行协议的绑定:

  1. 当协议驱动使用NdisRegisterProtocolDriver注册驱动的时候,NDIS框架就会遍历当前系统所有的小端口驱动,对每个小端口驱动调用BindAdapterHandlerEx回调函数。
  2. 当有新的网卡设备插入并启动的时候(IRP_MN_START),就会对该小端口驱动遍历所有的协议驱动,然后调用其BindAdapterHandlerEx回调函数进行绑定。

BindAdapterHandlerEx只是绑定的回调函数,该函数声明如下:

PROTOCOL_BIND_ADAPTER_EX ProtocolBindAdapterEx;

NDIS_STATUS ProtocolBindAdapterEx(
  NDIS_HANDLE ProtocolDriverContext,
  NDIS_HANDLE BindContext,
  PNDIS_BIND_PARAMETERS BindParameters
)
{...}

该函数只是将小端口驱动和协议驱动的信息当作回调函数的参数传递过来,小端口驱动的信息通过PNDIS_BIND_PARAMETERS进行描述。真实的绑定是通过NdisOpenAdapterEx函数来完成的,该函数如下:

NDIS_STATUS NdisOpenAdapterEx(
  NDIS_HANDLE           NdisProtocolHandle,
  NDIS_HANDLE           ProtocolBindingContext,
  PNDIS_OPEN_PARAMETERS OpenParameters,
  NDIS_HANDLE           BindContext,
  PNDIS_HANDLE          NdisBindingHandle
);

NdisOpenAdapterEx其实是建立小端口和协议驱动的桥梁,大致如下:

基于ndis protocol driver的底层数据包发送,Windows驱动开发,驱动开发,c++,c语言,网络,windows

这样小端口的数据可以在NDIS框架中通过NDIS_OPEN_BLOCK回调给协议驱动,协议驱动也可以通过NDIS_OPEN_BLOCK调用小端口驱动。

2.2 SendNetBufferListsCompleteHandler

在协议驱动中,我们通过NdisSendNetBufferLists将以太包发送数据,该函数声明如下:

void NdisSendNetBufferLists(
  NDIS_HANDLE                       NdisBindingHandle,
  __drv_aliasesMem PNET_BUFFER_LIST NetBufferLists,
  NDIS_PORT_NUMBER                  PortNumber,
  ULONG                             SendFlags
);

NET_BUFFER_LIST描述着我们需要发送的数据包集合,当小端口驱动将数据包发送成功之后,就会调用NdisMSendNetBufferListsComplete来通知协议驱动数据包被发送完成,该函数如下:

void NdisMSendNetBufferListsComplete(
  NDIS_HANDLE      MiniportAdapterHandle,
  PNET_BUFFER_LIST NetBufferList,
  ULONG            SendCompleteFlags
);

NdisMSendNetBufferListsComplete完成回调的函数就是SendNetBufferListsCompleteHandler,该函数如下:

PROTOCOL_SEND_NET_BUFFER_LISTS_COMPLETE ProtocolSendNetBufferListsComplete;

void ProtocolSendNetBufferListsComplete(
  NDIS_HANDLE ProtocolBindingContext,
  PNET_BUFFER_LIST NetBufferList,
  ULONG SendCompleteFlags
)
{...}

SendNetBufferListsCompleteHandler这个函数将会重新获取NdisSendNetBufferLists发送的数据包,在该函数中可以释放发送分配的NET_BUFFER_LIST内存。

不过这里需要注意的是NET_BUFFER_LIST是一个链表结构,在底层可能会被断链发送;因此SendNetBufferListsCompleteHandler这个函数中的NET_BUFFER_LIST可能是NdisSendNetBufferLists中的子链。

2.3 ReceiveNetBufferListsHandler

当我们的网卡接收到数据的时候,就会通过硬件方式(例如中断等)通知数据包到来,然后小端口驱动通过NdisMIndicateReceiveNetBufferLists将数据包传送给协议层驱动进行协议解析和数据包的传递,该函数如下:

void NdisMIndicateReceiveNetBufferLists(
  NDIS_HANDLE      MiniportAdapterHandle,
  PNET_BUFFER_LIST NetBufferList,
  NDIS_PORT_NUMBER PortNumber,
  ULONG            NumberOfNetBufferLists,
  ULONG            ReceiveFlags
);

NdisMIndicateReceiveNetBufferLists内部就会调用ReceiveNetBufferListsHandler将接收到的网络数据包通过NET_BUFFER_LIST进行传递,该函数声明如下:

PROTOCOL_RECEIVE_NET_BUFFER_LISTS ProtocolReceiveNetBufferLists;

void ProtocolReceiveNetBufferLists(
  NDIS_HANDLE ProtocolBindingContext,
  PNET_BUFFER_LIST NetBufferLists,
  NDIS_PORT_NUMBER PortNumber,
  ULONG NumberOfNetBufferLists,
  ULONG ReceiveFlags
)
{...}

在该函数中NetBufferLists表示数据包,通过解析该数据包我们就可以进行TCPIP网络栈的协议解析了。

2.4 ProtocolNetPnpEvent

改例程是NDIS框架对于网络PNP事件响应的回调函数,该函数声明如下:

PROTOCOL_NET_PNP_EVENT ProtocolNetPnpEvent;

NDIS_STATUS ProtocolNetPnpEvent(
  NDIS_HANDLE ProtocolBindingContext,
  PNET_PNP_EVENT_NOTIFICATION NetPnPEventNotification
)
{...}

NET_PNP_EVENT_NOTIFICATION描述了一个PNP事件的信息,该结构如下:

typedef struct _NET_PNP_EVENT_NOTIFICATION {
  NDIS_OBJECT_HEADER       Header;
  NDIS_PORT_NUMBER         PortNumber;
  NET_PNP_EVENT            NetPnPEvent;
  ULONG                    Flags;
  NDIS_NIC_SWITCH_ID       SwitchId;
  NDIS_NIC_SWITCH_VPORT_ID VPortId;
} NET_PNP_EVENT_NOTIFICATION, *PNET_PNP_EVENT_NOTIFICATION;

typedef struct _NET_PNP_EVENT {
  NET_PNP_EVENT_CODE NetEvent;
  PVOID              Buffer;
  ULONG              BufferLength;
  ULONG_PTR          NdisReserved[4];
  ULONG_PTR          TransportReserved[4];
  ULONG_PTR          TdiReserved[4];
  ULONG_PTR          TdiClientReserved[4];
} NET_PNP_EVENT, *PNET_PNP_EVENT;

NET_PNP_EVENT_CODE描述网络PNP事件的类型,有如下:

typedef enum _NET_PNP_EVENT_CODE
{
    NetEventSetPower,
    NetEventQueryPower,
    NetEventQueryRemoveDevice,
    NetEventCancelRemoveDevice,
    NetEventReconfigure,
    NetEventBindList,
    NetEventBindsComplete,
    NetEventPnPCapabilities,
    NetEventPause,
    NetEventRestart,
    NetEventPortActivation,
    NetEventPortDeactivation,
    NetEventIMReEnableDevice,
    NetEventNDKEnable,
    NetEventNDKDisable,
    NetEventFilterPreDetach,
    NetEventBindFailed,
    NetEventSwitchActivate,
    NetEventAllowBindsAbove,
    NetEventInhibitBindsAbove,
    NetEventAllowStart,
    NetEventRequirePause,
    NetEventUploadGftFlowEntries,
    NetEventMaximum
} NET_PNP_EVENT_CODE, *PNET_PNP_EVENT_CODE;

NET_PNP_EVENT_CODE的具体值,参见MSDN(例如NetEventSetPower类型Buffer表示了NDIS_DEVICE_POWER_STATE结构,描述电源状态)。

3. NET_BUFFER_LIST

在NDIS中,网络数据包通过NET_BUFFER_LIST来进行抽象,这个结构表示着网络数据包的集合;该数据结构如下:
基于ndis protocol driver的底层数据包发送,Windows驱动开发,驱动开发,c++,c语言,网络,windows

NET_BUFFER_LIST是一个NET_BUFFER_LIST的链表集合;单个NET_BUFFER_LISTNET_BUFFER的集合,NET_BUFFER表示一个数据包,该结构如下:
基于ndis protocol driver的底层数据包发送,Windows驱动开发,驱动开发,c++,c语言,网络,windows

NET_BUFFER其实就是使用MDL来描述数据包的真实类容,因此对于NET_BUFFER_LIST的全部结构可以描述为如下:
基于ndis protocol driver的底层数据包发送,Windows驱动开发,驱动开发,c++,c语言,网络,windows

对于NET_BUFFER_LISTNET_BUFFER提供了如下宏来操作该结构的成员:

#define NET_BUFFER_LIST_NEXT_NBL(_NBL)              ((_NBL)->Next)
#define NET_BUFFER_LIST_FIRST_NB(_NBL)              ((_NBL)->FirstNetBuffer)

#define NET_BUFFER_NEXT_NB(_NB)                     ((_NB)->Next)
#define NET_BUFFER_FIRST_MDL(_NB)                   ((_NB)->MdlChain)
#define NET_BUFFER_DATA_LENGTH(_NB)                 ((_NB)->DataLength)
#define NET_BUFFER_DATA_OFFSET(_NB)                 ((_NB)->DataOffset)
#define NET_BUFFER_CURRENT_MDL(_NB)                 ((_NB)->CurrentMdl)
#define NET_BUFFER_CURRENT_MDL_OFFSET(_NB)          ((_NB)->CurrentMdlOffset)

我们需要对接收或者发送的数据包进行处理,都是解析NET_BUFFER_LIST的过程。

4. ndisprot实例

对于NDIS协议驱动,WDK提供了一个示例ndisprot,该示例展示了协议驱动的工作原理,该实例提供如下功能:

  1. ndisprot驱动可以绑定到网卡上面。
  2. 通过NdisprotReceiveNetBufferLists接收底层的网络数据包,并将其放入队列中。
  3. 用户层程序可以通过ReadFile读取协议驱动的网络数据包。
  4. 用户层程序可以通过WriteFile往网络协议驱动写入数据包,协议驱动通过NdisSendNetBufferLists将数据包发送到底层小端口驱动。

该协议否是可以支持网络通信呢?本人没有进行实验验证,但是从原理上来说是可行的,只是它是一个面向非连接,并且没有校验的原始通信手段的协议。

对于该驱动我们可以简单的使用如下方式手动安装和验证:
基于ndis protocol driver的底层数据包发送,Windows驱动开发,驱动开发,c++,c语言,网络,windows

5. 总结

对于NDIS协议层驱动平时我们的使用场景不多,我们也没有能力(也没必要)设计一个完整的网络协议驱动。但是协议驱动是我们后面NDIS过滤驱动的基础,NDIS过滤驱动可以帮助我们获取本机接收到的网络帧数据包,并且对以太帧数据包进行过滤(例如ARP数据包等)。

因此NDIS协议驱动还是非常值得我们去学习的,它是NDIS过滤驱动的基础,也是以太帧数据包过滤的重要手段。文章来源地址https://www.toymoban.com/news/detail-776760.html

到了这里,关于NDIS协议驱动开发指南的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • Linux SPI-NAND 驱动开发指南

    1.1 编写目的 介绍 Sunxi SPINand mtd/ubi 驱动设计, 方便相关驱动和应用开发人员 1.2 适用范围 本设计适用于所有 sunxi 平台 1.3 相关人员 Nand 模块开发人员,及应用开发人员等 2 术语、缩略语及概念 MTD :(Memory Technology device)是用于访问存储设备的 linux 子系统。本模块是MTD 子系统

    2024年02月04日
    浏览(35)
  • 第22章 自旋锁死锁实验(iTOP-RK3568开发板驱动开发指南 )

    瑞芯微RK3568芯片是一款定位中高端的通用型SOC,采用22nm制程工艺,搭载一颗四核Cortex-A55处理器和Mali G52 2EE 图形处理器。RK3568 支持4K 解码和 1080P 编码,支持SATA/PCIE/USB3.0 外围接口。RK3568内置独立NPU,可用于轻量级人工智能应用。RK3568 支持安卓 11 和 linux 系统,主要面向物联网

    2024年02月09日
    浏览(45)
  • 第19章 并发与竞争实验(iTOP-RK3568开发板驱动开发指南 )

    瑞芯微RK3568芯片是一款定位中高端的通用型SOC,采用22nm制程工艺,搭载一颗四核Cortex-A55处理器和Mali G52 2EE 图形处理器。RK3568 支持4K 解码和 1080P 编码,支持SATA/PCIE/USB3.0 外围接口。RK3568内置独立NPU,可用于轻量级人工智能应用。RK3568 支持安卓 11 和 linux 系统,主要面向物联网

    2024年02月09日
    浏览(45)
  • Taro小程序隐私协议开发指南填坑

    一. 配置文件 app.config.js 二. 开发者工具基础库修改 原因: 从基础库 2.32.3 开始支持 修改路径:详情-本地设置-调试基础库 三. 用户隐私保护指引更新 修改路径:mp后台-设置-服务内容声明-用户隐私保护指引 隐私接口: 直达文档 报错: { \\\"errMsg\\\": \\\"A:fail api scope is not declared in

    2024年02月07日
    浏览(44)
  • 嵌入式linux驱动开发之移远4G模块EC800驱动移植指南

    回顾下移远4G模块移植过程, 还是蛮简单的。一通百通,无论是其他4G模块都是一样的。这里记录下过程,分享给有需要的人。环境使用正点原子的imax6ul开发板,板子默认支持中兴和移远EC20的驱动,这里要移植使用的是移远4G模块EC800。 imax6ul开发板 虚拟机(Ubuntu18.04) 交叉编译

    2024年02月17日
    浏览(63)
  • 嵌入式linux之iMX6ULL驱动开发 | 移远4G模块EC800驱动移植指南

    回顾下移远4G模块移植过程, 还是蛮简单的。一通百通,无论是其他4G模块都是一样的。这里记录下过程,分享给有需要的人。环境使用正点原子的imax6ul开发板,板子默认支持中兴和移远EC20的驱动,这里要移植使用的是移远4G模块EC800。 imax6ul开发板 虚拟机(Ubuntu18.04) 交叉编译

    2024年02月12日
    浏览(55)
  • uniapp适配微信隐私协议开发指南[uniapp+vue3+js]

    没怎么做过uniapp,找了一些文章做了出来,给大家分享一下 2023.9.15以后需要适配微信的隐私协议开发指南 目前uniapp的说法是微信小程序隐私协议开发指南 | uni-app官网 微信小程序小程序隐私协议开发指南 | 微信开放文档 微信官方提供了几个demo demo1: 演示使用  wx.getPrivacySet

    2024年02月07日
    浏览(55)
  • HarmonyOS鸿蒙开发指南:基于ArkTS开发 音频渲染开发指导

    目录 场景介绍 音频中断 状态检查 异步操作 开发步骤

    2024年01月16日
    浏览(58)
  • HarmonyOS鸿蒙开发指南:基于ArkTS开发 音频录制开发指导

    目录 场景介绍 开发步骤 全流程场景 正常录制场景 音频录制的主要工作是捕获音频信号

    2024年01月19日
    浏览(69)
  • HarmonyOS鸿蒙开发指南:基于ArkTS的声明式开发范式 声明式UI开发实例 绘图与动画

    目录 绘制图形 绘制基本几何图形 绘制自定义几何图形 animateTo实现闪屏动画 页面转场动画 绘制能力主要是通过框架提供的绘制组件来支撑,支持svg标准绘制命令。 本节主要学习如何使用绘制组件,绘制详情页食物成分标签(基本几何图形)和应用Logo(自定义图形)。

    2024年01月17日
    浏览(60)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包