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

 
 
> Вывод float в arm-none-eabi-gcc для stmf4
zemlemer
сообщение Oct 20 2016, 15:09
Сообщение #1


Участник
*

Группа: Участник
Сообщений: 69
Регистрация: 10-06-08
Пользователь №: 38 190



Добрый день.

Я меня возникли некоторые проблемы с выводом в терминал значений с плавающей точкой. Самый обычный sprintf прекрасно работает с целыми числами и отказывается работать с float. Например

CODE

float flttest = 123.45678;

sprintf(TestArray, "-= %lf =-", flttest);
Terminal("Test1: STR %s", TestArray);

uint16_t integer= 12345;
sprintf(TestArray, "*- %d -*", integer);
Terminal("Test2: STR %s", TestArray);


При этот вывод:
003 INFO: Test1: STR -= 403099659660755.687500 =-
004 INFO: Test2: STR *- 12345 -*

Основные ключи компилятора: arm-none-eabi-gcc -c -mthumb -mcpu=cortex-m4 -mfloat-abi=hard -mfpu=fpv4-sp-d16 -mlittle-endian -mthumb-interwork -ffunction-sections -fdata-sections -Wl,--gc-sections
Основные ключи линкера: arm-none-eabi-gcc -mthumb -mcpu=cortex-m4 -mfloat-abi=hard -mfpu=fpv4-sp-d16 -Xlinker -Map="project.map" -L/opt/gcc-arm-none-eabi-5_4/arm-none-eabi/lib/armv7e-m/fpu/ --specs=rdimon.specs -T/Core/STM32F4XX_FLASH.ld

Кто-либо сталкивался с подобным?

Go to the top of the page
 
+Quote Post
2 страниц V   1 2 >  
Start new topic
Ответов (1 - 14)
scifi
сообщение Oct 20 2016, 15:26
Сообщение #2


Гуру
******

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



В командную строку линкера нужно добавить "-u _printf_float". Первоисточник.
Go to the top of the page
 
+Quote Post
zemlemer
сообщение Oct 20 2016, 17:02
Сообщение #3


Участник
*

Группа: Участник
Сообщений: 69
Регистрация: 10-06-08
Пользователь №: 38 190



Цитата(scifi @ Oct 20 2016, 18:26) *
В командную строку линкера нужно добавить "-u _printf_float". Первоисточник.


На самом деле пробовал с -u _printf_float и с -u _sprintf_float с одинаковым результатом.

Сообщение отредактировал zemlemer - Oct 20 2016, 17:18
Go to the top of the page
 
+Quote Post
scifi
сообщение Oct 20 2016, 17:22
Сообщение #4


Гуру
******

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



Я невнимательно посмотрел. Если линкуется целочисленная версия, не будет вот этих странных чисел.
Обратите внимание на выравнивание стека по границе 8 байт. Помню, это приводило к чудесам в плавучке.
Go to the top of the page
 
+Quote Post
Lagman
сообщение Oct 20 2016, 20:10
Сообщение #5


Знающий
****

Группа: Свой
Сообщений: 875
Регистрация: 28-10-05
Пользователь №: 10 245



может поможет http://www.openstm32.org/tiki-view_forum_t...s_parentId=2108
Go to the top of the page
 
+Quote Post
dxp
сообщение Oct 21 2016, 05:29
Сообщение #6


Adept
******

Группа: Свой
Сообщений: 3 469
Регистрация: 6-12-04
Из: Novosibirsk
Пользователь №: 1 343



Ещё вариант: https://github.com/emb-lib/io

библиотека, оптимизированная для embedded применений. Плавучку печатает без нареканий (только нужно определить макро PRINTF_FLOAT, лучше всего передать в командной строке -DPRINTF_FLOAT).


--------------------
«Отыщи всему начало, и ты многое поймёшь» К. Прутков
Go to the top of the page
 
+Quote Post
zemlemer
сообщение Oct 21 2016, 19:00
Сообщение #7


Участник
*

Группа: Участник
Сообщений: 69
Регистрация: 10-06-08
Пользователь №: 38 190



Цитата(scifi @ Oct 20 2016, 20:22) *
Я невнимательно посмотрел. Если линкуется целочисленная версия, не будет вот этих странных чисел.
Обратите внимание на выравнивание стека по границе 8 байт. Помню, это приводило к чудесам в плавучке.


Вроде к линковщике секция с правильным ALIGN, или я не правильно понял.
Код
  ._user_heap_stack :
  {
    . = ALIGN(4);
    PROVIDE ( end = . );
    PROVIDE ( _end = . );
    PROVIDE ( __end__ = . );
    . = . + _Min_Heap_Size;
    . = . + _Min_Stack_Size;
    . = ALIGN(4);
  } >RAM

