基于CANopenNode协议栈在正点原子探索者开发板上进行移植

本文介绍了CANopen协议栈在STM32平台上的使用,包括基于CUBEMX的配置、定时器设置、CANopenNode的集成以及简单通信测试,如心跳包发送和接收。还详细解释了CANID映射和协议内容解析示例。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

参考资源

CANopen协议栈源码:

https://github.com/CANopenNode/CANopenNode

CANopen协议栈基于STM32平台开发的demo,不过这里是基于STM32CUBEIDE开发的:

https://github.com/CANopenNode/CanOpenSTM32

协议栈的配置工具:

https://github.com/CANopenNode/CANopenEditor

程序源码在这里

CUBEMX配置

正点原子的开发板引出的是CAN1接口
在这里插入图片描述
CAN1接口的波特率125k:
在这里插入图片描述
定时器timer14的配置:
在这里插入图片描述
配置好之后进行采用keil IDE进行代码生成即可,还需要进行CANopenNode的代码集成,并参照demo进行代码移植。
集成的代码主要包括CANopenNode和CANopenNode_stm32两个文件夹内:
在这里插入图片描述
在这里插入图片描述
相关代码的移植主要是main函数:

/**
  * @brief  The application entry point.
  * @retval int
  */
int main(void)
{
  /* USER CODE BEGIN 1 */

  /* USER CODE END 1 */

  /* MCU Configuration--------------------------------------------------------*/

  /* Reset of all peripherals, Initializes the Flash interface and the Systick. */
  HAL_Init();

  /* USER CODE BEGIN Init */

  /* USER CODE END Init */

  /* Configure the system clock */
  SystemClock_Config();

  /* USER CODE BEGIN SysInit */

  /* USER CODE END SysInit */

  /* Initialize all configured peripherals */
  MX_GPIO_Init();
  MX_CAN1_Init();
  MX_TIM14_Init();
  MX_USART1_UART_Init();
  /* USER CODE BEGIN 2 */
  CANopenNodeSTM32 canOpenNodeSTM32;
  canOpenNodeSTM32.CANHandle = &hcan1;
  canOpenNodeSTM32.HWInitFunction = MX_CAN1_Init;
  canOpenNodeSTM32.timerHandle = &htim14;
  canOpenNodeSTM32.desiredNodeID = 5;
  canOpenNodeSTM32.baudrate = 125;
  canopen_app_init(&canOpenNodeSTM32);
  /* USER CODE END 2 */

  /* Infinite loop */
  /* USER CODE BEGIN WHILE */
  while (1)
  {
    /* USER CODE END WHILE */

    /* USER CODE BEGIN 3 */
      
    HAL_GPIO_WritePin(LD3_GPIO_Port, LD3_Pin, !canOpenNodeSTM32.outStatusLEDGreen);
	HAL_GPIO_WritePin(LD4_GPIO_Port, LD4_Pin, !canOpenNodeSTM32.outStatusLEDRed);
    canopen_app_process();
      
  }
  /* USER CODE END 3 */
}

定时器的回调函数:

void HAL_TIM_PeriodElapsedCallback(TIM_HandleTypeDef* htim) {
    if (htim == canopenNodeSTM32->timerHandle) {
        canopen_app_interrupt();
    }
}

协议栈的配置

进行简单的通信测试,最简单的测试就是心跳包测试,采用CANopenEditor配置进行心跳包的配置,设备作为从机,配置内容为下图:
在这里插入图片描述
这里的心跳时间间隔设置为100ms。

协议栈的测试

程序编译后下载到开发板中,心跳包报文如下:
在这里插入图片描述
可以采用SDO方式更改心跳包为0,发送及反馈报文如下:
在这里插入图片描述
则将心跳包周期设置为0,心跳包停止发送。
可以采用SDO方式更改心跳包为1000ms,发送及反馈报文如下:
在这里插入图片描述
发送完成后心跳包以1s的周期重新开始发送。

协议内容解析

其中CANID的映射关系如下图所示:
在这里插入图片描述
客户端向服务器发送指令的CAN报文ID为 600+节点ID,服务器向客户端响应的指令为580+节点ID。特别需要注意的是,这里的客户端指的是CAN采集设备,这里用的是周立功的USBCANFD_200U,服务器指的是正点原子的开发板。
在这里插入图片描述
协议分析如下,对0x05节点,的0x1017配置为0x0000
客户端(USBCANFD_200U)通知服务器(开发板)需要写2字节,索引为0x1017,子索引为0x00,数据为0x0000
服务器(开发板)反馈客户端(USBCANFD_200U),写入成功,索引为0x1017,子索引0x00。

### 将CANOpenNode移植到RT-Thread操作系统 #### 移植准备 为了成功将CANOpenNode移植至RT-Thread环境,需先理解两者特性并完成必要的前期准备工作。确保拥有最新版本的RT-Thread以及目标硬件支持包;同时获取CANOpenNode源码及其依赖项[^1]。 #### 配置开发环境 安装适用于所选平台的编译工具链,并配置好用于调试的目标板连接方式。对于大多数嵌入式项目而言,这通常意味着设置交叉编译器路径变量、下载对应的固件烧录软件等前置作业。 #### 修改CANOpenNode代码适配层 由于不同RTOS间API存在差异,在此阶段主要工作集中在调整底层驱动接口上。具体来说就是重写`can.c/h`文件中的函数实现部分,使其能够调用RT-Thread提供的相应服务来管理CAN控制器资源。例如: ```c // 原始定义可能如下所示 (来自 CANOpenNode ) void canWriteMsgBuf(uint8_t id, uint8_t ext, uint8_t rtr, uint8_t len, const uint8_t *data); // 调整后的版本应利用 RT-Thread 的 API 进行改写 static rt_err_t can_write(const struct device *dev, int wait_ms, void *buf, size_t size); ``` 上述例子展示了如何把标准C语言风格的过程抽象转换成符合RT-Thread设备模型的方法签名形式。 #### 整合CO_OD应用对象字典 考虑到实际应用场景的需求多样性,还需针对特定产品定制化OD(Object Dictionary),即通过编辑XML描述文档指定节点属性参数值范围及默认设定等内容。这部分改动不会直接影响到底层通信机制层面的工作逻辑,但却关系着高层协议交互能否顺利达成预期效果的关键所在。 #### 测试验证与优化改进 最后一步便是全面测试整个系统的功能表现情况,包括但不限于基本收发消息准确性检验、异常处理流程模拟演练等方面。基于初步运行反馈信息进一步微调性能瓶颈处直至满足工业级可靠性指标为止。
评论 11
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

三环西北角

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值