stm32fx, RTX и внешние прерывания - PullRequest
0 голосов
/ 29 января 2020

Я работаю над проектом на плате STM32F1x с RTX RTOS, и мне нужно некоторое внешнее прерывание для работы с некоторым пользовательским вводом. По этой причине я позволил внешнему прерыванию работать со входом на выводе A.0.

/* Handle PA0 interrupt: user input */
__irq void EXTI0_IRQHandler(void) {
     /* Make sure that interrupt flag is set */
     if (EXTI_GetITStatus(EXTI_Line0) != RESET) {
         isr_evt_set(0x0003, task1);
         EXTI_ClearITPendingBit(EXTI_Line0);
     }
}

__task void task1(void) {
     OS_RESULT result;
     unsigned int i;
     while(1) {
         result = os_evt_wait_or(0x0003, 0xFFFF);
         if (result == OS_R_TMO)
         {
            // event not received
            counter--;  // variable used for debugging purposes
         }
         else
         {
         // event received
            counter++;
     }
}

Однако всякий раз, когда я даю ввод на вывод GPIOA.0, я получаю странное поведение, которое можно увидеть в анализаторе logi c (все другие периодические c задачи перестают выполняться и go плоская линия). РЕДАКТИРОВАТЬ: Чтобы дать дополнительную информацию: Этот код работает с кодом переключения светодиодов, помещенным в обработчик.

Вот моя конфигурация GPIO / EXTI

void Configure_PA0(void) {
     /* PA0 */
     GPIO_InitTypeDef GPIO_InitStruct;
     EXTI_InitTypeDef EXTI_InitStruct;
     NVIC_InitTypeDef NVIC_InitStruct;

     /* Enable clock for GPIOA */
     /* Enable clock for GPIO */
     RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);

     /* Set pin 0 as input */
     GPIO_InitStruct.GPIO_Mode = GPIO_Mode_IN_FLOATING;
     GPIO_InitStruct.GPIO_Pin = GPIO_Pin_0;
     GPIO_Init(GPIOA, &GPIO_InitStruct);

     /* Tell system that you will use PDn for EXTI_Line n */
     GPIO_EXTILineConfig(GPIO_PortSourceGPIOA, GPIO_PinSource);

     /* PA0 is connected to EXTI_Line0 */
     EXTI_InitStruct.EXTI_Line = EXTI_Line0;
     /* Enable interrupt */
     EXTI_InitStruct.EXTI_LineCmd = ENABLE;
     /* Interrupt mode */
     EXTI_InitStruct.EXTI_Mode = EXTI_Mode_Interrupt;
     /* Triggers on rising and falling edge */
     EXTI_InitStruct.EXTI_Trigger = EXTI_Trigger_Rising_Falling;
     /* Add to EXTI */
     EXTI_Init(&EXTI_InitStruct);

     /* Add IRQ vector to NVIC */

     /* PA0 is connected to EXTI_Line0, which has EXTI0_IRQn vector */
     NVIC_InitStruct.NVIC_IRQChannel = EXTI0_IRQn;
     /* Set priority */
     NVIC_InitStruct.NVIC_IRQChannelPreemptionPriority = 0x00;
     /* Set sub priority */
     NVIC_InitStruct.NVIC_IRQChannelSubPriority = 0x00;
     /* Enable interrupt */
     NVIC_InitStruct.NVIC_IRQChannelCmd = ENABLE;
     /* Add to NVIC */
     NVIC_Init(&NVIC_InitStruct);
}

Что я делаю неправильно? Спасибо всем, кто поможет мне с этим.

1 Ответ

1 голос
/ 29 января 2020

После небольшого изучения кода я узнал, что пошло не так, и хотел бы поделиться с теми, кто может столкнуться с той же проблемой. У меня было несколько задач, созданных внутри задачи инициализации и перечисленных до этой задачи, скажем:

init_task() {
  id_tsk0 = os_tsk_create(task1, 50);
  id_tsk1 = os_tsk_create(task2, 40);
  id_tsk2 = os_tsk_create(task3, 30);
  //...and the task I want to call from interrupt..
  id_tskn = os_tsk_create(taskn, 254);
}

Я переместил задачу N с последнего места на первое и ... теперь она работает!

...