你的浏览器版本过低,可能导致网站不能正常访问!
为了你能正常使用网站功能,请使用这些浏览器。

【经验分享】STM32实例-CAN通信④-程序编写

[复制链接]
STMCU小助手 发布时间:2022-6-26 17:47
硬件设计
    本实验使用到硬件资源如下:
(1)D1 指示灯
(2)K_UP 和 K_DOWN按键
(3)串口 1
(4)CAN 控制器
(5)CAN 收发器:TJA1040
    D1指示灯、K_UP 和 K_DOWN 按键、串口 1 电路在前面章节都介绍过,这里就不多说,至于 CAN 控制器它属于 STM32F1 芯片内部的资源,只要通过软件配置好即可使用,下面我们看下 STM32F1 的 CAN 与 TJA1040 的连接,如下图所示:
微信图片_20220626174723.png

微信图片_20220626174721.png

微信图片_20220626174718.png

    从电路图中可以看到,TJA1040 芯片的 CAN_TX 和 CAN_RX 管脚连接在P16 排针上,因为 PA11 和 PA12 是 USB 和 CAN1 共用的管脚,所以使用 P16排针来做切换,本章实验我们使用的是 CAN1,所以将 P16 排针短接到CAN 端,PCB 板上P16插针短接图如下图所示:

微信图片_20220626174714.png

    在TJA1040 的 CAN_H和CAN_L 管脚上连接了一个 120 欧的终端电阻, 如果我们的开发板不是作为 CAN 终端的话,需要把这个电阻去掉,以免影响通信。最后我们用两根导线将两块开发板上 CAN 收发器的 CAN_H 和 CAN_L 对应连接,不要交叉,否则通信错误。CAN_H 和 CAN_L 示意图如下图所示:

微信图片_20220626174711.png

    连接好后, 我们 STM32F1 的 CAN 控制器就可以通过 CAN 收发器 TJA1040与外部 CAN总线进行通信了。
    D1指示灯用来提示系统运行状态, K_UP 按键用来切换 CAN 工作模式, K_DOWN键用来发送数据,通过串口 1将切换的模式和发送与接收到的数据打印输出。

函数编写
    本实例所要实现的功能是:通过 K_UP 键切换 CAN 通信模式,K_DOWN 键控制数据发送,将切换的 CAN 模式,发送和接收的数据通过串口打印输出,整个过程D1 指示灯不断闪烁,提示系统正常运行。本章我们使用的是 STM32F1 的 CAN1,程序框架如下:
(1)初始化 CAN1,包括工作模式、波特率、筛选器设置等
(2)编写 CAN1 的发送和接收函数
(3)编写主函数
    前面介绍 CAN 配置步骤时, 就已经讲解如何初始化 CAN。下面我们打开 “CAN 通信实验”工程,在 APP 工程组中添加can.c 文件(里面包含了 CAN 驱动程序),在 StdPeriph_Driver 工程组中添加stm32f10x_can.c 库文件。CAN 操作的库函数都放在 stm32f10x_can.c 和stm32f10x_can.h 文件中,所以使用到 CAN 就必须加入 stm32f10x_can.c文件,同时还要包含对应的头文件路径。
    这里我们分析几个重要函数,其他部分程序大家可以打开工程查看。

