0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內(nèi)不再提示

CAN總線收發(fā)節(jié)點設計

CHANBAEK ? 來源:愛研究的小胡同學 ? 作者: 從嵌入式到微電子 ? 2023-11-08 15:29 ? 次閱讀

寫在前面

這是微機接口的一個項目作業(yè)。

這段時間一直在宿舍隔離,沒辦法進行焊接和測試,但原理和代碼已經(jīng)在學習板子上經(jīng)過驗證。

設計目標

CAN工業(yè)現(xiàn)場大量應用,尤其是汽車工業(yè),設計一種CAN總線收發(fā)節(jié)點,該節(jié)點自身帶8路模擬信號采集,采集結(jié)果通過CAN總線發(fā)送到上位機并顯示。

技術要求

(1)系統(tǒng)以51單片機控制器,和具有CAN接口的器件能通信;

(2)CAN控制器用SJA1000,總線收發(fā)器用PCA82C250;

(3)節(jié)點帶8路模擬信號采集,信號范圍0-5V;

(4)用USB轉(zhuǎn)CAN模塊,通過USB口接收CAN節(jié)點發(fā)送的數(shù)據(jù),驗證結(jié)果是否正確。

提交材料

(1)提交紙質(zhì)版設計報告1份(包括測量原理、主要電路設計、主要器件選型、程序設計原理、關鍵程序設計舉例等);

(2)電子版設計報告、系統(tǒng)電氣原理圖、完整的程序代碼

(3)提交實物1套,能現(xiàn)場演示(所需的can調(diào)試助手 can總線分析儀可以找我);

(4)該題目4人完成(1人負責硬件設計、1人負責單片機軟件設計、1人負責調(diào)試、1人負責設計報告的整理編輯以及答辯PPT制作),在設計報告上寫清楚每人所承擔的工作。

項目實現(xiàn)

設計成果展示

實物展示

原理圖設計

圖片

PCB設計

圖片

圖片

上位機效果

圖片

圖片

測量原理

ADC數(shù)模轉(zhuǎn)換原理,這里采用PCF8591AD采樣芯片,測量原理如下:

圖片

通過引腳AIN0、AIN1、AIN2、AIN3輸入的模擬信號(電壓),經(jīng)過模擬信號多路復用器、采樣與保持、比較器,把處理后的數(shù)據(jù)放入ADC數(shù)據(jù)寄存器中,通過I2C總線接口傳遞給51主控芯片。

主要電路設計

供電與程序燒錄電路

圖片

考慮到板子尺寸的限制以及器件的選型,這里采用UCB轉(zhuǎn)串口芯片CH340,在USB供電的同時,又可通過串口進行程序的燒錄,一路雙用。

VCC直接作為5V電源輸入,在串口芯片那邊有一個保險絲12V/1000mA的保險絲進行保護,并通過電容進行濾波與穩(wěn)壓,來防止熱插拔效應的干擾。

撥動電源開關,可看到電源指示燈亮起。

51主控芯片附近電路

圖片

主控芯片選取的是STC90C51RD+,國產(chǎn)51MCU芯片,簡單易學,入門容易。工作電壓:5.5V-3.4V,工作溫度范圍:-40-+85°C,工作頻率范圍:0-40MHz,用戶應用程序空間4K,片上集成1280字節(jié)RAM,32個通用I/O口,4個外部中斷?;究梢詽M足項目要求,實現(xiàn)對應的功能。

復位電路

圖片

采用阻容復位電路,電容C7是10μF,電阻R7是10K。

晶振電路

圖片

使用外部晶振12MHz,此時電容選取47pF。

AD采樣電路

圖片

采用PCF8591這款芯片,該芯片具有4路模擬輸入通道,8位AD采樣精度,以及一個DA輸出。

電路部分,使用兩個電位器作為采樣目標,通過改變電位器阻值來改變采樣的數(shù)值,并通過開關來切換AD采樣的通道。

把采集到的數(shù)據(jù)存放至8位的數(shù)據(jù)寄存器中,通過I2C總線傳輸?shù)?1主控芯片中。

利用該芯片的一個DA輸出通道,可以連接一個LED的燈,轉(zhuǎn)動電位器,可以觀察燈的的亮度發(fā)生細微的變化,更加的直觀。

數(shù)碼管驅(qū)動電路

圖片

為了便于調(diào)試,這里使用8位共陰數(shù)碼管進行實時顯示,數(shù)碼管驅(qū)動芯片選擇74HC573,作為最常見的鎖存芯片,在這里一個作為位鎖存,一個作為段鎖存。

數(shù)碼管這里采用的是共陰數(shù)碼管,兩個4位的數(shù)碼管,合成一個8位的數(shù)碼管。

CAN總線通訊電路

圖片

