新聞中心

        EEPW首頁(yè) > 嵌入式系統(tǒng) > 設(shè)計(jì)應(yīng)用 > AVR單片機(jī)串行接口SPI接口應(yīng)用設(shè)計(jì)

        AVR單片機(jī)串行接口SPI接口應(yīng)用設(shè)計(jì)

        作者: 時(shí)間:2011-11-24 來(lái)源:網(wǎng)絡(luò) 收藏

          使用的同步串行三線,可以方便的連接采用SPI通信協(xié)議的外圍或另一片,實(shí)現(xiàn)在短距離內(nèi)的高速同步通信。ATmega128的SPI采用硬件方式實(shí)現(xiàn)面向字節(jié)的全雙工3線同步通信,支持主機(jī)、從機(jī)和2種不同極性的SPI時(shí)序,通信速率有7種選擇,主機(jī)方式的最高速率為1/2系統(tǒng)時(shí)鐘,從機(jī)方式最高速率為1/4系統(tǒng)時(shí)鐘。

          ATmega128單片機(jī)內(nèi)部的也被用于程序存儲(chǔ)器和數(shù)據(jù)E2PROM的編程下載和上傳。但特別需要注意的是,此時(shí)SPI的MOSI和MISO接口不再對(duì)應(yīng)PB2、PB3引腳,而是轉(zhuǎn)換到PE0、PE1引腳上(PDI、PDO),其詳見(jiàn)第二章中關(guān)于程序存儲(chǔ)器的串行編程和校驗(yàn)部分的內(nèi)容。

          ATmega128的SPI為硬件接口和傳輸完成中斷申請(qǐng),所以使用SPI傳輸數(shù)據(jù)的有效方法是采用中斷方式+數(shù)據(jù)緩存器的設(shè)計(jì)方法。在對(duì)SPI初始化時(shí),應(yīng)注意以下幾點(diǎn):

          .正確選擇和設(shè)置主機(jī)或從機(jī),以及工作模式(極性),數(shù)據(jù)傳輸率;

          .注意傳送字節(jié)的順序,是低位優(yōu)先(LSB First)還是高位優(yōu)先(MSB Frist);

          .正確設(shè)置MOSI和MISO接口的輸入輸出方向,輸入引腳使用上拉電阻,可以節(jié)省總線上的吊高電阻。

          下面一段是SPI主機(jī)方式連續(xù)發(fā)送(接收)字節(jié)的例程:

          #Define SIZE 100

          Unsigned Char SPI_rx_buff[SIZE];

          Unsigned Char SPI_tx_buff[SIZE];

          Unsigned Char Rx_wr_index,Rx_rd_index,Rx_counter,Rx_buffer_overflow;

          Unsigned Char Tx_wr_index,Tx_rd_index,Tx_counter;

          #Pragma Interrupt_handler Spi_stc_isr:18

          Void Spi_stc_isr(Void)

          {

          SPI_rx_buff[Rx_wr_index] = SPDR; //從ISP口讀出收到的字節(jié)

          If (++Rx_wr_index == SIZE) Rx_wr_index = 0; //放入接收緩沖區(qū),并調(diào)整隊(duì)列指針

          If (++Rx_counter == SIZE)

          {

          Rx_counter = 0;

          Rx_buffer_overflow = 1;

          }

          If (Tx_counter) //如果發(fā)送緩沖區(qū)中有待發(fā)的數(shù)據(jù)

          {

          --Tx_counter;

          SPDR = SPI_tx_buff[Tx_rd_index]; //發(fā)送一個(gè)字節(jié)數(shù)據(jù),并調(diào)整指針

          If (++Tx_rd_index == SIZE) Tx_rd_index = 0;

          }

          }

          Unsigned Char GetSPIchar(Void)

          {

          Unsigned Char Data;

          While (Rx_counter == 0); //無(wú)接收數(shù)據(jù),等待

          Data = SPI_rx_buff[Rx_rd_index]; //從接收緩沖區(qū)取出一個(gè)SPI收到的數(shù)據(jù)

          If (++Rx_rd_index == SIZE) Rx_rd_index = 0; //調(diào)整指針

          CLI();

          --Rx_counter;

          SEI();

          Return Data;

          }

          Void PutSPIchar(Char C)

          {

          While (Tx_counter == SIZE);//發(fā)送緩沖區(qū)滿,等待

          CLI();

          If (Tx_counter || ((SPSR 0x80) == 0))//發(fā)送緩沖區(qū)已中有待發(fā)數(shù)據(jù)

          { //或SPI正在發(fā)送數(shù)據(jù)時(shí)

          SPI_tx_buffer[Tx_wr_index] = C; //將數(shù)據(jù)放入發(fā)送緩沖區(qū)排隊(duì)

          If (++Tx_wr_index == SIZE) Tx_wr_index = 0; //調(diào)整指針

          ++Tx_counter;

          }

          Else

          SPDR = C; //發(fā)送緩沖區(qū)中空且SPI口空閑,直接放入SPDR由SIP口發(fā)送

          SEI();

          }

          Void Spi_init(Void)

          {

          Unsigned Chat Temp;

          DDRB |= 0x080; //MISO=Input And MOSI,SCK,SS = Output

          PORTB |= 0x80; //MISO上拉電阻有效

          SPCR = 0xD5; //SPI允許,主機(jī)模式,MSB,允許SPI中斷,極性方式01,1/16系統(tǒng)時(shí)鐘速率

          SPSR = 0x00;

          Temp = SPSR;

          Temp = SPDR; //清空SPI,和中斷標(biāo)志,使SPI空閑

          }

          Void Main(Void)

          {

          Unsigned Char I;

          CLI(); //關(guān)中斷

          Spi_init(); //初始化

          SEI(); //開(kāi)中斷

          While()

          {

          PutSPIchat(I); //發(fā)送一個(gè)字節(jié)

          I++;

          GetSPIchar(); //接收一個(gè)字節(jié)(第一個(gè)字節(jié)為空字節(jié))

          ………

          }

          }

          這個(gè)典型的SPI例程比較簡(jiǎn)單,主程序中首先對(duì)ATmega128的硬件SPI進(jìn)行初始化。在初始化過(guò)程中,將PORTB的MOSI、SCLK和SS引腳作為輸出,同時(shí)將MISO作為輸入引腳,并打開(kāi)上拉電阻。接著對(duì)SPI的寄存器進(jìn)行初始化設(shè)置,并空讀一次SPSR、SPDR寄存器(讀SPSR后再對(duì)SPDR操作將自動(dòng)清零SPI中斷標(biāo)志自動(dòng)清零),使ISP空閑等待發(fā)送數(shù)據(jù)。

          AVR的SPI由一個(gè)16位的循環(huán)移位寄存器構(gòu)成,當(dāng)數(shù)據(jù)從主機(jī)方移出時(shí),從機(jī)的數(shù)據(jù)同時(shí)也被移入,因此SPI的發(fā)送和接收在一個(gè)中斷服務(wù)中完成。在SPI中斷服務(wù)程序中,先從SPDR中讀一個(gè)接收的字節(jié)存入接收數(shù)據(jù)緩沖器中,再?gòu)陌l(fā)送數(shù)據(jù)緩沖器取出一個(gè)字節(jié)寫入SPDR中,由ISP發(fā)送到從機(jī)。數(shù)據(jù)一旦寫入SPDR,ISP硬件開(kāi)始發(fā)送數(shù)據(jù)。下一次ISP中斷時(shí),表示發(fā)送完成,并同時(shí)收到一個(gè)數(shù)據(jù)。類似本章介紹的USART接口的使用,程序中PutSPIchar()和GetSPIchar()為應(yīng)用程序的底層接口函數(shù)(SPI驅(qū)動(dòng)程序是SPI中斷服務(wù)程序),同時(shí)也使用了兩個(gè)數(shù)據(jù)緩沖器,分別構(gòu)成循環(huán)隊(duì)列。這種程序設(shè)計(jì)的思路,不但程序的結(jié)構(gòu)性完整,同時(shí)也適當(dāng)?shù)慕鉀Q了高速M(fèi)CU和低速串口之間的矛盾,實(shí)現(xiàn)程序中任務(wù)的并行運(yùn)行,提高了MCU的運(yùn)行效率。

          本例程是通過(guò)SPI批量輸出、輸入數(shù)據(jù)的示例,用戶可以使用一片ATmega128,將其MOSI和MISO兩個(gè)引腳連接起來(lái),構(gòu)成一個(gè)ISP接口自發(fā)自收的系統(tǒng),對(duì)程序進(jìn)行演示驗(yàn)證。需要注意,實(shí)際接收到的字節(jié)為上一次中斷時(shí)發(fā)出的數(shù)據(jù),即第一個(gè)收到的字節(jié)是空字節(jié)。

          讀懂和了解程序的處理思想,讀者可以根據(jù)需要對(duì)程序進(jìn)行改動(dòng),適合實(shí)際系統(tǒng)的使用。如在實(shí)際應(yīng)用中外接的從機(jī)是一片SPI接口的溫度芯片,協(xié)議規(guī)程為:主機(jī)先要連續(xù)發(fā)送3個(gè)字節(jié)的命令,然后從機(jī)才返回一個(gè)字節(jié)的數(shù)據(jù)。那么用戶程序可以先循環(huán)調(diào)用PutSPIchar()函數(shù)4次,將3個(gè)字節(jié)的命令和一個(gè)字節(jié)的空數(shù)據(jù)發(fā)送到從機(jī),然后等待一段時(shí)間,或處理一些其它的操作后,再循環(huán)調(diào)用GetSPIchar()函數(shù)4次,從接收數(shù)據(jù)緩沖器中連續(xù)讀取4個(gè)字節(jié),放棄前3個(gè)空字節(jié),第4個(gè)字節(jié)即為從機(jī)的返回?cái)?shù)據(jù)了。



        評(píng)論


        相關(guān)推薦

        技術(shù)專區(qū)

        關(guān)閉
        主站蜘蛛池模板: 盐亭县| 荣昌县| 江西省| 普兰县| 客服| 新安县| 宜川县| 贞丰县| 邳州市| 兴仁县| 宁波市| 布尔津县| 阿鲁科尔沁旗| 化德县| 都昌县| 信宜市| 思茅市| 瑞丽市| 井研县| 贵南县| 英山县| 福泉市| 拉孜县| 盐池县| 元阳县| 通江县| 南汇区| 新和县| 浦县| 嫩江县| 凤山县| 光泽县| 甘南县| 景宁| 苏州市| 确山县| 洪江市| 晋州市| 云阳县| 黑水县| 隆安县|