LoopBack
This sample demonstrates using I2C for self-transmitting and self-receiving communication.
I2C0 acts as the master and reads data length and information transmitted by I2C1, while I2C1 acts as the slave and receives data length and information sent by I2C0, thereby implementing I2C communication functionality.
Requirements
The sample supports the following development kits:
Hardware Platforms |
Board Name |
---|---|
RTL87x2G HDK |
RTL87x2G EVB |
For more requirements, please refer to Quick Start.
Wiring
Connect P4_0(I2C0 SCL) to P4_2(I2C1 SCL), P4_1(I2C0 SDA) to P4_3(I2C1 SDA) on the EVB, need to connect 2.2k pull-up resistor.
The connection schematic of the pull-up resistor is shown below:

I2C pull-up resistor schematic
Configurations
The example configurable macros are as follows:
I2C_MASTER_SEND_SLAVE_RECEIVE
: Enable this macro to configure the example as a master sends and a slave receives.I2C_MASTER_RECEIVE_SLAVE_SEND
: Enable this macro to configure the example as a master receives and a slave sends.I2C_MASTER_REPEAT_READ
: Enable this macro to configure the example for the master to continuously send and receive data.
Building and Downloading
This sample can be found in the SDK folder:
Project file: samples\peripheral\i2c\trx_loopback\proj\rtl87x2g\mdk
Project file: samples\peripheral\i2c\trx_loopback\proj\rtl87x2g\gcc
To build and run the sample, follow the steps listed below:
Open sample project file.
To build the target, follow the steps listed on the Generating App Image in Quick Start.
After a successful compilation, the app bin
app_MP_xxx.bin
will be generated in the directorymdk\bin
orgcc\bin
.To download app bin into EVB board, follow the steps listed on the MP Tool Download in Quick Start.
Press reset button on EVB board and it will start running.
Experimental Verification
When enabling I2C master send slave receive I2C_MASTER_SEND_SLAVE_RECEIVE
:
When the data received at the SLAVE side reaches the set threshold level, trigger the
I2C_INT_RX_FULL
interrupt and print log.I2C1 Rx Full detected
When the MASTER finishes sending data and the SLAVE detects the stop signal, the MASTER and SLAVE trigger the
I2C_INT_STOP_DET
interrupt and print log.I2C0 Stop signal detected I2C1 Stop signal detected
Within the SLAVE interrupt function, the SLAVE prints the length and content of the data received.
Slave Recv Date Lenth = 24 I2C1_Slave_ReceiveData=0 ... I2C1_Slave_ReceiveData=23
When enabling I2C master to receive slave send I2C_MASTER_RECEIVE_SLAVE_SEND
:
The SLAVE receives the read request from the MASTER, the SLAVE triggers
I2C_INT_RD_REQ
interrupt and prints log.Enter I2C1 interrupt I2C1_INT_RD_REQ!
Print the data received at the MASTER.
Master Read data = 10 ... Master Read data = 33
When enabling I2C master repeat read I2C_MASTER_REPEAT_READ
:
The SLAVE receives the read request from the MASTER, the SLAVE triggers
I2C_INT_RD_REQ
interrupt and prints log.Enter I2C1 interrupt I2C1_INT_RD_REQ!
The SLAVE receives the data from the MASTER and prints the length and values of the received data in the SLAVE ISR.
Slave Recv Date Lenth = 4 I2C1_Slave_ReceiveData=1 I2C1_Slave_ReceiveData=2 I2C1_Slave_ReceiveData=3 I2C1_Slave_ReceiveData=4
The SLAVE finishes receiving data and prints log.
I2C1 RX_DONE detected
The SLAVE sends data back to the MASTER, and after the MASTER finishes receiving the data, it prints log.
I2C0 Stop signal detected
Print the data received by the MASTER side.
Master Repeat Read data = 10 ... Master Repeat Read data = 33
Code Overview
This chapter will be introduced according to the following several parts:
Peripheral initialization will be introduced in chapter Initialization .
Functional implementation after initialization will be introduced in chapter Functional Implementation .
Source Code Directory
Project Directory:
sdk\samples\peripheral\i2c\trx_loopback\proj
Source Code Directory:
sdk\samples\peripheral\i2c\trx_loopback\src
Source files are currently categorized into several groups as below.
└── Project: trx_loopback
└── secure_only_app
└── Device includes startup code
├── startup_rtl.c
└── system_rtl.c
├── CMSIS includes CMSIS header files
├── CMSE Library Non-secure callable lib
├── Lib includes all binary symbol files that user application is built on
└── rtl87x2g_io.lib
├── Peripheral includes all peripheral drivers and module code used by the application
├── rtl_rcc.c
├── rtl_pinmux.c
├── rtl_nvic.c
├── rtl_gdma.c
└── rtl_i2c.c
└── APP includes the ble_peripheral user application implementation
├── main_ns.c
└── io_i2c.c
Initialization
The initialization process includes board_i2c_master_init
, board_i2c_slave_init
,
driver_i2c_master_init
, driver_i2c_slave_init
and nvic_i2c_config
.
board_i2c_master_init
and board_i2c_slave_init
contain the PAD and PINMUX settings for I2C0 and I2C1:
Config PAD: Set pins as PINMUX mode, PowerOn, internal Pull-Up.
Config PINMUX: Assign pins for I2C0_CLK, I2C0_DAT, I2C1_CLK, I2C1_DAT functions respectively.
driver_i2c_master_init
contains the initialization of the I2C0 peripheral:
Enable PCC clock.
Set the I2C clock frequency to 100k.
Set the device mode to master mode.
Set the I2C address to 7-bit address mode.
Set the slave address bit to 0x50.
Enable the ACK function.
Enable I2C0.
RCC_PeriphClockCmd(APBPeriph_I2C0, APBPeriph_I2C0_CLOCK, ENABLE);
...
I2C_InitStruct.I2C_ClockSpeed = 100000;
I2C_InitStruct.I2C_DeviveMode = I2C_DeviveMode_Master;
I2C_InitStruct.I2C_AddressMode = I2C_AddressMode_7BIT;
I2C_InitStruct.I2C_SlaveAddress = 0x50;
I2C_InitStruct.I2C_RxThresholdLevel = 8;
I2C_InitStruct.I2C_Ack = ENABLE;
driver_i2c_slave_init
contains the initialization of I2C1 peripherals:
Enable PCC clock.
Set the I2C clock frequency to 100k.
Set the device mode to slave mode.
Set the I2C address to 7-bit address mode.
Set slave address bit 0x50.
Enable the ACK function.
Enable I2C1.
RCC_PeriphClockCmd(APBPeriph_I2C1, APBPeriph_I2C1_CLOCK, ENABLE);
...
I2C_InitStruct.I2C_ClockSpeed = 100000;
I2C_InitStruct.I2C_DeviveMode = I2C_DeviveMode_Slave;
I2C_InitStruct.I2C_AddressMode = I2C_AddressMode_7BIT;
I2C_InitStruct.I2C_SlaveAddress = 0x50;
I2C_InitStruct.I2C_RxThresholdLevel = 8;
I2C_InitStruct.I2C_Ack = ENABLE;
nvic_i2c_config
contains the initialization of interrupts:
Clear the interrupt pending bit.
Enable I2C0 and I2C1 interrupts according to the specific transmit or receive settings in the program.
I2C_ClearINTPendingBit(I2C0, I2C_INT_STOP_DET);
...
#if I2C_MASTER_SEND_SLAVE_RECEIVE
I2C_INTConfig(I2C0, I2C_INT_STOP_DET, ENABLE);
I2C_INTConfig(I2C1, I2C_INT_RX_FULL, ENABLE);
I2C_INTConfig(I2C1, I2C_INT_STOP_DET, ENABLE);
I2C_INTConfig(I2C1, I2C_INT_RX_DONE, ENABLE);
#endif
#if I2C_MASTER_RECEIVE_SLAVE_SEND
I2C_INTConfig(I2C1, I2C_INT_RD_REQ, ENABLE);
#endif
#if I2C_MASTER_REPEAT_READ
I2C_INTConfig(I2C0, I2C_INT_STOP_DET, ENABLE);
I2C_INTConfig(I2C1, I2C_INT_RX_FULL, ENABLE);
I2C_INTConfig(I2C1, I2C_INT_STOP_DET, ENABLE);
I2C_INTConfig(I2C1, I2C_INT_RX_DONE, ENABLE);
I2C_INTConfig(I2C1, I2C_INT_RD_REQ, ENABLE);
#endif
...
Functional Implementation
When enabling I2C master send slave receive I2C_MASTER_SEND_SLAVE_RECEIVE
:
Execute
I2C_MasterWrite()
and the master sends the data in the write_data array to the slave.The slave receives the data sent by the master, and when the slave reaches the threshold, it triggers the
I2C_INT_RX_FULL
or detects the stop signal, it triggers theI2C_INT_STOP_DET
interrupt, and enters the interrupt handler function :cpp:any:I2C1_Handler
.Determine when the receive full state is detected as SET:
Get the length of the received data in the RX FIFO and save the received data.
Clear the
I2C_INT_RX_FULL
interrupt pending bit.
When the slave detects the stop signal status as SET:
Obtain the length of the received data in the RX FIFO and save the received data.
Print the total length of the received data and the received data information.
Clear the
I2C_INT_STOP_DET
interrupt pending bit.
if (I2C_GetINTStatus(I2C1, I2C_INT_STOP_DET) == SET)
{
DBG_DIRECT("I2C1 Stop signal detected");
/*read I2C receive data*/
lenth = I2C_GetRxFIFOLen(I2C1);
for (uint32_t i = 0; i < lenth; i++)
{
I2C_Rev_Data[I2C_Rev_Index++] = I2C_ReceiveData(I2C1);
}
I2C_Rev_Data_Lenth += lenth;
I2C_ClearINTPendingBit(I2C1, I2C_INT_STOP_DET);
...
}
...
if (I2C_GetINTStatus(I2C1, I2C_INT_RX_FULL) == SET)
{
DBG_DIRECT("I2C1 Rx Full detected");
lenth = I2C_GetRxFIFOLen(I2C1);
/*read I2C data*/
for (uint32_t i = 0; i < lenth; i++)
{
I2C_Rev_Data[I2C_Rev_Index++] = I2C_ReceiveData(I2C1);
}
I2C_Rev_Data_Lenth += lenth;
I2C_ClearINTPendingBit(I2C1, I2C_INT_RX_FULL);
}
When enable I2C master receive slave send I2C_MASTER_RECEIVE_SLAVE_SEND
:
Execute
I2C_MasterRead()
, the master sends master read command to the slave.The slave detects the RD_REQ request and triggers the
I2C_INT_RD_REQ
interrupt, and enters the interrupt handler functionI2C1_Handler
.Execute
I2C_SendCmd()
command to send data to the host.Clear the
I2C_INT_RD_REQ
interrupt pending bit after sending data.
for (uint32_t i = 0; i < TransferLength; i++)
{
I2C_SendCmd(I2C1, I2C_WRITE_CMD, send_data_buffer[i], DISABLE);
}
I2C_ClearINTPendingBit(I2C1, I2C_INT_RD_REQ);
Print the data received by the master.
When enabling I2C master repeat read I2C_MASTER_REPEAT_READ
:
Execute
I2C_RepeatRead()
, the master sends the repeat read command to the slave, sends the data in tx_data, and receives the data stored in rx_data.
uint8_t tx_data[10] = {01, 02, 03, 04};
uint8_t rx_data[TransferLength] = {0};
I2C_RepeatRead(I2C0, tx_data, 4, rx_data, TransferLength);
The slave receives the data sent by the host and triggers the
I2C_INT_RX_FULL
interrupt when the slave receives a full FIFO, or triggers theI2C_INT_STOP_DET
interrupt when it detects a stop signal. TheI2C_INT_RD_REQ
interrupt is triggered when the slave detects the RD_REQ application and enters the interrupt handler functionI2C1_Handler
(same as above)