当前位置:   article > 正文

linux can总线接收数据串口打包上传_【Linux应用】CAN总线编程

linux怎么通过串口工具发送can数据

1 CAN总线简介

CAN是控制器局域网络(Controller Area Network,CAN)的简称,由德国BOSCH公司开发,并最终成为国际标准(ISO 11898-1)。CAN总线主要应用于工业控制和汽车电子领域,是国际上应用最广泛的现场总线之一。 CAN总线是一种串行通信协议,能有效地支持具有很高安全等级的分布实时控制。CAN总线的应用范围很广,从高速的网络到低价位的多路接线都可以使用CAN。在汽车电子行业里,使用CAN 连接发动机的控制单元、传感器、防刹车系统等,传输速度可达1Mbps。

5562f242cdb7d18e1aac1e62d9bf755f.png

与前面介绍的一般通信总线相比,CAN总线的数据通信具有突出的可靠性、实时性和灵活性,在汽车领域的应用最为广泛,世界上一些著名的汽车制造厂商都采用CAN总线来实现汽车内部控制系统与各检测和执行机构之间的数据通信。目前,CAN总线的应用范围已不仅仅局限于汽车行业,而且已经在自动控制、航空航天、航海、过程工业、机械工业、纺织机械、农用机械、机器人、数控机床、医疗器械及传感器等领域中得到了广泛应用。CAN总线规范从最初的CAN 1.2规范(标准格式)发展为兼容CAN 1.2规范的CAN 2.0规范(CAN 2.0A为标准格式,CAN 2.0B为扩展格式),目前应用的CAN器件大多符合CAN 2.0规范。

2 CAN总线的工作原理

当CAN总线上的节点发送数据时,以报文形式广播给网络中的所有节点,总线上的所有节点都不使用节点地址等系统配置信息,只根据每组报文开头的 11 位标识符(CAN 2.0A 规范)解释数据的含义来决定是否接收。这种数据收发方式称为面向内容的编址方案。

当某个节点要向其他节点发送数据时,这个节点的处理器将要发送的数据和自己的标识符传送给该节点的CAN总线接口控制器,并处于准备状态;当收到总线分配时,转为发送报文状态。数据根据协议组织成一定的报文格式后发出,此时网络上的其他节点处于接收状态。处于接收状态的每个节点对接收到的报文进行检测,判断这些报文是否是发给自己的以确定是否接收。

由于CAN总线是一种面向内容的编址方案,因此很容易建立高水准的控制系统并灵活地进行配置我们可以很容易地在CAN总线上加进一些新节点而无须在硬件或软件上进行修改。

当提供的新节点是纯数据接收设备时,数据传输协议不要求独立的部分有物理目的地址。此时允许分布过程同步化。也就是说,当总线上的控制器需要测量数据时,数据可由总线上直接获得,而无需每个控制器都有自己独立的传感器。

3 CAN总线的工作特点

CAN总线的有以下三方面特点:可以多主方式工作,网络上的任意节点均可以在任意时刻主动地向网络上的其他节点发送信息,而不分主从,通信方式灵活。网络上的节点(信息)可分成不同的优先级,可以满足不同的实时要求。采用非破坏性位仲裁总线结构机制,当两个节点同时向网络上传送信息时,优先级低的节点主动停止数据发送,而优先级高的节点可不受影响地继续传输数据。

4 CAN总线协议层次结构

与前面介绍的简单总线逻辑不同,CAN是一种复杂逻辑的总线结构。从层次上可以将 CAN 总线划分为三个不同层次:

(1) 物理层

在物理层中定义实际信号的传输方法,包括位的编码和解码、位的定时和同步等内容,作用是定义不同节点之间根据电气属性如何进行位的实际传输。在物理连接上,CAN总线结构提供两个引脚:CANH和CANL,总线通过CANH和CANL之间的差分电压完成信号的位传输。在不同系统中,CAN总线的位速率不同;在系统中,CAN总线的位速率是唯一的,并且是固定,这需要对总线中的每个节点配置统一的参数。

(2) 传输层

传输层是CAN总线协议的核心。传输层负责把接收到的报文提供给对象层,以及接收来自对象层的报文。传输层负责位的定时及同步、报文分帧、仲裁、应答、错误检测和标定、故障界定。

(3) 对象层

在对象层中可以为远程数据请求以及数据传输提供服务,确定由实际要使用的传输层接收哪一个报文,并且为恢复管理和过载通知提供手段。

5 CAN总线的报文结构

CAN 总线上的报文传输由以下 4 个不同的帧类型表示和控制。

(1) 数据帧

数据帧携带数据从发送器至接收器。总线上传输的大多是这种帧。从标识符长度上,又可以把数据帧分为标准帧(11 位标识符)和扩展帧(29 位标识符)。

