|
|
  |
__LDREX __STREX в STM32F407 |
|
|
|
Jun 8 2017, 05:01
|
Профессионал
    
Группа: Свой
Сообщений: 1 241
Регистрация: 15-11-05
Из: Челябинск
Пользователь №: 10 882

|
AHTOXA, спасибо, ознакомился. Но, опять противоречивые результаты: AVI-crak говорит, что флаг сбрасывает любая запись по адресу, хоть DMA. ArtDenis в своих тестах показывает, что даже явная запись не сбрасывает. AlexandrY утверждает, что у него явная запись сбрасывает флаг, а неявная (PUSH) - не меняет состояние. Походу, собака порылась в Цитата(ARM ® v7-M Architecture Reference Manual) A3.4.5 Load-Exclusive and Store-Exclusive usage restrictions The Load-Exclusive and Store-Exclusive instructions are designed to work together, as a pair, for example a LDREX/STREX pair or a LDREXB/STREXB pair. As mentioned in Context switch support, ARM recommends that the Store-Exclusive instruction always follows within a few instructions of its associated Load-Exclusive instructions. In order to support different implementations of these functions, software must follow the notes and restrictions given here. ... В частности, по поводу простой записи: Цитата • An explicit store to memory can cause the clearing of exclusive monitors associated with other processors, therefore, performing a store between the LDREX and the STREX can result in a livelock situation. As a result, code must avoid placing an explicit store between an LDREX and an STREX in a single code sequence. • LDREX and STREX operations must be performed only on memory with the Normal memory attribute. То есть: - LDREX/STREX применять только в паре, они для этого сделаны. Все остальное - от лукавого. Может сработать, а может и нет, зависит от конкретной реализации ядра - LDREX/STREX применять только на Normal памяти: Цитата(ARM ® v7-M Architecture Reference Manual) A3.5.1 Memory types For each memory region, the most significant memory attribute specifies the memory type. There are three mutually exclusive memory types: • Normal. • Device. • Strongly-ordered. - вроде как сброс локального монитора текущего ядра может происходить при ЛЮБОЙ записи (хоть DMA), при ЯВНОЙ записи могут сбросится только мониторы других ядер. Но по факту может быть что угодно в зависимости от конкретной реализации. Извращения проверять обязательно! Цитата(ARM ® v7-M Architecture Reference Manual) A3.4.1 Exclusive access instructions and Non-shareable memory regions When a processor writes using any instruction other than a Store-Exclusive: • If the write is to a physical address that is covered by its local monitor it is IMPLEMENTATION DEFINED whether the write affects the state of the local monitor. - локальный монитор сбрасывается при возникновении исключения. Т.е. для одноядерной системы неважно, была ли в другом процессе запись(эксклюзивная) - все равно по входу и выходу из исключения происходит сброс. Цитата A3.4.4 Context switch support In ARMv7-M, the local monitor is changed to Open Access automatically as part of an exception entry or exit sequence.
|
|
|
|
|
Jun 8 2017, 06:21
|

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

|
Цитата(Alechek @ Jun 8 2017, 10:01)  - LDREX/STREX применять только в паре, они для этого сделаны. Все остальное - от лукавого. Может сработать, а может и нет, зависит от конкретной реализации ядра Именно. Различия в результатах тестов вызваны, скорее всего, различиями в реализации. (Ну или кто-то накосячил в тестах  ) Способ использования от этого не меняется - только в паре. Добавьте в ваш список выводов, что монитор может срабатывать на любое изменение уровня прерываний (вход в прерывание или выход из него). ЗЫ. Хотя насчёт DMA - я лично не верю. В этом случае при непрерывно идущих быстрых транзакциях DMA можно наглухо зависнуть в LDREX/STREX.
--------------------
Если бы я знал, что такое электричество...
|
|
|
|
|
Jun 8 2017, 11:54
|
Профессионал
    
Группа: Свой
Сообщений: 1 241
Регистрация: 15-11-05
Из: Челябинск
Пользователь №: 10 882

|
Цитата(jcxz @ Jun 8 2017, 11:17)  И правильно! Через DMA и пр. - ЗАЧЕМ??? Импульсивный Вы наш, по делу бы что сказали! Цитата(AHTOXA @ Jun 8 2017, 11:21)  Добавьте в ваш список выводов, что монитор может срабатывать на любое изменение уровня прерываний Добавил, куда можно было. Весь этот раздел A3.4 - сплошные выводы. Читать и читать. Цитата(AHTOXA @ Jun 8 2017, 11:21)  ЗЫ. Хотя насчёт DMA - я лично не верю. В этом случае при непрерывно идущих быстрых транзакциях DMA можно наглухо зависнуть в LDREX/STREX. Мне тоже идея с DMA не очень нравится. Вроде как не обязательно Store от ядра должно исходить.... В одном месте так написано, в другом - по другому... Вобщем, все на откуп реализации:
Опыты, только опыты... Ни в одной докуметации на MCU не видел таких тонкостей.
|
|
|
|
|
Jun 8 2017, 13:39
|

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

