目前,许多工业现场如电力系统、化工系统等大量使用控制器局部网(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通信卡设备驱动程序的模块化程序设计
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是设备名;fopsa是对各个系统调用的入口点的说明 。
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 {
log 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时,缓冲区空;当usedbytes=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)(strut inode *inode,struct file *file,off_toff,int pos);
int (*read)(struct inode *inode,struct file *filp,char *buf,int count);
int (*write)(struct inode *inode,struct file *file,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_inoctl(struct inode *,struct file *,unsigned int cmd,unsigned long arg)入口点负责向CAN设备下发各种操作命令,命令代码通过cmd参数传送,命令参数通过arg参数传送。本驱动程序提供了一些命令,配合 can_read()和can_write()可以实现对CAN通信卡的控制。CAN_IOCREADFFRAME命令可以从CAN通信卡上读取数据帧; CAN_IOCWRITEFRAME命令可以向CAN通信卡发送数据;CAN_IOCSETCONF命令可以设备CAN通信卡的运行参数; CAN_IOCGETCONF命令可以获取CAN控制器的运行参数;CAN_IOCQUERY-BUSSTATE命令可以查询CAN总线状态; CAN_INCCLEARBUF命令可以清除CAN通信卡的收发缓冲区。
本设备驱动程序考虑到CAN通信卡的特点如CAN网络传输数据的特点,设计了合理的数据结构和缓存管理方法,使得当有大量数据进出CAN通信卡时,既可以保证数据帧丢失和出错几率在允许范围内,又可以保证数据帧能被快速下发和接收,实现应用中性能很好。Linux擅长通信,支持大多数以太网卡。如果将CAN 通信卡的设备驱动程序加入到Linux系统,由于Linux的可裁减性和对硬件资源要求低的特点,可以用小硬盘、小内存和低档CPU构成通信机连接高速以太网和低速现场总线CAN网络,经济实惠而且实用。
免责声明: 凡注明来源本网的所有作品,均为本网合法拥有版权或有权使用的作品,欢迎转载,注明出处。非本网作品均来自互联网,转载目的在于传递更多信息,并不代表本网赞同其观点和对其真实性负责。