搞嵌入式必懂的CAN總線(xiàn)知識
嵌入式的工程師一般都知道CAN總線(xiàn)廣泛應用到汽車(chē)中,其實(shí)船艦電子設備通信也廣泛使用CAN,隨著(zhù)國家對海防的越來(lái)越重視,對CAN的需求也會(huì )越來(lái)越大。
本文引用地址:http://dyxdggzs.com/article/202310/451257.htm概述
CAN(Controller Area Network)即控制器局域網(wǎng),是一種能夠實(shí)現分布式實(shí)時(shí)控制的串行通信網(wǎng)絡(luò )。
想到CAN就要想到德國的Bosch公司,因為CAN就是這個(gè)公司開(kāi)發(fā)的(和Intel)。
CAN有很多優(yōu)秀的特點(diǎn),使得它能夠被廣泛地應用。比如:傳輸速度最高到1Mbps,通信距離最遠到10km,無(wú)損位仲裁機制,多主結構。
近些年來(lái),CAN控制器價(jià)格越來(lái)越低,很多MCU也集成了CAN控制器?,F在每一輛汽車(chē)上都裝有CAN總線(xiàn)。
一個(gè)典型的CAN應用場(chǎng)景:
CAN總線(xiàn)標準
CAN總線(xiàn)標準只規定了物理層和數據鏈路層,需要用戶(hù)自定義應用層。不同的CAN標準僅物理層不同。
CAN收發(fā)器負責邏輯電平和物理信號之間的轉換。
將邏輯信號轉換成物理信號(差分電平),或者將物理信號轉換成邏輯電平。
CAN標準有兩個(gè),即IOS11898和IOS11519,兩者差分電平特性不同。
高低電平幅度低,對應的傳輸速度快;
* 雙絞線(xiàn)共模消除干擾,是因為電平同時(shí)變化,電壓差不變。
物理層
CAN有三種接口器件:
多個(gè)節點(diǎn)連接,只要有一個(gè)為低電平,總線(xiàn)就為低電平,只有所有節點(diǎn)輸出高電平時(shí),才為高電平。所謂"線(xiàn)與"。
CAN總線(xiàn)有5個(gè)連續相同位后,就插入一個(gè)相反位,產(chǎn)生跳變沿,用于同步。從而消除累積誤差。
和485、232一樣,CAN的傳輸速度與距離成反比。
CAN總線(xiàn),終端電阻的接法:
為什么是120Ω,因為電纜的特性阻抗為120Ω,為了模擬無(wú)限遠的傳輸線(xiàn)。
數據鏈路層
CAN總線(xiàn)傳輸的是CAN幀,CAN的通信幀分成五種,分別為數據幀、遠程幀、錯誤幀、過(guò)載幀和幀間隔。
數據幀用來(lái)節點(diǎn)之間收發(fā)數據,是使用最多的幀類(lèi)型;遠程幀用來(lái)接收節點(diǎn)向發(fā)送節點(diǎn)接收數據;錯誤幀是某節點(diǎn)發(fā)現幀錯誤時(shí)用來(lái)向其他節點(diǎn)通知的幀;過(guò)載幀是接收節點(diǎn)用來(lái)向發(fā)送節點(diǎn)告知自身接收能力的幀;用于將數據幀、遠程幀與前面幀隔離的幀。
數據幀根據仲裁段長(cháng)度不同分為標準幀(2.0A)和擴展幀(2.0B)
幀起始
幀起始由一個(gè)顯性位(低電平)組成,發(fā)送節點(diǎn)發(fā)送幀起始,其他節點(diǎn)同步于幀起始;
幀結束由7個(gè)隱形位(高電平)組成。
仲裁段
CAN總線(xiàn)是如何解決多點(diǎn)競爭的問(wèn)題?
由仲裁段給出答案。
CAN總線(xiàn)控制器在發(fā)送數據的同時(shí)監控總線(xiàn)電平,如果電平不同,則停止發(fā)送并做其他處理。如果該位位于仲裁段,則退出總線(xiàn)競爭;如果位于其他段,則產(chǎn)生錯誤事件。
幀ID越小,優(yōu)先級越高。由于數據幀的RTR位為顯性電平,遠程幀為隱性電平,所以幀格式和幀ID相同的情況下,數據幀優(yōu)先于遠程幀;由于標準幀的IDE位為顯性電平,擴展幀的IDE位為隱形電平,對于前11位ID相同的標準幀和擴展幀,標準幀優(yōu)先級比擴展幀高。
控制段
共6位,標準幀的控制段由擴展幀標志位IDE、保留位r0和數據長(cháng)度代碼DLC組成;擴展幀控制段則由IDE、r1、r0和DLC組成。
數據段
為0-8字節,短幀結構,實(shí)時(shí)性好,適合汽車(chē)和工控領(lǐng)域;
CRC段
CRC校驗段由15位CRC值和CRC界定符組成。
ACK段
當接收節點(diǎn)接收到的幀起始到CRC段都沒(méi)錯誤時(shí),它將在A(yíng)CK段發(fā)送一個(gè)顯性電平,發(fā)送節點(diǎn)發(fā)送隱性電平,線(xiàn)與結果為顯性電平。
遠程幀
遠程幀分為6個(gè)段,也分為標準幀和擴展幀,且RTR位為1(隱性電平)
CAN是可靠性很高的總線(xiàn),但是它也有五種錯誤:
CRC錯誤:發(fā)送與接收的CRC值不同發(fā)生該錯誤; 格式錯誤:幀格式不合法發(fā)生該錯誤; 應答錯誤:發(fā)送節點(diǎn)在A(yíng)CK階段沒(méi)有收到應答信息發(fā)生該錯誤; 位發(fā)送錯誤:發(fā)送節點(diǎn)在發(fā)送信息時(shí)發(fā)現總線(xiàn)電平與發(fā)送電平不符發(fā)生該錯誤; 位填充錯誤:通信線(xiàn)纜上違反通信規則時(shí)發(fā)生該錯誤。
當發(fā)生這五種錯誤之一時(shí),發(fā)送節點(diǎn)或接受節點(diǎn)將發(fā)送錯誤幀。
為防止某些節點(diǎn)自身出錯而一直發(fā)送錯誤幀,干擾其他節點(diǎn)通信,CAN協(xié)議規定了節點(diǎn)的3種狀態(tài)及行為。
過(guò)載幀
當某節點(diǎn)沒(méi)有做好接收的"準備"時(shí),將發(fā)送過(guò)載幀,以通知發(fā)送節點(diǎn)。
幀間隔
用來(lái)隔離數據幀、遠程幀與他們前面的幀,錯誤幀和過(guò)載幀前面不加幀間隔。
構建CAN節點(diǎn)
構建節點(diǎn),實(shí)現相應控制,由底向上分為四個(gè)部分:CAN節點(diǎn)電路、CAN控制器驅動(dòng)、CAN應用層協(xié)議、CAN節點(diǎn)應用程序。
雖然不同節點(diǎn)完成的功能不同,但是都有相同的硬件和軟件結構。
CAN收發(fā)器和控制器分別對應CAN的物理層和數據鏈路層,完成CAN報文的收發(fā);功能電路,完成特定的功能,如信號采集或控制外設等;主控制器與應用軟件按照CAN報文格式解析報文,完成相應控制。
CAN硬件驅動(dòng)是運行在主控制器(如P89V51)上的程序,它主要完成以下工作:基于寄存器的操作,初始化CAN控制器、發(fā)送CAN報文、接收CAN報文;
如果直接使用CAN硬件驅動(dòng),當更換控制器時(shí),需要修改上層應用程序,移植性差。在應用層和硬件驅動(dòng)層加入虛擬驅動(dòng)層,能夠屏蔽不同CAN控制器的差異。
一個(gè)CAN節點(diǎn)除了完成通信的功能,還包括一些特定的硬件功能電路,功能電路驅動(dòng)向下直接控制功能電路,向上為應用層提供控制功能電路函數接口。特定功能包括信號采集、人機顯示等。
CAN收發(fā)器是實(shí)現CAN控制器邏輯電平與CAN總線(xiàn)上差分電平的互換。實(shí)現CAN收發(fā)器的方案有兩種,一是使用CAN收發(fā)IC(需要加電源隔離和電氣隔離),另一種是使用CAN隔離收發(fā)模塊。推薦使用第二種。
CAN控制器是CAN的核心元件,它實(shí)現了CAN協(xié)議中數據鏈路層的全部功能,能夠自動(dòng)完成CAN協(xié)議的解析。CAN控制器一般有兩種,一種是控制器IC(SJA1000),另一種是集成CAN控制器的MCU(LPC11C00)。
MCU負責實(shí)現對功能電路和CAN控制器的控制:在節點(diǎn)啟動(dòng)時(shí),初始化CAN控制器參數;通過(guò)CAN控制器讀取和發(fā)送CAN幀;在CAN控制器發(fā)生中斷時(shí),處理CAN控制器的中斷異常;根據接收到的數據輸出控制信號;
接口管理邏輯:解釋MCU指令,尋址CAN控制器中的各功能模塊的寄存器單元,向主控制器提供中斷信息和狀態(tài)信息。
發(fā)送緩沖區和接收緩沖區能夠存儲CAN總線(xiàn)網(wǎng)絡(luò )上的完整信息。
驗收濾波是將存儲的驗證碼與CAN報文識別碼進(jìn)行比較,跟驗證碼匹配的CAN幀才會(huì )存儲到接收緩沖區。
CAN內核實(shí)現了數據鏈路的全部協(xié)議。
CAN協(xié)議應用層概述
CAN總線(xiàn)只提供可靠的傳輸服務(wù),所以節點(diǎn)接收報文時(shí),要通過(guò)應用層協(xié)議來(lái)判斷是誰(shuí)發(fā)來(lái)的數據、數據代表了什么含義。常見(jiàn)的CAN應用層協(xié)議有:CANOpen、DeviceNet、J1939、iCAN等。
CAN應用層協(xié)議驅動(dòng)是運行在主控制器(如P89V51)上的程序,它按照應用層協(xié)議來(lái)對CAN報文進(jìn)行定義、完成CAN報文的解析與拼裝。例如,我們將幀ID用來(lái)表示節點(diǎn)地址,當接收到的幀ID與自身節點(diǎn)ID不通過(guò)時(shí),就直接丟棄,否則交給上層處理;發(fā)送時(shí),將幀ID設置為接收節點(diǎn)的地址。
CAN收發(fā)器
SJA1000的輸出模式有很多,使用最多的是正常輸出模式,輸入模式通常不選擇比較器模式,可以增大通信距離,并且減少休眠下的電流。
收發(fā)器按照通信速度分為高速CAN收發(fā)器和容錯CAN收發(fā)器。
同一網(wǎng)絡(luò )中要使用相同的CAN收發(fā)器。
CAN連接線(xiàn)上會(huì )有很多干擾信號,需要在硬件上添加濾波器和抗干擾電路:
也可以使用CAN隔離收發(fā)器(集成濾波器和抗干擾電路)。
CAN控制器與MCU的連接方式:
SJA1000可被視為外擴RAM,地址寬度8位,最多支持256個(gè)寄存器
#define REG_BASE_ADDR 0xA000 // 寄存器基址
unsigned char *SJA_CS_Point = (unsigned char *) REG_BASE_ADDR ;
// 寫(xiě)SJA1000寄存器
void WriteSJAReg(unsigned char RegAddr, unsigned char Value)
{
*(SJA_CS_Point + RegAddr) = Value;
return;
}
// 讀SJA1000寄存器
unsigned char ReadSJAReg(unsigned char RegAddr)
{
return (*(SJA_CS_Point + RegAddr));
}
將緩存區的數據連續寫(xiě)入寄存器:
……
for (i=0; i<len; i++)
{
WriteSJAReg(RegAdr + i, ValueBuf[i]);
}
……
將連續多個(gè)寄存器連續讀入緩存區:
……
for (i=0; i<len; i++)
{
ReadSJAReg(RegAdr + i, ValueBuf[i]);
}
……
頭文件包含方案:
1. 每個(gè)程序包含用到的頭文件
#ifndef __CONFIG_H__ // 防止頭文件被重復包含
#define __CONFIG_H__
#include <8051.h> // 包含80C51寄存器定義頭文件
#include "SJA1000REG.h" // 包含SJA1000寄存器定義頭文件
// 定義取字節運算
#define LOW_BYTE(x) (unsigned char)(x)
#define HIGH_BYTE(x) (unsigned char)((unsigned int)(x) >> 8)
// 定義振蕩器時(shí)鐘和處理器時(shí)鐘頻率(用戶(hù)可以根據實(shí)際情況作出調整)
#define OSCCLK 11059200UL
// 宏定義MCU的時(shí)鐘頻率
#define CPUCLK (OSCCLK / 12)
#endif // __CONFIG_H__
SJA1000上電后處于復位狀態(tài),必須初始化后才能工作:
(1)置位模式寄存器Bit0位進(jìn)入復位模式;
(2)設置時(shí)鐘分頻寄存器選擇時(shí)鐘頻率、CAN模式;
(3)設置驗收濾波,設定驗證碼和屏蔽碼;
(4)設置總線(xiàn)定時(shí)器寄存器0、1設定CAN波特率;
(5)設置輸出模式;
(6)清零模式寄存器Bit0位退出復位模式;
模式寄存器
只檢測模式:SJA1000發(fā)送CAN幀時(shí)不檢查應答位;
只聽(tīng)模式:此模式下SJA1000不會(huì )發(fā)送錯誤幀,用于自動(dòng)檢測波特率;SJA1000以不同的波特率接收CAN幀,當收到CAN幀時(shí),表明當前波特率與總線(xiàn)波特率相同。
波特率設置
CAN總線(xiàn)無(wú)時(shí)鐘,使用異步串行傳輸;波特率是1秒發(fā)送的數據位;
CAN幀發(fā)送:
發(fā)送CAN幀的步驟:
1.檢測狀態(tài)寄存器,等待發(fā)送緩沖區可用;
2.填充報文到發(fā)送緩沖區;
3.啟動(dòng)發(fā)送。
SJA1000具有一個(gè)12字節的緩沖區,要發(fā)送的報文可以通過(guò)寄存器16-28寫(xiě)入,也可通過(guò)寄存器96-108寫(xiě)入或讀出:
設置發(fā)送模式:
char SetSJASendCmd(unsigned char cmd)
{
unsigned char ret;
switch (cmd)
{
default:
case 0:
ret = SetBitMask(REG_CAN_CMR, TR_BIT); //正常發(fā)送
break;
case 1:
ret = SetBitMask(REG_CAN_CMR, TR_BIT|AT_BIT); //單次發(fā)送
break;
case 2:
ret = SetBitMask(REG_CAN_CMR, TR_BIT|SRR_BIT);//自收自發(fā)
break;
case 0xff:
ret = SetBitMask(REG_CAN_CMR, AT_BIT);//終止發(fā)送
break;
}
return ret;
}
發(fā)送函數:
unsigned char SJA_CAN_Filter[8] =
{
// 定義驗收濾波器的參數,接收所有幀
0x00, 0x00, 0x00, 0x00,
// ACR0~ACR3
0xff, 0xff, 0xff, 0xff
// AMR0~AMR3
};
unsigned char STD_SEND_BUFFER[11] =
{
// CAN 發(fā)送報文緩沖區
0x08, // 幀信息,標準數據幀,數據長(cháng)度 = 8
0xEA, 0x60, // 幀ID = 0x753
0x55, 0x55, 0x55, 0x55, 0xaa, 0xaa, 0xaa, 0xaa // 幀數據
};
void main(void) // 主函數,程序入口
{
timerInit();// 初始化
D1 = 0;
SJA1000_RST = 1; // 硬件復位SJA1000
timerDelay(50); // 延時(shí)500ms
SJA1000_RST = 0;
SJA1000_Init(0x00, 0x14, SJA_CAN_Filter); // 初始化SJA1000,設置波特率為1Mbps
// 無(wú)限循環(huán),main()函數不允許返回
for(;;)
{
SJASendData(STD_SEND_BUFFER, 0x0);
timerDelay(100); // 延時(shí)1000ms
}
}
為什么幀ID是0x753,這與CAN幀在緩沖區的存儲格式有關(guān)。
終端電阻非常重要,當波特率較高而且沒(méi)加終端電阻時(shí),信號過(guò)沖非常嚴重。
SJA1000有64個(gè)字節的接收緩沖區(FIFO),這可以降低對MCU的要求。
MCU可以通過(guò)查詢(xún)或中斷的方式確定SJA1000接收到報文后讀取報文。
評論