您当前的位置:首页 > 生活常识

can总线详细介绍,基于proteus的51单片机开发实例27-I2C总线的读写

时间:2023-04-05 08:26:11

关于【can总线详细介绍】,今天向乾小编给您分享一下,如果对您有所帮助别忘了关注本站哦。

内容导航:1、can总线详细介绍:基于proteus的51单片机开发实例27-I2C总线的读写2、can总线详细介绍,一文看懂CAN总线

1、can总线详细介绍:基于proteus的51单片机开发实例27-I2C总线的读写

本实例意在了解I2C总线的接口电路设计,IC总线数据的写入和读出。由于51单片机没有内置I2C模块,所以本实例中我们使用51单片机的I/O口模拟I2C总线的读写。

1.2. 设计思路","content"

在单片机系统设计中,完成一个功能的前提是首先了解该功能的实现原理,其次要知道完成该功能的电路设计,第三是能够实现编程控制。所以本例中我们要首先了解I2C器件的基础知识,以此进行电路设计,最后编程实现预期的功能。

1.3. 基础知识","content"

I2C总线是一种双向二进制总线,用于连接单片机和外围设备,是单片机应用系统中应用较多的串行总线之一。

目前有很多外围器件都是IC2总线,例如常见的EEPROM存储器AT24系列,日历时钟芯片PCF8563,DS1302,LCD驱动器PCF8576,AD转换器PCF8591等。

1.3.1. I2C总线接口","content"

I2C只有两根信号线:SDA-双向数据/地址线;SCL-串行时钟线。I2C总线上可以连接多个I2C器件,如下图所示。

can总线详细介绍,基于proteus的51单片机开发实例27-I2C总线的读写

图2 I2C总线连接示意图

需要说明的是,I2C总线的接口是开漏型的,其目的就是为了能够连接多个相同器件。所以进行I2C总线的电路设计时,需要加上拉电阻,如图所示。

既然I2C总线上可以挂多个器件,那么这些器件怎么一一识别呢?我们以AT24C02为例来说明。如下图所示。

A0,A1,A2作为地址设置端口,可以将这三个端口分别连接不同的电平,这样就能区别了。

前面我们学习的RS-232串口通信的数据发送和接收分别用不同的线,一根专门用来发送数据,另一根专门用来接收数据。而I2C总线的数据发送和接收都使用同一根线。I2C总线通信时,有主器件和从器件的区别。当某个器件向总线上发送数据时,这个器件就是主器件,而当这个器件从总线上接收数据时,它又变成了从器件。因此对于参与I2C通信的器件,主器件和从器件是相对的。

1.3.2. I2C总线器件的地址","content"

前面我们已经说过,I2C总线上可以挂接多个器件,每个器件都必须有唯一的地址,这样才能确保总线通信的正确性。I2C总线上所以期间的地址都按照一定的规范来设定。器件地址由7位二进制位组成,然后与一位读写控制位(方向位)合成一个字节的寻址字节(对于存储容量大的,寻址字节之外还要有高地址字节)。

can总线详细介绍,基于proteus的51单片机开发实例27-I2C总线的读写

寻址字节

1.3.3. I2C总线上的时钟信号","content"

I2C总线上的数据传输必须在时钟信号的驱动下进行。

1.3.4. I2C总线的传输协议","content"

1.3.4.1. 起始和结束条件","content"

如下图所示。在数据传送时,必须确认数据传送的开始和结束。

can总线详细介绍,基于proteus的51单片机开发实例27-I2C总线的读写

I2C总线的开始和停止

开始条件

SCL为高电平时,SDA由高电平向低电平跳变。开始传送数据。

结束条件

SCL为高电平时,SDA由低电平向高电平跳变。结束传送数据。

1.3.4.2. 数据格式","content"

开始信号发出后,主器件送出的第一个字节是用来发送从器件地址的。数据的发送是按照字节数据高位在前,低位在后的原则发送的。前7位是地址码,第8位是方向位(通俗来说就是读写控制位,对于主器件来说,读数据是从从器件获得数据,写数据是向从器件发送数据)。I2C总线上的所以器件在收到开始信号后,将收到的第一个字节的数据与自己的地址比较,如果地址一致,则该器件被选定,然后该器件根据方向位进行数据的接收还是发送操作。

I2C总线数据传送采用时钟脉冲逐位串行传送方式。时序图如下。在SCL的低电平期间,SDA线上的数据可以变化。SCL高电平器件,SDA线上的数据必须保持不变。因为此时SDA状态的改变已被用来表示起始和结束条件。

