Создание собственной системы стабилизации
тут даже смысла не вижу изобретать
Сам не хотел изобретать, сначала взял готовый кусок кода, но фиг там … Похоже надо с компилятором повнимательней разобраться (с указанием типов), где то переполнение вылазит., так что “скопипастить” не получилось…
Мда… Все ж компиллер как то теряет знаковый тип при сдвиге (делении) больших чисел, отсюда и косяк у меня,…
Не было печали…
Keil вроде с оффсайта качал, правда он с ограничениями… (не в этом ли дело?)…
Мда… Все ж компиллер как то теряет знаковый тип при сдвиге (делении) больших чисел, отсюда и косяк у меня,…
Не было печали…
Keil вроде с оффсайта качал, правда он с ограничениями… (не в этом ли дело?)…
кейл с ограничениями тут не причем, компилятор “считает” с права налево и выполняет действия согласно приоритетам операций. Может он что то не то сначала вычисляет.
эти формулы не правильно считаются?
dT=(int32_t)(D2 -C5*256);
MS_temper=2000 + dT*C6/8388608;
покажите кусок кода и какие типы переменных используются.
покажите кусок кода
Вот переменные:
int32_t dT,MS_temper;
uint32_t D2;
uint16_t C5,C6;
Вот расчет:
dT=D2 - (uint32_t)C5<<8; //( dT здесь получается 1700, проверял)
MS_temper=((int64_t)dT*28312)>>23;
MS_temper выходит -131, хотя должно быть 5…
Если пишу прям так: MS_temper=(1700*28312)>>23;
или присваиваю ранее dT=1700; - то как раз 5 и выходит…
Такое ощущение что c типом int64_t что то не так…
при умножении вылазит за 32 бита, попробуйте заменить на double, если заработает посмотрите как определен тип int64.
Возможно int64 определен как long, а компилятор его считает 32 битным, вспомнил (long long) 64 бита целочисленное.
Возможно int64 определен как long
Да, спасибо Вам за наводку, похоже где то здесь косяк… (первый признак, проверенные примеры из инета не работают…) буду копать.
Если “быстро” поправить то можно написать:
MS_temper=((long long)dT*28312)>>23;
Камрады, а зачем вы геммороитесь с такими операцыями типа <<8 и >>23? кто мешает нормально делить и перемножать числа как и указано в даташите? Сами себе проблемы ищете.
Или наносекунды экономите?
это касается и возведения в степень
заранее двойку в степень возведите и усе
Вот расчет:
dT=D2 - (uint32_t)C5<<8; //( dT здесь получается 1700, проверял)
вот здесь еще косяк, не хватает скобочек:
dT=D2 - ((uint32_t)C5<<8);
Камрады, а зачем вы геммороитесь с такими операцыями типа <<8 и >>23? кто мешает нормально делить и перемножать числа как и указано в даташите? Сами себе проблемы ищете.
Или наносекунды экономите?
это касается и возведения в степень
заранее двойку в степень возведите и усе
можно и без << >>, просто это пример из интернета, вот его и разбирали почему не работает
с такими операцыями типа <<8 и >>23
В моем случае деление на число=2^23 дает тот же результат…
Если “быстро” поправить то можно написать:
MS_temper=((long long)dT*28312)>>23;
К сожалению, Андрей, не прошло и так…
Тут я понял однозначно: компиллер отказывается (?) понимать беззнаковые переменные при операциях деления или сдвига вправо, … он их считает знаковыми, независимо от непосредственного указания типа…
Хотя на сам тип реагирует - “warning”, если ему подсунуть uint16_t на проверку на “<0”…
не знаю что может быть с кейлом, вроде не должно быть такого, сам пишу в иаре, попробовал в иаре все работает:
unsigned short int C5 = 33464;
unsigned short int C6 = 28312;
unsigned int D2 = 16777215;
signed int dT = D2 - ((unsigned int)C5<<8);
signed int TEMP = 2000 + (((long long)dT*C6)>>23);
под ARM пишете?
MS_temper=((int64_t)dT*28312)>>23; MS_temper выходит -131, хотя должно быть 5…
MS_temper ранее был объявлен как int32_t …почему он вдруг должен стать int64_t ?
И наверное MS_temper = 2000+dT*C6/2^23; Не?
не знаю что может быть с кейлом
Тут, скорее всего, ключ какой то нужен для компиллера, или еще какая “магия”…
Был удивлен, но : пишу
uint32_t K=57438; при компиляции он мне:
<main.c(78): warning: #68-D: integer conversion resulted in a change of sign>
т.е. он мне “великодушно” решил преобразовать беззнаковую в константу в знаковую (увидел старший бит)…
Не скажу что я программист, STM32 недавно только освоил, но ранее как то не сталкивался с этим (заморочки)…
почему он вдруг должен стать int64_t
произведение двух переменных в скобках больше 32-х разрядов, а они мне все нужны 😃…
напишите вот так, посчитает ли кейл:
unsigned short int C5 = 33464;
unsigned short int C6 = 28312;
unsigned int D2 = 16777216;
signed int dT = D2 - ((unsigned int)C5<<8);
signed int TEMP = 2000 + (((long long)dT*C6)>>23);
дабы простыню не выкладывать, вот рабочий драйвер github.com/SergDoc/…/drv_ms5611.c
дабы простыню не выкладывать
Не работает…
Вопрос снят, всем спасибо (фиг с ним с этим Keil-ом, у меня лимит в 32-Кб все равно уже заканчивается, буду какой нибуть CooCox мучить…)
буду какой нибуть CooCox
Велком ту волшебный мир кокоса!
Не работает…
работает - на 103м и 407 точно 😃 и под кеил и под gcc…
работает - на 103м и 407 точно
Спорить не буду, но если интересно, можете посмотреть какие данные выдает датчик температуры MS5611 (в реале при нагреве и охлаждении), не знаю что за глюк, но у меня при <17,4 градуса сразу 22,… (явно в расчете переполнение на 256 ед.),
а так то да, все работает… циферки бегут… и высоту даже можно посчитать… 😃
Вопрос снят, всем спасибо (фиг с ним с этим Keil-ом, у меня лимит в 32-Кб все равно уже заканчивается, буду какой нибуть CooCox мучить…)
Эт не правильно! Кейл отличный компиллер, а вот gcc никаких стандартов не соблюдает, можно в чистом Ц файле объявить переменную внутри кода и тд и тп
По поводу вашей переменной со знаком, компиллер вас предупреждает, что беззнаковая переменная участвует в действиях со знаковыми числами и будет преобразована, у ГЦЦ тут было бы всё шоколадно, но при вычислениях могли половить нереальный глюк. Чтоб предупреждения не выдавалось надо в коде выполнить явное преобразование типа (int32_t)peremennaya.
но у меня при <17,4 градуса сразу 22,… (явно в расчете переполнение на 256 ед.),
это как раз тот глюк о котором выше написано, результат сдвига uint8_t на 8 и более разрядов теряет всю значимую часть, за типами надо следить.
буду какой нибуть CooCox мучить…
Кейл отличный компиллер
Саша, может оно и так. У них компиллер вообще свой?
Но в плане удобности IDE как бы сказать, не в ногу со временем…
Поставил вот сегодня CoIDE-1.7.5 + свежий 4.8 arm-none-eabi тулчейн. Создал новый проект под STM32F407VG, выбрал периферийные модули, закинул сорсы нужные. Готово! Единственное с чем про$%ся это с настройкой клоков, т.к. на дискавери 8мгц, а прописано в системных файлах 25мгц вроде… Кейл тут силен конечно, не знаю каким макаром, но вне зависимости от настроек проекта, чип всегда стартовал на положенных 168мгц…
А вот далее вкусности CoIDE, чего явно Кейлу не хватает на первый взляд:
- скорость навигации. Кейл мягко говоря тупит на переходах + сами переходы только через контекстное меню, морока… а CoIDE через Ctrl+click прыгает куда надо без пауз…
- code completion - сила! Вмиг на старте сканит проект и дает подсказки по всем свойствам классов, переменным и дефайнам по ctrl+space…
- сканит и подсвечивает блоки неактивных дефайн секций
- организация и менеджмент окон
Был бы он под линукс, ценыБ ему не было 😃