根據(jù)實際的項目需求,CAN總線電路采用SJA1000作為CAN控制器,PCA82C250作為CAN總線收發(fā)器。

但因單獨購買芯片沒有相關渠道,轉(zhuǎn)而選擇集成化的CAN通訊模塊。

主要器件選型

器件選型方面本著有現(xiàn)成的就使用現(xiàn)成的、能簡單實現(xiàn)的功能就不做的復雜的原則。器件選型上可以分為兩類:芯片類,其他電子器件類。

芯片類

51主控芯片STC90C516RD+

USB轉(zhuǎn)串口芯片CH340G

AD采樣芯片PCF8591

鎖存器芯片74HC573

CAN通訊模塊

其他器件

開關

按鍵

四位共陰數(shù)碼管

晶振

供電USB接口

電阻、電容、二極管、LED

程序設計原理

程序框架

圖片

整個程序框架包含了:main.c、display.c、i2c.c、uart.c、delay.c。

圖片

主函數(shù)模塊main.c,延時函數(shù)模塊delay.c、數(shù)碼管驅(qū)動函數(shù)模塊display.c、i2c驅(qū)動函數(shù)模塊i2c.c、串口通訊函數(shù)模塊uart.c。相關函數(shù)都用.h文件進行封裝,提供相關的接口,供主函數(shù)調(diào)用。

主函數(shù)模塊main.c
/*
    Date:2022.03.22
    Author:
    Target:主函數(shù)
*/
#include < reg52.h >                
#include "i2c.h"
#include "delay.h"
#include "display.h"
#include < uart.h >


#define AddWr 0x90   //寫數(shù)據(jù)地址 
#define AddRd 0x91   //讀數(shù)據(jù)地址

extern bit ack;
bit ReadADFlag;

unsigned char VoltData[5]; //存儲電壓的全局變量,用于串口通訊

unsigned char numback(unsigned char s);
unsigned char ReadADC(unsigned char Chl);
bit WriteDAC(unsigned char dat);
/*------------------------------------------------
              主程序
------------------------------------------------*/
main()
{
 unsigned char num=0,num0=0,num1=0,num2=0,num3=0,i;
 Init_Timer0();
 DelayMs(20);

 InitUART();

while (1)         //主循環(huán)
  {
 if(ReadADFlag)
  {
  ReadADFlag=0; 
//連續(xù)讀5次,輸入通道后多讀幾次,取最后一次值,以便讀出穩(wěn)定值
 for(i=0;i< 5;i++)
    num0=ReadADC(0);
  num0=num0*5*10/256;// x10表示把實際值擴大10,如4.5 變成 45 方便做下一步處理 x5 表示基準電壓5V 
  TempData[0]=dofly_DuanMa[num0/10]|0x80;  
  TempData[1]=dofly_DuanMa[num0%10];

 for(i=0;i< 5;i++)
   num1=ReadADC(1);
 num1=num1*5*10/256;   // x10表示把實際值擴大10,如4.5 變成 45 方便做下一步處理
 TempData[2]=dofly_DuanMa[num1/10]|0x80;    
 TempData[3]=dofly_DuanMa[num1%10];

 for(i=0;i< 5;i++)
    num2=ReadADC(2);
 num2=num2*5*10/256;   // x10表示把實際值擴大10,如4.5 變成 45 方便做下一步處理
 TempData[4]=dofly_DuanMa[num2/10]|0x80;    
 TempData[5]=dofly_DuanMa[num2%10];

 for(i=0;i< 5;i++)
     num=ReadADC(3);
 num3=num3*5*10/256;   // x10表示把實際值擴大10,如4.5 變成 45 方便做下一步處理
 TempData[6]=dofly_DuanMa[num3/10]|0x80;    
 TempData[7]=dofly_DuanMa[num3%10];
 //主循環(huán)中添加其他需要一直工作的程序

 VoltData[0]=num0;
 VoltData[1]=num1;
 VoltData[2]=num2;
 VoltData[3]=num3;
 VoltData[4]=0xff;

 SendStr1(VoltData);
    DelayMs(240);//延時循環(huán)發(fā)送
    DelayMs(240);
   }
 /*
   SendStr1(VoltData);
    DelayMs(240);//延時循環(huán)發(fā)送
    DelayMs(240);
 */
  }
}
/*------------------------------------------------
             讀AD轉(zhuǎn)值程序
輸入參數(shù) Chl 表示需要轉(zhuǎn)換的通道,范圍從0-3
返回值范圍0-255
------------------------------------------------*/
unsigned char ReadADC(unsigned char Chl)
 {
  unsigned char Val;
   Start_I2c();               //啟動總線
   SendByte(AddWr);             //發(fā)送器件地址
     if(ack==0)return(0);
   SendByte(Chl);            //發(fā)送器件子地址
     if(ack==0)return(0);
   Start_I2c();
   SendByte(AddRd);
      if(ack==0)return(0);
   Val=RcvByte();
   NoAck_I2c();                 //發(fā)送非應位
   Stop_I2c();                  //結(jié)束總線
  return(Val);
 }
