基于WDF的PCI/PCIe接口卡Windows驱动程序(5)-如何为硬件移植驱动程序

时间:2022-09-05 10:41:30

原文地址:http://www.cnblogs.com/jacklu/p/6139347.html

正如前几篇博客所说,使用WDF开发PCIe驱动程序是我本科毕业设计的主要工作。在读研的两年,我也分别为所在课题组移植了自己编写的驱动程序,在Windows 32位和64位平台下的PXI、PXIe、PCI、PCIe板卡分别得到了验证。

这篇文章根据自己最新编写的驱动代码(源代码请找博主索取),主要讲述如何为自己的硬件板卡移植驱动程序,并简单讲述如何使用Altera系列FPGA配置PCI IP核,然后对INF文件作简要描述,最后描述如何使用Qt编写上位机软件调用底层驱动。

准备去读博了,这一篇将作为这个系列的完结,之后将把更多精力放到机器学习上来。

1概述

所编写的驱动代码程序包括7个源代码文件,分别是Device.h, Driver.h, Public.h, Trace.h, Device.c,Driver.c, Queue.c。其中Device.h 定义了与硬件相关的地址偏移;Public.h定义了DeviceIoControl 用到的用户自定义命令字, 此头文件由上层应用程序和驱动程序共同使用;Queue.c定义了I/O回调例程,分别使用了read、write和I/O Control 三个队列。 除了这三个文件外, 不建议更改其他4个文件的代码。三个源文件函数列表分别如图 1-1、1-2、1-3所示:

基于WDF的PCI/PCIe接口卡Windows驱动程序(5)-如何为硬件移植驱动程序

2驱动程序移植说明

2.1 Public.h说明

  • 代码中定义了GUID值,开发者可以使用 VS2013 下的工具 GUIDGen.exe 生成 GUID值,该GUID标识驱动程序,应用程序根据这个GUID值来找到对应的驱动程序。
  • 代码中定义了CTL_CODE, I/O 处理例程 DeviceIoControl 的第二个参数dwIoControlCode 就是由 CTL_CODE 宏定义的。 CTL_CODE 是一个用于创建一个唯一的32 位系统I/O控制代码的宏,这个控制代码包括4部分组成:DeviceType(设备类型,高16位(16-31 位)),Access(访问限制,14-15 位),Function(功能2-13位),Method(I/O访问内存使用方式)。CTL_CODE 定义中有一个Method域,该2域定义了驱动程序中获取应用程序数据缓冲区的地址方式。 已经定义的 CTL_CODE 命令字说明如表 2-1 所示, 用户可以根据格式自定义 CTL_CODE, 实现不同功能。

基于WDF的PCI/PCIe接口卡Windows驱动程序(5)-如何为硬件移植驱动程序

什么是 CRA 寄存器组? 打开 quartus 的 sopc builder,可以看到sopc架构如图 2-1 所示,在PCI IP核一栏中有 Control Register Access 寄存器组,地址范围0x00000000-0x00003fff。 里面关键的寄存器地址如图 2-2 所示。 通过读黄色标识的寄存器,可以通过驱动程序调试验证 PCI 核。 关于 CRA 寄存器组的配置说明会在 2.3 节详细说明。

基于WDF的PCI/PCIe接口卡Windows驱动程序(5)-如何为硬件移植驱动程序

基于WDF的PCI/PCIe接口卡Windows驱动程序(5)-如何为硬件移植驱动程序

2.2 Device.h说明

  • 代码对 FPGA 上硬件资源的偏移地址进行宏定义,在 Altera 系列的 FPGA 里,这些偏移地址也叫 Avalon 地址, 在 sopc builder 可以自定义分配, 如图 2-1 中的 Base和End 所示, “ 小锁头” 标志表示地址锁定, 点击该标志解锁后可以自定义便宜地址。这些地址必须与驱动程序中所用的地址一一对应;
  • 代码定义了设备对象结构体, 对几个重要的成员变量注释如下:

基于WDF的PCI/PCIe接口卡Windows驱动程序(5)-如何为硬件移植驱动程序

  • 代码对一些事件回调例程进行了说明, 一般不需要用户进行二次修改;

