реклама на сайте
подробности

 
 
 
Reply to this topicStart new topic
> STM32 USB и FreeRTOS, Падает FreeRTOS
Lyrri
сообщение Apr 11 2012, 15:53
Сообщение #1


Участник
*

Группа: Участник
Сообщений: 27
Регистрация: 15-01-11
Из: Киев
Пользователь №: 62 244



Есть плата с STM32F207. USB используется как виртуальный com порт (точнее, эмуляция ft4232). Библиотека USB - стандартная USB-Host-Device_Lib_V2.0.0 от STM. При подключении к ПК устройство видится как 4 com порта, принимает данные и возвращает их в эхо-режиме. Все ок.
Запускаю FreeRTOS без использования USB. Для простоты создал одну задачу, которая каждую секунду сбрасывает в UART тестовую строку. Тоже все работает. Но при попытке обьединить эти два действия возникла проблема: при подключении по USB к ПК, та частьу программы, которая отвечает за обработку USB продолжает работать (поскольку там вся работа проходит в прерывании USB) а rtos падает где-то через 5 секунд после подключения.
После падения rtos, перестает сбрасываться тестовая строка в порт. Попытка остановится на брекпоинте в vApplicationIdleHook() тоже ни к чему не привела. vApplicationStackOverflowHook() переполнения не находит. Выделил по максимуму стека для idle и своей задачи - все равно не работает. Нашел описание подобной проблеммы на форуме STM. Там решение свелось к понижению приоритета прерывания USB ниже системного таймера. Попробовал у себя, хотя в прерывании USB не используются api вызовы rtos. Но и это не спасло отца русской демократии (((. И вот теперь назрело несколько вопросов:
1 Кто-нибуть использовал FreeRTOS совместно с usb device на stm32?
2 В какую сторону копать?
Go to the top of the page
 
+Quote Post
Forger
сообщение Apr 11 2012, 16:06
Сообщение #2


Профессионал
*****

Группа: Свой
Сообщений: 1 215
Регистрация: 22-02-05
Пользователь №: 2 831



Цитата(Lyrri @ Apr 11 2012, 19:53) *
2 В какую сторону копать?

Вытаскивать всю обработку USB из прерываний и загонять это в отдельную задачу/задачи.
В прерывания оставлять только формирование сообщений и (возможно) копирование принятых данных куда-нить в буферы.
Разбор всего добра вести только в задаче.
Суть - с применением RTOS в прерываниях только минимум работы и как можно быстрее, а всю остальную работу выполнять в фоне задач.
Задачи ессно просто ждуть неких событий, т.е. работают по событиям.


--------------------
Кругозор некоторых людей - круг с нулевым радиусом. Они называют его "точкой зрения".
Go to the top of the page
 
+Quote Post
EugenyAM
сообщение Apr 12 2012, 02:49
Сообщение #3


Участник
*

Группа: Свой
Сообщений: 73
Регистрация: 14-10-08
Из: Omsk
Пользователь №: 40 929



Цитата(Lyrri @ Apr 11 2012, 22:53) *
Попробовал у себя, хотя в прерывании USB не используются api вызовы rtos.

В функциях EPx_Callback используются вызовы rtos? Эти функции вызываются из дебрей обработчика прерывания USB.
Если используются, лучше, как было сказано скопировать данные в свой буфер или очередь и поставить флаг по приему данных

Go to the top of the page
 
+Quote Post
adnega
сообщение Apr 12 2012, 04:00
Сообщение #4


Гуру
******

Группа: Свой
Сообщений: 2 724
Регистрация: 14-05-07
Из: Ярославль, Россия
Пользователь №: 27 702



А с приоритетами прерываний раобрались? Все грабли, которые я когда-либо собирал во FreeRTOS были из-за
Код
  NVIC_PriorityGroupConfig(NVIC_PriorityGroup_4);
и
  NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority    = (uint8_t)(configKERNEL_INTERRUPT_PRIORITY >> 4);
.
Точнее, из-за их отсутствия.
Go to the top of the page
 
+Quote Post
Lyrri
сообщение Apr 12 2012, 07:05
Сообщение #5


Участник
*

Группа: Участник
Сообщений: 27
Регистрация: 15-01-11
Из: Киев
Пользователь №: 62 244



Цитата(EugenyAM @ Apr 12 2012, 05:49) *
В функциях EPx_Callback используются вызовы rtos?

В функциях EPx_Callback вызовы rtos не используются. Данные принятые от ПК отправляются обратно.

Цитата(adnega @ Apr 12 2012, 07:00)
А с приоритетами прерываний раобрались?

При инициализации USB использую следующий код:
Код
NVIC_PriorityGroupConfig(NVIC_PriorityGroup_4);
NVIC_InitStructure.NVIC_IRQChannel = OTG_HS_IRQn;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 10;
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
NVIC_Init(&NVIC_InitStructure);

Поскольку вызовы rtos в прерывании не используются, приоритет прерывания выше чем
Код
#define configMAX_SYSCALL_INTERRUPT_PRIORITY     191 /* equivalent to 0xb0, or priority 11. */

Мои рассуждения по поводу прерываний верны? Или я где-то ошибся?
Go to the top of the page
 
+Quote Post
Lyrri
сообщение Apr 12 2012, 12:04
Сообщение #6


Участник
*

Группа: Участник
Сообщений: 27
Регистрация: 15-01-11
Из: Киев
Пользователь №: 62 244



Нашел в чем проблема, может для кого-то будет актуально.

Как только производится запись в ненулевую IN endpoint (функция DCD_EP_Tx()), вызывается функция USB_OTG_EPStartXfer() и устанавливает
DREGS->DIEPEMPMSK в 1 для заданого endpoint. После этого, когда TX FIFO окажется пустым, получим прерывание "empty FIFO", флаг которого нигде в библиотеке не сбрасывается. В итоге получаем прерывание, обработчик которого выполняется постоянно.

Как исправить:
- в файле usb_dcd_init.c находим функцию DCD_WriteEmptyTxFifo()
- в этой функции находим следующие строки
Код
ep->xfer_buff  += len;
ep->xfer_count += len;

- и после этих строк добавляем следующий код
Код
if( ep->xfer_count >= ep->xfer_len){
        uint32_t fifoemptymsk = 1 << ep->num;
        USB_OTG_MODIFY_REG32(&pdev->regs.DREGS->DIEPEMPMSK, fifoemptymsk, 0);
        break;
}
Go to the top of the page
 
+Quote Post

Reply to this topicStart new topic
1 чел. читают эту тему (гостей: 1, скрытых пользователей: 0)
Пользователей: 0

 


RSS Текстовая версия Сейчас: 7th July 2025 - 06:37
Рейтинг@Mail.ru


Страница сгенерированна за 0.01415 секунд с 7
ELECTRONIX ©2004-2016