EM9280系列產(chǎn)品包括EM9280、EM9287和EM9281,是英創(chuàng)公司新一代的低成本嵌入式主板產(chǎn)品。該主板的SPI接口,在內(nèi)部DMA(Direct Memory Access直接內(nèi)存存?。C(jī)制的驅(qū)動(dòng)下,其最高數(shù)據(jù)傳輸速度可達(dá)20Mbps。另外SPI接口可支持4bit、8bit、16bit位長(zhǎng)的數(shù)據(jù)通訊;也可對(duì)SPI時(shí)序的極性及相位進(jìn)行設(shè)置。
針對(duì)SPI接口的應(yīng)用特點(diǎn),EM9280的SPI的驅(qū)動(dòng)進(jìn)行了專門的優(yōu)化,不僅可支持常規(guī)的SPI讀、寫操作,還可支持外部中斷觸發(fā)的讀寫操作。中斷觸發(fā)的讀寫操作,主要應(yīng)用于工業(yè)控制的高速數(shù)據(jù)采集。另一方面,針對(duì)AD芯片控制需求,SPI驅(qū)動(dòng)還支持混合讀寫模式的數(shù)據(jù)傳輸操作。
本文以下部分重點(diǎn)介紹SPI驅(qū)動(dòng)API的使用方法。
操作SPI設(shè)備的基本步驟
1、打開SPI設(shè)備文件,其設(shè)備文件名為“SPI1:”
2、根據(jù)應(yīng)用需求,設(shè)置SPI數(shù)據(jù)幀的基本參數(shù),包括數(shù)據(jù)長(zhǎng)度、波特率、時(shí)鐘極性等參數(shù)。
3、若需要用到外部中斷觸發(fā)SPI讀取操作,則需要設(shè)置外部GPIO中斷管腳,及中斷后的讀取數(shù)據(jù)的長(zhǎng)度。
4、設(shè)置完成后,對(duì)常規(guī)操作,即可使用標(biāo)準(zhǔn)的ReadFile函數(shù)接收SPI數(shù)據(jù)、使用WriteFile發(fā)送SPI數(shù)據(jù)。
5、對(duì)需要讀寫混合操作的,則需要調(diào)用DeviceIoControl來(lái)實(shí)現(xiàn)。
6、當(dāng)啟動(dòng)了外部中斷,則通過(guò)調(diào)用DeviceIoControl來(lái)等待外部事件,然后再調(diào)用ReadFile函數(shù)來(lái)讀取已緩沖在驅(qū)動(dòng)程序內(nèi)部的SPI數(shù)據(jù)。
7、調(diào)用CloseHandle將關(guān)閉SPI接口并清除相關(guān)設(shè)置。即使重新打開SPI設(shè)備文件,需重新設(shè)置SPI的參數(shù),才能進(jìn)行讀寫。
SPI數(shù)據(jù)幀參數(shù)設(shè)置
初始化SPI,需要用到下面這個(gè)數(shù)據(jù)結(jié)構(gòu):
typedef struct _SPIFrame
{
UCHAR ucBitLength; //SPI數(shù)據(jù)bit長(zhǎng)度,= 4、8、16
DWORD dwBitRate; //SPI波特率,20000000對(duì)應(yīng)20Mbps
BOOL bPhase; //時(shí)鐘相位
BOOL bPolarity; //時(shí)鐘極性
} SPIFrame , *PSPIFrame;
該數(shù)據(jù)結(jié)構(gòu)在hw_spi.h頭文件中進(jìn)行的定義,數(shù)據(jù)結(jié)構(gòu)中的變量說(shuō)明:
ucBitLength:SPI通訊的數(shù)據(jù)位長(zhǎng),EM9280/EM9287支持4bit、8bit、16bit三種數(shù)據(jù)位長(zhǎng)格式,在hw_spi.h中定義了這三種數(shù)據(jù)位長(zhǎng)的常量。
dwBitRate:SPI時(shí)鐘速率,為每秒傳輸?shù)腷it數(shù),參數(shù)20000000表示20Mbps,
bPhase:SPI時(shí)序相位設(shè)置(如下圖所示)
bPolarity:SPI時(shí)序極性設(shè)置(如下圖所示)
bPhase=0 , bPolarity=0
bPhase=1 , bPolarity=0
bPhase=0 , bPolarity=1
bPhase=1 , bPolarity=1
SPI設(shè)備的初始化例子
HANDLE hSPI;
SPIFrame ConfigSPI;
//打開設(shè)備驅(qū)動(dòng)文件
hSPI = CreateFile(L”SPI1:”, //name of device
GENERIC_READ|GENERIC_WRITE, //desired access
FILE_SHARE_READ|FILE_SHARE_WRITE, //sharing mode
NULL, //security attributes (ignored)
OPEN_EXISTING, //creation disposition
FILE_FLAG_RANDOM_ACCESS, //flags/attributes
NULL); //template file (ignored)
if(hSPI == FALSE )
{
printf('SPI Open False!!!\r\n');
return 0;
}
//配置SPI參數(shù)
ConfigSPI.ucBitLength=SSP_WORD_LENGTH_8BITS; //Len_8BITS
ConfigSPI.dwBitRate=10000000; //10Mbps
ConfigSPI.bPhase=0;
ConfigSPI.bPolarity=0;
DeviceIoControl(hSPI, //file handle to the driver
SPI_IOCTL_SSPCONFIGURE, //I/O control code
&ConfigSPI, //in buffer
sizeof(ConfigSPI), //in buffer size
NULL,
0,
NULL,
NULL)
SPI接口的單向讀寫操作
用標(biāo)準(zhǔn)的ReadFile和WriteFile就可實(shí)現(xiàn)常規(guī)的SPI數(shù)據(jù)接收(讀)或發(fā)送(寫)。
SPI數(shù)據(jù)接收的函數(shù)調(diào)用:
ReadFile(hSPI, //設(shè)備驅(qū)動(dòng)文件句柄
pDatBuf, //數(shù)據(jù)buffer指針,注意指針類型!
dwBufLength, //數(shù)據(jù)buffer的字節(jié)長(zhǎng)度
pdwBytesRead, //實(shí)際讀取的SPI數(shù)據(jù)字節(jié)數(shù)
NULL)
pDatBuf:數(shù)據(jù)BUFF指針。需要注意的是SPI數(shù)據(jù)幀長(zhǎng)度若為4-bit或8-bit,則每個(gè)SPI數(shù)據(jù)占用一個(gè)字節(jié),而對(duì)16-bit的SPI數(shù)據(jù),則占用2個(gè)字節(jié)。一般來(lái)說(shuō),對(duì)4-bit或8-bit的SPI傳輸,其數(shù)據(jù)buffer應(yīng)當(dāng)是BYTE類型的;對(duì)16-bit的SPI傳輸,數(shù)據(jù)buffer則為WORD類型的。
dwBufLength:需要傳輸?shù)臄?shù)據(jù)字節(jié)長(zhǎng)度。該參數(shù)是以字節(jié)為單位,其涵義也與SPI數(shù)據(jù)長(zhǎng)度有關(guān),對(duì)16-bit的SPI傳輸,dwBufLength應(yīng)為2的倍數(shù)。
pdwBytesRead:SPI數(shù)據(jù)實(shí)際接收的字節(jié)數(shù)。一個(gè)正確的SPI數(shù)據(jù)接收調(diào)用后,指針pdwByteRead所包含的數(shù)據(jù)應(yīng)等于dwBufLength,才能表示SPI數(shù)據(jù)接收?qǐng)?zhí)行完全正確。
SPI數(shù)據(jù)發(fā)送的函數(shù)調(diào)用:
WriteFile(hSPI, //設(shè)備驅(qū)動(dòng)文件句柄
pDatBuf, //數(shù)據(jù)buffer指針,事先應(yīng)把數(shù)據(jù)填入
dwBufLength, //數(shù)據(jù)buffer的字節(jié)長(zhǎng)度
pdwBytesWritten, //實(shí)際發(fā)送的SPI數(shù)據(jù)字節(jié)數(shù)
NULL)
發(fā)送函數(shù)的參數(shù)定義與接收函數(shù)的參數(shù)定義是一致的。特別的,一個(gè)正確的SPI數(shù)據(jù)發(fā)送調(diào)用后,指針pdwByteWritten所包含的數(shù)據(jù)應(yīng)等于dwBufLength。
讀寫混合型的SPI操作
在SPI的實(shí)際應(yīng)用,有時(shí)需要在一個(gè)連續(xù)的片選過(guò)程中,既有讀操作,也有寫操作。這時(shí)間需要用到所謂的混合型SPI操作。
混合型SPI操作需要用到以下數(shù)據(jù)結(jié)構(gòu):
typedef struct _SPITransfer
{
LPVOID pTxBuff; //SPI發(fā)送buffer指針
LPVOID pRxBuff; //SPI接收buffer指針
DWORD dwBufLength; //本次SPI傳輸?shù)淖止?jié)數(shù)
} SPITransfer;
pTxBuff:SPI輸出數(shù)據(jù)BUFF指針
pRxBuff:SPI讀入數(shù)據(jù)BUFF指針
dwBufLength:SPI數(shù)據(jù)傳輸長(zhǎng)度,以字節(jié)為單位
注意,EM9280的SPI接口僅支持半雙工操作,因此在上述結(jié)構(gòu)中,只能有一個(gè)buffer指針為有效指針,另一個(gè)必須為NULL。dwBufLength的定義與單向讀寫的定義一致。具體的傳輸是通過(guò)DeviceIoControl來(lái)實(shí)現(xiàn)的,舉例說(shuō)明,本例首先進(jìn)行發(fā)送1個(gè)字節(jié)(8-bit SPI),然后接收2個(gè)字節(jié)。
SPITransfer Trans[2];
BYTE Tx[16], Rx[16]; //buffer足夠大
Tx[0] = 0xE5; //發(fā)送的字節(jié)
Trans[0].pTxBuf = Tx;
Trans[0].pRxBuf = NULL;
Trans[0].dwBufLength = 1; //要發(fā)送1字節(jié)
Trans[1].pTxBuf = NULL;
Trans[1].pRxBuf = Rx;
Trans[1].dwBufLength = 2; //要接收2字節(jié)
DeviceIoControl(hSPI,
SPI_IOCTL_EXCHANGE,
Trans, //in buffer
sizeof(Trans) , //in buffer size
NULL,
0,
NULL,
NULL))
在上述調(diào)用中需要注意的是,DeviceIoControl()輸入?yún)?shù)中的buffer長(zhǎng)度必須是數(shù)據(jù)結(jié)構(gòu)SPITransfer大小的整倍數(shù),否則將被視作無(wú)效參數(shù)。
外部中斷觸發(fā)的SPI操作
外部中斷觸發(fā)的SPI操作,主要是利用SPI的高速特性,進(jìn)行實(shí)時(shí)的大數(shù)據(jù)量讀取。因?yàn)镾PI的接線非常簡(jiǎn)單,作為一種高效低成本的接口模式在工業(yè)控制領(lǐng)域有廣泛的應(yīng)用。使用這種SPI操作方式,需要用到以下數(shù)據(jù)結(jié)構(gòu):
typedef struct _SPI_IrqTransfer
{
DWORD dwGpioPin; //外部中斷管腳,上升沿觸發(fā)中斷
DWORD dwBufLength; //中斷觸發(fā)的SPI傳輸?shù)淖止?jié)數(shù),小于64KB
DWORD dwRVSD; //保留,必須設(shè)置為0
} SPI_IrqTransfer;
dwGpioPin:要用作外部中斷源的GPIO引腳
dwBufLength:要讀取的數(shù)據(jù)字節(jié)長(zhǎng)度
dwRVSD:系統(tǒng)保留,必須設(shè)置為0
在上述結(jié)構(gòu)中,dwBufLength的定義與單向讀寫的定義一致,如果dwGpioPin與dwBufLength同時(shí)設(shè)置為0,則將關(guān)閉已打開的GPIO中斷資源并禁止該功能啟動(dòng)。dwGpioPin為EM9280主板的GPIO引腳編號(hào),與GPIO操作時(shí)的引腳數(shù)據(jù)一致。注意:由于系統(tǒng)功能的占用,不是所有的GPIO引腳都可以用作外部中斷觸發(fā)源。
?EM9280可以使用的GPIO引腳有:GPIO0、GPIO1、GPIO6、GPIO7、GPIO10、GPIO11、GPIO20、GPIO21、GPIO22、GPIO23。
?EM9287和EM9281可以使用的GPIO引腳有:GPIO0 - GPIO23。
該操作的具體的設(shè)置操作仍然需要調(diào)用DeviceIoControl()來(lái)實(shí)現(xiàn)。
SPI_IrqTransfer irq_transfer;
irq_transfer. dwGpioPin=GPIO0; //使用GPIO0作為SPI的外部中斷源
irq_transfer. dwBufLength=1024; //中斷產(chǎn)生后需要讀取1024字節(jié)的數(shù)據(jù)
irq_transfer. dwRVSD=0;
DeviceIoControl(hSPI,
SPI_IOCTL_SSP_IRQTransfer,
& irq_transfer, //輸入?yún)?shù)
sizeof(SPI_IrqTransfer), //輸入?yún)?shù)字節(jié)數(shù)
NULL,
0,
NULL,
NULL);
設(shè)置完成即啟動(dòng)外部中斷自動(dòng)觸發(fā)SPI操作,一旦中斷產(chǎn)生,驅(qū)動(dòng)程序?qū)⒆詣?dòng)接收dwBufLength長(zhǎng)度的數(shù)據(jù),存儲(chǔ)在驅(qū)動(dòng)程序的內(nèi)部緩沖區(qū)中。數(shù)據(jù)接收完成后,將發(fā)送事件通知應(yīng)用層。應(yīng)用程序可通過(guò)DeviceIoControl()調(diào)用來(lái)等待該事件,得到事件后再調(diào)用ReadFile讀取數(shù)據(jù)。通過(guò)調(diào)用DeviceIOControl()等待SPI事件,可以給定一個(gè)時(shí)間參數(shù)作為等待超時(shí)的條件,以ms為單位。成功等到SPI執(zhí)行完成的消息時(shí),DeviceIoControl會(huì)返回TRUE,否則返回FALSE。SPI事件等待的調(diào)用方法如下:
DeviceIoControl(hSPI,
SPI_IOCTL_SSP_WaitSPIEvent,
&DelayTime, //等待超時(shí),時(shí)間為ms
Sizeof(DWORD),
NULL,
0,
NULL,
NULL)
調(diào)用上述方法啟動(dòng)了外部中斷觸發(fā)SPI讀取數(shù)據(jù)的功能后,該功能將一直存在,即每次在所設(shè)置的GPIO引腳上產(chǎn)生中斷信號(hào),都會(huì)執(zhí)行一次SPI讀取操作,直到應(yīng)用程序關(guān)閉該中斷,即設(shè)置dwGpioPin和dwBufLength等于0,再調(diào)用DeviceIoControl()進(jìn)行設(shè)置操作。
SPI操作相關(guān)的范例代碼請(qǐng)參考光盤中的EM9280_SPIDemo,或來(lái)郵件索取或咨詢。
-
API
+關(guān)注
關(guān)注
2文章
1508瀏覽量
62225 -
SPI
+關(guān)注
關(guān)注
17文章
1717瀏覽量
91859 -
嵌入式主板
+關(guān)注
關(guān)注
7文章
6086瀏覽量
35490
發(fā)布評(píng)論請(qǐng)先 登錄
相關(guān)推薦
評(píng)論