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

 
 
 
Reply to this topicStart new topic
> Стек main после запуска планировщика
glonium
сообщение Feb 26 2014, 15:10
Сообщение #1


Частый гость
**

Группа: Участник
Сообщений: 77
Регистрация: 27-05-11
Пользователь №: 65 303



Доброго времени суток всем!
Использую в своём проекте freeRTOS на stm32 компилятор Keil.
Вот недавно столкнулся с особенностью что после запуска планировщика из main портиться итек main, то есть при попытке обратиться к локальным переменным main типа класс с виртуальными функциями внутри по заданному адресу из задачи ввалимся в HardFault_Handler. При анализе обнаружил мусор по адресу данной переменной хотя адрес верный и до запуска планировщика было всё нормально. Кто нибудь сталкивался почему портиться стек main?

Сейчас выяснил что стек портиться после вызова функции:
Код
__asm void vPortStartFirstTask( void )
{
    PRESERVE8

    /* Use the NVIC offset register to locate the stack. */
    ldr r0, =0xE000ED08
    ldr r0, [r0]
    ldr r0, [r0]
    /* Set the msp back to the start of the stack. */
    msr msp, r0
    /* Globally enable interrupts. */
    cpsie i
    /* Call SVC to start the first task. */
    svc 0
    nop
}


Я просто в асме не очень кто нибудь подскажите что тут твориться?

Сообщение отредактировал glonium - Feb 26 2014, 14:59
Go to the top of the page
 
+Quote Post
_Артём_
сообщение Feb 26 2014, 15:17
Сообщение #2


Гуру
******

Группа: Свой
Сообщений: 2 128
Регистрация: 21-05-06
Пользователь №: 17 322



Цитата(glonium @ Feb 26 2014, 19:10) *
Использую в своём проекте freeRTOS на stm32 компилятор Keil.
Вот недавно столкнулся с особенностью что после запуска планировщика из main портиться итек main, то есть при попытке обратиться к локальным переменным main типа класс с виртуальными функциями внутри по заданному адресу из задачи ввалимся в HardFault_Handler.

Видимо стек портится потому, что после запуска планировщика он используется как стек прерываний. Переключение стеков реализовано в Cortex аппаратно.
Если хотите использовать переменные из стека main (а оно надо такое?), то расположите стек прерываний в другой области ОЗУ.
Go to the top of the page
 
+Quote Post
megajohn
сообщение Feb 26 2014, 15:31
Сообщение #3


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

Группа: Свой
Сообщений: 1 080
Регистрация: 16-11-04
Из: СПб
Пользователь №: 1 143



>Я просто в асме не очень кто нибудь подскажите что тут твориться?
включается прерывание SvCall и это по сути запрос на переключение контекста

увеличьте стек
либо поставьте бряк к примеру на 0x10000020 и смотрите кто пишет
для удобства можно до startup_main проиннициализировать main-стек

Цитата(megajohn @ Feb 26 2014, 19:27) *
для удобства можно до startup_main проиннициализировать main-стек


для IAR делается так, открываете startup*.s

MODULE ?cstartup

;; Forward declaration of sections.
SECTION CSTACK:DATA:NOROOT(3)

SECTION .intvec:CODE:NOROOT(2)

EXTERN _InitStacks добавляете строку
EXTERN SystemInit
EXTERN __iar_program_start

берете из аттача файл, переименовываете в fill_stacks.s и добавляете к проекту. И всё
Прикрепленные файлы
Прикрепленный файл  fill_stacks.txt ( 560 байт ) Кол-во скачиваний: 19
 


--------------------
Марс - единственная планета, полностью населенная роботами (около 7 штук).
Go to the top of the page
 
+Quote Post
glonium
сообщение Feb 26 2014, 15:31
Сообщение #4


Частый гость
**

Группа: Участник
Сообщений: 77
Регистрация: 27-05-11
Пользователь №: 65 303



Не надо впринципе просто хотелось выяснить причины. Невыясненная причина потом может вылезти!
Go to the top of the page
 
+Quote Post
megajohn
сообщение Feb 26 2014, 15:39
Сообщение #5


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

Группа: Свой
Сообщений: 1 080
Регистрация: 16-11-04
Из: СПб
Пользователь №: 1 143



>Если хотите использовать переменные из стека main (а оно надо такое?), то расположите стек прерываний в другой области ОЗУ.

