stm32 can总线通信[库函数]

发布者:恬淡岁月最新更新时间:2016-03-08 来源: eefocus关键字:stm32  can总线通信  库函数 手机看文章 扫描二维码
随时随地手机看文章
    CAN是控制器局域网络(Controller Area Network, CAN)的简称,是由研发和生产汽车电子产品著称的德国BOSCH公司开发了的,并最终成为国际标准(ISO118?8)。近年来,其所具有的高可靠性和良好的错误检测能力受到重视,被广泛应用于汽车计算机控制系统和环境温度恶劣、电磁辐射强和振动大的工业环境。相比于I2C和SPI总线结构,can总线定义了更为优秀的物理层、数据链路层,并拥有种类丰富、繁简不一的上层协议。
        
        CAN总线的物理连接只需要两根线,通常称为CAN_H和CAN_L,通常查分信号进行数据的传输。 CAN总线有两种电平,分为隐性电平和显性电平,这两种电平表现为“与”的关系。
  • 若隐性电平相遇,则总线表现为隐性电平。
  • 若显性电平相遇,则总线表现为显性电平。
  • 若隐性电平和显性电平相遇,则总线表现为显性电平。
 
    CAN总线网络是一种多主机网络,在总线处于空闲时,任何一个节点都可以申请成为主机,向总线发送消息,最先访问总线的节点单元可以获得总线的控制权。
 
    CAN总线的所有消息都是以固定的形式打包发送的。两个以上的节点单元同时发送数据时,根据节点标志符决定各自优先关系。CAN总线并没有类似其他的总线上的地址的概念,在总线上增加节点时,连接节点的其他单元软硬件什么都不需要改变。
 
    CAN总线的通信速率和总线长度有关,在总线长度小于40m的场合中,数据传输速率可以达到1Mbps,即便长度达到1000m,数据传输数率也可以达到50Kbps,无论在数率和传输距离上都由于常见的RS232、RS485和I2C总线。
    
    CAN总线在理论上节点数没有上限,但是实际中受到总线上的时间延时和电气负载的限制。降低节点数可以增大通信速率。
 
    Stm32 至少配备一个CAN总线控制器,最高速率可以达到1Mbps,支持11位的标准帧格式和29为的拓展帧格式的接收和发送,具备三个邮箱和2个接收FIFO,此外还有3级可编程滤波器。
 
本例主要实现使用stm32的CAN总线实现数据的发送和接收,使用串口观察数据。
 

 

库函数操作
 
CAN 通信 过滤器 和 屏蔽器 :
 
例如设置某接收滤波寄存器00000000001(11位),接收屏蔽寄存器11111111101(11位),则该对组合会拒绝接收00000000011和00000000001之外所有的标识符对应的CAN帧,因为屏蔽器规定第二位(为0)以外的所有标识符位要严格匹配(与滤波器值一致),第二位的滤波器值和收到的CAN标识符第二位值是否一致都可以.
 