can总线详细介绍,基于proteus的51单片机开发实例27-I2C总线的读写

I2C器件的数据交互

1.3.4.3. 响应","content"

I2C总线协议规定,每传送一个字节数据(包括地址和命令)完毕,都要有一个应答信号ACK,以确定数据传送是否正确。应答位的时钟脉冲由主器件产生,主器件需要在应答脉冲的高电平期间释放数据线SDA,转由从器件控制。通常的做法是从器件在这个时钟期间必须向SDA发送一个低电平,以产生有效的应答信号。表示接受正常,若接收不正常或不能产生应答信号,则保持SDA为高电平,此时,主器件产生一个停止信号,使传送异常结束。

1.4. 电路设计","content"

本实例使用常见的EEPROM存储器AT24C02。该芯片引脚图如下所示。

其中SCL-时钟线;SDA-数据线;WP-写保护,当该引脚接高电平,不允许写入数据,接地,不允许写入数据;A2A1A0-地址输入端,可通过接高电平或低电平确定芯片地址。

can总线详细介绍,基于proteus的51单片机开发实例27-I2C总线的读写

AT24C02引脚图

本实例的电路如图1所示。电路中单片机的P3.4和P3.4连接I2C器件AT24C02。P0口和P2口分别连接两个LED排,用于指示将要写入和读出的数据。

1.5. 程序设计","content"

本实例程序代码如下。

本程序实现的功能是,讲一个数据写入到I2C器件AT24C02的相应地址,同时将要写入的数据通过连接在P0口的8个发光二极管显示,然后读出该位置的数据,并通过连接P2口的8个发光二极管显示。