по стартапу контроллер находится в Thread Mode с MSP
по запуску операционки задачи работают в Thread Mode с PSP
в прерывании режим Handler Mode с MSP

то есть до старта ОС и прерывания это одна область памяти. Зачем её разделять - не понятно, наоборот трата ресурсов


--------------------
Марс - единственная планета, полностью населенная роботами (около 7 штук).
Go to the top of the page
 
+Quote Post
_Артём_
сообщение Feb 26 2014, 16:35
Сообщение #6


Гуру
******

Группа: Свой
Сообщений: 2 128
Регистрация: 21-05-06
Пользователь №: 17 322



Цитата(megajohn @ Feb 26 2014, 19:39) *
по стартапу контроллер находится в Thread Mode с MSP

МК стартует в handler mode и потом может переключиться в Thread mode. Если не путаю конечно...Иначе что мешает задаче переключится в режим обработчика?
Go to the top of the page
 
+Quote Post
glonium
сообщение Feb 26 2014, 18:06
Сообщение #7


Частый гость
**

Группа: Участник
Сообщений: 77
Регистрация: 27-05-11
Пользователь №: 65 303



Спасибо за ответы!
Буду разбираться, впринципе static помогает!

Разобрался!
В упор не видел!
Код
    /* Use the NVIC offset register to locate the stack. */
    ldr r0, =0xE000ED08
    ldr r0, [r0]
    ldr r0, [r0]
    /* Set the msp back to the start of the stack. */
    msr msp, r0

восстанавливает верхушку стека таким образом сбрасывая её!
Всем спасибо большое за помощь!
Думаю тему можно считать исчерпанной!
Go to the top of the page
 
+Quote Post
_Артём_
сообщение Feb 26 2014, 21:43
Сообщение #8


Гуру
******

Группа: Свой
Сообщений: 2 128
Регистрация: 21-05-06
Пользователь №: 17 322



Цитата(glonium @ Feb 26 2014, 22:06) *
Буду разбираться, впринципе static помогает!

static - тоже что и глобальная переменная. Может так и объявить?
Цитата(glonium @ Feb 26 2014, 22:06) *
Код
    /* Use the NVIC offset register to locate the stack. */
    ldr r0, =0xE000ED08
    ldr r0, [r0]
    ldr r0, [r0]
    /* Set the msp back to the start of the stack. */
    msr msp, r0

И что там было? Можно ведь понятней писать...Нужно стремится к этому.
Go to the top of the page
 
+Quote Post
kolobok0
сообщение Feb 27 2014, 20:06
Сообщение #9


практикующий тех. волшебник
*****

Группа: Участник
Сообщений: 1 190
Регистрация: 9-09-05
Пользователь №: 8 417



Цитата(glonium @ Feb 26 2014, 22:06) *
..восстанавливает верхушку стека...


да, там нужно немного подправить азм. чтоб корректно сохранять-восстанавливать данную область памяти.
делал когда-то, для останова FreeRTOS. Правда по второму кругу так и не удалось застартовать ось уже, пришлось сделать
кондово(и стабильней) - через ресет.
Go to the top of the page
 
+Quote Post
AHTOXA
сообщение Feb 28 2014, 05:06
Сообщение #10


фанат дивана
******

Группа: Свой
Сообщений: 3 387
Регистрация: 9-08-07
Из: Уфа
Пользователь №: 29 684



Цитата(_Артём_ @ Feb 27 2014, 03:43) *
И что там было? Можно ведь понятней писать...Нужно стремится к этому.

Там MAIN стек переставляют в начало перед запуском оси. Чтоб стек зря не пропадал. После старта оси этот стек будет использоваться для прерываний.
Им не пришло в голову, что кто-то перед запуском оси может завести локальные переменные на стеке и потом работать с ними.


--------------------
Если бы я знал, что такое электричество...
Go to the top of the page
 
+Quote Post
glonium
сообщение Mar 1 2014, 17:21
Сообщение #11


Частый гость
**

Группа: Участник
Сообщений: 77
Регистрация: 27-05-11
Пользователь №: 65 303



Да все верно! Но использовать переменные из стека маин не критично, просто хотелось разобраться в чем дело исключить баги в программе!
Go to the top of the page
 
+Quote Post

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

 


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


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