Вообще ситуация еще более печальна. Задавая вопрос, я просто упростил сложившуюся ситуацию. На самом деле я пишу небольшую библиотеку для отладочного вывода. Парсер разбора форматной строки - va_arg - корректно работает с целочисленными переменными. Т.е. intarg = va_arg(ap, int32_t); отрабатывает корректно. Как только речи идет о передаче плавучки - doublearg = va_arg(ap, double); - передается мусор. Если передать float через *((int32_t*)&floatarg и передать через va_arg(ap, int32_t); - передается правильно.

Сообщение отредактировал IgorKossak - Oct 22 2016, 13:27
Причина редактирования: [codebox] для длинного кода, [code] - для короткого!
Go to the top of the page
 
+Quote Post
iiv
сообщение Oct 21 2016, 19:03
Сообщение #8


вопрошающий
*****

Группа: Свой
Сообщений: 1 726
Регистрация: 24-01-11
Пользователь №: 62 436



Цитата(zemlemer @ Oct 20 2016, 21:09) *
float flttest = 123.45678;

sprintf(TestArray, "-= %lf =-", flttest);

если у вас одинарная точность, sprintf хочет %f или %e или %g, а если двойная, то да, с "l" надо писать.
Go to the top of the page
 
+Quote Post
aaarrr
сообщение Oct 21 2016, 19:18
Сообщение #9


Гуру
******

Группа: Свой
Сообщений: 10 713
Регистрация: 11-12-04
Пользователь №: 1 448



Цитата(iiv @ Oct 21 2016, 22:03) *
если у вас одинарная точность, sprintf хочет %f или %e или %g, а если двойная, то да, с "l" надо писать.

У sprintf не бывает одинарной точности, "l" игнорируется.
Go to the top of the page
 
+Quote Post
zemlemer
сообщение Oct 21 2016, 19:25
Сообщение #10


Участник
*

Группа: Участник
Сообщений: 69
Регистрация: 10-06-08
Пользователь №: 38 190



Цитата(iiv @ Oct 21 2016, 22:03) *
если у вас одинарная точность, sprintf хочет %f или %e или %g, а если двойная, то да, с "l" надо писать.

Знаю. Просто va_arg не работает с float, поэтому перед передачей я преобразовывал в double а после - обратно во float.
Go to the top of the page
 
+Quote Post
scifi
сообщение Oct 21 2016, 19:26
Сообщение #11


Гуру
******

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



Цитата(zemlemer @ Oct 21 2016, 22:00) *
Вроде к линковщике секция с правильным ALIGN, или я не правильно понял.
._user_heap_stack :
{
. = ALIGN(4);

Ещё раз: выравнивать по границе 8 байт. Проверяйте на входе в sprintf(). Внутрисхемным отладчиком остановите на входе в функцию и посмотрите на SP. Если не кратно 8 - беда.
Go to the top of the page
 
+Quote Post
Genadi Zawidowsk...
сообщение Oct 21 2016, 20:05
Сообщение #12


Профессионал
*****

Группа: Участник
Сообщений: 1 620
Регистрация: 22-06-07
Из: Санкт-Петербург, Россия
Пользователь №: 28 634



Цитата
перед передачей я преобразовывал в double а после - обратно во float.

про "после" подробнее... желательно с кодом.
Go to the top of the page
 
+Quote Post
demiurg_spb
сообщение Oct 22 2016, 21:21
Сообщение #13


неотягощённый злом
******

Группа: Свой
Сообщений: 2 746
Регистрация: 31-01-08
Из: Санкт-Петербург
Пользователь №: 34 643



Цитата(aaarrr @ Oct 21 2016, 22:18) *
У sprintf не бывает одинарной точности, "l" игнорируется.
Я каждый раз задаюсь вопросом: в чём был смысл такого решения при создании сишной библиотеки?


--------------------
“Будьте внимательны к своим мыслям - они начало поступков” (Лао-Цзы)
Go to the top of the page
 
+Quote Post
zemlemer
сообщение Oct 24 2016, 10:53
Сообщение #14


Участник
*

Группа: Участник
Сообщений: 69
Регистрация: 10-06-08
Пользователь №: 38 190



Цитата(demiurg_spb @ Oct 23 2016, 00:21) *
Я каждый раз задаюсь вопросом: в чём был смысл такого решения при создании сишной библиотеки?


Смыслов несколько. У меня статически выделен массив для передачи строки в терминал, а стандартные работают с динамическим перераспределением памяти. Использовать malloc или realloc нужно когда совсем припрет. Мне не нужны все возможности printf или sprintf (кто-нить использовал \b или \v или ..... ), но все лишнее намертво прилепится к выходному бинарнику.
Go to the top of the page
 
+Quote Post
scifi
сообщение Oct 24 2016, 11:19
Сообщение #15


Гуру
******

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



Цитата(demiurg_spb @ Oct 23 2016, 00:21) *
Я каждый раз задаюсь вопросом: в чём был смысл такого решения при создании сишной библиотеки?

Между прочим, если функция имеет переменное число аргументов (как sprintf), аргументы типа float приводятся к типу double перед вызовом функции. Отцы-основатели не подумали о микроконтроллерах. Да их (МК) тогда и не было, наверное laughing.gif

Цитата(zemlemer @ Oct 24 2016, 13:53) *
Мне не нужны все возможности printf или sprintf (кто-нить использовал \b или \v или ..... ), но все лишнее намертво прилепится к выходному бинарнику.

Уже проверили выравнивание стека, кратное 8 байт? Просто интересно.
Go to the top of the page
 
+Quote Post

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

 


RSS Текстовая версия Сейчас: 25th July 2025 - 12:28
Рейтинг@Mail.ru


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