« Utiliser les PIC 16F et 18F/Exercices/Interruption timer0 en langage C » : différence entre les versions

Une page de Wikiversité, la communauté pédagogique libre.
Contenu supprimé Contenu ajouté
m Robot : Changement de type cosmétique
m Robot : Remplacement de texte automatisé (-(\d{1,})\s?MHz +{{Unité|\1|{{abréviation|Mhz|mégahertz}}}})
Ligne 2 : Ligne 2 :
Si vous voulez savoir comment est réalisé pratiquement ce qui est décrit dans cet exercice, visitez le chapitre [[Very High Speed Integrated Circuit Hardware Description Language#Embarquer un PIC 16F84|Embarquer un PIC 16F84]] d'un autre livre.
Si vous voulez savoir comment est réalisé pratiquement ce qui est décrit dans cet exercice, visitez le chapitre [[Very High Speed Integrated Circuit Hardware Description Language#Embarquer un PIC 16F84|Embarquer un PIC 16F84]] d'un autre livre.


Un [[w:PIC_16F84_de_Microchip|PIC16F84]] est enfoui dans un [[w:FPGA|FPGA]]. Sa seule particularité est de fonctionner à 50MHz contre 10 (resp. 20 MHz) de fréquence maximale d'horloge pour les PIC 16F84 (resp. 16F84A). Il exécute le programme suivant (écrit avec le compilateur Hitech C) :
Un [[w:PIC_16F84_de_Microchip|PIC16F84]] est enfoui dans un [[w:FPGA|FPGA]]. Sa seule particularité est de fonctionner à {{Unité|50|{{abréviation|Mhz|mégahertz}}}} contre 10 (resp. {{Unité|20|{{abréviation|Mhz|mégahertz}}}}) de fréquence maximale d'horloge pour les PIC 16F84 (resp. 16F84A). Il exécute le programme suivant (écrit avec le compilateur Hitech C) :
<source lang="c">
<source lang="c">
//#include <pic1684.h>
//#include <pic1684.h>
Ligne 43 : Ligne 43 :
Les deux questions suivantes sont issues du Devoir Surveillé de juin 2010.
Les deux questions suivantes sont issues du Devoir Surveillé de juin 2010.


5°) Le programme suivant est donné comme exemple du compilateur MikroC et tourne dans un PIC 16F84 qui a un quartz de 4 MHz.
5°) Le programme suivant est donné comme exemple du compilateur MikroC et tourne dans un PIC 16F84 qui a un quartz de {{Unité|4|{{abréviation|Mhz|mégahertz}}}}.
<source lang="c">
<source lang="c">
unsigned cnt;
unsigned cnt;
Ligne 107 : Ligne 107 :
N'oubliez pas la division par 16 qui est réalisée avec le if (!(nb % 16)) dans l'interruption.
N'oubliez pas la division par 16 qui est réalisée avec le if (!(nb % 16)) dans l'interruption.


<u>Calcul précis</u> : 50 MHz / 4*(256*256*16) = 11,92 Hz (la division par 4 a toujours lieu).
<u>Calcul précis</u> : {{Unité|50|{{abréviation|Mhz|mégahertz}}}} / 4*(256*256*16) = 11,92 Hz (la division par 4 a toujours lieu).


3°) "if (!(nb % 16))" est une façon pas très efficace de calculer le reste de la division par 16. J'ignore la technique utilisé par le compilateur, mais ce calcul est forcément long puisqu'il n'y a pas d'instruction de division sur le 16F84.
3°) "if (!(nb % 16))" est une façon pas très efficace de calculer le reste de la division par 16. J'ignore la technique utilisé par le compilateur, mais ce calcul est forcément long puisqu'il n'y a pas d'instruction de division sur le 16F84.
Ligne 171 : Ligne 171 :
unsigned char SEGMENT[] = {0x3F,....
unsigned char SEGMENT[] = {0x3F,....
</source>
</source>
3°) Réaliser le programme main() responsable de l'initialisation de l'interruption qui doit avoir lieu toutes les 10ms (avec un quartz de 4MHz) et qui compte de 00 à 99 toutes les secondes environ (avec un "Delay_ms(1000);")
3°) Réaliser le programme main() responsable de l'initialisation de l'interruption qui doit avoir lieu toutes les 10ms (avec un quartz de {{Unité|4|{{abréviation|Mhz|mégahertz}}}}) et qui compte de 00 à 99 toutes les secondes environ (avec un "Delay_ms(1000);")


