PCI驱动程序开发

2019-08-01 15:26发布

 0 引 言
  计算机系统总是通过总线(Bus)实现相互间信息或数据交换的。这些定向的信息流和数据流在总线中流动,就形成计算机系统的各种操作,它能实现各种不同部件和设备之间的互连。
  PCI总线广泛使用在计算机中,一方面是因为该总线的数据吞吐量大,另一方面是因为该总线与具体的处理器无关。PCI总线的设计也使各种PCI外设卡可以直接插入PCI总线插槽中,而不需要考虑各种额外的特殊逻辑,在设计和使用PCI设备时,需要访问和控制硬件设备,如存储器读写、I/O端口访问、中断响应等。
  与工业标准结构(Industry Standard Architecture,ISA)设备不同的是:PCI硬件设备资源的分配不是硬件设计所决定的,而是由Windows操作系统根据PC机中所有硬件设备对资源的占有统一分配的。为了保证系统的安全性、稳定性和可移植性,对应用程序访问硬件资源加以限制。这就要求设计设备驱动程序以跨越操作系统的边界,对物理硬件进行操作。
  1 DSP芯片中集成的PCI接口特点
  1.1 PCI接口的内部结构
  DM642片内集成一个主/从模式的PCI接口,它相当于专用的PCI接口芯片,这样可以不必深究PCI总线规范,将工作重点放在系统功能的实现上。DSP可以通过这个接口实现与PCI主机的互连。
  从图1可以看出,PCI接口的内部结构包括7个部分:
  (1)PCI总线接口模块(PCI Bus Interface Unit,PBIN):该模块对主/从模式下的总线交易都不会插入等待周期,可以实现最大的总线传输带宽。
  (2)E2PROM控制器模块:控制器与外部的4线串行E2PROM相连。PCI接口复位时,控制器读取E2PROM中的数据,配置PCI接口。DSP可以通过映射寄存器访问E2PROM。
  (3)DSP从模式写模块:包括一个多路复用器和一个PBIN到DSP的FIFO。它完成的功能是:外部PCI设备通过PCI接口写数据到DSP从设备。外部主设备往DSP的Base0空间执行写操作时,PCI地址与DSPP寄存器中的固定偏移值结合,形成DSP目的地址,在传输过程中目的地址自动递增。
  (4)DSP从模式读模块:包括一个多路复用器和一个DSP到PBIN的FIFO。它完成的功能是:外部PCI设备通过PCI接口能够从DSP从设备读取数据。在外部主设备从DSP的Base0空间执行读操作时,PCI地址与DSPP寄存器中的固定偏移值结合,形成DSP源地址,在传输过程中此地址自动递增。
  (5)DSP主模式模块:包括读/写两个子模块,DSP是该模块的主控方。DSP主模式读这个子模块,完成DSP主设备通过PCI接口从外部PCI从设备中读取数据。DSP主模式写这个子模块完成DSP主设备通过PCI接口写数据到外部PCI从设备。
  (6)PCI I/O接口模块:它包括PCI的I/O寄存器,HSR,HDCR,DSPP。只能由PCI主机通过基址1寄存器或基址2寄存器的空间映射进行访问。
  (7)DSP寄存器接口模块:包含DSP的映射寄存器,用于控制主模式接口,产生PCI中断以及电源管理。

友情提示: 此问题已得到解决,问题已经关闭,关闭后问题禁止继续编辑,回答。
该问题目前已经被作者或者管理员关闭, 无法添加新回复
9条回答
biechedan
1楼-- · 2019-08-01 20:44
 精彩回答 2  元偷偷看……
biechedan
2楼-- · 2019-08-01 22:23
  2 驱动程序设计

  设备驱动程序提供连接到计算机硬件的软件接口。它是操作系统的信任部分,由I/O管理器(I/O Manag-er)管理和调动。

  用户应用程序以一种规范的方式访问硬件,而不必考虑如何控制硬件。驱动程序总是使设备看起来像一个文件,可以打开设备的一个句柄,然后应用程序可以在设备句柄最后关闭之前向驱动程序发出读写请求。

  I/O管理器每收到一个来自用户应用程序的请求就创建一个I/O请求包(IRP)的数据结构,并将其作为参数传递给驱动程序。
