linux设备驱动学习9

linux设备驱动学习9

ID:34417577

大小:96.16 KB

页数:16页

时间:2019-03-05

linux设备驱动学习9_第1页
linux设备驱动学习9_第2页
linux设备驱动学习9_第3页
linux设备驱动学习9_第4页
linux设备驱动学习9_第5页
资源描述:

《linux设备驱动学习9》由会员上传分享,免费在线阅读,更多相关内容在工程资料-天天文库

1、Linux设备驱动程序学习(9)-与硬件通信在学习有关I/O总线的内容时,最好先看看相关的知识:从PC总线到ARM的内部总线 I/O端口和I/O内存每种外设都是通过读写寄存器来进行控制。在硬件层,内存区和I/O区域没有概念上的区别:它们都是通过向在地址总线和控制总线发出电平信号来进行访问,再通过数据总线读写数据。因为外设要与IO总线匹配,而大部分流行的I/O总线是基于个人计算机模型(主要是x86家族:它为读和写I/O端口提供了独立的线路和特殊的CPU指令),所以即便那些没有单独I/O端口地址空间的处理器,在访问外设时也

2、要模拟成读写IO端口。这一功能通常由外围芯片组(PC中的南北桥)或CPU中的附加电路实现(嵌入式中的方法)。Linux在所有的计算机平台上实现了I/O端口。但不是所有的设备都将寄存器映射到I/O端口。虽然ISA设备普遍使用I/O端口,但大部分PCI设备则把寄存器映射到某个内存地址区,这种I/O内存方法通常是首选的。因为它无需使用特殊的处理器指令,CPU核访问内存更有效率,且编译器在访问内存时在寄存器分配和寻址模式的选择上有更多自由。I/O寄存器和常规内存在进入这部分学习的时候,首先要理解一个概念:sideeffect,

3、书中译为边际效应,第二版译为副作用。我觉得不管它是怎么被翻译的,都不可能精准表达原作者的意思,所以我个人认为记住sideeffect就好。下面来讲讲sideeffect的含义。我先贴出两个网上已有的两种说法(在这里谢谢两位高人的分享):第一种说法:3.sideeffect(译为边际效应或副作用):是指读取某个地址时可能导致该地址内容发生变化,比如,有些设备的中断状态寄存器只要一读取,便自动清零。I/O寄存器的操作具有sideeffect,因此,不能对其操作不能使用cpu缓存。原文网址:http://qinbh.blog.

4、sohu.com/62733495.html第二种说法:说一下我的理解:I/O端口与实际外部设备相关联,通过访问I/O端口控制外部设备,“边际效应”是指控制设备(读取或写入)生效,访问I/O口的主要目的就是边际效应,不像访问普通的内存,只是在一个位置存储或读取一个数值,没有别的含义了。我是基于ARM平台理解的,在《linux设备驱动程序》第二版中的说法是“副作用”,不是“边际效应”。原文网址:http://linux.chinaunix.net/bbs/viewthread.php?tid=890636&page=1#p

5、id6312646 结合以上两种说法和自己看《Linux设备驱动程序(第3版)》的理解,我个人认为可以这样解释: sideeffect是指:访问I/O寄存器时,不仅仅会像访问普通内存一样影响存储单元的值,更重要的是它可能改变CPU的I/O端口电平、输出时序或CPU对I/O端口电平的反应等等,从而实现CPU的控制功能。CPU在电路中的意义就是实现其sideeffect。 I/O寄存器和RAM的主要不同就是I/O寄存器操作有sideeffect,而内存操作没有。因为存储单元的访问速度对CPU性能至关重要,编译器会对源代码进行

6、优化,主要是:使用高速缓存保存数值和重新编排读/写指令顺序。但对I/O寄存器操作来说,这些优化可能造成致命错误。因此,驱动程序必须确保在操作I/O寄存器时,不使用高速缓存,且不能重新编排读/写指令顺序。解决方法:硬件缓存问题:只要把底层硬件配置(自动地或者通过Linux初始化代码)成当访问I/O区域时(不管内存还是端口)禁止硬件缓存即可。硬件指令重新排序问题:在硬件(或其他处理器)必须以一个特定顺序执行的操作之间设置内存屏障(memorybarrier)。Linux提供以下宏来解决所有可能的排序问题:#include

7、inux/kernel.h>voidbarrier(void)/*告知编译器插入一个内存屏障但是对硬件没有影响。编译后的代码会将当前CPU寄存器中所有修改过的数值保存到内存中,并当需要时重新读取它们。可阻止在屏障前后的编译器优化,但硬件能完成自己的重新排序。其实中并没有这个函数,因为它是在kernel.h包含的头文件compiler.h中定义的*/#include#definebarrier()__memory_barrier()#include

8、/system.h>voidrmb(void);/*保证任何出现于屏障前的读在执行任何后续的读之前完成*/voidwmb(void);/*保证任何出现于屏障前的写在执行任何后续的写之前完成*/voidmb(void);/*保证任何出现于屏障前的读写操作在执行任何后续的读写操作之前完成*/voidread_barrie

当前文档最多预览五页,下载文档查看全文

此文档下载收益归作者所有

当前文档最多预览五页,下载文档查看全文
温馨提示:
1. 部分包含数学公式或PPT动画的文件,查看预览时可能会显示错乱或异常,文件下载后无此问题,请放心下载。
2. 本文档由用户上传,版权归属用户,天天文库负责整理代发布。如果您对本文档版权有争议请及时联系客服。
3. 下载前请仔细阅读文档内容,确认文档内容符合您的需求后进行下载,若出现内容与标题不符可向本站投诉处理。
4. 下载文档时可能由于网络波动等原因无法下载或下载错误,付费完成后未能成功下载的用户请联系客服处理。