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

目前,许多工业现场如电力系统、化工系统等大量使用控制器局部网(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。

目前,许多工业现场如电力系统、化工系统等大量使用控制器局部网(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。

在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网络,经济实惠而且实用。

can总线linux程序,CAN通信卡的Linux设备驱动程序 - 嵌入式操作系统 - 电子发烧友网...相关推荐

  1. 智能硬件Linux效能更高,Android软硬件巧妙整合的开发技 - 嵌入式操作系统 - 电子发烧友网...

    1.Linux驱动程序的导入 由于Android是根源于Linux所延伸出来的操作系统,因此各种关键功能的驱动程序也必须要能顺利的写入其中,举凡像是字符设备.存储器的空间配置.中断处理.网路通信.萤幕 ...

  2. linux安全模式改文件,嵌入式Linux的安全模式设计 - 嵌入式操作系统 - 电子发烧友网...

    本系统的架构如下图: 产品所使用的flash总大小为16M. 系统包括三大部分,即Bootloader,config, kernel + rootfs: 另外,/dev/mtdblock/0,在系统中 ...

  3. 嵌入式linux系统电源管理器,嵌入式Linux系统的动态电源管理技 - 嵌入式操作系统 - 电子发烧友网...

    嵌入式Linux系统的动态电源管理技术 通过用户层制定策略与内核提供管理功能交互,实时调整电源参数而同时满足系统实时应用的需求,允许电源管理参数在短时间的空闲或任务运行在低电源需求时,可以被频繁地.低 ...

  4. linux步进电机驱动程序,基于Linux的步进电机驱动程序设 - 嵌入式操作系统 - 电子发烧友网...

    定义时钟节拍函数time_tick() static void time_tick(unsigned long data) { static int i=0; switch(status) { cas ...

  5. Linux下查看硬盘序列号、设备序列号、操作系统版本和安装时间、系统启用时间等命令

    Linux下查看硬盘序列号.设备序列号.操作系统版本和安装时间.启用时间等命令 最近由于工作需要查询一些硬盘序列号.设备序列号.操作系统版本和安装时间.系统启用时间等信息.所以对用到的命令进行一下总结 ...

  6. linux 程序定时重启tomcat,linux下实现tomcat定时自动重启

    tomcat自带的脚本中没有提供直接restart的模式,但是有start和shutdown两种模式.要实现restart模式,实际上只需要判断是否已经启动tomcat,若已经启动则限制性shutdo ...

  7. linux串口驱动及应用程序,基于华邦W90P710处理器的Linux内核应用及串口驱动的实现-嵌入式系统-与非网...

    嵌入式Linux是一种很受欢迎的操作系统,具有开放源码.不存在黑箱技术.内核小.功能强大.运行稳定.效率高.易于定制裁减等特点[1],广泛应用于工控产品.很多工控产品需要和外部设备进行信息交换,而串口 ...

  8. linux程序分析工具下载,linux 程序分析工具

    http://blog.csdn.net/denny_233/article/details/7477282 2012 linux程序分析工具介绍(一)--"/proc" 写在最前 ...

  9. linux程序设计基础——概述,6.Linux程序设计基础—Makefile.pdf

    6.Linux程序设计基础-Makefile Haubo Training Center Linux开发基础-Makfile 张勇涛 GNU make 和makefile  GNU make概述  ...

最新文章

  1. nginx rewrite重写与防盗链配置
  2. 安卓学习UI组件-ExpandablelistView-可拓展的两级列表
  3. mysql error manager,MYSQL Starting MySQL. ERROR! Manager of pid-file quit without updating file
  4. 简单示例立马搞懂Java日期格式中yyyy-MM-dd HH:mm:ss和YYYY-MM-dd hh:mm:ss的区别
  5. 洛谷3605 Promotion Counting
  6. SAP 电商云 Spartacus UI 如何在应用程序里捕捉到成功下单的事件
  7. 有效IT运维 效率提高 成本降低
  8. 算法学习之路|方格分割
  9. Android7.1 Offload模式下的音频数据播放流程
  10. 软中断上下文能够睡眠吗?
  11. Android知识笔记:特效系列,仿instagram文字排版特效,android应用开发基础学堂在线答案
  12. JTAG接口简要介绍
  13. 计算机科学成为独立学科的奠基人,详解卡内基梅隆大学计算机学院
  14. 高铁轮毂表面缺陷的<视觉显著性>超像素图像检测方法
  15. Openwrt如何添加SD_Card?看这里
  16. RobotFramework中,选择注释套件中的某些用例的方法(主要用于RED/IntelliBot从套件执行单个用例)
  17. 我们DevSkim、Jarvis和USG是这样子的
  18. 在线选课系统UML大作业
  19. Pandas学习-Day3
  20. Rust P2P网络应用实战-1 P2P网络核心概念及Ping程序

热门文章

  1. 三花集金花茶“茶族皇后”不仅貌美,还有丰富的营养价值
  2. 20230123使AIO-3568J开发板在Android12下永不休眠
  3. Linux打开最大文件数限制
  4. 大学生对计算机的认识调查报告,大学生电脑使用情况调查报告
  5. 对泊松分布和泊松过程已经爱尔朗分布的理解和他们的联系(排队论和运筹学中的重要理论)
  6. 煤矿生产高精专!选矿厂 3D 可视化监管,实现提质增效
  7. gridview分页的问题
  8. Unity 获取手机键盘弹出高度
  9. 解决IDEA中出现Error:Cannot run program “D:\ideaJDK\jdk1.8.0_241\bin\java“导致程序无法运行的问题。
  10. 讨论电子工程师的前途