biechedan
3楼-- · 2019-08-02 01:07
  2.1 设备驱动程序的组成部分

  可以把一个完整的驱动程序看作是一个容器,它包含许多例程。当操作系统遇到一个I/O请求包(I/ORequest Packet,IRP)时,它就调用这个容器中的例程来执行该IRP的各种操作。驱动程序包含以下几个基本例程:

  (1)DriverEntry例程:它是驱动程序的初始化入口点,必须叫作DriverEntry。它负责驱动程序的初始化,用来初始化驱动程序范围内的数据结构和资源。它主要有以下三个功能:设置Adddevice,Unload和其他例程的入口指针;可以从注册表中获取一些需要的信息以初始化驱动程序;初始化其他的在驱动程序范围内的数据结构和资源。所有的驱动程序都必须包含它。当装载驱动程序时,PnP管理器为每个驱动程序调用一次 DriverEntry例程。

  (2)AddDevice例程:在驱动程序初始化以后,PnP管理器调用驱动程序的Add Device例程来初始化由该驱动程序所控制的设备。在Add Device例程中,驱动程序创建一个设备对象作为目标设备,并将设备对象附着到设备堆栈中。

  (3)PnP例程:PCI设备都是即插即用设备,PCI设备的驱动程序必须具备PnP例程。PnP管理器使用PnP例程来管理驱动程序启动、停止和删除设备。

  (4)分发例程(Dispatch):用于管理驱动程序与应用程序之间的通信,从而实现应用程序控制PCI设备的目的。

  严格地说,驱动程序中只有“初始化”模块Drivet-Entry例程是一定不能少的。在实际工作中,所有驱动程序都有分发例程处理用户I/O请求。
biechedan
4楼-- · 2019-08-02 06:42
 2.2 IRP处理
  I/O请求包(IRP)是驱动程序操作的中心,是一个预先定义的数据结构,带有一组对它进行操作的I/O管理器例程。一个IRP有固定的首部和可变数目的 IRP栈单元。IRP的固定部分含有IRP的固定属性,每个栈单元含有大多数有关的IRP参数。当IRP由多个驱动程序处理时,使用多个IRP栈单元。每个驱动程序从当前IRP栈单元得到它的IRP参数。如果把IRP沿当前设备的驱动程序栈向下传递,必须在当前驱动程序中使用正确的参数设置下一个栈单元,然后在此驱动程序中利用函数IoCalldriver()调用更低层的驱动程序。驱动程序不必处理所有的IRP,但至少需要处理“创建”和“关闭”这两个 IRP。I/O管理器接收I/O请求,然后在把它传递到合适的驱动程序栈中的最高驱动程序之前,分配并初始化IRP。驱动程序处理IRP的过程如图2所示。

biechedan
5楼-- · 2019-08-02 12:35
  IRP首先到达最高层的驱动程序1,驱动程序1使用函数IoGetCurrentIrpStackLocation()获得指向当前栈单元的指针。
  然后驱动程序1使用IoCallDriver()函数调用下一个驱动程序。I/O管理器现在改变“当前IRF’栈单元”指针,所以驱动程序2看到向下的第二个IRP栈单元(驱动程序1为它设置的栈单元)。这个过程继续,直到最底层的的驱动程序4收到这个IRP。
  驱动程序4现在处理这个IRP。当它完成IRP的处理时,驱动程序4调用IoCompleteRequest()函数。指示它已经完成IRP的处理。IRP再沿设备栈向上传递,直到它最终弹出栈顶,回到用户。
  2.3 IRP的完成
  当一个驱动程序完成对IRP的处理时,它必须告诉I/O管理器,这称为IRP完成。如下面代码所示,必须设置IRP IoStatus域结构中的几个域。IoStatus,Status设置为一个NTSTATUS状态码,IoStatus.In-formation通常存储传输的字节数。如:
  Irp一>loStatus.Status=S T ATUS_SUCCESS
  Irp一>IoStatus.Information=info;
  IoCompleteRequest(Irp,IO_NO_INCREMENT);
  调用IoCompleteRequest()表明低层驱动程序已经完成了IRP的请求,并将这个IRP返回给I/O管理器。IO_No_INCREMENT是个系统定义的常量,指定启动该IRP的优先级,需要驱动程序快速处理。
biechedan
6楼-- · 2019-08-02 18:30
 精彩回答 2  元偷偷看……

一周热门 更多>