#include<AT89x51.h> //#include<intrins.h> //#defineOP_READ0xa1// #defineOP_WRITE 0xa0// Æ÷¼þµØÖ·ÒÔ¼°Ð´Èë²Ù×÷,0xa1¼´Îª1010 0000Bsbit SDA=P3^4; //½«´®ÐÐÊý¾Ý×ÜÏßSDA붨ÒåÔÚΪP3.4Òý½Åsbit SCL=P3^3; //½«´®ÐÐʱÖÓ×ÜÏßSDA붨ÒåÔÚΪP3.3Òý½Å/*****************************************************º¯Êý¹¦ÄÜ£ºÑÓʱ1ms(3j+2)*i=(3¡Á33+2)¡Á10=1010(΢Ãë)£¬¿ÉÒÔÈÏΪÊÇ1ºÁÃë***************************************************/void delay1ms();/*****************************************************º¯Êý¹¦ÄÜ£ºÑÓʱÈô¸ÉºÁÃëÈë¿Ú²ÎÊý£ºn***************************************************/ void delaynms(unsigned int n);//ÒÔÏÂÊǶÔAT24C02µÄ¶Áд²Ù×÷³ÌÐòvoid start();void stop();unsigned char ReadData();bit WriteCurrent(unsigned char y);void WriteSet(unsigned char add, unsigned char dat);unsigned char ReadCurrent();unsigned char ReadSet(unsigned char set_add);/*********************************************************************º¯Êý¹¦ÄÜ£ºÖ÷º¯Êý***********************************************************************/ void main(void) { unsigned char sum=0x01; //´¢´æ¼ÆÊýÖµ unsigned char x=0; //´¢´æ´ÓAT24C02¶Á³öµÄÖµ P0=0xff; while(1) //ÎÞÏÞÑ­»·{ P0=sum; WriteSet(0x01,sum); //½«¼Æ¼þֵдÈëAT24C02ÖеÄÖ¸¶¨µØÖ·"0x01" x=ReadSet(0x01); //´ÓAT24C02ÖжÁ³ö¼Æ¼þÖµP2=x; delaynms(300); sum++;P0=sum; WriteSet(0x01,sum); //½«¼Æ¼þֵдÈëAT24C02ÖеÄÖ¸¶¨µØÖ·"0x01" x=ReadSet(0x01); //´ÓAT24C02ÖжÁ³ö¼Æ¼þÖµP2=x; delaynms(300); }}/*****************************************************º¯Êý¹¦ÄÜ£ºÑÓʱ1ms(3j+2)*i=(3¡Á33+2)¡Á10=1010(΢Ãë)£¬¿ÉÒÔÈÏΪÊÇ1ºÁÃë***************************************************/void delay1ms(){ unsigned char i,j; for(i=0;i<10;i++) for(j=0;j<33;j++) ; }/*****************************************************º¯Êý¹¦ÄÜ£ºÑÓʱÈô¸ÉºÁÃëÈë¿Ú²ÎÊý£ºn***************************************************/ void delaynms(unsigned int n) { unsigned int i;for(i=0;i<n;i++) delay1ms(); }/*******************************************************************************ÒÔÏÂÊǶÔAT24C02µÄ¶Áд²Ù×÷³ÌÐò********************************************************************************//***************************************************º¯Êý¹¦ÄÜ£º¿ªÊ¼Êý¾Ý´«ËÍ***************************************************/void start()// ¿ªÊ¼Î»{SDA = 1; //SDA³õʼ»¯Îª¸ßµçƽ¡°1¡±SCL = 1; //¿ªÊ¼Êý¾Ý´«ËÍʱ£¬ÒªÇóSCLΪ¸ßµçƽ¡°1¡±_nop_(); //µÈ´ýÒ»¸ö»úÆ÷ÖÜÆÚ_nop_(); //µÈ´ýÒ»¸ö»úÆ÷ÖÜÆÚSDA = 0; //SDAµÄϽµÑر»ÈÏΪÊÇ¿ªÊ¼ÐźÅ_nop_(); //µÈ´ýÒ»¸ö»úÆ÷ÖÜÆÚ_nop_(); //µÈ´ýÒ»¸ö»úÆ÷ÖÜÆÚ_nop_(); //µÈ´ýÒ»¸ö»úÆ÷ÖÜÆÚ_nop_(); //µÈ´ýÒ»¸ö»úÆ÷ÖÜÆÚSCL = 0; //SCLΪµÍµçƽʱ£¬SDAÉÏÊý¾Ý²ÅÔÊÐí±ä»¯(¼´ÔÊÐíÒÔºóµÄÊý¾Ý´«µÝ£©}/***************************************************º¯Êý¹¦ÄÜ£º½áÊøÊý¾Ý´«ËÍ***************************************************/void stop()// ֹͣλ{SDA = 0; //SDA³õʼ»¯ÎªµÍµçƽ¡°0¡±_nop_(); //µÈ´ýÒ»¸ö»úÆ÷ÖÜÆÚ_nop_(); //µÈ´ýÒ»¸ö»úÆ÷ÖÜÆÚSCL = 1; //½áÊøÊý¾Ý´«ËÍʱ£¬ÒªÇóSCLΪ¸ßµçƽ¡°1¡±_nop_(); //µÈ´ýÒ»¸ö»úÆ÷ÖÜÆÚ_nop_(); //µÈ´ýÒ»¸ö»úÆ÷ÖÜÆÚ_nop_(); //µÈ´ýÒ»¸ö»úÆ÷ÖÜÆÚ_nop_(); //µÈ´ýÒ»¸ö»úÆ÷ÖÜÆÚSDA = 1; //SDAµÄÉÏÉýÑر»ÈÏΪÊǽáÊøÐźÅ}/***************************************************º¯Êý¹¦ÄÜ£º´ÓAT24Cxx¶ÁÈ¡Êý¾Ý³ö¿Ú²ÎÊý£ºx***************************************************/unsigned char ReadData()// ´ÓAT24CxxÒÆÈëÊý¾Ýµ½MCU{unsigned char i;unsigned char x; //´¢´æ´ÓAT24CxxÖжÁ³öµÄÊý¾Ýfor(i = 0; i < 8; i++){SCL = 1; //SCLÖÃΪ¸ßµçƽx<<=1; //½«xÖеĸ÷¶þ½øλÏò×óÒÆһλx|=(unsigned char)SDA; //½«SDAÉϵÄÊý¾Ýͨ¹ý°´Î»¡°»ò¡°ÔËËã´æÈëxÖÐSCL = 0; //ÔÚSCLµÄϽµÑضÁ³öÊý¾Ý}return(x); //½«¶ÁÈ¡µÄÊý¾Ý·µ»Ø}/***************************************************º¯Êý¹¦ÄÜ£ºÏòAT24CxxµÄµ±Ç°µØַдÈëÊý¾ÝÈë¿Ú²ÎÊý£ºy (´¢´æ´ýдÈëµÄÊý¾Ý£©***************************************************///ÔÚµ÷ÓôËÊý¾ÝдÈ뺯ÊýÇ°ÐèÊ×Ïȵ÷ÓÿªÊ¼º¯Êýstart(),ËùÒÔSCL=0bit WriteCurrent(unsigned char y){unsigned char i;bit ack_bit; //´¢´æÓ¦´ðλfor(i = 0; i < 8; i++)// Ñ­»·ÒÆÈë8¸öλ{ SDA = (bit)(y&0x80); //ͨ¹ý°´Î»¡°Ó롱ÔËË㽫×î¸ßλÊý¾ÝË͵½S //ÒòΪ´«ËÍʱ¸ßλÔÚÇ°£¬µÍλÔÚºó_nop_(); //µÈ´ýÒ»¸ö»úÆ÷ÖÜÆÚ SCL = 1; //ÔÚSCLµÄÉÏÉýÑؽ«Êý¾ÝдÈëAT24Cxx _nop_(); //µÈ´ýÒ»¸ö»úÆ÷ÖÜÆÚ _nop_(); //µÈ´ýÒ»¸ö»úÆ÷ÖÜÆÚ SCL = 0; //½«SCLÖØÐÂÖÃΪµÍµçƽ£¬ÒÔÔÚSC£ÌÏßÐγɴ«ËÍÊý¾ÝËùÐèµÄ£¸¸öÂö³åy <<= 1; //½«yÖеĸ÷¶þ½øλÏò×óÒÆһλ}SDA = 1; // ·¢ËÍÉ豸£¨Ö÷»ú£©Ó¦ÔÚʱÖÓÂö³åµÄ¸ßµçƽÆÚ¼ä(SCL=1)ÊÍ·ÅSDAÏߣ¬ //ÒÔÈÃSDAÏßתÓɽÓÊÕÉ豸(AT24Cxx)¿ØÖÆ_nop_(); //µÈ´ýÒ»¸ö»úÆ÷ÖÜÆÚ _nop_(); //µÈ´ýÒ»¸ö»úÆ÷ÖÜÆÚ SCL = 1; //¸ù¾ÝÉÏÊö¹æ¶¨£¬SCLӦΪ¸ßµçƽ_nop_(); //µÈ´ýÒ»¸ö»úÆ÷ÖÜÆÚ _nop_(); //µÈ´ýÒ»¸ö»úÆ÷ÖÜÆÚ _nop_(); //µÈ´ýÒ»¸ö»úÆ÷ÖÜÆÚ _nop_(); //µÈ´ýÒ»¸ö»úÆ÷ÖÜÆÚ ack_bit = SDA; //½ÓÊÜÉ豸£¨AT24Cxx)ÏòSDAË͵͵çƽ£¬±íʾÒѾ­½ÓÊÕµ½Ò»¸ö×Ö½Ú //ÈôË͸ߵçƽ£¬±íʾûÓнÓÊÕµ½£¬´«ËÍÒì³£SCL = 0; //SCLΪµÍµçƽʱ£¬SDAÉÏÊý¾Ý²ÅÔÊÐí±ä»¯(¼´ÔÊÐíÒÔºóµÄÊý¾Ý´«µÝ£©return ack_bit;// ·µ»ØAT24CxxÓ¦´ðλ}/***************************************************º¯Êý¹¦ÄÜ£ºÏòAT24CxxÖеÄÖ¸¶¨µØַдÈëÊý¾ÝÈë¿Ú²ÎÊý£ºadd (´¢´æÖ¸¶¨µÄµØÖ·£©£»dat(´¢´æ´ýдÈëµÄÊý¾Ý£©***************************************************/void WriteSet(unsigned char add, unsigned char dat)// ÔÚÖ¸¶¨µØÖ·addr´¦Ð´ÈëÊý¾ÝWriteCurrent{start(); //¿ªÊ¼Êý¾Ý´«µÝWriteCurrent(OP_WRITE); //Ñ¡ÔñÒª²Ù×÷µÄAT24CxxоƬ£¬²¢¸æÖªÒª¶ÔÆäдÈëÊý¾ÝWriteCurrent(add); //дÈëÖ¸¶¨µØÖ·WriteCurrent(dat); //Ïòµ±Ç°µØÖ·£¨ÉÏÃæÖ¸¶¨µÄµØÖ·£©Ð´ÈëÊý¾Ýstop(); //Í£Ö¹Êý¾Ý´«µÝdelaynms(4); //1¸ö×Ö½ÚµÄдÈëÖÜÆÚΪ1ms, ×îºÃÑÓʱ1msÒÔÉÏ}/***************************************************º¯Êý¹¦ÄÜ£º´ÓAT24CxxÖеĵ±Ç°µØÖ·¶ÁÈ¡Êý¾Ý³ö¿Ú²ÎÊý£ºx (´¢´æ¶Á³öµÄÊý¾Ý£© ***************************************************/unsigned char ReadCurrent(){unsigned char x;start(); //¿ªÊ¼Êý¾Ý´«µÝWriteCurrent(OP_READ); //Ñ¡ÔñÒª²Ù×÷µÄAT24CxxоƬ£¬²¢¸æÖªÒª¶ÁÆäÊý¾Ýx=ReadData(); //½«¶ÁÈ¡µÄÊý¾Ý´æÈëxstop(); //Í£Ö¹Êý¾Ý´«µÝreturn x; //·µ»Ø¶ÁÈ¡µÄÊý¾Ý}/***************************************************º¯Êý¹¦ÄÜ£º´ÓAT24CxxÖеÄÖ¸¶¨µØÖ·¶ÁÈ¡Êý¾ÝÈë¿Ú²ÎÊý£ºset_add³ö¿Ú²ÎÊý£ºx ***************************************************/unsigned char ReadSet(unsigned char set_add)// ÔÚÖ¸¶¨µØÖ·¶ÁÈ¡{start(); //¿ªÊ¼Êý¾Ý´«µÝWriteCurrent(OP_WRITE); //Ñ¡ÔñÒª²Ù×÷µÄAT24CxxоƬ£¬²¢¸æÖªÒª¶ÔÆäдÈëÊý¾ÝWriteCurrent(set_add); //дÈëÖ¸¶¨µØÖ·return(ReadCurrent()); //´ÓÖ¸¶¨µØÖ·¶Á³öÊý¾Ý²¢·µ»Ø}

