先看看工作原理
當 CAN 總線上的一個節(jié)點(站)發(fā)送數(shù)據(jù)時,它以報文的形式廣播給網(wǎng)絡(luò)中所有節(jié)點,對每個節(jié)點來說,無論數(shù)據(jù)是否是發(fā)給自己的,都對其接收。
每組報文開頭的 11 位字符為標識符,定義了報文的優(yōu)先級,這種報文格式成為面向內(nèi)容的編制方案。同一系統(tǒng)中標識符是唯一的,不可能有兩個站發(fā)送具有相同標識符的報文,當幾個站同時競爭總線讀取時,這種配置十分重要。
大體的工作原理我們搞清了,但是根本的協(xié)議我們還要花一番功夫。下面介紹一個重要的名詞,“顯性”和“隱性”:
在我看到的很多文章里,有很多顯性和隱性的地方,為此我頭痛不已,最終我把它們徹底弄明白了。
首先 CAN 數(shù)據(jù)總線有兩條導(dǎo)線,一條是黃色的,一條是綠色的 ------ 分別是 CAN_High 線和 CAN_Low 線,當靜止狀態(tài)時,這兩條導(dǎo)線上的電平一樣,這個電平稱為靜電平,大約為 2.5 伏。
這個靜電平狀態(tài)就是隱形狀態(tài),也稱隱性電平,也就是沒有任何干擾的時候的狀態(tài)稱為隱性狀態(tài)。當有信號修改時,CAN_High 線上的電壓值變高了,一般來說會升高至少 1V;而 CAN_Low 線上的電壓值會降低一個同樣值,也是 1v。
那么這時候,CAN_High 就是 2.5v+1v=3.5v,它就處于激活狀態(tài)了。而 CAN_Low 降為 2.5v-1v=1.5v??梢钥纯催@個圖
由此我們得到
在隱性狀態(tài)下,CAN_High 線與 CAN_Low 沒有電壓差,這樣我們看到?jīng)]有任何變化也就檢測不到信號。但是在顯性狀態(tài)時,改值最低為 2V,我們就可以利用這種變化才傳輸數(shù)據(jù)了。所以出現(xiàn)了那些幀,那些幀中的場,那些場中的位,云云。
在總線上通常邏輯 1 表示隱性。而 0 表示顯性。這些 1 啊,0 啊,就可以利用起來為我們傳數(shù)據(jù)了。利用這種電壓差,我們可以接收信號。
一般來說,控制單元通過收發(fā)器連接到 CAN 驅(qū)動總線上,這個收發(fā)器(顧名思義,可發(fā)送,可接收)內(nèi)有一個接收器,該接收器是安裝在接收一側(cè)的差動信號放大器。然后,這個放大器很自然地就放大了 CAN_High 和 CAN_Low 線的電平差,然后傳到接收區(qū)。如下圖
由上圖可知,當有電壓差,差動信號放大器放大傳輸,將相應(yīng)的數(shù)據(jù)位轉(zhuǎn)化為 0。
下面我們進入重點難點 ----- 報文
所謂報文,就是 CAN 總線上要傳輸?shù)臄?shù)據(jù)報,為了安全,我們要給我們傳輸?shù)臄?shù)據(jù)報編碼定一下協(xié)議,這樣才能不容易出錯,所以出現(xiàn)了很多的幀,以及仲裁啊,CRC 效驗。這些都是難點。
識別符的概念
識別符顧名思義,就是為了區(qū)分不同報文的可以鑒別的好多字符位。有標準的,和擴展的。標準的是 11 位,擴展的是 29 位。他有一個功能就是可以提供優(yōu)先級,也就是決定哪個報文優(yōu)先被傳輸,報文標識符的值越小,報文具有越高的優(yōu)先權(quán)。
CAN 的報文格式有兩種,不同之處其實就是識別符長度不同,具有 11 位識別符的幀稱為標準幀,而還有 29 位識別符的幀為擴展幀,CAN 報文有以下 4 個不同的幀類型。分別是:
(1) 數(shù)據(jù)幀:數(shù)據(jù)幀將數(shù)據(jù)從發(fā)送器傳輸?shù)浇邮掌鳎?/p>
(2) 遠程幀:總線節(jié)點發(fā)出遠程幀,請求發(fā)送具有同一標識符的數(shù)據(jù)幀;
(3) 錯誤幀:任何節(jié)點檢測到總線錯誤就發(fā)出錯誤幀;
(4) 過載幀:過載幀用已在先行的后續(xù)的數(shù)據(jù)幀(或遠程幀)之間提供一附加的延時。
我們先研究數(shù)據(jù)幀吧。
一,數(shù)據(jù)幀由 7 個不同位場組成(幀起始、仲裁場、控制場、數(shù)據(jù)場、CRC 場、應(yīng)答場、幀結(jié)尾)。
這里的位場,就是不同位的組合,這名字起的很爛,讓人看了感覺很抽象。我們來看看這些個不同的位場吧。一開始是一位幀起始,也叫 SOF。它用顯性位表示,也就是 0;它告訴我們,兩個線上有電壓差了,也就是有數(shù)據(jù)了。
這個幀起始看起來只有一位,其實不簡單了。為了讓所有的分站都同步于發(fā)送報文的發(fā)送站,好接收數(shù)據(jù),有很多要考慮的地方。
報文的數(shù)據(jù)幀結(jié)構(gòu)
然后下一個場是仲裁場。這個仲裁很抽象,其實在這里就是為了解決一個問題。如果 2 個或 2 個以上的單元同時開始傳送報文,那么就會有總線訪問沖突,那么仲裁機制就是用來根據(jù)標識符優(yōu)先級來一個一個的去掉低級別的數(shù)據(jù)。我們可以詳細的描述這場生動的爭搶總線的戰(zhàn)斗。
當總線處于空閑狀態(tài)時呈隱性電平,此時任何節(jié)點都可以向總線發(fā)送顯性電平作為幀的開始。2 個或 2 個以上的節(jié)點同時發(fā)送開始爭搶總線,但是總線只能被一個人搶走。這時候到底怎么決定誰留下,誰滾蛋呢。我們開始思索,我們以前定義了標識符,標識符有優(yōu)先級,它越小,它優(yōu)先級越高。那么怎么實現(xiàn)的呢??聪旅鎴D:
首先搞明白兩點,
一、下圖中, 低波形代表 0(顯性),高波形代表 1(隱性);
二、當隱性碰到顯性,就變?yōu)轱@性。
如圖所示,節(jié)點 A 和節(jié)點 B 的標識符的第 10、9、8 位電平相同,因此兩個節(jié)點偵聽到的信息和它們發(fā)出的信息相同。第 7 位節(jié)點 B 發(fā)出一個“1”,但從節(jié)點上接收到的消息卻是“0”。
為什么呢,因為 A 節(jié)點同時發(fā)出顯性位,讓總線也變成
顯性了,也就是 0。節(jié)點 B 會退出發(fā)送處于單純監(jiān)聽方式而不發(fā)送數(shù)據(jù);節(jié)點 A 成功發(fā)送仲裁位從而獲得總線的控制權(quán),繼而發(fā)送全部消息。
總線中的信號持續(xù)跟蹤最后獲得總線控制權(quán)發(fā)出的報文,本例中節(jié)點 A 的報文將被跟蹤。這種非破壞性位仲裁方法的優(yōu)點在于,在網(wǎng)絡(luò)最終確定哪個節(jié)點被傳送前,報文的起始部分已經(jīng)在網(wǎng)絡(luò)中傳輸了,因此具有高優(yōu)先級的節(jié)點的數(shù)據(jù)傳輸沒有任何延時。
在獲得總線控制權(quán)的節(jié)點發(fā)送數(shù)據(jù)過程中,其他節(jié)點成為報文的接收節(jié)點,并且不會在總線再次空閑之前發(fā)送報文,在這逐位的比較中,最終節(jié)點 B 因為第七位的偏差丟掉了總線。從此單純監(jiān)聽,江山就拱手讓給了節(jié)點 A 了。這就是仲裁機制。
上面我們說過,報文有兩種格式,標準和擴展。這里,不同的格式仲裁場是不一樣的。標準格式下,仲裁場由 11 位識別符和 RTR 位組成。
但在擴展格式里,包括 29 位識別符、SRR 位、IDE 位、RTR 位。
RTR 位,Remote Tranmission Request BIT 全稱為遠程發(fā)送請求位。它在數(shù)據(jù)幀里必須為顯性 0 ,但在遠程幀里為隱性 1。
我暈,為什么這么搞呢,不急,先留著這個問題。
SRR 位,替代遠程請求位,SRR 是一隱性位,也就是 1,它在擴展格式的標準幀 RTR 位位置,那么標準幀怪不得優(yōu)先于擴展幀了,因為在傳輸完 11 位標識符之后(擴展幀的后 18 位在最后發(fā)送,先發(fā)送 11 位標識符),輪到標準幀的 RTR 位和擴展幀的 SRR 位了。
這時候,標準幀的 RTR 為顯性,而擴展幀 SRR 為隱性,這樣,總線自然就被標準幀占據(jù)。
同時上面那個問題,也一目了然了,CAN 總線協(xié)議設(shè)計者,肯定是設(shè)計了數(shù)據(jù)幀優(yōu)先于遠程幀。所以 IDE(Identifier Extension Bit),全稱識別符擴展位,它屬于擴展格式的仲裁場。
對于擴展格式,IDE 位屬于仲裁場;對于標準格式,IDE 位屬于控制場。標準格式的 IDE 位為“顯性”,而擴展格式的 IDE 位為“隱性”。
標準格式中的數(shù)據(jù)幀
拓展格式中的數(shù)據(jù)幀
控制場
控制場由 6 個位組成,標準格式和擴展格式的控制場格式不同。標準格式里的幀包括數(shù)據(jù)長度代碼、IDE 位(為顯性位)及保留位 r0。擴展格式里的幀包括數(shù)據(jù)長度代碼和兩個保留位:r1 和 r0。其保留位必須發(fā)送為顯性,但是接收器認可“顯性”和“隱性”位的任何組合。其結(jié)構(gòu)如圖所示:
控制場結(jié)構(gòu)
數(shù)據(jù)長度代碼(標準格式以及擴展格式)DLC,如下表所示
數(shù)據(jù)幀長度代碼 DLC
數(shù)據(jù)長度代碼指示了數(shù)據(jù)場里的字節(jié)數(shù)量。其中:d—“顯性”, r—“隱性”,數(shù)據(jù)幀允許的數(shù)據(jù)字節(jié)數(shù)為{0,1,...,7,8}。其他的數(shù)值不允許使用。
數(shù)據(jù)場
數(shù)據(jù)場由數(shù)據(jù)幀里的發(fā)送數(shù)據(jù)組成。它可以為 0~8 個字節(jié),每字節(jié)包含了 8 個位,首先發(fā)送最高有效位(MSB)。
循環(huán)冗余碼 CRC 場是數(shù)據(jù)通信領(lǐng)域中最常用的一種差錯校驗碼,其特征是信息字段和校驗字段的長度可以任意選定。
CRC 場包括 CRC 序列(CRC Sequence),其后是 CRC 界定符(CRC Delimiter),結(jié)構(gòu)如圖:
生成 CRC 碼的基本原理:
任意一個由二進制位串組成的代碼都可以和一個系數(shù)僅為‘0’和‘1’取值的多項式一一對應(yīng)。例如:代碼 1010111 對應(yīng)的多項式為 x6+x4+x2+x+1,而多項式為 x5+x3+x2+x+1 對應(yīng)的代碼 101111。
參考一下下面的例題,自已再領(lǐng)悟一下吧!已知信息位為 1101,生成多項式 G(x)= x3+x+1,求 CRC 碼。
要傳輸?shù)男畔⑿蛄袨?1101,在末尾添加所給多項式的最高次階個 0,如本題為 x^3,則添加 3 個 0,變?yōu)椋?101000;
由多項式 G(X)=X3+X+1,得其階數(shù)為 1 的二進制編碼為:1011;1101000 對 1011 進行模二除法,所得到的余數(shù)即為校驗碼,把校驗碼添加在原數(shù)據(jù)尾部即為所求的編碼,則實際發(fā)送的數(shù)據(jù)序列為 1101001。校驗碼計算過程如圖所示:
模二除法
應(yīng)答場(ACK Field)
應(yīng)答場長度為 2 個位,包含應(yīng)答間隙(ACK Slot)和應(yīng)答界定符(ACK Delimiter),如圖所示。在 ACK 場(應(yīng)答場)里,發(fā)送節(jié)點發(fā)送兩個“隱性”位。
當接收器正確地接收到有效的報文,接收器就會在應(yīng)答間隙(ACK Slot)期間向發(fā)送器發(fā)送一“顯性”位以示應(yīng)答。
幀結(jié)尾
每一個數(shù)據(jù)幀和遠程幀均由一標志序列界定。這個標志序列由 7 個“隱性”位組成。
二、遠程幀
通過發(fā)送遠程幀,總線的節(jié)點發(fā)出遠程幀,請求以前發(fā)送給它數(shù)據(jù)幀的節(jié)點再發(fā)送一遍。具體發(fā)送哪個數(shù)據(jù)幀,由遠程幀的標識符決定。
與數(shù)據(jù)幀類似,遠程幀也有標準格式和擴展格式,而且都由 6 個不同的位場組成:幀起始、仲裁場、控制場、CRC 場、應(yīng)答場、幀結(jié)尾。
與數(shù)據(jù)幀相反,遠程幀的 RTR 位是“隱性”的。它沒有數(shù)據(jù)場,數(shù)據(jù)長度代碼 DLC 的數(shù)值是不受制約的(可以標注為容許范圍0~8 里的任何數(shù)值),此數(shù)值是相應(yīng)于數(shù)據(jù)幀的數(shù)據(jù)長度代碼。遠程幀結(jié)構(gòu)如圖所示:
遠程幀結(jié)構(gòu)
三、錯誤幀
錯誤幀由兩個不同的場組成,第一個場是不同節(jié)點提供的錯誤標志(Error Flag)的疊加,第二個場是錯誤界定符。
為了能正確地終止錯誤幀,“錯誤認可”的節(jié)點要求總線至少有長度為 3 個位時間的總線空閑。因此,總線的載荷不應(yīng)為 100%。錯誤幀結(jié)構(gòu)如圖:
錯誤幀結(jié)構(gòu)(圖中統(tǒng)一起見出錯幀改為錯誤幀)
(1) 錯誤標志,有兩種形式的錯誤標志:激活錯誤標志 和 認可錯誤標志
“激活錯誤”標志由 6 個連續(xù)的“顯性”位組成;“認可錯誤”標志由 6 個連續(xù)的“隱性”的位組成,除非被其他節(jié)點的“顯性”位重寫。
(2) 錯誤界定符,錯誤界定符包括 8 個“隱性”的位。
錯誤標志傳送了以后,每一個節(jié)點就發(fā)送一個“隱性”的位,并一直監(jiān)視總線直到檢測出一個“隱性”的位為止,然后就開始發(fā)送其余 7 個“隱性”位。
四、過載幀
過載幀(Overload Frame)包括兩個位場:過載標志 和 過載界定符,其結(jié)構(gòu)如圖:
過載幀結(jié)構(gòu)圖
有三種過載的情況會引發(fā)過載標志的傳送:接收器的內(nèi)部情況,需要延遲下一個數(shù)據(jù)幀和遠程幀。
在間歇(Intermission)的第 1 和第 2 字節(jié)檢測到一個“顯性”位。這里有個間歇的概念。我們可以講講。間歇屬于幀間空間的一部分。它包含三個隱性位。間歇期間,所有的站不允許傳送數(shù)據(jù)幀或遠程幀。它唯一要做的就是標示一個過載條件。
如果 CAN 節(jié)點在錯誤界定符或過載界定符的第 8 位(最后一位)采樣到一個顯性位,節(jié)點會發(fā)送一個過載幀。該幀不是錯誤幀,錯誤計數(shù)器不會增加。
(1)過載標志(Overload Flag)
過載標志由 6 個“顯性”的位組成。過載標志的所有形式和“激活錯誤”標志的一樣。
(2)過載界定符(Overload Delimiter)
過載界定符包括 8 個“隱性”的位。
五、幀間空間
數(shù)據(jù)幀(或遠程幀)與先行幀的隔離是通過幀間空間實現(xiàn)的,無論此先行幀類型如何(數(shù)據(jù)幀、遠程幀、錯誤幀、過載幀)。
幀間空間包括間歇、總線空閑的位場。如果“錯誤認可”的節(jié)點已作為前一報文的發(fā)送器,則其幀間空間除了間歇、總線空閑外,還包括稱作“掛起傳送”(暫停發(fā)送)(Suspend Transmission)的位場。
對于不是“錯誤認可”的節(jié)點,或作為前一報文的接收器的節(jié)點,其幀間空間如圖:
非 “錯誤認可”幀間空間
對于作為前一報文發(fā)送器的“錯誤認可”的節(jié)點,其幀間空間如圖:
“錯誤激活”幀間空間
(1)總線空閑(Bus Idle)
總線空閑的時間是任意的。只要總線被認定為空閑,任何等待發(fā)送報文的節(jié)點就會訪問總線。在發(fā)送其他報文期間,有報文被掛起,對于這樣的報文,其傳送起始于間歇之后的第一個位??偩€上檢測到的“顯性”的位可被解釋為幀的起始。
(2)掛起傳送(Suspend Transmission)
“錯誤認可”的節(jié)點發(fā)送報文后,節(jié)點就在下一報文開始傳送之前或總線空閑之前發(fā)出 8 個“隱性”的位跟隨在間歇的后面。如果與此同時另一節(jié)點開始發(fā)送報文(由另一節(jié)點引起),則此節(jié)點就作為這個報文的接收器。
最后用腦圖總結(jié):
再來幾張實物圖:
CAN 總線接收芯片
CAN 總線控制器芯片
標準數(shù)據(jù)幀波形捕獲
擴展數(shù)據(jù)幀波形捕獲