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

 
 
> Проверка на равенство векторов прерывания директивой #if
ARV
сообщение Jun 7 2016, 19:42
Сообщение #1


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

Группа: Свой
Сообщений: 1 143
Регистрация: 30-09-08
Из: Новочеркасск
Пользователь №: 40 581



Уважаемые знатоки GCC!
Захотел сделать "гибкий" код (для AVR), в котором бы можно было гибко варьировать обработчики прерываний. Как-то так:
Код
#define MY_INTERRUPT     TIMER0_OVF_vect
...
ISR(MY_INTERRUPT){
...
}
Разумеется, это работает.

Теперь вторая хотелка: по значению константы-макроса MY_INTERRUPT делать ту или иную инициализацию периферии, вот примерно так:
Код
#if MY_INTERRUPT == TIMER0_OVF_vect
// тут инициализация таймера 0
#elif MY_INTERRUPT == ADC_vect
// тут инициализация АЦП
#elif MY_INTERRUPT == RXC_vect
// тут инициализация USART
// и т.д.
#endif
И вот это не работает: ВСЕГДА срабатывает первый #if, какой бы вектор не был присвоен MY_INTERRUPT!

Что я делаю не так? Как надо?


--------------------
Я бы взял частями... но мне надо сразу.
Go to the top of the page
 
+Quote Post
 
Start new topic
Ответов (1 - 12)
Сергей Борщ
сообщение Jun 8 2016, 06:25
Сообщение #2


Гуру
******

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



QUOTE (ARV @ Jun 7 2016, 22:42) *
Что я делаю не так? Как надо?
Есть подозрение, что перед этими проверками или перед определением MY_INTERRUPT не включен заголовочный файл, в котором определены #define TIMER0_OVF_vect и т.д. Поэтому все эти символы как бы равны пустой строке и все равны MY_INTERRUPT. Сделайте перед этими условиями простую проверку:

CODE
#if !defined(MY_INTERRUPT) || !defined(TIMER0_OVF_vect) || !defined(ADC_vect) || !defined(RXC_vect)
#error Попался!!!
#endif



--------------------
На любой вопрос даю любой ответ
"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
ARV
сообщение Jun 8 2016, 18:04
Сообщение #3


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

Группа: Свой
Сообщений: 1 143
Регистрация: 30-09-08
Из: Новочеркасск
Пользователь №: 40 581



Вы правы - попался!
Но КАК?! blink.gif
Векторы прерываний определены в "стандартных" для avr-gcc заголовках, в проект включается самый "главный" avr/io.h, который уже подтягивает остальные нужные. Макрос ISR отлично "видит" и ADC_vect и любой иной... как в простом if-е эти макросы вдруг оказываются неопределеными?!


--------------------
Я бы взял частями... но мне надо сразу.
Go to the top of the page
 
+Quote Post
Владивольт
сообщение Jun 8 2016, 20:55
Сообщение #4


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

Группа: Участник
Сообщений: 168
Регистрация: 14-02-10
Пользователь №: 55 490



Когда препроцессор закончит разворачивать символьные идентификаторы вида xxx_vect
по дефайнам из системных заголовочных файлов, получатся идентификаторы вида
Код
__vector_12
__vector_13
__vector_16
(Последним сработает определение
Код
#define _VECTOR(N) __vector_ ## N
из файла SFR_DEFS.H)

Далее, для директивы проверки #if все неопределённые символьные идентификаторы -- нули.
Цитата
Identifiers that are not macros, which are all considered to be the number zero.

Так что срабатывает первый #if
Код
#if 0 == 0


--------------------
#define TRUE (4==(2*2))
Go to the top of the page
 
+Quote Post
jorikdima
сообщение Jun 8 2016, 20:55
Сообщение #5


тут может быть ваша реклама
*****

Группа: Свой
Сообщений: 1 164
Регистрация: 15-03-06
Из: Санкт-Петербург/CA
Пользователь №: 15 280



Найдите (скорее всего в настройках надо квлючить) результат препроцессора. Там увидите, что именно идет компилятору. Ну а далее внимательно искать почему дефайн не виден, как тут подскажешь.

Во, ответили уже.
Go to the top of the page
 
+Quote Post
arhiv6
сообщение Jun 9 2016, 03:55
Сообщение #6


Знающий
****

Группа: Свой
Сообщений: 633
Регистрация: 21-05-10
Из: Томск
Пользователь №: 57 423



Цитата
Найдите (скорее всего в настройках надо включить) результат препроцессора.

Для gcc это делается ключом -E
Код
gcc -E test.c > test.txt


--------------------
Go to the top of the page
 
+Quote Post
ARV
сообщение Jun 10 2016, 18:45
Сообщение #7


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

Группа: Свой
Сообщений: 1 143
Регистрация: 30-09-08
Из: Новочеркасск
Пользователь №: 40 581



хм... Получается, __vector_12 для компилятора такой же ноль, как и __vector_11? как же тогда разные вектора прерываний получаются? ISR(__vector_12) совсем не совпадает с ISR(__vector_11)... что-то я не понимаю...


--------------------
Я бы взял частями... но мне надо сразу.
Go to the top of the page
 
+Quote Post
Владивольт
сообщение Jun 10 2016, 22:38
Сообщение #8


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

Группа: Участник
Сообщений: 168
Регистрация: 14-02-10
Пользователь №: 55 490



