《电子技术应用》
您所在的位置:首页 > 其他 > 业界动态 > CAN通信卡的Linux设备驱动程序设计实现

CAN通信卡的Linux设备驱动程序设计实现

2009-05-11
作者:杨 峰 吉吟东 薛 明

    摘  要: 介绍了Linux下设备驱动程序的结构,描述了CAN通信卡设备驱动程序的软件框架以及如何将CAN设备驱动程序加入到Linux系统内核中。讨论了具体实现中为了提高通信效率和通信能力,改进设备驱动程序的缓冲区管理以及利用Linux的特点合理设计中断处理程序。 

    关键词: Linux操作系统     设备驱动程序  CAN通信卡  中断处理程序

 

    目前,许多工业现场如电力系统、化工系统等大量使用控制器局部网(CAN——Controller Area Network)现场总线网络,CAN通信卡作为计算机的外设将计算机接入CAN网络。市场上有不少CAN通信卡,但基本上都不带Linux驱动程序,当需要在Linux下使用CAN通信卡设备时,需自己开发Linux的驱动程序。开发Linux驱动程序不但要求程序员要非常熟悉Linux系统,而且要熟悉Linux驱动程序开发的规范。本文将详细介绍CAN通信卡的Linux驱动设备程序的设计和实现。 

1 CAN通信卡的Linux设备驱动程序结构 

    Linux系统内核通过设备驱动程序与外围设备进行交互,设备驱动程序是Linux内核的一部分,它是一组数据结构和函数,这些数据结构和函数通过定义的接口控制一个或多个设备。对用户程序而言,设备驱动程序隐藏了设备的具体细节,对各种不同设备提供一致的接口,一般来说是把设备映射为一个特殊的设备文件,用户程序可以象对普通文件一样对此设备文件进行操作。 

    Linux将每个设备看作一个文件,即可以像对待文件那样使用read、write等系统调用进行读写。Linux的设备文件分为两类:一是字符设备,只能对该类设备进行顺序读写,对外提供字节流方式的操作;二是块设备,可以对该类设备进行随机访问,一般是磁盘设备等大容量存储设备。CAN通信卡设备属于字符型设备。 

    对设备的访问是由设备驱动程序提供的。Linux的设备驱动程序可以用模块的方式加载入内核,设备驱动程序与Linux系统的关系如图1所示。 

1.1  CAN通信卡设备的特点 

    控制器局部网(CAN)属于现场总线的范畴,它是一种有效支持分布式控制或实时控制的串行通信网络。由于其性能优异、价格低廉,很快被推广到工业测控现场。 

    CAN通信卡硬件实现CAN定义的物理层和数据链路层功能,收发报文中数据长度为0~8个字节,有2032个报文标识符。工作时通过报文标识符确定总线访问优先权,高优先级报文具有低延迟时间,数据传送速率可编程(最高为1Mbps)。发送期间若丢失仲裁或由于出错而遭破坏的报文可自动重发。具有成组和广播报文功能。 

    当CAN通信卡接收到一个报文时,数据保存在CAN通信卡上的接收缓存器中,并产生一个接收中断。当一个报文被成功发送后,发送缓冲器可再次被访问,产生一个发送中断信号。CAN通信卡发送报文,将数据送入CAN通信卡上的发送缓存器中,CAN通信卡将数据串行化发到CAN总线上。 

1.2 CAN通信卡设备驱动程序的任务 

    由于CAN一帧的数据长度最大为8个字节,可以用多帧的Hilon A协议来使CAN传输数据任意长。CAN通信卡驱动程序主要完成按照Hilon A协议解包接收和打包发送任务,并要对接收的多帧进行管理。 

    CAN通信卡驱动程序应该完成以下任务: 

    (1)为应用程序提供通过CAN卡发送和接收任意长度数据的能力; 

    (2)为应用程序提供设置CAN卡上CAN控制器运行参数的能力; 

    (3)以阻塞或非阻塞方式读写CAN设备文件; 

    (4)允许CAN卡同时收发多路数据。 

