|
Как перейти к началу функции по внешнему прерыванию?, подробнее внутри |
|
|
|
Aug 15 2008, 13:30
|
Местный
  
Группа: Свой
Сообщений: 413
Регистрация: 15-12-06
Пользователь №: 23 563

|
Господа есть код в CodeVision примерно такой: Код void loop() { for (i=0;i<360;i++) { ................. }
}
void afterINT() { ..... }
void main () { while (1==1) { ...... loop(); ..... } } Как сделать чтобы по внешнему прерыванию от пина AVR функция loop завершала свою работу? выполнялась функция afterINT(), после чего вновь запускалась функция loop(). Пишу в CodeVision, но буду рад примеру и в WinAVR. И еще - в ATMEGA32 INT0, INT1 и INT2 чемн-ниьбудь отличаются?
|
|
|
|
|
 |
Ответов
|
Aug 17 2008, 17:42
|
Профессионал
    
Группа: Свой
Сообщений: 1 719
Регистрация: 13-09-05
Из: Novosibirsk
Пользователь №: 8 528

|
Цитата(zltigo @ Aug 17 2008, 18:10)  Рассуждай токмо о том, о чём понятия твои тебе сие дозволяют. Так: не зная законов языка ирокезского, можешь ли ты делать такое суждение по сему предмету, которое не было бы неосновательно и глупо? (с) Козьма Прутков. Афоризмы  Понятия дозволяют. Я законы языка ирокез С ещё на PDP-11 изучал. И с работой setjmp/longjmp тогда же разбирался. Цитата(aaarrr @ Aug 17 2008, 18:14)  Замечательно. А вы умеете готовить программы, которые будут нормально работать при вываливании из любой точки после прерывания в другую? Чтобы из любой - это гораздо сложнее. Но Вас же не удивляет что любая (некооперативная) RTOS может передать управление другому процессу после того как текущий был прерван прерыванием? Разбираемый сейчас случай попроще будет. Требуется поведение программы типа "всё бросить и вернуться на заранее подготовленные позиции", как раз для такого эти функции и предназначены. В популярных учебниках тему setjmp/longjmp предпочитают не затрагивать, вот народ про них почти ничего и не знает. Попробую рассказать что знаю, с некоторыми упрощениями. Функция setjmp() всего лишь запоминает в специальном буфере содержимое нескольких регистров, указатель стека и счётчик команд (адрес возврата, лежащий в стеке). Эта информация потом, при исполнении longjmp(), позволяет восстановить состояние программы таким, словно только что произошел возврат из setjmp(). Отличается только возвращаемое значение, setjmp() возвращает 0, а longjmp() возвращает свой второй аргумент. Это позволяет понять откуда попало управление в этот раз. Разумеется, восстанавливается далеко не всё, скажем если программа успела изменить глобальные переменные то эти изменения не будут откачены назад, ну так это-то уже в руках программиста, и он может это предусмотреть, если бывает нужно "прибрать за собой". Поскольку longjmp() переставляет указатель стека на прежнее место, то существует естественное ограничение на её применение: состояние той части стека, которая существовала на момент возврата из setjmp() не должно изменяться до выполнения longjmp(). Другими словами, не должно быть выхода из функции или блока, содержащего setjmp(). А с учетом архитектуры AVR и имеющейся реализации этих функций не вижу никаких препятствий для примения longjmp() прямо из обработчика прерывания. Реализацию в CodeVision надо проверить, но скорее всего и там будет работать. Примерно так (для IAR): Код jmp_buf jbuf;
int foo() { if( setjmp( jbuf ) ) { // исполняется только после перехода по longjmp() //если переход делать прямо из обработчика, то прерывания останутся запрещёнными __enable_interrupt(); afterINT(); } else { // исполняется после возврата из setjmp() }
for(;;) { loop(); //........ } }
#pragma vector=INT0_vect __interrupt void INT0_isr(void) { longjmp( jbuf, 1); }
--------------------
Russia est omnis divisa in partes octo.
|
|
|
|
|
Aug 17 2008, 22:15
|

Гуру
     
Группа: Свой
Сообщений: 13 372
Регистрация: 27-11-04
Из: Riga, Latvia
Пользователь №: 1 244

|
Цитата(SSerge @ Aug 17 2008, 19:42)  Требуется поведение программы типа "всё бросить и вернуться на заранее подготовленные позиции", как раз для такого эти функции и предназначены. Ну насчет все грубо бросить в непонятном состоянии И ЗАБЫТЬ НА ВСЕГДА, это еще большой вопрос о допустимости такого. Цитата Попробую рассказать что знаю, с некоторыми упрощениями. Спасибо, не надо - плавали. Особенно с упрощениями, ибо дявольщина таится в деталях. А деталей реализации там много и некоторые даже подбираются ключами в пределах одного компилятора. Цитата Разумеется, восстанавливается далеко не всё... ..И при прерывании/восстановлении в Цитата ..уже в руках программиста... столько всего окажется, что разбираться в отсутствии побочных эффектах придется долго. Цитата А с учетом архитектуры AVR и имеющейся реализации этих функций не вижу никаких препятствий для примения longjmp() прямо из обработчика прерывания. Ага, самая примтивная архитектура и железо - скомпенсировали отсутствие RETI и типа все... А на ARM слабо? Цитата Реализацию в CodeVision надо проверить... Ага, проверить, и при получении свежего компилятора проверить... Цитата , но скорее всего и там будет работать. Примерно так (для IAR): Ну и чего добились этими финтами? Эквивалент внесения afterINT() в обработчик прерывания + непонятные эффекты (особенно в случае прерывания afterINT()) + опасно переносимый или гарантированно не переносимый код. Нет это не "руками хирурга" сделано  И даже перечисленные лично Вами подводные камни, камня на камне не оставляют от Вашего-же утверждения Цитата Вообще-то для таких "фокусов" придуманы функции setjmp() и longjmp(). с которого все и началось...
--------------------
Feci, quod potui, faciant meliora potentes
|
|
|
|
Сообщений в этой теме
Зверюга Как перейти к началу функции по внешнему прерыванию? Aug 15 2008, 13:30 GDI ЦитатаКак сделать чтобы по внешнему прерыванию от ... Aug 15 2008, 13:38 Зверюга Цитатапрерывания, или вам надо чтобы после прерыва... Aug 15 2008, 13:46 GDI Можно в цикле что внутри loop() проверять некий фл... Aug 15 2008, 14:00 AHTOXA Можно в прерывании подменять адрес возврата. Но эт... Aug 15 2008, 14:14 bzx Цитата(AHTOXA @ Aug 15 2008, 18:14) Можно... Aug 15 2008, 16:14 rezident В plain Си функция не может перезапустить саму себ... Aug 15 2008, 15:01 Kuzmi4 2 Зверюга
ЦитатаА все-таки?
0 от 1 отличаются вро... Aug 15 2008, 15:01 Зверюга ЦитатаТолько не забудьте присвоить переменной флаг... Aug 15 2008, 16:00 zltigo Цитата(Зверюга @ Aug 15 2008, 18:00) Так ... Aug 15 2008, 16:16 rezident Цитата(Зверюга @ Aug 15 2008, 22:00) Так ... Aug 15 2008, 16:27 Зверюга Цитата{ cntr--;
}
Эта строчка "от балды... Aug 15 2008, 16:57 rezident Цитата(Зверюга @ Aug 15 2008, 22:57) Эта ... Aug 15 2008, 17:17  zltigo Цитата(rezident @ Aug 15 2008, 19:17) ...... Aug 15 2008, 17:49   rezident Цитата(zltigo @ Aug 15 2008, 23:49) грамо... Aug 15 2008, 19:42    zltigo Цитата(rezident @ Aug 15 2008, 21:42) ...... Aug 15 2008, 20:53     rezident Цитата(zltigo @ Aug 16 2008, 02:53) Масло... Aug 15 2008, 21:18      zltigo Цитата(rezident @ Aug 15 2008, 23:18) ..м... Aug 15 2008, 22:05      Aesthete Animus Цитата(rezident @ Aug 16 2008, 01:18) К т... Aug 15 2008, 23:12       rezident Цитата(Aesthete Animus @ Aug 16 2008, 05... Aug 15 2008, 23:34 Зверюга Спасибо, теперь я разобрался. Это точно для CodeVi... Aug 15 2008, 17:37 zltigo Цитата(rezident @ Aug 16 2008, 01:34) ОК.... Aug 16 2008, 08:58 rezident Цитата(zltigo @ Aug 16 2008, 14:58) Э... ... Aug 16 2008, 12:17  zltigo Цитата(rezident @ Aug 16 2008, 14:17) Я н... Aug 16 2008, 12:35 aaarrr Цитата(rezident @ Aug 16 2008, 16:17) Я н... Aug 16 2008, 12:36 rezident Цитата(aaarrr @ Aug 16 2008, 18:36) Так з... Aug 16 2008, 12:40  zltigo Цитата(rezident @ Aug 16 2008, 14:40) Доб... Aug 16 2008, 12:52   rezident Цитата(zltigo @ Aug 16 2008, 18:52) Для н... Aug 16 2008, 13:23    aaarrr Цитата(rezident @ Aug 16 2008, 17:23) Пос... Aug 16 2008, 13:43    zltigo Цитата(rezident @ Aug 16 2008, 15:23) А в... Aug 16 2008, 13:52 SSerge Вообще-то для таких "фокусов" придуманы ... Aug 16 2008, 21:20 aaarrr Цитата(SSerge @ Aug 17 2008, 01:20) Вообщ... Aug 16 2008, 21:43  SSerge Цитата(aaarrr @ Aug 17 2008, 04:43) Не дл... Aug 17 2008, 11:03   zltigo Цитата(SSerge @ Aug 17 2008, 13:03) Вот к... Aug 17 2008, 11:10   aaarrr Цитата(SSerge @ Aug 17 2008, 15:03)
Вы и... Aug 17 2008, 11:14 aaarrr Цитата(SSerge @ Aug 17 2008, 21:42) Чтобы... Aug 17 2008, 22:00 @Ark Цитата(SSerge @ Aug 17 2008, 21:42) ... Р... Aug 18 2008, 13:37  AndrewN Прежде всего, спасибо всем за возможность освежить... Aug 18 2008, 15:25   aaarrr Цитата(AndrewN @ Aug 18 2008, 19:25) Для ... Aug 18 2008, 15:31   @Ark Цитата(AndrewN @ Aug 18 2008, 19:25) ... ... Aug 18 2008, 17:28    AndrewN Цитата(@Ark @ Aug 18 2008, 20:28) Вы, воз... Aug 18 2008, 19:06 GDI не кажется что за спором вы забыли одну важную вещ... Aug 18 2008, 10:04 MMos Недавно в одной ветке, я обмолвился, что пишу на а... Aug 19 2008, 08:25 SSerge Цитата(MMos @ Aug 19 2008, 15:25) Почитал... Aug 19 2008, 09:08 zltigo Цитата(MMos @ Aug 19 2008, 10:25) В ассем... Aug 19 2008, 09:21 MMos Я суть темы понял так: при наступлении некоего соб... Aug 19 2008, 10:19 aaarrr Цитата(MMos @ Aug 19 2008, 14:19) Я суть ... Aug 19 2008, 10:25  MMos Цитата(aaarrr @ Aug 19 2008, 12:25) Не по... Aug 19 2008, 10:36 aaarrr Я тоже много лет программирую на ассемблерах (если... Aug 19 2008, 10:53 MMos Язык, на котором пишет программист, для меня не яв... Aug 19 2008, 11:16  Herz Цитата(MMos @ Aug 19 2008, 13:16) Язык, н... Aug 19 2008, 12:02   zltigo Цитата(Herz @ Aug 19 2008, 14:02) А что, ... Aug 19 2008, 13:34 SysRq Как вариант, функцию loop можно сделать полностью ... Aug 19 2008, 22:19 Dog Pawlowa Цитата(SysRq @ Aug 20 2008, 01:19) Как ва... Aug 20 2008, 05:54
2 чел. читают эту тему (гостей: 2, скрытых пользователей: 0)
Пользователей: 0
|
|
|