Sto lavorando a un progetto che coinvolge un MCU STM32 (sulla scheda STM32303C-EVAL per l'esattezza) che deve rispondere a un interrupt esterno. Voglio che la reazione all'interrupt esterno sia il più veloce possibile. Ho modificato un esempio di libreria periferica standard dalla pagina Web ST e il programma corrente commuta semplicemente un LED su ogni fronte di salita successivo su PE6:
#include "stm32f30x.h"
#include "stm32303c_eval.h"
EXTI_InitTypeDef EXTI_InitStructure;
GPIO_InitTypeDef GPIO_InitStructure;
NVIC_InitTypeDef NVIC_InitStructure;
static void EXTI9_5_Config(void);
int main(void)
{
/* Initialize LEDs mounted on STM32303C-EVAL board */
STM_EVAL_LEDInit(LED1);
/* Configure PE6 in interrupt mode */
EXTI9_5_Config();
/* Infinite loop */
while (1)
{
}
}
// Configure PE6 and PD5 in interrupt mode
static void EXTI9_5_Config(void)
{
/* Enable clocks */
RCC_AHBPeriphClockCmd(RCC_AHBPeriph_GPIOD | RCC_AHBPeriph_GPIOE, ENABLE);
RCC_APB2PeriphClockCmd(RCC_APB2Periph_SYSCFG, ENABLE);
/* Configure input */
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN;
GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_DOWN;
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6;
GPIO_Init(GPIOD, &GPIO_InitStructure);
/* Connect EXTI6 Line to PE6 pin */
SYSCFG_EXTILineConfig(EXTI_PortSourceGPIOE, EXTI_PinSource6);
/* Configure Button EXTI line */
EXTI_InitStructure.EXTI_Line = EXTI_Line6;
EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt;
EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Rising;
EXTI_InitStructure.EXTI_LineCmd = ENABLE;
EXTI_Init(&EXTI_InitStructure);
/* Enable and set interrupt to the highest priority */
NVIC_InitStructure.NVIC_IRQChannel = EXTI9_5_IRQn;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0x00;
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0x00;
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
NVIC_Init(&NVIC_InitStructure);
}
Il gestore di interrupt si presenta così:
void EXTI9_5_IRQHandler(void)
{
if((EXTI_GetITStatus(EXTI_Line6) != RESET))
{
/* Toggle LD1 */
STM_EVAL_LEDToggle(LED1);
/* Clear the EXTI line 6 pending bit */
EXTI_ClearITPendingBit(EXTI_Line6);
}
}
In questo caso particolare, gli interrupt vengono creati da un generatore di funzioni programmabile esterno che funziona a 100 Hz. Dopo aver esaminato la risposta della MCU su un oscilloscopio, sono rimasto piuttosto sorpreso dal fatto che ci vogliono quasi 1,32 noi affinché la MCU inizi a elaborare l'interruzione:
Con l'MCU a 72 MHz (in precedenza ho verificato l'uscita SYSCLK sul pin MCO) ciò equivale a quasi 89 cicli di clock. La risposta MCU all'interrupt non dovrebbe essere molto più veloce?
PS Il codice è stato compilato con IAR Embedded Workbench e ottimizzato per la massima velocità.
if{}
istruzione è necessaria perché la routine di interrupt non sa quale sia l'origine dell'interrupt.