CAN1 初始化函数
    要使用 STM32F1 的 CAN1,我们必须先对它进行配置。初始化代码如下:
  1. //CAN 初始化
  2. //tsjw:重新同步跳跃时间单元.范围:CAN_SJW_1tq~ CAN_SJW_4tq
  3. //tbs2:时间段 2 的时间单元. 范围:CAN_BS2_1tq~CAN_BS2_8tq;
  4. //tbs1:时间段 1 的时间单元. 范围:CAN_BS1_1tq ~CAN_BS1_16tq
  5. //brp :波特率分频器.范围:1~1024; tq=(brp)*tpclk1
  6. //波特率=Fpclk1/((tbs1+tbs2+1)*brp);
  7. //mode:CAN_Mode_Normal,普通模式;CAN_Mode_LoopBack,回环模式;
  8. //Fpclk1 的 时 钟 在 初 始 化 的 时 候 设 置 为 36M, 如 果 设 置
  9. CAN_Mode_Init(CAN_SJW_1tq,CAN_BS2_8tq,CAN_BS1_9tq,4,CAN_Mode_LoopBack);
  10. //则波特率为:36M/((8+9+1)*4)=500Kbps
  11. //返回值:0,初始化 OK;
  12. // 其他,初始化失败;
  13. void CAN_Mode_Init(u8 tsjw,u8 tbs2,u8 tbs1,u16 brp,u8 mode)
  14. {
  15. GPIO_InitTypeDef GPIO_InitStructure;
  16. CAN_InitTypeDef CAN_InitStructure;
  17. CAN_FilterInitTypeDef CAN_FilterInitStructure;
  18. #if CAN_RX0_INT_ENABLE
  19. NVIC_InitTypeDef NVIC_InitStructure;
  20. #endif
  21. RCC_APB1PeriphClockCmd(RCC_APB1Periph_CAN1, ENABLE); // 打 开CAN1时钟
  22. RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE); //PA 端口时钟打开
  23. GPIO_InitStructure.GPIO_Pin = GPIO_Pin_11; //PA11
  24. GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU; //上拉输入模式
  25. GPIO_Init(GPIOA, &GPIO_InitStructure);
  26. GPIO_InitStructure.GPIO_Pin = GPIO_Pin_12; //PA12
  27. GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; //复用推挽输出
  28. GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; //IO 口速度为 50MHz
  29. GPIO_Init(GPIOA, &GPIO_InitStructure);
  30. //CAN 单元设置
  31. CAN_InitStructure.CAN_TTCM=DISABLE; //非时间触发通信模式
  32. CAN_InitStructure.CAN_ABOM=DISABLE; //软件自动离线管理
  33. CAN_InitStructure.CAN_AWUM=DISABLE;//睡眠模式通过软件唤醒(清除CAN->MCR 的 SLEEP 位)
  34. CAN_InitStructure.CAN_NART=ENABLE; //使用报文自动传送
  35. CAN_InitStructure.CAN_RFLM=DISABLE; //报文不锁定,新的覆盖旧的
  36. CAN_InitStructure.CAN_TXFP=DISABLE; //优先级由报文标识符决定
  37. CAN_InitStructure.CAN_Mode= mode; //模式设置
  38. CAN_InitStructure.CAN_SJW=tsjw; //重新同步跳跃宽度(Tsjw)为tsjw+1个时间单位 CAN_SJW_1tq~CAN_SJW_4tq
  39. CAN_InitStructure.CAN_BS1=tbs1; //Tbs1 范 围 CAN_BS1_1tq~CAN_BS1_16tq
  40. CAN_InitStructure.CAN_BS2=tbs2;//Tbs2 范 围 CAN_BS2_1tq ~CAN_BS2_8tq
  41. CAN_InitStructure.CAN_Prescaler=brp; //分频系数(Fdiv)为 brp+1
  42. CAN_Init(CAN1, &CAN_InitStructure); // 初始化 CAN1
  43. //配置过滤器
  44. CAN_FilterInitStructure.CAN_FilterNumber=0; //过滤器0
  45. CAN_FilterInitStructure.CAN_FilterMode=CAN_FilterMode_IdMask;
  46. CAN_FilterInitStructure.CAN_FilterScale=CAN_FilterScale_32bit;
  47. //32位
  48. CAN_FilterInitStructure.CAN_FilterIdHigh=0x0000;////32 位ID
  49. CAN_FilterInitStructure.CAN_FilterIdLow=0x0000;
  50. CAN_FilterInitStructure.CAN_FilterMaskIdHigh=0x0000;//32 位MASK
  51. CAN_FilterInitStructure.CAN_FilterMaskIdLow=0x0000;
  52. CAN_FilterInitStructure.CAN_FilterFIFOAssignment=CAN_Filter_FIFO0;
  53. //过滤器 0 关联到 FIFO0
  54. CAN_FilterInitStructure.CAN_FilterActivation=ENABLE; //激活过滤器 0
  55. CAN_FilterInit(&CAN_FilterInitStructure);//滤波器初始化
  56. #if CAN_RX0_INT_ENABLE
  57. CAN_ITConfig(CAN1,CAN_IT_FMP0,ENABLE); //FIFO0 消 息挂号中断允许.
  58. NVIC_InitStructure.NVIC_IRQChannel = USB_LP_CAN1_RX0_IRQn;
  59. NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1;
  60. // 主优先级为 1
  61. NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;
  62. // 次优先级为 0
  63. NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
  64. NVIC_Init(&NVIC_InitStructure);
  65. #endif
  66. }