/*------------------------------------------------
               寫入DA轉(zhuǎn)換數(shù)值
輸入?yún)?shù):dat 表示需要轉(zhuǎn)換的DA數(shù)值,范圍是0-255
------------------------------------------------*/
/*bit WriteDAC(unsigned char dat)
{
   Start_I2c();               //啟動總線
   SendByte(AddWr);             //發(fā)送器件地址
     if(ack==0)return(0);
   SendByte(0x40);            //發(fā)送器件子地址
     if(ack==0)return(0);
   SendByte(dat);             //發(fā)送數(shù)據(jù)
     if(ack==0)return(0);
   Stop_I2c();  
}*/

c

延時函數(shù)模塊delay.c
/*
    Date:2022.03.22
    Author:
    Target:提供延時
*/

#include< delay.h >

//uS延時函數(shù),輸入?yún)?shù)t,無返回值,延時時間=t*2+5 uS
void DelayUs2x(unsigned int t)
{
    while(--t);
}

//mS延時函數(shù),輸入?yún)?shù)t,無返回值,延時時間1mS
void DelayMs(unsigned int t)
{
    while(t--)
    {
        DelayUs2x(245);
        DelayUs2x(245);
    }
}
#ifndef _DELAY_H_
#define _DELAY_H_
void DelayUs2x(unsigned int t);
void DelayMs(unsigned int t);
#endif
數(shù)碼管驅(qū)動函數(shù)模塊display.c
/*
    Date:2022.03.22
    Author:
    Target:數(shù)碼管驅(qū)動
*/

#include< display.h >
#include< delay.h >

#define DataPort P0 //定義數(shù)據(jù)端口 程序中遇到DataPort 則用P0 替換
//sbit LATCH1 = P2^0;//定義鎖存使能端口 段鎖存
//sbit LATCH2 = P2^3;//                 位鎖存
extern bit ReadADFlag;//extern聲明,不是定義,外部變量
unsigned char code dofly_DuanMa[10] = {0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f};// 顯示段碼值0~9
unsigned char code dofly_WeiMa[] = {0xfe,0xfd,0xfb,0xf7,0xef,0xdf,0xbf,0x7f};//分別對應相應的數(shù)碼管點亮,即位碼
unsigned char TempData[8]; //存儲顯示值的全局變量

/*
顯示函數(shù),動態(tài)掃描數(shù)碼管,
參數(shù)FirstBit 表示需要顯示的是第一位,比如0就是從第一個數(shù)碼管顯示,2就是從第三個數(shù)碼管顯示,
參數(shù)Num表示要顯示的位數(shù),也就是幾個數(shù)碼管顯示,如要顯示兩位數(shù),就應該輸入2
*/
void Display(unsigned char FirstBit,unsigned char Num)
{
    static unsigned char i = 0;

    DataPort=0;   //清空數(shù)據(jù),防止有交替重影
    LATCH1=1;     //段鎖存
    LATCH1=0;

    DataPort=dofly_WeiMa[i+FirstBit]; //取位碼 
    LATCH2=1;     //位鎖存
    LATCH2=0;

    DataPort=TempData[i]; //取顯示數(shù)據(jù),段碼
    LATCH1=1;     //段鎖存
    LATCH1=0;

    i++;
    if(i==Num)
        i=0;
}

/*  定時器初始化  */
void Init_Timer0(void)
{
    TMOD |= 0x01;      //使用模式1,16位定時器,使用"|"符號可以在使用多個定時器時不受影響
    EA=1;            //總中斷打開
    ET0=1;           //定時器中斷打開
    TR0=1;           //定時器開關打開
}

/*  定時器中斷子程序    */
void Init_Timer0_isr(void) interrupt 1
{
    static unsigned int num;
    TH0=(65536-2000)/256;          //重新賦值 高位 低位
    TL0=(65536-2000)%256;         //可以理解成,提前減去2000,就是2ms倒計時

    Display(0,8);       // 調(diào)用數(shù)碼管掃描
    num++;
    if(num==50)        //中斷50次,大致100ms
    {
        num = 0;
        ReadADFlag=1;//AD標志位1
    }
}
#include< reg52.h >

#ifndef __DISPLAY_H__
#define __DISPLAY_H__

#define DataPort P0 //定義數(shù)據(jù)端口 程序中遇到DataPort 則用P0 替換
sbit LATCH1=P2^2;//定義鎖存使能端口 段鎖存
sbit LATCH2=P2^3;//               位鎖存


extern unsigned char TempData[8]; //存儲顯示值的全局變量
extern unsigned char code dofly_DuanMa[10];

