CANopen开源库canfestival的移植

news2025/1/23 10:25:04

本文记录将CANopen开源库CANfestival移植到GD32F470单片机的过程。CANopen协议理解请参考博客:CANopen协议的理解-CSDN博客

CANfestival开源库下载链接

CSDN链接: https://download.csdn.net/download/heqiunong/89774627

官网链接:https://hg.beremiz.org/canfestival/file/de1fc3261f21

objdictedit字典工具下载链接

https://download.csdn.net/download/heqiunong/89774674

https://download.csdn.net/download/supcool/12492303

视频参考链接

1 canfestival移植_哔哩哔哩_bilibili

移植正文

上述各种链接都是些准备工作,主要有两个东西需要下载,一个是CANfestival库,一个是objdictedit对象字典工具。

1、CANfestival库文件预处理

下载好了CANfestval库后,里面有些文件是多余的,需要删除

1.1 src文件夹下的文件删除,src重命名为source

左边红色框住的部分删除掉

1.2 include文件夹下的文件删除,AVR文件夹重命为gd32

 

1.3 gd32文件夹(原AVR文件夹)下的文件删除

2、将CANfestival库放入工程文件当中

先把预处理的CANfestivel库拷贝到keil的工程文件夹当中,具体拷到哪里由自己定。

2.1 将所有源文件添加进入项目

 

2.2 添加include路径

3、处理编译的报错

3.1 注释config.h文件中的部分内容

3.2 缺函数的问题

完成3.1后,编译工程会缺函数

 

3.3 处理前两个缺函数的报错

start_and_seek_node 和 start_node这两个函数,是有的,只是因为inline关键字没有被识别,去掉inline就可以了。

全局搜索一下,找到位置。(这两个函数定义是在dcf.c文件中)

 

 删除这两个inline,现在就只缺下面3个函数了。

3.4 添加canSend函数以及CAN接收的处理

在添加canSend函数之前,我们需要对GD32F470的CAN进行外设层的配置,这部分配置完最好拿个USB-to-CAN的工具验证一下配置成功没有。 这部分工作可GD32F470提供的参考例程,本文的重点不在这里。

假设我们已经把CAN的外设部分都配置好了,下面来添加canSend函数。

#include "canfestival.h"

// This function is called by CANopen library
uint8_t canSend(CAN_PORT notused, Message *message)
{
    uint8_t transmit_mailbox = 0;
    uint32_t timeout = 0xFFFF;
    
    transmitMessage.tx_dlen = message->len;
    
    memcpy(transmitMessage.tx_data, message->data, message->len);
    transmitMessage.tx_ff = CAN_FF_STANDARD;
    
    transmitMessage.tx_sfid = message->cob_id;
    // Check here if an accident occurs
    transmitMessage.tx_ft = (message->rtr == 0) ? CAN_FT_DATA : CAN_FT_REMOTE;
            
    // Transmit message
    transmit_mailbox = can_message_transmit(CAN0, &transmitMessage);
    
    // Waiting for transmit completed
    timeout = 0xFFFF;
    while((CAN_TRANSMIT_OK != can_transmit_states(CAN0, transmit_mailbox)) && (0 != timeout)){
        timeout--;
    }
    return (timeout!=0) ? 0:1;
}

Message这个结构体是canfestival的库里面定义的,所以这里需要包含canfestival.h。我们需要做的是理解Message结构体里面的内容,然后把信息通过gd32的CAN对应的外设函数把Message发送出去。

★那么同样的道理,当CAN在接收外部发来的信息的时候,我们也要把接收到的信息,按照Message的格式,存到Message里面去。gd32的CAN接收是用中断来做的,下面给出代码参考一下。

void CAN0_RX0_IRQHandler(void)
{

    // For CANopen communication   
    Message Rx_Message;
    can_message_receive(CAN0, CAN_FIFO0, &receiveMessage);
    Rx_Message.cob_id = receiveMessage.rx_sfid;
    Rx_Message.rtr = (receiveMessage.rx_ft == CAN_FT_DATA) ? 0:1;  // be careful
    Rx_Message.len =receiveMessage.rx_dlen;
    memcpy(Rx_Message.data, receiveMessage.rx_data, receiveMessage.rx_dlen);
    
    // TODO we need objdictedit
    // canDispatch(&GD32Master_Data,&Rx_Message);
    
}

 

注意函数最后一行的// canDispatch(&GD32Master_Data,&Rx_Message);是需要添加完对象字典后,需要解开注释的,