1.6. 实例仿真","content"

编写程序代码,编译生成HEX文件,将HEX文件装载到proteus电路的单片机中,开始仿真,连接在P0口和P2口的两组发光二极管,看看他们显示的是否一致。

视频加载中...

1.7. 总结","content"

通过本例,我们学习了I2C总线的原理、电路设计、编程方法。

2、can总线详细介绍,一文看懂CAN总线

一文看懂CAN总线前言

CAN总线的应用在现在看来越来越广泛,我厂设备从最初的ARM9与ARM7平台、期间升级过度到CortexA8与Cortex M3平台,再到现在的Cortex M4平台,围绕CAN进行了一系列产品的开发,CAN总线的稳定性是毋庸置疑的。

CAN总线物理结构与特性CAN总线网络

CAN总线网络主要挂在CAN_H和CAN_L,各个节点通过这两条线实现信号的串行差分传输,为了避免信号的反射和干扰,还需要在CAN_H和CAN_L之间接上120欧姆的终端电阻,但是为什么是120欧姆呢?那是因为电缆的特性阻抗为120欧。

can总线详细介绍,基于proteus的51单片机开发实例27-I2C总线的读写

CAN收发器

CAN收发器的作用是负责逻辑电平和信号电平之间的转换。

can总线详细介绍,基于proteus的51单片机开发实例27-I2C总线的读写

