Cours:Atmega328p : Différence entre versions

De troyesGEII
Aller à : navigation, rechercher
m ({{Bleu|Fonctions d'interruption}})
m ({{Vert|Exemple}})
Ligne 102 : Ligne 102 :
 
ISR(INT0_vect)    // programme d'interruption : le programme principal est interrompu,
 
ISR(INT0_vect)    // programme d'interruption : le programme principal est interrompu,
 
{                // l'interruption exécutée et ensuite le programme principal continu normalement son exécution
 
{                // l'interruption exécutée et ensuite le programme principal continu normalement son exécution
   PORTB^=1<<PB0;  // modification de la sortie PB0
+
   PORTB ^= 1<<PB0;  // modification de la sortie PB0
 
}  
 
}  
  
 
int main()
 
int main()
 
{
 
{
   DDRB=1<<PB0;    // configuration de PB0 en sortie
+
   DDRB = 1<<PB0;    // configuration de PB0 en sortie
 
   cli();          // arrêt des interruptions
 
   cli();          // arrêt des interruptions
   EICRA=1<<ISC00; // mode de déclenchement de l'interruption
+
   EICRA = 1<<ISC00; // mode de déclenchement de l'interruption
   EIMSK=1<<INT0;  // choix des interruptions actives
+
   EIMSK = 1<<INT0;  // choix des interruptions actives
 
   sei();          // autorisation des interruptions
 
   sei();          // autorisation des interruptions
 
   while(1)
 
   while(1)

Version du 21 octobre 2016 à 19:15

Interruptions

On rappelle qu'une interruption peut être imaginée comme un sous-programme appelé par un événement (et non pas par un autre programme). Les événements dont il s'agit sont en général des positionnements de bits spéciaux qui sont appelés drapeaux (on gardera souvent la terminologie anglaise de "flag"). Ces flags sont toujours positionnés à un par le matériel et non par le logiciel (ou par le programmeur si vous voulez). Ainsi le programmeur n'a absolument aucun moyen de positionner un flag à 1. Même l'utilisation de masques et d'opérateurs OU associé ne produira pas le résultat désiré.

Activer/Désactiver les interruptions

Les fonctions suivantes permettent d'activer et désactiver globalement les interruptions :

cli(); //désactive toute interruption
sei(); //autorise les interruptions

Attention

Quand une interruption est appelée, les autres interruptions sont automatiquement désactivées par le µcontrôleur

Les variables partagées entre interruption et programme principal doivent être de type volatile

Les opérations de lecture/écriture sur des variables de plus de 8 bits doivent être protégées en désactivant les interruptions.

Exemple :

unsigned long c;
uint8_t SaveSREG = SREG;
cli();
c = PulseCounts;
SREG = SaveSREG;
sei();

Interruption externe

Cette partie détaille l'utilisation des interruptions INT0 et INT1, attachées aux pin PD2 et PD3.

Registre EICRA

Le registre EICRA permet de choisir le mode de déclenchement de l'interruption.

EICRA bit 7 6 5 4 3 2 1 0
Fonction ----- ----- ----- ----- ISC11 ISC10 ISC01 ISC00
Valeur initiale 0 0 0 0 0 0 0 0

Le tableau suivant donne la valeur des bits ISCx0 et ISCx1 pour configurer le mode de déclenchement associé à l'interruption INTx :

ISCx1 ISCx0 Déclenchement de l'interruption sur :
0 0 Un niveau bas sur l'entrée INTx
0 1 Un changement d'état sur l'entrée INTx
1 0 Un front descendant sur l'entrée INTx
1 1 Un front montant sur l'entrée INTx

Registre EIMSK

Le registre EIMSK permet d'autoriser ou non les interruptions INT1 et INT0.

EIMSK bit 7 6 5 4 3 2 1 0
Fonction ----- ----- ----- ----- ----- ----- INT1 INT0
Valeur initiale 0 0 0 0 0 0 0 0

Une mise à '1' du bit INTx permet d'autoriser l'interruption associée.

Registre EIFR Exernal Interrupt Flag Register

Le registre EIFR permet d'observer l'état des interruptions INT1 et INT0.

EIFR bit 7 6 5 4 3 2 1 0
Fonction ----- ----- ----- ----- ----- ----- INTF1 INTF0
Valeur initiale 0 0 0 0 0 0 0 0

Le bit INTFx passe à '1' lors du déclenchement de l'interruption.

Exemple

Pour déclencher une interruption à chaque changement d'état de la patte PD2 (donc sur les fronts montant et descendant), on pourra utiliser le code suivant :

ISR(INT0_vect)    // programme d'interruption : le programme principal est interrompu,
{                 // l'interruption exécutée et ensuite le programme principal continu normalement son exécution
  PORTB ^= 1<<PB0;  // modification de la sortie PB0
} 

int main()
{
  DDRB = 1<<PB0;    // configuration de PB0 en sortie
  cli();          // arrêt des interruptions
  EICRA = 1<<ISC00; // mode de déclenchement de l'interruption
  EIMSK = 1<<INT0;  // choix des interruptions actives
  sei();          // autorisation des interruptions
  while(1)
  {
  }  
}

Pour la première fois vous voyez apparaître un sous-programme particulier ISR(...). C'est lui qui va être appelé quand l'événement qui déclenche l'interruption correspondante arrive. Ce qui apparaît dans les parenthèses dépend de l'événement : ici INT0_vect est une constante associée à l'interruption sur un changement sur la broche PD2. On vous donnera toujours la constante associée à un événement.

Timer

Registre TIMSKx

Le registre TIMSKx permet d'autoriser les différents modes d'interruptions associés aux Timers. Le registre se décompose comme suit :

TIMSKx bit 7 6 5 4 3 2 1 0
Fonction ----- ----- ICIEx ----- ----- OCIExB OCIExA TOIEx

Le bit TOIEx autorise l'interruption de débordement associé au TIMERx.

ISR(TIMER0_OVF_vect)
{
  ...
}

...
TIMSK0 |= 1 << TOIE1;
sei();
...

Les 2 bits OCIEx(A/B) autorise une interruption lorsque le timer atteint la valeur de comparaison OCRx(A/B).

ISR(TIMER2_COMPA_vect)
{
  ...
}

...
TIMSK2 |= 1 << OCIE2A;
sei();
...

Fonctions d'interruption

Le tableau donne la liste des interruptions et le vecteur d'interruption associé :

Nom de l'interruption Description Fonction
INT0 Changement d'état de la patte PD2 ISR(INT0_vect)
INT1 Changement d'état de la patte PD3 ISR(INT1_vect)
TIMERx_OVF Débordement du Timer(0/1/2) ex : ISR(TIMER2_OVF_vect)
TIMERx_COMPY Comparaison Timer(0/1/2) avec OCRxY ex : ISR(TIMER2_COMPA_vect)