4°) Réaliser enfin l'interruption qui affichera tantôt les dizaines, tantôt les unités.
4°) Réaliser enfin l'interruption qui affichera tantôt les dizaines, tantôt les unités.
Ligne 211 : Ligne 211 :
Les compilateurs C compilent cela certainement de manière peu optimisée. Il faut savoir qu'il existe un bit "half carry" dans le registre '''Status''' qui permet certainement d'optimiser, mais nous laissons tomber ce genre de détail.
Les compilateurs C compilent cela certainement de manière peu optimisée. Il faut savoir qu'il existe un bit "half carry" dans le registre '''Status''' qui permet certainement d'optimiser, mais nous laissons tomber ce genre de détail.


Pour la période de l'interruption, on part de la fréquence quartz divisée par 4 soit : 1 MHz qu'il faut diviser par {{formatnum:10000}} pour avoir une période d'overflow de 10 ms. Le timer gère une division par 256 qu'on peut ramener à 250 en l'initialisant à 6 mais il nous reste à réaliser une division par 40 qui n'est pas une puissance de 2. On va prendre une division par 64 et 10000 / 64 = 156,25 donc notre timer0 sera initialisé à 256-156 = 100.
Pour la période de l'interruption, on part de la fréquence quartz divisée par 4 soit : {{Unité|1|{{abréviation|Mhz|mégahertz}}}} qu'il faut diviser par {{formatnum:10000}} pour avoir une période d'overflow de 10 ms. Le timer gère une division par 256 qu'on peut ramener à 250 en l'initialisant à 6 mais il nous reste à réaliser une division par 40 qui n'est pas une puissance de 2. On va prendre une division par 64 et 10000 / 64 = 156,25 donc notre timer0 sera initialisé à 256-156 = 100.
<source lang="c">
<source lang="c">
//****** Mikro C ********
//****** Mikro C ********

Version du 2 janvier 2012 à 14:24

Exercice 1 (Hitech C et Mikro C)

Si vous voulez savoir comment est réalisé pratiquement ce qui est décrit dans cet exercice, visitez le chapitre Embarquer un PIC 16F84 d'un autre livre.

Un PIC16F84 est enfoui dans un FPGA. Sa seule particularité est de fonctionner à 50 Mhz contre 10 (resp. 20 Mhz) de fréquence maximale d'horloge pour les PIC 16F84 (resp. 16F84A). Il exécute le programme suivant (écrit avec le compilateur Hitech C) :

//#include <pic1684.h>
#include <htc.h> //A changer si autre compilateur
void interrupt decalage(void); //A changer si autre compilateur
unsigned char nb; 
main(void) { 
  TRISA = 0xF9; // 6 entrees, 2 sorties pour A 
  TRISB = 0x00; // 8 sorties pour B 
  OPTION = 0x07; // prescaler 256 , entree sur quartz (A changer si autre compilateur : OPTION_REG)
  INTCON = 0xA0;  // autorise l'interruption timer 
  PORTB = 0x01; // une seule diode allumee		 
  TMR0 = 0x00 ; 
  nb=0; 
  while(1) { 
   // on ne fait rien que recopier sur 2 segments la valeur de SW1 
      if ((PORTA & 0x01) == 1) PORTA = 0x06; 
    } 
} 

void interrupt decalage(void) { //A changer si autre compilateur
  nb++; 
  //TMR0 = 0x00; //c'est fait car ici par overflow
  if (!(nb % 16)) 
  PORTB = (PORTB << 1) ; 
  if (PORTB == 0x00)   PORTB = 0x01; 
  T0IF = 0;  // acquittement interruption 
}

Remarquez comment est écrit une interruption avec ce compilateur.

1°) Repérer et modifier les lignes de ce programmes pour qu'il fonctionne avec le compilateur MikroC.

2°) Calculer si le chenillard réalisé par ce programme est visible à l'œil humain (fréquence de changement de position des LEDs inférieure à 20 Hz).

3°) Comment peut-on écrire l'instruction "if (!(nb % 16))" pour plus d'efficacité.

4°) Quelle est la suite des états (LEDs allumées) réalisée par ce programme.

Les deux questions suivantes sont issues du Devoir Surveillé de juin 2010.

5°) Le programme suivant est donné comme exemple du compilateur MikroC et tourne dans un PIC 16F84 qui a un quartz de 4 Mhz.

unsigned cnt; 
void interrupt() { 
  if (TMR0IF_bit) { 
    cnt++;                 // increment counter 
    TMR0IF_bit = 0;        // clear TMR0IF 
    TMR0   = 96; 
  } 
} 

void main() { 
  OPTION_REG = 0x84;       // Assign prescaler to TMR0 
  ANSEL  = 0;              // Configure AN pins as digital 
  ANSELH = 0; 
  C1ON_bit = 0;            // Disable comparators 
  C2ON_bit = 0; 
  TRISB = 0;               // PORTB is output 
  PORTB = 0xFF;            // Initialize PORTB 
  TMR0  = 96;              // Timer0 initial value 
  INTCON = 0xA0;           // Enable TMRO interrupt 
  cnt = 0;                 // Initialize cnt 

  do { 
    if (cnt >= 400) { 
      PORTB = ~PORTB;      // Toggle PORTB LEDs 
      cnt = 0;             // Reset cnt 
    } 
  } while(1); 
}

Quelle est la fréquence de clignotement des LEDs reliées au PORTB ?

6°) Modifier l'interruption (du programme de départ) pour qu'elle réalise un chenillard d'une LED se déplaçant vers les poids faibles.

Exercice 2

Une partie matérielle est constituée de deux afficheurs sept segments multiplexés. Les sept segments sont commandés par le PORTC, tandis que les commandes d'affichages sont réalisée par les bits b0 et b1 du PORTB. Un schéma de principe est donné ci-après.

Comment utiliser deux afficheurs multiplexés

1°) A l'aide de la documentation calculer les valeurs dans un tableau "unsigned char SEGMENT[] = {0x3F,...};" pour un affichage des chiffres de 0 à 9.

2°) réaliser une fonction responsable du transcodage :

unsigned char Display(unsigned char no) { 
    unsigned char Pattern; 
    unsigned char SEGMENT[] = {0x3F,....

3°) Réaliser le programme main() responsable de l'initialisation de l'interruption qui doit avoir lieu toutes les 10ms (avec un quartz de 4 Mhz) et qui compte de 00 à 99 toutes les secondes environ (avec un "Delay_ms(1000);")

4°) Réaliser enfin l'interruption qui affichera tantôt les dizaines, tantôt les unités.