可以从上述函数分析,CAN接收中断函数把接收到的信息,存到了一个Message类型的结构体变量里面,最后调用canDispatch(&GD32Master_Data,&Rx_Message)函数,把接收到的信息和对象字典两个变量传进canfestival库进行处理。

那么搞完3.4这一步,就只剩下两个错误了。

 

3.5 添加getElapsedTime和setTimer函数

canfestival库的运行是需要一个定时器的,这个定时器需要由单片机给它提供,因此我们需要配置一个gd32的定时器给canfestival库。关于GD32的定时器配置内容,不是本文的重点,这里直接给出代码供参考。

static void CanOpenTimerConfig(void)
{
    timer_parameter_struct initPara;
    
    rcu_periph_clock_enable(RCU_TIMER2);    // Timer 0 1 3 4 7 has been used for other purposes 
                                            // TIMER2_CLK = 240MHz
    timer_deinit(TIMER2);
                                            // CANopen requires a 10us timer, which is 100kHz
    initPara.prescaler  = 240 - 1;               // 240MHz -> 100kHz
    initPara.period     = TIMEVAL_MAX - 1;         
    initPara.alignedmode       = TIMER_COUNTER_EDGE;
    initPara.counterdirection  = TIMER_COUNTER_UP;
    initPara.clockdivision     = TIMER_CKDIV_DIV1;  
    initPara.repetitioncounter = 0;
    timer_init(TIMER2, &initPara);
    
    timer_auto_reload_shadow_enable(TIMER2);        // Auto-reload preload enable
    timer_flag_clear(TIMER2, TIMER_FLAG_UP);
    timer_interrupt_enable(TIMER2, TIMER_INT_UP);   // Enable count up interrupt
    
    nvic_irq_enable(TIMER2_IRQn, 1U, 1U);           // Enable and set timer interrupt priority
    timer_enable(TIMER2);
}

// This function is called by CANopen library
void setTimer(TIMEVAL value)  
{  
    TIMER_CAR(TIMER2) =  value;  
}

// This function is called by CANopen library
TIMEVAL getElapsedTime(void)  
{  
    return TIMER_CNT(TIMER2);  
}

// TIMER2 is assigned as a CANopen timer 
void TIMER2_IRQHandler(void)  
{  
     if(SET == timer_interrupt_flag_get(TIMER2,TIMER_INT_UP))
     {  
         TimeDispatch();  
         timer_interrupt_flag_clear(TIMER2,TIMER_INT_UP);  
    }       
}

我们把定时器的计数频率配置成了1MHz。计数周期这里配置成TIMEVAL_MAX - 1;

注意:canfestival库里面默认的频率是125kHz,所以canfestival库里面timerscfg.h文件几个定义需要改改

 

注意:除了报错所要求我们添加的getElapsedTime和setTimer函数以外, 定时器的计数溢出中断里面,也调用了一个canfestival库里的函数TimeDispatch(); 而且canfestival库规定,计数中断周期是2ms。所以我们在配置定时器的时候才使用TIMEVAL_MAX-1来配置的。 如何理解#define TIMEVAL_MAX 2000的意思?定时器是1MHz,2000即表示2000*(1/1MHz)= 2000us = 2ms,这个细节需要去理解和注意的。

同理,比如我是100kHz(10us计数一次)定时器呢?那TIMEVAL_MAX这里就是 200了,200*(1/100kHz)= 2000us = 2ms。 一个计数值 = 10us, 下面这两个define应该这么改。

最需要注意的就是3.4和3.5,这里很容易出问题。  

4、添加对象字典

如果前面的操作都没有问题,那么这时候编译工程是不会报错的啦,但是这时候移植是没有完成的。 我们还需要添加单片机主节点的CANopen对象字典。 这时候就要用到前面我们提到的objdictedit字典啦。Objdictedit这个工具可以保存你的配置到一个xxxxx.od文件中,在开发的过程中,你可以每次只修改一部分。然后保存到.od文件中。下一次再改呢,又把这个.od文件再打开。 我们keil工程里面,需要的不是.od文件,而是利用.od文件生成的.c和.h文件。所以我们每次修改完.od文件保存之后,同时,我们还要利用objdictedit来生成一次.c和.h文件,把更新后的.c和.h文件替换keil工程里面原来的对象字典.c和.h文件。

下面我们以配置一个1ms的同步报文为例,来举例说明objdictedit的使用过程。

 

 

 把objdictedit生成的Master.c Master.h添加进入项目中。

 将Master.c代码最底部的对象字典变量名拷贝一下

 