2.3 Queue.c说明

  • 代码是用户需要针对功能开发的代码。 以从应用程序向驱动程序写入偏移地址为例,即原代码第 xx-xx 行 , 首先在 Public.h 文件里定义 IoControlCode 为qd41_IOCTL_WRITE_OFFSETADDRESS, 第 48-53 行为获取应用程序输入缓存数据的指针( 代码中为 inBuffer) , 通过赋值语句将 inBuffer 指向的数据内容赋给设备对象的成员变量 OffseAddressFromApp, 即完成了本功能;
  • 代码定义了写队列功能, 对 DMA 的写功能配置在此函数中完成。 Altera的 DMA IP 核共有 5 个寄存器, 如图 2-3 所示。

基于WDF的PCI/PCIe接口卡Windows驱动程序(5)-如何为硬件移植驱动程序

  • 在配置 DMA 前需要配置 PCI CRA 寄存器, 使能 PCI 中断, 配置 Avalon-PCI 地址转换表, 如代码所示;
  • 为什么需要配置 Avalon-PCI 地址转换表? 因为 PCI IP核一端是 PCI 总线,一端是Avalon总线,地址转换过程图 2-4 所示,类似MMU地址翻译原理,不在赘述,此时需要把PC机获得的DMA传输缓存物理地址的高16位地址写入地址转换表;

基于WDF的PCI/PCIe接口卡Windows驱动程序(5)-如何为硬件移植驱动程序

  • 配置好 PCI 后就可以配置 DMA 控制寄存器了, 首先将状态寄存器和控制寄存器清零,如代码所示;
  • 然后将读写地址分别写入读写寄存器, 注意PC机上的内存地址为低16位,而高16位要配置在 Avalon-PCI 地址转换表中, 如代码所示, 其中 0x20000 为 PCI IP核的 Avalon 地址, 如图 2-1 所示;
  • 向长度寄存器写入 DMA 传输长度(单位:字节) , 如代码所示;
  • 向控制寄存器写入 DMA 控制字 0x8c, 即长度寄存器降低为0时传输完成, 使能DMA,32 位字传输, 如代码所示;

3应用程序如何调用驱动程序

3.1 GUID说明

GUID( Globally Unique Identifier) 是微软推出的全局唯一标识符, 通过使用某个特定的算法( 比如根据时间或地点等信息) 生成一组128 位二进制数,来标识某一个实体,比如硬盘上的一张图片。 GUID 广泛应用于微软的产品中, 用于识别接口、文件等对象。开发者可以使用 VS2013 下的工具 GUIDGen.exe 生成 GUID 值, 该 GUID 标识驱动程序, 应用程序根据这个 GUID 值来找到对应的驱动程序。

应用程序总体流程设计为: 首先通过 Win32API 函数 CreateFile 打开设备, 然后调用DeviceIoControl 函数与驱动程序通信,即读写数据,当应用程序退出时,调用CloseHandle函数关闭设备。

应用程序根据底层设置的 GUID 获取设备路径, 从而与指定设备建立连接: 首先调用SetupDiGetClassDevs 函数获得符合传入参数 GUID 的设备信息集合 hDevInfo; 然后根据设备信息集合 hDevInfo 和 GUID 调用 SetupDiEnumDeviceInterfaces 函数枚举设备信息集合中的设备,并输出设备接口数据信息 DeviceInterfaceData ; 再 根 据 hDevInfo和 DeviceInterfaceData 调用 SetupDiGetDeviceInterfaceDetail函数得到保存设备接口详细信息DeviceInterfaceDetailData 的缓冲区大小 ,接着为其开辟空间 , 最后再调用一次SetupDiGetDeviceInterfaceDetail 函数获得设备接口详细信息 pDeviceInterfaceDetailData; 最后根据设备接口详细信息的设备接口路径 pDeviceInterfaceDetailData->DevicePath 调用CreateFile 函数来创建设备句柄。

3.2 应用程序如何打开设备

在测试程序 Source.c 的代码第 62-70 行,完成打开设备句柄功能, 不需要用户改动。

3.3 应用程序如何读设备