main.c
001 #include "stm32f10x.h"
002 #include "stdio.h"
003   
004 #define  PRINTF_ON  1
005   
006 void RCC_Configuration(void);
007 void GPIO_Configuration(void);
008 void USART_Configuration(void);
009 void CAN_Configuration(void);
010 void NVIC_Configuration(void);
011   
012 u8 TransmitMailbox = 0;
013 CanTxMsg TxMessage;
014 CanRxMsg RxMessage;
015   
016 int main(void)
017 {
018     RCC_Configuration();
019     GPIO_Configuration();
020     USART_Configuration();
021     CAN_Configuration();
022   
023   
024     TxMessage.ExtId = 0x00AA0000;
025     TxMessage.RTR = CAN_RTR_DATA;
026     TxMessage.IDE = CAN_ID_EXT;
027     TxMessage.DLC = 8;
028     TxMessage.Data[0] = 0x00;
029     TxMessage.Data[1] = 0x12;
030     TxMessage.Data[2] = 0x34;
031     TxMessage.Data[3] = 0x56;
032     TxMessage.Data[4] = 0x78;
033     TxMessage.Data[5] = 0xAB;
034     TxMessage.Data[6] = 0xCD;
035     TxMessage.Data[7] = 0xEF;
036   
037     TransmitMailbox = CAN_Transmit(CAN1,&TxMessage);
038     while(CAN_TransmitStatus(CAN1,TransmitMailbox) != CANTXOK);
039     printf("\r\n The CAN has send data: %d ,%d,%d ,%d,%d ,%d,%d ,%d \r\n",
040         TxMessage.Data[0],
041         TxMessage.Data[1],
042         TxMessage.Data[2],
043         TxMessage.Data[3],
044         TxMessage.Data[4],
045         TxMessage.Data[5],
046         TxMessage.Data[6],
047         TxMessage.Data[7]
048     );
049   
050     while(CAN_MessagePending(CAN1,CAN_FIFO0) == 0); 
051       
052     //RxMessage.StdId = 0x00;
053     RxMessage.IDE = CAN_ID_EXT;
054     RxMessage.DLC = 0;
055     RxMessage.Data[0] = 0x00;
056     RxMessage.Data[1] = 0x00;
057     RxMessage.Data[2] = 0x00;
058     RxMessage.Data[3] = 0x00;
059     RxMessage.Data[4] = 0x00;
060     RxMessage.Data[5] = 0x00;
061     RxMessage.Data[6] = 0x00;
062     RxMessage.Data[7] = 0x00;
063   
064     CAN_Receive(CAN1,CAN_FIFO0,&RxMessage);
065     printf("\r\n The CAN has receive data : %d,%d,%d,%d,%d,%d,%d,%d \r\n",
066         RxMessage.Data[0],
067         RxMessage.Data[1],
068         RxMessage.Data[2],
069         RxMessage.Data[3], 
070         RxMessage.Data[4],
071         RxMessage.Data[5],
072         RxMessage.Data[6],
073         RxMessage.Data[7]           
074     );
075   
076     while(1);
077 }
078   
079 void CAN_Configuration(void)
080 {
081     CAN_InitTypeDef CAN_InitStructure;
082     CAN_FilterInitTypeDef CAN_FilterInitStructure;
083   
084     CAN_DeInit(CAN1);
085     CAN_StructInit(&CAN_InitStructure);
086   
087     CAN_InitStructure.CAN_TTCM = DISABLE;
088     CAN_InitStructure.CAN_ABOM = DISABLE;
089     CAN_InitStructure.CAN_AWUM = DISABLE;
090     CAN_InitStructure.CAN_NART = DISABLE;
091     CAN_InitStructure.CAN_RFLM = DISABLE;
092     CAN_InitStructure.CAN_TXFP = DISABLE;
093     CAN_InitStructure.CAN_Mode = CAN_Mode_LoopBack;
094     CAN_InitStructure.CAN_SJW = CAN_SJW_1tq;
095     CAN_InitStructure.CAN_BS1 = CAN_BS1_8tq;
096     CAN_InitStructure.CAN_BS2 = CAN_BS2_7tq;
097     CAN_Init(CAN1,&CAN_InitStructure);
098   
099     CAN_FilterInitStructure.CAN_FilterNumber = 0;
100     CAN_FilterInitStructure.CAN_FilterMode = CAN_FilterMode_IdMask;
101     CAN_FilterInitStructure.CAN_FilterScale = CAN_FilterScale_32bit;
102     CAN_FilterInitStructure.CAN_FilterIdHigh = 0x00AA << 3;           //匹配过滤寄存器,因为数据标志符段 还有 IDE ,RTR 和一个补零位  所以左移三位
103     CAN_FilterInitStructure.CAN_FilterIdLow = 0x0000;
104     CAN_FilterInitStructure.CAN_FilterMaskIdHigh = 0x00FF << 3 ;      //匹配屏蔽寄存器
105     CAN_FilterInitStructure.CAN_FilterFIFOAssignment = 0;
106     CAN_FilterInitStructure.CAN_FilterActivation = ENABLE;
107   
108     CAN_FilterInit(&CAN_FilterInitStructure);
109   
110   
111 }
112   
113   
114 void GPIO_Configuration(void)
115 {
116     GPIO_InitTypeDef    GPIO_InitStructure;
117   
118     GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
119   
120     GPIO_InitStructure.GPIO_Pin = GPIO_Pin_11;
121     GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;           
122     GPIO_Init(GPIOB , &GPIO_InitStructure); 
123   
124     GPIO_InitStructure.GPIO_Pin = GPIO_Pin_12;
125     GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; 
126     GPIO_Init(GPIOB , &GPIO_InitStructure); 
127   
128   
129     GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;
130     GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;         
131     GPIO_Init(GPIOA , &GPIO_InitStructure); 
132       
133     GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;
134     GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;           
135     GPIO_Init(GPIOA , &GPIO_InitStructure); 
136 }
137   
138 void RCC_Configuration(void)
139 {
140     
141     ErrorStatus HSEStartUpStatus;
142   
143     
144     RCC_DeInit();
145     
146     RCC_HSEConfig(RCC_HSE_ON);
147     
148     HSEStartUpStatus = RCC_WaitForHSEStartUp();
149     
150     if(HSEStartUpStatus == SUCCESS)
151     {
152         
153         RCC_HCLKConfig(RCC_SYSCLK_Div1); 
154         
155         RCC_PCLK2Config(RCC_HCLK_Div1); 
156         
157         RCC_PCLK1Config(RCC_HCLK_Div2);
158         
159         FLASH_SetLatency(FLASH_Latency_2);
160         
161         FLASH_PrefetchBufferCmd(FLASH_PrefetchBuffer_Enable);
162         
163         RCC_PLLConfig(RCC_PLLSource_HSE_Div1, RCC_PLLMul_9);
164          
165         RCC_PLLCmd(ENABLE);
166         
167         while(RCC_GetFlagStatus(RCC_FLAG_PLLRDY) == RESET);
168         
169         RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK);
170         
171         while(RCC_GetSYSCLKSource() != 0x08);
172     
173     
174     RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|RCC_APB2Periph_USART1, ENABLE);
175   
176     //RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);
177   
178     RCC_APB1PeriphClockCmd(RCC_APB1Periph_CAN1,ENABLE);
179     //RCC_APB1PeriphClockCmd(RCC_APB1Periph_PWR|RCC_APB1Periph_BKP|RCC_APB1Periph_WWDG|RCC_APB1Periph_SPI2|RCC_APB1Periph_I2C1|RCC_APB1Periph_I2C2, ENABLE);
180           
181 }
182   
183    
184 void USART_Configuration(void)
185 {
186     USART_InitTypeDef USART_InitStructure;
187     USART_ClockInitTypeDef USART_ClockInitStructure;
188   
189     USART_ClockInitStructure.USART_Clock = USART_Clock_Disable;
190     USART_ClockInitStructure.USART_CPOL = USART_CPOL_Low;
191     USART_ClockInitStructure.USART_CPHA = USART_CPHA_2Edge;                                                                                                                                                      
192     USART_ClockInitStructure.USART_LastBit = USART_LastBit_Disable;
193     USART_ClockInit(USART1 , &USART_ClockInitStructure);
194   
195     USART_InitStructure.USART_BaudRate = 9600;
196     USART_InitStructure.USART_WordLength = USART_WordLength_8b;
197     USART_InitStructure.USART_StopBits = USART_StopBits_1;
198     USART_InitStructure.USART_Parity = USART_Parity_No;
199     USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;
200     USART_InitStructure.USART_Mode = USART_Mode_Rx|USART_Mode_Tx;
201     USART_Init(USART1,&USART_InitStructure);
202   
203     USART_Cmd(USART1,ENABLE);
204 }
205   
206 #if  PRINTF_ON
207   
208 int fputc(int ch,FILE *f)
209 {
210     USART_SendData(USART1,(u8) ch);
211     while(USART_GetFlagStatus(USART1,USART_FLAG_TC) == RESET);
212     return ch;
213 }
214   
215 #endif