void Display(unsigned char FirstBit,unsigned char Num);

void Init_Timer0(void);

#endif
i2c驅(qū)動函數(shù)模塊i2c.c
/*
    Date:2022.03.22
    Author:
    Target:i2c驅(qū)動
*/

#include < i2c.h >
#include < delay.h >

#define  _Nop()  _nop_()  //定義空指令 一個空指令大致為1us

bit ack;

sbit SDA=P2^1;//數(shù)據(jù)線
sbit SCL=P2^0;//時鐘

/*  啟動i2c總線  */
void Start_I2c()
{
  SDA=1;   //發(fā)送起始條件的數(shù)據(jù)信號
  _Nop();
  SCL=1;
  _Nop();    //起始條件建立時間大于4.7us,延時
  _Nop();
  _Nop();
  _Nop();
  _Nop();    
  SDA=0;     //發(fā)送起始信號
  _Nop();    //起始條件鎖定時間大于4μ
  _Nop();
  _Nop();
  _Nop();
  _Nop();       
  SCL=0;    //鉗住I2C總線,準備發(fā)送或接收數(shù)據(jù)
  _Nop();
  _Nop();
}

/*  關閉i2c總線 */
void Stop_I2c()
{
  SDA=0;    //發(fā)送結(jié)束條件的數(shù)據(jù)信號
  _Nop();   //發(fā)送結(jié)束條件的時鐘信號
  SCL=1;    //結(jié)束條件建立時間大于4μ
  _Nop();
  _Nop();
  _Nop();
  _Nop();
  _Nop();
  SDA=1;    //發(fā)送I2C總線結(jié)束信號
  _Nop();
  _Nop();
  _Nop();
  _Nop();
}