成功打开设备后, 调用 DeviceIoControl 即可通过 IOControl 队列与驱动程序通信, 以读32bit 数据为例, 首先向驱动程序写入需要读的存储单元地址, 如代码第 257-272 行所示,然后向驱动程序传入读数据缓存 outBuffer, 如代码第 274-289 所示, 即可获得FPGA 上对应偏移地址的数据。

3.4 应用程序如何写设备

成功打开设备后, 调用 DeviceIoControl 即可通过 IOControl 队列与驱动程序通信, 以写32bit 数据为例, 首先向驱动程序写入需要写的存储单元地址, 如代码第 303-317 行所示,然后向驱动程序传入缩写数据缓存 inBuffer, 如代码第 322-336 所示, 即可将数据写入 FPGA上对应偏移地址的内存单元。

3.5 应用程序如何对设备进行DMA传输

通过Win32API 函数ReadFile和WriteFile对设备句柄进行文件读写操作, 在内核驱动中会调用驱动程序的 qd41EvtIoRead和qd41EvtIoWrite,实际的DMA配置在这两个驱动回调例程中实现, ReadFile和WriteFile 只是完成了数据从用户层到内核层的搬运。详细代码在第 340-378 行,不再赘述。

4 INF文件与如何调用Qt编写上位机软件调用底层驱动

基于WDF的PCI/PCIe接口卡Windows驱动程序(5)-如何为硬件移植驱动程序

基于WDF的PCI/PCIe接口卡Windows驱动程序(5)-如何为硬件移植驱动程序

基于WDF的PCI/PCIe接口卡Windows驱动程序(5)-如何为硬件移植驱动程序

基于WDF的PCI/PCIe接口卡Windows驱动程序(5)-如何为硬件移植驱动程序

我把C语言编写的应用程序每一个独立的控制命令编译成一个可执行文件,这样Qt可以在新的进程中调用 C 程序编写的命令字读写程序。使用Qt 封装好的进程类 QProcess, 使用其成员函数 start 即可调用一个外部程序。

基于WDF的PCI/PCIe接口卡Windows驱动程序(5)-如何为硬件移植驱动程序的更多相关文章

  1. 基于WDF的PCI/PCIe接口卡Windows驱动程序(2)-开发者需要了解的WDF中的一些重要的概念

    原文出处:http://www.cnblogs.com/jacklu/p/4646601.html 本科毕业设计是这方面的工作,所以想开几篇博客来介绍使用WDF开发PCI/PCIe接口卡的驱动程序方法 ...

  2. 基于WDF的PCI/PCIe接口卡Windows驱动程序(1)-WDF概述及开发环境搭建

    原文出处:http://www.cnblogs.com/jacklu/p/4619110.html 本科毕业设计是这方面的工作,所以想开几篇博客来介绍使用WDF开发PCI/PCIe接口卡的驱动程序方法 ...

  3. 基于WDF的PCI/PCIe接口卡Windows驱动程序(4)- 驱动程序代码(源文件)

    原文出处:http://www.cnblogs.com/jacklu/p/4687325.html 本篇文章将对PCIe驱动程序的源文件代码作详细解释与说明.整个WDF驱动程序工程共包含4个头文件(已 ...

  4. 基于WDF的PCI/PCIe接口卡Windows驱动程序(3)- 驱动程序代码(头文件)

    原文出处:http://www.cnblogs.com/jacklu/p/4679304.html 在WDF的PCIe驱动程序中,共有四个.h文件(Public.h  Driver.h  Device ...

  5. Q35+uefi or bios+legacy // PCI | PCIE

    1:首先统一可扩展固件接口(UEFI)是一种规范定义操作系统和平台固件之间的软件接口. UEFI旨在替代基本输入/输出系统(BIOS)固件接口.(legacy) 硬件平台厂商越来越多地采用UEFI管理 ...

  6. PCIE_DMA实例五:基于XILINX XDMA的PCIE高速采集卡

    PCIE_DMA实例五:基于XILINX XDMA的PCIE高速采集卡 一:前言 这一年关于PCIE高速采集卡的业务量激增,究其原因,发现百度"xilinx pcie dma",出 ...

  7. KVM 介绍(4):I/O 设备直接分配和 SR-IOV [KVM PCI/PCIe Pass-Through SR-IOV]

    学习 KVM 的系列文章: (1)介绍和安装 (2)CPU 和 内存虚拟化 (3)I/O QEMU 全虚拟化和准虚拟化(Para-virtulizaiton) (4)I/O PCI/PCIe设备直接分 ...

  8. windows ODBC数据源里没有Oracle的驱动程序

    windows ODBC数据源里没有Oracle的驱动程序   直接在“控制面板---管理工具----数据源(ODBC)”   打开数据源配置,发现只有SQLServer的驱动,其他的都没有了.   ...

  9. Linux PCI/PCI-E设备配置空间读取与修改

    Linux PCI/PCI-E设备配置空间读取与修改 1 前言 PCI和PCI Express,是计算机常使用的一种高速总线.操作系统中的PCI/PCI-E设备驱动以及操作系统内核,都需要访问PCI及 ...