下面把代码贴出了,方便copy 

// user_can.c


...
#include "canfestival.h"


extern CO_Data Master_Data;

...
...
...

void CAN0_RX0_IRQHandler(void)
{

    // For CANopen communication   
    Message Rx_Message;
    can_message_receive(CAN0, CAN_FIFO0, &receiveMessage);
    Rx_Message.cob_id = receiveMessage.rx_sfid;
    Rx_Message.rtr = (receiveMessage.rx_ft == CAN_FT_DATA) ? 0:1;  // be careful
    Rx_Message.len =receiveMessage.rx_dlen;
    memcpy(Rx_Message.data, receiveMessage.rx_data, receiveMessage.rx_dlen);
    
    // ★
    canDispatch(&Master_Data,&Rx_Message);
    
}

...

另外,在main主函数这边也需要做一个CANopen的基本的初始化操作。

// main.c

...
#include "Master.h"
...

void main()
{
...
    setNodeId(&Master_Data,0x00);
    setState(&Master_Data,Initialisation);
    setState(&Master_Data,Pre_operational);
    setState(&Master_Data,Operational);

...
}

5、实验现象

温故而知新,写这篇文章的时候对3.4,3.5这部分内容理解又加深了。 CANfestival的移植相比于CANopen协议的理解还是要简单一些,后续应该会根据实际的项目,更新一些除了SYNC操作以外的其他操作,欢迎关注/阅订。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2160701.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

2024年主动降噪头戴式耳机该如何选择?四款品牌高性价比推荐

今天与大家聊一聊头戴式降噪蓝牙耳机。无论是沉浸在游戏的世界中,还是专注于观看视频课程汲取知识,它都能为我带来清晰、震撼的音质体验。对程序员来说,在嘈杂的工作环境中(比如机房里),头戴式耳机都能让我…

Linux shell编程学习笔记83:time命令——争分夺秒

0 引言 在DOS或Windows中,我们可以使用time命令来查看或修改系统时间。 但是在Linux中,time命令的功能却与DOS或Windows迥然不同。 1 time命令 的功能、帮助信息、命令格式和参数说明 1.1 time命令 的功能 在Linux,time命令的功能是测量…

鸿蒙 OS 开发单词打卡 APP 项目实战 20240922 笔记和源码分享