1.3 CAN通信卡驱动程序的处理流程 

    用户进程通过系统调用向驱动程序传送一帧任意长度的数据,驱动程序中发送数据的程序按照Hilon A协议将该帧分段打包,放入发送队列,并向CAN控制器请求发送,由中断处理程序中发送部分负责发送完所有的数据包。 

    当CAN通信卡接收到数据时,产生接收中断,启动接收中断处理程序上半部将CAN控制器中接收缓冲器中的内容复制到接收队列中,由中断处理的下半部负责解包和组帧的任务,并将处理完的帧放入帧队列中,最后用户使用系统调用从接收帧队列中读取完整的一帧。 

    CAN通信卡设备驱动程序处理框架如图2所示。 

 

 

2 CAN通信卡设备驱动程序的模块化程序设计 

    根据Linux对设备驱动程序的要求,模块化的CAN驱动程序软件结构如图3。 

 

 

2.1 初始化加载和卸载部分 

    如果设备驱动程序以模块方式加入内核,则一定会包括两个模块init_module和clear_module。init_module模块用来加载设备,系统初始化时调用;clear_module模块用来卸载设备,取消设备时调用。 

    设备驱动程序是系统内核的一部分。在任何程序使用设备驱动程序之前,设备驱动程序应该向系统进行登记,以便系统在适当的时候调用。Linux系统里,通过调用register_chrdev函数向系统注册字符型设备驱动程序。register_chrdev定义为:  

    int register_chrdev(unsigned int major, const char *name, struct file_operations *fops);  

    其中,major是为设备驱动程序向系统申请的主设备号,如果为0则系统为此驱动程序动态分配一个主设备号;name是设备名;fops是对各个系统调用的入口点的说明。  

    CAN通信卡使用中断与系统交换数据,CAN设备驱动程序需要使用内存来缓存接收到的数据和发送的数据,中断和内存等资源是由Linux系统统一管理的,设备驱动程序在初始化时,需要申请资源。在资源不用的时候,应该释放它们,以利于资源的共享。在Linux系统里,对中断的处理是属于系统核心的部分,设备驱动程序通过调用request_irq()函数来申请中断,通过free_irq()函数来释放中断。作为系统核心的一部分,设备驱动程序在申请和释放内存时不是调用malloc()函数和free()函数,而代之以调用kmalloc函数和kfree()函数。 

    在init_module模块中,先检查是否存在CAN通信卡,如果不存在则退出设备驱动程序的加载;如果存在,使用request_irq()函数为CAN通信卡申请系统中空闲的中断,使用kmalloc()为设备驱动程序申请输入输出缓存队列,如果这些资源申请不成功,则释放已经申请到的系统资源,然后退出设备驱动程序的加载,如果申请成功,使用register_chrdev()函数将CAN通信卡驱动程序注册到Linux系统中,加载完成。 

    在cleanup_module模块中先使用free_irq()函数释放init_module模块中申请到的中断,然后使用kfree()函数释放init_module模块中申请到的内存空间,最后使用unregister_chrdev()函数释放init_module模块中注册的设备驱动程序,卸载完成。 

2.2  CAN通信卡设备驱动程序的中断处理部分 

    Linux中断处理程序可以分为上半部和下半部。上半部即一般的中断服务程序,由硬件中断触发,它一般运行在关中断的方式下,应当尽可能短小,处理尽可能快;而下半部是单独的一段处理程序,一般将其挂入立即队列中以便快速执行。立即队列中的任务在退出系统调用或调度器获得运行时,将最优先地被执行。下半部运行在一个安全的环境,即开中断和任务串行化,可以处理一些较花时间的任务。 

    这样,驱动程序上半部在处理完实时性很强的任务后,用queue_task()函数将下半部处理函数挂入立即队列,并用mark_bh()函数来激活立即队列,则下半部可以最优先地被执行。 

    当有接收中断时,CAN通信卡设备驱动程序的中断处理程序上半部首先获取驱动程序接收缓存中的空闲块,将CAN通信卡上的接收缓冲器中数据复制到驱动程序的接收缓存中,释放CAN通信卡接收缓冲器,然后将下半部处理函数挂入立即队列,最后激活立即队列。 

    当有发送中断时,CAN通信卡设备驱动程序的中断处理程序上半部首先获取发送缓冲队列中的数据,将需要发送的数据写入CAN控制器的发送缓冲器,最后请求发送。 

    CAN通信卡设备驱动程序的中断处理程序下半部使用Hilon A协议对要传送到CAN网络上的数据打包,并对从CAN网络上接收到的数据进行解包和组帧。这样中断处理程序的上半部只需要从CAN通信卡的缓冲器中将数据复制到驱动程序的缓冲区,系统开销很小,但很费CPU时间;系统开销较大的打包、解包和组帧处理则放在中断处理程序的下半部,使用系统非中断时间调度,可以使系统响应中断更快,通信更稳定。 