关键字:stm32  can总线通信  库函数 引用地址:stm32 can总线通信[库函数]

上一篇:Linux下的串口总线驱动(四)
下一篇:s3c2410的timer定时器的使用

推荐阅读最新更新时间:2024-03-16 14:46

STM32库函数EXTI_GetFlagStatus和EXTI_GetITStatus区别
一个是获取状态的,一个是获取中断的。 EXTI_GetFlagStatus只是纯粹读取中断标志位的状态,但是不一定会响应中断(EXT_IMR寄存器对该中断进行屏蔽);而EXTI_GetITStatus除了读取中断标志位,还查看EXT_IMR寄存器是否对该中断进行屏蔽,在中断挂起&没有屏蔽的情况下就会响应中断。仔细看看代码就知道区别了 FlagStatus EXTI_GetFlagStatus(uint32_t EXTI_Line) { FlagStatus bitstatus = RESET; assert_param(IS_GET_EXTI_LINE(EXTI_Line)); if ((EXTI- PR & EX
[单片机]
STM32学习笔记---名词解释
1、JTAG是联合测试工作组(Joint Test Action Group)的简称,是在名为标准测试访问端口和边界扫描结构的 IEEE 的标准1149.1的常用名称。此标准用于 测试访问端口 ,使用 边界扫描 的方法来测试 印刷电路板 。使得程序员可以经由JTAG去读取集成在CPU上的调试模块。调试模块可以让程序员调试 嵌入式系统 中的软件。 2、J-Link是SEGGER公司为支持仿真 ARM 内核芯片推出的 JTAG仿真器 。 3、CMSIS:ARM Cortex™ 微控制器软件接口标准(CMSIS:Cortex Microcontroller Software Interface Standard) 是 Cortex-M
[单片机]
STM32步进伺服电机梯形加速源程序 单轴简易运动控制器
步进电机梯形加速程序 单片机源程序如下: /*基于STM32的单轴简易运动控制器/脉冲发生器*/ /*脉冲+方向控制步进伺服电机*/ /* 优化记录: 增加急停GPIOC.0、正向极限GPIOC.1、负向极限GPIOC.2等输入IO接点 中断修改TIMx_PSC一个寄存器的值,而不是修改TIMx_ARR预加载寄存器+TIMx_CCRx比较值寄存器两个值,缩短中断处理时间 定位指令DRVI/DRVA中,目标频率设定过高、而实际输出脉冲数过少时,则不必加速到目标频率即进入减速区 */ /* DRVI(A);相对定位,输出A(A取绝对值)个脉冲 A不能为0 若A为正数,则方向为正、GPIOB.0为高电平 若A为负数,则
[单片机]
STM32 Flash详解
本文将根据ST官方Flashprogramming manual,文档编号:PM0059,讲解STM32F207内部Flash编程。 01、概述 这里的flash是指STM32F207内部集成的Flash Flash存储器有以下特点 最大1M字节的能力 128位,也就是16字节宽度的数据读取 字节,半字,字和双字写入 扇区擦除和批量擦除 存储器的构成 主要存储区块包含4个16K字节扇区,1个64K字节扇区和7个128K字节扇区。 系统存储器是用于在系统boot模式启动设备的。这一块是预留给ST的。包括bootloader程序,boot程序用于通过以下接口对Flash进行编程。USART1、USART3、CAN2、
[单片机]
<font color='red'>STM32</font> Flash详解
STM32 IAP 在线更新程序问题
搞了一天才发现 在MDK设置选项中有个 linker 选项卡 , 需要在Project- Options- Linker中将Use Memory Layout from Target Dialog前面的复选框勾上 一切就正常了 Use Memory Layout from Target Dialog这个选项有个“分散加载文件”的概念。 下边的三段话是我从网上摘抄的(MDK程序下载到flash--分散加载文件scatter),大概了解了一下,以后如果会用到这方面的知识再来深入研究。 前边三个分散加载文件是从别的地方拷过来的,用在自己的程序中可能会有问题,因为如果不修改它的话就固定了加载地址和运行地址,如果程序简单又比较
[单片机]
STM32入门系列-创建寄存器模板
介绍如何使用 KEIL5 软件创建寄存器模板, 方便之后使用寄存器方式来操作STM32开发板上的LED,让大家创建属于自己的寄存器工程模板。 获取工程模板的基础文件 首先我们在电脑任意位置创建一个文件夹,命名为“寄存器模板创建”,然后在其下面新建 2 个文件夹,如下: Obj 文件夹: 用于存放编译产生的 c/汇编/链接的列表清单、 调试信息、hex文件、预览信息、封装库等文件。 User 文件夹: 用于存放用户编写的 main.c、 STM32F1 启动文件、stm32f10x.h头文件。 我们需要将寄存器工程模板所需的基础文件 main.c、STM32F1 启动文件及stm32f10x.h 头文件放入到
[单片机]
<font color='red'>STM32</font>入门系列-创建寄存器模板
stm32查看存储器的详细准确的地址的方法
由于之前都是看中文参考手册及hal库函数开发指南,有些内容可能和实际中使用的地址有所不同,那么如何看具体的芯片的存储器地址呢? 1、打开工程,找到stm32f767xx.h这个头文件,大概在1312行左右,用宏定义详细列出 2、或者可以搜索关键词RAMITCM_BASE,找到对应的头文件位置,它下面的就是具体的真实地址
[单片机]
基于STM32模拟UART串口通信
UART工作原理 UART即通用异步收发器,是一种串行通信方式。数据在传输过程中是通过一位一位地进行传输来实现通信的,串行通信方式具有传输线少,成本底等优点,缺点是速度慢。串行通信分为两种类型:同步通信方式和异步通信方式。 但一般多用异步通信方式,主要因为接受和发送的时钟是可以独立的这样有利于增加发送与接收的灵活性。异步通信是一个字符接着一个字符传输,一个字符的信息由起始位、数据位、奇偶校验位和停止位组成。 每一个字符的传输靠起始位来同步,字符的前面一位是起始位,用下降沿通知收方开始传输,紧接着起始位之后的是数据位,传输时低位在前高位在后,字符本身由5~8位数据位组成。 数据位后面是奇偶校验位,
[单片机]
基于<font color='red'>STM32</font>模拟UART串口<font color='red'>通信</font>
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved