PlayBack
Implement the function of outputting audio by encoding a 1KHz sine wave signal through the CODEC.
Use DMA to transfer I2S data from memory to the Transmit FIFO of I2S, and then encode it into a PDM signal through the CODEC.
By connecting an external PDM player, it can play a fixed frequency sine wave voice.
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
EVB external recording and playback module, connect P4_3 and CLK, P4_2 and DATA.
Building and Downloading
This sample can be found in the SDK folder:
Project file: samples\peripheral\codec\playback\proj\rtl87x2g\mdk
Project file: samples\peripheral\codec\playback\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
You can hear the sound of a 1KHz sine wave in the PDM player.
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\codec\playback\proj
Source Code Directory:
sdk\samples\peripheral\codec\playback\src
Source files are currently categorized into several groups as below.
└── Project: playback
└── 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_gdma.c
├── rtl_nvic.c
├── rtl_codec.c
└── rtl_i2s.c
└── APP includes the ble_peripheral user application implementation
├── main_ns.c
└── io_codec.c
Initialization
The initialization process includes the initialization of the CODEC, UART, I2S, and DMA modules as follows.
board_codec_init
contains the PAD and PINMUX settings for PDM.
Configure PAD: set pins, PINMUX mode, PowerOn, no internal Pull-Up.
Configure PINMUX: Assign pins to PDM_CLK, PDM_DATA functions respectively.
board_i2s_init
contains the PAD and PINMUX settings for I2S.
Configure PAD: set pins, PINMUX mode, PowerOn, no internal Pull-Up.
Configure PINMUX: Assign pins to BCLK_SPORT0, LRC_SPORT0, DACDAT_SPORT0 functions respectively.
driver_i2s_init
contains the initialization of I2S peripherals.
Enable PCC clock.
Set the I2S bit clock to 16kHz.
Set master device mode, dual channel output, data format to I2S mode.
Set data width to 24 bits and channel width to 32 bits.
driver_gdma_init
contains the initialization of GDMA peripherals.
Enable PCC clock.
Use GDMA channel 0.
GDMA transfer direction is memory to peripheral transfer.
Source address is
codec_drv_cos_table
and destination address is(&(I2S0->I2S_TX_FIFO_WR_ADDR)
.Enable GDMA channel 0 total transfer completion interrupt
GDMA_INT_Transfer
.
driver_codec_init
contains the initialization of CODEC peripherals.
Enable PCC clock.
Initialize the configuration of the CODEC’s analog circuit.
Set the playback mode to un-muted.
Set the playback gain.
Set DAC dither to DISABLE.
Set data format to I2S format.
Set channel length, data width, and channel order.
RCC_PeriphClockCmd(APBPeriph_CODEC, APBPeriph_CODEC_CLOCK, ENABLE);
CODEC_AnalogCircuitInit();
...
CODEC_InitStruct.CODEC_DaMute = CODEC_UNMUTE;
CODEC_InitStruct.CODEC_DaGain = 0xAF;
CODEC_InitStruct.CODEC_DaC_Dither = DAC_DA_DITHER_DISABLE;
CODEC_InitStruct.CODEC_I2SFormat = CODEC_I2S_DataFormat_I2S;
CODEC_InitStruct.CODEC_I2SChannelLen = I2S_CHANNELLEN_32;
CODEC_InitStruct.CODEC_I2STxDataWidth = CODEC_I2S_Tx_DataWidth_24Bits;
CODEC_InitStruct.CODEC_I2SChSequence = CODEC_I2S_CH_L_R;
Functional Implementation
Execute
preload_cos_table
to prepare the sine wave data to be sent.Enable I2S transmit mode and receive mode; enable GDMA channel 0.
I2S_Cmd(I2S0, I2S_MODE_TX | I2S_MODE_RX, ENABLE);
GDMA_Cmd(GDMA_Channel_DMIC_NUM, ENABLE);
Use GDMA to carry the sine wave data to the I2S0 transmit FIFO; when the data transfer is complete, trigger the
GDMA_INT_Transfer
interrupt and enter the interrupt handler functionCODEC_GDMA_Channel_Handler
.Reset the source and destination addresses.
Reset the GDMA transfer data size.
Clear the GDMA channel 0
GDMA_INT_Transfer
interrupt pending bit and enable GDMA channel 0.
GDMA_SetSourceAddress(CODEC_GDMA_Channel, (uint32_t)codec_drv_cos_table);
GDMA_SetDestinationAddress(CODEC_GDMA_Channel, (uint32_t)(&(I2S0->I2S_TX_FIFO_WR_ADDR)));
GDMA_SetBufferSize(CODEC_GDMA_Channel, CODEC_DRV_COS_TABLE_SAMPLE);
GDMA_ClearINTPendingBit(CODEC_GDMA_Channel_NUM, GDMA_INT_Transfer);
GDMA_Cmd(CODEC_GDMA_Channel_NUM, ENABLE);