Цитата(ARV @ Jun 10 2016, 22:45) *
хм... Получается, __vector_12 для компилятора такой же ноль, как и __vector_11? как же тогда разные вектора прерываний получаются? ISR(__vector_12) совсем не совпадает с ISR(__vector_11)... что-то я не понимаю...

не для компилятора, а для препроцессора.
Процедуры расставит линкер и скорректирует адреса переходов
Код
50:○0c 94 7c 10 ○ jmp○0x20f8○ ; 0x20f8 <__vector_20>
54:○0c 94 5e 11 ○ jmp○0x22bc○ ; 0x22bc <__vector_21>
58:○0c 94 50 11 ○ jmp○0x22a0○ ; 0x22a0 <__vector_22>
5c:○b1 c2       ○ rjmp○ .+1378   ○; 0x5c0 <__bad_interrupt>
5e:○00 00       ○ nop
60:○af c2       ○ rjmp○ .+1374   ○; 0x5c0 <__bad_interrupt>
62:○00 00       ○ nop
64:○ad c2       ○ rjmp○ .+1370   ○; 0x5c0 <__bad_interrupt>
66:○00 00       ○ nop


--------------------
#define TRUE (4==(2*2))
Go to the top of the page
 
+Quote Post
Сергей Борщ
сообщение Jun 11 2016, 05:27
Сообщение #9


Гуру
******

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



QUOTE (ARV @ Jun 8 2016, 21:04) *
Вы правы - попался!
Но КАК?! blink.gif
Нет такого вектора RXC_vect, он называется USART_RX_vect. Ответа на основной вопрос не знаю, увы.


--------------------
На любой вопрос даю любой ответ
"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
ARV
сообщение Jun 13 2016, 06:22
Сообщение #10


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

Группа: Свой
Сообщений: 1 143
Регистрация: 30-09-08
Из: Новочеркасск
Пользователь №: 40 581



В общем, получается, что взять один из предопределенных макросов и использовать его, как селектор вариантов собственного кода, я не могу...
Очень это огорчительно... и не понятно.


--------------------
Я бы взял частями... но мне надо сразу.
Go to the top of the page
 
+Quote Post
aiwa
сообщение Jun 13 2016, 11:27
Сообщение #11


Местный
***

Группа: Участник
Сообщений: 301
Регистрация: 13-12-15
Из: Харьков
Пользователь №: 89 682



Вы можете слегка изменить конструкцию, воспользовавшись стандартными определениями номеров векторов:
Код
#define MY_INTERRUPT_num     __TIMER0_OVF_vect_num
#define MY_INTERRUPT     _VECTOR(MY_INTERRUPT_num)
...
ISR(MY_INTERRUPT){
...
}

#if MY_INTERRUPT_num == TIMER0_OVF_vect_num
// тут инициализация таймера 0
#elif MY_INTERRUPT_num == ADC_vect_num
// тут инициализация АЦП
#elif MY_INTERRUPT_num == RXC_vect_num
// тут инициализация USART
// и т.д.
#endif


Сообщение отредактировал aiwa - Jun 13 2016, 11:38
Go to the top of the page
 
+Quote Post
ARV
сообщение Jun 22 2016, 11:26
Сообщение #12


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

Группа: Свой
Сообщений: 1 143
Регистрация: 30-09-08
Из: Новочеркасск
Пользователь №: 40 581



Цитата(aiwa @ Jun 13 2016, 15:27) *
Вы можете слегка изменить конструкцию, воспользовавшись стандартными определениями номеров векторов

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

в случае с нестандартными определениями векторов смысл в этом пропадает, т.к. чем нагружать пользователя дополнительными инструкциями по использованию дополнительных макросов, лучше уж вообще этим не заниматься...


--------------------
Я бы взял частями... но мне надо сразу.
Go to the top of the page
 
+Quote Post
aiwa
сообщение Jun 22 2016, 14:42
Сообщение #13


Местный
***

Группа: Участник
Сообщений: 301
Регистрация: 13-12-15
Из: Харьков
Пользователь №: 89 682



Цитата(ARV @ Jun 22 2016, 14:26) *
во-первых, я не понимаю, чем это будет отличаться от использования "стандартных" макросов, уже определенных
.................
в случае с нестандартными определениями векторов смысл в этом пропадает.....

Во-первых, определения "TIMER0_OVF_vect_num" имеют такую же "стандартность" как и "TIMER0_OVF_vect", потому что определены в стандартных заголовочных файлах равносильно:
Код
например, для мега16:
/* Timer/Counter0 Overflow */
#define TIMER0_OVF_vect_num        9
#define TIMER0_OVF_vect            _VECTOR(9)
#define SIG_OVERFLOW0            _VECTOR(9)


код с определениями с "_num", будет отличаться от "стандартных" тем, что препроцессор его все-таки кушает правильно.
Проверял на Вашем примере, единственное - не удалил два лидирующих подчеркивания в дефайне.

В принципе, отличие от первоначальной идеи всего лишь в суффиксе.
Вместо "#define MY_INTERRUPT TIMER0_OVF_vect" пользватель будет писать "#define MY_INTERRUPT_num TIMER0_OVF_vect_num".

Конструкция "#define MY_INTERRUPT _VECTOR(MY_INTERRUPT_num)" может быть реализована без его участия:
#ifdef MY_INTERRUPT_num
#define MY_INTERRUPT _VECTOR(MY_INTERRUPT_num)
#endif



Сообщение отредактировал aiwa - Jun 22 2016, 14:54
Go to the top of the page
 
+Quote Post

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

 


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


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