随机推荐

  1. BZOJ1568: [JSOI2008]Blue Mary开公司

    可以平衡树或线段树维护斜率来做. 还有一种线段树直接打标记的做法: 线段树每个节点存一条线段作为标记,打标记时如果已有标记,则把占优区间小的那个线段下放. #include<cstdio> ...

  2. 序列GCD和问题&lpar;题目&rpar;

    序列GCD和 题目描述 Massacc有一个序列$A_1,A_2,A_3,\dots ,A_n$. Popbab说:我要知道这个序列的和$\pmod{1\times10^9+7}$. Massacc在 ...

  3. cocos2d-x 3&period;0 使用最新物理引擎的一个源代码实例

    1.碰撞函数參数由两个变成一个了 2.检測不到碰撞.须要设置那三个參数.同一时候还要设置成动态的. body进行设置. 3.初始入口文件也发生了改变. 附录上我近期调试好的cocos2d-x 3.1 ...

  4. PXE搭建

    前提最好是防火墙规则-F,关闭,selinux 是disable 这个在以后更新linux系统的时候还可以在这个基础上再次增加可以一体化安装的系统. 1.用yum来安装所需要的软件包,先来搭建yum光 ...

  5. tomcat和iis共用80端口的简明手册

    ​​对于使用tomcat-connector实现iis与tomcat实现80端口共用的问题,网上的信息异常混乱,很多地方误人子弟,浪费时间.本文给出简明手册式的做法: 首先列出我们需要做的事项: 1. ...

  6. p211有界自共轭算子T是实数集合的子集

    对条件 取非   是 ∉谱集合的实数 才对  现在是  入  属于正则点集 他  然后  又说T 的谱是实数 这不矛盾吗 这里根据   必要性    推出 蓝色和红色矛盾    矛盾就是 这是谱点 然 ...

  7. Python中使用RabbitMQ

    一 RabbitMQ简介 RabbitMQ是一个在AMQP基础上完整的,可复用的企业消息系统.他遵循Mozilla Public License开源协议. MQ全称为Message Queue, 消息 ...

  8. android 的几个黄色警告解决办法&lpar;转&rpar;

    转自:http://my.eoe.cn/864234/archive/5162.html 1:Handler 1 2 3 4 5 6 7 8 // This Handler class should ...

  9. 谷歌笔试题--给定一个集合A&equals;&lbrack;0&comma;1&comma;3&comma;8&rsqb;&lpar;该集合中的元素都是在0,9之间的数字,但未必全部包含&rpar;, 指定任意一个正整数K,请用A中的元素组成一个大于K的最小正整数。

    谷歌笔试题--给定一个集合A=[0,1,3,8](该集合中的元素都是在0,9之间的数字,但未必全部包含), 指定任意一个正整数K,请用A中的元素组成一个大于K的最小正整数. Google2009华南地 ...

  10. 为通过 ATS 检测 Tomcat 完全 TLS v1&period;2、完全正向加密及其结果检验

    2017 年起 app store 要求 app 对接的服务器支持 TLS v1.2,否则 ats 检测不予通过.有点强制推 TLS v1.2 的意味.本文介绍如何使 tomcat 强制执行 TLS ...