|
Цитата(Alechek @ Jun 8 2017, 16:54)  Вобщем, все на откуп реализации: Хорошие картинки, теперь понятны отличия в реализации. По идее, получается, могут существовать реализации, где DMA сбрасывает монитор. Надо будет учесть. Цитата(Alechek @ Jun 8 2017, 16:54)  Опыты, только опыты... Ни в одной докуметации на MCU не видел таких тонкостей. Думаю, это плата за универсальность ядра. АРМу пришлось при проектировании закладывать разные варианты реализации.
--------------------
Если бы я знал, что такое электричество...
|
|
|
|
|
Jun 8 2017, 13:44
|
Частый гость
 
Группа: Участник
Сообщений: 180
Регистрация: 5-04-09
Пользователь №: 47 205

|
QUOTE (AHTOXA @ Jun 8 2017, 09:21)  Именно. Различия в результатах тестов вызваны, скорее всего, различиями в реализации. (Ну или кто-то накосячил в тестах  ) Способ использования от этого не меняется - только в паре. Добавьте в ваш список выводов, что монитор может срабатывать на любое изменение уровня прерываний (вход в прерывание или выход из него). ЗЫ. Хотя насчёт DMA - я лично не верю. В этом случае при непрерывно идущих быстрых транзакциях DMA можно наглухо зависнуть в LDREX/STREX. Есть архитектура, есть реализация. На Cortex-M, которые по определению одноядерные, монитор как таковой отсутствует. Есть просто один-единственный триггер. LDREX его взводит, STREX проверяет что установлен, выполняет запись и сбрасывает. Любое исключение сбрасывает бит. Принудительно можно сбросить через CLREX. На на навороченных многоядерных камнях с MMU, многоуровневыми кэшами, множеством bus-master устройств реализация может быть совсем другой.
|
|
|
|
|
Jun 8 2017, 14:52
|

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

|
Цитата(LightElf @ Jun 8 2017, 18:44)  Есть архитектура, есть реализация. На Cortex-M, которые по определению одноядерные, монитор как таковой отсутствует. Есть просто один-единственный триггер. LDREX его взводит, STREX проверяет что установлен, выполняет запись и сбрасывает. Любое исключение сбрасывает бит. Принудительно можно сбросить через CLREX. Не вижу в вашем высказывании противоречия с моим. (Или вы не возражали, а дополняли?) И на Cortex-M есть нюансы. Где-то обычная запись сбрасывает монитор, а где-то -- нет. Где-то DMA сбрасывает монитор, а где-то -- нет.
--------------------
Если бы я знал, что такое электричество...
|
|
|
|
|
Jun 8 2017, 20:32
|
Гуру
     
Группа: Свой
Сообщений: 5 228
Регистрация: 3-07-08
Из: Омск
Пользователь №: 38 713

|
Цитата(AVI-crak @ Jun 8 2017, 15:21)  Неявная запись (PUSH) - является отложенным событием (как и любая запись кучи регистров), код выполняется параллельно, сбросить физику с подтверждением не получится, но можно немного подождать Сколько? Сколько вешать в граммах? Цитата(AVI-crak @ Jun 8 2017, 15:21)  в пустом цикле или сбросить конвейер. Это что такое? И как оно поможет? Цитата(AVI-crak @ Jun 8 2017, 15:21)  И наверное главное, давайте указывать явно - какой чип в тесте на данный момент. DMB/DSB/и прочие B - спасут отца русской демократии  Цитата(LightElf @ Jun 8 2017, 15:44)  На Cortex-M, которые по определению одноядерные Вы это расскажите NXP с их линейкой LPC43xx. Про свои определения. Или Ваш мир Cortex-M ограничен STM32? Курица - не птица, Cortex-M - не STM! Рифма одна-ко. Надо в подпись внести ©
|
|
|
|
|
Jun 9 2017, 07:50
|

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

