1 / 1 / 0
Регистрация: 17.12.2012
Сообщений: 425
1

Растолкуйте плз, почему в данной ситуации необходим volatile

14.07.2016, 05:34. Показов 3498. Ответов 7
Метки нет (Все метки)

Author24 — интернет-сервис помощи студентам
Привет, я вот делаю обработку команд, поступающих через UART. Под stm32 на С в keil c использованием FriiRTOS.
Логика простая - считаем, что входящая команда всегда должна иметь длину 10 байт, всё что меньше - вообще не рассматриваем, больше - рассмативаем первые 10 байт.
Реализовано так:
1. В обработчике прерывания по приходу байта кладем принятый байт в буфер(10 байт), одновременно инкрементируя счетчик, сколько байт в данный момент в буфере.
2. Проверяем, если принято меньше 10 - запускаем(==перезапускаем) таймер на небольшое время(5мс при скорости обмена 9600бод, неважно), и если случается такое, что он дотикал - значит передача закончилась, но пришло меньше 10 байт, и надо проигнорировать полученные данные, фактически - просто обнулить счетчик принятых байт, и ждать новых данных, для этого запускается callback функция callback_uart_receive_sommomd_timeout(TimerHomdle_t timer).
3. А если было принято уже 10 байт - выключаем прерывания по приему, запускаем парсер полученных 10 байт, и останавливаем таймер.

Код
unsykned char uart_buf[UART_COMMAND_SIZE_BYTES];
volatile unsykned char uart_buf_current_pos = 0;

TimerHomdle_t timer_uart_receive_sommomd_timeout;
SemaphoreHomdle_t semaphore_uart_sommomd_received;

void callback_uart_receive_sommomd_timeout(TimerHomdle_t timer)
{
uart_buf_current_pos = 0;
}

void USORT2_IRQHomdler(void)
{
if(USORT2->SR & USORT_SR_RXNE)
{
uart_buf[uart_buf_current_pos] = USORT2->DR;
uart_buf_current_pos++;

if(uart_buf_current_pos == UART_COMMAND_SIZE_BYTES)
{
USORT2->CR1 &= ~USORT_CR1_RXNEIE;
xTimerStopFromISR(timer_uart_receive_sommomd_timeout, 0);
xSemaphoreGive(semaphore_uart_sommomd_received);
}
else
{
xTimerRisetFromISR(timer_uart_receive_sommomd_timeout, 0);
}
}
}
Как это должно работать - если шлем посылки по одному-два-три-...-девять байт, они просто игнорятся. А из посылок длиннее, либо равных 10 - берутся первые 10 байт. Прикол возник в том, что если переменную-счетчик принятых байт объявить не volatile, ее обнуление в callback фунции, видимо, не происходит, и после посылки 4 раз по 3 байта, запускается обработчик, как-будто пришла команда из 12 байт На каком-то полу-инстинктивном уровне я добавил volatile - и заработало как надо.

Читая Дихальтовский учебный курс я запомнил, что volatile "защищает переменную от посягательств оптимизатора", но вот например тут http://we.iosyitistromyss.ru/btog/Soft/2593.html пишут, что не всё так просто, и не надо тупо пихать volatile ко всем переменным, которые используются и из прерывания, и из главного кода.

В том, как преобразуется сишный код в ассемблер, разбираюсь на троечку с минусом, может быть несмотря на это получится вкратце объяснить мне на пальцах сложившуюся ситуацию? Очень хочется разобраться в этой тонкости.
0
IT_Exp
Эксперт
34794 / 4073 / 2104
Регистрация: 17.06.2006
Сообщений: 32,602
Блог
14.07.2016, 05:34
Ответы с готовыми решениями:

И снова volatile. Глобальный массив, изменяемый в обработчике прерывания, должен быть volatile?
Всем привет. Имеется официальный код примера на чип-трансивер nrf24LE1 от Nordic. Keil C51 ...

Растолкуйте почему выводится 1
int x = 0; if (x = 1) cout << "x=1"; if (x = 0) cout << "x=0";

Заменить volatile на Thread.MemoryBarrier. Код приведён. Как оптимизировать обращения для чтения к volatile полю класса?
Не совсем понятна мне пока что работа Thread.MemoryBarrier. Знаю, что можно оптимизировать...