即从CAN控制芯片输出逻辑电平到CAN收发器,然后经过CAN收发器内部转换将逻辑电平转换为差分信号输出到CAN总线上,CAN总线上的节点都可以决定自己是否需要总线上的数据。具体的管教定义如下:

can总线详细介绍,基于proteus的51单片机开发实例27-I2C总线的读写

信号表示

CAN总线采用不归零码位填充技术,也就是说CAN总线上的信号有两种不同的信号状态,分别是显性的(Dominant)逻辑0和隐形的(recessive)逻辑1,信号每一次传输完后不需要返回到逻辑0(显性)的电平。

位填充规则:发送器只要检测到位流里有5个连续相同值的位,便自动在位流里插入补充位。

观察下图:

can总线详细介绍,基于proteus的51单片机开发实例27-I2C总线的读写

可以看到上图中的当第一段为隐性(recessive),CAN_H和CAN_L电平几乎一样,也就是说CAN_H和CAN_L电平很接近甚至相等的时候,总线表现隐性的,而两线点位差较大时表现为显性的,按照定义的:

CAN_H-CAN_L < 0.5V 时候为隐性的,逻辑信号表现为"逻辑1"- 高电平。

CAN_H-CAN_L > 0.9V 时候为显性的,逻辑信号表现为"逻辑0"- 低电平。

下面将差分信号和显隐性之间对应关系总结为下表:

状态

逻辑信号

电压范围

显性Dominant

0

CAN_H-CAN_L > 0.9

隐性recessive

1

CAN_H-CAN_L < 0.5

由上面的分析我们可以知道:

CAN总线采用的"线与"的规则进行总线冲裁。即1&0=0;所以0为显性。

这句话隐含的意思是,如果总线上只要有一个节点将总线拉到低电平(逻辑0)即显性状态,总线就为低电平(逻辑0)即显性状态而不管总线上有多少节点处于传输隐性状态(高电平或是逻辑1),只有所有节点都为高(隐性),总线才为高,即隐性。

