Arbitrary Length Transmit and Receive - DMA
This sample demonstrates arbitrary length data communication with a PC terminal using UART and DMA .
The SoC receives data from the PC terminal using DMA. Simultaneously, the UART_FLAG_RX_IDLE
case gets the arbitrary length of data, handles the data, and sets received_flag to indicate that the reception is complete.
The SoC sends the data back to the PC terminal using DMA when the received_flag is set.
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 P3_0 to the RX pin of the FT232 and P3_1 to the TX pin of the FT232.
Building and Downloading
This sample can be found in the SDK folder:
Project file: samples\peripheral\uart\unfixedlen_gdma\proj\rtl87x2g\mdk
Project file: samples\peripheral\uart\unfixedlen_gdma\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
Preparation Phase
Start a PC terminal like PuTTY or UartAssist and connect to the used COM port with the following UART settings:
Baud rate: 115200
8 data bits
1 stop bit
No parity
No hardware flow control
Testing Phase
After resetting the EVB, observe the log information as shown in the Debug Analyzer.
Start uart tx rx unfixedlen by gdma test!
Type strings on the PC terminal and observe that the same string appears on the PC terminal. Simultaneously, observe that received data and interrupt information appear in the Debug Analyzer. Assuming the input data length on the PC terminal is number, the interrupt log for different cases is as follows:
GDMA0_Channel4_Handler /* time = (number / block_size) > 0 triggers time interrupts and print logs. */ UART_FLAG_RX_IDLE /* time = (number % block_size) > 0 and meeting the UART_FLAG_RX_IDLE interrupt condition triggers time interrupts and print logs. */ value is 0x.. value is 0x.. value is 0x.. ... UART_TX_GDMA_Handler /* Once received_flag is set, the SoC uses GDMA to send number length data back to the PC terminal. Upon completion, it triggers the interrupt and prints logs. */
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\uart\unfixedlen_gdma\proj
Source code directory:
sdk\samples\peripheral\uart\unfixedlen_gdma\src
Source files are currently categorized into several groups as below.
└── Project: output_toggle
└── 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_gdma.c
├── rtl_nvic.c
├── rtl_pinmux.c
├── rtl_rcc.c
└── rtl_uart.c
└── APP includes the ble_peripheral user application implementation
├── io_uart.c
└── main_ns.c
Initialization
The initialization process includes board_uart_init
and driver_uart_init
.
board_uart_init
contains the PAD and PINMUX settings:
Config PAD: Set the pin as PINMUX mode, PowerOn, internal Pull-Up.
Config PINMUX: Assign pins for UART3_TX, UART3_RX functions respectively.
driver_uart_init
contains the initialization of the UART peripheral.
Enable PCC Clock.
Set the baud rate to 115200, no parity, 1-bit stop bit, 8-bits data length.
Enable UART DMA transmission.
Set UART_TxWaterLevel to 1, the recommended setting is
GDMA_MSize
.Set UART_RxWaterLevel to 1, the recommended setting is
GDMA_MSize
.Enable UART TX DMA, RX DMA.
Enable UART receive idle interrupt
UART_INT_RX_IDLE
.
RCC_PeriphClockCmd(APBPeriph_UART3, APBPeriph_UART3_CLOCK, ENABLE);
...
UART_InitStruct.UART_Div = BaudRate_Table[BAUD_RATE_115200].div;
UART_InitStruct.UART_Ovsr = BaudRate_Table[BAUD_RATE_115200].ovsr;
UART_InitStruct.UART_OvsrAdj = BaudRate_Table[BAUD_RATE_115200].ovsr_adj;
...
UART_InitStruct.UART_IdleTime = UART_RX_IDLE_2BYTE; //idle interrupt wait time
UART_InitStruct.UART_TxWaterLevel = 1; /* Better to equal: GDMA_MSize */
UART_InitStruct.UART_RxWaterLevel = 1; /* Better to equal: GDMA_MSize */
UART_InitStruct.TxDmaEn = ENABLE;
UART_InitStruct.RxDmaEn = ENABLE;
UART_InitStruct.dmaEn = UART_DMA_ENABLE;
UART_Init(UART_DEMO, &UART_InitStruct);
UART_INTConfig(UART_DEMO, UART_INT_RX_IDLE | UART_INT_LINE_STS, ENABLE);
...
driver_gdma3_init
contains the initialization of GDMA TX peripheral.
Use GDMA channel 3.
The GDMA transfer direction is Memory to Peripheral.
The source address is set to
GDMA_Tx_Buf
and the destination address is set to(&(UART_DEMO->UART_RBR_THR))
.Enable the GDMA channel 3 total transfer completion interrupt
GDMA_INT_Transfer
.
GDMA_InitStruct.GDMA_ChannelNum = UART_TX_GDMA_CHANNEL_NUM;
GDMA_InitStruct.GDMA_DIR = GDMA_DIR_MemoryToPeripheral;
...
GDMA_InitStruct.GDMA_SourceAddr = (uint32_t)GDMA_Tx_Buf;
GDMA_InitStruct.GDMA_DestinationAddr = (uint32_t)(&(UART_DEMO->UART_RBR_THR));
GDMA_InitStruct.GDMA_DestHandshake = GDMA_Handshake_UART3_TX;
GDMA_Init(UART_TX_GDMA_CHANNEL, &GDMA_InitStruct);
/* Enable transfer finish interrupt */
GDMA_INTConfig(UART_TX_GDMA_CHANNEL_NUM, GDMA_INT_Transfer, ENABLE);
...
driver_gdma4_init
contains the initialization of GDMA RX peripheral.
Use GDMA channel 4.
The GDMA transfer direction is peripheral to memory.
The source address is set to
(&(UART_DEMO->UART_RBR_THR))
and the destination address is set toGDMA_Rx_Buf
.Enable the GDMA channel 4 total transfer completion interrupt
GDMA_INT_Transfer
.Enable GDMA channel 4 transfer.
GDMA_InitStruct.GDMA_ChannelNum = UART_RX_GDMA_CHANNEL_NUM;
GDMA_InitStruct.GDMA_DIR = GDMA_DIR_PeripheralToMemory;
GDMA_InitStruct.GDMA_BufferSize = GDMA_BLOCK_SIZE;
...
GDMA_InitStruct.GDMA_SourceAddr = (uint32_t)(&(UART_DEMO->UART_RBR_THR));
GDMA_InitStruct.GDMA_DestinationAddr = (uint32_t)GDMA_Rx_Buf;
GDMA_InitStruct.GDMA_SourceHandshake = GDMA_Handshake_UART3_RX;
GDMA_Init(UART_RX_GDMA_CHANNEL, &GDMA_InitStruct);
/* Enable transfer finish interrupt */
GDMA_INTConfig(UART_RX_GDMA_CHANNEL_NUM, GDMA_INT_Transfer, ENABLE);
...
GDMA_Cmd(UART_RX_GDMA_CHANNEL_NUM, ENABLE);
Functional Implementation
Typing characters on the PC terminal initiates the GDMA transfer from the UART RX FIFO to memory.
Assuming the input data length on the PC terminal is number, the interrupt for different cases is as follows:
if number / block_size > 0, it triggers n_time
GDMA_INT_Transfer
interrupts.if number % block_size > 0 and meets the
UART_FLAG_RX_IDLE
interrupt condition (If no data is put into RX FIFO within RX idle timeout period after reading all data in the RX FIFO), it triggers n_timeUART_INT_RX_IDLE
interrupts.
When the
UART_FLAG_RX_IDLE
interrupt is triggered and the UART interrupt handling functionUART3_Handler
is entered.Suspend UART_RX_GDMA_CHANNEL channel transmission.
Disable the
UART_INT_RX_IDLE
interrupt.Get the GDMA transfer length (i.e., the length of data received by GDMA) and save the received data into GDMA_Tx_Buf.
Set the receive_flag to indicate that the reception is complete.
Reinitialize the GDMA after resuming and disabling the GDMA transfer for the next data reception.
Enable the
UART_INT_RX_IDLE
interrupt again after clearing the UART FIFO.
if (UART_GetFlagState(UART3, UART_FLAG_RX_IDLE) == SET)
{
DBG_DIRECT("UART_FLAG_RX_IDLE");
/* Suspend GDMA_Channel2 */
GDMA_SuspendCmd(UART_RX_GDMA_CHANNEL, ENABLE);
UART_INTConfig(UART3, UART_INT_RX_IDLE, DISABLE);
data_len = GDMA_GetTransferLen(UART_RX_GDMA_CHANNEL);
if (data_len)
{
receive_offset += data_len;
memcpy(GDMA_Tx_Buf + GDMA_BLOCK_SIZE * count, GDMA_Rx_Buf, data_len);
...
GDMA_Cmd(UART_RX_GDMA_CHANNEL_NUM, DISABLE);
GDMA_SuspendCmd(UART_RX_GDMA_CHANNEL, DISABLE);
driver_gdma4_init();
/* GDMA TX flag */
receiveflg = true;
}
......
UART_ClearRxFIFO(UART3);
UART_INTConfig(UART3, UART_INT_RX_IDLE, ENABLE);
}
...
When the
GDMA_INT_Transfer
interrupt is triggered and the GDMA interrupt handling functionGDMA0_Channel4_Handler
is entered.Disable the GDMA RX channel and clear the GDMA channel 3 interrupt flag.
Save the received data into GDMA_Tx_Buf.
Reconfigure the destination address and enable the GDMA RX channel transmission.
GDMA_Cmd(UART_RX_GDMA_CHANNEL_NUM, DISABLE);
GDMA_ClearAllTypeINT(UART_RX_GDMA_CHANNEL_NUM);
...
memcpy(GDMA_Tx_Buf + GDMA_BLOCK_SIZE * (count - 1), GDMA_Rx_Buf, GDMA_BLOCK_SIZE);
...
GDMA_SetDestinationAddress(UART_RX_GDMA_CHANNEL, (uint32_t)GDMA_Rx_Buf);
GDMA_Cmd(UART_RX_GDMA_CHANNEL_NUM, ENABLE);
Once the received_flag is set, execute
GDMA_SetBufferSize()
to update the GDMA TX Block Size, and executeGDMA_Cmd()
to initiate the GDMA transfer from memory to the UART TX FIFO.Set the GDMA TX block size to the number which is typed on the PC terminal.
Enable the GDMA TX channel again.
while (1)
{
if (receiveflg)
{
GDMA_SetBufferSize(UART_TX_GDMA_CHANNEL, receive_offset);
GDMA_Cmd(UART_TX_GDMA_CHANNEL_NUM, ENABLE);
...
}
}
When transfer data by GDMA is completed, the GDMA interrupt handling function
GDMA0_Channel3_Handler
is entered.Print the completion information of GDMA transport data.
Disable GDMA TX channel transfer and clear the interrupt flag of
GDMA_INT_Transfer
of GDMA channel 3.
void GDMA0_Channel3_Handler(void)
{
DBG_DIRECT("UART_TX_GDMA_Handler");
GDMA_Cmd(UART_TX_GDMA_CHANNEL_NUM, DISABLE);
GDMA_ClearAllTypeINT(UART_TX_GDMA_CHANNEL_NUM);
}