1、北邮嵌入式实验报告嵌入式系统开发实验报告班 级: 姓 名: 班内序号: 学 号: 日 期: 目录一、 实验目的 1二、 实验设备 1三、 基础实验(实验一实验七) 11. 实验五 22. 实验六 23. 实验七 2四、 驱动程序 31. 设备驱动程序的概念 32. 驱动程序结构 33. 设备注册和初始化 44. 设备驱动程序的开发过程 6五、 基本接口实验 61. 实验十二 简单设备驱动程序 62. 实验十三 CPU GPIO驱动程序设计 73. 实验十四 中断实验 84. 实验十五 数码管显示实验 95. 实验十六 LED点阵驱动程序设计 116. 实验十七 AD驱动实验 127. 实验十八
2、 DA驱动实验 13六、 实验中遇到的问题及解决方法 15七、 实验总结及心得 15一、 实验目的通过实验熟悉Linux环境,并掌握一些基本接口驱动的写法和用C语言编写简单的实验程序。学习LINUX开发环境的搭建,通讯配置等。并熟练掌握LINUX驱动程序的编写及开发流程。对嵌入式系统有进一步的了解。二、 实验设备1.一套PXA270EP嵌入式实验箱2.安装Redhat9的宿主PC机,并且配置好ARMLinux的开发环境三、 基础实验(实验一实验七)实验一到六为基础实验, 主要是为了在熟悉实验操作平台的同时为后续实验搭建好软、硬件环境,配置好相关的协议、服务。其中实验一是各个硬件的互联,搭建好了
3、实验的硬件环境。实验二是在宿主PC端安装虚拟机,提供了实验需要的Linux操作系统。实验三是宿主PC端开发环境的安装与配置。实验四是配置宿主PC机端的超级终端,使PC机与PXA270目标板之间可以通过串口通讯。在每次重启宿主PC机时,都需要重新将超级终端挂载到虚拟机上,挂载之前须通过ifconfig命令查看该机的IP地址,若其已经复位,须用命令:ifconfig eth0 192.168.0.100 up重置宿主PC机的IP地址。挂载虚拟机的代码为:rootifconfig eth0 192.168.0.50 upmount o nolock 192.168.0.100:/ /mnt实验五是配
4、置宿主PC机的TFTP服务。TFTP是简单文件传输协议。每次重启宿主PC机时,都要重启该服务,重启命令为:service xinetd restart。实验六是配置宿主PC机端NFS服务。NFS是指网络文件系统,它实现了文件在不同的系统间使用。当使用者想用远端档案时,只需调用“mount”就可以远端系统挂接在自己的档案系统之下。每次重启宿主PC机时,也都要重启该服务,重启命令为:service nfs restartservice nfs restart1. 实验五实验五为宿主PC机配置了TFTP服务。TFTP(Trivial File Transfer Protocol)是简单文件传输协议,
5、由于特定开发环境的制约,这一服务是必须的。在配置完成后,每次重启宿主PC机时,都须先输入命令:service xinetd restart,以启动TFTP服务。2. 实验六实验六为宿主PC机配置了NFS服务。NFS(Network File System)指网络文件系统,它实现了文件在不同的系统间使用。当我们想用远端档案时,只需调用“mount”就可以远端系统挂接在自己的档案系统之下。每次重启宿主PC机时,都须先输入命令:service nfs restart,以启动nfs服务。3. 实验七实验七通过用c语言编写的简单程序HelloWorld,测试前面几个实验是否成功配置好环境,从超级终端可以
6、看到HelloWorld程序的运行结果。四、 驱动程序4. 设备驱动程序的概念设备驱动程序实际是处理和操作硬件控制器的软件,从本质上讲,是内核中具有最高特权级的、驻留内存的、可共享的底层硬件处理例程。驱动程序是内核的一部分,是操作系统内核与硬件设备的直接接口,驱动程序屏蔽了硬件的细节,完成以下功能: 对设备初始化和释放; 对设备进行管理,包括实时参数设置,以及提供对设备的操作接口; 读取应用程序传送给设备文件的数据或者回送应用程序请求的数据; 检测和处理设备出现的错误。Linux操作系统将所有的设备全部看成文件,并通过文件的操作界面进行操作。对用户程序而言,设备驱动程序隐藏了设备的具体细节,对
7、各种不同设备提供了一致的接口,一般来说,是把设备映射为一个特殊的设备文件,用户程序可以像对其他文件一样对此设备文件进行操作。这意味着: 由于每一个设备至少由文件系统的一个文件代表,因而都有一个“文件名”。 应用程序通常可以通过系统调用open()打开设备文件,建立起与目标设备的连接。 打开了代表着目标设备的文件,即建立起与设备的连接后,可以通过read()、write()、ioctl()等常规的文件操作对目标设备进行操作。设备文件的属性由三部分信息组成:第一部分是文件的类型,第二部分是一个主设备号,第三部分是一个次设备号。其中类型和主设备号结合在一起惟一地确定了设备文件驱动程序及其界面,而次设
8、备号则说明目标设备是同类设备中的第几个。由于Linux 中将设备当做文件处理,所以对设备进行操作的调用格式与对文件的操作类似,主要包括open()、read()、write()、ioctl()、close()等。应用程序发出系统调用命令后,会从用户态转到核心态,通过内核将open()这样的系统调用转换成对物理设备的操作。 5. 驱动程序结构一个设备驱动程序模块的基本框架在系统内部,I/O设备的存取通过一组固定的入口点来进行,入口点也可以理解为设备的句柄,就是对设备进行操作的基本函数。字符型设备驱动程序提供如下几个入口点: open入口点。打开设备准备I/O操作。对字符设备文件进行打开操作,都会
9、调用设备的open入口点。open子程序必须对将要进行的I/O操作做好必要的准备工作,如清除缓冲区等。如果设备是独占的,即同一时刻只能有一个程序访问此设备,则open子程序必须设置一些标志以表示设备处于忙状态。 close入口点。关闭一个设备。当最后一次使用设备完成后,调用close子程序。独占设备必须标记设备方可再次使用。 read入口点。从设备上读数据。对于有缓冲区的I/O操作,一般是从缓冲区里读数据。对字符设备文件进行读操作将调用read子程序。 write入口点。往设备上写数据。对于有缓冲区的I/O操作,一般是把数据写入缓冲区里。对字符设备文件进行写操作将调用write子程序。 ioc
10、tl入口点。执行读、写之外的操作。select入口点。检查设备,看数据是否可读或设备是否可用于写数据。select系统调用在检查与设备文件相关的文件描述符时使用select入口点。6. 设备注册和初始化设备的驱动程序在加载的时候首先需要调用入口函数init_module(),该函数最重要的一个工作就是向内核注册该设备,对于字符设备调用register_chrdev()完成注册。register_chrdev 的定义为:int register_chrdev(unsigned int major, const char *name, struct file_ operations *fops);
11、其中,major是为设备驱动程序向系统申请的主设备号,如果为0,则系统为此驱动程序动态分配一个主设备号。name是设备名,fops是对各个调用的入口点说明。此函数返回0时表示成功;返回-EINVAL,表示申请的主设备号非法,主要原因是主设备号大于系统所允许的最大设备号;返回-EBUSY,表示所申请的主设备号正在被其他设备程序使用。如果动态分配主设备号成功,此函数将返回所分配的主设备号。如果register_chrdev()操作成功,设备名就会出现在/proc/dvices文件中。Linux在/dev目录中为每个设备建立一个文件,用ls l命令列出函数返回值,若小于0,则表示注册失败;返回0或者
12、大于0的值表示注册成功。注册以后,Linux将设备名与主、次设备号联系起来。当有对此设备名的访问时,Linux通过请求访问的设备名得到主、次设备号,然后把此访问分发到对应的设备驱动,设备驱动再根据次设备号调用不同的函数。当设备驱动模块从Linux内核中卸载,对应的主设备号必须被释放。字符设备在cleanup_ module()函数中调用unregister_chrdev()来完成设备的注销。unregister_chrdev()的定义为:int unregister_chrdev(unsigned int major, const char *name);包括设备注册在内,设备驱动的初始化函数
13、主要完成的功能是有以下5项。(1)对驱动程序管理的硬件进行必要的初始化。对硬件寄存器进行设置。比如,设置中断掩码,设置串口的工作方式、并口的数据方向等。(2)初始化设备驱动相关的参数。一般说来,每个设备都要定义一个设备变量,用以保存设备相关的参数。在这一步骤里对设备变量中的项进行初始化。(3)在内核注册设备。调用register_chrdev()函数来注册设备。(4)注册中断。如果设备需要IRQ支持,则要使用request_irq()函数注册中断。(5)其他初始化工作。初始化部分一般还负责给设备驱动程序申请包括内存、时钟、I/O端口等在内的系统资源,这些资源也可以在open子程序或者其他地方申
14、请。这些资源不用时,应该释放,以利于资源的共享。若驱动程序是内核的一部分,初始化函数则要按如下方式声明:int _init chr_driver_init(void);其中_init是必不可少的,在系统启动时会由内核调用chr_driver_init,完成驱动程序的初始化。当驱动程序是以模块的形式编写时,则要按照如下方式声明:int init_module(void)当运行后面介绍的insmod命令插入模块时,会调用init_module函数完成初始化工作。 7. 设备驱动程序的开发过程由于嵌入式设备由于硬件种类非常丰富,在默认的内核发布版中不一定包括所有驱动程序。所以进行嵌入式Linux系统
15、的开发,很大的工作量是为各种设备编写驱动程序。除非系统不使用操作系统,程序直接操纵硬件。嵌入式Linux系统驱动程序开发与普通Linux开发没有区别。可以在硬件生产厂家或者Internet上寻找驱动程序,也可以根据相近的硬件驱动程序来改写,这样可以加快开发速度。实现一个嵌入式Linux设备驱动的大致流程如下。(1)查看原理图,理解设备的工作原理。一般嵌入式处理器的生产商提供参考电路,也可以根据需要自行设计。(2)定义设备号。设备由一个主设备号和一个次设备号来标识。主设备号惟一标识了设备类型,即设备驱动程序类型,它是块设备表或字符设备表中设备表项的索引。次设备号仅由设备驱动程序解释,区分被一个设
16、备驱动控制下的某个独立的设备。(3)实现初始化函数。在驱动程序中实现驱动的注册和卸载。(4)设计所要实现的文件操作,定义file_operations结构。(5)实现所需的文件操作调用,如read、write等。(6)实现中断服务,并用request_irq向内核注册,中断并不是每个设备驱动所必需的。(7)编译该驱动程序到内核中,或者用insmod命令加载模块。(8)测试该设备,编写应用程序,对驱动程序进行测试。 五、 基本接口实验在完成了基本实验后,我们开始着手基本接口实验。在这些实验中,我们学习如何编写设备驱动程序,及如何用测试程序检验驱动程序是否正确,并通过改写测试程序正常地对驱动程序进
17、行相关操作。8. 实验十二 简单设备驱动程序本次实验的任务是编写一个字符型设备驱动程序,并学习在应用程序中调用驱动。考虑到我们初次接触驱动程序的编写,对此还十分陌生,因此指导书中提供了本次实验所要用到的程序源代码。虽然这样一个字符型设备驱动程序并没有任何实际作用,但是它让我们轻松掌握了嵌入式驱动的编写过程,因为复杂繁琐的驱动,其骨架都是相同的。因此,看懂本实验的源代码,学习并模仿其编写方法,对于后续实验有着非常重要的意义。9. 实验十三 CPU GPIO驱动程序设计在本实验中,我们要编写第一个针对实际硬件的驱动程序。我们知道,凡是操作系统控制外部设备,即使是最简单的硬件电路,也是需要驱动的。本
18、实验涉及的外部硬件只有电阻和发光二极管。我们使用自己编写的驱动程序与应用程序控制 GPIO96的电平,通过 LED 的亮灭来判断,是否 CPU 做出了正确的响应。实验第一步是编写 PXA270 GPIO 驱动程序。驱动的写法参照实验十二,大体相同,主要区别如下:/ -控制IO设备-switch (cmd)case LED_ON : GPCR3 |= 0x1;break; /如果cmd=LED_ON,那么GPCR3置为1case LED_OFF: GPSR3 |= 0x1;break; /如果cmd=LED_OFF,那么GPSR3置为1 default : printk (lcd control
19、 : no cmd run -kernel- n); return (-EINVAL);/ -驱动程序初始化-GPDR3 |= 0x00000001; /设置GPIO96输出模式:开灯GPSR3 |= 0x00000001; / 关灯对Makefile中的目标体和依赖文件也要做相应的修改,此处省略。在测试程序中有这样一段代码:while(1) ioctl(fd, LED_OFF);sleep(1); / 休眠1秒ioctl(fd,LED_ON);sleep(1); 实验作业要求在目标板上LED闪烁产生亮7秒,灭5秒的效果,很容易实现,只需将上面的代码改为如下代码即可:while(1) ioct
20、l(fd, LED_OFF);sleep(5); / 灭5秒ioctl(fd,LED_ON);sleep(7); / 亮7秒10. 实验十四 中断实验在理论课中,我们学习了许多中断方面的知识,包括中断向量、中断优先级、中断过程等。在PXA270系统里,中断控制器分外部设备和 PXA270X 处理器设备产生的两个层次的中断,前者是初级的中断源,后者是次级中断源,大量的次级中断源通常被映射为一个初级中断源。在此实验中,我们要编写一个中断程序,利用目标板上的按键SW2来产生中断,使得当每次按下此按键时,在超级终端上打印出响应的信息。编写中断程序与前两个实验的主要区别如下:/* -初始化request
21、_irq申请硬件中断,参数包括申请的硬件中断号、设备id、中断处理的一些属性(SA_INTERRUPT是快速处理程序,调用时屏蔽所有中断)等- */ret = request_irq(SIMPLE_INT_IRQ, &SIMPLE_INT_interrupt, SA_INTERRUPT, int_ctl, NULL);/ - 卸载,对应request_irq释放中断 -free_irq(SIMPLE_INT_IRQ,NULL);通过此实验,我了解了硬件中断管脚与中断号的对应关系,以及中断号与中断处理程序的对应关系,对于今后编写更为复杂的中断程序打下基础。11. 实验十五 数码管显示实验在此实验
22、中,我们要编写针对 74LV164 的驱动程序,并用其串并转换功能来控制八段LED数码管的显示。在编写驱动程序时,主要有以下需要注意的:/ 按位写入void write_bit(int data) GPCR2 |= (0x1 27);if(data & 0x80) = 0x80) GPSR2 |= (0x1 26);elseGPCR2 |= (0x1 26);GPSR2 |= (0x1 27);/ 按字节写,一次写8位void write_byte(int data) int i;for(i=0;i8;i+) write_bit( data i );使用测试程序看到的测试结果是数码管按0-9显
23、示输出。实验作业要求在上述基础上,分别实现一下两个功能: 要求您再编写一个测试程序,实现 PXA270-EP 目标板上的 LED 数码管循环显示的数字9-0。 要求您再编写一个测试程序,实现 PXA270-EP 目标板上的 LED 数码管循环显示的数字02468。由于在测试程序中定义了数组buf10分别存储了0-9是个数,因此上述功能的实现方法是,分别对测试程序做如下修改:原测试程序:while(1) for(count=0;count=0;count-) / 倒序显示数字 data0 = bufcount; ret=write(fd,data,1); sleep(1);实现功能:while(
24、1) for(count=0;count9;count=count+2) / 更改显数顺序 data0 = bufcount; ret=write(fd,data,1); sleep(1);通过更改显数的顺序,很容易实现实验作业里要求的功能。12. 实验十六 LED点阵驱动程序设计通过本实验的操作,我们将 8X8 的点阵 LED 驱动起来并通过编写测试程序,使其能够按照您的意图进行显示。要求您还编写更多的测试程序 要求您再编写一个测试程序,实现按横的方向隔行顺序扫描 LED 点阵数码管。 要求您再编写一个测试程序,实现按竖的方向顺序扫描 LED 点阵数码管。作业一,隔行扫描: for (i=1
25、;i=8;i2+) buf0=c; buf1=r; / row for (j=1;j=8;j+) write(fd,buf,2); printf (buf0,buf1: %x,%xn,buf0,buf1); usleep(200000); / sleep 0.2 second r=r1; buf1=r; / column r = 1; c = c1;作业二,竖向扫描: for (i=1;i=8;i+) buf0=c; buf1=r; / row for (j=1;j=8;j+) write(fd,buf,2); printf (buf0,buf1: %x,%xn,buf0,buf1); usl
26、eep(200000); / sleep 0.2 second r=r1; buf1=r; / column r = 1; c = c1;13. 实验十七 AD驱动实验通过本实验的操作,我们将 AD 转换器驱动起来并通过编写测试程序,使其能够将模拟信 号量按照我们的要求转换成数字信号量。为了更加清楚地理解 AD 转换器的工作过程,请您再 编写一个测试程序,将 UCB_ADC_INP_AD0 换成其他通道,来观察其他 AD 通道情况。主要代码: for(i=0;i50;i+) val0 = ioctl(fd,UCB_ADC_INP_AD1,0); usleep(100); val1 = ioct
27、l(fd,UCB_ADC_INP_AD0,0); usleep(100); val2 = ioctl(fd,UCB_ADC_INP_AD2,0); usleep(100);14. 实验十八 DA驱动实验通过本实验的操作,我们使用示波器看到了通过 DA 转换而输出的波形。在此基础上,要求试写一个实现输出三角波的测试程序。主要代码: while (flag_func_run = FUNC_RUN) print_prompt(); / print select functions scanf(%d,&flag_select_func); / user input select getchar();
28、/ get ENTER switch(flag_select_func) case DA_SIN : da_create_sin(fd); break; case DA_FANG : da_create_fang(fd); break; case FUNC_QUIT : flag_func_run = FUNC_NOT_RUN; printf(Quit DA function. byebyen); break; case DA_TRI :da_create_tri(fd); break; default : printf (input = %xn,flag_select_func); prin
29、tf (statys = %xn,flag_func_run); printf( - please input your select use 1 to 4 -n); 六、 实验中遇到的问题及解决方法每一次上课重新启动后,当需要将宿主PC机的根目录挂在到PXA270-EP目标板的mnt目录下(即在超级终端中输入命令“mount o soft,timeo=100,rsize=1024 192.168.0.100:/ /mnt”)时,常显示无法挂载。解决方法:在超级终端下的挂载命令应该用”mount o nolock 192.168.0.100:/ /mnt”,如果依然不能挂载需要重启NFS服务,即在PC机终端中输入命令”service nfs restart”两遍后就可以挂载,当然有时候也可能是因为网线没插好。在每次
copyright@ 2008-2022 冰豆网网站版权所有
经营许可证编号:鄂ICP备2022015515号-1