《Linux内核修炼之道》之高效学习linux驱动开发

<h1 class="title_txt">
<img src="http://blog.csdn.net/images/authorship.gif" border="0" alt="原创" width="15" height="15">

《Linux内核修炼之道》 之 高效学习Linux驱动开发
<cite class="fav_csdnstylebykimi"><a class="fav_csdnstylebykimi" title="收藏到我的网摘中,并分享给我的朋友">收藏</a>
</cite>
<span style="height: 0px; width: 0px;"><img style="" src="http://blog.csdn.net/images/tuijian.gif" alt="" align="absmiddle"></span>
</h1>
<p class="MsoNormal" style="margin-top: 6pt; text-indent: 21pt;"><span>这本《</span>
<span lang="EN-US">Linux</span>
<span>内核修炼之道》已经开卖(网上的链接为:</span>
<span lang="EN-US"><a href="http://www.amazon.cn/gp/product/B003U3N09C?ver=gp"><span>卓越</span>
</a>
</span>
<span>、</span>
<span lang="EN-US"><a href="http://product.dangdang.com/product.aspx?product_id=20872348"><span>当当</span>
</a>
</span>
<span>、</span>
<span lang="EN-US"><a href="http://www.china-pub.com/51884&amp;ref=xiangguan">china-pub</a>
)</span>
<span>,虽然是严肃文学,但为了保证流畅性,大部分文字我还都是斟词灼句,反复的念几遍才写上去的,尽量考虑到写上去的</span>

