DBI-C

This sample demonstrates the application of the DBI-C interface of the LCDC peripheral. The DBI-C interface is usually known as QSPI.

Requirements

The sample supports the following development kits:

Development Kit

Hardware Platforms

Board Name

RTL87x2G HDK

RTL87x2G EVB

For more requirements, please refer to Quick Start.

Wiring

A QSPI screen or logical analyzer is required.

  1. Connect P3_6 to the Reset pin.

  2. Connect P4_2 to D0 pin.

  3. Connect P4_1 to D1 pin.

  4. Connect P3_2 to D2 pin.

  5. Connect P3_3 to D3 pin.

  6. Connect P4_3 to CS pin.

  7. Connect P4_0 to CLK pin.

  8. Connect P0_5 to TE pin if available.

Or connect all the pins to a logic analyzer instead.

Building and Downloading

This sample can be found in the SDK folder:

Project file: samples\peripheral\lcdc\lcdc_qspi\proj\rtl87x2g\mdk

Project file: samples\peripheral\lcdc\lcdc_qspi\proj\rtl87x2g\gcc

To build and run the sample, follow the steps listed below:

  1. Open Sample Project File.

  2. To build the target, follow the steps listed on the Generating App Image in Quick Start.

  3. After a successful compilation, the app bin app_MP_xxx.bin will be generated in the directory mdk\bin.

  4. To download app bin into EVB board, follow the steps listed on the MP Tool Download in Quick Start.

  5. Press the reset button on the EVB board and it will start running.

Experimental Verification

  1. The screen displays green or the correct waveform is observed in the logical analyzer.

Code Overview

This chapter will be introduced according to the following several parts:

  1. Source code directory.

  2. Peripheral initialization will be introduced in chapter Initialization.

  3. Functional implementation after initialization will be introduced in chapter Functional implementation.

Source Code Directory

  1. Project directory: sdk\samples\peripheral\lcdc\lcdc_qspi\proj

  2. Source code directory: sdk\samples\peripheral\lcdc\lcdc_qspi\src

Source files are currently categorized into several groups as below.

└── Project: qspi
    └── lcdc
        ├── config                       includes project construct configuration
        ├── display                      includes the peripheral drivers related with display
        └── 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
        ├── peripheral                   includes all peripheral drivers and module code used by the application
        └── APP                          source code of LCDC qspi sample
            ├── main_ns.c
            └── icna3311_280x456_qspi.c  driver of LCD screen

Initialization

The initialization process is included in rtk_lcd_hal_init().

  1. Enable clock of LCDC, and select suitable clock source.

/* config LCD dev info */
RCC_PeriphClockCmd(APBPeriph_DISP, APBPeriph_DISP_CLOCK, ENABLE);

//from XTAL SOURCE = 40M
PERIBLKCTRL_PERI_CLK->u_324.BITS_324.disp_ck_en = 1;
PERIBLKCTRL_PERI_CLK->u_324.BITS_324.disp_func_en = 1;
PERIBLKCTRL_PERI_CLK->u_324.BITS_324.r_disp_mux_clk_cg_en = 1;

//From PLL1, SOURCE = 100M
PERIBLKCTRL_PERI_CLK->u_324.BITS_324.r_disp_div_en = 1;
PERIBLKCTRL_PERI_CLK->u_324.BITS_324.r_disp_clk_src_sel0 = 0; //pll1_peri(0) or pll2(1, pll2 = 160M)
PERIBLKCTRL_PERI_CLK->u_324.BITS_324.r_disp_clk_src_sel1 = 1; //pll(1) or xtal(0)
PERIBLKCTRL_PERI_CLK->u_324.BITS_324.r_disp_div_sel = 1; //div
  1. Configure pins to be used.

