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

 
 
 
Reply to this topicStart new topic
> iar вычисление константных выражений, препроцессором и самим компилятором
AndryG
сообщение Feb 26 2015, 21:13
Сообщение #1


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

Группа: Свой
Сообщений: 139
Регистрация: 23-05-05
Из: UA
Пользователь №: 5 317



Код
#define F_CPU 8000000u
#define MB_BAUD_RATE 38400
#define MB_TIMER_PRESCALER 64

#if MB_BAUD_RATE > 19200
// не растет кокос :(
  #define MB_OCR_T25  (F_CPU / MB_TIMER_PRESCALER * (11 / MB_BAUD_RATE +  0.00075))
  #define MB_OCR_T45  (F_CPU / MB_TIMER_PRESCALER * (11 / MB_BAUD_RATE +  0.00175))
#else
// тут я вроде как разобрался как на форуме советовали: сперва умножать побольше и потом делить поменьше
  #define MB_OCR_T25  (F_CPU / MB_TIMER_PRESCALER * 11 * 25 /10 / MB_BAUD_RATE)
  #define MB_OCR_T45  (F_CPU / MB_TIMER_PRESCALER * 11 * 45 /10 / MB_BAUD_RATE)
#endif


Компилятор считает не то, что хочется. Результат выходит буд-то "11 / MB_BAUD_RATE" вообще нет
Код
код
volatile uint8_t ocr25 = (uint8_t)MB_OCR_T25;

листинг после препроцессора
volatile uint8_t ocr25 = (uint8_t)(8000000u / 64 * (11 / 38400 + 0.00075));

и асм
//   48   volatile uint8_t ocr25 = (uint8_t)MB_OCR_T25;
        LDI     R16, 93  !!! а должно быть 129
        ST      Y, R16


Ещё препроцессору не нравится вот это:
Код
  #if mb_ocr_test < MB_OCR_T25 || mb_ocr_test < MB_OCR_T45

#Error[Pe031]: expression must have integral type

Во втором варианте расчета было "* 2.5". После замены на " * 25 / 10" ругаться перестал. На сим сделал вывод, что препроцессор знает только целочисленную математику.


Компилятор ещё намекает, что не всё в порядке с вычислением:
Код
uint8_t ocr25 = MB_OCR_T25;

#Warning[Pa092]: implicit conversion to integer does not preserve the value

OCR0A  = TCNT0 + MB_OCR_T45;

#Warning[Pa093]: implicit conversion from floating point to integer

