Micro contrôleurs AVR/Le Timer 0
Présentation du timer 0
modifierLe timer 0 est un compteur sur 8 bits qui peut être commandé par l'horloge du processeur (quartz) ou par un bit externe. Nous allons présenter de manière graphique le fonctionnement du timer 0.
Documentation pour l'ATMega8
modifierVoici présenté de manière schématique le fonctionnement du timer0.
Les conventions de cette figure ne sont pas universelles et ne correspondent pas à la documentation officielle. Donc pas de problème de copyright.
Le cœur du timer 0 est un compteur sur 8 bits (c'est précisé dans le dessin) en haut à gauche de la figure appelé TCNT0. Comme tout compteur il nécessite une horloge. Celle-ci peut être
- soit une entrée externe (bit b4 du PORTD couramment appelé T0,
- soit l'horloge du processeur (quartz externe ou horloge interne) qui est traitée par un diviseur. Ce diviseur est appelé prescaler en anglais et on le nommera ainsi dans la suite.
Un bit d'un registre TIFR est appelé TOV0 (Timer overflow 0 = débordement du timer 0). Ce bit est automatiquement positionné à 1 lors du débordement du timer 0, c'est-à-dire pour un passage de 0xFF à 0x00.
Le registre TCCR0 gère le prescaler à l'aide de trois bits CS02, CS01, CS00. La suite de ce qui est marqué dans le dessin correspond à la suite binaire sur ces trois bits. Par exemple vous en déduisez qu'une division par 1024 est réalisée par le nombre binaire 101...
Documentation pour l'ATMega328
modifierDepuis la version de l'ATMega8, le timer 0 même s'il est resté sur 8 bits s'est progressivement étoffé de fonctionnalités additionnelles.
Une comparaison rapide entre les deux documentations vous montre que le registre TIFR de l'ATMega8 s'est transformé en registre TIFR0 et que le registre TCCR0 s'est transformé en TCCR0B. Il y a eu d'autres modifications qui seront présentées au fur et à mesure.
Mesure du temps d'exécution d'un algorithme
modifierL'optimisation d'un algorithme en vitesse (ou en taille) est très importante dans les systèmes embarqués réalisés par des micro-contrôleurs. Une recherche d'algorithmes sur Internet vous donnera des résultats qu’il vous faudra évaluer. Par exemple, le site : convert base vous propose un algorithme de division par 10 que voici :
unsigned int A;
unsigned int Q; /* the quotient */
Q = ((A >> 1) + A) >> 1; /* Q = A*0.11 */
Q = ((Q >> 4) + Q) ; /* Q = A*0.110011 */
Q = ((Q >> 8) + Q) >> 3; /* Q = A*0.00011001100110011 */
/* either Q = A/10 or Q+1 = A/10 for all A < 534,890 */
Exercice 1
modifier1°) Sans chercher à comprendre l'algorithme de division, on vous demande de le transformer en une fonction unsigned int div10(unsigned int A);
2°) Les LEDs d'un shield maison sont couplés à l'arduino MEGA2560. Une lecture de son schéma fait apparaître la correspondance :
Numéro | f5 | f4 | f3 | f2 | f1 | f0 | p1 | p0 |
---|---|---|---|---|---|---|---|---|
Couleur | r | o | v | r | o | v | v | r |
Arduino Pin | 13 | 12 | 11 | 10 | 9 | 8 | 7 | 6 |
Port Arduino UNO | PB5 | PB4 | PB3 | PB2 | PB1 | PB0 | PD7 | PD6 |
Port Arduino LEONARDO | PC7 | PD6 | PB7 | PB6 | PB5 | PB4 | PE6 | PD7 |
Port Arduino MEGA2560 | PB7 | PB6 | PB5 | PB4 | PH6 | PH5 | PH4 | PH3 |
Écrire un sous-programme capable d'afficher un nombre sur 8 bits sur les LEDs
3°) Écrire un programme complet qui mesure le temps d'exécution du sous programme de division par 10, puis modifier le programme pour qu’il puisse comparer avec une division par 10 normale.
1°)
unsigned int div10(unsigned int A){
unsigned int Q; /* the quotient */
Q = ((A >> 1) + A) >> 1; /* Q = A*0.11 */
Q = ((Q >> 4) + Q) ; /* Q = A*0.110011 */
Q = ((Q >> 8) + Q) >> 3; /* Q = A*0.00011001100110011 */
/* either Q = A/10 or Q+1 = A/10 for all A < 534,890 */
return Q; // ne pas oublier le return pour une fonction
}
2°)
void afficheLeds(unsigned char ch){
unsigned char ch_partie;
ch_partie = (ch << 3) & 0x78;
PORTH = ch_partie;
ch_partie = ch & 0xF0;
PORTB = ch_partie;
}
3°)
//***** Compilateur avr-gcc ************
//**** OK : Arduino MEGA2560 + shield
#include <avr/io.h>
#undef F_CPU
#define F_CPU 16000000UL
#include <util/delay.h>
unsigned int div10(unsigned int A);
void afficheLeds(unsigned char ch);
int main(void){
unsigned int res,i;
unsigned char temps;
DDRH = 0x78; // 4 sorties pour H
DDRB = 0xF0; // 4 sorties pour B
// initialisation du timer avec prescaler 256 ,
TCCR0B |= _BV(CS02);
TCCR0B &= ~(_BV(CS01) | _BV(CS00));
while(1) {
// départ de la mesure
// while ((TIFR & (0x01<<TOV0)) == 0);
TCNT0 = 0x00;
// algorithme de calcul ici plusieurs fois
for(i=0;i<55;i++) {
res=div10(i+15558);
PORTC = res; // autrement simplifié
}
// fin de la mesure : lecture du timer
temps=TCNT0;
afficheLeds(temps);
//afficheLeds(0x55);
_delay_ms(500);
} // while(1)
return 0;
}
unsigned int div10(unsigned int A){
unsigned int Q; /* the quotient */
Q = ((A >> 1) + A) >> 1; /* Q = A*0.11 */
Q = ((Q >> 4) + Q) ; /* Q = A*0.110011 */
Q = ((Q >> 8) + Q) >> 3; /* Q = A*0.00011001100110011 */
/* either Q = A/10 or Q+1 = A/10 for all A < 534,890 */
return Q; // ne pas oublier le return pour une fonction
}
void afficheLeds(unsigned char ch){
unsigned char ch_partie;
ch_partie = (ch << 3) & 0x78;
PORTH = ch_partie;
ch_partie = ch & 0xF0;
PORTB = ch_partie;
}
L'ajout dans la boucle de "PORTC = res;" a été essentielle pour éviter la suppression de la boucle avec juste un warning comme quoi res n'était jamais utilisé ! C'est normal c’est TCNTO qui m'intéresse pas res !
Pour comparer avec l'algorithme classique, remplacer
// algorithme de calcul ici plusieurs fois
for(i=0;i<50;i++) {
res=div10(i+15558);
PORTC = res; // autrement simplifié
}
par
// algorithme de calcul ici plusieurs fois
for(i=0;i<50;i++) {
res=(i+15558)/10; // division classique
PORTC = res; // autrement simplifié
}
Le premier algorithme avec div10 me donne 16 ou 17.
Le deuxième algorithme avec /10 me donne 45 ou 46.
Ceci est conforme à ce que j'attendais contrairement à ce qui se passait avec le PIC et mikroC.
Le mode de scrutation du flag
modifierNous devons savoir à ce niveau, que tout débordement du timer0 (passage de 0xFF à 0x00) entraîne le positionnement du flag TOV0, bit b0 du registre TIFR0. Vous pouvez donc utiliser ce flag pour déterminer si vous avez eu débordement du timer0, ou, en d’autres termes, si le temps programmé est écoulé. Cette méthode à l’inconvénient de vous faire perdre du temps inutilement dans une boucle d'attente.
Petit exemple d'attente passive :
while ((TIFR0 & 0x01) == 0); //attente passive
ou encore pour plus de lisibilité sur les bits manipulés :
while ((TIFR0 & 1<<T0V0) == 0); //attente passive
Le drapeau TOV0 ne se remet pas tout seul à 0... et il faut écrire un 1 pour le remettre à 0 !!! |
Exercice 2
modifierLe quartz est choisi à 4MHz dans ce problème.
Question 1
modifierOn donne le programme suivant concernant le timer 0 :
int main(void){
// initialisation du timer division par 8
TCCR0B = 0x02; // prescaler 8 , entrée sur quartz
TCNT0 = 0x00; // tmr0 : début du comptage dans 2 cycles
// bit RB0 du PORTB en sortie
DDRB |= 0x01; //RB0 as output
while(1) {
while ((TIFR & (1<<TOV0)) == 0);
// ce qui est fait ici est fait tous les 256 comptages de TCNT0
PORTB ^= 0x01; // on bascule avec ou exclusif
TIFR0 |= 0x01; // clr TOV0 with 1 : obligatoire !!!
// TIFR0 |= (1<<TOV0); // fait le même chose
}
return 0;
}
Pouvez-vous donner la fréquence d'oscillation du bit b0 du PORTB avec quelques explications ?
(4 000 000) / (8 * 256 * 2) = 977 Hz environ
Le facteur 2 apparaissant dans cette formule est lié au fait que "PORTB ^= 0x01;" fait basculer le bit et qu'ainsi il faut deux basculements pour faire une période. Il est bon de retenir que toute technique de basculement nécessite la création d'une fréquence double. Ainsi pour créer 1 kHz il vous faudra 2 kHz ...
Question 2
modifierÉcrire en langage C un programme qui fait la même chose que le programme ci-dessus : initialise le timer0, efface le flag et attend à l'aide d'une boucle le positionnement de ce dernier mais 100 incrémentations seulement. On vous demande aussi d’utiliser au mieux les déclarations du fichier d'inclusion avr/io.h :
/* TIFR0 */
#define OCF2 7
#define TOV2 6
#define ICF1 5
#define OCF1A 4
#define OCF1B 3
#define TOV1 2
/* bit 1 reserved (OCF0?) */
#define TOV0 0
//***** plein de lignes ici mais cachées *****
/* TCCR0 */
/* bits 7-3 reserved */
#define CS02 2
#define CS01 1
#define CS00 0
L'utilisation des valeurs prédéfinies permet de rendre vos programmes un peu plus lisibles.
#include <avr/io.h>
void main(void){
// initialisation du timer division par 8
TCCR0B |= _BV(CS01); // prescaler 8 , entrée sur quartz
TCCR0B &= ~(_BV(CS02) | _BV(CS00)); // pour être sûr
TCNT0 = 255-99; // tmr0 : début du comptage dans 2 cycles
// bit RB0 du PORTB en sortie
DDRB |= 0x01; //RB0 as output
while(1) {
TIFR0 |= _BV(TOV0); // clr TOV0 with 1
while (TIFR0 & _BV(TOV0)) == 0);
// ce qui est fait ici est fait tous les 256 comptages de TCNT0
PORTB ^= _BV(0); // on bascule avec ou exclusif
//TIFR0 &= ~_BV(TOV0); // reset the overflow flag
TCNT0 = 255-99; // remplace le reset de l'overflow
}
}
Question 3
modifierGénérer un signal de fréquence 1 kHz. Pour cela :
- calculer la valeur de la pré division
- calculer la valeur de décomptage
- Écrire le programme.
On va en fait générer 2kHz et changer sans arrêt un bit B0 du PORTB. 4 MHz / 2kHz=2000. Si l’on divise par 8 on peut rester en 8 bits avec comme valeur 250 à compter donc une initialisation à 256-250=6. Ceci n'est qu'une valeur théorique qui dépend du compilateur. Il faut donc faire des essais pour trouver la valeur exacte. Une autre technique pour être quasi indépendant du compilateur c’est de remplacer :
TCNT0 = 6;
par
TCNT0 = TCNT0+6;
qui dépend moins du temps que l’on a mis pour réaliser les instructions.
#include <avr/io.h>
void main(void){
// initialisation du timer division par 8
TCCR0B |= _BV(CS01); // prescaler 8 , entrée sur quartz
TCCR0B &= ~(_BV(CS02) | _BV(CS00)); // pour être sûr
TCNT0 = 6; // tmr0 : début du comptage dans 2 cycles
// bit RB0 du PORTB en sortie
DDRB |= 0x01; //RB0 as output
while(1) {
TIFR0 |= _BV(TOV0); // reset the overflow flag
while (TIFR0 & _BV(TOV0)) == 0);
TCNT0 += 6; // remplace le reset de l'overflow
// ce qui est fait ici est fait tous les 250 comptages de TCNT0
PORTB ^= _BV(0); // on bascule avec ou exclusif
}
}
Question 4
modifierGénérer un signal de sortie de rapport cyclique 1/4 sur le même principe.
La solution de la question 3 nous a montré qu’il fallait compter 250 pour sortie à 0 et 250 pour sortie à 1. Ici il faut donc compter 125 pour la valeur 1 et 250+125=375 pour la valeur 0. J'espère que tout le monde a vu le problème ! 375 ne tient plus dans 8 bits. Il faut donc encore diviser l'horloge par 2. Mais cela n’est pas possible : on ne peut que diviser par 8 pour avoir le prescaler à 64.
Cela nous donne un comptage de 125/8 = 16 (environ) 375/8 = 47 (environ).
void main(void){
// initialisation du timer division par 64
TCCR0B |= (_BV(CS01) | _BV(CS00)); // prescaler 64 , entrée sur quartz
TCCR0B |= ~_BV(CS02); // pour être sûr
//TCNT0 = 6; // tmr0 : début du comptage dans 2 cycles
// bit RB0 du PORTB en sortie
DDRB |= 0x01; //RB0 as output
PORTB |= _BV(0);// pour être sûr de la façon dont on démarre
while(1) {
//TIFR0 &= ~_BV(TOV0); // reset the overflow flag
TCNT0 += 240; // 256-16 : remplace le reset de l'overflow
while (TIFR0 & _BV(TOV0)) == 0);
// ce qui est fait ici est fait tous les 256 comptages de TCNT0
PORTB ^= _BV(0); // on bascule avec ou exclusif
//TIFR0 &= ~_BV(TOV0); // reset the overflow flag
TCNT0 += 209; // 256-47remplace le reset de l'overflow
while (TIFR0 & _BV(TOV0)) == 0);
PORTB ^= _BV(0); // on bascule avec ou exclusif
}
}
Nous testerons ce programme (un jour je n'ai pas de scope sous la main chez moi) sur une Arduino MEGA2560 avec une horloge à 16MHz. Si l’on veut rester à 1kHz il nous faut utiliser le programme suivant :
#include <avr/io.h>
void main(void){
// initialisation du timer division par 64
TCCR0B |= (_BV(CS01) | _BV(CS00)); // prescaler 64 , entrée sur quartz
TCCR0B |= ~_BV(CS02); // pour être sûr
//TCNT0 = 6; // tmr0 : début du comptage dans 2 cycles
// bit RB0 du PORTB en sortie
DDRB |= 0x01; //RB0 as output
PORTB |= _BV(0);// pour être sûr de la façon dont on démarre
while(1) {
//TIFR0 &= ~_BV(TOV0); // reset the overflow flag
TCNT0 += 192; // 256-(16*4) : remplace le reset de l'overflow
while (TIFR0 & _BV(TOV0)) == 0);
// ce qui est fait ici est fait tous les 256 comptages de TCNT0
PORTB ^= _BV(0); // on bascule avec ou exclusif
//TIFR0 &= ~_BV(TOV0); // reset the overflow flag
TCNT0 += 68; // 256-(47*4)remplace le reset de l'overflow
while (TIFR0 & _BV(TOV0)) == 0);
PORTB ^= _BV(0); // on bascule avec ou exclusif
}
}
Il y a mieux à faire avec les AVRs, utiliser le module CCP (détaillé plus loin).
Exercice 3
modifierMettre en œuvre le timer0 sur le MEGA2560 avec un prescaler à 1024 et une attente active du bit T0V0 du registre TIFR0. On basculera le bit b4 du PORTB qui est relié à une LED. Quelle est la fréquence de basculement du bit b4 si la fréquence du quartz est 16 MHz ?
Solution de l'exercice 3
modifierLe nom des registres est un peu changé par rapport à un ATMega8.
//*** testé OK sur Arduino MEGA2560 + shield
#include <avr/io.h>
int main(void){
// initialisation du timer division par 1024
TCCR0B = 0x05; // prescaler 1024 , entrée sur quartz
TCNT0 = 0x00; // tmr0 : début du comptage dans 2 cycles
// bit RB4 du PORTB en sortie
DDRB |= 0x10; //RB4 as output
while(1) {
TIFR0 |= 0x01; // clr TOV0 with 1
while ((TIFR0 & (1<<TOV0)) == 0); //attente passive
PORTB ^= 0x10; // on bascule avec ou exclusif
}
return 0;
}
Fréquence 16 000 000 /(1024*256*2) = 30,51 Hz
Timer 0 et interruption
modifierDocumentation
modifierLa mise à zéro du bit TOV0 est complètement automatique dans les ATMega contrairement aux autres architectures connues par nous. Ceci doit être certainement lié au fait que les interruptions sont vectorisées et qu'ainsi il est possible pour le processeur de savoir à tout moment l'événement qui a déclenché l'interruption.
Pour comprendre cette figure, il suffit de se rappeler qu'un front montant dans l'ellipse rouge réalisera cette-interruption. Ce front montant est réalisé par la mise à un, par le matériel, de TOV0. En C cette interruption est désignée par "TIMER0_OVF_vect". Si vous avez compris que ce n’est pas le logiciel qui positionnera le bit TOV0 mais le matériel, alors vous déduisez que pour réaliser une interruption il suffit de
- mettre à 1 le bit TOIE0 du registre TIMSK pour l'ATMega8. Pour l'ATMega328 ce registre s’appelle TIMSK0.
- mettre à 1 le bit I du registre SREG. Ceci se réalise par l'instruction "sei();" en C et "interrupts();" avec l’Arduino.
Le registre TIFR de l'ATMega8 a été rebaptisé TIFR0 pour l'ATMega328.
Mise en œuvre
modifierSi la fréquence de la platine MEGA2560 est de 16 MHz et que l’on utilise un préscalaire de 1024 et un basculement nous avons déjà trouvé une fréquence de 32 Hz dans un exercice précédent. Ainsi une division par 32 doit donner 1 Hz. On le fait par exemple par interruption, toujours avec notre MEGA2560 :
//*** testé OK sur Arduino MEGA2560 + shield
#include <avr/io.h>
#include <avr/interrupt.h>
// compteur
volatile unsigned char cpt=0;
// Fonction de traitement Timer 0 OverFlow
ISR(TIMER0_OVF_vect){
cpt++;
if(cpt==31) {
PORTB ^=(1<<PB4);
cpt=0;
}
}
void main(){
// IT Timer0 Over Flow Active
TIMSK0=(1<<TOIE0);
// Prescaler 1024 (Clock/1024)
TCCR0B = (1<<CS02) | (1<<CS00);
//Configuration PORTB.4 en sortie
DDRB |= (1<<DDB4);
PORTB &= ~(1<<PB4); // PORTB.4 <-0
//activation des IT (SREG.7=1)
sei();
// SREG |= 0x80; // équivalent à sei()
while(1);
}
Le bit TOV0 est repositionné à 0 automatiquement (certainement par le retour d'interruption) ! Pour le réaliser par programme on peut mettre une valeur dans le timer0 (TCNT0) ou écrire un '1' dans le bit TOV0!
Exercice 4
modifierUn ATMega8 est enfoui dans un FPGA (Voir Embarquer un ATMega8 dans un FPGA). Sa seule particularité est de fonctionner à 25 MHz contre 20 MHz de fréquence maximale d'horloge pour celui du commerce. Il exécute le programme suivant écrit pour le compilateur gcc :
/*********************************************************************
Includes
***********************************************************************/
#include <avr/io.h>
#include <stdbool.h>
#include <avr/interrupt.h>
volatile unsigned char nb=0,vPORTB=1;
/*********************************************************************
Interrupt Routine
**********************************************************************/
// timer0 overflow
ISR(TIMER0_OVF_vect) {
nb++;
if (!(nb % 16))
vPORTB = (vPORTB << 1);
if (vPORTB == 0x00) vPORTB = 0x01;
PORTB = vPORTB;
}
/**********************************************************************
Main
**********************************************************************/
int main( void ) {
// Configure PORTB as output
DDRB = 0xFF;
PORTB = 0x01;
// enable timer overflow interrupt for both Timer0
TIMSK=(1<<TOIE0);
// set timer0 counter initial value to 0
TCNT0=0x00;
// start timer0 with 1024 prescaler
TCCR0 = (1<<CS02) | (1<<CS00);
// enable interrupts
sei();
while(true) { // grace a stdbool.h
}
return 0;
}
L'utilisation d'une variable vPORTB plutôt que directement PORTB est liée au matériel un peu particulier : l'ATMega8 enfoui n’est pas complet et une lecture de PORTB donnerait absolument n’importe quoi.
1°) 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).
2°) Comment peut-on écrire l'instruction "if (!(nb % 16))" pour plus d'efficacité.
3°) Quelle est la suite des états (LEDs allumées) réalisée par ce programme.
Pour les deux questions suivantes ce ne sera pas la routine d'interruption qui sera chargée de mettre le PORTB.
4°) Le programme suivant est donné et tourne dans un ATMega8 cadencé avec un quartz de 4 MHz.
#include <avr/interrupt.h>
volatile unsigned int cnt;
ISR(TIMER0_OVF_vect) {
cnt++; // increment counter
TCNT0 = 96;
}
}
int main() {
TCCR0 = (1<<CS01) | (1<<CS00); // Assign prescaler to TCNT0
DDRB = 0xFF; // PORTB is output
PORTB = 0xFF; // Initialize PORTB
TCNT0 = 96; // Timer0 initial value
TIMSK=(1<<TOIE0); // Enable TMRO interrupt
sei();
cnt = 0; // Initialize cnt
do {
if (cnt >= 400) {
PORTB = ~PORTB; // Toggle PORTB LEDs
cnt = 0; // Reset cnt
}
} while(1);
return 0;
}
Quelle est la fréquence de clignotement des LEDs reliées au PORTB ?
5°) Modifier le programme principal pour réaliser un chenillard d'une LED se déplaçant vers les poids faibles en gardant le traitement en dehors de l'interruption.
1°) N'oubliez pas la division par 16 qui est réalisée avec le if (!(nb % 16)) dans l'interruption.
Calcul précis : 25 Mhz / (1024*256*16) = 5,96 Hz.
2°) "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 l'ATMega8.
Comme la division se fait par 16 qui est une puissance de deux, on peut utiliser un masque pour faire ce calcul bien plus rapidement :
// langage C
if (!(nb & 0x0F)) // idem à if (!(nb % 16)) mais plus rapide
3°) Décalage tout simple d'une LED vers les poids forts. L'utilisation d'une variable "vPORTB" complique un peu la lecture du programme ! Ceci est lié au fait que dans le FPGA on ne peut pas lire PORTB (ce que l’on peut faire dans la vraie vie.
4°)
TCCR0 = (1<<CS01) | (1<<CS00);
met le prescaler à 64
4 000 000 / 2*64*(256-96)*400 = 0,488 281 25 Hz Soit pratiquement 0,5 Hz
5°)
#include <avr/io.h>
#include <avr/interrupt.h>
unsigned int cnt;
ISR(TIMER0_OVF_vect) {
cnt++; // increment counter
TCNT0 = 96;
}
}
int main() {
TCCR0 = (1<<CS01) | (1<<CS00); // Assign prescaler to TCNT0
DDRB = 0xFF; // PORTB is output
PORTB = 0x80; // Initialize PORTB
TCNT0 = 96; // Timer0 initial value
TIMSK=(1<<TOIE0); // Enable TMRO interrupt
sei();
cnt = 0; // Initialize cnt
do {
if (cnt >= 400) {
PORTB = PORTB>>1; // PORTB LEDs *** changé ici ****
if (PORTB == 0x00) PORTB = 0x80; // *** changé ici ****
cnt = 0; // Reset cnt
}
} while(1);
return 0;
}
Exercice 5
modifierUne 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.
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 4MHz) 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.
Exemple de réalisation à l'IUT de Troyes
modifierUn exemple de réalisation d'afficheur sept segments sur deux digits peut être trouvé ICI. Nous allons l’utiliser pour un exercice similaire à celui de la section précédente.
Description
modifierLes 2 afficheurs ne peuvent pas être utilisés simultanément. L'état de la sortie mux (arduino port 4 ou PD4) permet de sélectionner l'un ou l'autre. En allumant successivement l'un puis l'autre rapidement, on a l'illusion qu’ils sont tous 2 allumés.
Les segments des afficheurs sont câblés de façon analogue comme décrit ci-dessous :
Segment | pt | g | f | e | d | c | b | a |
---|---|---|---|---|---|---|---|---|
Arduino Pin | 11 | 9 | 10 | 8 | 7 | 6 | 12 | 13 |
Port UNO | PB3 | PB1 | PB2 | PB0 | PD7 | PD6 | PB4 | PB5 |
Voici sous forme schématique la documentation correspondante :
Programme d'utilisation en langage Arduino
modifierMême si l’Arduino possède son propre chapitre dans ce livre, nous vous proposons ici du code le concernant
const char pinMux = 4;
const char pinAff[8]={13,12,6,7,8,10,9,11};
void setup() {
char i;
for (i=0;i<8;i++) pinMode(pinAff[i],OUTPUT); // Déclaration des 8 sorties des afficheurs
pinMode(pinMux,OUTPUT); // + sortie de multiplexage (choix de l'afficheur)
}
void loop() {
char i,c;
for (i=0;i<8;i++) digitalWrite(pinAff[i],HIGH); // Les segments s'allument
for (c=0;c<20;c++) {
digitalWrite(pinMux,1); // sur l'afficheur 1
delay(10);
digitalWrite(pinMux,0); // puis sur l'afficheur 2
delay(10);
}
for (i=0;i<8;i++) digitalWrite(pinAff[i],LOW); // Les segments s'éteignent
for (c=0;c<20;c++) {
digitalWrite(pinMux,1); // sur l'afficheur 1
delay(10);
digitalWrite(pinMux,0); // puis sur l'afficheur 2
delay(10);
}
}
Programme d'utilisation en langage C
modifierUtiliser le langage C pour utiliser le Shield ci-dessus est pas simple. L'idée est de faire un tableau qui considère que tout est sur un PORT et de gérer l'affichage qui gère correctement dans un sous-programme.
Voici un programme de test pour un sous-programme d'affichage.
#include<util/delay.h>
// Pour Arduino UNO et le shield de l'{{abréviation|IUT|institut universitaire de technologie}} de Troyes
// ch sous la forme [pt|g |f |e |d |c |b |a ] doit réaliser
// PORTB = [- |- |a |b |DP |f |g |e ]
// PORTD = [d |c |- |- |- |- |- |- ]
// Trouver les décalages à partir des trois dessins de registres ci-dessus
void affiche7segs(unsigned char ch){
unsigned char port;
port = 0;
port = ((ch & 0x01)<<5) | ((ch & 0x02)<<3) | ((ch & 0x10)>>4) | ((ch & 0x20)>>3) | ((ch & 0x40)>>5);
PORTB = port;
port = 0;
port = ((ch & 0x0C)<<4);
PORTD = port;
}
int main(void){
unsigned char transcod[16]={0x3F,0x06,0x5B,0x4F,0x66,0x6D,0x7D,0x07,0x7F,0x6F,0x77,0x7C,0x39,0x5E,0x79,0x71},cmpt=0;
DDRD = 0xC0; // 2 sorties pour D
DDRB = 0x3F; // 6 sorties pour B
while(1) {
affiche7segs(transcod[cmpt]);
_delay_ms(1000);
cmpt++;
if (cmpt > 15) cmpt=0;
} // while(1)
return 0;
}
Il manque quelques fichiers d'inclusion, car on travaille directement avec l'environnement Arduino (sans setup() et sans loop() mais avec un main !
Pour utiliser les deux digits, le programme C suivant fonctionne correctement sans TIMER.
#include<util/delay.h>
// Pour Arduino UNO et le shield de l'{{abréviation|IUT|institut universitaire de technologie}} de Troyes
// ch sous la forme [pt|g |f |e |d |c |b |a ] doit réaliser
// PORTB = [- |- |a |b |DP |f |g |e ]
// PORTD = [d |c |- |- |- |- |- |- ]
// Trouver les décalages à partir des trois dessins de registres ci-dessus
void affiche7segs(unsigned char ch){
unsigned char port;
port = 0;
port = ((ch & 0x01)<<5) | ((ch & 0x02)<<3) | ((ch & 0x10)>>4) | ((ch & 0x20)>>3) | ((ch & 0x40)>>5);
PORTB = port;
port = 0;
port = ((ch & 0x0C)<<4);
PORTD = port;
}
void afficheDiz(unsigned char ch){
affiche7segs(ch);
PORTD &= ~(1<<PD4);
}
void afficheUnit(unsigned char ch){
affiche7segs(ch);
PORTD |= (1<<PD4);
}
int main(void){
unsigned char transcod[16]={0x3F,0x06,0x5B,0x4F,0x66,0x6D,0x7D,0x07,0x7F,0x6F,0x77,0x7C,0x39,0x5E,0x79,0x71},cmpt=0,i;
DDRD = 0xD0; // 2 sorties pour D
DDRB = 0x3F; // 6 sorties pour B
while(1) {
for (i=0;i<50;i++) {
afficheDiz(transcod[cmpt>>4]);
_delay_ms(10);
afficheUnit(transcod[cmpt&0x0F]);
_delay_ms(10);
}
cmpt++;
} // while(1)
return 0;
}
Exercice 6
modifierReprendre l'exercice 5 avec le shield présenté dans cette section.
1°) Pour compliquer un peu on utilisera le MEGA2560 pour lequel la correspondance entre les numéros Arduino et les PORTs est :
Arduino Pin | 13 | 12 | 11 | 10 | 9 | 8 | 7 | 6 |
---|---|---|---|---|---|---|---|---|
Port MEGA2560 | PB7 | PB6 | PB5 | PB4 | PH6 | PH5 | PH4 | PH3 |
Trouver la correspondance entre les segments a,b, c, d, e, f et DP et les PORTs du MEGA2560.
2°) Écrire un sous-programme capable de prendre un octet et de l'afficher avec la convention "a" poids faible.
3°) Écrire le programme qui réalise une interruption à 100 Hz et affiche tantôt sur l'afficheur des poids faibles tantôt sur l'afficheur des poids forts.
Solution exercice 6
modifierLa question 1°) n’est pas cachée car nous ne savons pas mettre un tableau dans le bandeau solution qui se déroule.
1°) Il vient assez facilement :
7 segments Pin | pt | g | f | e | d | c | b | a |
---|---|---|---|---|---|---|---|---|
Port MEGA2560 | PB5 | PH6 | PB4 | PH5 | PH4 | PH3 | PB6 | PB7 |
2°) Le code ci-dessous ne serait pas le même avec une carte Arduino UNO !
// Pour Arduino MEGA2560 et le shield de l'{{abréviation|IUT|institut universitaire de technologie}} de Troyes
// ch sous la forme [pt|g |f |e |d |c |b |a ] doit réaliser
// PORTB = [a |b |pt|f |- |- |- |- ]
// PORTH = [- |g |e |d |c |- |- |- ]
// Trouver les décalages à partir des trois dessins de registres ci-dessus
void affiche7segs(unsigned char ch){
unsigned char port;
port = 0;
port = ((ch & 0x01)<<7) | ((ch & 0x02)<<5) | ((ch & 0x20)>>1) | ((ch & 0x80)>>2);
PORTB = port;
port = 0;
port = ((ch & 0x1C)<<1) | (ch & 0x40);
PORTH = port;
}
Un programme principal du genre
//*** testé OK sur Arduino MEGA2560 + shield
int main(void){
unsigned char transcod[16]={0x3F,0x06,0x5B,0x4F,0x66,0x6D,0x7D,0x07,0x7F,0x6F,0x77,0x7C,0x39,0x5E,0x79,0x71},cmpt=0;
DDRH = 0x78; // 4 sorties pour H
DDRB = 0xF0; // 4 sorties pour B
DDRG = 0x20; // PG5 en sortie
while(1) {
affiche7segs(transcod[cmpt]);
_delay_ms(1000);
cmpt++;
if (cmpt > 15) cmpt=0;
} // while(1)
return 0;
}
permet les tests.
3°) La documentation du schield dit que pinMUX = 4. La documentation Arduino MEGA2560 donne 4 en PG5. C'est donc PG5 qui permet de commuter les afficheurs.
Pour obtenir 100 Hz à partir de 16MHz, il faut diviser par 1024 puis par 156. Cela peut se faire avec le programme suivant :
//*** testé OK sur Arduino MEGA2560 + shield
#include <avr/io.h>
#undef F_CPU
#define F_CPU 16000000UL
#include <util/delay.h>
#include <avr/interrupt.h>
void affiche7segs(unsigned char ch){
unsigned char port;
port = 0;
port = ((ch & 0x01)<<7) | ((ch & 0x02)<<5) | ((ch & 0x20)>>1) | ((ch & 0x80)>>2);
PORTB = port;
port = 0;
port = ((ch & 0x1C)<<1) | (ch & 0x40);
PORTH = port;
}
unsigned char mux=0,cmpt=0;
// on utilisera seulement les dix premières cases du tableau suivant :
unsigned char transcod[16]={0x3F,0x06,0x5B,0x4F,0x66,0x6D,0x7D,0x07,0x7F,0x6F,0x77,0x7C,0x39,0x5E,0x79,0x71};
// timer0 overflow
ISR(TIMER0_OVF_vect) {
TCNT0 = 100;
mux++;
if ((mux & 0x01)==0) {
PORTG &= ~(1<<5);
affiche7segs(transcod[cmpt >> 4]);
} else {
PORTG |= (1<<5);
affiche7segs(transcod[cmpt & 0X0F]);
}
}
int main(void){
DDRH = 0x78; // 4 sorties pour H
DDRB = 0xF0; // 4 sorties pour B
DDRG = 0x20; // PG5 en sortie
//** gestion du timer0
// enable timer overflow interrupt for both Timer0
TIMSK0=(1<<TOIE0);
// set timer0 counter initial value to 0
TCNT0=0x00;
// start timer0 with 1024 prescaler
TCCR0B = (1<<CS02) | (1<<CS00);
// enable interrupts
sei();
while(1) {
// réalisation d'un compteur BCD
cmpt ++;
if ((cmpt & 0x0F)>9) cmpt+=6;
if ((cmpt & 0xF0)>0x90) cmpt+=0x60;
// on attend
_delay_ms(1000);
} // while(1)
return 0;
}
Timer 0 et comparaison
modifierCette fonctionnalité n'existait pas dans l'ATMega8. Elle a été ajoutée pour gérer le PWM (Modulation de largeur d'impulsion = MLI et Pulse Width Modulation en anglais). Ce PWM sert essentiellement à commander des moteurs de Robots mais éventuellement à moduler des éclairages.
Les fonctionnalités ajoutées sont difficiles à appréhender. En effet le mode comparaison permet de gérer différents modes pour pouvoir réaliser toute une gamme de signaux du carré au PWM. Ceci complique un peu la gestion pour le programmeur. La bonne nouvelle c’est que les autres timers ont un fonctionnement un peu similaire.
Documentation de la comparaison
modifierLes différents modes de comparaison sont choisis à l'aide des bits WGM02, WGM01 et WGM00. Ces choix sont conformes au tableau suivant.
- Description des bits pour la génération de forme d'onde
Mode | WGM02 | WGM01 | WGM00 | Mode de fonctionnement | Bas si | Mise à jour de OCRAx si | Drapeau TOV0 positionné si |
---|---|---|---|---|---|---|---|
0 | 0 | 0 | 0 | Normal | 0XFF | immédiatement | MAX |
1 | 0 | 0 | 1 | PWM à phase correct | OXFF | TOP | BOTTOM |
2 | 0 | 1 | 0 | CTC | OCR0A | immédiatement | MAX |
3 | 0 | 1 | 1 | PWM rapide | 0XFF | BOTTOM | MAX |
4 | 1 | 0 | 0 | Reservé | - | - | - |
5 | 1 | 0 | 1 | PWM à phase correct | OCR0A | TOP | BOTTOM |
6 | 1 | 1 | 0 | Reservé | - | - | - |
7 | 1 | 1 | 1 | PWM rapide | OCR0A | BOTTOM | TOP |
- MAX = 0xFF
- BOTTOM = 0x00
- CTC : Clear Timer on Compare match = RAZ du timer quand comparaison
Pour chacun des modes de ce tableau ci-dessus, les bits COM0A1 et COM0A0 auront un fonctionnement différent. Ces bits sont destinés à gérer les formes d'onde du signal de sortie.
Comparaison simple
modifierNous présentons la comparaison simple ici. Elle est essentiellement utilisée avec le mode "Bascule OC0A sur la comparaison" ci-dessous quand l'objectif est de réaliser un signal de fréquence déterminé sans utiliser de logiciel (sauf pour tout initialiser bien sûr). Le mode de fonctionnement le plus adapté du timer dans ce cas est appelé CTC (Clear Timer on Compare match). Ce mode n’est pas choisi avec le tableau ci-dessous mais avec le tableau précédent.
- Mode non PWM pour la comparaison
COM0A1 | COM0A0 | Description |
---|---|---|
0 | 0 | Opération Normale PORT, OC0A déconnecté |
0 | 1 | Bascule OC0A sur la comparaison |
1 | 0 | Mise à 0 de OC0A sur la comparaison, mise à 1 sur overflow |
1 | 1 | Mise à 1 de OC0A sur la comparaison, mise à 0 sur overflow |
Et voici donc la documentation correspondante :
Il n'est plus possible de déclencher une interruption de débordement dans le mode CTC sauf si vous mettez OCR0A à 0xFF.
En résumé, le mode CTC s'utilise de la manière suivante :
- mise à un de WGM01, WGM02 et WGM00 sont supposés à 0
- choix de COM0A1 et COM0A0 pour la logique de sortie
- choix du préscaler pour le démarrage du timer 0
Sans interruption, seul le mode "basculement du bit OC0A" a un intérêt mais il impose d’utiliser ce bit (qui est le bit b6 du PORTD).
L'interruption de comparaison peut servir à utiliser un bit de sortie quelconque. Elle n’est pas documentée mais le sera à travers un exercice (Exercice 8).
Exercice 7
modifier1°) Donner le squelette d'un programme qui utilise le mode CTC pour réaliser un signal de période 8 ms sur le bit OC0A. La fréquence du quartz sera de 16MHz comme sur la carte Arduino UNO.
2°) Dessiner sur un chronogramme le comptage du timer et le signal généré sur OC0A.
3°) Quelle est la fréquence la plus basse que l’on peut réaliser sur le bit OC0A ? Quel mode du générateur de signal utilise-t-on ?
4°) Quelle est la fréquence la plus haute que l’on peut réaliser sur le bit OC0A ? Quel mode du générateur de signal utilise-t-on ?
1°)
#include <avr/io.h>
int main(void){
// set OC0A as output
DDRD |= 0x40;
// Set the Timer Mode to CTC
TCCR0A |= (1 << WGM01);
// Set the value that you want to count to
OCR0A = 0xF9;
// Set OC0A toggle mode
TCCR0A |= (1 << COM0A0);
// set prescaler to 256 and start the timer
TCCR0B |= (1 << CS02);
while (1)
{
//main loop
}
return 0;
}
3°) En mode basculement (mode 1) :
Le 2 qui apparaît au dénominateur est lié au mode basculement.
4°) En mode CTC encore : . À vérifier quand même !!!
Le premier 2 qui apparaît au dénominateur est lié à la valeur minimale de OCR0A soit 1. Ainsi, pour faire une période il faut compter de 0 à 1 puis que l’on repasse à 0 (soit deux comptages). Le deuxième est lié au mode basculement.
Exercice 8 : comparaison simple
modifierCompléter l'exemple ci-dessous trouvé sur Internet pour réaliser un clignotement d'un Hertz sur une LED connectée sur le bit b4 du PORTB.
// this code sets up a timer0 for 4ms @ 16Mhz clock cycle
// an interrupt is triggered each time the interval occurs.
#include <avr/io.h>
#include <avr/interrupt.h>
int main(void){
// Set the Timer Mode to CTC
TCCR0A |= (1 << WGM01);
// Set the value that you want to count to
OCR0A = 0xF9;
TIMSK0 |= (1 << OCIE0A); //Set the ISR COMPA vect
sei(); //enable interrupts
// set prescaler to 256 and start the timer
TCCR0B |= (1 << CS02);
while (1)
{
//main loop
}
return 0;
}
ISR (TIMER0_COMPA_vect) // timer0 overflow interrupt
{
//event to be exicuted every 4ms here
}
Même si la documentation de l'interruption de la comparaison n'a pas été évoquée dans ce chapitre, le code ci-dessous n’est pas difficile à comprendre. On admettant qu'effectivement la valeur de 4 ms du commentaire soit correcte, il nous faut multiplier par 125 pour avoir 500 ms de période (soit compter de 0 à 124). On calcule 500ms à cause du basculement : il faut deux basculements pour une période.
// this code sets up a timer0 for 4ms @ 16Mhz clock cycle
// an interrupt is triggered each time the interval occurs.
//***** testé OK sur Arduino MEGA2560 + shield
#include <avr/io.h>
#include <avr/interrupt.h>
// compteur
volatile unsigned char cpt=0;
int main(void){
// Set the Timer Mode to CTC
TCCR0A |= (1 << WGM01);
// Set the value that you want to count to
OCR0A = 0xF9;
TIMSK0 |= (1 << OCIE0A); //Set the ISR COMPA vect
sei(); //enable interrupts
// set prescaler to 256 and start the timer
TCCR0B |= (1 << CS02);
//Configuration PORTB.4 en sortie
DDRB |= (1<<DDB4);
PORTB &= ~(1<<PB4); // PORTB.4 <-0
while (1)
{
//main loop
}
return 0;
}
ISR (TIMER0_COMPA_vect) { // timer0 overflow interrupt
//event to be exicuted every 4ms here
cpt++;
if (cpt == 124) { // 500ms/4ms = 125
cpt = 0;
PORTB ^=(1<<PB4); // basculement
}
}
Les 4 ms annoncées sont très théoriques et ne tiennent pas compte du temps que le processeur utilise pour réaliser l'interruption. Avant d'envisager de modifier ce code prenez un fréquencemètre (un oscilloscope n’est pas simple à utiliser autour du Hertz).
Mode PWM rapide
modifierPour information, le mode PWM rapide est abordé dans un autre chapitre sur le robot miniQ. Nous allons le décrire quand même ici : mieux vaut deux fois qu'une.
- Mode PWM rapide et comparaison
COM0A1 | COM0A0 | Description |
---|---|---|
0 | 0 | Opération Normale PORT, OC0A déconnecté |
0 | 1 | WGM02=0 Opération Normale PORT, OC0A déconnecté |
0 | 1 | WGM02=1 Basculement de OC0A sur la comparaison |
1 | 0 | Mise à 0 de OC0A sur la comparaison et à 1 à BOTTOM |
1 | 1 | Mise à 1 de OC0A sur la comparaison et à 0 à BOTTOM |
Voici un exemple de programme qui réalise une PWM sur une carte Arduino UNO :
#include <avr/io.h>
#undef F_CPU
#define F_CPU 16000000UL
#include <util/delay.h>
int main(void){
// Set the Timer Mode to PWM fast
TCCR0A |= ((1 << WGM01) | (1<<WGM00));
// clear OC0A on compare match set OC0A at TOP
TCCR0A |= (1 << COM0A1);
//Configuration (PORTB.6 pour UNO)
DDRD |= ((1<<DDD6)|(1<<DDD5)|(1<<DDD4));
PORTD |= (1<<PD5);
PORTD &= ~(1<<PD4);
TCNT0 = 0x00;
// set prescaler to 1024 and start the timer
TCCR0B |= ((1 << CS02) | (1 << CS00));
while(1) {
// Set the value that you want to count to
OCR0A = 128;
//_delay_ms(300);
} // while(1)
return 0;
}
Ce programme connecté à une carte L298N est capable d'entraîner un moteur courant continu d'un Robot. La carte de puissance en question nécessite de mettre un seul des deux bits (PD5 ou PD4) à 1 pour tourner dans un sens. Pour inverser le sens, il faut inverser les bits PD5 et PD4. La mise à 0 des deux bits PD4 et PD5 permet de freiner.
Mode PWM à phase correcte
modifierCe mode utilise le compteur 0 dans les deux sens : en comptage et décomptage.
- Mode PWM à phase correcte et comparaison
COM0A1 | COM0A0 | Description |
---|---|---|
0 | 0 | Opération Normale PORT, OC0A déconnecté |
0 | 1 | WGM02=0 Opération Normale PORT, OC0A déconnecté |
0 | 1 | WGM02=1 Basculement de OC0A sur la comparaison |
1 | 0 | Mise à 0 de OC0A sur la comparaison quand comptage et mise à 1 de OC0A sur la comparaison quand décomptage |
1 | 1 | Mise à 1 de OC0A sur la comparaison quand comptage et mise à 0 de OC0A sur la comparaison quand décomptage |
Exercice 9 : PWM rapide
modifierOn désire changer l'intensité d'éclairage d'une LED à l'aide d'une PWM rapide. La valeur du rapport cyclique varie entre 0 et 255 et sera systématiquement envoyée par la liaison série sous forme de deux caractères 0,...,9,A,...,F.
1°) Un sous-programme sera donc chargé de lire ces deux caractères d’en vérifier la syntaxe. Écrire ce sous-programme et le tester avec les afficheurs de l'exercice 6. On affichera "--" en cas d'erreur de syntaxe.
Lisez le chapitre Les communications en tout genre pour vous aider sur la communication série.
Voici un programme qui fait l'acquisition de deux caractères sensés représenter un nombre hexadécimal. Si un mauvais caractère est rentré un message est envoyé par la RS232 et sur les afficheurs comme demandé !
//***** testé OK sur Arduino MEGA2560 sans shield
#include <avr/io.h>
#undef F_CPU
#define F_CPU 16000000UL
#define BaudRate 9600
#define MYUBRR (F_CPU / 16 / BaudRate ) - 1
#include <util/delay.h>
//#include <avr/interrupt.h>
void serialInit(void);
unsigned char serialCheckTxReady(void);
unsigned char serialCheckRxComplete(void);
void serialWrite(unsigned char DataOut);
unsigned char serialRead(void);
void usart_puts(char str[]);
void usart_puts_hexa(unsigned char val);
int main(void){
unsigned char ch,rapport_cyclique,erreur=0;
serialInit();
while(1) {
if (serialCheckRxComplete()) {
ch = UDR0;
rapport_cyclique = ch-'0';
if (rapport_cyclique > 9) rapport_cyclique -= 7;
if (rapport_cyclique > 15) {
usart_puts("Erreur Caractère non reconnu\n");
erreur=1;
} else {
rapport_cyclique <<= 4;
ch = serialRead();
ch -= '0';
if (ch > 9) ch -= 7;
if (ch > 15) {
usart_puts("Erreur Caractère non reconnu\n");
erreur = 1;
} else {
rapport_cyclique += ch;
usart_puts_hexa(rapport_cyclique);
} // else
} // if (!erreur) else
}
// on attend
//_delay_ms(300);
} // while(1)
return 0;
}
// Written by Windell Oskay, http://www.evilmadscientist.com/
// Copyright 2009 Windell H. Oskay
// Distributed under the terms of the GNU General Public License, please see below.
void serialWrite(unsigned char DataOut)
{
while (serialCheckTxReady() == 0) // while NOT ready to transmit
{;;}
UDR0 = DataOut;
}
unsigned char serialCheckRxComplete(void)
{
return( UCSR0A & _BV(RXC0)) ; // nonzero if serial data is available to read.
}
unsigned char serialCheckTxReady(void)
{
return( UCSR0A & _BV(UDRE0) ) ; // nonzero if transmit register is ready to receive new data.
}
unsigned char serialRead(void)
{
while (serialCheckRxComplete() == 0) // While data is NOT available to read
{;;}
return UDR0;
}
void serialInit(void) {
//Serial Initialization
/*Set baud rate 9600 */
UBRR0H = (unsigned char)(MYUBRR>>8);
UBRR0L = (unsigned char) MYUBRR;
/* Enable receiver and transmitter */
UCSR0B = (1<<RXEN0)|(1<<TXEN0);
/* Frame format: 8data, No parity, 1stop bit */
UCSR0C = (3<<UCSZ00);
}
//************************************************************************
// function usart_puts()
// purpose: puts characters in first rs232 PORT
// arguments:
// corresponding string
// return:
// note:
//************************************************************************
void usart_puts(char str[]){
uint8_t i=0;
do {
serialWrite(str[i]);
i++;
} while(str[i]!=0);
}
//************************************************************************
// function usart_puts_hexa()
// purpose: puts number in hexadecimel in first rs232 PORT
// arguments:
// corresponding number
// return:
// note:
//************************************************************************
void usart_puts_hexa(unsigned char val){
int8_t digit=0;
char char_digit;
usart_puts("\n0X"); //debut hexadécimal du C
digit = (val >> 4) & 0x0F;
char_digit=digit+0x30;
if (char_digit>0x39) char_digit += 7;
serialWrite(char_digit);
digit = val & 0x0F;
char_digit=digit+0x30;
if (char_digit>0x39) char_digit += 7;
serialWrite(char_digit);
}
La vérification syntaxique est trop élémentaire, mais ce n’est pas sur ce point précis que l’on veut travailler.
2°) Changer votre programme de test précédent pour qu’il réalise un rapport cyclique sur le bit OC0A du PORTB.
Le bit OC0A correspond au bit PD6 pour l'ATMega328p de l'Arduino UNO, mais au bit B7 pour l'ATMega2560 de l'Arduino MEGA2560. Nos tests seront comme d'habitude réalisés avec l'ATMega2560.
Voici un programme qui change le rapport cyclique en fonction d'une valeur entrée par la liaison série :
#include <avr/io.h>
#undef F_CPU
#define F_CPU 16000000UL
#define BaudRate 9600
#define MYUBRR (F_CPU / 16 / BaudRate ) - 1
#include <util/delay.h>
//#include <avr/interrupt.h>
void serialInit(void);
unsigned char serialCheckTxReady(void);
unsigned char serialCheckRxComplete(void);
void serialWrite(unsigned char DataOut);
unsigned char serialRead(void);
void usart_puts(char str[]);
void usart_puts_hexa(unsigned char val);
int main(void){
unsigned char ch,rapport_cyclique,erreur=0;
TCCR0A = 0x00;
// Set the Timer Mode to PWM fast
TCCR0A |= ((1 << WGM01) | (1<<WGM00));
// clear OC0A on compare match set OC0A at TOP
TCCR0A |= (1 << COM0A1);
// Set the value that you want to count to
OCR0A = 0x88; // rapport cyclique
// set prescaler to 1024 and start the timer
TCCR0B = 0x00;
TCCR0B |= ((1 << CS02) | (1 << CS00));
//Configuration PORTB.7 en sortie (PORTB.6 pour UNO)
DDRB |= (1<<DDB7);
TCNT0 = 0x00;
serialInit();
while(1) {
if (serialCheckRxComplete()) {
erreur=0;
ch = UDR0;
rapport_cyclique = ch-'0';
if (rapport_cyclique > 9) rapport_cyclique -= 7;
if (rapport_cyclique > 15) {
usart_puts("Erreur Caractère non reconnu\n");
erreur=1;
} else {
rapport_cyclique <<= 4;
ch = serialRead();
ch -= '0';
if (ch > 9) ch -= 7;
if (ch > 15) {
usart_puts("Erreur Caractère non reconnu\n");
erreur = 1;
} else {
rapport_cyclique += ch;
usart_puts_hexa(rapport_cyclique);
} // else
} // if (!erreur) else
// ** non ! if (PORTB & 0x80) PORTB |= 0x10; else PORTB &= 0xEF;
}
if (!erreur) { // gérer rapport cyclique
OCR0A = rapport_cyclique;
}
// on attend
//_delay_ms(300);
} // while(1)
return 0;
}
// Written by Windell Oskay, http://www.evilmadscientist.com/
// Copyright 2009 Windell H. Oskay
// Distributed under the terms of the GNU General Public License, please see below.
void serialWrite(unsigned char DataOut)
{
while (serialCheckTxReady() == 0) // while NOT ready to transmit
{;;}
UDR0 = DataOut;
}
unsigned char serialCheckRxComplete(void)
{
return( UCSR0A & _BV(RXC0)) ; // nonzero if serial data is available to read.
}
unsigned char serialCheckTxReady(void)
{
return( UCSR0A & _BV(UDRE0) ) ; // nonzero if transmit register is ready to receive new data.
}
unsigned char serialRead(void)
{
while (serialCheckRxComplete() == 0) // While data is NOT available to read
{;;}
return UDR0;
}
void serialInit(void) {
//Serial Initialization
/*Set baud rate 9600 */
UBRR0H = (unsigned char)(MYUBRR>>8);
UBRR0L = (unsigned char) MYUBRR;
/* Enable receiver and transmitter */
UCSR0B = (1<<RXEN0)|(1<<TXEN0);
/* Frame format: 8data, No parity, 1stop bit */
UCSR0C = (3<<UCSZ00);
}
//************************************************************************
// function usart_puts()
// purpose: puts characters in first rs232 PORT
// arguments:
// corresponding string
// return:
// note:
//************************************************************************
void usart_puts(char str[]){
uint8_t i=0;
do {
serialWrite(str[i]);
i++;
} while(str[i]!=0);
}
//************************************************************************
// function usart_puts_hexa()
// purpose: puts number in hexadecimel in first rs232 PORT
// arguments:
// corresponding number
// return:
// note:
//************************************************************************
void usart_puts_hexa(unsigned char val){
int8_t digit=0;
char char_digit;
usart_puts("\n0X"); //debut hexadécimal du C
digit = (val >> 4) & 0x0F;
char_digit=digit+0x30;
if (char_digit>0x39) char_digit += 7;
serialWrite(char_digit);
digit = val & 0x0F;
char_digit=digit+0x30;
if (char_digit>0x39) char_digit += 7;
serialWrite(char_digit);
}
Exercice 10 : PWM rapide et servomoteur
modifierRéaliser une commande d'un servomoteur à l'aide du timer 0.
Indications :
- La fréquence de 50 Hz (période T=20 ms) n'est pas importante pour la commande des servomoteurs.
- Le rapport cyclique par contre a besoin de varier entre 5% et 10%.
- Un calcul simple montre donc que OCR0A doit varier entre 255/10 et 255/20 ce qui donnera donc 26 et 13.
- On utilisera la fréquence minimale du Timer0.
- OC0A est le bit PD6 du PORTD. Sur une carte Arduino, il s'agit de la broche -6
#include <avr/io.h>
#include <util/delay.h>
// Frequence 16MHz
//*********** Pour platine UNO *************
int main() {
//Configuration PORTD.6 pour UNO
DDRD |= (1<<DDD6);
// set prescaler to 1024 and start the timer
TCCR0B |= (1 << CS02)|(1 << CS00);
// Set the Timer Mode to PWM fast
TCCR0A |= ((1 << WGM01) | (1<<WGM00));
// clear OC0A on compare match set OC0A at TOP
TCCR0A |= (1 << COM0A1);
OCR0A = 26;
TCNT0 = 0x00;
while(1) {
OCR0A -= 1;
if (OCR0A == 13) OCR0A = 26;
_delay_ms(1000);
}
return 0;
}