段话能够让读者产生什么疑惑,然后也都会紧接着尽量的去进行解释清楚,中间的很多概念也有反复纠结过怎么解释能够更容易的理解,力求即使对于初学者也可以
有很少阻碍的一气读完。同时我也把书中一部分自己的感悟抽出来整理了精华版,share出来。当然水平有限,错漏之处有发现而修订时遗漏的,也有尚没有发
现的。这本书如果对您有用,乃我之幸事,如果无用,就在此先诚惶诚恐的向大家拜个不是了。</p>
<p class="MsoNormal">下面仍然是之前5月份一次presentation的部分内容及讲义,不过当时的题目叫“驱动开发的方法论”,或许对大家有用吧。至于这几次讲座的视频貌似网上都能找到。</p>
<p class="MsoNormal">************************************************************************</p>
<div style="text-indent: 0px;">
<span style="font-family: Verdana,Arial,Helvetica,sans-serif;"><span style="">  </span>
<span style="">前一篇我们谈到了如何高效学习Linux内核,现在我们开始另外一个话题,就是如何高效学习</span>
<span style="" lang="EN-US">linux</span>
<span style="">驱动开发。至于为什么会选择这样一个</span>
<span style="" lang="EN-US">topic</span>
<span style="">,主要是基于这样两个原因:</span>
</span>
</div>
<p class="MsoNormal" style="margin-top: 6pt; text-indent: 21pt;"><span style="">第一个原因是:目前几乎所有的驱动开发方面的参考书,内容结构都是先介绍介绍什么是</span>
<span style="" lang="EN-US">linux</span>
<span style="">驱动,它分为哪些种类,然后是各种类型设备的驱动程序的内容细节。大都是只注重各种驱动
本身的细节,而没有站在一个全局整体的角度讲解一下驱动开发的方法。这样导致的后果就是,大多数的驱动开发者虽然可以正确的编写驱动程序,但往往都是只知
其一不知其二,知其然而不知其所以然。</span>
</p>
<p class="MsoNormal" style="margin-top: 6pt; text-indent: 21pt;"><span style="">第二个原因是:目前很多驱动开发者,即使是已经有多年经验的开发者,在开发驱动的时候也就是填充填充</span>
<span style="" lang="EN-US">driver</span>
<span style="">的结构体,对于比较成熟的平台,就是网上找个类似的驱动修改一下,即使写十个百个千个驱动,也就是对某些硬件比较熟,遇到全新的芯片全新的平台就束手无策。应该说这样对驱动的理解是很有限的。这也是目前</span>
<span style="" lang="EN-US">linux</span>
<span style="">驱动开发领域的现状。</span>
</p>
<p><img src="http://hi.csdn.net/attachment/201007/22/0_1279802475XKNq.gif" alt="" width="685" height="512"></p>
<p class="MsoNormal" style="margin-top: 6pt; text-indent: 21pt;"><span style="">我们首先认识一下</span>
<span style="" lang="EN-US">linux</span>
<span style="">驱动的基本面,我们认识一个新事物的的第一件事就是了解它的一些基本信息,就像我们人与人之间互相认识首先也是通过个人的基本信息一样。</span>
</p>
<p class="MsoNormal" style="margin-top: 6pt; text-indent: 21pt;"><span style="" lang="EN-US">linux</span>
<span style="">驱动在本质上就是一种软件程序,上层软件可以在不用了解硬件特性的情况下,通过驱动提供的接口,和计算机硬件进行通信。</span>
</p>
<p class="MsoNormal" style="margin-top: 6pt; text-indent: 21pt;"><span style="">系统调用是内核和应用程序之间的接口,而驱动程序是内核和硬件之间的接口,也就是内核和硬件之间的桥梁。它为应用程序屏蔽了硬件的细节,这样在应用程序看来,硬件设备只是一个设备文件,应用程序可以象操作普通文件一样对硬件设备进行操作。</span>
</p>
<p class="MsoNormal" style="margin-top: 6pt; text-indent: 21pt;"><span style="" lang="EN-US">linux</span>
<span style="">驱动程序是内核的一部分,管理着系统中的设备控制器和相应的设备。它主要完成这么几个功能:对设备初始化和释放;传送数据到硬件和从硬件读取数据;检测和处理设备出现的错误。</span>
</p>
<p class="MsoNormal" style="margin-top: 6pt; text-indent: 21pt;"><span style="">一般来说,一个驱动可以管理一种类型的设备。例如不同的</span>
<span style="" lang="EN-US">U</span>
<span style="">盘都属于</span>
<span style="" lang="EN-US">mass storage</span>
<span style="">设备,我们不需要为每一个</span>
<span style="" lang="EN-US">U</span>
<span style="">盘编写驱动,而只需要一个驱动就可以管理所有这些</span>
<span style="" lang="EN-US">mass storage</span>
<span style="">设备。</span>
</p>
<p class="MsoNormal" style="margin-top: 6pt; text-indent: 21pt;"><span style="">为方便我们加入各种驱动来支持不同的硬件,内核抽象出了很多层次结构,这些层次结构是</span>
<span style="" lang="EN-US">linux</span>
<span style="">设备驱动的上层。它们抽象出各种的驱动接口,驱动只需要填写相应的回调函数,就能很容易把新的驱动添加到内核。</span>
</p>
<p class="MsoNormal" style="margin-top: 6pt; text-indent: 21pt;"><span style="">一般来说,</span>
<span style="" lang="EN-US">linux</span>
<span style="">驱动可以分为三类,就是块设备驱动,字符设备驱动和网络设备驱动。块设备的读写都有缓存来支持,并且块设备必须能够随机存取。块设备驱动主要用于磁盘驱动器。</span>
</p>
<p class="MsoNormal" style="margin-top: 6pt; text-indent: 21pt;"><span style="">而字符设备的</span>
<span style="" lang="EN-US">I/O</span>
<span style="">操作没有通过缓存。字符设备操作以字节为基础,但不是说一次只能执行一个字节操作。例如对于字符设备我们可以通过</span>
<span style="" lang="EN-US">mmap</span>
<span style="">一次进行大量数据交换。字符设备实现比较简单和灵活。</span>
</p>
<p>

