can总线用java怎么解析_CAN信号值解析

本文提供一种可以解析CAN信号各信号值的一种方法并进行说明。

一般情况下,高端一点的设备会计算每一个信号的值,但是接受到CAN信号的报文实际上有各种情况,如何通过设定的起始位和数据长度来获取某一信号的值,详情请参考,本文方法比较笨拙,如有更好的办法,欢迎指教。

假设,我们获取到的数据长度为8byte,即DLC为8。CAN Matrix表格如下:

94972c475f51836da2c90d544355da4b.png

我们约定,一种表示信号的方式如下:

1bd89cd37b6dbd656dec36657f8de701.png

lsb起始位

数据长度

按照上面的图来说,该信号起始位为40,长度为10。所以,该信号表示值的范围在0-1023之间,至于具体表示的物理量的含义需要有其他的文件规定。

获取到一帧数据以后,如何通过一系列的运算来得到信号值呢?

这里提供一种思路:

根据起始位和长度信息确定信息跨越的byte位

根据起始位确定LSB所在的bit位

根据跨越的byte位组合为数据

把以上组合的数据进行位与,得到信号值。

大概分为以上的4部分,以下分别说明:

设定:lsbbit,lsbbyte,msbbyte,start,length.按照以上的例子就是:

lsbbit  = 0

lsbbyte = 5

msbbyte = 4

start = 40

length = 10

计算lsbbit:

lsbbit = start & 7

计算lsbbyte:

lsbbyte = start >> 3

计算msbbyte:

由于数据是向byte减少的方向进行的。所以:

msbbyte = lsbbyte - x

x为跨越的位数,也就是:lsbbit + length - 1所占的长度,如果大于7说明为另一行,即:

msbbyte = lsbbyte - ((lsbbit + length - 1) >> 3)

组合data:

我们知道数据介于msbbyte-lsbbyte之间,那么我们可以组合这两个数据:

for(index = msbbyte -> (lsbbyte+1)):

data_merge  += data[index] << (( lsbbyte - index ) << 3)

获取数据value:

去尾:value = data_merge >> lsbbit

按位与:value = value & ((1 << length) - 1)

所以,最终得到的结果为筛选出来的信号值,然后根据该信号值关联上实际的物理值,就可以得到具体的物理数据了。

代码实现的方式有很多,大体思路为此。

很久以前写的,今天用的时候居然发现了一个bug,所以记录下来,表示注意。

  • 1
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
CAN(Controller Area Network)总线协议是一种串行通信协议,广泛应用于汽车电子、工业自动化、医疗设备等领域。在CAN总线中,每个节点都可以发送和接收消息,而这些消息都是通过CAN总线传输的。因此,为了能够在嵌入式系统中使用CAN总线,需要编写相应的驱动程序对总线信号进行解析。 下面是一个基于CAN通信协议解析总线信号的驱动程序的示例: ```c #include <linux/can.h> #include <linux/can/raw.h> #include <linux/netdevice.h> struct can_filter filter[1]; int main(void) { int s; struct sockaddr_can addr; struct ifreq ifr; struct can_frame frame; /* 创建socket */ s = socket(PF_CAN, SOCK_RAW, CAN_RAW); if (s < 0) { perror("socket"); return 1; } /* 设置CAN设备 */ strcpy(ifr.ifr_name, "can0"); ioctl(s, SIOCGIFINDEX, &ifr); addr.can_family = AF_CAN; addr.can_ifindex = ifr.ifr_ifindex; bind(s, (struct sockaddr *)&addr, sizeof(addr)); /* 设置CAN过滤器 */ filter[0].can_id = 0x123; filter[0].can_mask = CAN_SFF_MASK; setsockopt(s, SOL_CAN_RAW, CAN_RAW_FILTER, &filter, sizeof(filter)); /* 接收CAN消息 */ while (1) { int nbytes = read(s, &frame, sizeof(struct can_frame)); if (nbytes < 0) { perror("read"); return 1; } printf("ID: 0x%x, DLC: %d, Data: %s\n", frame.can_id, frame.can_dlc, frame.data); } close(s); return 0; } ``` 该驱动程序使用Linux的CAN socket API实现了对CAN总线读取解析。首先,通过调用socket函数创建一个CAN socket。然后,通过ioctl函数获取CAN设备的索引,并将CAN socket与该设备绑定。接着,通过setsockopt函数设置CAN过滤器,只接收ID为0x123的CAN消息。最后,通过read函数从CAN总线读取消息,并解析出ID、数据长度和数据内容。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值