STM32F2-CAN通信-发送任意长度

STM32F2-CAN通信-发送任意长度,调试好的,与大家分享


#include "include.h"
//************************************************************************************************************************************************
//20190611调试完成
//By Wang
//************************************************************************************************************************************************

CanRxMsg RxMessage;
CanTxMsg TxMessage;
//************************************************************************************************************************************************
//函数名:void NVIC_Config(void)
//功  能:  CAN通信接收RX0中断配置
//参  数:无
//返回值:无
//************************************************************************************************************************************************

void NVIC_Config(void)
{
  NVIC_InitTypeDef  NVIC_InitStructure;

//#ifdef  USE_CAN1 
  
  NVIC_InitStructure.NVIC_IRQChannel = CAN1_RX0_IRQn;//与CAN1_RX1_IRQn;区别
  NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0x00;//不同的中断优先级一样会怎样
  NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0x00;
  
  NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
  NVIC_Init(&NVIC_InitStructure);
  
}

//************************************************************************************************************************************************
//函数名:void CAN_Config(void)
//功  能:  CAN1配置 PD0-RX;PD1-TX
//参  数:无
//返回值;无
//************************************************************************************************************************************************
void CAN_Config(void)
{
  GPIO_InitTypeDef GPIO_InitStructure;
  CAN_InitTypeDef CAN_InitStructure;
  CAN_FilterInitTypeDef CAN_FilterInitStructure;
    
  /* CAN GPIOs configuration ***********************************/
  /* Enable GPIOD clock */
  RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOD, ENABLE);
  
  /* Connect PD1 to CAN1_Tx pin */
  GPIO_PinAFConfig(GPIOD, GPIO_PinSource0, GPIO_AF_CAN1);
  /* Connect PD0 to CAN1_Rx pin */
  GPIO_PinAFConfig(GPIOD, GPIO_PinSource1, GPIO_AF_CAN1);
  
  /* Configure CAN1_Rx(PD0) and CAN1_Tx(PD1) pins */
  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1;
  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;
  GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;
  GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP;
  GPIO_Init(GPIOD, &GPIO_InitStructure);
  
   GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0;//Rx要单独初始化,否则进入不了中断,不能按照ST官方库来设定
   GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;
   GPIO_Init(GPIOD, &GPIO_InitStructure);
  /* CAN configuration *****************************************/
    /* Enable CAN1 clock */
  RCC_APB1PeriphClockCmd(RCC_APB1Periph_CAN1, ENABLE);
  
  /* CAN register init */
  CAN_DeInit(CAN1);
  /* CAN cell init */
  CAN_InitStructure.CAN_TTCM = DISABLE;
  CAN_InitStructure.CAN_ABOM = DISABLE;
  CAN_InitStructure.CAN_AWUM = DISABLE;
  CAN_InitStructure.CAN_NART = DISABLE;
  CAN_InitStructure.CAN_RFLM = DISABLE;
  CAN_InitStructure.CAN_TXFP = DISABLE;
  CAN_InitStructure.CAN_Mode = CAN_Mode_Normal;
  CAN_InitStructure.CAN_SJW = CAN_SJW_1tq;
  /* CAN Baudrate = 1MBps (CAN clocked at 30 MHz) */ //30/(1+6+8)/4=500k
  CAN_InitStructure.CAN_BS1 = CAN_BS1_6tq;
  CAN_InitStructure.CAN_BS2 = CAN_BS2_8tq;
  CAN_InitStructure.CAN_Prescaler = 4;
  CAN_Init(CAN1, &CAN_InitStructure);
  
  /* CAN filter init */
  CAN_FilterInitStructure.CAN_FilterNumber = 0;
  CAN_FilterInitStructure.CAN_FilterMode = CAN_FilterMode_IdMask;
  CAN_FilterInitStructure.CAN_FilterScale = CAN_FilterScale_32bit;
  CAN_FilterInitStructure.CAN_FilterIdHigh = 0x0000;
  CAN_FilterInitStructure.CAN_FilterIdLow = 0x0000;
  CAN_FilterInitStructure.CAN_FilterMaskIdHigh = 0x0000;
  CAN_FilterInitStructure.CAN_FilterMaskIdLow = 0x0000;
  CAN_FilterInitStructure.CAN_FilterFIFOAssignment = 0;
  CAN_FilterInitStructure.CAN_FilterActivation = ENABLE;
  CAN_FilterInit(&CAN_FilterInitStructure);

  /* Enable FIFO 0 message pending Interrupt */  
  Init_TxMes();//发送结构体参数初始化
  Init_RxMes();//接收结构体参数初始化
  NVIC_Config();//中断向量配置
  CAN_ITConfig(CAN1, CAN_IT_FMP0, ENABLE);

}
//************************************************************************************************************************************************
//函数名:void Init_TxMes(CanRxMsg *RxMessage)
//功  能:  Initializes the Tx Message
//参  数:无
//返回值:无
//************************************************************************************************************************************************
void Init_TxMes()
{
    /* Transmit Structure preparation */
  TxMessage.StdId = 0x123;//标准标识符,必须在( TxMessage.IDE = CAN_ID_STD;//标准帧)情况下才有效
  TxMessage.ExtId = 0x00;//设置拓展标识符(29位),必须在( TxMessage.IDE = CAN_ID_EXT;//标准帧)情况下才有效
  TxMessage.RTR = CAN_RTR_DATA;//数据帧
  TxMessage.IDE = CAN_ID_STD;//标准帧 
  TxMessage.DLC = 8;//发送数据长度  DLC=0-8
}
//************************************************************************************************************************************************
//函数名:void Init_RxMes(CanRxMsg *RxMessage)
//功  能:  Initializes the Rx Message
//参  数:无
//返回值:无
//************************************************************************************************************************************************
void Init_RxMes()
{
  u8 i = 0;
  RxMessage.StdId = 0x00;//这个可以在接收中断中判断 如果RxMessage->StdId = TxMessage.StdId 接收正确 
  RxMessage.ExtId = 0x00;//这个可以在接收中断中判断 如果RxMessage->ExtId = TxMessage.ExtId 接收正确 
  RxMessage.IDE = CAN_ID_STD;//这个可以在接收中断中判断 如果RxMessage->IDE = TxMessage.IDE 接收正确 
  RxMessage.DLC = 8;//接收数据长度
  RxMessage.FMI =16;//????????????????
  
  for (i = 0;i < 8;i++)
  {
    RxMessage.Data[i] = 0x00;
  }
}
//************************************************************************************************************************************************
//函数名:void CAN1_RX0_IRQHandler()
//功  能:  CAN1-RX0接收中断服务函数
//参  数:无
//返回值:无
//************************************************************************************************************************************************
void CAN1_RX0_IRQHandler()  //与CAN1_RX1_IRQHandler()区别
{
  if(CAN_GetITStatus(CAN1,CAN_IT_FMP0)!= RESET)//判断是否发生CAN-R0中断
  {
    CAN_ClearITPendingBit(CAN1,CAN_IT_FMP0);//清除中断标志
    if(CAN_MessagePending(CAN1,CAN_FIFO0)!=0)//接收到数据
       CAN_Receive(CAN1,CAN_Filter_FIFO0,&RxMessage);//接收数据最大只能8字节
  }
  Usart_Send_Buff(RxMessage.Data,8);//发送数据RS232通信    
}
//************************************************************************************************************************************************
//函数名:u8 CAN_SendData()
//功  能:  CAN发送数据  发送第二步
//参  数:Msg数据指针;Length数据长度
//返回值:返回成功/失败标志
//************************************************************************************************************************************************
u8 CAN_SendData(u8* Msg,u8 Length)
{
  u16 i=0;
  u8 mbox=0x00; 
  TxMessage.DLC = Length;//发送数据长度  DLC=0-8
  for(i=0;i<Length;i++)
  {
     TxMessage.Data[i]=Msg[i];
  }
  mbox=CAN_Transmit(CAN1,&TxMessage);//发送数据  返回邮箱使用数量
  
//  while((CAN_TransmitStatus(CAN1,mbox)!=CAN_TxStatus_Ok)&&(i<0xfff))
//  {
//    i++;
//  }
//  if(i>=0xfff) 
//  {
//    return 0;//发送数据失败
//  }
//  return 1;//发送数据成功
}
//************************************************************************************************************************************************
//函数名:void CAN_SendDataProces(u8* data,u8 Num)
//功  能:CAN发送数据前期处理   发送第一步
//参  数:data数据指针;Num数据长度
//返回值:无
//************************************************************************************************************************************************
void CAN_SendDataProces(u8* data,u8 Num)
{
  u8 flag=0x00,i=0;
  u8 Count=0x00,Left=0x00;

  Count=Num/8;//求多少个8字节倍数
  Left=Num%8;//剩余小于8字节的数量
  for(i=0;i<Count;i++)
  {
    flag=CAN_SendData((data+i*8),8);//发送整8字节的数据
  } 
  if(Left>0)//有余数才单独再发送一次
  {
    flag=CAN_SendData((data+8*Count),Left);//发送剩余字节
  }
}
//************************************************************************************************************************************************
 

  • 5
    点赞
  • 25
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
### 回答1: STM32F-J1939 CAN通信协议是一种适用于物联网设备的协议,该协议是基于CAN总线实现的。其主要应用于移动设备与汽车之间的通信,旨在提高通信效率和可靠性。 该协议使用标准的J1939协议,并通过STM32F系列单片机实现CAN总线通信。通过该协议,不仅能够实现数据的高速传输,还能够保证数据的完整性和准确性。 当使用STM32F-J1939 CAN通信协议时,需要注意的是必须按照协议要求设置CAN节点地址、数据帧格式等参数才能正常使用。此外,在实际应用中还需要进行一些必要的调试和测试工作,以确保数据传输的可靠性和稳定性。 总之,STM32F-J1939 CAN通信协议是一种可靠高效的通信协议,其在物联网和汽车领域具有广泛应用价值,未来会有更多厂商将其应用于各种智能设备和系统中。 ### 回答2: STM32F-J1939是一种嵌入式系统模块,支持CAN通信协议。CAN(Controller Area Network控制器局域网)是一种用于高速数据传输和实时控制的通信协议。它最初是为汽车行业开发的,但现在已广泛应用于许多不同的应用程序领域。 STM32F-J1939模块是在STM32系列微控制器上实现的,它配备了可编程的CAN控制器,可以很好地支持CAN通信标准。这意味着模块已经具备了CAN通信所需的硬件资源并且通过软件可以进行配置和控制。 J1939是一种基于CAN通信协议的高层协议,最初用于商用车辆和重型机械设备中。这种协议定义了一组通用的信号类型和数据格式,用于在车辆和机器之间传递信息。STM32F-J1939模块可以很好地支持这种高层协议,并且可以通过J1939协议指定和处理消息。 总结来说,STM32F-J1939模块通过CAN通信协议和J1939高层协议,使嵌入式系统在车辆和机械设备中实现快速、可靠的数据传输和实时控制。同时,它也为开发人员提供了方便的软件配置和控制接口。 ### 回答3: STM32F-J1939是一种可以实现CAN通信协议的芯片。J1939通信协议被广泛应用于重型商用车辆和机械设备的网络通信中,因为它具有高效稳定、安全可靠、数据传输速度快等优点。这种协议的实现需要支持CAN通信的硬件和软件,并需要作者根据实际需求进行编程。 利用STM32F-J1939芯片实现CAN通信协议非常方便,可以实现广泛的应用场景,例如汽车和设备之间的数据传输、监控、诊断等。同时,通过使用CAN通信协议,可以避免数据传输中的干扰和信号丢失等问题,从而提高了通信的稳定性和可靠性。 总之,STM32F-J1939芯片的CAN通信协议实现将为重型商用车辆和机械设备带来更高效、更安全的通信方式,这种协议具有广泛的应用前景和市场前景。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值