数据帧由 7 个不同的位场组成:帧起始、仲裁场、控制场、数据场、CRC场、应答场、帧结束。其中,数据场的长度为0~8个字节。标识符位于仲裁场中,报文接收节点通过标识符进行报文滤波。帧结构如图所示

efe810f7c383e34243e24a1b67f827bd.png

(2) 远程帧

由总线上的节点发出,用于请求其他节点发送具有同一标识符的数据帧。当某个节点需要数据时,可以发送远程帧请求另一节点发送相应数据帧。与数据帧相比,远程帧没有数据场,结构如图所示。

6854cdaded8ebdab3b8da42b865db2db.png

(3) 错误帧

任何单元,一旦检测到总线错误就发出错误帧。错误帧由两个不同的场组成,第一个场是由不同站提供的错误标志的叠加(错误标志),第二个场是错误界定符。3bda54a511792672c1ad31f8a731d74a.png

(4) 过载帧

过载帧用于在先行的和后续的数据帧(或远程帧)之间提供附加延时。过载帧包括两个场:过载标志和过载界定符。

6 CAN接口配置

在 Linux 系统中,CAN总线接口设备作为网络设备被系统进行统一管理。在控制台下,CAN总线的配置和以太网的配置使用相同的命令。

7053a7c62248a129274312c73ad95de6.png      在上面的结果中,eth0 和 eth1 设备为以太网接口,can0 设备为 CAN 总线接口。接下来使用 ip 命 令来配置 CAN 总线的位速率:

ip link set can0 type cantq 125 prop-seg 6phase-seg1 7 phase-seg2 2 sjw 1ip link set can0 type can bitrate 500000 triple-sampling onifconfig can0 up cansend can0 145#1122334455667788candump can0

当设置完成后,可以通过下面的命令查询 can0 设备的参数设置:

ip -details link show can0

7 代码  

/*1.报文发送程序*/#include#include#include#include#include#include#include#include#includeint main(){    int s,nbytes;    struct sockaddr_can addr;    struct ifreq ifr;    struct can_frame frame[2] = {{0}};    s = socket(PF_CAN, SOCK_RAW, CAN_RAW);//创建套接字    strcpy(ifr.ifr_name, "can0");    ioctl(s, SIOCGIFINDEX, &ifr);//指定can0设备    addr.can_family = AF_CAN;    addr.can_ifindex = ifr.ifr_ifindex;    bind(s, (structsockaddr*)&addr, sizeof(addr));//将套接字与can0绑定    //禁用过滤规则,本进程不接收报文,只负责发送    setsockopt(s, SOL_CAN_RAW, CAN_RAW_FILTER, NULL, 0);    //生成两个报文    frame[0].can_id = 0x11;    frame[0].can_dlc = 1;    frame[0].data[0] = 'Y';    frame[0].can_id = 0x22;    frame[0].can_dlc = 1;    frame[0].data[0] = 'N';    //循环发送两个报文    while(1)    {        nbytes = write(s,&frame[0], sizeof(frame[0]));//发送frame[0]        if(nbytes! = sizeof(frame[0]))        {            printf("Send Error frame[0]\n!");            break;//发送错误,退出        }        sleep(1);        nbytes = write(s,&frame[1], sizeof(frame[1]));//发送frame[1]        if(nbytes != sizeof(frame[0]))        {            printf("SendErrorframe[1]\n!");            break;        }        sleep(1);    }    close(s);    return0;}/*2.报文过滤接收程序*/#include#include#include#include#include#include#include#include#includeint main(){    int s,nbytes;    struct sockaddr_can addr;    struct ifreq ifr;    struct can_frame frame;    struct can_filter rfilter[1];    s = socket(PF_CAN, SOCK_RAW, CAN_RAW);//创建套接字    strcpy(ifr.ifr_name, "can0");    ioctl(s, SIOCGIFINDEX, &ifr);//指定can0设备    addr.can_family = AF_CAN;    addr.can_ifindex = ifr.ifr_ifindex;    bind(s,(struct sockaddr*)&addr, sizeof(addr));//将套接字与can0绑定    //定义接收规则,只接收表示符等于0x11的报文    rfilter[0].can_id=0x11;    rfilter[0].can_mask = CAN_SFF_MASK;    //设置过滤规则    setsockopt(s, SOL_CAN_RAW, CAN_RAW_FILTER, &rfilter, sizeof(rfilter));    while(1)    {        nbytes = read(s, &frame, sizeof(frame));//接收报文        //显示报文        if(nbytes>0)        {            printf("ID=0x%XDLC=%ddata[0]=0x%X\n",frame.can_id, frame.can_dlc,frame.data[0]);        }    }    close(s);    return 0;}
声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/小小林熬夜学编程/article/detail/582336
推荐阅读
相关标签
  

闽ICP备14008679号