设备驱动调试和移植的一般方法-程序员宅基地

技术标签: 操作系统  嵌入式  

做linux底层软件工作也有两年了,算上研究生时期对底层软件的研究,加起来也快四年了。慢慢地发现有必要总结一些一般性的方法了。因为一般性的方法有宏观上的指导意义,以后调试和移植驱动时,经常性地回味这些一般性的方法可以防止自己犯同样的错误,进而少走弯路,以最高的效率完成工作。

当谈到底层软件,我们一般都会想到bootloader、BSP、device driver、linux kernel等等。这篇文章将会着重介绍linux device driver调试的一般性方法。另外,关于设备驱动移植的方法会在文章最后以补充说明的形式给出。

注意,这里说的一般方法不是固定套路,也就是说你不用把它当做教科书来读。你只需要根据具体情况灵活的选用其中某些步骤即可。

    一般情况下,进行设备驱动的调试之前,你必须具备以下条件:

l   设备驱动的代码和应用层测试软件(或者代码)

因为你做驱动的最终目的还是要给上层应用去操作的。所以,一个针对驱动的测试应用程序是必须的。这里的应用层测试软件一般都是供应商提供的,或者系统现有的可以用的应用,如果没有就必须自己写了。

l   你要调试的设备对应的datasheet、规格书和相关资料

尽可能地去搞到详尽的芯片相关资料和相关说明,因为这些都是你随后调试驱动的指引。

l   开发板的原理图和贴片图

一般情况下,你可能不需要贴片图,但是如果你们的硬件工程师对你的支持力度不够,那你就需要一个板子的贴片图。原因可能有以下几点:

(1)    如果是飞线过来的模块,而且你又不确定硬件工程师的操作是完全正确的,是不是每条线都接的是对的,这时候你就必须搞到贴片图和原理图,然后和板子上的飞线一条一条线的核对一遍。如果准确无误,那你就开工。

(2)    当你们的硬件工程师没有给你提供硬件板子的引脚具体在板子上的哪个位置的说明时,你就必须搞到贴片图。

(3)    此外,经常贴片图有助于提高一个驱动工程师的硬件素养。

l   万用表、示波器

万用表和示波器是必备的,其次,根据需要,你可能会用到频谱分析仪和逻辑分析仪。

OK!当你拿到这些资料后,你就可以大刀阔斧地进行设备驱动调试了。一般的,设备驱动的调试都会经历如下几个步骤:

一、 把设备驱动的代码添加进kernel

这一点要注意了,不同的方案提供商给出的添加方法是不同的。例如,对于Mediatek方案提供商,添加设备驱动的方法就和一般化的不一样,

因为Mediatek的体系架构不同于像telechips、Qualcomm等这样的kernel传统架构。所以,对于像Mediatek这样的特殊供应商,你必须遵循其提供的一套添加驱动的流程(mtk支持比较好,一般会提供sop)。传统架构的话,就很简单了,一般都会涉及到一下几步:

l   新建设备驱动目录

l   添加头文件

l   创建当前目录的Makefile

l   创建上层目录的Makefile

l   在上层目录添加Kconfig选项

l   在板文件添加对应项。  

l   如果不想每次都用menuconfig配置,就在kernel/arch/arm/configs/XXX_defconfig里面添加编译项

具体的实例我之类就不写了,google一下一大把,这篇文章也不会涉及到过多的细节问题。

二、 使得添加进去的设备驱动可以编译通过

设备驱动添加进来之后,一般的,都需要你添加、删除、修改一些内容,才能保证其编译通过。如果是调试,这里可能改动不大,因为驱动文

件一般是由芯片原厂提供的,他们已经修改的基本可以编译通过了。但如果是移植,在这里花费的精力一般是比较大的,这会在后文中的附加说明里面提及到。

三、 检查设备驱动是否正常初始化

进行到这一步,你需要检查设备驱动是否可以顺利初始化,这个很简单,你只需要在设备驱动的probe函数或者init函数里面添加一条打印

信息即可在串口观察是否有打印,如果没有打印,说明驱动根本就没有得到初始化。此时,你应该:

l   首先,你赶紧跑过去问和你合作的硬件工程师:“hi,我这块板子,那个XXX芯片是否贴上了?”,或者问他们:“这些飞线是不是都连接的OK了?”。 问这些问题都是很有必要的,因为很有可能硬件上没有准备好,或者硬件连线本来就有问题。所以,为了避免浪费时间,你还是主动去问下他们。

l   其次,如果芯片确实贴上了而且硬件连线都接对了(后来你不放心,自己也核对了一遍),这时候问题就在你喽。马上检查设备驱动的添加是否有什么问题,是否是板文件没有修改好,是否驱动本身就没有注册好等等。

 

 

一、 检查应用程序是否可以正常使用

如果驱动得到了正常的初始化,这时候你开始检查该设备驱动对应的应用程序是否可以正常使用(或者自己写设备驱动的测试应用程序)。如果可以正常使用,那你太幸运了,这说明芯片原厂为你考虑的很周到,你可以节省N多时间干别的事儿了。

但遗憾的是,往往事情不会这么顺利,当你检查应用程序的时候,你发现应用程序没有做出应有的反应——例如当你辛苦把camera的驱动添加到kernel而且顺利初始化之后,你兴奋地进入系统打开camera发现:黑屏!你郁闷了,但是郁闷不能解决问题,这时候真正的所谓设备驱动的“调”和“试”才来了。赶紧去找一个万用表和一个示波器吧。

二、 设备驱动的“调”和“试”

所谓“调”,就是你依据设备芯片datasheet、主控datasheet、相关的规格书等资料,以及观察与测量的结果,不断地对设备驱动 code进

行添加、删除和修改。

所谓“试”,就是你在添加、删除和修改的同时要不断地:

l   通过串口对反馈信息进行观察;

l   通过万用表对各电源引脚电压进行测量;

l   通过示波器对芯片的时钟(输入的和输出的)频率(包括测量晶振)、数据信号的波形、某种总线信号的波形,等各个引脚进行测量。

观察设备的上电时序,数据交互时序等等是否正常(参考datasheet)。

l   通过测试应用程序对驱动进行测试,并反复观察设备驱动的运行状态。

测试不光是测试员的事情,你必须保证自己调试的驱动可以运行稳定才可以提交代码,发给测试员进行大量测试。测试自己调试的设备驱动有以下几种:

(1)    利用系统里面现有的应用程序进行测试

这个是最直接的测试,也省去了你自己编写测试程序的时间。

(2)    自己编写linux应用程序来测试驱动

如果你对系统现有的应用程序不放心,或者不满意,就自己编写测试驱动吧。

(3)    利用proc或者sys文件系统的读和写函数

如果你在设备驱动里面添加了proc或者sys文件系统的读和写函数,你就可以在命令行通过cat或者echo命令来对设备驱动的各项参数进行手动测试。

注:以上测试方法是我习惯用的,除了这些,还是有其他一些测试方法我没有列举出来,感兴趣的话可以google一下。

此外,应该养成一个好习惯,那就是在测量各个引脚的同时,你最好创建一份excel表格,把各个引脚的电压、波形等情况记录下来,目的是在N多次修改之间作比较,也更有备忘的作用。

    这里的“调”和“试”的 过程是最耗时的,也是设备驱动调试的最关键的步骤,所以你应该非常有耐心的走好这一步。还应该注意的是,设备驱动的调试不光是你一个人的事情,有时候你闷 头苦干一个星期都不一定能搞定。所以,进行调试的同时,必须有一个硬件工程师配合你,因为驱动不能正常运行也很有可能是硬件的设计纰漏导致的。此外,不要 闭门造车,毕竟你在一个大的开发团队里面,你需要及时和同事进行有效的交流,避免走弯路。最后,如果还是存在问题,你就得邀请芯片原厂的FAE过来喽。

补充说明

    要进行设备驱动的移植,一般的,你必须具备以下几个条件:

l   和当前设备芯片相近的或者相类似的芯片的现有驱动代码。

这句话有点拗口了,举个例子来讲吧。例如,你现在要移植一份驱动到g-sensor(BMA250E)上,你现在就应该尽可能

地去找到和芯片BMA250E相近的芯片所对应的驱动代码。这时候,你高兴的发现,咦?!手头上还有一份BMA250的代码,太幸运了。此时你就可以基于BMA250的代码来做移植。但有时候并没有这么幸运,那怎么办呢?这时候你完全可以去找其他类型的g-sensor啊,假如你手头上并没有BMA250的代码,但是你有一份ADXL345的代码,那当然也是可以的,你完全可以基于ADXL345的代码来做移植。至于为什么要选择与当前设备芯片相近的芯片所对应的驱动代码,原因就四个字“提高效率”。

l   当前芯片的详细datasheet,以及芯片的相关说明

l   开发板的原理图

如果有pads版本的原理图和贴片图,那是最好的了,方便以后测量引脚的各项参数。

l   万用表和示波器

这是调试、移植、开发设备驱动必不可少的设备。

具备了以上几个条件之后,接着走上文所讲的第一个和第二个步骤,尤其在第二个步骤,你可能得费好大力气来针对你当前的开发板进行驱动文件的修改。修改一般都包括I\O配置、中断的设置、总线的配置(例如I2C、SPI、SDIO等)、寄存器的配置、甚至驱动代码的框架都得修改。等你这一步做好了,接着进行上文谈到的第三步,然后一步一步往下走完即可。

用一句话概括一下驱动移植吧。驱动的移植实际上就是:“通过添加、删除和修改,使现有的一份芯片相近的驱动代码可以在目标芯片和目标板上跑起来并且跑的很稳定!

转载于:https://www.cnblogs.com/welhzh/p/4786934.html

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/dielucui7698/article/details/101400960

智能推荐

Java面试题 (线程的生命周期?线程有几种状态)_面试题 线程的就绪状态?-程序员宅基地

文章浏览阅读276次。一、线程通常有五种状态:创建,就绪,运行,阻塞和死亡。(1).新建状态(New):新创建了一个线程对象。(2).就绪状态(Runnable):线程对象创建后,其他线程调用了该对象的start方法。该状态的线程位于可运行线程池中,变得可运行,等待获取CPU的使用权。(3).运行状态(Running):就绪状态的线程获取了CPU,执行程序代码。(4).阻塞状态(Blocked):阻塞状态是线程因为某种原因放弃CPU使用权,暂时停止运行。直到线程进 入就绪状态,才有机会转到运行状态。(5).死_面试题 线程的就绪状态?

hdoj-程序员宅基地

文章浏览阅读880次。#include int team[505][505];int degree[505];int N,M;int main(){ while(scanf("%d%d",&N,&M)!=EOF) { int i,j; //初始化 for(i=1;i<=N;i++) { degree[i]=0; for(j=1;j<=N;j++)

react:react-cropper插件,实现图片裁剪upload上传功能-程序员宅基地

文章浏览阅读1.1w次,点赞2次,收藏14次。因业务需求,需要用户自定义个人头像(图片剪切上传);调研了之后,决定使用react-cropper插件来实现头像上传功能!react-cropper插件还是很全面不错的, 满足了功能的需要。步骤实现:1、安装 npm install --save react-cropper2、文件引入 import "cropperjs/dist/cropper.css..._react-cropper

青龙之腾讯自选股_青龙 腾讯自选股-程序员宅基地

文章浏览阅读8.4k次,点赞3次,收藏33次。1、青龙面板中选择添加任务2、拉取脚本扣扣裙获取脚本:9846983523、手动运行一次此任务,拉取脚本到青龙4、编辑一下拉下来的定时任务的定时,改成35 11,16 * * *5、准备工作5-1、下载【腾讯自选股】APP,可以直接appstore下载。5-2、关注腾讯自选股微信版|微证券 公众号腾讯自选股微信版|微证券6、打开抓包软件准备抓包。我下面直接用圈x举例了。圈x长按风车,弹出菜单中点击HTTP数据抓取按钮,将其打开共需要获取三个c_青龙 腾讯自选股

笔记8:vb.net的二进制数据流BinaryReader、BinaryWriter_vb.netbinaryreader-程序员宅基地

文章浏览阅读7.9k次。前面的FileStream、BufferedStream、MemoryStream类都是派生于Stream类。 二进制数据流不同,它并不派生于Stream,而是直接派生于System.Object类。 因此,天生不足,它只有串接到Stream类或派生类中进行操作,即二进制数据流的构造都要用Stream类。 二进_vb.netbinaryreader