复制代码

    在CAN1_Mode_Init()函数中,首先使能 CAN1 和 GPIOA 时钟,并将管脚配置为复用功能。然后配置 CAN的工作模式、波特率及筛选器等参数。这一过程在前面步骤介绍中已经提了。CAN1_Mode_Init()函数有 4个参数,用来设置 CAN1 的波特率和工作模式,方便大家修改 CAN 的波特率和工作模式。参数的具体意义在函数开始处已注释。

CAN1 发送与接收函数
    初始化了 CAN1 后,我们就可以编写它的发送和接收函数,具体代码如下:
  1. //can 发送一组数据(固定格式:ID 为 0X12,标准帧,数据帧)
  2. //len:数据长度(最大为8)
  3. //msg:数据指针,最大为8个字节.
  4. //返回值:0,成功;
  5. // 其他,失败;
  6. u8 CAN1_Send_Msg(u8* msg,u8 len)
  7. {
  8. u8 mbox;
  9. u16 i=0;
  10. CanTxMsg TxMessage;
  11. TxMessage.StdId=0x12; // 标准标识符为 0
  12. TxMessage.ExtId=0x12; // 设置扩展标示符(29 位)
  13. TxMessage.IDE=0; // 使用扩展标识符
  14. TxMessage.RTR=0; // 消息类型为数据帧,一帧 8位
  15. TxMessage.DLC=len; // 发送两帧信息
  16. for(i=0;i<len;i++)
  17. TxMessage.Data[i]=msg[i]; // 第一帧信息
  18. mbox= CAN_Transmit(CAN1, &TxMessage);
  19. i=0;
  20. while((CAN_TransmitStatus(CAN1,
  21. mbox)==CAN_TxStatus_Failed)&&(i<0XFFF))i++; //等待发送结束
  22. if(i>=0XFFF)return 1;
  23. return 0;
  24. }
  25. //can 口接收数据查询
  26. //buf:数据缓存区;
  27. //返回值:0,无数据被收到;
  28. // 其他,接收的数据长度;
  29. u8 CAN1_Receive_Msg(u8 *buf)
  30. {
  31. u32 i;
  32. CanRxMsg RxMessage;
  33. if( CAN_MessagePending(CAN1,CAN_FIFO0)==0)return 0; //没有接收到数据,直接退出
  34. CAN_Receive(CAN1, CAN_FIFO0, &RxMessage);//读取数据
  35. for(i=0;i<RxMessage.DLC;i++)
  36. buf[i]=RxMessage.Data[i];
  37. return RxMessage.DLC;
  38. }
复制代码

    CAN1_Send_Msg 函数为发送函数,用于 CAN 报文的发送,主要是设置标识符 ID等信息,写入数据长度和数据,并请求发送,实现一次报文的发送。CAN1_Receive_Msg 函数为接收函数,用于接受报文数据并且将接受到的数据存放到 buf 数组内。