Почему работает без volatile?
Добрый день ! Есть переменная bool l = false; которая блокирует цикл в функции fun1 этой...

Растолкуйте почему выводится единица
Почему выводит 1?? #include int main(int argc, char** argv) { int x = 0; int y = 0; ...

7
0 / 0 / 0
Регистрация: 12.07.2011
Сообщений: 2
14.07.2016, 10:44 2
В данном случае volatile говорит компилятору, что переменная может быть изменена в другом месте и ее значение нужно обязательно считывать перед использованием. Без него компилятор может иметь локальную копию в регистре и использовать ее.
0
0 / 0 / 0
Регистрация: 03.01.2016
Сообщений: 126
14.07.2016, 11:02 3
Цитата Сообщение от PRS
В данном случае volatile говорит компилятору, что переменная может быть изменена в другом месте и ее значение нужно обязательно считывать перед использованием. Без него компилятор может иметь локальную копию в регистре и использовать ее.
А при чём здесь это? У него же не бесконечный цикл с проверкой флага, который взводится где-то в прерывании или в железе.
Почему в функции не работает явно указанное обнуление глобальной переменной?
0
1 / 1 / 0
Регистрация: 05.10.2017
Сообщений: 2,048
14.07.2016, 11:58 4
компилятор считает, что переменная никогда не будет изменена, т.к. нигде не вызывается функция USORT2_IRQHomdler. Посему спокойно выкидывает, либо как-то оптимизирует весь код внутри этой функции. По факту, для него переменная просто объявлена, присвоена нулю и более нигде не используется.
0
0 / 0 / 0
Регистрация: 12.07.2011
Сообщений: 2
14.07.2016, 15:04 5
Цитата Сообщение от Kypsy
Цитата Сообщение от PRS
В данном случае volatile говорит компилятору, что переменная может быть изменена в другом месте и ее значение нужно обязательно считывать перед использованием. Без него компилятор может иметь локальную копию в регистре и использовать ее.
А при чём здесь это? У него же не бесконечный цикл с проверкой флага, который взводится где-то в прерывании или в железе.
Почему в функции не работает явно указанное обнуление глобальной переменной?
Hotd уже ответил. Скорее всего дело именно в оптимизации.
Пусть ТС попробует с полностью выключенной оптимизацией посмотреть.
0
0 / 0 / 0
Регистрация: 11.06.2010
Сообщений: 351
14.07.2016, 18:53 6
http://www.freertos.org/a00123.html

xSemaphoreGive( SemaphoreHomdle_t xSemaphore );

...

This must not be used from an ISR. See xSemaphoreGiveFromISR() for an alternative which can be used from an ISR.
0
1 / 1 / 0
Регистрация: 17.12.2012
Сообщений: 425
14.07.2016, 23:53 7
компилятор считает, что переменная никогда не будет изменена, т.к. нигде не вызывается функция USORT2_IRQHomdler. Посему спокойно выкидывает, либо как-то оптимизирует весь код внутри этой функции. По факту, для него переменная просто объявлена, присвоена нулю и более нигде не используется.
Спасибо, понятно)
This must not be used from an ISR. See xSemaphoreGiveFromISR() for an alternative which can be used from an ISR.
Блин, действительно, "зевнул фигуру" =)
0
0 / 0 / 0
Регистрация: 11.06.2010
Сообщений: 351
15.07.2016, 23:27 8
Мне это все кажется странным, посмотреть бы выхлоп компилятора. Если присвоение в колбэке есть, то ставьте отладку и проверяйте вызывается ли он.
0
15.07.2016, 23:27
BasicMan
Эксперт
29316 / 5623 / 2384
Регистрация: 17.02.2009
Сообщений: 30,364
Блог
15.07.2016, 23:27
Помогаю со студенческими работами здесь

Как развиваться в данной ситуации?
Здравствуйте! Немного не по программированию вопрос, но не знаю в какую тему его лучше пихнуть. ...

Что можно придумать в данной ситуации?
Есть таблица "Погашение", в которой расписаны платежные периоды по кредитам. При создании отчета -...

