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

 
 
 
Reply to this topicStart new topic
> Преобразовать uint8_t в int32_t, что-то в Keil для STM32 не идет
ViKo
сообщение Dec 14 2010, 10:31
Сообщение #1


Универсальный солдатик
******

Группа: Модераторы
Сообщений: 8 634
Регистрация: 1-11-05
Из: Минск
Пользователь №: 10 362



Имею буфер приема от панели управления (в составе структуры FPI)
uint8_t FP_RBf[2];
Хочу прочитать этот буфер и вызвать функцию по значению первого байта, и значение второго байта передать в функцию, как int32_t.
Код
uint32_t KeyNum = (uint32_t)FPI.FP_RBf[0];
int32_t  KeyCnt = (int32_t) FPI.FP_RBf[1];
FpKey_hand[KeyNum](KeyCnt);

Однако заметил, что при считывании RBf[1] компилятор использует команду LDRB, а не LDRSB
Код
;;;295      int32_t  KeyCnt = (int32_t) FPI.FP_RBf[1];
0001a4  78c0              LDRB     r0,[r0,#3] ; FPI

Т.е. не происходит знакового расширения байта. Как сделать правильно?

Заменил тип в буфере на int8_t. Получилось следующее:
Код
;;;294      uint32_t KeyNum = (uint32_t)FPI.FP_RBf[0];
0001a4  f9901002          LDRSB    r1,[r0,#2] ; FPI
;;;295      int32_t  KeyCnt = (int32_t)FPI.FP_RBf[1];
0001a8  f9900003          LDRSB    r0,[r0,#3] ; FPI

Так оно работает, но непонятки остались... Теперь обе переменные размножили знак.
Go to the top of the page
 
+Quote Post
dxp
сообщение Dec 14 2010, 11:07
Сообщение #2


Adept
******

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



Цитата(ViKo @ Dec 14 2010, 16:31) *
Имею буфер приема от панели управления (в составе структуры FPI)
uint8_t FP_RBf[2];
Хочу прочитать этот буфер и вызвать функцию по значению первого байта, и значение второго байта передать в функцию, как int32_t.
Код
uint32_t KeyNum = (uint32_t)FPI.FP_RBf[0];
int32_t  KeyCnt = (int32_t) FPI.FP_RBf[1];
FpKey_hand[KeyNum](KeyCnt);

Однако заметил, что при считывании RBf[1] компилятор использует команду LDRB, а не LDRSB
Код
;;;295      int32_t  KeyCnt = (int32_t) FPI.FP_RBf[1];
0001a4  78c0              LDRB     r0,[r0,#3]; FPI

Т.е. не происходит знакового расширения байта. Как сделать правильно?

Не знаток АРМа, но навскидку: во-первых, зачем ручное преобразование типов - ведь

uint32_t KeyNum = FPI.FP_RBf[0];

даст то же самое.

Во-вторых, исходный операнд у вас беззнаковый, т.е. знака там нет в принципе, поэтому расширять нечего и по правилам стандартных преобразований типов компилятор просто честно забивает нулями старшие биты. Когда вы поменяли тип источника на знаковый, то знак появился и опять же по тем же правилам стандартных преобразований С/С++ компилятор должен выполнить расширение знака.


--------------------
«Отыщи всему начало, и ты многое поймёшь» К. Прутков
Go to the top of the page
 
+Quote Post
ViKo
сообщение Dec 14 2010, 11:29
Сообщение #3


Универсальный солдатик
******

Группа: Модераторы
Сообщений: 8 634
Регистрация: 1-11-05
Из: Минск
Пользователь №: 10 362



Цитата(dxp @ Dec 14 2010, 13:07) *
зачем ручное преобразование типов - ведь
uint32_t KeyNum = FPI.FP_RBf[0];
даст то же самое.

Ну, эта строка написана "по инерции" за компанию, чтобы не отличалась от следующей.
Вот какое решение нашлось, для буфера из uint8_t
Код
;;;305      int8_t temp = FPI.FP_RBf[1];  
0001a4  f9900003          LDRSB    r0,[r0,#3]; FPI
;;;306      int32_t KeyCnt = (int32_t)temp;

Как видите, нашлось что расширять sm.gif
Go to the top of the page
 
+Quote Post
XVR
сообщение Dec 15 2010, 04:29
Сообщение #4


Гуру
******

Группа: Свой
Сообщений: 3 123
Регистрация: 7-04-07
Из: Химки
Пользователь №: 26 847



Код
int32_t  KeyCnt = (int8_t) FPI.FP_RBf[1];
Go to the top of the page
 
+Quote Post
ViKo
сообщение Dec 15 2010, 05:57
Сообщение #5


Универсальный солдатик
******

Группа: Модераторы
Сообщений: 8 634
Регистрация: 1-11-05
Из: Минск
Пользователь №: 10 362



Цитата(XVR @ Dec 15 2010, 09:29) *
Код
int32_t  KeyCnt = (int8_t) FPI.FP_RBf[1];

Да, проходит и такое.
А я уж было "изобрел" конструкцию "с двойным преобразованием" sm.gif
Код
int32_t KeyCnt = (int32_t)(int8_t)FPI.FP_RBf[1];

А, может, так, как я написал, надежнее, правильнее? Все-таки, конкретное указание...
Go to the top of the page
 
+Quote Post
XVR
сообщение Dec 16 2010, 04:25
Сообщение #6


Гуру
******

Группа: Свой
Сообщений: 3 123
Регистрация: 7-04-07
Из: Химки
Пользователь №: 26 847



Цитата(ViKo @ Dec 15 2010, 11:57) *
Код
int32_t KeyCnt = (int32_t)(int8_t)FPI.FP_RBf[1];

А, может, так, как я написал, надежнее, правильнее? Все-таки, конкретное указание...
Второе преобразование (int32_t) компилятор сделает сам (причем молча - это вполне штатная ситуация). Так что от наличия или отсутствия (int32_t) в приведении ничего не изменится.

Go to the top of the page
 
+Quote Post
ViKo
сообщение Dec 16 2010, 06:16
Сообщение #7


Универсальный солдатик
******

Группа: Модераторы
Сообщений: 8 634
Регистрация: 1-11-05
Из: Минск
Пользователь №: 10 362



Цитата(XVR @ Dec 16 2010, 09:25) *
Второе преобразование (int32_t) компилятор сделает сам (причем молча - это вполне штатная ситуация). Так что от наличия или отсутствия (int32_t) в приведении ничего не изменится.

Полностью согласен. Правая часть выражения должна быть приведена к размерности переменной в левой части.
Остался маленький вопросик - преобразование беззнакового байта в знаковый - это просто "подмена представления", не вызывающая у компилятора даже варнинга? Все-таки, например, 255 превращается в -1, и ничего страшного?
Go to the top of the page
 
+Quote Post
dxp
сообщение Dec 16 2010, 07:10
Сообщение #8


Adept
******

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



Цитата(ViKo @ Dec 16 2010, 15:16) *
Остался маленький вопросик - преобразование беззнакового байта в знаковый - это просто "подмена представления", не вызывающая у компилятора даже варнинга? Все-таки, например, 255 превращается в -1, и ничего страшного?

С чего компилятору тут ругаться - ведь ему же программистом указано. Предупреждения компилятор выдает, когда имеет место неявное преобразование, которое делается как бы скрытно и поэтому может привести к неожиданному (для программиста) поведению кода. Если программист сам руками указал, что этот объект в данном случае трактовать так и не иначе, то компилятор должен выполнять.

Другое дело, что сишная форма явного преобразования типов этакая "неброская", ее иногда легко пропустить при чтении кода - похожа на аргумент выражения или функции. Поэтому в С++ специально ввели новый синтаксис для явного преобразования типов, который своим безобразным видом сразу привлекает к себе внимание - чтобы в случае чего можно быстро увидеть потенциально опасные места. Ну, и еще разделили единую форму на две различные, отличающиеся степенью опасности: static_cast и reinterpret_cast.


--------------------
«Отыщи всему начало, и ты многое поймёшь» К. Прутков
Go to the top of the page
 
+Quote Post

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

 


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


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