STM32F042 CAN使用例子

代码如下:

#include "mycan.h"

//CAN初始化
//tsjw:重新同步跳跃时间单元.范围:1~3; CAN_SJW_1tq	 CAN_SJW_2tq CAN_SJW_3tq CAN_SJW_4tq
//tbs2:时间段2的时间单元.范围:1~8;
//tbs1:时间段1的时间单元.范围:1~16;	  CAN_BS1_1tq ~CAN_BS1_16tq
//brp :波特率分频器.范围:1~1024;(实际要加1,也就是1~1024) tq=(brp)*tpclk1
//注意以上参数任何一个都不能设为0,否则会乱.
//波特率=Fpclk1/((tsjw+tbs1+tbs2)*brp);
//mode:0,普通模式;1,回环模式;
//Fpclk1的时钟在初始化的时候设置为36M,如果设置CAN_Normal_Init(1,8,7,5,1);
//则波特率为:48M/((1+8+7)*5)=450Kbps

void CAN_Mode_Init(uint8_t tsjw,uint8_t tbs2,uint8_t tbs1,uint16_t brp,uint8_t mode)
{

	GPIO_InitTypeDef GPIO_InitStructure; 
	CAN_InitTypeDef        CAN_InitStructure;
 	CAN_FilterInitTypeDef  CAN_FilterInitStructure;
#if CAN_RX0_INT_ENABLE 
   	NVIC_InitTypeDef  NVIC_InitStructure;
#endif
		
		RCC_APB2PeriphClockCmd(RCC_APB2Periph_SYSCFG, ENABLE);
		RCC_AHBPeriphClockCmd(RCC_AHBPeriph_GPIOA, ENABLE);//使能PORTA时钟	                   											 
	  SYSCFG->CFGR1 |= SYSCFG_CFGR1_PA11_PA12_RMP;                  //RMAP	
	  GPIO_PinAFConfig(GPIOA,GPIO_PinSource12,GPIO_AF_4);
		GPIO_PinAFConfig(GPIOA,GPIO_PinSource11,GPIO_AF_4);
  	RCC_APB1PeriphClockCmd(RCC_APB1Periph_CAN, ENABLE);//使能CAN时钟	

    GPIO_InitStructure.GPIO_Pin   = GPIO_Pin_11|GPIO_Pin_12;
    GPIO_InitStructure.GPIO_Mode  = GPIO_Mode_AF;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_10MHz;
    GPIO_InitStructure.GPIO_PuPd  = GPIO_PuPd_UP;
    GPIO_Init(GPIOA, &GPIO_InitStructure);
		
	  CAN_DeInit(CAN);
 	//CAN单元设置
		CAN_InitStructure.CAN_TTCM=DISABLE;						 //非时间触发通信模式  //
		CAN_InitStructure.CAN_ABOM=ENABLE;						 //软件自动离线管理	 //
		CAN_InitStructure.CAN_AWUM=DISABLE;						 //睡眠模式通过软件唤醒(清除CAN->MCR的SLEEP位)//
		CAN_InitStructure.CAN_NART=ENABLE;						 	//禁止报文自动传送 //
		CAN_InitStructure.CAN_RFLM=DISABLE;						 //报文不锁定,新的覆盖旧的 // 
		CAN_InitStructure.CAN_TXFP=DISABLE;						 //优先级由报文标识符决定 //
		CAN_InitStructure.CAN_Mode= mode;	         //模式设置: mode:0,普通模式;1,回环模式; //
  	//设置波特率
  	CAN_InitStructure.CAN_SJW=tsjw;				//重新同步跳跃宽度(Tsjw)为tsjw+1个时间单位  CAN_SJW_1tq	 CAN_SJW_2tq CAN_SJW_3tq CAN_SJW_4tq
  	CAN_InitStructure.CAN_BS1=tbs1; //Tbs1=tbs1+1个时间单位CAN_BS1_1tq ~CAN_BS1_16tq
  	CAN_InitStructure.CAN_BS2=tbs2;//Tbs2=tbs2+1个时间单位CAN_BS2_1tq ~	CAN_BS2_8tq
  	CAN_InitStructure.CAN_Prescaler=brp;            //分频系数(Fdiv)为brp+1	//
  	CAN_Init(CAN, &CAN_InitStructure);            // 初始化CAN1 

		CAN_FilterInitStructure.CAN_FilterNumber=0;	  //过滤器0
		CAN_FilterInitStructure.CAN_FilterMode=CAN_FilterMode_IdMask; 
		CAN_FilterInitStructure.CAN_FilterScale=CAN_FilterScale_32bit; //32位 
		CAN_FilterInitStructure.CAN_FilterIdHigh=0x0000;32位ID
		CAN_FilterInitStructure.CAN_FilterIdLow=0x0000;
		CAN_FilterInitStructure.CAN_FilterMaskIdHigh=0x0000;//32位MASK
		CAN_FilterInitStructure.CAN_FilterMaskIdLow=0x0000;
		CAN_FilterInitStructure.CAN_FilterFIFOAssignment=CAN_Filter_FIFO0;//过滤器0关联到FIFO0
		CAN_FilterInitStructure.CAN_FilterActivation=ENABLE; //激活过滤器0

		CAN_FilterInit(&CAN_FilterInitStructure);//滤波器初始化
#if CAN_RX0_INT_ENABLE
	
		CAN_ITConfig(CAN1,CAN_IT_FMP0,ENABLE);//FIFO0消息挂号中断允许.		    

		NVIC_InitStructure.NVIC_IRQChannel = USB_LP_CAN1_RX0_IRQn;
		NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1;     // 主优先级为1
		NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;            // 次优先级为0
		NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
		NVIC_Init(&NVIC_InitStructure);
#endif
}
#if CAN_RX0_INT_ENABLE	//使能RX0中断
//中断服务函数			    
void CEC_CAN_IRQHandler(void)
{
		CanRxMsg RxMessage;
		int i=0;
		CAN_Receive(CAN, 0, &RxMessage);
		//for(i=0;i<8;i++)
		//printf("rxbuf[%d]:%d\r\n",i,RxMessage.Data[i]);
}
#endif