Pad_Config(LCD_QSPI_RST, PAD_PINMUX_MODE, PAD_IS_PWRON, PAD_PULL_UP, PAD_OUT_DISABLE, PAD_OUT_HIGH);
Pad_Config(LCD_QSPI_D0, PAD_PINMUX_MODE, PAD_IS_PWRON, PAD_PULL_UP, PAD_OUT_DISABLE, PAD_OUT_HIGH);
Pad_Config(LCD_QSPI_D1, PAD_PINMUX_MODE, PAD_IS_PWRON, PAD_PULL_UP, PAD_OUT_DISABLE, PAD_OUT_HIGH);
Pad_Config(LCD_QSPI_D2, PAD_PINMUX_MODE, PAD_IS_PWRON, PAD_PULL_UP, PAD_OUT_DISABLE, PAD_OUT_HIGH);
Pad_Config(LCD_QSPI_D3, PAD_PINMUX_MODE, PAD_IS_PWRON, PAD_PULL_UP, PAD_OUT_DISABLE, PAD_OUT_HIGH);
Pad_Config(LCD_QSPI_CS, PAD_PINMUX_MODE, PAD_IS_PWRON, PAD_PULL_UP, PAD_OUT_DISABLE, PAD_OUT_HIGH);
Pad_Config(LCD_QSPI_CLK, PAD_PINMUX_MODE, PAD_IS_PWRON, PAD_PULL_UP, PAD_OUT_DISABLE, PAD_OUT_HIGH);
Pad_Config(LCD_QSPI_TE, PAD_PINMUX_MODE, PAD_IS_PWRON, PAD_PULL_UP, PAD_OUT_DISABLE, PAD_OUT_HIGH);

Pad_Dedicated_Config(LCD_QSPI_RST, ENABLE);
Pad_Dedicated_Config(LCD_QSPI_D0, ENABLE);
Pad_Dedicated_Config(LCD_QSPI_D1, ENABLE);
Pad_Dedicated_Config(LCD_QSPI_D2, ENABLE);
Pad_Dedicated_Config(LCD_QSPI_D3, ENABLE);
Pad_Dedicated_Config(LCD_QSPI_CS, ENABLE);
Pad_Dedicated_Config(LCD_QSPI_CLK, ENABLE);
Pad_Dedicated_Config(LCD_QSPI_TE, ENABLE);
  1. Select color format and DBI-C interface.

LCDC_InitTypeDef lcdc_init = {0};
lcdc_init.LCDC_Interface = LCDC_IF_DBIC;
lcdc_init.LCDC_GroupSel = 1; //QFN88 2 - QFN68 1
lcdc_init.LCDC_PixelInputFormat = LCDC_INPUT_RGB565;
lcdc_init.LCDC_PixelOutputFormat = LCDC_OUTPUT_RGB565;
lcdc_init.LCDC_PixelBitSwap = LCDC_SWAP_BYPASS;
lcdc_init.LCDC_TeEn = ENABLE;
lcdc_init.LCDC_TePolarity = LCDC_TE_EDGE_FALLING;
lcdc_init.LCDC_TeInputMux = LCDC_TE_LCD_INPUT;
lcdc_init.LCDC_DmaThreshold = 112;
LCDC_Init(&lcdc_init);
  1. Initialize DBI-C interface, select clock divider, polarity and phase.

LCDC_DBICCfgTypeDef dbic_init = {0};
dbic_init.DBIC_SPEED_SEL         = 1;
dbic_init.DBIC_TxThr             = 0; //0 or 4
dbic_init.DBIC_RxThr             = 0;
dbic_init.SCPOL                  = DBIC_SCPOL_LOW;
dbic_init.SCPH                   = DBIC_SCPH_1Edge;
DBIC_Init(&dbic_init);
  1. Control Reset pin to finish reset procedure.

LCDC_LCD_SET_RST(false);
platform_delay_ms(100);
LCDC_LCD_SET_RST(true);
platform_delay_ms(120);
LCDC_LCD_SET_RST(false);
platform_delay_ms(50);
  1. If the screen needs extra information for initialization, send command and data through the DBI-C interface to the screen.

