一. 概念
DMA:直接内存访问。它允许外围设备和主内存之间直接传输它的I/O数据,而不需要系统处理器的参与。如此可大大提高与设备通信的吞吐量和提高系统的工作效率
,免除大量的计算开销。
二. DMA两种传输数据方式(以输入为例)
1. 软件对数据的请求(如通过read调用),步骤为:
1.1 当进程调用read,驱动程序函数分配一个DMA缓冲区,并让硬件将数据传输到这个缓冲区,进程进入睡眠状态。
1.2 硬件将数据写入DMA缓冲区,写入完毕会产生一个中断。
1.3 中断处理程序获得输入的数据,应答中断,并且唤醒进程去读取数据。
2. 硬件异步地将数据传递给系统(异步使用DMA)。比如对于一个数据采集设备,即使没有进程读取数据,也不断地写入数据,驱动程序应该维护一个缓冲区,其后的read调用时将返回所有积累的数据给用户空间,步骤为:
2.1 硬件产生中断,宣告新数据的到来
2.2 中断处理程序分配一个缓冲区,并告诉硬件向哪里传输数据
2.3 外围设备将数据写入缓冲区,完成后产生另外一个中断。
2.4 处理程序分发新数据,唤醒任何相关进程,然后执行清理工作。
对于第二种方式,在网卡中可以看到,网卡期望在内存中建一个循环缓冲区(通常叫做DMA环形缓冲区),并与处理器共享;每个输入的数据包都放入缓冲器环中的下一个可用缓冲器中,然后引发中断。接着驱动程序将数据包发送给内核其他部分处理(接收完数据后,向上层协议层上报),并在环形缓冲区中放置一个新的DMA缓冲区。
由此可见高效的DMA处理依赖于中断。虽然可以使用轮询的驱动程序实现DMA,但这没有意义,因为一个轮询驱动程序会将DMA相对简单的处理器驱动I/O获得的性能优势抵消掉。DMA需要设备驱动程序分配一个或者多个适合执行DMA的特殊缓冲区(比如内核一开始就会为有些模块划分了内存空间)。
三、分配DMA缓冲区
这里主要讨论低层分配DMA缓冲区的方法。分配DMA缓冲区时,当大于一页时,它们必须占据连续物理页,这是因为设备使用ISA或者PCI系统总线传输数据,而这两种方式使用的都是物理地址。这种限制对SBus是无效的,因为它在外围总线上使用了虚拟地址。