//can发送一组数据(固定格式:ID为0X12,标准帧,数据帧)	
//len:数据长度(最大为8)				     
//msg:数据指针,最大为8个字节.
//返回值:0,成功;
//		 其他,失败;
uint8_t Can_Send_Msg(uint8_t* msg,uint8_t len)
{	
  uint8_t mbox;
  uint16_t i=0;
  CanTxMsg TxMessage;
  TxMessage.StdId=0x12;					 // 标准标识符为0
  TxMessage.ExtId=0x12;				 // 设置扩展标示符(29位)
  TxMessage.IDE=0;			 // 使用扩展标识符
  TxMessage.RTR=0;		 // 消息类型为数据帧,一帧8位
  TxMessage.DLC=len;							 // 发送两帧信息
  for(i=0;i<len;i++)
  TxMessage.Data[i]=msg[i];				 // 第一帧信息          
  mbox= CAN_Transmit(CAN, &TxMessage);   
  i=0;
  while((CAN_TransmitStatus(CAN, mbox)==CAN_TxStatus_Failed)&&(i < 0XFFF))i++;	//等待发送结束
  if(i >= 0XFFF)return 1;
  return 0;		
}
//can口接收数据查询
//buf:数据缓存区;	 
//返回值:0,无数据被收到;
//		 其他,接收的数据长度;
uint8_t Can_Receive_Msg(uint8_t *buf)
{		   		   
 	uint32_t i;
	CanRxMsg RxMessage;
    if( CAN_MessagePending(CAN,CAN_FIFO0)==0)return 0;		//没有接收到数据,直接退出 
    CAN_Receive(CAN, CAN_FIFO0, &RxMessage);//读取数据	
    for(i=0;i<8;i++)
    buf[i]=RxMessage.Data[i];  
	return RxMessage.DLC;	
}

注意:一定要根据自己的封装来设置
SYSCFG->CFGR1 |= SYSCFG_CFGR1_PA11_PA12_RMP;
如果不设置 CAN_Init(CAN, &CAN_InitStructure);会初始化失败

  • 5
    点赞
  • 25
    收藏
    觉得还不错? 一键收藏
  • 5
    评论
STM32F042是一款由STMicroelectronics公司推出的低功耗32位微控制器,具有强大的性能和丰富的外设功能。它是基于ARM Cortex-M0内核设计的,适用于广泛的应用领域,如工业自动化、消费电子、医疗设备等。 在使用STM32F042微控制器时,我们可以利用STMicroelectronics提供的例程来快速上手。例程是提供给开发者的一些示例代码和工程,旨在帮助开发者理解和使用微控制器的各种功能。通过参考例程,我们可以学习和了解如何使用STM32F042的各个外设和特性。 例程通常在官方网站上提供,我们可以根据自己的需求下载并使用。通过研究例程代码,我们可以学习如何配置和使用微控制器的外设,如GPIO、USART、SPI、I2C等,以及如何编写中断服务程序和处理器指令等。 在使用例程之前,我们需要确保已经搭建好了开发环境,包括安装Keil、IAR或者其他适用的开发工具,并且配置好了对应的编译器和调试器。 在开始使用例程之前,我们可以先了解例程的文件组成结构,通常包括源代码、头文件、链接脚本和编译选项等。我们可以使用开发工具打开例程的工程文件,然后编译、烧写和调试代码。 除了基本的外设功能,例程还可以提供一些高级应用场景的示例,如使用PWM控制电机速度、使用定时器实现延时等。这些例程可以帮助开发者更快速地了解和使用STM32F042的高级功能。 总而言之,通过使用STM32F042的例程,我们可以快速入门和掌握微控制器的开发,特别是熟悉外设的配置和使用。同时,我们也可以通过对例程的研究和理解,更深入地学习和应用STM32F042的各种功能和特性。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值