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

 
 
> Битовые сдвиги
CSB
сообщение Jun 17 2006, 15:01
Сообщение #1


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

Группа: Новичок
Сообщений: 100
Регистрация: 9-03-06
Пользователь №: 15 088



Код

#include <iom8.h>

void main(void)
{
    unsigned long int a, n;

    n = 16;
    a = 1 << n;
    PORTB = a;
}

По идее в переменной "а" должен быть установлен 16 бит, а на деле там 0. Какие есть идеи?

Компилятор:
IAR C/C++ Compiler for AVR
4.12A/W32 [Evaluation] (4.12.1.3)

Для просмотра значений переменных использую родной симулятор.
Go to the top of the page
 
+Quote Post
 
Start new topic
Ответов (1 - 11)
AndyBig
сообщение Jun 17 2006, 15:28
Сообщение #2


Иногдящий
****

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



Попробуйте определить переменную а как unsigned long.
Go to the top of the page
 
+Quote Post
CSB
сообщение Jun 17 2006, 15:51
Сообщение #3


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

Группа: Новичок
Сообщений: 100
Регистрация: 9-03-06
Пользователь №: 15 088



Не помогло sad.gif
Go to the top of the page
 
+Quote Post
GetSmart
сообщение Jun 17 2006, 15:57
Сообщение #4


.
******

Группа: Участник
Сообщений: 4 005
Регистрация: 3-05-06
Из: Россия
Пользователь №: 16 753



Что это за тип вообще: long int ?
Это на каком языке написано?


--------------------
Заблуждаться - Ваше законное право :-)
Go to the top of the page
 
+Quote Post
Сергей Борщ
сообщение Jun 17 2006, 15:58
Сообщение #5


Гуру
******

Группа: Модераторы
Сообщений: 8 455
Регистрация: 15-05-06
Из: Рига, Латвия
Пользователь №: 17 095



Цитата(CSB @ Jun 17 2006, 18:01) *
Код

    n = 16;
    a = 1 << n;

По идее в переменной "а" должен быть установлен 16 бит, а на деле там 0. Какие есть идеи?

1 имеет тип int. Для avr int имеет 16 бит. Соответственно любой int сдвинутый на 16 будет нулем, что и имеем. Решение: привести 1 к типу unsigned long (или к unsigned long long) или сразу указать что она такого типа:
Код
     a = (unsigned long long)1 << n;
или
     a = 1ULL << n;


--------------------
На любой вопрос даю любой ответ
"Write code that is guaranteed to work, not code that doesn’t seem to break" (C++ FAQ)
Go to the top of the page
 
+Quote Post
GetSmart
сообщение Jun 17 2006, 16:00
Сообщение #6


.
******

Группа: Участник
Сообщений: 4 005
Регистрация: 3-05-06
Из: Россия
Пользователь №: 16 753



try this
a = (long)1 << n;


--------------------
Заблуждаться - Ваше законное право :-)
Go to the top of the page
 
+Quote Post
CSB
сообщение Jun 17 2006, 16:07
Сообщение #7


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

Группа: Новичок
Сообщений: 100
Регистрация: 9-03-06
Пользователь №: 15 088



Спасибо.

Но в сязи с этим возникает другой вопрос. Делаем тип переменной "а" unsigned long long. Как теперь сдвинуть на, например, 40 бит? Тут проблемы возникают даже при сдвиге на 7 бит.

Если провести присваивание вида
a = 1ULL;
то получим в "а" значение 0x0100000000000000;
Получается какя-то путаница в байтах числа.

При
a = 0xAFFULL;
а == 0xFF00000000000000

a = 0xABCDEFABCDEFABCD;
a == 0xCD00000000000000

Сообщение отредактировал CSB - Jun 17 2006, 16:26
Go to the top of the page
 
+Quote Post
Сергей Борщ
сообщение Jun 17 2006, 16:32
Сообщение #8


Гуру
******

Группа: Модераторы
Сообщений: 8 455
Регистрация: 15-05-06
Из: Рига, Латвия
Пользователь №: 17 095



Цитата(CSB @ Jun 17 2006, 19:07) *
Спасибо.

