MAXQ3180微控制器為電表的多相模擬前端。它集成了現(xiàn)代多功能電能計量所需的所有功能。MAXQ3180通過串行外設接口(SPI?)總線將其讀數(shù)傳送給主機微控制器。本應用筆記描述了該接口是如何完成的,并提供了示例代碼,以幫助設計人員實現(xiàn)通信機制。
SPI概述
串行外設接口 (SPI) 是一種設備間總線協(xié)議,可在芯片之間提供快速、同步、全雙工通信。一個器件(主設備)驅動同步時鐘,并選擇多個從站中的哪一個被尋址。每個SPI外設由一個移位寄存器和控制電路組成,因此尋址串行外設接口SPI外設同時發(fā)送和接收。
圖1.SPI 從站的插圖。
SPI通信中使用了四種分立電路:
SCLK:所有設備使用的同步時鐘。主站驅動此時鐘,從設備接收時鐘。請注意,SCLK 可以門控,不需要在 SPI 事務之間驅動。
莫西:主人出來,奴隸進來。這是由主站驅動到SPI總線上所有從機的主數(shù)據(jù)線。只有選定的從站時鐘從MOSI數(shù)據(jù)。
MISO:主人進,奴出。這是由選定的從站驅動到主站的主數(shù)據(jù)線。只有選定的從機可以驅動該電路。事實上,它是SPI總線布置中唯一允許從站驅動的電路。
SSEL:此信號對于每個從站都是唯一的。當活動(通常為低電平)時,所選從機必須驅動MISO。
對于此討論,必須注意SPI外設同時發(fā)送和接收。想到這一點的一個方便方法是,主站總是發(fā)送一個字節(jié)并接收一個字節(jié)。
一些SPI外設犧牲了速度,轉而模擬半雙工操作。 MAXQ3180微控制器不是這種情況,它是一款真正的全雙工SPI從機。
本應用筆記的其余部分介紹如何在SPI總線上連接并成功使用MAXQ3180。
MAXQ3180通信概述
對于主機(即主機)來說,MAXQ3180看起來像一個由RAM和ROM組成的存儲器陣列。這是因為MAXQ3180中的ROM固件從RAM讀取其工作參數(shù),并將結果放在RAM中。因此,配置MAXQ3180就像對其RAM位置進行塊寫入一樣簡單。
一些MAXQ3180“存儲器”位置觸發(fā)器件內的動作,以“動態(tài)”計算電能計量結果。寫入這些位置是“nop”。RAM 和虛擬 ROM 位置的具體功能和用途超出了本文檔的范圍。這里重要的事實是,微控制器實際上只有兩種SPI通信操作:讀取和寫入。
MAXQ3180中的每個事務都從主站發(fā)送兩個字節(jié)開始,其中包含命令(即讀或寫)、訪問地址和訪問字節(jié)數(shù)。如上所述,每個SPI外設每接收一個字節(jié)返回一個字節(jié)。因此,MAXQ3180在接收到第一個命令字節(jié)后返回0xC1,在第二個命令字節(jié)后返回0xC2。該協(xié)議如圖 2 所示。
圖2.主機向MAXQ3180讀寫數(shù)據(jù)。
如果主服務器正在讀取一個或多個字節(jié),則必須發(fā)送虛擬字節(jié)。請記住,除非從站發(fā)送一些東西,否則主站無法從從站接收任何東西:發(fā)送一個字節(jié)來獲取一個字節(jié)。但是在收到命令后,MAXQ3180可能必須計算結果,因此當主機發(fā)送虛擬字節(jié)時可能還沒有準備好結果。因此,MAXQ3180在發(fā)送數(shù)據(jù)之前,總是發(fā)送零個或多個NAK字符(0x4E或ASCII'N'),后跟ACK字符(0x41或ASCII'A')。
如果主站正在寫入一個或多個字節(jié),它會在發(fā)送命令后立即發(fā)送要寫入的數(shù)據(jù)。MAXQ3180為每個數(shù)據(jù)字節(jié)返回ACK (0x41)。然后,它返回 NAK (0x4E),直到寫入周期完成,之后返回最終的 ACK。
注意,在最終ACK之后,MAXQ3180立即準備好開始下一筆交易;它不需要等待任何其他事件。甚至不需要切換 SSEL 即可開始下一個事務。MAXQ3180知道第一筆交易已經(jīng)結束,并準備好進行下一筆交易。
無論出于何種原因,如果需要復位主機與MAXQ3180之間的通信(例如,如果通信變得不同步),主機只需等待200ms即可從第一個命令字節(jié)重新啟動通信。200ms延遲通知MAXQ3180主機放棄前一個事務。
命令字節(jié)
命令字節(jié)告訴MAXQ3180:
請求的事務是讀取還是寫入
交易的長度
RAM 中要修改的地址(或要讀取的虛擬 ROM 地址)
圖3.命令字節(jié)的結構。
第一個命令字節(jié)(圖3)告訴MAXQ3180所請求的事務是讀還是寫,以及事務的長度。命令字節(jié)使用以下計劃:
長度代碼 | 數(shù)據(jù)長度 |
0b00 | 1 字節(jié) |
0b01 | 2 字節(jié) |
0b10 | 4 字節(jié) |
0b11 | 8 字節(jié) |
命令字節(jié) 1 的其余部分和所有命令字節(jié) 2 提供要訪問的 RAM 中字節(jié)的地址(或虛擬 ROM 函數(shù)的身份)。
主機軟件設計
雖然MAXQ3180包含硬件SPI控制器,但單個消息字節(jié)仍由ROM固件中的軟件例程處理。因此,連續(xù)字節(jié)之間需要延遲。在當前版本的MAXQ3180中,該延遲必須不小于100μs才能可靠工作。參見圖 4 和圖 5。
圖4.MAXQ3180讀數(shù)流程圖
圖5.編寫MAXQ3180的流程圖
代碼清單
提供代碼,用于將內置SPI主機的MAXQ2000微控制器連接至MAXQ3180。其他微控制器的用戶將需要提供自己的SPI基元,并可能修改高級子程序。
在下面的列表中,子例程導致程序線程在給定的微秒數(shù)內停止執(zhí)行。常量被定義為提供比字符超時更長的內容。dly_usSPI_TIMEOUT
在高級子例程中,ENUM 用于按名稱選擇寄存器。它索引數(shù)組,其中包括每個MAXQ3180寄存器的寄存器長度。請參見圖 6、7 和 8。register_lookup_table
|
圖6.基元的代碼。Send_SPI
long Read_AFE(enum METER_REGISTER_RECORD reg, uint16 reg_addr) { extern unsigned char record[8]; unsigned long x = 0; unsigned char i, regadd, command_code = 0; for(i=0; i<8; i++) record[i] = 0; switch(register_lookup_table[reg].register_length) { case 2: command_code |= 0x10; break; case 4: command_code |= 0x20; break; case 8: command_code |= 0x30; break; } command_code |= reg_addr >> 8; regadd = reg_addr & 0xff; /* Disable SPI to reset it */ SPICN_bit.SPIEN = 0; for(x=0; x<300; x++); SPICN_bit.SPIEN = 1; SPI_SELECT_0; i = 0; while((Send_SPI(command_code)!= 0xC1)&&(++i < SPI_COMMAND_RETRIES)) spi_comm_timeout(); x = 0xffffffff; if (i == SPI_COMMAND_RETRIES) goto spierror; Send_SPI(regadd); i = 0; while((Send_SPI(0) != 'A') && (++i < SPI_RETRIES)); if (i == SPI_RETRIES) goto spierror; x = 0; for(i=0; i |
圖7.() 子例程的代碼。ReadAFESPI_Read
long Read_AFE(enum METER_REGISTER_RECORD reg, uint16 reg_addr) { extern unsigned char record[8]; unsigned long x = 0; unsigned char i, regadd, command_code = 0; for(i=0; i<8; i++) record[i] = 0; switch(register_lookup_table[reg].register_length) { case 2: command_code |= 0x10; break; case 4: command_code |= 0x20; break; case 8: command_code |= 0x30; break; } command_code |= reg_addr >> 8; regadd = reg_addr & 0xff; /* Disable SPI to reset it */ SPICN_bit.SPIEN = 0; for(x=0; x<300; x++); SPICN_bit.SPIEN = 1; SPI_SELECT_0; i = 0; while((Send_SPI(command_code)!= 0xC1)&&(++i < SPI_COMMAND_RETRIES)) spi_comm_timeout(); x = 0xffffffff; if (i == SPI_COMMAND_RETRIES) goto spierror; Send_SPI(regadd); i = 0; while((Send_SPI(0) != 'A') && (++i < SPI_RETRIES)); if (i == SPI_RETRIES) goto spierror; x = 0; for(i=0; i |
圖8.() 子例程的代碼。Write_AFESPI_Write
審核編輯:郭婷
-
微控制器
+關注
關注
48文章
7552瀏覽量
151426 -
芯片
+關注
關注
455文章
50816瀏覽量
423672 -
接口
+關注
關注
33文章
8598瀏覽量
151163
發(fā)布評論請先 登錄
相關推薦
評論