使用LPC55S28的I2C從機(jī)接口,比如實(shí)現(xiàn) 24LC系列的從機(jī)EEPROM 。可以提供以下操作,如字節(jié)寫(xiě)入、頁(yè)面寫(xiě)入、當(dāng)前地址讀取、隨機(jī)讀取和順序讀取。接下來(lái),使用Byte Write和Random Read作為示例來(lái)說(shuō)明。首先明確需要判斷的狀態(tài),包括以下內(nèi)容:數(shù)據(jù)和地址-- 用于確定接收到的地址或數(shù)據(jù)。起始位-- 用于判斷是否接收到起始位。讀寫(xiě)-- 用于判斷是寫(xiě)狀態(tài)還是讀狀態(tài)。
隨機(jī)讀取
如果從設(shè)備生成kI2C_SlaveAddressMatchEvent,則表示從設(shè)備檢測(cè)到主設(shè)備發(fā)送的啟動(dòng)或重復(fù)啟動(dòng)。可以在程序中設(shè)置一個(gè)變量,記錄它是啟動(dòng)還是重復(fù)啟動(dòng),并根據(jù)傳輸情況合理設(shè)置該變量的值。假設(shè)從設(shè)備檢測(cè)到啟動(dòng),那么根據(jù)接受容量字節(jié)大小,接收到兩個(gè)字節(jié)或三個(gè)字節(jié),那么第一個(gè)字節(jié)是包含讀寫(xiě)操作的設(shè)備的地址,第二個(gè)和第三個(gè)字(如果存在)是要操作的數(shù)據(jù)的地址。接下來(lái),LPC5528將再次檢測(cè)啟動(dòng),即再次接收具有讀取操作的設(shè)備地址。LPC5528從相應(yīng)的數(shù)據(jù)地址讀取數(shù)據(jù)并將其返回給主機(jī)。
字節(jié)寫(xiě)
如果從設(shè)備生成kI2C_SlaveAddressMatchEvent,則表示從設(shè)備檢測(cè)到主設(shè)備發(fā)送的啟動(dòng)或重復(fù)啟動(dòng)。可以在程序中設(shè)置一個(gè)變量,記錄它是啟動(dòng)還是重復(fù)啟動(dòng),并根據(jù)傳輸情況合理設(shè)置該變量的值。假設(shè)從設(shè)備檢測(cè)到啟動(dòng),那么根據(jù)接收容量大小,接收到兩個(gè)字節(jié)或三個(gè)字節(jié),那么第一個(gè)字節(jié)是包含讀寫(xiě)操作的設(shè)備的地址,第二個(gè)和第三個(gè)字(如果存在)是要操作的數(shù)據(jù)的地址。緊隨其后的字節(jié)是數(shù)據(jù)。將此數(shù)據(jù)字節(jié)保存到上一步驟中獲得的地址中。
當(dāng)主機(jī)設(shè)置STOP條件停止傳輸?shù)臅r(shí)候,從機(jī)端需要獲得一個(gè)提醒。可以參考例程"i2c_interrupt_b2b_transfer_slave.c" 實(shí)現(xiàn)基于I2C從機(jī)的例子。
I2C驅(qū)動(dòng)的狀態(tài)機(jī)設(shè)置xfer->rxSize 和xfer->txSize 變量。
static void i2c_slave_callback(I2C_Type *base, volatile i2c_slave_transfer_t *xfer, void *userData)
{
switch (xfer->event)
{
/* Address match event */
case kI2C_SlaveAddressMatchEvent:
xfer->rxData = NULL;
xfer->rxSize = 0;
break;
/* Transmit request */
case kI2C_SlaveTransmitEvent:
/* Update information for transmit process */
xfer->txData = &g_slave_buff[2];
xfer->txSize = g_slave_buff[1];
break;
/* Setup the slave receive buffer */
case kI2C_SlaveReceiveEvent:
/* Update information for received process */
xfer->rxData = g_slave_buff;
xfer->rxSize = I2C_DATA_LENGTH;
break;
/* The master has sent a stop transition on the bus */
casekI2C_SlaveCompletionEvent:
g_SlaveCompletionFlag = true;
break;
default:
g_SlaveCompletionFlag = false;
break;
}
}
如果主機(jī)發(fā)送一定數(shù)據(jù)后,判斷分支casekI2C_SlaveCompletionEvent將會(huì)執(zhí)行,同時(shí)g_SlaveCompletionFlag 標(biāo)志將會(huì)設(shè)置為 true。在SDK中,使用I2C_SlaveTransferNonBlocking 函數(shù),有判斷字節(jié)傳輸完成的動(dòng)作。
/* Start accepting I2C transfers on the I2C slave peripheral */
reVal = I2C_SlaveTransferNonBlocking(EXAMPLE_I2C_SLAVE, &g_s_handle,
kI2C_SlaveAddressMatchEvent | kI2C_SlaveCompletionEvent);
if (reVal != kStatus_Success)
{
return -1;
}
/* Wait for transfer completed. */
while (!g_SlaveCompletionFlag)
{
}
g_SlaveCompletionFlag = false;
case k2IC_SlaveCompletionEvent 這個(gè)分支的進(jìn)入不是根據(jù)主機(jī)發(fā)送的stop條件進(jìn)入的,而是根據(jù)字節(jié)數(shù)進(jìn)入的,譬如:如果主機(jī)發(fā)送256字節(jié),接收字節(jié)數(shù)定義為16,當(dāng)?shù)?6個(gè)數(shù)據(jù)接收完成后,就會(huì)進(jìn)入該分支。但是 I2C 從機(jī)端事先并不知道 I2C 主機(jī)發(fā)送的字節(jié)數(shù),所以無(wú)法判定停止。有什么辦法能判斷停止呢?
“API需要編寫(xiě)代碼判斷在I2C STOP停止時(shí)達(dá)到完成狀態(tài),而不是通過(guò)rxData計(jì)數(shù)到0。”參考示例代碼只顯示raData計(jì)數(shù)減到了0。如果要在停止時(shí)達(dá)到完成狀態(tài),在 i2c 從機(jī)回調(diào)函數(shù)i2c_slave_callback中,需要添加case kI2C_SlaveDeselectedEvent。
static void i2c_slave_callback(I2C_Type*base, volatile i2c_slave_transfer_t *xfer, void *userData)
{
switch (xfer->event)
{
...
casekI2C_SlaveDeselectedEvent:
g_SlaveCompletionFlag = true;
// 用戶(hù)代碼
break;
...
}
}
在傳輸函數(shù)中,需要添加kI2C_SlaveDeselectedEvent:
/* 在 I2C 從機(jī)外設(shè)中, 啟動(dòng)接收 I2C傳輸*/
reVal = I2C_SlaveTransferNonBlocking(EXAMPLE_I2C_SLAVE,&g_s_handle,kI2C_SlaveAddressMatchEvent | kI2C_SlaveCompletionEvent|kI2C_SlaveDeselectedEvent);
當(dāng) I2C主機(jī)發(fā)送 STOP,從機(jī)的STAT標(biāo)志將會(huì)設(shè)置為1,函數(shù) I2C_SlaveTransferHandleIRQ將會(huì)觸發(fā) kI2C_SlaveDeselectedEvent。查看驅(qū)動(dòng)代碼如下:
void I2C_SlaveTransferHandleIRQ(I2C_Type *base,i2c_slave_handle_t *handle)
{
uint32_t i2cStatus = base->STAT;
if (i2cStatus &I2C_STAT_SLVDESEL_MASK)
{
I2C_SlaveInvokeEvent(base, handle,kI2C_SlaveDeselectedEvent);
I2C_SlaveClearStatusFlags(base,I2C_STAT_SLVDESEL_MASK);
}
...
}
在 LPC55 參考手冊(cè)第 637頁(yè)面有相關(guān)描述。
審核編輯:湯梓紅。
-
接口
+關(guān)注
關(guān)注
33文章
8885瀏覽量
152977 -
EEPROM
+關(guān)注
關(guān)注
9文章
1061瀏覽量
83014 -
I2C
+關(guān)注
關(guān)注
28文章
1514瀏覽量
126528 -
狀態(tài)機(jī)
+關(guān)注
關(guān)注
2文章
493瀏覽量
27974 -
從機(jī)
+關(guān)注
關(guān)注
0文章
3瀏覽量
982
原文標(biāo)題:使用 LPC55S28 I2C 從機(jī)功能,接收不定長(zhǎng)的數(shù)據(jù)(通過(guò)字節(jié)數(shù)或者STOP條件判斷)
文章出處:【微信號(hào):嵌入式 MCU,微信公眾號(hào):嵌入式 MCU】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。
發(fā)布評(píng)論請(qǐng)先 登錄
相關(guān)推薦
用STM32F030F4串口空閑中斷接收不定長(zhǎng)數(shù)據(jù),發(fā)生中斷后不知道如何計(jì)算接收到的字節(jié)數(shù)?
LPC55S28修改音頻數(shù)據(jù)格式后USB Aduio類(lèi)異常的原因?
STM32單片機(jī)的接收不定長(zhǎng)度字節(jié)數(shù)據(jù)的方法(適用于ARM單片機(jī))
STM32單片機(jī)的接收不定長(zhǎng)度字節(jié)數(shù)據(jù)的方法
STM32單片機(jī)接收不定長(zhǎng)度字節(jié)數(shù)據(jù)的方法解析相關(guān)資料推薦
STM32單片機(jī)的接收不定長(zhǎng)度字節(jié)數(shù)據(jù)的方法
STM32單片機(jī)的接收不定長(zhǎng)度字節(jié)數(shù)據(jù)的方法
STM32單片機(jī)接收不定長(zhǎng)度字節(jié)數(shù)據(jù)的方法是什么
請(qǐng)問(wèn)STM32單片機(jī)如何接收不定長(zhǎng)度字節(jié)數(shù)據(jù)?
STM32串口接收不定長(zhǎng)字節(jié)數(shù)據(jù)的方法
STM32單片機(jī)是如何利用IDLE中斷接收不定長(zhǎng)字節(jié)數(shù)據(jù)的
STM32單片機(jī)的接收不定長(zhǎng)度字節(jié)數(shù)據(jù)的方法
STM32單片機(jī)接收不定長(zhǎng)度字節(jié)數(shù)據(jù)的方法解析

stm32 串口接收不定長(zhǎng)度數(shù)據(jù)及黏包處理 + 串口DMA接收

評(píng)論