通讯速率与通讯距离

下面的SAE J2411为美国汽车标准。

类型

标准

最高速率

描述

高速CAN

CAN/ISO 11839-2

1Mbit/秒

最通用的CAN总线类型

低速CAN

ISO/ISO 11839-3

125Kbit/秒

容错,在一条总线短路的时候仍然能工作

单线CAN

SAE J2411

50Kbit/秒

高速模式可达到100Kbit/s主要用在汽车上,例如通用公司

CAN总线上任意两个节点的最大传输距离与其位速率有关,如下表:

位速率/kbps

1000

500

250

125

100

50

20

10

5

最大距离/m

40

130

270

530

620

1300

3300

6700

10000

这里的最大通信距离指的是同一条总线上两个节点之间的距离。可以看到速率越低通讯距离就越远,也就是说CAN总线的通讯距离和波特率成反比。在位速率为5千比特位每秒的时候达到最大的传输距离10公里。其中一般的工程中比较常用的为500K每秒的通讯速率。这个速率在实际测试的时候也是非常可靠的。

can总线详细介绍,基于proteus的51单片机开发实例27-I2C总线的读写

如果想要更远的传输(大于10公里);可以考虑用多个CAN控制器连接或是加其他通讯协议(如485或是TCP/IP)的接口芯片组成的一个设备,这样就可实现长距离的通讯需求。

CAN总线关键概念仲裁

只要总线空闲,总线上任何节点都可以发送报文,如果有两个或两个以上的节点开始传送报文,那么就会存在总线访问冲突的可能。但是CAN使用了标识符的逐位仲裁方法可以解决这个问题。

can总线详细介绍,基于proteus的51单片机开发实例27-I2C总线的读写

在仲裁期间,每一个发送器都对发送的电平与被监控的总线电平进行比较。如果电平相同,则这个单元可以继续发送。如果发送的是一"隐性"电平而监视到的是一"显性"电平,那么这个节点失去了仲裁,必须退出发送状态。如果出现不匹配的位不是在仲裁期间则产生错误事件。

帧ID越小,优先级越高。由于数据帧的RTR位为显性电平,远程帧为隐性电平,所以帧格式和帧ID相同的情况下,数据帧优先于远程帧;由于标准帧的IDE位为显性电平,扩展帧的IDE位为隐形电平,对于前11位ID相同的标准帧和扩展帧,标准帧优先级比扩展帧高。

CAN总线协议

CAN 总线是一个广播类型的总线,所以任何在总线上的节点都可以监听总线上传输的数据。也就是说总线上的传输不是点到点的,而是一点对多点的传输,这里多点的意思是总线上所有的节点。但是总线上的节点如何知道那些数据是传送给自己的呢?CAN总线的硬件芯片提供了一种叫做本地过滤的功能,通过这种本地过滤的功能可以过滤掉一些和自己无关的数据,而保留一些和自己有关的信息。

CAN消息机制

CAN标准定义了四种消息类型,每条消息用一种叫做比特位仲裁(Arbitration)机制来控制进入CAN总线,并且每条消息都标记了优先权。另外CAN标准还定义了一系列的错误处理机制。

CAN报文的四种消息类型:

数据帧:数据帧将数据从发送器传输到接收器。

远程帧:总线单元发出远程帧,请求发送具有同一标识符的数据帧。

错误帧:任何单元检测到总线错误就发出错误帧。

过载帧:过载帧用在相邻数据帧或远程帧之间的提供附加的延时。

数据帧

下图为基本的格式:

can总线详细介绍,基于proteus的51单片机开发实例27-I2C总线的读写

CAN总线中有标准帧和扩展帧两种格式,两种格式不同的地方在于仲裁域格式的不同,看下面两个表格可以很清楚的看出两者的不同,下面第一个表是标准帧(CAN2.0 A),第二个为扩展帧(CAN2.0 B):

can总线详细介绍,基于proteus的51单片机开发实例27-I2C总线的读写

下面为扩展帧格式(CAN2.0B):

can总线详细介绍,基于proteus的51单片机开发实例27-I2C总线的读写

其中

SOF为帧开始

SRR为"替代远程请求位

IDE为扩展标识符位

RTR为远程传输请求位

CRC delimiter 为CRC定界符。

ACK delimiter 为应答定界符.

我们看到上图中的基本帧格式可以总结为以下几个域:

描述

仲裁域

仲裁域决定了当总线上两个或是多个节点争夺总线时的优先权。

数据域

