dword

Ардуинщики всех стран - объединяйтесь! В этом форуме, конечно.
Ответить
Аватара пользователя
airnbrew
Открыл глаза
Сообщения: 72
Зарегистрирован: Вт май 16, 2017 11:32:19
Откуда: Казань
Контактная информация:

dword

Сообщение airnbrew »

Tiny88.
Нужно отмасштабировать сигнал с АЦП с коэффициентом 0..1.
Коэффициент 16-битный, точности достаточно.
Ну и разобраться, как оно вообще работает.
Думаю, должно быть быстрее и короче, чем через float.

Имею примерно следующее.

uint16_t result,k;
uint32_t x;

x=analogRead(an0)*k;
x=x>>16;
result=x;

С любого разворота умножение даёт ноль.

У большинства потребителей вход максимум 16-битный, не понял, как присвоить известно какую часть uint32 в uint16, младшую или старшую, где почитать? Что присваивается при обычном присвоении?
И вообще, такой вариант должен работать тупо в лоб, или нужно искать какие-то доп. библиотеки?
Реклама
Уош
Опытный кот
Сообщения: 700
Зарегистрирован: Вс мар 23, 2025 14:56:55

Re: dword

Сообщение Уош »

да, я смотрел одно, видел другое, думал о третьем, поэтому, удалил свой ответ, как вводящий в заблуждение, прошу прощения.

Добавлено after 1 minute 7 seconds:
присваиваются, разумеется, одинаковые разряды. То есть, начиная со младших.

А если явно указать приведение? например:

Код: Выделить всё

uint16_t result, k;

result= (uint16_t)((uint32_t)analogRead(an0) * k) >> 16);
Реклама
Аватара пользователя
muravei_
Потрогал лапой паяльник
Сообщения: 339
Зарегистрирован: Чт май 20, 2021 13:33:05

Re: dword

Сообщение muravei_ »

Может

Код: Выделить всё

 result = reinterpret_cast <uint16_t> x;
так?
Аватара пользователя
BOB51
Друг Кота
Сообщения: 15571
Зарегистрирован: Вт мар 16, 2010 22:02:27
Откуда: ДОНЕЦК

Re: dword

Сообщение BOB51 »

Для преобразования диапазона/масштабирования в референсе есть функции
constrain(x,a,b)
и
map(value, fromLow, fromHigh, toLow, toHigh)
другое делаем в случае, ежли ресурса тех функций недостаточно будет.
8)
Реклама
Эиком - электронные компоненты и радиодетали
Аватара пользователя
uldemir
Друг Кота
Сообщения: 7359
Зарегистрирован: Пт авг 28, 2009 21:34:30
Откуда: 845-й км.

Re: dword

Сообщение uldemir »

да вы что. Если аналогРид даёт 16 бит и умножаете на K, который 16 бит - в результат получаете 16 бит (и, возможно, теряете данные если происходит переполнение), который потом записываете в 32 битную переменную. Но там всё-равно будет только 16 бит, которые операцией x>>16 уйдут прочь и останется 0.

Чтобы при умножении не потерять точность, либо объявите k как 32битную или приведите её к 32 битам:
x = Analogread() * (uint32_t)k;

или аналогично приведите результат аналогрида к 32 битам.
Реклама
Аватара пользователя
BOB51
Друг Кота
Сообщения: 15571
Зарегистрирован: Вт мар 16, 2010 22:02:27
Откуда: ДОНЕЦК

Re: dword

Сообщение BOB51 »

Это при масштабировании "вверх".
Там соответствующий размер результата надо с самого начала выбирать. Да и решений гораздо больше.
Только где у АВРок АЦП более 12 бит то?
:roll:
Или таки задача состоит в получении младшего байта(слова) из более "крупноформатного" значения?
Тогда или lowByte()/highByte() обратно им свертка l и h байт в word - word(h,l)
или "фокусы" с union/struct lдля более "крупнразмерных" данных...
:roll:
Реклама
Аватара пользователя
airnbrew
Открыл глаза
Сообщения: 72
Зарегистрирован: Вт май 16, 2017 11:32:19
Откуда: Казань
Контактная информация:

Re: dword

Сообщение airnbrew »

x=(long)analogRead(an0)*k;

Правильно так.
Оказывается, его нужно явно ткнуть носом, сам не умеет при умножении, когда в операндах 32 бита, к ним приводить.
Аватара пользователя
HardWareMan
Мучитель микросхем
Сообщения: 431
Зарегистрирован: Ср сен 02, 2015 07:47:20

Re: dword

Сообщение HardWareMan »

[uquote="airnbrew",url="/forum/viewtopic.php?p=4713409#p4713409"]x=(long)analogRead(an0)*k;