Здесь, я так понял, надо приведение типа, ибо компилятор все же считает не целочисленно. Но проблему с ошибкой вычисления это не решает :(

OCR0A  = TCNT0 + (uint8_t)MB_OCR_T45; // ругаться перестал


Подскажите, пжлст, где про это почитать подробней (сколько листал - не нахожу по теме)?
Может встречал кто утилиты/онлайн сервисы для просмотра "и что нам насчитают"? Довольно неудобно лазить по листингам.
Go to the top of the page
 
+Quote Post
Xenia
сообщение Feb 26 2015, 21:31
Сообщение #2


Гуру
******

Группа: Модератор FTP
Сообщений: 4 479
Регистрация: 20-02-08
Из: Москва
Пользователь №: 35 237



Цитата(AndryG @ Feb 27 2015, 00:13) *
#define MB_TIMER_PRESCALER 64
...
// не растет кокос sad.gif
#define MB_OCR_T25 (F_CPU / MB_TIMER_PRESCALER * (11 / MB_BAUD_RATE + 0.00075))
...
Компилятор считает не то, что хочется. Результат выходит буд-то "11 / MB_BAUD_RATE" вообще нет.


Компилятор поступает совершенно правильно, полагая, что выражения "11 / MB_BAUD_RATE" вообще нет, поскольку в целых числах
11 / 64 = 0

А если хотите, чтобы результат получился дробным, то пишите так:
11.0 / MB_BAUD_RATE

Но с дробью
F_CPU / MB_TIMER_PRESCALER
у вас та же история - дробную часть вы теряете.

Цитата(AndryG @ Feb 27 2015, 00:13) *
// тут я вроде как разобрался как на форуме советовали: сперва умножать побольше и потом делить поменьше
#define MB_OCR_T25 (F_CPU / MB_TIMER_PRESCALER * 11 * 25 /10 / MB_BAUD_RATE)

Ну, так и делайте, как вам советовали на форуме, а то, что вы написали, этому совету противоречит, т.к там уже вторая операция - деление.
Но если последовать тому совету буквально, то у вас возникнет переполнение произведения в разрядной сетке целого.
Поэтому было бы лучше определить F_CPU как long, чтобы произведение не было урезанным.
Тогда оно будет выглядеть так:
#define F_CPU 8000000L
#define MB_OCR_T25 (F_CPU * 11 * 25 / 10 / MB_BAUD_RATE / MB_TIMER_PRESCALER)

Или еще (uint8_t) туда добавить, чтобы компилятор warning не выдавал:
#define MB_OCR_T25 ((uint8_t)(F_CPU * 11 * 25 / 10 / MB_BAUD_RATE / MB_TIMER_PRESCALER))
Go to the top of the page
 
+Quote Post
AndryG
сообщение Feb 26 2015, 23:27
Сообщение #3


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

Группа: Свой
Сообщений: 139
Регистрация: 23-05-05
Из: UA
Пользователь №: 5 317



Код
11.0 / MB_BAUD_RATE

Вот это я поймался! Спасибо.

Цитата(Xenia @ Feb 26 2015, 23:31) *
Но с дробью
F_CPU / MB_TIMER_PRESCALER
у вас та же история - дробную часть вы теряете.

Именно из-за переполнения я это деление и вынес наверх. Плюс поймался на частный случай 8000000 / 64 = 125000 "тю! Так тут целое и получается" sm.gif

Заработали оба варианта расчета с разными вариациями параметров. С excel совпадает.
Код
#if MB_BAUD_RATE > 19200
  #define MB_OCR_T25  (F_CPU * (11.0 / MB_BAUD_RATE + 0.00075) / MB_TIMER_PRESCALER)
  #define MB_OCR_T45  (F_CPU * (11.0 / MB_BAUD_RATE + 0.00175) / MB_TIMER_PRESCALER)
#else
  #define MB_OCR_T25  (F_CPU * 11 * 2.5 / MB_BAUD_RATE / MB_TIMER_PRESCALER)
  #define MB_OCR_T45  (F_CPU * 11 * 4.5 / MB_BAUD_RATE / MB_TIMER_PRESCALER)
#endif

  volatile uint8_t ocr25 = (uint8_t)MB_OCR_T25;
  volatile uint8_t ocr45 = (uint8_t)MB_OCR_T45;


А вариант проверки допустимых диапазонов препроцессором не прокатит? Препроцессору дробные числа никак не скормить?
Код
#define mb_ocr_test 255

#ifdef mb_ocr_test
  #if mb_ocr_test < MB_OCR_T25 || mb_ocr_test < MB_OCR_T45
    #warning [MB_OCR_T25] è/èëè [MB_OCR_T45] âûõîäÿò çà ðàçðÿäíîñòü OCR-ðåãèñòðà. Ïðîâåðüòå íàñòðîéêè òàéìåðà.
  #endif
#endif

Error[Pe031]: expression must have integral type

Go to the top of the page
 
+Quote Post
scifi
сообщение Feb 27 2015, 09:32
Сообщение #4


Гуру
******

Группа: Свой
Сообщений: 3 020
Регистрация: 7-02-07
Пользователь №: 25 136



Цитата(AndryG @ Feb 27 2015, 02:27) *
Препроцессору дробные числа никак не скормить?

Никак.

Цитата(AndryG @ Feb 27 2015, 02:27) *
А вариант проверки допустимых диапазонов препроцессором не прокатит?

Зачем именно препроцессором? Для этих дел существует assert_static().
Go to the top of the page
 
+Quote Post
kolobok0
сообщение Feb 27 2015, 19:29
Сообщение #5


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

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



Цитата(scifi @ Feb 27 2015, 12:32) *
Никак.


обычно для простоты душевной уходят от дробных вещей путём умножения на некую удобную константу, и в дальнейшем при вычислениях -
делением на неё. удобные константы = те что дробны на величину байта. Но если "по взрослому" с делением, то не принципиально по большому счёту.


в препроцессоре обычно можно сделать элементарные проверки на уровне иф-дефов и выводе прям не приличных сообщений на руском
в случае не правильного сочетания звёзд и кода sm.gif

Сообщение отредактировал kolobok0 - Feb 27 2015, 19:31
Go to the top of the page
 
+Quote Post

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

 


RSS Текстовая версия Сейчас: 22nd July 2025 - 07:54
Рейтинг@Mail.ru


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