主函数
    编写好 CAN1 初始化和发送接收函数后,接下来就可以编写主函数了,代码如下:
  1. /****************************************************************
  2. * 函 数 名 : main
  3. * 函数功能 : 主函数
  4. * 输 入 : 无
  5. * 输 出 : 无
  6. *****************************************************************/
  7. int main()
  8. {
  9. u8 i=0,j=0;
  10. u8 key;
  11. u8 mode=0;
  12. u8 res;
  13. u8 tbuf[8],char_buf[8];
  14. u8 rbuf[8];
  15. SysTick_Init(72);
  16. NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2); //中断优先级分组 分2 组
  17. LED_Init();
  18. USART1_Init(9600);
  19. KEY_Init();
  20. CAN_Mode_Init(CAN_SJW_1tq,CAN_BS2_8tq,CAN_BS1_9tq,4,CAN_Mode_Normal);//500Kbps 波特率
  21. while(1)
  22. {
  23. key=KEY_Scan(0);
  24. if(key==KEY_UP) //模式切换
  25. {
  26. mode=!mode;
  27. CAN_Mode_Init(CAN_SJW_1tq,CAN_BS2_8tq,CAN_BS1_9tq,4,mode);
  28. if(mode==0)
  29. {
  30. printf("Normal Mode\r\n");
  31. }
  32. else
  33. {
  34. printf("LoopBack Mode\r\n");
  35. }
  36. }
  37. if(key==KEY_DOWN) //发送数据
  38. {
  39. for(j=0;j<8;j++)
  40. {
  41. tbuf[j]=j;
  42. char_buf[j]=tbuf[j]+0x30;
  43. }
  44. res=CAN_Send_Msg(tbuf,8);
  45. if(res)
  46. {
  47. printf("Send Failed!\r\n");
  48. }
  49. else
  50. {
  51. printf("发送数据:%s\r\n",char_buf);
  52. }
  53. }
  54. res=CAN_Receive_Msg(rbuf);
  55. if(res)
  56. {
  57. for(j=0;j<res;j++)
  58. {
  59. char_buf[j]=rbuf[j]+0x30;
  60. }
  61. printf("接收数据:%s\r\n",char_buf);
  62. }
  63. i++;
  64. if(i%20==0)
  65. {
  66. led1=!led1;
  67. }
  68. delay_ms(10);
  69. }
  70. }
复制代码

    主函数实现的功能很简单,首先调用之前编写好的硬件初始化函数,包括SysTick 系统时钟,中断分组,LED 初始化等。然后调用我们前面编写的CAN1_Mode_Init 函数,我们将 CAN1 波特率设置为 500Kbps(波特率计算可参考前面内容),工作模式为正常模式 CAN_Mode_Normal。最后进入 while 循环,循环检测K_UP 和 K_DOWN 按键是否按下。如果 K_UP 键按下,通过 mode 值变化实现CAN1的工作模式的切换(正常/环回模式),并将切换的工作模式提示信息打印输出。如果 K_DOWN 键按下,调用 CAN1_Send_Msg 函数将数据发送出去,并将发送的数据内容打印输出,同时调用 CAN1_Receive_Msg 函数将发送的数据内容保存在 rbuf 数组内,通过串口打印输出。整个过程 D1 指示灯会间隔200ms闪烁,提示系统正常运行。
    将工程程序编译后下载到开发板内,可以看到 D1 指示灯不断闪烁,表示程序正常运行。默认我们设置的是正常模式,所以需要连接 2 块 STM32F1 开发板的CAN 接口才能互发数据(前提是两块开发板都下载了这个程序),当按下任意一块开发板上的 K_DOWN 键后,通过串口就可以查看它们的收发数据。如果想在串口调试助手上看到输出信息,可以打开“串口调试助手”,首先勾选下标号 1 DTR 框,然后再取消勾选。这是因为此串口助手启动时会把系统复位住,通过 DTR 状态切换下即可。然后设置好波特率等参数后,串口助手上即会收到串口发送过来的信息。(前提一定要用短接片将P16排针短接到 CAN 端,串口助手上先勾选下标号 1 DTR 框,然后再取消勾选)如下图所示:
微信图片_20220626174701.png

收藏 评论0 发布时间:2022-6-26 17:47

举报

0个回答

所属标签

相似分享

官网相关资源

关于
我们是谁
投资者关系
意法半导体可持续发展举措
创新与技术
意法半导体官网
联系我们
联系ST分支机构
寻找销售人员和分销渠道
社区
媒体中心
活动与培训
隐私策略
隐私策略
Cookies管理
行使您的权利
官方最新发布
STM32N6 AI生态系统
STM32MCU,MPU高性能GUI
ST ACEPACK电源模块
意法半导体生物传感器
STM32Cube扩展软件包
关注我们
st-img 微信公众号
st-img 手机版