Правильно так.
Оказывается, его нужно явно ткнуть носом, сам не умеет при умножении, когда в операндах 32 бита, к ним приводить.[/uquote]
Вы будете смеяться, но даже gcc для ARM так не умеет. Сам наступил на эти грабли при работе с STM32, который далеко не 8 бит.
Репозиторий STM32: https://cloud.mail.ru/public/2i19/Y4w8kKEiZ
Актуальность репозитория: 22 апреля 2026 года
Если чего-то не хватает с сайта st.com - пишите, докачаю.
/!\ Обновлений для STM32PowerMon и STM32PowerMon-UCPD временно не будет.
Аватара пользователя
BOB51
Друг Кота
Сообщения: 15571
Зарегистрирован: Вт мар 16, 2010 22:02:27
Откуда: ДОНЕЦК

Re: dword

Сообщение BOB51 »

А кто мешает обьявить k как long?
8)
Уош
Опытный кот
Сообщения: 700
Зарегистрирован: Вс мар 23, 2025 14:56:55

Re: dword

Сообщение Уош »

смотря где объявить.
Аватара пользователя
uldemir
Друг Кота
Сообщения: 7359
Зарегистрирован: Пт авг 28, 2009 21:34:30
Откуда: 845-й км.

Re: dword

Сообщение uldemir »

BOB51 писал(а):А кто мешает обьявить k как long
Жадность. Но...

Если мы в теме про ардуину, можно допустить, что речь идёт о 8 битных контроллерах, так избыточная разрядность может выйти боком. Например, если эта переменная будет использоваться в прерывании, то надо будет заботиться о блокировке прерываний на время каждого её изменения в теле программы. Впрочем, на восьмибитках так же надо поступать и с 16 битными переменными.
Аватара пользователя
BOB51
Друг Кота
Сообщения: 15571
Зарегистрирован: Вт мар 16, 2010 22:02:27
Откуда: ДОНЕЦК

Re: dword

Сообщение BOB51 »

Так в прерываниях свои правила.
Да и регистровые пары довольно частое явление во многих 8-битных МК.
Как там ресурсами компилятор распоряжается то только предположить можно (Си в этом смысле не ассемблер).
При том, что предположить формат результата вполне доступное дело.
Размерность результата АЦП известна, размерность коэффициента и конечного результата также.
Только функция чтения АЦП уже имеет свой формат возвращаемых данных.
Следовательно или результат надо приводить к наибольшему возможному или вторую участвующих в вычислениях переменную (константу) заранее с максимальным форматом объявлять.
То уже зависит от конкретики(где чего удобнее).
8)
Уош
Опытный кот
Сообщения: 700
Зарегистрирован: Вс мар 23, 2025 14:56:55

Re: dword

Сообщение Уош »

uldemir, полагаю, речь лишь о тех переменных, которые и вне и внутри прерывания?
veso74
Поставщик валерьянки для Кота
Сообщения: 1907
Зарегистрирован: Сб май 05, 2012 20:24:52
Откуда: KN34PC, Болгария
Контактная информация:

Re: dword

Сообщение veso74 »

Умножите ваш коэффициент 0..1 на 10, 100, 1000 (или больше), а в полученном результате просто напр. на дисплее переместите десятичную точку (столько раз, сколько нулей).

Пример:
567 * 0.123 = 69.741 -> 69.741V
будет
567 * 123 = 69741 -> 69.741V
Аватара пользователя
uldemir
Друг Кота
Сообщения: 7359
Зарегистрирован: Пт авг 28, 2009 21:34:30
Откуда: 845-й км.

Re: dword

Сообщение uldemir »

ну да. Обычно те, что обзывают волатильными. Компилятор может позаботиться, чтобы её "не соптимизировали", но позаботиться, чтобы при присваивании этой переменной какой-то другой не оказалось, так, что часть из этой жизни, а часть из другой - не всегда.
Спойлер

Код: Выделить всё

/******************************************************************************
 * ATOMIC OPERATIONS for ATMEGA328 ONLY
 * Since the ATMega328 is an 8 bit processor it is possible that you will end
 * up trying to read a multi-byte quantity that is modified in an interrupt while
 * you are doing the read or write. The result is a corrupt value. 32 bit processors
 * are unlikely to suffer from this since quantities are read in a single operation.
 *
 * The AVR compiler provides a method for you to disable interrupts for the
 * duration of a block of code and then restore the state at the end of the block.
 *
 * It is not enough to simply turn off interrupts and then turn them back on because
 * you need to remember the state of the interrupt enable flag at the start of the
 * block.
 *
 * These macros do this for you and should be either modified for different processors
 * or bypassed if needed.
 *
 * Use like this:
 * ATOMIC {
 * // code to protect
 * }
 *
 */
Ответить

Вернуться в «Arduino»