Но в сязи с этим возникает другой вопрос. Делаем тип переменной "а" unsigned long long. Как теперь сдвинуть на, например, 40 бит? Тут проблемы возникают даже при сдвиге на 7 бит.
Где проблемы возникают?
Код
  unsigned long long a;
  a = 1ULL << 40;

     21              a = 1ULL << 40;
   \   00000002   E000               LDI     R16, 0
   \   00000004   E051               LDI     R21, 1
   \   00000006   ....               LDI     R30, LOW(a)
   \   00000008   ....               LDI     R31, (a) >> 8
   \   0000000A   8300               ST      Z, R16
   \   0000000C   8301               STD     Z+1, R16
   \   0000000E   8302               STD     Z+2, R16
   \   00000010   8303               STD     Z+3, R16
   \   00000012   8304               STD     Z+4, R16
   \   00000014   8355               STD     Z+5, R21
   \   00000016   8306               STD     Z+6, R16
   \   00000018   8307               STD     Z+7, R16
IAR AVR v4.10B


Цитата(CSB @ Jun 17 2006, 19:07) *
Если провести присваивание вида
a = 1ULL;
то получим в "а" значение 0x0100000000000000;
Получается какя-то путаница в байтах числа.

При
a = 0xAFFULL;
а == 0xFF00000000000000

a = 0xABCDEFABCDEFABCD;
a == 0xCD00000000000000

Похоже вы не в ту сторону считаете, это у вас от кейла для x51 привычка к BigEndian. У AVR младший байт располагается по младшему адресу.


--------------------
На любой вопрос даю любой ответ
"Write code that is guaranteed to work, not code that doesn’t seem to break" (C++ FAQ)
Go to the top of the page
 
+Quote Post
CSB
сообщение Jun 17 2006, 16:45
Сообщение #9


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

Группа: Новичок
Сообщений: 100
Регистрация: 9-03-06
Пользователь №: 15 088



В ассемблере у меня получается две команды: очистка регистров и все!

Оптимизация: переключатель в положении "скорость", в списке выбрано None (Best debug support).

Кроме того получается разный результат при
a = 1ULL << 7;
и
n = 7;
a = 1ULL << n;
Go to the top of the page
 
+Quote Post
_Bill
сообщение Jun 17 2006, 16:48
Сообщение #10


Местный
***

Группа: Участник
Сообщений: 416
Регистрация: 18-04-06
Из: Челябинск
Пользователь №: 16 219



Цитата(GetSmart @ Jun 17 2006, 19:00) *
try this
a = (long)1 << n;

Обычно это делается так
Код
a = 1L << n;        // Константа со знаком
или так
Код
a = 1UL << n;      // Константа без знака
Go to the top of the page
 
+Quote Post
Сергей Борщ
сообщение Jun 17 2006, 17:14
Сообщение #11


Гуру
******

Группа: Модераторы
Сообщений: 8 455
Регистрация: 15-05-06
Из: Рига, Латвия
Пользователь №: 17 095



Цитата(CSB @ Jun 17 2006, 19:45) *
В ассемблере у меня получается две команды: очистка регистров и все!
Оптимизация: переключатель в положении "скорость", в списке выбрано None (Best debug support).
Кроме того получается разный результат при
a = 1ULL << 7;
и
n = 7;
a = 1ULL << n;
Что-то не так. Приведи весь код. Вот что получается у меня:
Код
volatile int a;
void main (void) {
    int n = 7;
    a = 1ULL << 7;
    a = 1ULL << n;
}
     19          void main (void) {
   \                     main:
   \   00000000   939A               ST      -Y, R25
   \   00000002   938A               ST      -Y, R24
     20              int n = 7;
     21              a = 1ULL << 7;
   \   00000004   E800               LDI     R16, 128
   \   00000006   E010               LDI     R17, 0
   \   00000008   ....               LDI     R30, LOW(a)
   \   0000000A   ....               LDI     R31, (a) >> 8
   \   0000000C   8300               ST      Z, R16
   \   0000000E   8311               STD     Z+1, R17
     22              a = 1ULL << n;
   \   00000010   8300               ST      Z, R16
   \   00000012   8311               STD     Z+1, R17
Результат одинаковый.


--------------------
На любой вопрос даю любой ответ
"Write code that is guaranteed to work, not code that doesn’t seem to break" (C++ FAQ)
Go to the top of the page
 
+Quote Post
CSB
сообщение Jun 17 2006, 17:37
Сообщение #12


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

Группа: Новичок
Сообщений: 100
Регистрация: 9-03-06
Пользователь №: 15 088



Господа, каюсь smile.gif
Я забыл про одну весьма немаловажную вещь: volatile!
Забыл с самого начала и что-бы компилятор не выкидывал код я делал PORTB = a; проверив перед этим: а вдруг "умный" компилятор посчитает старшую часть числа не нужной и не будет ее вычислять; для 16-битовых чисел все работало.

2 Сергей Борщ:
Большое спасибо!
Go to the top of the page
 
+Quote Post

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

 


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


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