GPIO Latch
This sample uses ENH_TIM0 to implement the functionality of triggering the latch state of GPIO and recording the latch count.
P2_4 is set as a GPIO output pin with its voltage constantly flipping over time.
P2_2 is set as a GPIO input pin and also set as the count trigger pin for ENHTIM.
Connect pin P2_2 to pin P2_4. When there is a rising edge voltage change on pin P2_2, it triggers a count. After three consecutive changes, it triggers an ENHTIM interrupt, and within the interrupt function, it prints the counter value at the time of the three triggers.
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 P2_4 to P2_2.
Building and Downloading
This sample can be found in the SDK folder:
Project file: samples\peripheral\enhtimer\latch_gpio\proj\rtl87x2g\mdk
Project file: samples\peripheral\enhtimer\latch_gpio\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 the EVB starts, observe the following log within the Debug Analyzer.
Start latch_gpio test!
After connecting the P2_2 and P2_4 pins, the counter value at the time of each GPIO triggering latch will be continuously printed on the Debug Analyzer.

The count value latched by the GPIO trigger
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\enhtimer\latch_gpio\proj
Source Code Directory:
sdk\samples\peripheral\enhtimer\latch_gpio\src
Source files are currently categorized into several groups as below.
└── Project: input_interrupt
└── 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_enh_tim.c
└── APP includes the ble_peripheral user application implementation
├── main_ns.c
└── io_latch_gpio.c
Initialization
The initialization process includes board_gpio_init
, driver_gpio_init
and driver_enhance_timer_init
.
board_gpio_init
contains the PAD and PINMUX settings.
Configure PAD: Set PIN as PINMUX mode, PowerOn, internal Pull-Up.
Configure PINMUX: Multiplex the pin to GPIO function.
driver_gpio_init
contains the initialization of GPIO peripherals.
Enable PCC clock.
Set
GPIO_Pin
to P2_4.Set
GPIO_Dir
to GPIO output mode.
RCC_PeriphClockCmd(APBPeriph_GPIOA, APBPeriph_GPIOA_CLOCK, ENABLE);
...
GPIO_InitStruct.GPIO_Pin = GPIO_PIN_OUTPUT;
GPIO_InitStruct.GPIO_Dir = GPIO_DIR_OUT;
driver_enhance_timer_init
contains the initialization of ENHTIM peripherals.
Enable PCC clock.
Set
ENHTIM_ClockDiv
toENHTIM_CLOCK_DIVIDER_1
, which means one division mode.Set
ENHTIM_Mode
toENHTIM_MODE_FreeRun
, which means the timer operates in free-running mode.Set
ENHTIM_LatchCountEn[0]
to ENABLE, which enables the GPIO triggering latch function.Set
ENHTIM_LatchCountTrigger[0]
to TRIGGER_RISING_EDGE, which means the latch is triggered on the rising edge.Set
ENHTIM_LatchCountThd
to 3, which sets the trigger interrupt FIFO threshold to 3.Set
ENHTIM_LatchTriggerPad
to INPUT_PIN.Set
ENHTIM_TimerGPIOTriggerEn
to ENABLE, which enables the GPIO-triggered timing function.Configure the ENHTIM interrupt, enable the ENHTIM timer interrupt, enable the FIFO threshold interrupt, and enable the ENHTIM peripheral.
RCC_PeriphClockCmd(APBPeriph_ENHTIMER, APBPeriph_ENHTIMER_CLOCK, ENABLE);
...
ENHTIM_InitStruct.ENHTIM_ClockDiv = ENHTIM_CLOCK_DIVIDER_1;
ENHTIM_InitStruct.ENHTIM_Mode = ENHTIM_MODE_FreeRun;
ENHTIM_InitStruct.ENHTIM_LatchCountEn[0] = ENABLE;
ENHTIM_InitStruct.ENHTIM_LatchCountTrigger[0] = ENHTIM_LATCH_TRIGGER_RISING_EDGE;
ENHTIM_InitStruct.ENHTIM_LatchCountThd = 3;
ENHTIM_InitStruct.ENHTIM_LatchTriggerPad = INPUT_PIN;
/* Enable ENHTIM IRQ */
NVIC_InitTypeDef NVIC_InitStruct;
NVIC_InitStruct.NVIC_IRQChannel = ENHTIMER_IRQn;
NVIC_InitStruct.NVIC_IRQChannelPriority = 3;
NVIC_InitStruct.NVIC_IRQChannelCmd = ENABLE;
NVIC_Init(&NVIC_InitStruct);
ENHTIM_ClearINTPendingBit(ENHTIMER_NUM, ENHTIM_INT_TIM);
ENHTIM_INTConfig(ENHTIMER_NUM, ENHTIM_INT_TIM, ENABLE);
ENHTIM_ClearINTPendingBit(ENHTIMER_NUM, ENHTIM_INT_LATCH_CNT_FIFO_FULL);
ENHTIM_INTConfig(ENHTIMER_NUM, ENHTIM_INT_LATCH_CNT_FIFO_FULL, ENABLE);
ENHTIM_ClearINTPendingBit(ENHTIMER_NUM, ENHTIM_INT_LATCH_CNT_FIFO_THD);
ENHTIM_INTConfig(ENHTIMER_NUM, ENHTIM_INT_LATCH_CNT_FIFO_THD, ENABLE);
ENHTIM_Cmd(ENHTIMER_NUM, ENABLE);
Functional Implementation
When the rising edge trigger count of the GPIO input pin reaches the set threshold of the FIFO, it triggers an RNHTIM interrupt and enters the interrupt handler function Enhanced_Timer0_Handler
.
Check if the interrupt status bit is
ENHTIM_INT_LATCH_CNT_FIFO_THD
, disable the interrupt.Execute
ENHTIM_GetLatchCountFIFOLength()
to get the number of data in the FIFO.Execute
ENHTIM_ReadLatchCountFIFO()
to get the count value and print the value data.Clear the interrupt flag and enable the interrupt.
void Enhanced_Timer0_Handler()
{
...
if (ENHTIM_GetINTStatus(ENH_TIM0, ENHTIM_INT_LATCH_CNT_FIFO_THD))
{
APP_PRINT_INFO0("ENH_TIM0 ENHTIM_INT_LATCH_CNT2_FIFO_THD\r\n");
ENHTIM_INTConfig(ENHTIMER_NUM, ENHTIM_INT_LATCH_CNT_FIFO_THD, DISABLE);
uint8_t length = ENHTIM_GetLatchCountFIFOLength(ENH_TIM0);
uint32_t data[4] = {0};
ENHTIM_ReadLatchCountFIFO(ENH_TIM0, data, length);
/* Only for debugging, removed in actual application. */
APP_PRINT_INFO1("ENH_TIM0 fifo length = %d\r\n", length);
for (uint8_t i = 0; i < length; i++)
{
/* Only for debugging, removed in actual application. */
APP_PRINT_INFO2("ENH_TIM0 data[%d] = 0x%x\r\n", i, data[i]);
}
ENHTIM_ClearINTPendingBit(ENH_TIM0, ENHTIM_INT_LATCH_CNT_FIFO_THD);
ENHTIM_INTConfig(ENH_TIM0, ENHTIM_INT_LATCH_CNT_FIFO_THD, ENABLE);
}
}