最近偶爾需要用到SPI模塊。正巧看到SpinalHDL中所提供的SPI-Master設(shè)計。看完之后尤為佩服如此簡潔而又全面的設(shè)計方式。本篇不對SPI協(xié)議進(jìn)行講解,僅針對SpinalHDL中的SpiMasterCtrl模塊做使用說明。
源代碼值得一讀和學(xué)習(xí)。
》》SpiMasterCtrl配置參數(shù) SpiMasterCtrl的配置參數(shù)類為SpiMasterCtrlGenerics:
case class SpiMasterCtrlGenerics( ssWidth : Int, timerWidth : Int, dataWidth : Int = 8){ def ssGen = ssWidth != 0}
這里主要有三個參數(shù):
ssWidth:SPI使用片選信號時片選信號個數(shù)。即指定了SPI從設(shè)備個數(shù)。當(dāng)該值為0時,則只有一個從設(shè)備,且SPI不使用片選信號。
timerWidth: 時鐘分頻計數(shù)器位寬。由于SPI時鐘信號是由主時鐘計數(shù)分頻而來,因此該值指定了計數(shù)器的位寬,根據(jù)SPI想要跑的最小頻率來定義該值。
dataWidth:指定單次傳輸?shù)奈粚挕?/p>
》》
接口定義
SpiMasterCtrl的接口主要包含下面四組接口:
val config = in(SpiMasterCtrlConfig(generics))val cmd = slave Stream(SpiMasterCmd(generics))val rsp = master Flow(Bits(dataWidth bits))val spi = master(SpiMaster(ssWidth))
》》
config接口
config接口組成為:
case class SpiMasterCtrlConfig(generics : SpiMasterCtrlGenerics) extends Bundle{ val kind = SpiKind() val sclkToogle = UInt(generics.timerWidth bits) val ss = if(generics.ssGen) new Bundle { val activeHigh = Bits(generics.ssWidth bits) val setup = UInt(generics.timerWidth bits) val hold = UInt(generics.timerWidth bits) val disable = UInt(generics.timerWidth bits) } else null}
kind接口指定了SPI接口的屬性CPOL及CPHA的值
case class SpiKind() extends Bundle { val cpol = Bool val cpha = Bool}
sclkToggle指定SPI的運行時鐘頻率。SPI時鐘頻率為:
系統(tǒng)時鐘頻率/((sclkToggle+1)*2)
當(dāng)SPI使用了片選信號時,ss接口包含的內(nèi)容為:
activeHigh:每個bit對應(yīng)一個從設(shè)備片選信號是高電平有效還是低電平有效。
setup:片選信號使能和開始發(fā)送數(shù)據(jù)之間的時鐘周期間隔數(shù)(加1)。
hold:發(fā)送完數(shù)據(jù)到片選信號釋放前的時鐘周期數(shù)(加1)。
disable:片選信號釋放后到重新使能需間隔的時鐘周期數(shù)(加1)。
》》
cmd接口
cmd接口類型為Stream類型接口,包含內(nèi)容為SpiMasterCmd:
case class SpiMasterCmd(generics : SpiMasterCtrlGenerics) extends Bundle{ val mode = if(generics.ssGen) SpiMasterCtrlCmdMode() else null val args = Bits(Math.max(widthOf(SpiMasterCtrlCmdData(generics)), log2Up(generics.ssWidth) + 1 ) bits)
def isData = if(generics.ssGen) mode === SpiMasterCtrlCmdMode.DATA else True
def argsData = { val ret = SpiMasterCtrlCmdData(generics) ret.assignFromBits(args) ret } def argsSs = { val ret = SpiMasterCtrlCmdSs(generics) ret.assignFromBits(args) ret }}
mode表明指令的類型:
object SpiMasterCtrlCmdMode extends SpinalEnum(binarySequential){ val DATA, SS = newElement()}
DATA:數(shù)據(jù)發(fā)送指令。
SS:片選操作指令。
args為指令數(shù)據(jù)。根據(jù)指令形式的不同,可通過提供的argsData、argsSS方法解析出相應(yīng)的指令內(nèi)容:
case class SpiMasterCtrlCmdData(generics : SpiMasterCtrlGenerics) extends Bundle{ val data = Bits(generics.dataWidth bits) val read = Bool//True表示讀操作,F(xiàn)alse表示寫操作}
case class SpiMasterCtrlCmdSs(generics : SpiMasterCtrlGenerics) extends Bundle{ val enable = Bool //True:使能,F(xiàn)alse:Disable val index = UInt(log2Up(generics.ssWidth) bits) //指定待操作的片選信號位置}
》》rsp接口 該接口為flow形式。當(dāng)操作為數(shù)據(jù)讀操作時,數(shù)據(jù)的讀返回結(jié)果從該接口返回?!贰稴PI接口 對外SPI接口?!贰穌riveFrom SpiMasterCtrl提供了一個driveFrom方法將接口映射到總線的slaveFactory上:
def driveFrom(bus : BusSlaveFactory, baseAddress : Int = 0)(generics : SpiMasterCtrlMemoryMappedConfig) = new Area {
其實現(xiàn)了讀寫8bit的功能,同時帶有終端功能。SpianlHDL中提供了一個APB總線的demo:
感興趣的小伙伴可以照此實現(xiàn)自己的需求。個人需求位寬多于8bit,自己實現(xiàn)了一個映射,很容易~
編輯:lyn
-
SPI
+關(guān)注
關(guān)注
17文章
1801瀏覽量
95781 -
config
+關(guān)注
關(guān)注
0文章
15瀏覽量
4822
原文標(biāo)題:值得學(xué)習(xí)的SPI-Master
文章出處:【微信號:zhuyandz,微信公眾號:FPGA之家】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。
發(fā)布評論請先 登錄
WH-M溫濕度模塊安裝使用說明書 V1.0
干貨分享 | TSMaster AUTOSAR E2E使用說明

CAN/CANFD卡接口函數(shù)使用說明

評論