2.3 缓冲区管理 

    在CAN通信卡设备驱动程序中,为了增强CAN通信卡的通信能力、提高通信效率,根据CAN的特点,使用两级缓冲区结构,即直接面向CAN通信卡的收发缓冲区和直接面向系统调用的接收帧缓冲区。 

    通讯中的收发缓冲区一般采用环形队列(或称为FIFO队列),使用环形的缓冲区可以使得读写并发执行,读进程和写进程可以采用“生产者和消费者”的模型来访问缓冲区,从而方便了缓存的使用和管理。然而,环形缓冲区的执行效率并不高,每读一个字节之前,需要判断缓冲区是否为空,并且移动尾指针时需要进行“折行处理”(即当指针指到缓冲区内存的末尾时,需重新将其定向到缓冲区的首地址);每写一个字节之前,需要判断缓冲区是否为满,并且移动尾指针时同样需要进行“折行处理”。程序大部分的执行过程都是在处理个别极端的情况,只有小部分在进行实际有效的操作。这就是软件工程中所谓的“8比2”关系。结合CAN通讯的实际情况,在本设计中对环形队列进行了改进,可以较大地提高数据的收发效率。 

    由于CAN通信卡上接收和发送缓冲器每次只接收一帧CAN数据,而且根据CAN的通讯协议,CAN控制器的发送缓冲器由1个字节的标识符、一个字节的RTR和DLC位及8个字节的数据区组成,共10个字节;接收缓冲器与之类似,也有10个字节的寄存器。所以CAN控制器收发的数据是短小的定长帧(数据可以不满8字节)。 

    于是,采用长度为10字节的数据块来分配内存比较方便,即每次需要内存缓冲区时,直接分配10个字节,由于这10个字节的地址是线性的,故不需要进行 “折行” 处理。更重要的是,在向缓冲区中写数据时,只需要判断一次是否有空闲块并获取其块首指针就可以了,从而减少了重复性的条件判断,大大提高了程序的执行效率;同样在从缓冲队列中读取数据时,也是一次读取10字节的数据块,同样减少了重复性的条件判断。 

    在CAN卡驱动程序中采用如下所示的称为“Block_Ring_t”的数据结构作为收发数据的缓冲区: 

        typedef struct { 

    long signature; 

    unsigned char *head_p; 

    unsigned char *tail_p; 

    unsigned char *begin_p; 

    unsigned char *end_p; 

    unsigned char buffer[BLOCK_RING_BUFFER_SIZE]; 

    int usedbytes; 

    }Block_Ring_t; 

    该数据结构在通用的环形队列上增加了一个数据成员usedbytes,它表示当前缓冲区中有多少字节的空间被占用了。使用usedbytes,可以比较方便地进行缓冲区满或空的判断。当usedbytes=0时,缓冲区空;当usedbyes=BLOCK_RING_BUFFER_SIZE时,缓冲区满。 

    本驱动程序除了收发缓冲区外,还有一个接收帧缓冲区,接收帧队列负责管理经Hilon A协议解包后得到的数据帧。由于有可能要同时接收多个数据帧,而根据CAN总线的通讯协议,高优先级的报文将抢占总线,则有可能在接收一个低优先级且被分为好几段发送的数据帧时,被一个优先级高的数据帧打断。这样会出现同时接收到多个数据帧中的数据包,因而需要有一个接收队列对同时接收的数据帧进行管理。 

    当有新的数据包到来时,应根据addr(通讯地址),mode(通讯方式),index(数据包的序号)来判断是否是新的数据帧。如果是,则开辟新的frame_node;否则如果已有相应的帧节点存在,则将数据附加到该帧的末尾;在插入数据的同时,应该检查接收包的序号是否正确,如不正确将丢弃这包数据。 

    每次建立新的frame_node时,需要向frame_queue申请内存空间;当frame_queue已满时,释放掉队首的节点(最早接收的但未完成的帧)并返回该节点的指针。 

    当系统调用读取了接收帧后,释放该节点空间,使设备驱动程序可以重新使用该节点。 