包含了0到8字节的数据。

CRC域

包含了15位的校验和,校验和用来做错误检测。

应答槽

任何一个已经正确接收到消息的控制器在每一条消息的末端发送一个应答位,发送器检查消息是否存在应答位,如果没有就重发消息。

远程帧

作为数据接收器的站,通过发送远程帧,可以启动其资源节点传送它们各自的数据。远程帧和数据帧非常类似,只是远程帧没有数据域。

can总线详细介绍,基于proteus的51单片机开发实例27-I2C总线的读写

上图就是远程帧的帧格式,它相对与数据帧没有远程帧,但是要注意发送远程帧的时候RTR位要置1,表示发送的是远程帧。下图更加清晰了呈现了这种结构。

can总线详细介绍,基于proteus的51单片机开发实例27-I2C总线的读写

错误帧

错误帧是当总线的某一个节点检测到错误后发送出来的,它会引起所有节点检测到一个错误,所以当有任何一个节点检测到错误,总线上的其他节点也会发出错误帧。CAN总线设计了一套详尽的错误计数机制来确保不会由于任何一个节点反复的发送错误帧而导致CAN总线的崩溃。

can总线详细介绍,基于proteus的51单片机开发实例27-I2C总线的读写

如上图所示错误标志和错误定界符组成,高低代表分别代表隐性和显性,其中错误标志为所有节点发过来的错误标志的叠加(Superposition)。下图更为清楚的看出各个数据位的分布:

can总线详细介绍,基于proteus的51单片机开发实例27-I2C总线的读写

下面通过以下数据结构框图概括各个部分的定义:

can总线详细介绍,基于proteus的51单片机开发实例27-I2C总线的读写

错误标志有两种形式:

主动错误标志,它由6个连续的显性位0组成,它是节点主动发送的错误标志。

被动错误标志,它由6个连续的隐性位1组成,除非被其他节点的显性位覆盖。

刚才说到一个节点上检测到错误会导致总线上所有的节点都会检测到错误并发送错误标志,这是为什么呢?

因为单一节点上的错误标志格式违背了从帧起始到CRC界定符的位填充规则,也破坏了ACK域或帧结尾的固定格式。下面简要说下位填充规则。

位填充规则:发送器只要检测到位流里有5个连续相同值的位,便自动在位流里插入补充位。

注意:位填充规则只是针对数据帧和远程帧,错误帧和过载帧格式固定。

所以所有其他的节点会检测到错误条件并且开始发送错误标志,因此错误帧就是各个站的不同错误标志叠加在一起的结果。

当某个节点发送错误帧(带有错误标志),其他节点收到了错误帧,检测到错误条件,就通过发送"被动错误标志"的错误帧来提示错误。

错误定界符:

传送了错误标志以后,每一个站就发送一个隐性位,并一直监视总线直到检测出一个隐性位为止,然后就开始发送其余7个隐性位。

过载帧

过载帧是接收节点用来向发送节点告知自身接收能力的帧。

过载帧,意思就是某个接收节点来不及处理数据了,希望其他节点慢点发送数据帧或者远程帧,所以告诉发送节点,我已经没有能力处理你发送过来的数据了。

过载帧跟错误帧结构类似包括过载标志和过载定界符,有3中情况会引起过载:

接收器内部的原因,它需要延迟下一个数据帧或是远程帧。

在间歇字段(看下面的帧间空间)的第一位和第二位检测到一个显性位(间歇字段都是隐性位的)

如果CAN节点在错误界定符或是过载界定符的第八位(最后一位)采样到一个显性位逻辑0,节点会发送一个过载帧,错误计数器不会增加。

can总线详细介绍,基于proteus的51单片机开发实例27-I2C总线的读写

can总线详细介绍,基于proteus的51单片机开发实例27-I2C总线的读写

上图中很清晰的表示了过载标志有6个显性位组成,而叠加部分和"主动错误"标志一样,过载的标志破坏的是间歇域的固定格式。所以导致其他的节点都检测到过载条件,并一同发出过载标志。

过载定界符:

也就是上图的过载结束符,过载标志被传送以后,节点就一直监听着总线,直到检测到有一个从显性位到隐性位的跳变为止。当从总线上检测到这样的跳变,则就标志着每一个节点都完成了各自过载标志的发送,并开始同时发送其余7个隐性位。

帧间空间(Interframe Space)

帧间空间说白了就是帧与帧之间的间隔,但是这种间隔在CAN的帧中只存在于数据帧和远程帧,其他的帧就不一定是帧间空间隔开的,而是其他形式,或是直接是没有间隔,例如过载帧和错误帧之间就没帧间空间。过个过载帧之间有间隔但是不是有帧间空间隔开的。