|
Коли SVC синхронное и его не получиться использовать нормально в прерываниях (для lockfree решений), то может быть для этой цели подойдет pendSV?
А, чтобы разделить переключатель контекста rtos и юзерские вызовы этого самого pendSV, скажем, в R0 кладем адрес функции, а в остальные регистры ее параметры, а уже внутри pendSV все это разбираем. Приоритет pendSV самый низкий (ниже лишь systick). Тогда другой вызов pendSV прервать его не сможет, а лишь поставить в очередь. Т.е. обеспечивается атомарность действий как в прерываниях, так и в основном коде. Вижу одно ограничение - передавать в такие функции указатели нежелательно, а можно лишь готовые данные (прямо в регистры R1...), т.е. ограничен объем передаваемых данных. В противном случае нет никакой гарантии, что к моменту вызова соотв. функции (внутри pendSV) данные по указателям будут актуальны.
Или тут есть какие-то подводные камни? зы. Предположительно вангую грабли со вложенными вызовами (вытесняющие прерывания), где оба обработчика вызывают pendSV...
--------------------
Кругозор некоторых людей - круг с нулевым радиусом. Они называют его "точкой зрения".
|
|
|
|
|
Jun 9 2017, 10:58
|

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

|
Цитата(jcxz @ Jun 9 2017, 13:33)  PendSV не "вызывают" ... Да, это-то понятно, это все - придирки к словам, а речь о другом: как сделать так, чтобы pendSV могли вызывать взводить разные обработчики, особенно вложенные друг в друга, но так, чтобы в стеке сформировалась очередь данных, которые позволят этот pendSV обработать несколько раз подряд и не потерять ни одного запроса и не разрушить стек? Понятно, что pendSV не для этого был задуман, но тем не менее )) Размышляя вслух дальше - выполняемый в данный момент pendSV может прервать другое более приоритетное прерывание и опять накидать в стек новых данных перед взводом pendSV, Или использовать внутри pendSV вызов соотв. SVC, который имеет приоритет выше pendSV... Заумно как-то выходит. Но возможно ли осуществить что-то подобное? Цель - избежать критических секций с запретом прерываний и "капризного" LDREX/STREX.
--------------------
Кругозор некоторых людей - круг с нулевым радиусом. Они называют его "точкой зрения".
|
|
|
|
|
Jun 9 2017, 13:02
|
Знающий
   
Группа: Участник
Сообщений: 758
Регистрация: 27-08-08
Пользователь №: 39 839

|
Цитата(Forger @ Jun 9 2017, 13:58)  как сделать так, чтобы pendSV могли вызывать взводить разные обработчики, особенно вложенные друг в друга, но так, чтобы в стеке сформировалась очередь данных, которые позволят этот pendSV обработать несколько раз подряд и не потерять ни одного запроса и не разрушить стек? Очередь данных в стеке не формируется - формируйте ее отдельно, а потом обрабатывайте в PendSV. Но для данной задачи это неэффективно и не имеет смысла - есть LDREX/STREX, если его не хватает, то есть запрет части или всех прерываний.
|
|
|
|
|
Jun 9 2017, 13:20
|

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

|
Цитата(Шаманъ @ Jun 9 2017, 16:02)  Очередь данных в стеке не формируется - формируйте ее отдельно, а потом обрабатывайте в PendSV. А все равно грабли - процесс формирования очереди должен быть защищен критической секцией, масло масляное ((( Цитата Но для данной задачи это неэффективно и не имеет смысла - есть LDREX/STREX, если его не хватает, то есть запрет части или всех прерываний. Согласен, но хочется ведь гипотетического универсального решения, вот и спрашиваю ))) Короче, задача довольно тривиальная: программная очередь, набиваемая в обработчике (скажем, очередь байтов), там же сигналим счетным семафором, переходим к соотв. задаче, которая это семафор ждет и выгребает из этой же очереди байты. Нужно защищать указатели (голова/хвост) очереди от прерывания. Сейчас сделано просто и логично - в конкретном экземпляре очереди запрещаются прерывания лишь от того, кто использует эту самую очередь - от USART.
--------------------
Кругозор некоторых людей - круг с нулевым радиусом. Они называют его "точкой зрения".
|
|
|
|
|
Jun 9 2017, 14:14
|
Знающий
   
Группа: Участник
Сообщений: 758
Регистрация: 27-08-08
Пользователь №: 39 839

|
Цитата(Forger @ Jun 9 2017, 16:20)  хочется ведь гипотетического универсального решения Если пишите под GCC, то можно посмотреть на atomic built-in функции https://gcc.gnu.org/onlinedocs/gcc-4.4.3/gc...c-Builtins.html . Цитата Короче, задача довольно тривиальная: программная очередь, набиваемая в обработчике (скажем, очередь байтов), там же сигналим счетным семафором, переходим к соотв. задаче, которая это семафор ждет и выгребает из этой же очереди байты. Нужно защищать указатели (голова/хвост) очереди от прерывания. Интересно зачем? Если это что-то типа кольцевого буфера приделанного к UART, то один указатель изменяется только в задаче, а второй только в прерывании. Пересечения между ними не происходит. Семафор это головная боль ОС. Вот и все. У меня сделано похожим образом.
|
|
|
|
|
  |
2 чел. читают эту тему (гостей: 2, скрытых пользователей: 0)
Пользователей: 0
|
|
|