Перейти к содержанию

Битовые Операции В Gcc


rmatveev

Рекомендуемые сообщения

Продолжаю биться с сабжом в рамках того, что я накопал.

Вот хочу я объявить PORTB как битовое поле. PORTB в хедере определяется как _SFR_IO8(0x18) (это AtTiny26). Как мне тогда объявить PORTB0?

struct {

char b0: 1;

char b1: 1;

char b2: 1;

char b3: 1;

char b4: 1;

char b5: 1;

char b6: 1;

char b7: 1;

} _SFR_IO8(0x18) PORTB;

Но не компилируется (AVR-GCC). Пишет error: expected identifier or '(' before 'volatile'

UPD: Пока суть да дело, пишу по-старинке:

#define _DAC_DIN PORTA,0

#define _DAC_SCLK PORTA,2

#define _DAC_SYNC PORTA,3

#define _setb(port,bit) ( (port)|= (1<<(bit)) )

#define _clrb(port,bit) ( (port)&=~(1<<(bit)) )

#define _wrib(port,bit,data) ( (data)?(_setb(port,bit) :(_clrb(port,bit)) )

#define _testb(port,bit) ( (port)&=(1<<(bit)) )

void WriteDAC( char value )

{

unsigned data = value;

data = data << 4;

_setb( _DAC_SYNC );

_setb( _DAC_SCLK );

_clrb( _DAC_SYNC );

for( int i=0; i<16; i++ )

{

_wrib( _DAC_DIN, _testb( data, 15 ) );

_clrb( _DAC_SCLK );

_setb( _DAC_SCLK );

}

}

Но компилятор ругается, что, например, макрос _clrb требует два параметра, а я ему даю только один. Т.е. вызов макроса внутри макроса отрабатывается не так как мне бы хотелось. Как можно форсить исполнить макрос внутри макроса?

Я для доступа к битам порта использую следующий способ:

union bfbyte

{

struct

{

unsigned char bit0 :1;

unsigned char bit1 :1;

unsigned char bit2 :1;

unsigned char bit3 :1;

unsigned char bit4 :1;

unsigned char bit5 :1;

unsigned char bit6 :1;

unsigned char bit7 :1;

};

unsigned char byte; // это поле для того чтоб сразу байт в порт вывести

};

#ifdef PORTB

#define portb (*((volatile bfbyte*)&PORTB))

#endif

int main(void)

{

while(1)

{

portb.bit0 = 1; // установка PB0 в 1

portb.bit0 = 0; // установка PB0 в 0

portb.byte = 0x03; // Запись в порт значения

}

}

Работает в AvrStudio 6.1. При вводе точки после имени (portb) штатная подсветка ситтаксиса предлагает варианты полей что очень удобно

Компилятором генерится следующий код:

00000019 LDI R24,0x03 Load immediate // загрузка значения порта в регистр

0000001A SBI 0x18,0 Set bit in I/O register // portb.bit0 = 1;

0000001B CBI 0x18,0 Clear bit in I/O register // portb.bit0 = 0;

0000001C OUT 0x18,R24 Out to I/O location // portb.byte = 0x03;

0000001D RJMP PC-0x0003 Relative jump // переход (бесконечный цикл)

как видно из листинга никаких излишеств!

определение

#ifdef PORTB

#define portb (*((volatile bfbyte*)&PORTB))

#endif

можно заменить на (объявление переменной по ссылке)

#ifdef PORTB

volatile bfbyte &portb = *((volatile bfbyte*)&PORTB);

#endif

Ссылка на комментарий
Поделиться на другие сайты

Реклама: ООО ТД Промэлектроника, ИНН: 6659197470, Тел: 8 (800) 1000-321

Присоединяйтесь к обсуждению

Вы можете написать сейчас и зарегистрироваться позже. Если у вас есть аккаунт, авторизуйтесь, чтобы опубликовать от имени своего аккаунта.
Примечание: Ваш пост будет проверен модератором, прежде чем станет видимым.

Гость
Unfortunately, your content contains terms that we do not allow. Please edit your content to remove the highlighted words below.
Ответить в этой теме...

×   Вставлено с форматированием.   Восстановить форматирование

  Разрешено использовать не более 75 эмодзи.

×   Ваша ссылка была автоматически встроена.   Отображать как обычную ссылку

×   Ваш предыдущий контент был восстановлен.   Очистить редактор

×   Вы не можете вставлять изображения напрямую. Загружайте или вставляйте изображения по ссылке.

Загрузка...
  • Последние посетители   0 пользователей онлайн

    • Ни одного зарегистрированного пользователя не просматривает данную страницу
×
×
  • Создать...