这里所说的针间空间包括"间歇"、"总线空闲"的位域。如果是发送前一报文的"被动错误"的站,则还包括叫做"挂起传输"的位域。

若不是"被动错误"的站,或作为前一报文的接收器的站,帧间空间格式为下图:

can总线详细介绍,基于proteus的51单片机开发实例27-I2C总线的读写

can总线详细介绍,基于proteus的51单片机开发实例27-I2C总线的读写

若是"被动错误"的站,如果想要发送8个隐性电平,在发送其他帧,帧间空间格式为下图,即包括了挂起传输,

can总线详细介绍,基于proteus的51单片机开发实例27-I2C总线的读写

可以看到

间歇字段有3个隐性位。

特别的在间歇期间,所有的节点都不允许传送数据帧和远程帧,唯一看做的是标示一个过载条件。

总线空闲:

只要总线空闲,任何节点就可以往总线发送数据,并且是开始于间歇之后的第一个位。一旦总线上检测到显性位即逻辑"0",可以认为是帧的开始。

1.

挂起传输

2.

"被动错误"的节点发送报文之后,在下一个报文开始传送之前或是确认总线空闲之前发出8个隐性位跟随在间歇的后面。如果这个时候有一个报文从其他的节点发过来,则这个节点就成为了接收器。

错误处理机制错误侦测

下标是几个错误类型:

错误类型

出错条件

出错域

帧测单元

比特错误

bit error

发送的位值与所监控的位值不相符合(填充比特和ACK比特除外)

数据帧(SOF~EOF)

远程帧(SOF~EOF)

错误帧

过载帧

发送单元

接收单元

填充错误stuff error

侦测到6个连续相同的电平

数据帧(SOF~CRC)

远程帧(SOF~CRC)

发送单元

接收单元

CRC 错误

计算结果和接收到的CRC不同

数据帧(CRC)

远程帧(CRC)

接收单元

格式错误

Form Error

某个固定的格式位置出现无效的比特

数据帧:

(CRC Delimiter, ACK Delimiter EOF)

远程帧:

(CRC Delimiter,ACK Delimiter)

错误帧:

(Error Frame Delimiter)

过载帧:

(Overload Delimiter)

接收单元

应答错误

Acknowledgment

发送端在应答间隙所监视的位不为显性,即逻辑0,发送器就检测到一个应答错误。

数据帧(ACK slot)

远程帧(ACK slot)

发送单元

错误计数

下面是错误计数表:

错误条件

Transmit Error Counter

Receive Error Counter

1

RECEIVER端侦测到一个位Error错误,除了发送ACTIVE ERROR FLAG 和OVERLOAD FLAG

-

1

2

TRANSMITIER 发送ERROR FLAG

8

3

TRANSMITTER发送ACTIVE ERROR FLAG OVERFLAG时侦测到BIT ERROR

8

4

当RECEIVER发送ACTIVE ERROR FLAG或OVERFLAG时侦测到BIT ERROR

8

5

一个帧被成功发送之后(取得ACK并且知道END OF FRAME完成都没有错误)

-1 IF TEC=0,TEC will not be changed

-

6

一个帧被成功接收(知道ACK域都没有检测到错误,并成功发送ACK比特)

-

1. if 1 <= REC <= 127 -> REC-1

2. if REC = 0 -> REC = 0

3. if REC > 127 -> REC = a value

Between 119 to 127

7

在总线上检测到128此连续的11个1,"bus off"的节点允许变成不再是"bus off"

Cleared to TEC = 0

Cleared to REC = 0

错误抑制

为防止某些节点自身出错而一直发送错误帧,干扰其他节点通信,CAN协议规定了节点的3种状态及行为,如下图:

can总线详细介绍,基于proteus的51单片机开发实例27-I2C总线的读写

一个节点挂到CAN总线上之后,处于ACTIVE状态;TEC>127或者REC>127导致节点进入passive状态;TEC>255之后节点处于bus off状态,就是不允许再往bus上发送东西了;处于bus off状态的节点,在检测到128个连续的11个1之后将回到active状态。

本文关键词:can总线百度百科,can总线特点及原理介绍,can总线hl,can总线简介,can总线是干嘛的。这就是关于《can总线详细介绍,基于proteus的51单片机开发实例27-I2C总线的读写》的所有内容,希望对您能有所帮助!更多的知识请继续关注《犇涌向乾》百科知识网站:http://www.029ztxx.com!

总线

最新文章