×
单片机 > 单片机程序设计 > 详情

ATMEGA48多机通讯程序

发布时间:2020-06-04 发布时间:
|

/*  
    通讯规则: 
  1:时钟7.3728 MHz/波特率9600/9个数据位/奇校验/1个停止位/硬件多机通讯功能/ 
  2:通讯连接采用硬件MAX485,双向单工 
  3:每个上行/下行的数据包的字节个数都是一样的(通讯数据量) 
  4:每个上行/下行的数据包都采用CRC8校验 

  5:数据接收采用中断+查询的方式 
  6:总是由主机向从机发送一个数据包,从机收到数据包后向主机回复一个数据包 
  7:不管是主机还是从机,如果收到的数据包有任何错误,都将丢弃该数据包,等效于没有接收 
  8:从机之间不能相互通讯,必须通过主机才能交换数据 
  9:无效地址是0,主机地址是1,从机地址是2.3.4......广播地址是255 
*/ 
#include   
#include   
#define   amount 10   //设定通讯数据量(包括1个地址帧,n个数据帧,1个校验帧) 
#include  
#include <1wire.h>    //CRC校验函数就在这个文件里面    

unsigned char  send[amount];           //发件箱 
unsigned char inbox[amount];           //收件箱 
unsigned char n=0;                     //记忆中断次数 

//-------------------------------------------------------------------- 
interrupt[19] Rxd_isr(void)            //接收中断 
{   
  unsigned char ERROR=0;  
  if( UCSR0A&4 || UCSR0A&16 ) ERROR=1; //奇偶效验错误或者帧错误就记录下来    
  inbox[n]=UDR0;                       //保存到收件箱 
  n++;                                 //记忆中断次数 
  if(ERROR) inbox[0]=0;                //如果通讯有错,收件箱的地址帧就标记成无效地址0 
}  

//--------------------------------------------------------------------- 
void main(void) 
{  
  USARTinit();                         //串口初始化 
  UCSR0A=0;                            //主机关闭地址筛选功能(多机通讯功能) 
  #asm("sei")                          //打开全局中断  
  while(1) 
  {  
    //-------------与从机2对话,与其他从机对话与下面的程序类似-------------------  
    n=0;                               //中断次数清0  
    inbox[0]=0;                        //收件箱地址清0 
    send[0]=2;                         //改变这个地址就可以实现与某个从机对话  
    send[amount-1]=w1_dow_crc8(send,amount-1);        //计算发件箱的crc8校验码 
    TXD(send);                         //将发件箱的数据send[]发送出去; 
     
    //等待,从机接收到数据后会回复数据的,如果是10个字节数据量,不能少于13ms!!!  
    //这个时间由人工计算,要考虑从机由于各种中断延长回复时间的可能 
    delay_ms(30); 
     
    //如果收件箱已经收到amount个数据,并且crc8校验成功就... 
    if(n==amount && inbox[amount-1]==w1_dow_crc8(inbox,amount-1)) 
      {  
        if(inbox[0]==1)                //如果收件箱地址帧属于本机就运行下面的测试代码 
          {  
            DDRD.3=1; 
            PORTD.3=1; delay_ms(50); 
            PORTD.3=0; delay_ms(950); 
          } 
           
        if(inbox[0]==255) 
          { 
            //请在这里添加收到广播数据的处理程序 
          } 
      }               
  } 
}   //end 





//************************************************************************************************** 

从机程序 

//************************************************************************************************** 


#include  
#include  
#define   amount 10    //设定通讯数据量(包括1个地址帧,n个数据帧,1个校验帧)   
#include  
#include <1wire.h> 
#define   address 2    //请在这里设定本机地址 

unsigned char  send[amount];               //发件箱 
unsigned char inbox[amount];               //收件箱 
unsigned char n=0;                         //记忆中断次数 

//-------------------------------------------------------------------- 
interrupt[19] Rxd_isr(void)                //接收中断 
{   
  unsigned char ERROR=0;  
  if( UCSR0A&4 || UCSR0A&16 ) ERROR=1;     //记录奇偶效验错误或者帧错误  
  inbox[n]=UDR0;                           //把接收到的数据保存到收件箱 
  n++;                                     //记忆接收的次数 
  if(ERROR)                                //如果通讯有错.... 
    { 
      n=0;                                 //接收计数清0  
      inbox[0]=0;                          //把地址改为无效地址0 
      UCSR0A|=0x01;                        //重新打开接收器的地址帧筛选功能 
    }  
   
  //如果地址匹配本机或者是广播地址就关闭地址筛选(多机通讯)功能   
  if(inbox[0]==address ||inbox[0]==255) UCSR0A&=254;  
   
  if(n==amount)                            //接收到amount个数据以后... 
    {   
      n=0;                                 //接收计数清0  
      UCSR0A|=0x01;                        //重新打开接收器的地址帧筛选功能 
      if(inbox[amount-1]==w1_dow_crc8(inbox,amount-1))   //如果crc8校验正确就... 
        {    
          if(inbox[0]==address)            //如果地址匹配本机就回复数据 
            {   
              send[0]=1;                   //发件箱地址指向主机 
              send[amount-1]=w1_dow_crc8(send,amount-1); //产生发件箱的crc8校验码 
              TXD(send);                   //发送发件箱的数据包send[]  
              //请在这里备份你的收件箱信息 
            } 
          if(inbox[0]==255)                //如果是广播地址就... 
            { 
              //请在这里添加你的代码 
              //收到广播数据请不要回复 
            } 
        } 
    } 
}  

//--------------------------------------------------------------------- 
void main(void) 
{  
  USARTinit();                             //串口初始化 
  UCSR0A=0x01;                             //从机打开地址帧筛选功能(多机通讯模式) 
  #asm("sei")                              //打开全局中断  
  while(1) 
  {   
    //请在这里添加你的代码         
  } 
}   //end 


//********************************************************************************************* 

 头文件 

//********************************************************************************************* 

//波特率9600/9个数据位/1个停止位/奇校验/收发开启/接收中断 
void USARTinit(void) 
     {   
       UCSR0B=0x9C; 
       UCSR0C=0x36; 
       UBRR0L=0x2F; 
       PORTD.4=0;                      //MAX485平时工作在接收状态 
       DDRD.4=1; 
     }  

//----------------------------------------------------------- 
//从数组datas[]的首地址开始发送amount个数据,其中第0个数据是地址帧,其他是数据帧 
void TXD(unsigned char *datas) 
     { 
       unsigned char i=0;   
       PORTD.4=1;                      //使MAX485处于发送状态 
       while(i            {   
              if(i==0) UCSR0B|=1; else UCSR0B&=254;    
              UDR0=*(datas+i);         //装载数据开始发送  
              while((UCSR0A&64)==0);   //等待发送结束 
              UCSR0A|=64;              //清除发送结束标志 
              i++;                     //发送次数统计 
            }  
       PORTD.4=0;                      //使MAX485处于接收状态 
     }    



『本文转载自网络,版权归原作者所有,如有侵权请联系删除』

热门文章 更多
C51 特殊功能寄存器SFR的名称和地址