Tip

About how to send command and data, refer to Troubleshooting

Functional Implementation

  1. Function rtk_lcd_hal_set_window() is used to set the window of screen that is to be updated. The following example demonstrates a window starting from point (0,0) with a width of 280 and a height of 456:

rtk_lcd_hal_set_window(0, 0, 280, 456);
  1. Send data with a certain length by function rtk_lcd_hal_start_transfer().

  2. In rtk_lcd_hal_start_transfer(), configure DMA with the desired parameters.

LCDC_DMA_InitTypeDef LCDC_DMA_InitStruct = {0};
LCDC_DMA_StructInit(&LCDC_DMA_InitStruct);
LCDC_DMA_InitStruct.LCDC_DMA_ChannelNum          = LCDC_DMA_CHANNEL_NUM;
LCDC_DMA_InitStruct.LCDC_DMA_DIR                 = 4;
LCDC_DMA_InitStruct.LCDC_DMA_SourceInc           = LCDC_DMA_SourceInc_Inc;
LCDC_DMA_InitStruct.LCDC_DMA_DestinationInc      = LCDC_DMA_DestinationInc_Fix;
LCDC_DMA_InitStruct.LCDC_DMA_SourceDataSize      = LCDC_DMA_DataSize_Word;
LCDC_DMA_InitStruct.LCDC_DMA_DestinationDataSize = LCDC_DMA_DataSize_Word;
LCDC_DMA_InitStruct.LCDC_DMA_SourceMsize         = LCDC_DMA_Msize_8;
LCDC_DMA_InitStruct.LCDC_DMA_DestinationMsize    = LCDC_DMA_Msize_8;
LCDC_DMA_InitStruct.LCDC_DMA_SourceAddr          = (uint32_t)buf;
LCDC_DMA_InitStruct.LCDC_DMA_Multi_Block_En      = 0;
LCDC_DMA_Init(LCDC_DMA_CHANNEL_INDEX, &LCDC_DMA_InitStruct);
  1. Reset DMA registers and assign with new value.

LCDC_ClearDmaFifo();
LCDC_ClearTxPixelCnt();

LCDC_SwitchMode(LCDC_AUTO_MODE);
LCDC_SwitchDirect(LCDC_TX_MODE);

LCDC_SetTxPixelLen(len);
  1. Enable DMA. If TE signal is available, start DMA transmission when the TE signal arrives.

LCDC_Cmd(ENABLE);
LCDC_DMAChannelCmd(LCDC_DMA_CHANNEL_NUM, ENABLE);
LCDC_DmaCmd(ENABLE);
#if (TE_VALID == 1)
    LCDC_TeCmd(ENABLE);
#endif
#if (TE_VALID == 0)
    LCDC_AutoWriteCmd(ENABLE);
#endif
  1. Use function rtk_lcd_hal_transfer_done() to wait for DMA to finish transmission.

Troubleshooting

Send Data Through DBI-C

  1. Take function qspi_write() as an example of sending command and data.

  2. Set the number of data lanes and the direction of data transmission.

DBIC_Cmd(DISABLE);//disable DBIC
DBIC->CTRLR0 &= ~(BIT_CMD_CH(3) | BIT_ADDR_CH(3) | BIT_DATA_CH(3));//SET CHANNEL NUM
DBIC->CTRLR0 &= ~(BIT_TMOD(3)); //tx mode
  1. Send the length of command and address.

DBIC_CmdLength(1);
DBIC_AddrLength(3);
DBIC_TX_NDF(len - 4);
DBIC_Cmd(ENABLE);
  1. Write data to DR register and wait until DBI-C interface enters idle state.

for (uint32_t i = 0; i < len; i++)
{
    DBIC->DR[0].byte = buf[i];
}
while (DBIC->SR & BIT0); // wait bus busy
DBIC_Cmd(DISABLE);//disable DBIC