SPI協(xié)議,MCP2515裸機(jī)驅(qū)動詳解,收藏吧用得著
SPI概述
Serial Peripheral interface 通用串行外圍設(shè)備接口
本文引用地址:http://www.butianyuan.cn/article/202402/455817.htm是Motorola首先在其MC68HCXX系列處理器上定義的。SPI接口主要應(yīng)用在 EEPROM,F(xiàn)LASH,實(shí)時時鐘,AD轉(zhuǎn)換器,還有數(shù)字信號處理器和數(shù)字信號解碼器之間。
SPI,是一種高速的,全雙工,同步的通信總線,并且在芯片的管腳上只占用四根線,節(jié)約了芯片的管腳,同時為PCB的布局上節(jié)省空間。
SPI特點(diǎn)
采用主-從模式(Master-Slave) 的控制方式
SPI 規(guī)定了兩個 SPI 設(shè)備之間通信必須由主設(shè)備 (Master) 來控制次設(shè)備 (Slave). 一個 Master 設(shè)備可以通過提供 Clock 以及對 Slave 設(shè)備進(jìn)行片選 (Slave Select) 來控制多個 Slave 設(shè)備, SPI 協(xié)議還規(guī)定 Slave 設(shè)備的 Clock 由 Master 設(shè)備通過 SCK 管腳提供給 Slave 設(shè)備, Slave 設(shè)備本身不能產(chǎn)生或控制 Clock, 沒有 Clock 則 Slave 設(shè)備不能正常工作。
而這里的SPI中的時鐘和相位,指的就是SCLk時鐘的特性,即保證主從設(shè)備兩者的時鐘的特性一致了,以保證兩者可以正常實(shí)現(xiàn)SPI通訊。
采用同步方式(Synchronous)傳輸數(shù)據(jù)
Master 設(shè)備會根據(jù)將要交換的數(shù)據(jù)來產(chǎn)生相應(yīng)的時鐘脈沖(Clock Pulse), 時鐘脈沖組成了時鐘信號(Clock Signal) , 時鐘信號通過時鐘極性 (CPOL) 和 時鐘相位 (CPHA) 控制著兩個 SPI 設(shè)備間何時數(shù)據(jù)交換以及何時對接收到的數(shù)據(jù)進(jìn)行采樣, 來保證數(shù)據(jù)在兩個設(shè)備之間是同步傳輸?shù)?
工作機(jī)制
概述
首先看下SPI Data Transfer模塊圖。
上圖只是對 SPI 設(shè)備間通信的一個簡單的描述, 下面詳細(xì)解釋一下圖中所示的幾個組件(Module):
SSPBUF
Synchronous Serial Port Buffer, 泛指 SPI 設(shè)備里面的內(nèi)部緩沖區(qū), 一般在物理上是以 FIFO 的形式, 保存?zhèn)鬏斶^程中的臨時數(shù)據(jù);
在這里插入圖片描述
我們知道, 在每個時鐘周期內(nèi), Master 與 Slave 之間交換的數(shù)據(jù)其實(shí)都是 SPI 內(nèi)部移位寄存器從 SSPBUF 里面拷貝的. 我們可以通過往 SSPBUF 對應(yīng)的寄存器 (Tx-Data / Rx-Data register) 里讀寫數(shù)據(jù), 間接地操控 SPI 設(shè)備內(nèi)部的 SSPBUF。
例如, 在發(fā)送數(shù)據(jù)之前, 我們應(yīng)該先往 Master 的 Tx-Data 寄存器寫入將要發(fā)送出去的數(shù)據(jù), 這些數(shù)據(jù)會被 Master-SSPSR 移位寄存器根據(jù) Bus-Width 自動移入 Master-SSPBUF 里, 然后這些數(shù)據(jù)又會被 Master-SSPSR 根據(jù) Channel-Width 從 Master-SSPBUF 中移出, 通過 Master-SDO 管腳傳給 Slave-SDI 管腳, Slave-SSPSR 則把從 Slave-SDI 接收到的數(shù)據(jù)移入 Slave-SSPBUF 里. 與此同時, Slave-SSPBUF 里面的數(shù)據(jù)根據(jù)每次接收數(shù)據(jù)的大小(Channel-Width), 通過 Slave-SDO 發(fā)往 Master-SDI, Master-SSPSR 再把從 Master-SDI 接收的數(shù)據(jù)移入 Master-SSPBUF.在單次數(shù)據(jù)傳輸完成之后, 用戶程序可以通過從 Master 設(shè)備的 Rx-Data 寄存器讀取 Master 設(shè)備數(shù)據(jù)交換得到的數(shù)據(jù)。
SSPSR
Synchronous Serial Port Register, 泛指 SPI 設(shè)備里面的移位寄存器(Shift Regitser), 它的作用是根據(jù)設(shè)置好的數(shù)據(jù)位寬(bit-width) 把數(shù)據(jù)移入或者移出 SSPBUF;
在這里插入圖片描述
SSPSR 是 SPI 設(shè)備內(nèi)部的移位寄存器(Shift Register). 它的主要作用是根據(jù) SPI 時鐘信號狀態(tài), 往 SSPBUF 里移入或者移出數(shù)據(jù), 每次移動的數(shù)據(jù)大小由 Bus-Width 以及 Channel-Width 所決定。 Bus-Width 的作用是指定地址總線到 Master 設(shè)備之間數(shù)據(jù)傳輸?shù)膯挝? 例如, 我們想要往 Master 設(shè)備里面的 SSPBUF 寫入 16 Byte 大小的數(shù)據(jù): 首先, 給 Master 設(shè)備的配置寄存器設(shè)置 Bus-Width 為 Byte; 然后往 Master 設(shè)備的 Tx-Data 移位寄存器在地址總線的入口寫入數(shù)據(jù), 每次寫入 1 Byte 大小的數(shù)據(jù)(使用 writeb 函數(shù)); 寫完 1 Byte 數(shù)據(jù)之后, Master 設(shè)備里面的 Tx-Data 移位寄存器會自動把從地址總線傳來的1 Byte 數(shù)據(jù)移入 SSPBUF 里; 上述動作一共需要重復(fù)執(zhí)行 16 次。
Channel-Width 的作用是指定 Master 設(shè)備與 Slave 設(shè)備之間數(shù)據(jù)傳輸?shù)膯挝? 與 Bus-Width 相似, Master 設(shè)備內(nèi)部的移位寄存器會依據(jù) Channel-Width 自動地把數(shù)據(jù)從 Master-SSPBUF 里通過 Master-SDO 管腳搬運(yùn)到 Slave 設(shè)備里的 Slave-SDI 引腳, Slave-SSPSR 再把每次接收的數(shù)據(jù)移入 Slave-SSPBUF里.通常情況下, Bus-Width 總是會大于或等于 Channel-Width, 這樣能保證不會出現(xiàn)因 Master 與 Slave 之間數(shù)據(jù)交換的頻率比地址總線與 Master 之間的數(shù)據(jù)交換頻率要快, 導(dǎo)致 SSPBUF 里面存放的數(shù)據(jù)為無效數(shù)據(jù)這樣的情況
Controller
泛指 SPI 設(shè)備里面的控制寄存器, 可以通過配置它們來設(shè)置 SPI 總線的傳輸模式。 通常情況下, 我們只需要對上圖所描述的四個管腳(pin) 進(jìn)行編程即可控制整個 SPI 設(shè)備之間的數(shù)據(jù)通信。
Master 設(shè)備里面的 Controller 主要通過時鐘信號(Clock Signal)以及片選信號(Slave Select Signal)來控制 Slave 設(shè)備. Slave 設(shè)備會一直等待, 直到接收到 Master 設(shè)備發(fā)過來的片選信號, 然后根據(jù)時鐘信號來工作。
Master 設(shè)備的片選操作必須由程序所實(shí)現(xiàn). 例如: 由程序把 SS/CS 管腳的時鐘信號拉低電平, 完成 SPI 設(shè)備數(shù)據(jù)通信的前期工作; 當(dāng)程序想讓 SPI 設(shè)備結(jié)束數(shù)據(jù)通信時, 再把 SS/CS 管腳上的時鐘信號拉高電平.
SCK
Serial Clock, 主要的作用是 Master 設(shè)備往 Slave 設(shè)備傳輸時鐘信號, 控制數(shù)據(jù)交換的時機(jī)以及速率;
SS/CS
Slave Select/Chip Select, 用于 Master 設(shè)備片選 Slave 設(shè)備, 使被選中的 Slave 設(shè)備能夠被 Master 設(shè)備所訪問。
SDO/MOSI
Serial Data Output/Master Out Slave In, 在 Master 上面也被稱為 Tx-Channel, 作為數(shù)據(jù)的出口, 主要用于 SPI 設(shè)備發(fā)送數(shù)據(jù)。
SDI/MISO
Serial Data Input/Master In Slave Out, 在 Master 上面也被稱為 Rx-Channel, 作為數(shù)據(jù)的入口, 主要用于SPI 設(shè)備接收數(shù)據(jù)。
SPI 設(shè)備在進(jìn)行通信的過程中, Master 設(shè)備和 Slave 設(shè)備之間會產(chǎn)生一個數(shù)據(jù)鏈路回環(huán)(Data Loop), 就像上圖所畫的那樣, 通過 SDO 和 SDI 管腳, SSPSR 控制數(shù)據(jù)移入移出 SSPBUF, Controller 確定 SPI 總線的通信模式, SCK 傳輸時鐘信號。
極性和相位
要想搞清楚SPI的數(shù)據(jù)傳輸,首先要搞清楚相位和極性的概念,即SPI的極性Polarity和相位Phase。
最常見的寫法是CPOL和CPHA,不過也有一些其他寫法,簡單總結(jié)如下:
(1) CKPOL (Clock Polarity) = CPOL = POL = Polarity = (時鐘)極性
(2) CKPHA (Clock Phase) = CPHA = PHA = Phase = (時鐘)相位
(3) SCK=SCLK=SPI的時鐘
(4) Edge=邊沿,即時鐘電平變化的時刻,即上升沿(rising edge)或者下降沿(falling edge)
對于一個時鐘周期內(nèi),有兩個edge,分別稱為:
(1)Leading edge=前一個邊沿=第一個邊沿,對于開始電壓是1, 那么就是1變成0的時候,對于開始電壓是0,那么就是0變成1的時候;
(2)Trailing edge=后一個邊沿=第二個邊沿,對于開始電壓是1, 那么就是0變成1的時候(即在第一次1變成0之后,才可能有后面的0變成1), 對于開始電壓是0,那么就是1變成0的時候;
本博文采用如下用法:
極性=CPOL
相位=CPHA
SCLK=時鐘
第一個邊沿和第二個邊沿
CPOL和CPHA,分別都可以是0或時1,對應(yīng)的四種組合就是:
模式 極性 相位 Mode 1 CPOL=0,CPHA=0 Mode 2 CPOL=0,CPHA=1 Mode 3 CPOL=1,CPHA=0 Mode 4 CPOL=1,CPHA=1
采樣示意圖
CPOL=0,CPHA=0
脈沖傳輸前和完成后都保持在低電平狀態(tài),所以 CPOL=0,即低電平是空閑時的電平。在第一個邊沿(上升沿)采樣數(shù)據(jù),第二個邊沿(下降沿)輸出數(shù)據(jù),對應(yīng)著CPHA=0。
CPOL=0,CPHA=1
脈沖傳輸前和完成后都保持在低電平狀態(tài),所以 CPOL=0,即低電平是空閑時的電平。在第二個邊沿(下降沿)采樣數(shù)據(jù),第一個邊沿(上升沿)輸出數(shù)據(jù),對應(yīng)著CPHA=1。
CPOL=1,CPHA=0
脈沖傳輸前和完成后都保持在高電平狀態(tài),所以 CPOL=1,即高電平是空閑時的電平。在第一個邊沿(下降沿)采樣數(shù)據(jù),第二個邊沿(上升沿)輸出數(shù)據(jù),對應(yīng)著CPHA=0。
CPOL=1,CPHA=1
脈沖傳輸前和完成后都保持在高電平狀態(tài),所以 CPOL=1,即高電平是空閑時的電平。在第二個邊沿(上升沿)采樣數(shù)據(jù),第一個邊沿(下降沿)輸出數(shù)據(jù),對應(yīng)著CPHA=1。
CPOL=1,CPHA=1
軟件中如何設(shè)置SPI的極性和相位
SPI分主設(shè)備和從設(shè)備,兩者通過SPI協(xié)議通訊。 而設(shè)置SPI的模式,是從設(shè)備的模式,決定了主設(shè)備的模式。 所以要先去搞懂從設(shè)備的SPI是何種模式,然后再將主設(shè)備的SPI的模式,設(shè)置和從設(shè)備相同的模式,即可正常通訊。
對于從設(shè)備的SPI是什么模式,有兩種:
(1)固定的,有SPI從設(shè)備硬件決定的 SPI從設(shè)備,具體是什么模式,相關(guān)的datasheet中會有描述,需要自己去datasheet中找到相關(guān)的描述,即: 關(guān)于SPI從設(shè)備,在空閑的時候,是高電平還是低電平,即決定了CPOL是0還是1; 然后再找到關(guān)于設(shè)備是在上升沿還是下降沿去采樣數(shù)據(jù),這樣就是,在定了CPOL的值的前提下,對應(yīng)著可以推算出CPHA是0還是1了。
(2)可配置的,由軟件自己設(shè)定 從設(shè)備也是一個SPI控制器,4種模式都支持,此時只要自己設(shè)置為某種模式即可。 然后知道了從設(shè)備的模式后,再去將SPI主設(shè)備的模式,設(shè)置為和從設(shè)備模式一樣即可。 對于如何配置SPI的CPOL和CPHA的話,不多細(xì)說,多數(shù)都是直接去寫對應(yīng)的SPI控制器中對應(yīng)寄存器中的CPOL和CPHA那兩位,寫0或?qū)?即可。
數(shù)據(jù)交換(Data Exchanges)
SPI只有主模式和從模式之分,沒有讀和寫的說法,因為實(shí)質(zhì)上每次SPI是主從設(shè)備在交換數(shù)據(jù)。也就是說,你發(fā)一個數(shù)據(jù)必然會收到一個數(shù)據(jù);你要收一個數(shù)據(jù)必須也要先發(fā)一個數(shù)據(jù)。
SPI 設(shè)備間的數(shù)據(jù)傳輸之所以又被稱為數(shù)據(jù)交換, 是因為 SPI 協(xié)議規(guī)定一個 SPI 設(shè)備不能在數(shù)據(jù)通信過程中僅僅只充當(dāng)一個 "發(fā)送者(Transmitter)" 或者 "接收者(Receiver)"。 在每個 Clock 周期內(nèi), SPI 設(shè)備都會發(fā)送并接收一個 bit 大小的數(shù)據(jù), 相當(dāng)于該設(shè)備有一個 bit 大小的數(shù)據(jù)被交換了。
一個 Slave 設(shè)備要想能夠接收到 Master 發(fā)過來的控制信號, 必須在此之前能夠被 Master 設(shè)備進(jìn)行訪問 (Access)。 所以, Master 設(shè)備必須首先通過 SS/CS pin 對 Slave 設(shè)備進(jìn)行片選, 把想要訪問的 Slave 設(shè)備選上。 在數(shù)據(jù)傳輸?shù)倪^程中, 每次接收到的數(shù)據(jù)必須在下一次數(shù)據(jù)傳輸之前被采樣. 如果之前接收到的數(shù)據(jù)沒有被讀取, 那么這些已經(jīng)接收完成的數(shù)據(jù)將有可能會被丟棄, 導(dǎo)致 SPI 物理模塊最終失效。
因此, 在程序中一般都會在 SPI 傳輸完數(shù)據(jù)后, 去讀取 SPI 設(shè)備里的數(shù)據(jù), 即使這些數(shù)據(jù)(Dummy Data)在我們的程序里是無用的。
SPI舉例
下面舉一個例子幫助大家理解。
SPI是一個環(huán)形總線結(jié)構(gòu),由ss(cs)、sck、sdi、sdo構(gòu)成,其時序其實(shí)很簡單,主要是在sck的控制下,兩個雙向移位寄存器進(jìn)行數(shù)據(jù)交換。
假設(shè)下面的8位寄存器裝的是待發(fā)送的數(shù)據(jù)10101010,上升沿發(fā)送、下降沿接收、高位先發(fā)送。 那么第一個上升沿來的時候 數(shù)據(jù)將會是sdo=1;寄存器=0101010x。下降沿到來的時候,sdi上的電平將所存到寄存器中去,那么這時寄存器=0101010sdi,這樣在 8個時鐘脈沖以后,兩個寄存器的內(nèi)容互相交換一次。這樣就完成里一個spi時序。
舉例: 假設(shè)主機(jī)和從機(jī)初始化就緒:并且主機(jī)的sbuff=0xaa,從機(jī)的sbuff=0x55,下面將分步對spi的8個時鐘周期的數(shù)據(jù)情況演示一遍:假設(shè)上升沿發(fā)送數(shù)據(jù)
這樣就完成了兩個寄存器8位的交換,上面的上表示上升沿、下表示下降沿,sdi、sdo相對于主機(jī)而言的。
下一步就是把 上面的過程轉(zhuǎn)為動畫
交換數(shù)據(jù)
交換前
在這里插入圖片描述
請仔細(xì)比較下交換后的bit順序。
了解了SPI協(xié)議說明之后,下面我們基于Cortex-A9架構(gòu)的exynos-4412,講解SPI控制器的使用。
Cortex-A9 SPI控制器
硬件設(shè)計
本例是基于FS4412開發(fā)板,SPI控制器外接了MCP2515,MCP2515與exynos-4412的硬件連接圖如下圖所示。
管腳連接說明
由上圖可知SPI個引腳與SOC的pin連接關(guān)系:
CS <---------> BUF_BK_LED <---------> GPC1_2
SO <---------> BUF_I2C_SDA6 <---------> GPC1_3
SI <---------> BUF_I2C_SCL6 <---------> GPC1_4
SCK <---------> BUF_GPC1_1 <---------> GPC1_1
INT <-------------------------------------------> BUF_GPX0_0
MCP2515芯片連接在4412芯片的SPI2上。
中斷連接在GPX0_0上; CS、SO、SI、SCK復(fù)用了GPIO引腳GPC1的引腳。
MCP2515輸出連接SN65HVD230 CAN總線收發(fā)器,SN65HVD230是德州儀器公司生產(chǎn)的3.3V CAN收發(fā)器。為了節(jié)省功耗,縮小電路體積,MCP2515 CAN總線控制器的邏輯電平采用LVTTL,SN65HVD230就是與其配套的收發(fā)器。
Cortex-A9 SPI控制器
exynos4412 scp中的串行外設(shè)接口(SPI)通過各種外設(shè)來傳輸串行數(shù)據(jù)。SPI包括兩個8、16和32位移位寄存器,用于傳輸和接收數(shù)據(jù)。在SPI傳輸過程中,它同時傳輸(串行移出)和接收(串行移位)數(shù)據(jù)。
特性
全雙工
用于Tx/Rx的8/16/32位移位寄存器
支持8位/16位/32位總線接口
支持摩托羅拉SPI協(xié)議和National Semiconductor Microwire
兩個獨(dú)立的32位寬的發(fā)送和接收FIFO:端口0的深度為64,端口1和2中的深度為16
主模式和從模式
接收而不發(fā)送操作
發(fā)送/接收最高頻率為50 MHz
SPI的操作
SPI在Exynos 4412 SCP和外部設(shè)備之間傳輸1位串行數(shù)據(jù)。 Exynos 4412 SCP中的SPI支持CPU或DMA分別同時發(fā)送或接收FIFO和雙向傳輸數(shù)據(jù)。 SPI有兩個信道,即Tx信道和Rx信道。Tx信道有來自Tx的路徑
FIFO到外部設(shè)備。Rx通道有從外部設(shè)備到Rx FIFO的路徑。
CPU(或DMA)必須將數(shù)據(jù)寫入寄存器SPI_TX_DATA,才能將數(shù)據(jù)寫入FIFO。寄存器上的數(shù)據(jù)會自動移動到Tx FIFO。要從Rx FIFO讀取數(shù)據(jù),CPU(或DMA)必須訪問寄存器SPI_RX_DATA,數(shù)據(jù)會自動發(fā)送到SPI_RX_DATA寄存器。
CMU寄存器可以控制SPI的工作頻率。
操作模式
SPI有兩種模式,即主模式和從模式。 在主模式下,生成SPICLK并將其傳輸?shù)酵獠吭O(shè)備。XspiCS#是選擇從機(jī)的信號,它指示在設(shè)置XspiCS時數(shù)據(jù)有效低水平。在發(fā)送或接收數(shù)據(jù)包之前,必須將XspiCS設(shè)置為低。
FIFO存取
SPI支持對fifo的CPU訪問和DMA訪問。 對fifo的CPU訪問和DMA訪問的數(shù)據(jù)大小從8位、16位或32位數(shù)據(jù)中選擇。當(dāng)它選擇8位數(shù)據(jù)大小時,有效位是0到7比特。用戶可以定義觸發(fā)閾值來引發(fā)CPU中斷。端口0中每個FIFO的觸發(fā)電平由從0到252字節(jié)的步進(jìn)為4個字節(jié),端口1中每個FIFO的字節(jié)從0到63字節(jié)按1個字節(jié)的步長設(shè)置。 SPI_MODE_CFG寄存器的TxDMAOn或RxDMAOn位必須設(shè)置為使用DMA訪問。DMA訪問支持只有單次傳輸和4突發(fā)傳輸。在Tx FIFO中,DMA請求信號是高的,直到Tx FIFO滿為止。在Rx FIFO中,如果FIFO不為空,DMA請求信號高。
片選控制
芯片選擇XspiCS是激活的低信號。換句話說,當(dāng)XspiCS輸入為0時,選擇芯片。 您可以自動或手動控制XspiCS。不需要改變。 手動模式 當(dāng)您使用手動控制模式時,您應(yīng)清除AUTO_N_MANUAL(默認(rèn)值為0)。NSSOUT位控制XspiCS級別。 自動模式 使用自動控制模式時,必須將AUTO_N_MANUAL設(shè)置為1。XspiCS在數(shù)據(jù)包和自動打包。NCS_TIME_COUNT 控制XspiCS的非激活期。NSSOUT在此時不可用。
SPI寄存器說明
配置寄存器CH_CFGn
【bit:5】 軟件復(fù)位的時候必須先將此位設(shè)置為1,給一個延時后再設(shè)置為0; 【bit:4】 設(shè)置SPI端口是主機(jī)還是從機(jī),0:主機(jī),1:從機(jī); 【bit:2-3】設(shè)置相位和極性; 【bit:1】接收數(shù)據(jù)的通道使能位; 【bit:0】發(fā)送數(shù)據(jù)的通道使能位。
數(shù)據(jù)寬度寄存器MODE_CFGn
設(shè)置總線數(shù)據(jù)寬度,一般設(shè)置為1個字節(jié)。
片選寄存器CS_REGn
只有NSSOUT 拉低,才會進(jìn)行數(shù)據(jù)的傳輸
移位寄存器狀態(tài)寄存器SPI_STATUSn
發(fā)送操作開始,如果移位寄存器空了,該值置1,通過該值判斷數(shù)據(jù)是否發(fā)送出去。
發(fā)送緩沖寄存器SPI_TX_DATA
SPI_TX_DATA
接收緩沖寄存器SPI_RX_DATA
SPI_RX_DATA
SPI初始化流程
設(shè)置GPIO引腳為SPI模式;
設(shè)置clock;
軟件復(fù)位;
設(shè)置CPOL CPHA 為 00 模式,并設(shè)置為主機(jī)模式;
設(shè)置數(shù)據(jù)位;
片選。
1. 設(shè)置GPIO引腳為SPI模式
因為CS、SO、SI、SCK復(fù)用了GPIO引腳GPC1的引腳。首先需要設(shè)置GPIO引腳為SPI模式。
如上所示,該寄存器設(shè)置方式如下:
GPC1.CON = (GPC1.CON & ~0xffff0) | 0x55550;//設(shè)置IO引腳為SPI模式
Step 1 設(shè)置CPOL CPHA 為 00 模式
在這里插入圖片描述
SPI2.CH_CFG&=~((0x1<< 4)|(0x1<<3)|(0x1<< 2)|0x3);
//master mode, CPOL = 0, CPHA = 0 (Format A)
2 設(shè)置clock
時鐘的設(shè)置需要依賴鎖相環(huán)(PLL)時鐘產(chǎn)生器。
從30.2.1節(jié)可知,時鐘配置需要參考CMU這一章。
由上圖可知SPI 的clock源是SCLK_SPI。
搜索SCLK_SPI
從第七章搜所有的SPI
繼續(xù)搜索
由此可知SPI0~2的之中受CLKMPLL_USER_T 控制,繼續(xù)搜索。 而此時鐘位于寄存器CLK_SRC_PERIL1的bit[27:24]。
繼續(xù)搜索SPI2。
搜索到CLK_SRC_MASK_PERIL1、CLK_DIV_PERIL2。 可知寄存器CLK_SRC_MASK_PERIL1默認(rèn)是打開的,所以不用設(shè)置。 CLK_DIV_PERIL2用來設(shè)置SPI2分頻的,于是寄存器設(shè)置如下:
CLK_SRC_PERIL1 = (CLK_SRC_PERIL1 & ~(0xF<<24)) | 6<<24;// 0x6: 0110 = SCLKMPLL_USER_T 800MhzCLK_DIV_PERIL2 = 19 <<8 | 3;//SPI_CLK = 800/(19+1)/(3+1)
3. 軟件復(fù)位
在這里插入圖片描述
void soft_reset(void){ SPI2.CH_CFG |= 0x1 << 5; delay(1); //延時 SPI2.CH_CFG &= ~(0x1 << 5); }
4. 設(shè)置CPOL CPHA 為 00 模式,并設(shè)置為主機(jī)模式
SPI2.CH_CFG &= ~( (0x1 << 4) | (0x1 << 3) | (0x1 << 2) | 0x3);
5. 設(shè)置數(shù)據(jù)位MODE_CFG
MODE_CFG
SPI2.MODE_CFG &= ~((0x3 << 17) | (0x3 << 29)); //BUS_WIDTH=8bit,CH_WIDTH=8bit
6. 片選
SPI2.CS_REG &= ~(0x1 << 1); //選擇手動選擇芯片
初始化搞定后,下面我們來看如何利用SPI收發(fā)數(shù)據(jù)。
收發(fā)數(shù)據(jù)流程
收發(fā)數(shù)據(jù)流程如下:
spi復(fù)位
片選從機(jī)
收發(fā)數(shù)據(jù)
取消片選
【注意】一下代碼為設(shè)置每次只讀寫1個byte數(shù)據(jù),每次讀寫1個byte數(shù)據(jù)都要按照上述步驟操作。
1 spi復(fù)位
void soft_reset(void){ SPI2.CH_CFG |= 0x1 << 5; delay(1); //延時 SPI2.CH_CFG &= ~(0x1 << 5); }
2 片選從機(jī)
void slave_enable(void){ SPI2.CS_REG &= ~0x1; //enable salve delay(3); }
3.1 發(fā)送數(shù)據(jù)
void send_byte(unsigned char data){ SPI2.CH_CFG |= 0x1; // enable Tx Channel delay(1); SPI2.SPI_TX_DATA = data; while( !(SPI2.SPI_STATUS & (0x1 << 25)) ); SPI2.CH_CFG &= ~0x1; // disable Tx Channel}
3.2 接收數(shù)據(jù)
unsigned char recv_byte(){ unsigned char data; SPI2.CH_CFG |= 0x1 << 1; // enable Rx Channel delay(1); data = SPI2.SPI_RX_DATA; delay(1); SPI2.CH_CFG &= ~(0x1 << 1); //disable Rx Channel return data; }
取消片選
void slave_disable(void){ SPI2.CS_REG |= 0x1; //disable salve delay(1); }
OK,到底為止,如何通過SPI收發(fā)數(shù)據(jù),我們已經(jīng)可以實(shí)現(xiàn)了,但是SPI往往外部回接各種各樣的從設(shè)備。下面我們來看SPI如何操作從設(shè)備。
上面我們說了,fs4412開發(fā)板的SPI2外設(shè)外接了MCP2515,現(xiàn)在我們來分析一下MCP2515。
MCP2515
MCP2515詳細(xì)資料,大家自行搜索 MCP2515 datasheet 《帶有 SPI 接口的獨(dú)立 CAN 控制器》。
簡介
MCP2515是一種獨(dú)立的CAN總線通信控制器,是Microchip公司首批獨(dú)立CAN解決方案的升級器件,其傳輸能力較Microchip公司原有CAN控制器(MCP2510)高兩倍,高通信速率可達(dá)到1Mbps。MCP2515能夠接收和發(fā)送標(biāo)準(zhǔn)數(shù)據(jù)幀和擴(kuò)展數(shù)據(jù)幀以及遠(yuǎn)程幀,通過兩個接收屏蔽寄存器和六個接收過濾寄存器濾除無關(guān)報文,從而減輕CPU負(fù)擔(dān)。
特性
MCP2515主要功能參數(shù)及電氣特性如下:
(1)支持CAN技術(shù)規(guī)范2.0A/B, 高傳輸速率達(dá)到1Mbps;
(2)支持標(biāo)準(zhǔn)數(shù)據(jù)幀、擴(kuò)展數(shù)據(jù)幀和遠(yuǎn)程幀,每幀數(shù)據(jù)域長度可為0~8個字節(jié);
(3)內(nèi)含兩個的接收緩沖器和三個發(fā)送緩沖器,并且可編程設(shè)定優(yōu)先級;
(4)內(nèi)含六個29位(bit)的接收過濾寄存器和兩個29位(bit)的接收屏蔽寄存器;
(5)高速SPI接口,支持SPI 0,0和1,1模式;
(6)一次性模式可確保報文被一次性傳輸;
(7)具有可編程時鐘脈沖輸出引腳,可作為其他芯片時鐘信號源;
(8) 幀起始(SOF)信號輸出功能可被用于在確定的系統(tǒng)中(如時間觸發(fā)CAN-TTCAN)執(zhí)行時隙功能,或在CAN總線診斷中決定早期總線出級;
(9) 采用低功耗CMOS技術(shù),工作電壓:2.7V~5.5V, 工作電流:5mA(待機(jī)狀態(tài)1μA);
(10)工作溫度范圍:(I)-40℃到+85℃,(E)-40℃到+125℃。
結(jié)構(gòu)框圖
在這里插入圖片描述
CAN模塊
MCP2515 是一款獨(dú)立 CAN 控制器, 可簡化需要與 CAN總線連接的應(yīng)用。如上圖 簡要顯示了 MCP2515 的結(jié)構(gòu)框圖。該器件主要由三個部分組成:
CAN 模塊,包括 CAN 協(xié)議引擎、驗收濾波寄存 器、驗收屏蔽寄存器、發(fā)送和接收緩沖器。
用于配置該器件及其運(yùn)行的控制邏輯和寄存器。
SPI 協(xié)議模塊。
CAN 模塊的功能是處理所有 CAN 總線上的報文接收和發(fā)送。報文發(fā)送時,首先將報文裝載到正確的報文緩沖器和控制寄存器中。通過 SPI 接口設(shè)置控制寄存器中的相應(yīng)位或使用發(fā)送使能引腳均可啟動發(fā)送操作。通過讀取相應(yīng)的寄存器可以檢查通訊狀態(tài)和錯誤。 會對在 CAN總線上檢測到的任何報文進(jìn)行錯誤檢查,然后與用戶定義的濾波器進(jìn)行匹配,以確定是否將報文移到兩個接收緩沖器中的一個。
SPI 協(xié)議模塊
MCU通過SPI接口與該器件連接。使用標(biāo)準(zhǔn)的SPI讀/寫指令以及專門的 SPI 命令來讀 / 寫所有的寄存器。 MCP2515 設(shè)計可與許多單片機(jī)的串行外設(shè)接口( SPI)直接相連,支持 0,0 和 1,1 運(yùn)行模式。外部數(shù)據(jù)和命令通過 SI 引腳傳送到器件中,且數(shù)據(jù)在 SCK 時鐘信號的上升沿傳送進(jìn)去。 MCP2515 在 SCK 的下降沿通過 SO引腳傳送出去。在進(jìn)行任何操作時, CS 引腳都必須保持為低電平。
與MCP2515通信
SPI指令集
我們要想操作MCP2515,只能用過SPI總線向MCP2515發(fā)送一些數(shù)據(jù),根據(jù)規(guī)定,發(fā)送不同的數(shù)據(jù)就代表不同的操作,于是就形成了對應(yīng)的指令集。
如上圖所示,比如我們要執(zhí)行復(fù)位操作,那么我只需要通過SPI總線寫入11000000,即0xC0即可。
下面我們來詳細(xì)分析如何向MCP2515發(fā)送復(fù)位命令,如何讀寫數(shù)據(jù)。
復(fù)位
因為復(fù)位只需要發(fā)送0XC0即可,并不需要其他額外操作,所以我們只需要根據(jù)我們之前章節(jié)所講的SPI發(fā)送數(shù)據(jù)流程操作接口。
void reset_2515() { soft_reset(); //復(fù)位spi控制器 slave_enable() ; //片選從機(jī) send_byte(0xc0); //發(fā)送復(fù)位命令 slave_disable() ; //取消片選}
讀取數(shù)據(jù)
由上圖可知,讀取數(shù)據(jù)流程如下:
片選從機(jī)
通過SPI發(fā)送指令0x03
發(fā)送地址
讀取數(shù)據(jù)
取消片選
unsigned char read_byte_2515(unsigned char Addr){ unsigned char ret; slave_enable(); send_byte(0x03); send_byte(Addr); ret = recv_byte(); slave_disable(); return(ret); }
發(fā)送數(shù)據(jù)
由上圖可知,發(fā)送數(shù)據(jù)流程如下:
片選從機(jī)
通過SPI發(fā)送指令0x02
發(fā)送地址
發(fā)送數(shù)據(jù)
取消片選
void write_byte_2515(unsigned char addr,unsigned char data){ slave_enable(); send_byte(0x02); send_byte(addr); send_byte(data); slave_disable(); }
讀RX緩沖器
在這里插入圖片描述
裝載TX緩沖器
在這里插入圖片描述
請求發(fā)送(RTS)指令
在這里插入圖片描述
位修改指令
有些寄存器要修改對應(yīng)的bit,必須先發(fā)送該指令,發(fā)送屏蔽字節(jié),然后才可以修改屏蔽字節(jié)中對應(yīng)位為1的值,后面會有例子給出。
CAN
知道該如何和CAN通信,下面我們來看如何初始化CAN。
CAN初始化
CAN的初始化步驟如下:
MCP2515復(fù)位
設(shè)置MCP2515為配置模式
位定時配置,有配置寄存器CNF1,CNF2,CNF3控制
中斷使能
接收緩沖器配置
引腳控制寄存器和狀態(tài)寄存器
此外為方便測試,我們再加一步: 7. 回環(huán)模式,用于測試
1. MCP2515復(fù)位
上一節(jié)已經(jīng)實(shí)現(xiàn)。
void reset_2515()
2. 設(shè)置MCP2515為配置模式
由上圖所示,只需要向地址0X0F寫入數(shù)據(jù)0x80即可。
write_byte_2515(0x0f, 0x80); //CANCTRL寄存器--進(jìn)入配置模式 中文DATASHEET 58頁
3. 位定時配置,有配置寄存器CNF1,CNF2,CNF3控制
CAN總線上的所有節(jié)點(diǎn)都必須具有相同的標(biāo)稱比特率。CAN 協(xié)議采用不歸零( Non Return to Zero, NRZ)編碼方式,在數(shù)據(jù)流中不對時鐘信號進(jìn)行編碼。因此,接收時鐘信號必須由接收節(jié)點(diǎn)恢復(fù)并與發(fā)送器的時鐘同步。
由于不同節(jié)點(diǎn)的振蕩器頻率和傳輸時間不同,接收器應(yīng)具有某種能與數(shù)據(jù)傳輸邊沿同步的鎖相環(huán)( Phase Lock Loop, PLL)來同步時鐘并保持這種同步。
鑒于數(shù)據(jù)采用 NRZ 編碼,有必要進(jìn)行位填充以確保至少每 6 位時間發(fā)生一次邊沿,使數(shù)字鎖相環(huán) ( Digital Phase LockLoop, DPLL)同步。MCP2515 通過 DPLL 實(shí)現(xiàn)位定時。 DPLL 被配置成同輸入數(shù)據(jù)同步,并為發(fā)送數(shù)據(jù)提供標(biāo)稱定時。 DPLL 將每一 個 位 時 間 分 割 為 由 最 小 單 位 為 時 間 份 額 ( Time Quanta, TQ)所組成的多個時間段。
在位時間幀中執(zhí)行的總線定時功能,例如與本地振蕩器同步、網(wǎng)絡(luò)傳輸延遲補(bǔ)償和采樣點(diǎn)定位等,都是由DPLL 的可編程位定時邏輯來規(guī)定的。
CNF1 BRP<5:0> 控制波特率預(yù)分頻比的設(shè)置。這些位根據(jù)OSC1 輸入頻率設(shè)置 TQ 的時間長度。當(dāng) BRP<5:0> =‘b000000’, TQ 最小值取 2 TOSC。通過 SJW<1:0> 選擇以 TQ 計的同步跳轉(zhuǎn)寬度。
CNF2 PRSEG<2:0> 位設(shè)定以 TQ 計的傳播段時間長度。PHSEG1<2:0>位設(shè)定以TQ計的相位緩沖段PS1的時間長度。
CNF3 如果 CNF2.BTLMODE 位為 1,則相位緩沖段 PS2 的時間長度將由 PHSEG2<2:0> 位設(shè)定,以 TQ 計。如果BTLMODE 位為 0,則 PHSEG2<2:0> 位不起作用。
MCP2515 波特率配置以16M晶振為例, SJW 段數(shù)(1+PRSEG+PRSEG1+PRSEG2)
#define CNF1_20K 0xd3 //4 20(1+4+8+7) #define CNF2_20K 0xfb #define CNF3_20K 0x46 //可以設(shè)置的波特率 5K 10K 15K 20K 25K 40K 50K 80K 100K 125K 200K 400K 500K 667K 800K 1Mwrite_byte_2515(0x2A, CNF1_20K); //CNF1位定時配置寄器 write_byte_2515(0x29, CNF2_20K); //CNF2位定時配置寄器 write_byte_2515(0x28, CNF3_20K); //CNF3位定時配置寄器
4. 中斷使能
MCP2515 有八個中斷源。 CANINTE 寄存器包含了使能各中斷源的中斷使能位。 CANINTF 寄存器包含了各中斷源的中斷標(biāo)志位。當(dāng)發(fā)生中斷時, INT 引腳將被MCP2515 拉為低電平, 并保持低電平狀態(tài)直至 MCU 清除中斷。中斷只有在引起相應(yīng)中斷的條件消失后,才會被清除。 建議在對 CANINTF 寄存器中的標(biāo)志位進(jìn)行復(fù)位操作時,采用位修改命令而不要使用一般的寫操作。這是為了避免在寫命令執(zhí)行過程中無意間修改了標(biāo)志位,進(jìn)而導(dǎo)致中斷丟失。 應(yīng)該注意的是, CANINTF 中的中斷標(biāo)志位是可讀寫位,因此在相關(guān) CANINTE 中斷使能位置 1 的前提下,對上述任一位置 1 均可使 MCU 產(chǎn)生中斷請求。
write_byte_2515(0x2B, 0x1f); //CANINTE中斷使能寄存器
5. 接收緩沖器配置
MCP2515 具有兩個全接收緩沖器。每個接收緩沖器配備 有多 個 驗 收濾 波 器。除 上述 專 用 接收 緩 沖 器外,MCP2515 還具有單獨(dú)的報文集成緩沖器 ( Message Assembly Buffer, MAB) ,可作為第三個接收緩沖器。
bit5:6設(shè)置為1,其余位暫時不用,設(shè)置為0.
write_byte_2515(0x60, 0x60); //RXB0CTRL接收緩沖器0 控制寄存器
6. 引腳控制寄存器和狀態(tài)寄存器
當(dāng)引腳配置為數(shù)字輸出引腳時,相應(yīng)的接收緩沖器中的BFPCTRL.BxBFM位應(yīng)被清零, 而BFPCTRL.BnBFE位應(yīng)被置 1。在這種工作模式下,引腳的狀態(tài)由 BFPCTRL.BnBFS位控制。 BnBFS位寫入1時,將使相應(yīng)的緩沖器滿中斷引腳輸出高電平,寫入 0 將使該引腳輸出低電平。當(dāng)引腳處于這種模式時,該引腳的狀態(tài)只應(yīng)通過位修改 SPI 命令來修改,以避免任何緩沖器滿中斷引腳出現(xiàn)干擾。
在這里插入圖片描述
void bit_modify_2515(unsigned char addr, unsigned char mask, unsigned char data){// CS_SPI = 0 ; slave_enable() ; send_byte(0x05) ; send_byte(addr) ; send_byte(mask) ; send_byte(data) ; slave_disable() ;// CS_SPI = 1 ;} bit_modify_2515(0x0C, 0x0f, 0x0f); //BFPCTRL_RXnBF 引腳控制寄存器和狀態(tài)寄存器 中文DATASHEET 29 頁
7. 回環(huán)模式,用于測試
write_byte_2515(0x0f, 0x40); //CAN控制寄存器--回環(huán)模式,用于測試
can緩沖區(qū)數(shù)據(jù)收發(fā)
MCP2515 采用三個發(fā)送緩沖器。每個發(fā)送緩沖器占用14 字節(jié)的 SRAM,并映射到器件存儲器中。 其中第一個字節(jié) TXBnCTRL 是與報文緩沖器相關(guān)的控制寄存器。該寄存器中的信息決定了報文在何種條件下發(fā)送,并在報文發(fā)送時指示其狀態(tài) 。
用 5 個字節(jié)來裝載標(biāo)準(zhǔn)和擴(kuò)展標(biāo)識符以及其他報文仲裁信息(見寄存器 3-3 到寄存器 3-7) 。最后 8 個字節(jié)用于裝載等待發(fā)送報文的 8 個可能的數(shù)據(jù)字節(jié) 。
至少須將 TXBnSIDH、 TXBnSIDL 和 TXBnDLC 寄存器裝載數(shù)據(jù)。如果報文包含數(shù)據(jù)字節(jié),還需要對 TXBnDm寄存器進(jìn)行裝載。若報文采用擴(kuò)展標(biāo)識符,應(yīng)對 TXBnEIDm 寄存器進(jìn)行裝載,并將 TXBnSIDL.EXIDE 位置1。
下面我們看如何向CAN的緩沖區(qū)0發(fā)送和接收數(shù)據(jù)。
數(shù)據(jù)發(fā)送
can緩沖區(qū)0數(shù)據(jù)發(fā)送流程如下:
設(shè)置為發(fā)送最高優(yōu)先級
設(shè)置發(fā)送緩沖器0標(biāo)準(zhǔn)標(biāo)識符高位
設(shè)置發(fā)送緩沖器0標(biāo)準(zhǔn)標(biāo)識符低位
設(shè)置發(fā)送緩沖器0數(shù)據(jù)長度碼8字節(jié)
向緩沖區(qū)寫入數(shù)據(jù),地址從0x36起
發(fā)送請求命令0x81,發(fā)送數(shù)據(jù)
1. 設(shè)置為發(fā)送最高優(yōu)先級
TXBnCTRL
write_byte_2515(0x30, 0x03); //設(shè)置為發(fā)送最高優(yōu)先級
2. 設(shè)置發(fā)送緩沖器0標(biāo)準(zhǔn)標(biāo)識符高位
在這里插入圖片描述
write_byte_2515(0x31, 0xff); //發(fā)送緩沖器0標(biāo)準(zhǔn)標(biāo)識符高位
3. 設(shè)置發(fā)送緩沖器0標(biāo)準(zhǔn)標(biāo)識符低位
在這里插入圖片描述
write_byte_2515(0x32, 0x00); //發(fā)送緩沖器0標(biāo)準(zhǔn)標(biāo)識符低位
4. 設(shè)置發(fā)送緩沖器0數(shù)據(jù)長度碼8字節(jié)
在這里插入圖片描述
write_byte_2515(0x35, 0x08); //發(fā)送緩沖器0數(shù)據(jù)長度碼8字節(jié)
5. 向緩沖區(qū)寫入數(shù)據(jù),地址從0x36起
在這里插入圖片描述
write_byte_2515(0x36+i ,tx_buff[i]); //向txb緩沖器中寫入8個字節(jié)
6. 發(fā)送請求命令0x81,發(fā)送數(shù)據(jù)
void send_req_2515() { // CS_SPI = 0; //復(fù)位 soft_reset(); //復(fù)位spi控制器 slave_enable() ; //片選從機(jī) send_byte(0x81); //發(fā)送請求命令 slave_disable() ; //取消片選// CS_SPI=1;}
Can數(shù)據(jù)的接收
從CAN緩沖區(qū)讀取數(shù)據(jù)流程如下:
讀取中斷標(biāo)志寄存器0x2c的value,判斷bit0是否為1
從接收緩沖區(qū)讀走數(shù)據(jù),地址從0X66開始
軟復(fù)位
向中斷標(biāo)志寄存器0x2c寫入位掩碼
向中斷標(biāo)志寄存器0x2c,寫入數(shù)據(jù)0,清終端
1. 讀取中斷標(biāo)志寄存器0x2c的value,判斷bit0是否為1
當(dāng)報文傳送至某一接收緩沖器時,與該接收緩沖器對應(yīng)的 CANINTF.RXnIF 位將置 1。一旦緩沖器中的報文處理完畢, MCU 就必須將該位清零以接收下一條報文。該控制位提供的鎖定功能確保 MCU 尚未處理完上一條報文前, MCP2515 不會將新的報文載入接收緩沖器。
如果 CANINTE.RXnIE 位被置 1,器件會在 INT 引腳產(chǎn)生一個中斷,顯示接收到報文有效。另外,如果被配置為接收緩沖器滿中斷引腳,與之相應(yīng)的 RXnBF 引腳會被拉低。
MCP2515 有八個中斷源。 CANINTE 寄存器包含了使能各中斷源的中斷使能位。 CANINTF 寄存器包含了各中斷源的中斷標(biāo)志位。當(dāng)發(fā)生中斷時, INT 引腳將被MCP2515 拉為低電平, 并保持低電平狀態(tài)直至 MCU 清除中斷。中斷只有在引起相應(yīng)中斷的條件消失后,才會被清除。
建議在對 CANINTF 寄存器中的標(biāo)志位進(jìn)行復(fù)位操作時,采用位修改命令而不要使用一般的寫操作。這是為了避免在寫命令執(zhí)行過程中無意間修改了標(biāo)志位,進(jìn)而導(dǎo)致中斷丟失。
應(yīng)該注意的是, CANINTF 中的中斷標(biāo)志位是可讀寫位,因此在相關(guān) CANINTE 中斷使能位置 1 的前提下,對上述任一位置 1 均可使 MCU 產(chǎn)生中斷請求。
2. 從接收緩沖區(qū)讀走數(shù)據(jù)
rx_buff[i]= read_byte_2515(0x66+i);
3. 軟復(fù)位
soft_reset();
4. 向中斷標(biāo)志寄存器0x2c寫入位掩碼
bit_modify_2515(0x2c,0x01,0x00);//修改bit 0
5.清中斷
write_byte_2515(0x2c, 0x00);
最終操作代碼如下
節(jié)省篇幅,重復(fù)函數(shù)不貼了。
#define CNF1_20K 0xd3 //4 20(1+4+8+7) #define CNF2_20K 0xfb #define CNF3_20K 0x46 void Init_can(void){ reset_2515(); //復(fù)位 write_byte_2515(0x0f, 0x80); //CANCTRL寄存器--進(jìn)入配置模式 中文DATASHEET 58頁 //可以設(shè)置的波特率 5K 10K 15K 20K 25K 40K 50K 80K 100K 125K 200K 400K 500K 667K 800K 1M write_byte_2515(0x2A, CNF1_20K); //CNF1位定時配置寄器 中文DATASHEET 41-42頁 write_byte_2515(0x29, CNF2_20K); //CNF2位定時配置寄器 中文DATASHEET 41-42頁 write_byte_2515(0x28, CNF3_20K); //CNF3位定時配置寄器 中文DATASHEET 41-43頁 write_byte_2515(0x2B, 0x1f); //CANINTE中斷使能寄存器 中文DATASHEET 50 頁 write_byte_2515(0x60, 0x60); //RXB0CTRL接收緩沖器0 控制寄存器 中文DATASHEET 27 頁 //write_byte_2515(0x70, 0x20); //接收緩沖器1控制寄存器 bit_modify_2515(0x0C, 0x0f, 0x0f); //BFPCTRL_RXnBF 引腳控制寄存器和狀態(tài)寄存器 中文DATASHEET 29 頁 write_byte_2515(0x0f, 0x40); //CAN控制寄存器--回環(huán)模式,用于測試}void send_byte(unsigned char data){ SPI2.CH_CFG |= 0x1; // enable Tx Channel delay(1); SPI2.SPI_TX_DATA = data; while( !(SPI2.SPI_STATUS & (0x1 << 25)) ); SPI2.CH_CFG &= ~0x1; // disable Tx Channel}unsigned char recv_byte(){ unsigned char data; SPI2.CH_CFG |= 0x1 << 1; // enable Rx Channel delay(1); data = SPI2.SPI_RX_DATA; delay(1); SPI2.CH_CFG &= ~(0x1 << 1); //disable Rx Channel return data; }void bit_modify_2515(unsigned char addr, unsigned char mask, unsigned char data){// CS_SPI = 0 ; slave_enable() ; send_byte(0x05) ; send_byte(addr) ; send_byte(mask) ; send_byte(data) ; slave_disable() ;// CS_SPI = 1 ;}void Can_send(unsigned char *tx_buff){ unsigned char i; write_byte_2515(0x30, 0x03); //設(shè)置為發(fā)送最高優(yōu)先級 write_byte_2515(0x31, 0xff); //發(fā)送緩沖器0標(biāo)準(zhǔn)標(biāo)識符高位 write_byte_2515(0x32, 0x00); //發(fā)送緩沖器0標(biāo)準(zhǔn)標(biāo)識符低位 write_byte_2515(0x35, 0x08); //發(fā)送緩沖器0數(shù)據(jù)長度碼8字節(jié) for(i = 0; i < 8; i++) { write_byte_2515(0x36+i ,tx_buff[i]); //向txb緩沖器中寫入8個字節(jié)// printf("%x ",tx_buff[i]); } send_req_2515(); }unsigned char Can_receive(unsigned char *rx_buff){ unsigned char i,flag; flag = read_byte_2515(0x2c); //CANINTF——中斷標(biāo)志寄存器 printf("flag=%xn",flag); // printf(" SPI2.SPI_STATUS =%xn", SPI2.SPI_STATUS ); // soft_reset(); if (flag&0x1) //接收緩沖器0滿中斷標(biāo)志位 { for(i = 0; i < 16; i++) { rx_buff[i]= read_byte_2515(0x66+i); // printf("%x ",rx_buff[i]); // printf(" SPI2.SPI_STATUS =%xn", SPI2.SPI_STATUS ); soft_reset(); } bit_modify_2515(0x2c,0x01,0x00); write_byte_2515(0x2c, 0x00); if (!(rx_buff[1]&0x08)) return(1); //接收標(biāo)準(zhǔn)數(shù)據(jù)幀 } return(0); }int main(void){ GPX2.CON = 0x1 << 28; uart_init(); unsigned char ID[4],buff[8]; //狀態(tài)字 unsigned char key; unsigned char ret;//,j,k,ret0,ret1,ret2,ret3,ret4,ret5,ret6,ret7,ret8,ret9; unsigned int rx_counter; volatile int i=0; GPC1.CON = (GPC1.CON & ~0xffff0) | 0x55550;//設(shè)置IO引腳為SPI模式/*spi clock config*/ CLK_SRC_PERIL1 = (CLK_SRC_PERIL1 & ~(0xF<<24)) | 6<<24;// 0x6: 0110 = SCLKMPLL_USER_T 800Mhz CLK_DIV_PERIL2 = 19 <<8 | 3;//SPI_CLK = 800/(19+1)/(3+1) soft_reset(); // 軟復(fù)位SPI控制器 SPI2.CH_CFG &= ~( (0x1 << 4) | (0x1 << 3) | (0x1 << 2) | 0x3);//master mode, CPOL = 0, CPHA = 0 (Format A) SPI2.MODE_CFG &= ~((0x3 << 17) | (0x3 << 29)); //BUS_WIDTH=8bit,CH_WIDTH=8bit SPI2.CS_REG &= ~(0x1 << 1); //選擇手動選擇芯片 mydelay_ms(10); //延時 Init_can(); //初始化MCP2515 printf("n************ SPI CAN test!! ************n"); while(1) { //Turn on led GPX2.DAT = GPX2.DAT | 0x1 << 7; mydelay_ms(50); printf("nplease input 8 bytesn"); for(i=0;i<8;i++) { src[i] = getchar(); putc(src[i]); } printf("n"); Can_send(src); //發(fā)送標(biāo)準(zhǔn)幀 mydelay_ms(100); ret = Can_receive(dst); //接收CAN總線數(shù)據(jù) printf("ret=%xn",ret); printf("src="); for(i=0;i<8;i++) printf(" %x", src[i]);//將CAN總線上收到的數(shù)據(jù)發(fā)到串行口 printf("n"); printf("dst="); for(i=0;i<8;i++) printf(" %x",dst[6+i]); //將CAN總線上收到的數(shù)據(jù)發(fā)到串行口 printf("n"); //Turn off GPX2.DAT = GPX2.DAT & ~(0x1 << 7); mydelay_ms(100); } //while(1) return 0; } //main
問題
1 什么是feddback 時鐘?
2 Can的緩沖寄存器組 原理 更多信息,請關(guān)注 公號 一口Linux
評論