<span style="">网络设备在</span>
<span style="" lang="EN-US">Linux</span>
<span style="">里做专门的处理。</span>
<span style="" lang="EN-US">Linux</span>
<span style="">的网络系统主要是基于</span>
<span style="" lang="EN-US">BSD</span>
<span style="">的</span>
<span style="" lang="EN-US">socket</span>
<span style="">机制。网络设备驱动为网络操作提供接口,管理网络数据的接送和收发。为了屏蔽网络环境中物理网络设备的多样性,</span>
<span style="" lang="EN-US">Linux</span>
<span style="">对所有的物理设备进行抽象并定义了一个统一的概念,称之为接口(</span>
<span style="" lang="EN-US">interface</span>
<span style="">)。所有对网络硬件的访问都是通过接口进行的,接口对上层协议提供一致化的操作集合来处理基本数据的发送和接收,对下层屏蔽硬件差异。它与字符设备及块设备不同之处其一就是网络接口不存在于</span>
<span style="" lang="EN-US">Linux</span>
<span style="">的设备文件系统</span>
<span style="" lang="EN-US">/dev/</span>
<span style="">中。</span>
</p>
<p><img src="http://hi.csdn.net/attachment/201007/22/0_1279802494dCTD.gif" alt="" width="685" height="513"></p>
<p class="MsoNormal" style="margin-top: 6pt; text-indent: 21pt;"><span style="">和前一篇的介绍一样,看完外表,我们再看内涵,就是</span>
<span style="" lang="EN-US">Linux</span>
<span style="">驱动的工作流程。大概有四个部分:使用</span>
<span style="" lang="EN-US">insmod</span>
<span style="">加载,模块的初始化,进行设备操作,使用</span>
<span style="" lang="EN-US">rmmod</span>
<span style="">卸载。</span>
</p>
<p class="MsoNormal" style="margin-top: 6pt; text-indent: 21pt;"><span style="" lang="EN-US">Linux</span>
<span style="">驱动有两种存在形式,一种是直接编译进内核,就是我们在配置内核的时候,在相应选项上选</span>
<span style="" lang="EN-US">Y</span>
<span style="">,另外一种就是编译成模块,按需加载和卸载。通常我们使用</span>
<span style="" lang="EN-US">insmod</span>
<span style="">命令完成模块的加载,在加载时还可以指定模块参数。另外一个常用的加载工具是</span>
<span style="" lang="EN-US">modprobe</span>
<span style="">,它与</span>
<span style="" lang="EN-US">insmod</span>
<span style="">的不同在于它会检查模块之间的依赖关系,将该模块依赖的模块也加载到内核。</span>
</p>
<p>
<span style="">每个驱动都有自己的初始化函数,完成一些新功能的注册,这个初始化函数只是在初始化的时候被使用。在</span>
<span style="" lang="EN-US">linux</span>
<span style="">系统里,设备以文件的形式存在,应用程序可以通过</span>
<span style="" lang="EN-US">open</span>
<span style="">、</span>
<span style="" lang="EN-US">read</span>
<span style="">等函数操作设备,通过设备文件实现对设备的访问。设备不再使用时,我们使用</span>
<span style="" lang="EN-US">rmmod</span>
<span style="">命令来卸载它,卸载的过程会调用到驱动的推出函数,每个驱动都必须有一个退出函数,没有的话,内核就不会允许去卸载它。</span>
</p>
<p><img src="http://hi.csdn.net/attachment/201007/22/0_1279802509k6ef.gif" alt="" width="686" height="513"></p>
<p class="MsoNormal" style="margin-top: 6pt; text-indent: 21pt;"><span style="">在对</span>
<span style="" lang="EN-US">linux</span>
<span style="">驱动的外表和内涵都有了一个初步的认识之后,我们来看看作为一个驱动开发者,我们需要注意哪些问题。</span>
</p>
<p class="MsoNormal" style="margin-top: 6pt; text-indent: 21pt;"><span style="">首先,对模块机制的了解是开发</span>
<span style="" lang="EN-US">linux</span>
<span style="">驱动的基础,因为我们编写驱动的过程也就是在编写一个内核模块的过程。早期版本的内核是整体式的,也就是说所有的部分都静态地连接成一个很大的执行文件。但是现在的内核采用的是新的机制,即模块机制:许多功能包含在模块内,当你需要时可以使用</span>
<span style="" lang="EN-US">insmod</span>
<span style="">去拥抱它,将它动态地载入到内核里,当你不需要时,则可以使用</span>
<span style="" lang="EN-US">rmmod</span>
<span style="">将它一脚踢开。这就使得</span>
<span style="" lang="EN-US">kernel</span>
<span style="">的内核很小,而且在运行的时候可以不用</span>
<span style="" lang="EN-US">reboot</span>
<span style="">就能够载入和替代模块。</span>
</p>
<p class="MsoNormal" style="margin-top: 6pt; text-indent: 21pt;"><span style="">其次,我们要注重对设备模型的理解。其实从</span>
<span style="" lang="EN-US">2.6</span>
<span style="">内核开始,随着设备模型的出现,驱动的开发就不再是个困难的问题,毫不夸张得说,理解了
设备模型,再去看那些五花八门的驱动程序,你会发现自己站在了另一个高度,从而有了一种俯视的感觉,就像凤姐俯视知音和故事会,韩峰同志俯视女下属。不过
貌似大部分驱动开发者都没意识到这个问题。</span>
</p>
<p>
<span style="">最后,是要养成使用协议的</span>
<span style="" lang="EN-US">spec</span>
<span style="">、设备的</span>
<span style="" lang="EN-US">datasheet</span>
<span style="">、内核参考代码去解决问题的习惯,而不是一碰到问题就到处寻找所谓的牛人去问怎么解决。</span>
</p>
<p><span style="font-family: mceinline;">中间的那些内容和前面精华版的博文里差不多,就不贴了,…………</span>
</p>
<p><img src="http://hi.csdn.net/attachment/201007/22/0_12798025562sHv.gif" alt="" width="689" height="516"></p>
<p class="MsoNormal" style="margin-top: 6pt; text-indent: 21pt;"><span style="">前面介绍了我个人感觉开发驱动需要注意的三个方面,现在说个实际的例子。前些天一个网友在自己的</span>
<span style="" lang="EN-US">csdn</span>
<span style="">博客上写了篇文章,名字就叫给</span>
<span style="" lang="EN-US">fudan_abc</span>
<span style="">的一封信,信里说了自己的问题,我觉得应该很多人都存在这样类似的问题,这里咱们来看一下。</span>
</p>
<p class="MsoNormal" style="margin-top: 6pt; text-indent: 21pt;"><span style="">先说他个人的情况:有一定的</span>
<span style="" lang="EN-US">linux c</span>
<span style="">基础,熟悉</span>
<span style="" lang="EN-US">linux</span>
<span style="">内核</span>
<span style="" lang="EN-US">/</span>
<span style="">驱动的开发环境搭建和编译。现在想做个</span>
<span style="" lang="EN-US"> i2c </span>
<span style="">芯片的</span>
<span style=""> </span>
<span style="">驱动,该驱动要跑在</span>
<span style="" lang="EN-US">x86</span>
<span style="">平台上。</span>
</p>
<p class="MsoNormal" style="margin-top: 6pt; text-indent: 21pt;"><span style="">手边的资料有:</span>
</p>
<p class="MsoNormal" style="margin-top: 6pt; text-indent: 21pt;"><span style="" lang="EN-US">1. </span>
<span style="">芯片手册</span>
<span style=""> </span>
<span style="">看了几次,基本了解</span>
<span style=""> </span>
<span style="">上边的</span>
<span style=""> </span>
<span style="">资源和</span>
<span style=""> </span>
<span style="">他的</span>
<span style="" lang="EN-US">i2c</span>
<span style="">地址</span>
</p>
<p class="MsoNormal" style="margin-top: 6pt; text-indent: 21pt;"><span style="" lang="EN-US">2. </span>
<span style="">找到</span>
<span style="" lang="EN-US">linux + arm </span>
<span style="">平台上的</span>
<span style=""> </span>
<span style="">对应的</span>
<span style="" lang="EN-US"> .c </span>
<span style="">文件和</span>
<span style="" lang="EN-US">.h</span>
<span style="">文件。但是对应的</span>
<span style="" lang="EN-US">arm</span>
<span style="">平台的</span>
<span style="" lang="EN-US">kernel source</span>
<span style="">没有。</span>
</p>
<p class="MsoNormal" style="margin-top: 6pt; text-indent: 21pt;"><span style="" lang="EN-US">3. .c
</span>
<span style="">文件里的</span>
<span style=""> </span>
<span style="">代码看过</span>
<span style=""> </span>
<span style="">一遍,从函数名</span>
<span style=""> </span>
<span style="">可以知道他的功能。</span>
</p>
<p class="MsoNormal" style="margin-top: 6pt; text-indent: 21pt;"><span style="">然后开始</span>
<span style=""> </span>
<span style="">操作</span>
<span style=""> </span>
<span style="">怎么移植这个</span>
<span style="" lang="EN-US">i2c </span>
<span style="">芯片到</span>
<span style="" lang="EN-US"> x86</span>
<span style="">平台上。之后遇到了一些列的问题,比如</span>
<span style="" lang="EN-US">i2c</span>
<span style="">的地址信息等等,具体我就不叙述了。</span>
</p>
<p class="MsoNormal" style="margin-top: 6pt; text-indent: 21pt;"><span style="">然后他就去请教了一些人</span>
</p>
<p class="MsoNormal" style="margin-top: 6pt; text-indent: 21pt;"><span style="">请教某人</span>
<span style="" lang="EN-US">A</span>
<span style="">:</span>
<span style="" lang="EN-US">“</span>
<span style="">其实</span>
<span style="" lang="EN-US">i2c</span>
<span style="">很简单,你填充那个</span>
<span style="" lang="EN-US">driver</span>
<span style="">结构体就</span>
<span style="" lang="EN-US">ok</span>
<span style="">了。</span>
<span style="" lang="EN-US">”</span>
<span style="">我反问</span>
<span style="" lang="EN-US">“</span>
<span style="">我对里边的</span>
<span style=""> </span>
<span style="">流程和调用不熟悉啊</span>
<span style="" lang="EN-US">“</span>
</p>
<p class="MsoNormal" style="margin-top: 6pt; text-indent: 21pt;"><span style="" lang="EN-US">A</span>
<span style="">回:</span>
<span style="" lang="EN-US">”</span>
<span style="">你把他们想象成一个黑盒就可以了</span>
<span style="" lang="EN-US">”</span>
<span style="">我:</span>
<span style="" lang="EN-US">“</span>
<span style="">无法想象,我怎么想象?我想看看里边的代码到底是怎样的。</span>
<span style="" lang="EN-US">”A:"....."</span>
<span style="">。</span>
<span style=""> </span>
<span style="">对</span>
<span style="" lang="EN-US">A</span>
<span style="">的请教结果,无法解决我的问题:我到底该怎么办?</span>
</p>
<p class="MsoNormal" style="margin-top: 6pt; text-indent: 21pt;"><span style="">再请教</span>
<span style="" lang="EN-US">B:</span>
</p>
<p class="MsoNormal" style="margin-top: 6pt; text-indent: 21pt;"><span style="" lang="EN-US">B:“</span>
<span style="">其实</span>
<span style="" lang="EN-US">i2c</span>
<span style="">移植就是注册那几个函数</span>
<span style="" lang="EN-US">, </span>
<span style="">你想看内核代码实现,内核这么大你怎么可能搞清楚,我做了那么多移植,有时候连芯片手册都搞不清,直接</span>
<span style="" lang="EN-US">prink</span>
<span style="">吧代码给调出来的</span>
<span style="" lang="EN-US">” </span>
<span style="">我:</span>
<span style="" lang="EN-US">“</span>
<span style="">。。。</span>
<span style="" lang="EN-US">”</span>
<span style="">无法理解,无话可说。</span>
</p>
<p class="MsoNormal" style="margin-top: 6pt; text-indent: 21pt;"><span style="">其实这里边的</span>
<span style="" lang="EN-US">A</span>
<span style="">和</span>
<span style="" lang="EN-US">B</span>
<span style="">说的也没错,很多人写驱动大概就这么做的,但是这样子就是写成百上千个驱动也不能说就理解</span>
<span style="" lang="EN-US">linux</span>
<span style="">驱动了,面试时碰到的绝大部分人都属于这种情况,能回答自己做了什么,但一谈到一些相关的基本的问题就往往回答不上来。</span>
</p>
<p class="MsoNormal" style="margin-top: 6pt; text-indent: 21pt;"><span style="">我觉得首要的问题是缺乏好奇心,做技术的好奇心应该是原动力,特别对于搞</span>
<span style="" lang="EN-US">linux</span>
<span style="">内核和驱动的,好奇心有多强,你的水平就可能会增长到多高。</span>
</p>
<p class="MsoNormal" style="margin-top: 6pt; text-indent: 21pt;"><span style="">其次,对于做驱动的来说,对于</span>
<span style="" lang="EN-US">2.6</span>
<span style="">内核,重要的是去理解设备模型,很多人都本末倒置了,很多专门写驱动的书也不注重设备模
型的理解,只去应付一种种类型的协议和设备的驱动,即使写个一万个也仅仅是对比较成熟的芯片熟了些。而且,不理解设备驱动,难道写驱动的时候不觉得很多东
西很朦胧么?这也是我个人觉得很奇怪的一方面。</span>
</p>
<p class="MsoNormal" style="margin-top: 6pt; text-indent: 21pt;"><span style="">像我之前设备模型的文章里说的那样,理解了设备模型,对各种类型的驱动就有种俯视的感觉了。这个时候再你去看特定类型的协议和设备的实现,脉络就很清晰,比如你看</span>
<span style="" lang="EN-US">i2c</span>
<span style="">的实现。这个时候重要的就是</span>
<span style="" lang="EN-US">i2c</span>
<span style="">的协议,具体芯片的</span>
<span style="" lang="EN-US">datasheet</span>
<span style="">,加上看看内核里现有的</span>
<span style="" lang="EN-US">i2c</span>
<span style="">驱动作为参考,就是我所说的驱动开发的三件宝。</span>
</p>
<p>
<span style="">所以说如果希望做</span>
<span style="" lang="EN-US">2.6</span>
<span style="">的驱动话,关键还是要先去理解下设备模型,将这个比较抽象的概念在心里形象化,然后再去看具体的驱动比较好。</span>
</p>

你可能感兴趣的:(linux)