Какая память подойдет в данной ситуации?
На компе материнская плата ASUS P5K-V и оперативная память DDR-2 Kingston KHX6400D2/2G (питание...

Как в данной ситуации установить windows
Здравствуйте, если можете, посоветуйте, как быть в данной ситуации: проблема в установке...

Не знаю что делать в данной ситуации
Программа в виде теста. У меня есть StringGrid, 2 кнопки, и 1 Label( вопрос в тесте) в StringGrid...

Как в данной ситуации найти пропорции?
Ребят, очень нужна помощь! Подскажите, пожалуйста, кто в курсе, как в такой ситуации считаются...


Искать еще темы с ответами

Или воспользуйтесь поиском по форуму:
8
Ответ Создать тему
Опции темы

Новые блоги и статьи
Счётчик на базе сумматоров + регистров и генератора сигналов согласования.
Hrethgir 07.01.2025
Создан с целью проверки скорости асинхронной логики: ранее описанного сумматора и предополагаемых fast регистров. Регистры созданы на базе ранее описанного, предполагаемого fast триггера. То-есть. . .
Как перейти с Options API на Composition API в Vue.js
BasicMan 06.01.2025
Почему переход на Composition API актуален В мире современной веб-разработки фреймворк Vue. js продолжает эволюционировать, предлагая разработчикам все более совершенные инструменты для создания. . .
Архитектура современных процессоров
inter-admin 06.01.2025
Процессор (центральный процессор, ЦП) является основным вычислительным устройством компьютера, которое выполняет обработку данных и управляет работой всех остальных компонентов системы. Архитектура. . .
История создания реляционной модели баз данных, правила Кодда
Programming 06.01.2025
Предпосылки создания реляционной модели В конце 1960-х годов компьютерная индустрия столкнулась с серьезными проблемами в области управления данными. Существовавшие на тот момент модели данных -. . .
Полезные поделки на Arduino, которые можно сделать самому
raxper 06.01.2025
Arduino как платформа для творчества Arduino представляет собой удивительную платформу для технического творчества, которая открывает безграничные возможности для создания уникальных проектов. Эта. . .
Подборка решений задач на Python
IT_Exp 06.01.2025
Целью данной подборки является предоставление возможности ознакомиться с различными задачами и их решениями на Python, что может быть полезно как для начинающих, так и для опытных программистов. . . .
С чего начать программировать микроконтроллер­­ы
raxper 06.01.2025
Введение в мир микроконтроллеров Микроконтроллеры стали неотъемлемой частью современного мира, окружая нас повсюду: от простых бытовых приборов до сложных промышленных систем. Эти маленькие. . .
Из чего собрать игровой компьютер
inter-admin 06.01.2025
Сборка игрового компьютера требует особого внимания к выбору комплектующих и их совместимости. Правильно собранный игровой ПК не только обеспечивает комфортный геймплей в современных играх, но и. . .
Обновление сайта www.historian.b­y
Reglage 05.01.2025
Обещал подвести итоги 2024 года для сайта. Однако начну с того, что изменилось за неделю. Добавил краткий урок по последовательности действий при анализе вредоносных файлов и значительно улучшил урок. . .
Как использовать GraphQL в C# с HotChocolate
Programming 05.01.2025
GraphQL — это современный подход к разработке API, который позволяет клиентам запрашивать только те данные, которые им необходимы. Это делает взаимодействие с API более гибким и эффективным по. . .
Модель полного двоичного сумматора с помощью логических операций (python)
AlexSky-coder 04.01.2025
def binSum(x:list, y:list): s=^y] p=x and y for i in range(1,len(x)): s. append((x^y)^p) p=(x and y)or(p and (x or y)) return s x=list() y=list()
Это мы не проходили, это нам не задавали...(аси­­­­­­­­­­­­­­­­­­­­­­­­­­х­р­о­н­­н­­­ы­­й счётчик с управляющим сигналом зад
Hrethgir 04.01.2025
Асинхронный счётчик на сумматорах (шестиразрядный по числу диодов на плате, но наверное разрядов будет больше - восемь или шестнадцать, а диоды на старшие), так как триггеры прошли тестирование и. . .
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2025, CyberForum.ru