当前位置:   article > 正文

STM32 CAN通讯实验程序_stm32的can通信发送与接收实例

stm32的can通信发送与接收实例

目录

STM32 CAN通讯实验

CAN硬件原理图

CAN外设原理图

TJA1050T硬件描述

实验线路图

回环实验

CAN头文件配置

 CAN_GPIO_Config初始化

CAN初始化结构体

CAN筛选器结构体

 接收中断优先级配置

接收中断函数

main文件

实验现象

补充


STM32 CAN通讯实验

CAN硬件原理图

CAN外设原理图

野火STM32F103ZET6霸道板载原理图

我们的开发板没有使用GPIO外设的复用功能PA11和PA12,而使用了重定义(重映射)功能PB8和PB9

TJA1050T硬件描述

实验线路图

图中为两个霸道开发板,如果使用指南针开发板需要外接CAN收发器和电阻。

        是否使用RX和TX引脚是根据实际情况来确认是否使用的。如果我们使用回环模式时,在STM32芯片内部的CAN控制器的发送端和接收端就已经通过硬件逻辑连接起来了,比如回环静默模式,根本不用使用STM32的发送和接收引脚。甚至使用回环测试的时候,CAN收发器就算不供电都可以工作。

回环实验

我们配置外设的GPIO功能时,可以参考手册的外设GPIO功能配置

