i.MX6Q -- Linux系统下FlexCAN总线的使用记录

开发板平台:珠海鼎芯D338 – 基于i.MX6Q
系统平台:Linux3.0.35 + Busybox1.22

1、 需要的工具
FlexCAN工具包:
libsocketcan-0.0.10.tar.bz2
canutils-4.0.6.tar.bz2
交叉编译器:
fsl-linaro-toolchain.tar.bz2

2、编译过程:
交叉编译器:
环境变量配置命令:
exportPATH=<YourCompilerPath>/fsllinarotoolchain/bin/: PATH

   libsocketcan-0.0.10:

使用tar命令解压libsocketcan-0.0.10.tar.bz2后,使用如下命令交叉编译并安装。
./configureprefix=/flexcan/installhost=armnonelinuxgnueabi make && make install

   canutils-4.0.6:

同样,交叉编译并安装命令如下。
./configureprefix=/flexcan/installhost=armnonelinuxgnueabiCC=armnonelinuxgnueabigcclibsocketcanCFLAGS=I//flexcan/install/includelibsocketcanLIBS=lsocketcanLDFLAGS=L//flexcan/install/libenablestatic make && make install

   移植拷贝安装目录下(<YourInstallPath>/flexcan/install)的所有文件到D338开发板/usr/目录下即可;

至此,FlexCAN工具的移植就完成了,直接将文件系统打包为rootfs.tar.bz2,然后拷贝到dchip_MFGTools_Linux3.0.35_MX6Q_UPDATER工具下进行烧写启动就可以使用了,接下来,主要是FlexCAN总线的使用方法及详细说明。

3、错误与解决:
错误1:
编译canutils错误
canconfig.c:34:26: fatal error: libsocketcan.h: No such fileor directory
解决1:
将libsocketcan已编译安装目录(/flexcan/install)下的include/can_netlink.h和libsocketcan.h文件,拷贝到canutils4.0.6/include目录下

4、测试过程:
准备:两个板子CAN总线进行相应管脚对接;

   开发板命令:

设置比特率: iplinksetcan0uptypecanbitrate125000使can0线 ifconfig can0 up
查看can0信息: $ ifconfig can0

发送数据到0x123地址:
$ cansend can0 -i 0x123 0x11 0x22 0x33 0x44 0x55 0x66 0x77 0x88

接收0x123地址数据:
canechocan0v candump can0 –filter 0x123:/未设置ID滤波 /
或$ candump can0 –filter 0x123: 0x7FF/设置使用ID滤波 /

说明:
接收时,未使用ID滤波,可以接收总线上的所有数据,无论ID是否正确;
接收时,使用ID滤波,就只有符合要求的ID的数据,才做接收,其他ID的数据均丢弃不管。

使用以上测试方法测试时,发现filter id最大可用值为0x7FF;
ID = 0x7FF = 1111111 1111 = 11bit地址,说明当前使用的是标准格式;
如果想使用更多地址,需要设置为扩展帧格式。

如何设置CAN总线数据帧为扩展帧格式呢?

经过查看CAN驱动源码发现:
在d-chip-linux3.0.35/drivers/net/can/flexcan.c中,有如下开始传输函数:
static int flexcan_start_xmit(struct sk_buff* skb,struct net_device *dev)
{
if (cf->can_id & CAN_EFF_FLAG) {
can_id = cf->can_id & CAN_EFF_MASK;
ctrl |= FLEXCAN_MB_CNT_IDE | FLEXCAN_MB_CNT_SRR;
} else {
can_id = (cf->can_id & CAN_SFF_MASK) << 18;
}

if (cf->can_id & CAN_RTR_FLAG)
ctrl |= FLEXCAN_MB_CNT_RTR;

if (cf->can_dlc > 0) {
u32 data = be32_to_cpup((__be32 *)&cf->data[0]);
writel(data,&regs->cantxfg[FLEXCAN_TX_BUF_ID].data[0]);
}
if (cf->can_dlc > 3) {
u32 data = be32_to_cpup((__be32 *)&cf->data[4]);
writel(data,&regs->cantxfg[FLEXCAN_TX_BUF_ID].data[1]);
}
}
查看以上源码发现:
由cf->can_id & CAN_EFF_FLAG 按位’与’的运算结果来判断当前是否是扩展帧格式,cf->can_id 是命令行设置的Filter ID, CAN_EFF_FLAG宏定义为0x80000000;因此,当命令行下设置ID时,使其最高位( id[31] )为1,即表示选择为扩展帧格式;0表示选择为标准帧格式。
而cf->can_id & CAN_RTR_FLAG用来判断是否设置远程帧请求;

以下是源码中的结构说明:
/*
*Controller Area Network Identifier structure
*
*bit 0-28 : CAN identifier (11/29 bit)
*bit 29 : error frame flag (0 = dataframe, 1 = error frame)
*bit 30 : remote transmission requestflag (1 = rtr frame)
*bit 31 : frame format flag (0 =standard 11 bit, 1 = extended 29 bit)
*/
/**
*struct can_frame - basic CAN frame structure
*@can_id: the CAN ID of the frame andCAN_*_FLAG flags, see above.
*@can_dlc: the data length field of the CAN frame
*@data: the CAN frame payload.
*/
struct can_frame {
canid_t can_id; /* 32bit CAN_ID + EFF/RTR/ERR flags */
__u8 can_dlc; /* datalength code: 0 .. 8 */
u8 data[8] __attribute((aligned(8))); /* 设定为8字节对齐 */
};

根据以上研究表明,想使用扩展帧格式,可按如下方法使用:
ifconfigcan0down/CANCAN/ ip link set can0 up type can bitrate 125000 /* 配置比特率并打开 */
ifconfigcan0/CAN/ cansend can0 -i 0x81234567 0x11 0x22 0x33 0x440x55 0x66 0x77 0x88 /* 发送8bytes数据到ID=0x1234567 */

canechocan0v candump can0 –filter 0x81234567: 或
$ candump can0 –filter 0x81234567:0x1FFFFFFF
最大ID地址为0x1FFFFFFF ,即:–filter 0x9FFFFFFF:

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值