2.4 服务于I/O请求的设备驱动程序部分 

    这部分实际上是应用程序唯一可见的,应用程序通过系统来调用这部分程序,是设备驱动程序对应用程序的接口。本驱动程序提供文件操作接口。Linux系统中,字符型设备驱动程序提供的文件操作入口点由一个结构来向系统说明,此结构定义为:  

    struct file_operations {  

    int (*lseek)(struct inode *inode,struct file *filp, off_t off,int pos);  

    int (*read)(struct inode *inode,struct file *filp, char *buf, int count);  

    int (*write)(struct inode *inode,struct file *filp, char *buf,int count);  

    int (*readdir)(struct inode *inode,struct file *filp,struct dirent *dirent,int count);  

    int (*select)(struct inode *inode,struct file *filp, int sel_type,select_table *wait);  

    int (*ioctl) (struct inode *inode,struct file *filp, unsigned int cmd,unsigned int arg);  

    int (*mmap) (void);  

    int (*open) (struct inode *inode, struct file *filp);  

    void (*release) (struct inode *inode, struct file *filp);   int (*fsync) (struct inode *inode, struct file *filp);  

    }; 

    该结构定义了10个操作入口点,但是驱动程序没有必要对每个入口点进行定义。根据需要,本驱动程序定义了如下的入口点。 

    can_open(struct inode *inode, struct file *filp)入口点负责打开can设备,检查can卡是否已被打开,完成can卡的初始化,设置设备的占用标志。can_release(struct inode *inode, struct file *filp)入口点负责关闭can设备。 

    can_read(struct inode *, struct file *, off_t, int) 入口点负责检查设备有没有接收到完整的帧,can_read函数只是判断是否有完整的数据帧可读。要获取数据帧,可以使用ioctl 的CAN_READFRAME命令。can_write(struct inode*, struct file *, const char *, int) 入口点负责向CAN发送数据。如果发送队列有足够的空间,则向设备传送数据,也可以使用ioctl的CAN_WRITEFRAME命令来实现can_write。 

    can_ioctl(struct inode *, struct file *, unsigned int cmd, unsigned long arg)入口点负责向CAN设备下发各种操作命令,命令代码通过cmd参数传送,命令参数通过arg参数传送。本驱动程序提供了一些命令,配合can_read()和can_write() 可以实现对CAN通信卡的控制。CAN_IOCREADFRAME命令可以从CAN通信卡上读取数据帧;CAN_IOCWRITEFRAME命令可以向CAN通信卡发送数据;CAN_IOCSETCONF命令可以设置CAN通信卡的运行参数;CAN_IOCGETCONF命令可以获取CAN控制器的运行参数;CAN_IOCQUERYBUSSTATE命令可以查询CAN总线状态;CAN_IOCCLEARBUF命令可以清除CAN通信卡的收发缓冲区。 

    本设备驱动程序考虑到CAN通信卡的特点和CAN网络传输数据的特点,设计了合理的数据结构和缓存管理方法,使得当有大量数据进出CAN通信卡时,既可以保证数据帧丢失和出错几率在允许范围内,又可以保证数据帧能被快速下发和接收,实际应用中性能很好。Linux擅长通信,支持大多数以太网卡。如果将CAN通信卡的设备驱动程序加入到Linux系统,由于Linux的可裁减性和对硬件资源要求低的特点,可以用小硬盘、小内存和低档CPU构成通信机连接高速以太网和低速现场总线CAN网络,经济实惠而且实用。 

参考文献 

1 Uresh Vahalia著.聊鸿斌译.UNIX高级教程-系统技术内幕.北京:清华大学出版社,1999 

2 Alessandro Rubini著.Lisoleg译.Linux设备驱动程序.北京:中国电力出版社,2000 

3 邬宽明.CAN总线原理和应用系统设计.北京:北京航空航天大学出版社,1996

本站内容除特别声明的原创文章之外,转载内容只为传递更多信息,并不代表本网站赞同其观点。转载的所有的文章、图片、音/视频文件等资料的版权归版权所有权人所有。本站采用的非本站原创文章及图片等内容无法一一联系确认版权者。如涉及作品内容、版权和其它问题,请及时通过电子邮件或电话通知我们,以便迅速采取适当措施,避免给双方造成不必要的经济损失。联系电话:010-82306118;邮箱:aet@chinaaet.com。