Помощь - Поиск - Пользователи - Календарь
Полная версия этой страницы: Как определить где зависает?
Форум разработчиков электроники ELECTRONIX.ru > Сайт и форум > В помощь начинающему > MCS51, AVR, PIC, STM8, 8bit
troy97
В общем, где-то год назад было сделано устройство на ATmega168, которое нормально работает, но вот обнаружилось, что оно периодически (до 5 раз в день) программно зависает и перезагружается по сторожевому таймеру. Как определить где именно зависает? Устройство имеет светодиод, но им отлаживать, учитывая частоту зависания, не хочется. Есть выход в локальную сеть для передачи UDP-пакетов, в режиме прерывания по watchdog всё вполне работоспособно, как узнать в каком месте зависает? Программа написана на CVAVR. Заранее спасибо.
MrYuran
Расставить контрольные точки и в каждой выдавать тестовые сообщения в UART или ещё куда-нибудь.
При старте - специальная метка старта.
Потом посмотрите полученный лог и увидите последнее сообщение перед рестартом
Проверьте ещё целостность стека - не наползает ли он на область глобальных переменных.
troy97
UART не предусмотрен, поэтому хотелось бы использовать имеющийся ethernet интерфейс, возможно ли, например по прерыванию от вотчдога, сохранить в еепром какую-то информацию о месте где была программа в момент этого прерывания, чтоб после перезагрузки передать её по сети? Например значение программного счётчика? Как в CVAVR обратиться к этому счётчику? Просто давно это уже было и сейчас приходится заново вникатьsad.gif
V_G
Цитата(troy97 @ Oct 6 2010, 05:26) *
UART не предусмотрен

Замените в сообщении MrYuran UART на LAN, далее по тексту
xelax
Можно также при зависании, в прерывании по вачдогу, сливать весь стек и регистры... Далее пишите какой-нибудт скриптик например на питоне, который получает на вход слитый стек и дизасм вашего проекта и по процессорному стеку разматывает стек вызовов функций.... 99% найдёте место зависания. Мы таким образом все jump на NULL отловили в своём коде.
king2
Все тупо, но придется много писать. smile.gif
Заводите две глобальные переменные, пусть будет trace_fn и trace_cp.

На входе в функцию:
unsigned int old_tfn = trace_fn;
unsigned int old_tcp = trace_cp;
trace_fn = <номер_функции>;

Внутри функции:
trace_cp = 1;
trace_cp = 2;
и так далее.

На выходе из функции восстанавливаете значения из переменных old.
Внутри функции watchdog отправляете на отладку обе глобальные переменные и видите, какой checkpoint в какой функции был последним.
Там поблизости и ищите...

_Pasha
Цитата(troy97 @ Oct 4 2010, 19:29) *
в режиме прерывания по watchdog всё вполне работоспособно

Перепишите его так, чтобы не сохранять в стеке регистры. РС будет сверху. Дальше-дамп
troy97
Всем спасибо отловил место зависания весьма нехитрым способом: наставил в коде несколько меток, которые устанавливались при выполнении конкретной части кода, при зависании WDT давал прерывание, в обработчике которого значения меток сливались в eeprom, далее происходит ресет по WDT и слив полученных меток в сеть при старте программы. Соответственно, последняя активированная метка перед не активированной и показала место зависания.
Для просмотра полной версии этой страницы, пожалуйста, пройдите по ссылке.
Invision Power Board © 2001-2025 Invision Power Services, Inc.