/*  
發(fā)送字節(jié)數(shù)據(jù)
將數(shù)據(jù)c發(fā)送出去,可以是地址,也可以是數(shù)據(jù),發(fā)完后等待應答,并對
此狀態(tài)位進行操作.(不應答或非應答都使ack=0 假)     
發(fā)送數(shù)據(jù)正常,ack=1; ack=0表示被控器無應答或損壞。
*/
void  SendByte(unsigned char c)
{
 unsigned char BitCnt;

 for(BitCnt=0;BitCnt< 8;BitCnt++)  //要傳送的數(shù)據(jù)長度為8位
    {
     if((c<
#ifndef __I2C_H__
#define __I2C_H__  

#include < reg52.h >          //頭文件的包含
#include < intrins.h >

#define  _Nop()  _nop_()        //定義空指令

/*------------------------------------------------
                    啟動總線
------------------------------------------------*/
void Start_I2c();
/*------------------------------------------------
                    結(jié)束總線
------------------------------------------------*/
void Stop_I2c();
/*----------------------------------------------------------------
                 字節(jié)數(shù)據(jù)傳送函數(shù)               
函數(shù)原型: void  SendByte(unsigned char c);
功能:  將數(shù)據(jù)c發(fā)送出去,可以是地址,也可以是數(shù)據(jù),發(fā)完后等待應答,并對
     此狀態(tài)位進行操作.(不應答或非應答都使ack=0 假)     
     發(fā)送數(shù)據(jù)正常,ack=1; ack=0表示被控器無應答或損壞。
------------------------------------------------------------------*/
void  SendByte(unsigned char c);
/*----------------------------------------------------------------
                 字節(jié)數(shù)據(jù)傳送函數(shù)               
函數(shù)原型: unsigned char  RcvByte();
功能:  用來接收從器件傳來的數(shù)據(jù),并判斷總線錯誤(不發(fā)應答信號),
     發(fā)完后請用應答函數(shù)。  
------------------------------------------------------------------*/    
unsigned char  RcvByte();
/*----------------------------------------------------------------
                     應答子函數(shù)
原型:  void Ack_I2c(void);
----------------------------------------------------------------*/
void Ack_I2c(void);
/*----------------------------------------------------------------
                     非應答子函數(shù)
原型:  void NoAck_I2c(void);
----------------------------------------------------------------*/
void NoAck_I2c(void);
/*----------------------------------------------------------------
                    向無子地址器件發(fā)送字節(jié)數(shù)據(jù)函數(shù)               
函數(shù)原型: bit  ISendByte(unsigned char sla,ucahr c);  
功能:     從啟動總線到發(fā)送地址,數(shù)據(jù),結(jié)束總線的全過程,從器件地址sla.
           如果返回1表示操作成功,否則操作有誤。
注意:    使用前必須已結(jié)束總線。
----------------------------------------------------------------*/
bit ISendByte(unsigned char sla,unsigned char c);

/*----------------------------------------------------------------
                    向有子地址器件發(fā)送多字節(jié)數(shù)據(jù)函數(shù)               
函數(shù)原型: bit  ISendStr(unsigned char sla,unsigned char suba,ucahr *s,unsigned char no);  
功能:     從啟動總線到發(fā)送地址,子地址,數(shù)據(jù),結(jié)束總線的全過程,從器件
          地址sla,子地址suba,發(fā)送內(nèi)容是s指向的內(nèi)容,發(fā)送no個字節(jié)。
           如果返回1表示操作成功,否則操作有誤。
注意:    使用前必須已結(jié)束總線。
----------------------------------------------------------------*/
bit ISendStr(unsigned char sla,unsigned char suba,unsigned char *s,unsigned char no);

/*----------------------------------------------------------------
                    向無子地址器件讀字節(jié)數(shù)據(jù)函數(shù)               
函數(shù)原型: bit  IRcvByte(unsigned char sla,ucahr *c);  
功能:     從啟動總線到發(fā)送地址,讀數(shù)據(jù),結(jié)束總線的全過程,從器件地
          址sla,返回值在c.
           如果返回1表示操作成功,否則操作有誤。
注意:    使用前必須已結(jié)束總線。
----------------------------------------------------------------*/
bit IRcvByte(unsigned char sla,unsigned char *c);
/*----------------------------------------------------------------
                    向有子地址器件讀取多字節(jié)數(shù)據(jù)函數(shù)               
函數(shù)原型: bit  ISendStr(unsigned char sla,unsigned char suba,ucahr *s,unsigned char no);  
功能:     從啟動總線到發(fā)送地址,子地址,讀數(shù)據(jù),結(jié)束總線的全過程,從器件
          地址sla,子地址suba,讀出的內(nèi)容放入s指向的存儲區(qū),讀no個字節(jié)。
           如果返回1表示操作成功,否則操作有誤。
注意:    使用前必須已結(jié)束總線。
----------------------------------------------------------------*/
bit IRcvStr(unsigned char sla,unsigned char suba,unsigned char *s,unsigned char no);

#endif
串口通訊函數(shù)模塊uart.c
#include < reg52.h > //包含頭文件,一般情況不需要改動,頭文件包含特殊功能寄存器的定義                        
#include "delay.h"
#include < uart.h >

//串口初始化    
 void InitUART  (void)
{

    SCON  = 0x50;                // SCON: 模式 1, 8-bit UART, 使能接收  
    TMOD |= 0x20;               // TMOD: timer 1, mode 2, 8-bit 重裝
    TH1   = 0xFD;               // TH1:  重裝值 9600 波特率 晶振 11.0592MHz  
    TR1   = 1;                  // TR1:  timer 1 打開                         
    EA    = 1;                  //打開總中斷
    //ES    = 1;                  //打開串口中斷
}        



//發(fā)送一個字節(jié)
void SendByte1(unsigned char dat)
{
 SBUF = dat;
 while(!TI);
      TI = 0;
}


//發(fā)送一個字符串
void SendStr1(unsigned char *s)
{
 while(*s!=0xff)// ? 表示字符串結(jié)束標志,通過檢測是否字符串末尾
  {
  SendByte1(*s);
  s++;
  }
}
#ifndef __uart_H__
#define __uart_H__  

void InitUART  (void);

void SendByte1(unsigned char dat);

void SendStr1(unsigned char *s);

#endif
上位機部分程序
namespace 微機上位機
{
    public partial class Form1 : Form
    {
    //初始化
        private void Form1_Load(object sender, EventArgs e)            
        {
            comboBox1.Text = "COM1";
            comboBox2.Text = "9600";
            serialPort1.DataReceived += new System.IO.Ports.SerialDataReceivedEventHandler(senddata);
        }

    //接受數(shù)據(jù)
        private void senddata(object sender, System.IO.Ports.SerialDataReceivedEventArgs e)
        {
            length = serialPort1.BytesToRead;   //獲取緩沖區(qū)字節(jié)數(shù)
            serialPort1.Read(R_data, 0, length);
            this.Invoke(new EventHandler(display));
        }

    //顯示數(shù)據(jù)
        private void display(object sender, EventArgs e)
        {
            double[] sample = new double[8];
            sample[0] = (Convert.ToDouble(R_data[0])) / 10;
            sample[1] = (Convert.ToDouble(R_data[1])) / 10;
            sample[2] = (Convert.ToDouble(R_data[2])) / 10;
            sample[3] = (Convert.ToDouble(R_data[3])) / 10;
            sample[4] = (Convert.ToDouble(R_data[4])) / 10;
            sample[5] = (Convert.ToDouble(R_data[5])) / 10;
            sample[6] = (Convert.ToDouble(R_data[6])) / 10;
            sample[7] = (Convert.ToDouble(R_data[7])) / 10;

            textBox1.Text = sample[0].ToString();
            textBox2.Text = sample[1].ToString();
            textBox3.Text = sample[2].ToString();
            textBox4.Text = sample[3].ToString();
            textBox5.Text = sample[0].ToString();
            textBox6.Text = sample[1].ToString();
            textBox7.Text = sample[2].ToString();
            textBox8.Text = sample[3].ToString();
            textBox9.Text = sample[4].ToString();
            textBox10.Text = sample[5].ToString();
            textBox11.Text = sample[6].ToString();
            textBox12.Text = sample[7].ToString();
            ovalShape1.FillColor = Color.LightGreen;
            ovalShape2.FillColor = Color.LightGreen;
            ovalShape3.FillColor = Color.LightGreen;
            ovalShape4.FillColor = Color.LightGreen;
            ovalShape5.FillColor = Color.LightGreen;
            ovalShape6.FillColor = Color.LightGreen;
            ovalShape7.FillColor = Color.LightGreen;
            ovalShape8.FillColor = Color.LightGreen;
            ovalShape9.FillColor = Color.LightGreen;
            ovalShape10.FillColor = Color.LightGreen;
            ovalShape11.FillColor = Color.LightGreen;
            ovalShape12.FillColor = Color.LightGreen;

        }

    //開啟串口
        private void button1_Click_1(object sender, EventArgs e)
        {
            R_Flag = 1;
            serialPort1.ReceivedBytesThreshold = 4;
            serialPort1.RtsEnable = true;
            if (serialPort1.IsOpen)
            {
                try
                {
                    timer1.Stop();
                    serialPort1.Close();
                    button1.Text = "打開串口";
                }
                catch
                {
                    MessageBox.Show("端口錯誤", "Error");
                    button1.Text = "關閉串口";
                }
            }
            else
            {
                try
                {
                    serialPort1.PortName = comboBox1.Text;
                    serialPort1.BaudRate = Convert.ToInt16(comboBox2.Text, 10);
                    serialPort1.Parity = System.IO.Ports.Parity.None;
                    serialPort1.StopBits = System.IO.Ports.StopBits.One;
                    serialPort1.DataBits = 8;
                    serialPort1.Open();
                    timer1.Start();
                    button1.Text = "關閉串口";
                }
                catch
                {
                    MessageBox.Show("端口錯誤", "Error");
                    serialPort1.Close();
                    button1.Text = "打開串口";
                }
            }
        }

    //配置報文長度
        private void button2_Click(object sender, EventArgs e)
        {
            serialPort1.ReceivedBytesThreshold = Convert.ToInt16(textBox13.Text, 10);
        }

    }
}

關鍵程序設計

延時模塊

12MHz晶振,一個指令周期大約是1μs,這里封裝了兩個函數(shù),一個μs級別的,一個ms級別的。

//uS延時函數(shù),輸入?yún)?shù)t,無返回值,延時時間=t*2+5 uS
void DelayUs2x(unsigned int t)
{
    while(--t);
}

//mS延時函數(shù),輸入?yún)?shù)t,無返回值,延時時間1mS
void DelayMs(unsigned int t)
{
    while(t--)
    {
        DelayUs2x(245);
        DelayUs2x(245);
    }
}

數(shù)碼管驅(qū)動模塊

段碼位碼的的編寫
unsigned char code dofly_DuanMa[10] = {0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f};// 顯示段碼值0~9
unsigned char code dofly_WeiMa[] = {0xfe,0xfd,0xfb,0xf7,0xef,0xdf,0xbf,0x7f};//分別對應相應的數(shù)碼管點亮,即位碼
unsigned char TempData[8]; //存儲顯示值的全局變量

這里是使用到了一個小工具:段碼值分別編寫0-9的數(shù)值。

圖片

位碼則是8位,比如fd代表1111 1101

圖片

使用的芯片屬于鎖存器,打開或者關閉制定鎖存器,就可實現(xiàn)數(shù)據(jù)的顯示。

定時器模塊的調(diào)用
/*  定時器初始化  */
void Init_Timer0(void)
{
    TMOD |= 0x01;      //使用模式1,16位定時器,使用"|"符號可以在使用多個定時器時不受影響
    EA=1;            //總中斷打開
    ET0=1;           //定時器中斷打開
    TR0=1;           //定時器開關打開
}

/*  定時器中斷子程序    */
void Init_Timer0_isr(void) interrupt 1
{
    static unsigned int num;
    TH0=(65536-2000)/256;          //重新賦值 高位 低位
    TL0=(65536-2000)%256;         //可以理解成,提前減去2000,就是2ms倒計時

    Display(0,8);       // 調(diào)用數(shù)碼管掃描
    num++;
    if(num==50)        //中斷50次,大致100ms
    {
        num = 0;
        ReadADFlag=1;//AD標志位1
    }
}

定時器初始化,定時器有四個模式,這里選擇模式1,十六位定時器/計數(shù)器。

圖片

把數(shù)碼管掃描函數(shù),放到中斷函數(shù)中,每隔100ms掃描一次。

i2c模塊的編寫

數(shù)據(jù)線,時鐘線,主要參考時序圖,什么時候開始發(fā)送數(shù)據(jù),什么時候結(jié)束發(fā)送數(shù)據(jù)。一個空指令是1μs。

/*  啟動i2c總線  */
void Start_I2c()
{
  SDA=1;   //發(fā)送起始條件的數(shù)據(jù)信號
  _Nop();
  SCL=1;
  _Nop();    //起始條件建立時間大于4.7us,延時
  _Nop();
  _Nop();
  _Nop();
  _Nop();    
  SDA=0;     //發(fā)送起始信號
  _Nop();    //起始條件鎖定時間大于4μ
  _Nop();
  _Nop();
  _Nop();
  _Nop();       
  SCL=0;    //鉗住I2C總線,準備發(fā)送或接收數(shù)據(jù)
  _Nop();
  _Nop();
}

/*  關閉i2c總線 */
void Stop_I2c()
{
  SDA=0;    //發(fā)送結(jié)束條件的數(shù)據(jù)信號
  _Nop();   //發(fā)送結(jié)束條件的時鐘信號
  SCL=1;    //結(jié)束條件建立時間大于4μ
  _Nop();
  _Nop();
  _Nop();
  _Nop();
  _Nop();
  SDA=1;    //發(fā)送I2C總線結(jié)束信號
  _Nop();
  _Nop();
  _Nop();
  _Nop();
}

串口通訊模塊的編寫

發(fā)送字符串,要在最后設置一個校驗位,就是告訴計算機,這次的數(shù)據(jù)發(fā)完了,停下吧。

//發(fā)送一個字符串
void SendStr1(unsigned char *s)
{
 while(*s!=0xff)// ff表示數(shù)據(jù)發(fā)完了
  {
  SendByte1(*s);
  s++;
  }
}

主函數(shù)模塊的數(shù)據(jù)數(shù)據(jù)處理

讀取到的數(shù)據(jù)是一個0~256(二的八次方)之間的數(shù),參考電壓這里是5V,所以要把讀取到的數(shù)帶入公式中計算,然后分小數(shù)點前的數(shù)據(jù),因為要在數(shù)碼管顯示,所以|0x80,加上小數(shù)點,小數(shù)點后直接保留就好。

//連續(xù)讀5次,輸入通道后多讀幾次,取最后一次值,以便讀出穩(wěn)定值
 for(i=0;i< 5;i++)
    num0=ReadADC(0);
  num0=num0*5*10/256;// x10表示把實際值擴大10,如4.5 變成 45 方便做下一步處理 x5 表示基準電壓5V 
  TempData[0]=dofly_DuanMa[num0/10]|0x80;  
  TempData[1]=dofly_DuanMa[num0%10];

通訊部分數(shù)據(jù)處理,十六進制的數(shù)據(jù)報文。

VoltData[0]=num0;
 VoltData[1]=num1;
 VoltData[2]=num2;
 VoltData[3]=num3;
 VoltData[4]=0xff;

SendStr1(VoltData);
DelayMs(240);//延時循環(huán)發(fā)送
DelayMs(240);
聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權轉(zhuǎn)載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學習之用,如有內(nèi)容侵權或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • 單片機
    +關注

    關注

    6042

    文章

    44617

    瀏覽量

    637890
  • 收發(fā)器
    +關注

    關注

    10

    文章

    3454

    瀏覽量

    106195
  • 控制器
    +關注

    關注

    112

    文章

    16444

    瀏覽量

    179164
  • CAN總線
    +關注

    關注

    145

    文章

    1955

    瀏覽量

    130993
  • CAN
    CAN
    +關注

    關注

    57

    文章

    2766

    瀏覽量

    464213
收藏 人收藏

    評論

    相關推薦

    影響CAN總線節(jié)點數(shù)的因素:為何考慮CAN接口負載?

    CAN-bus電路設計中,理論上收發(fā)器支持節(jié)點數(shù)最多可做到110個,但實際應用中往往達不到這個數(shù)量。今天我們就來談談如何通過合理的CAN-bus
    的頭像 發(fā)表于 09-05 09:24 ?2.2w次閱讀
    影響<b class='flag-5'>CAN</b><b class='flag-5'>總線</b><b class='flag-5'>節(jié)點</b>數(shù)的因素:為何考慮<b class='flag-5'>CAN</b>接口負載?

    未知多節(jié)點CAN總線網(wǎng)絡,如何準確識別?

    CAN網(wǎng)絡中,所有節(jié)點的數(shù)據(jù)收發(fā)共享一條總線。當面對未知的多節(jié)點CAN
    的頭像 發(fā)表于 02-27 09:29 ?1.6w次閱讀
    未知多<b class='flag-5'>節(jié)點</b><b class='flag-5'>CAN</b><b class='flag-5'>總線</b>網(wǎng)絡,如何準確識別?

    基于CAN總線的溫度測量節(jié)點設計

    任務。同時為了增加CAN總線節(jié)點的抗干擾能力,更好地實現(xiàn)了總線上各CAN節(jié)點間的電氣隔離,SJA
    發(fā)表于 11-13 10:38

    CAN總線節(jié)點通信異常問題解決

    總線上的節(jié)點都可以決定自己是否需要總線上的數(shù)據(jù)。市場上常用的收發(fā)器(例如: VP230、TJA1040、TCAN337等)多為ISO11898標準。在此標準中,對于
    發(fā)表于 07-01 19:20

    基于STM32的CAN總線通信節(jié)點控制設計

    說明:這是我的畢業(yè)設計,基于STM32的CAN總線通信節(jié)點設計。用的是STM32F103微控制器,LCD1602液晶顯示,MQ-4甲烷氣體傳感器,CTM1050CAN
    發(fā)表于 07-19 07:18

    基于CAN總線的煤礦瓦斯報警節(jié)點系統(tǒng)的設計

    文章介紹了一種基于CAN 總線的煤礦瓦斯報警節(jié)點系統(tǒng)的設計,給出了基于CAN 總線的智能監(jiān)控節(jié)點
    發(fā)表于 03-30 16:40 ?47次下載

    CAN總線節(jié)點電路的設計與實現(xiàn)

    CAN 總線作為一種現(xiàn)場總線,應用極為普及。本文給出了一種基于AT89C51 和SJA1000 的CAN 總線
    發(fā)表于 05-25 13:26 ?60次下載

    基于CAN總線智能節(jié)點設計

    CAN總線是一種應用極為普及的現(xiàn)場總線。文中提出了一種CAN總線通信接口的設計方案。CAN
    發(fā)表于 01-25 15:04 ?52次下載

    基于LPC2294的CAN總線節(jié)點設計

    為引入CAN總線技術以實現(xiàn)運動控制系統(tǒng)的網(wǎng)絡化,提出了基于LPC2294的CAN總線節(jié)點的硬件及軟件設計方案。硬件采用基于ARM7內(nèi)核的微
    發(fā)表于 06-21 15:17 ?61次下載
    基于LPC2294的<b class='flag-5'>CAN</b><b class='flag-5'>總線</b>主<b class='flag-5'>節(jié)點</b>設計

    基于LPC2294的CAN總線智能節(jié)點設計

    CAN總線是一種應用廣泛的實時性現(xiàn)場總線。本文提出了一種基于具有ARM7DMI內(nèi)核的32位ARM微控制器LPC2294的CAN總線智能
    發(fā)表于 05-29 09:27 ?3904次閱讀
    基于LPC2294的<b class='flag-5'>CAN</b><b class='flag-5'>總線</b>智能<b class='flag-5'>節(jié)點</b>設計

    基于CAN總線的智能節(jié)點設計

    應用51單片機為控制核心結(jié)合其他的器件設計了一種能連接于CAN總線上的智能節(jié)點。通過單片機控制CAN總線控制器SJA1000,并進一步通過
    發(fā)表于 01-10 16:54 ?120次下載

    CAN總線節(jié)點的可靠性設計

    CAN總線通訊已經(jīng)從汽車電子行業(yè)逐漸向各行各業(yè)鋪開使用了,例如軌道交通、礦井監(jiān)控等。在設計CAN總線接口電路時需要注意, 對于提高CAN
    發(fā)表于 06-08 11:41 ?3128次閱讀

    CAN總線節(jié)點軟件的設計與實現(xiàn)

    CAN總線節(jié)點軟件的設計與實現(xiàn)說明。
    發(fā)表于 04-19 16:59 ?20次下載

    基于FPGA的CAN總線通信節(jié)點設計

    點擊上方 藍字 關注我們 摘要:以FPGA 代替?zhèn)鹘y(tǒng)的單片機和外圍擴展芯片, 給出了CAN 總線通信節(jié)點的詳細設計方案。其中以SJA1000為CAN
    的頭像 發(fā)表于 06-18 11:15 ?3075次閱讀

    基于CAN總線的智能節(jié)點設計

    電子發(fā)燒友網(wǎng)站提供《基于CAN總線的智能節(jié)點設計.pdf》資料免費下載
    發(fā)表于 10-20 14:44 ?21次下載
    基于<b class='flag-5'>CAN</b><b class='flag-5'>總線</b>的智能<b class='flag-5'>節(jié)點</b>設計