CAN头文件配置

  1. #ifndef __BSP_CAN_H
  2. #define __BSP_CAN_H
  3. #include "stm32f10x.h"
  4. #define PASS_ID ((uint32_t)0x1314)
  5. #define CAN_TX_GPIO_PROT GPIOB
  6. #define CAN_TX_GPIO_PIN GPIO_Pin_9
  7. #define CAN_RX_GPIO_PORT GPIOB
  8. #define CAN_RX_GPIO_PIN GPIO_Pin_8
  9. #define CAN_GPIO_CLK RCC_APB2Periph_GPIOB
  10. /*信息输出*/
  11. #define CAN_DEBUG_ON 1
  12. #define CAN_INFO(fmt,arg...) printf("<<-CAN-INFO->> "fmt"\n",##arg)
  13. #define CAN_ERROR(fmt,arg...) printf("<<-CAN-ERROR->> "fmt"\n",##arg)
  14. #define CAN_DEBUG(fmt,arg...) do{\
  15. if(CAN_DEBUG_ON)\
  16. printf("<<-CAN-DEBUG->> [%d]"fmt"\n",__LINE__, ##arg);
  17. void CAN_Config(void) ;
  18. #endif /* __BSP_CAN_H */

 CAN_GPIO_Config初始化

  1. void CAN_GPIO_Config(void)
  2. {
  3. GPIO_InitTypeDef GPIO_InitStructure;
  4. /* 使能CAN时钟 */
  5. RCC_APB1PeriphClockCmd (RCC_APB1Periph_CAN1 , ENABLE );
  6. /* 使能CAN引脚相关的时钟 */
  7. RCC_APB2PeriphClockCmd ( CAN_GPIO_CLK|RCC_APB2Periph_AFIO, ENABLE );
  8. //使用PA8 9引脚的第二功能
  9. GPIO_PinRemapConfig (GPIO_Remap1_CAN1 ,ENABLE);
  10. /* 配置CAN的 引脚,普通IO即可 */
  11. GPIO_InitStructure.GPIO_Pin = CAN_TX_GPIO_PIN;
  12. GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  13. GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
  14. GPIO_Init(CAN_TX_GPIO_PROT, &GPIO_InitStructure);
  15. /* 配置CAN的 引脚,普通IO即可 */
  16. GPIO_InitStructure.GPIO_Pin = CAN_RX_GPIO_PIN;
  17. GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  18. GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;
  19. GPIO_Init(CAN_RX_GPIO_PORT, &GPIO_InitStructure);
  20. }

PB8和PB9使用重映射功能

我们要使用GPIO_PinRemapConfig函数来使能PA8 9引脚的第二功能,此外一定要记得开启相应的AFIO时钟,否则第二功能是无法使用的。

CAN初始化结构体

  1. void CAN_Mode_Config(void)
  2. {
  3. CAN_InitTypeDef CAN_InitTypeStruct;
  4. CAN_InitTypeStruct.CAN_ABOM = ENABLE;
  5. CAN_InitTypeStruct.CAN_AWUM = ENABLE;
  6. CAN_InitTypeStruct.CAN_Mode = CAN_Mode_LoopBack;//CAN_Mode_Normal;//调试时建议使用回环模式,调试完再改成NORMAL
  7. CAN_InitTypeStruct.CAN_NART = ENABLE; //错误重传
  8. CAN_InitTypeStruct.CAN_RFLM = ENABLE;
  9. CAN_InitTypeStruct.CAN_TTCM = DISABLE;
  10. CAN_InitTypeStruct.CAN_TXFP = DISABLE; //按ID优先级发送
  11. //配置成1Mbps
  12. CAN_InitTypeStruct.CAN_BS1 = CAN_BS1_5tq;
  13. CAN_InitTypeStruct.CAN_BS2 = CAN_BS2_3tq;
  14. CAN_InitTypeStruct.CAN_SJW = CAN_SJW_2tq;
  15. CAN_InitTypeStruct.CAN_Prescaler = 4;
  16. CAN_Init(CAN1,&CAN_InitTypeStruct);
  17. }

其中位时序及波特率按照下表配置

CAN筛选器结构体

  1. void CAN_Filter_Config(void)
  2. {
  3. CAN_FilterInitTypeDef CAN_FilterInitTypeStruct;
  4. CAN_FilterInitTypeStruct.CAN_FilterActivation = ENABLE;
  5. CAN_FilterInitTypeStruct.CAN_FilterFIFOAssignment = CAN_Filter_FIFO0 ;
  6. CAN_FilterInitTypeStruct.CAN_FilterNumber = 0;
  7. CAN_FilterInitTypeStruct.CAN_FilterScale = CAN_FilterScale_32bit;
  8. CAN_FilterInitTypeStruct.CAN_FilterMode = CAN_FilterMode_IdMask ;
  9. CAN_FilterInitTypeStruct.CAN_FilterIdHigh = ((PASS_ID<<3 |CAN_Id_Extended |CAN_RTR_Data)&0xFFFF0000)>>16;
  10. CAN_FilterInitTypeStruct.CAN_FilterIdLow = ((PASS_ID<<3 |CAN_Id_Extended |CAN_RTR_Data)&0xFFFF);
  11. CAN_FilterInitTypeStruct.CAN_FilterMaskIdHigh = 0xFFFF;
  12. CAN_FilterInitTypeStruct.CAN_FilterMaskIdLow =0xFFFF;
  13. CAN_FilterInit(&CAN_FilterInitTypeStruct);
  14. CAN_ITConfig (CAN1,CAN_IT_FMP0,ENABLE);
  15. }

其中假如我们要过滤的ID为0x1314

使用stm32f10x_can.h文件末尾定义的相关宏

由于使用的是32位筛选器且标识符掩码,所以其中CAN_FilterIdHigh和CAN_FilterIdLow为我们过滤ID格式的高16位和低16位,首先将ID号左移三位,然后或上IDE、RTR位

  1. CAN_FilterInitTypeStruct.CAN_FilterIdHigh = ((PASS_ID<<3 |CAN_Id_Extended |CAN_RTR_Data)&0xFFFF0000)>>16;
  2. CAN_FilterInitTypeStruct.CAN_FilterIdLow = ((PASS_ID<<3 |CAN_Id_Extended |CAN_RTR_Data)&0xFFFF);

而CAN_FilterMaskIdHigh和CAN_FilterMaskIdLow为要过滤的ID掩码,全为1,表示完全过滤

  1. CAN_FilterInitTypeStruct.CAN_FilterMaskIdHigh = 0xFFFF;
  2. CAN_FilterInitTypeStruct.CAN_FilterMaskIdLow =0xFFFF;

 接收中断优先级配置

  1. void CAN_NVIC_Config(void)
  2. {
  3. NVIC_InitTypeDef NVIC_InitStructure;
  4. /* 配置NVIC为优先级组1 */
  5. NVIC_PriorityGroupConfig(NVIC_PriorityGroup_1);
  6. NVIC_InitStructure.NVIC_IRQChannel = USB_LP_CAN1_RX0_IRQn;
  7. /* 配置抢占优先级 */
  8. NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1;
  9. /* 配置子优先级 */
  10. NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1;
  11. /* 使能中断通道 */
  12. NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
  13. NVIC_Init(&NVIC_InitStructure);
  14. }

在标准库头文件中找到CAN接收数据中断源,我们使用的是接收邮箱0,所以选择USB_LP_CAN1_RX0_IRQn

使能中断放在CAN筛选器结构体配置中

接收中断函数

  1. extern CanRxMsg CAN_Rece_Data;
  2. extern uint8_t flag;
  3. void USB_LP_CAN1_RX0_IRQHandler(void)
  4. {
  5. CAN_Receive(CAN1,CAN_FIFO0, &CAN_Rece_Data);
  6. flag = 1;
  7. }

要注意的是在CAN里,我们设置完CAN_Receive(CAN1,CAN_FIFO0, &CAN_Rece_Data);之后不需要手动清除接收标志,该函数会自动清除。

flag用于在main函数中判断是否接收到数据,从而做相应的应用处理

我们也可以再加个判断进一步确认接收数据的准确性

main文件

  1. #include "stm32f10x.h"
  2. #include "./usart/bsp_usart.h"
  3. #include "./led/bsp_led.h"
  4. #include "./can/bsp_can.h"
  5. #include "./key/bsp_key.h"
  6. CanRxMsg CAN_Rece_Data;
  7. CanTxMsg CAN_Tran_Data;
  8. uint8_t flag = 0;
  9. void Delay(__IO uint32_t nCount);
  10. /*
  11. * 函数名:main
  12. * 描述 :主函数
  13. * 输入 :无
  14. * 输出 :无
  15. */
  16. int main(void)
  17. {
  18. LED_GPIO_Config();
  19. LED_BLUE;
  20. /* 配置串口为:115200 8-N-1 */
  21. USART_Config();
  22. printf("\r\n 这是一个CAN通讯实验 \r\n");
  23. CAN_Config() ;
  24. Key_GPIO_Config();
  25. printf("\r\n 按KEY1按键发送数据\r\n");
  26. while(1)
  27. {
  28. if( Key_Scan(KEY1_GPIO_PORT,KEY1_GPIO_PIN) == KEY_ON )
  29. {
  30. uint8_t box;
  31. CAN_Tran_Data.StdId = 0;
  32. CAN_Tran_Data.ExtId = PASS_ID;
  33. CAN_Tran_Data.RTR = CAN_RTR_Data;//使用数据帧
  34. CAN_Tran_Data.IDE = CAN_Id_Extended ; //使用扩展帧
  35. CAN_Tran_Data.DLC = 1;
  36. CAN_Tran_Data.Data[0] = 10;
  37. box = CAN_Transmit(CAN1,&CAN_Tran_Data);
  38. while(CAN_TransmitStatus(CAN1,box) == CAN_TxStatus_Failed);
  39. printf("\r\n 数据包发送完成\r\n");
  40. }
  41. if(flag == 1)
  42. {
  43. printf("\r\n接收到的数据:%d\r\n",CAN_Rece_Data.Data[0]);
  44. flag = 0;
  45. }
  46. else
  47. {
  48. }
  49. }
  50. }
  51. void Delay(__IO uint32_t nCount)
  52. {
  53. for(; nCount != 0; nCount--);
  54. }

实验现象

补充

由于只有一个板子,无法演示双机实验,双机实验只需要将我们的回环实验中的回环模式换成正常模式,然后将程序分别下载到两个开发板即可。

学完基础的CAN通讯协议之后,如果想要今后从事CAN通讯相关工作,比如工业、汽车领域,我们还需要进一步学习一下CAN OPEN。

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/2023面试高手/article/detail/489390
推荐阅读
相关标签
  

闽ICP备14008679号