优化C语言代码(程序员必读)_编译器再什么时候会优化代码 c-程序员宅基地

文章浏览阅读232次。1、选择合适的算法和数据结构 应该熟悉算法语言,知道各种算法的优缺点,具体资料请参见相应的参考资料,有很多计算机书籍上都有介绍。将比较慢的顺序查找法用较快的二分查找或乱序查找法代替,插入排序或冒泡排序法用快速排序、合并排序或根排序代替,都可以大大提高程序执行的效率。.选择一种合适的数据结构也很重要,比如你在一堆随机存放的数中使用了大量的插入和删除指令,那使用链表要快得多。数组与指针语句具_编译器再什么时候会优化代码 c

随便推点

linux系统ps命令的参数a与-a表示的讨论-程序员宅基地

文章浏览阅读1.6w次,点赞7次,收藏15次。前言 在刘遄老师所著的《linux就该这么学》中ps 命令用于查看系统中的进程状态,格式为“ps [参数]”。书中对于ps命令有一定的介绍,为了加强自己的理解,所以我在自己的虚拟机上进行了实验,发现输出的和书上讲的不一样,可能是我理解的错了?以下就展开对此的讨论。 在讨论之前,我们先了解以下ps命令的用法。一、ps命令 ps 命令用于查看...

JAVA的Calendar类set月份的时候,月份会比输入的加1_(calendar.month)总是大1-程序员宅基地

文章浏览阅读4.7k次。今天产品跟我反馈后台管理系统上传数据的时候,命名输入的是 2018/11/16,但是上传之后显示的时间是 2018/12/16我用的是Calendar.set(Calendar.MONTH, )我就觉得很奇怪啊,我的代码没问题啊,然后去看数据库的时间戳,发现真的是 2018/12/16那么我们可以将问题锁定到下图的红框框中那么好,现在很显然我们可以改为month-1来修改我们..._(calendar.month)总是大1

crontab 定时执行php脚本文件_crontable 执行php-程序员宅基地

文章浏览阅读919次。什么是Cron和CrontabUnix和Linux系统的各个发行版本基本都支持Cron,Cron /Crontab允许我们在设定的时间自动执行或定时执行某个任务,如应用程序或脚本。更详细的概念和使用方法介绍请点击Cron和Crontab是什么?现在我们要讨论的是,许多的php程序员都可以很快完成Web应用的开发,PHP代码的调试也比PERL或C语言容易很多,不过经常会碰到有php_crontable 执行php

vue3 组合式风格setup笔记(定义变量、函数 data和method的setup写法)_vue3 methods和data写法-程序员宅基地

文章浏览阅读2.9k次,点赞2次,收藏2次。vue3 组合式风格 变量定义(对应data)、函数定义(对应method)_vue3 methods和data写法

子串判断_给定俩个字符串a和b,其中a 的长度大于b-程序员宅基地

文章浏览阅读1.9k次,点赞4次,收藏7次。从键盘输入两个长度小于80的字符串A和B,且A的长度大于B的长度,编程判断B是不是A的子串,如果是,则输出”Yes”,否则输出”No”。这里所谓的该串的子串是指字符串中任意多个连续的字符组成的子序列。函数原型:int IsSubString(char a[], char b[]);函数功能:判断b是否是a的子串,是则返回1,否则返回0程序运行结果示例1:Input the firs..._给定俩个字符串a和b,其中a 的长度大于b

企业微信支付提示请在微信客户端打开链接_微信h5支付?-程序员宅基地

文章浏览阅读4.1k次。如果您是使用浏览器的移动网页进行支付,那就是调用微信H5支付,如果是使用APP客户端进行支付,那就要调用微信APP支付。H5支付是指商户在微信客户端外的移动端网页展示商品或服务,用户在前述页面确认使用微信支付时,商户发起本服务呼起微信客户端进行支付。主要用于触屏版的手机浏览器请求微信支付的场景。可以方便的从外部浏览器唤起微信支付。顺便提醒一下,H5支付不建议在APP端使用,如需要在APP中使用微信...

推荐文章

热门文章

相关标签