配套有完整的录播课, 需要的私信. 零基础入门级别, 有点前端基础都能学会. 效果截图: 代码截图: 页面完整代码: import { AnswerStatus } from ../enums/AnswerStatus import { PracticeStatus } from ../enums/PracticeStatus import { getRandomQuestions, Question …

江协科技STM32学习- P17 TIM输入捕获

🚀write in front🚀 🔎大家好,我是黄桃罐头,希望你看完之后,能对你有所帮助,不足请指正!共同学习交流 🎁欢迎各位→点赞👍 收藏⭐️ 留言📝​…

【C++篇】探寻C++ STL之美:从string类的基础到高级操作的全面解析

文章目录 C string 类详解:从入门到精通前言第一章:C 语言中的字符串 vs C string 类1.1 C 语言中的字符串1.2 C string 类的优势 第二章:string 类的构造与基础操作2.1 string 类的构造方法2.1.1 示例代码:构造字符串 2.2 string…

Kotlin 多种形式的 when 表达式(七)

导读大纲 1.0.1 在变量中捕捉 when 表达式1.0.2 对任意对象使用 when 表达式1.0.3 使用不带参数的 when 表达式 when 表达式专题系列 从枚举类引出 when 表达式 1.0.1 在变量中捕捉 when 表达式 在前面的示例中,when 表达式的评估值是color变量 它是通过调用 measureColor() …

pip的安装和使用

pip的安装和使用 1、 pip 是一个现代的,通用的 Python 包管理工具。提供了对 Python 包的查找、下载、安装、卸载的功能。便于我们对Python的资源包进行管理。 2、注:pip 已内置于 Python 3.4 和 2.7 及以上版本,其他版本需另行安装。 3、在安…

java并发工具包JUC(Java Util Concurrent)

1. 什么是JUC 1.1 JUC简介 JUC(Java Util Concurrent)是Java中的一个并发工具包,提供了一系列用于多线程编程的类和接口,旨在简化并发编程并提高其效率和可维护性。JUC库包含了许多强大的工具和机制,用于线程管理、同…

多比特AI事业部VP程伟光受邀为第四届中国项目经理大会演讲嘉宾

全国项目经理专业人士年度盛会 武汉市多比特信息科技有限公司AI事业部VP程伟光先生受邀为PMO评论主办的全国项目经理专业人士年度盛会——2024第四届中国项目经理大会演讲嘉宾,演讲议题为“AI对于项目经理工作的影响和变化解析”。大会将于10月26-27日在北京举办&am…

如何将Vue项目部署至 nginx

一、准备工作 1.确保安装了开发软件 VS Code(此处可查阅安装 VS Code教程),确保相关插件安装成功 2.安装Node.js 和创建Vue项目(此处可查阅安装创建教程) 3.成功在VS Code运行一个Vue项目(此处可查阅运行…

【LeetCode】动态规划—打家劫舍(附完整Python/C++代码)

动态规划—#198. 打家劫舍 前言题目描述基本思路1. 问题定义:2. 理解问题和递推关系:3. 解决方法:4. 进一步优化:5. 小总结: 代码实现Python3代码实现Python 代码解释C代码实现C 代码解释 总结: 前言 在这个问题中,你是一个专业的小偷,计划偷窃沿街的房…

JinDouYun性能测试工具使用方法

1.功能介绍 2. 安卓端支持安卓6及以上的版本,ios支持大部分版本 3. 可以测试游戏,视频,普通应用的性能数据,数据精准,低延迟,无侵入 4.工具下载链接 筋斗云 5.后续功能添加,高版本支持&…

网页爬虫法律与道德:探索法律边界与道德规范

目录 引言 一、网络爬虫技术概述 1.1 定义与功能 1.2 技术原理 1.3 案例分析 二、网络爬虫的法律边界 2.1 合法性要求 2.2 刑事风险 2.3 案例分析 三、网络爬虫的道德规范 3.1 尊重版权和隐私 3.2 合理使用爬虫技术 3.3 透明度和社会责任 四、技术挑战与应对策略…

[linux 驱动]块设备驱动详解与实战

目录 1 描述 2 结构体 2.1 block_device_operations 2.2 gendisk 2.3 block_device 2.4 request_queue 2.5 request 2.6 bio 3.7 blk_mq_tag_set 3.8 blk_mq_ops 3 相关函数 3.1 注册注销块设备 3.1.1 register_blkdev 3.1.2 unregister_blkdev 3.2 gendisk 结构…

SpringBoot开发——整合Hutool工具类轻松生成验证码

文章目录 1、Hutool简介2、验证码效果展示2.1 扭曲干扰验证码2.2 线条干扰验证码2.3 圆圈干扰验证码3、验证码应用场景3.1. 用户注册与身份验证3.2. 支付验证3.3. 订单与物流通知3.4. 信息安全与隐私保护3.5. 通知与提醒3.6. 其他应用场景4、Hutool工具类实现验证码生成4.1 引入…

如何使用ssm实现基于VUE的儿童教育网站的设计与实现+vue

TOC ssm676基于VUE的儿童教育网站的设计与实现vue 第一章 课题背景及研究内容 1.1 课题背景 信息数据从传统到当代,是一直在变革当中,突如其来的互联网让传统的信息管理看到了革命性的曙光,因为传统信息管理从时效性,还是安全…

API公共开放平台设计

背景 随着业务发展未来会有更多的三方应用接入公司平台,目前为服务商定制的机制无法满足三方应用快速接入,所以需要一个更加通用的解决方案,开放平台势在必行。 目标 设计一套通用协议,可以支持其他应用快速接入。 说明 本方案旨在设计整体架构,以及对为何这样设计做…

React 理解 re-render 的作用、概念,并提供详细的例子解释

一、什么是 re-render 在 React 中 re-render(重新渲染) 是经常发生的行为,主要确保视图要时刻保持最新的数据来呈现。 但每次发生 re-render 也是有代价的,比如数据状态、focus 焦点、表单数据、都得重置, 遇到代码…

获取商品销量详情API:深入解析返回值,助力电商决策

在电商行业,了解商品的销量详情对于商家制定营销策略、优化库存管理和提升用户体验至关重要。通过调用获取商品销量详情的API接口,商家可以实时获取关键的销售数据,从而做出更加明智的决策。本文将深入解析获取商品销量详情API的返回值&#…

linux信号| 学习信号三步走 | 学习信号需要打通哪些知识脉络?

前言: 本节内容主要讲解linux下信号的预备知识以及信号的概念, 信号部分我们将会分为几个阶段进行讲解:信号的概念, 信号的产生, 信号的保存。本节主要讲解信号 ps:本节内容适合学习了进程相关概念的友友们进行观看哦 目录 什么是…