Cours:TPS XR207 tpTimer : Différence entre versions
(→{{Bleu|Allons un peu plus loin ... }}) |
(→{{Bleu|Configuration du Timer2}}) |
||
Ligne 255 : | Ligne 255 : | ||
<source lang=c> | <source lang=c> | ||
#define F_CPU 16000000UL | #define F_CPU 16000000UL | ||
+ | #define __DELAY_BACKWARD_COMPATIBLE__ | ||
#include <util/delay.h> | #include <util/delay.h> | ||
+ | |||
enum notes {Do, Dod, Re, Red, Mi, Fa, Fad, Sol, Sold, La, Lad, Si}; | enum notes {Do, Dod, Re, Red, Mi, Fa, Fad, Sol, Sold, La, Lad, Si}; | ||
− | unsigned char notes[12]={ .... }; | + | unsigned char freqNote[12]={ .... }; |
+ | |||
+ | int main() | ||
+ | { | ||
+ | } | ||
+ | </source> | ||
+ | |||
+ | |||
+ | =={{Bleu|Tester vos notes}}== | ||
+ | |||
+ | Commençons simplement par jouer toute la gamme. Il suffit pour cela de changer régulièrement de note. | ||
+ | |||
+ | Il convient bien évidemment de configurer correctement le timer; et de modifier régulièrement (à l'intérieur de la boucle "while(1)") la valeur de OCR2A. | ||
+ | |||
+ | {{Question|Écrire ce programme, et tester}} | ||
+ | |||
+ | <source lang=c> | ||
+ | #define F_CPU 16000000UL | ||
+ | #define __DELAY_BACKWARD_COMPATIBLE__ | ||
+ | #include <util/delay.h> | ||
+ | |||
+ | |||
+ | enum notes {Do, Dod, Re, Red, Mi, Fa, Fad, Sol, Sold, La, Lad, Si}; | ||
+ | unsigned char freqNote[12]={ .... }; | ||
+ | |||
+ | int main(){ // variables // e/s // configuration du timer2 (prédiviseur) // mode de génération de signaux (COM2A1 COM2A0) // boucle while(1) { // parcourir le tableau de notes et modifier la valeur de OCR2A | ||
+ | OCR2A= ...; | ||
+ | _delay_ms(xxx); | ||
+ | }} | ||
+ | </source> | ||
+ | |||
+ | '''Remarque :''' L'énumération est équivalente à dire que Do = 0, Dod = 1, Re = 2, ... | ||
+ | |||
+ | =={{Bleu|Préparons la suite}}== | ||
+ | |||
+ | Afin de pouvoir écrire facilement une mélodie, nous allons écrire une fonction qui permet de changer la note, et également de modifier sa durée. | ||
+ | |||
+ | le prototype de la fonction sera : | ||
+ | <source lang=c> | ||
+ | void playnote(char n,unsigned int duree); | ||
+ | </source> | ||
+ | |||
+ | {{Question|Modifier votre programme afin de pouvoir faire de la musique de la façon suivante :}} | ||
+ | |||
+ | <source lang=c> | ||
+ | #define F_CPU 16000000UL | ||
+ | #define __DELAY_BACKWARD_COMPATIBLE__ | ||
+ | #include <util/delay.h> | ||
+ | enum notes {Do, Dod, Re, Red, Mi, Fa, Fad, Sol, Sold, La, Lad, Si}; | ||
+ | unsigned char freqNote[12]={239,226,213,201,190,179,169,159,142,134,127}; | ||
+ | |||
+ | #define blanche 400 | ||
+ | #define noire 200 | ||
+ | #define croche 100 | ||
+ | |||
+ | void playnote(char n,unsigned int duree) | ||
+ | { | ||
+ | // modifier la valeur de OCR2A suivant la note n | ||
+ | OCR2A = ...; | ||
+ | // attendre la durée de la note | ||
+ | _delay_ms(duree); | ||
+ | } | ||
+ | |||
int main() | int main() | ||
{ | { | ||
+ | // variables // e/s // configuration du timer2 (prédiviseur) // mode de génération de signaux (COM2A1 COM2A0) // boucle | ||
+ | while(1) | ||
+ | { | ||
+ | playnote(Do,noire); | ||
+ | playnote(Re,noire); | ||
+ | playnote(Mi,noire); | ||
+ | playnote(Re,noire); | ||
+ | } | ||
} | } | ||
</source> | </source> | ||
+ | |||
+ | =={{Bleu|Petite amélioration}}== | ||
+ | |||
+ | On pourra légèrement modifier la fonction '''playnote''' pour introduire un "blanc" (pas de son) entre 2 notes. | ||
+ | |||
+ | Pour ce faire, il suffit d'arrêter pendant un temps très court (1ms) le timer en choisissant un prédiviseur nul, et ensuite remettre la "bonne" valeur de prédiviseur : | ||
+ | |||
+ | <source lang=c> | ||
+ | void playnote(char n,unsigned int duree) | ||
+ | { | ||
+ | // modifier la valeur de OCR2A suivant la note n | ||
+ | OCR2A = ...; | ||
+ | // attendre la durée de la note | ||
+ | _delay_ms(duree); | ||
+ | // arrêter le timer : | ||
+ | TCCR2A &= ~(...); | ||
+ | // pause dans le son | ||
+ | _delay_ms(1); | ||
+ | // remettre le timer en route | ||
+ | TCCR2A |= (.....); | ||
+ | } | ||
+ | |||
+ | </source> | ||
+ | |||
+ | =={{Bleu|Mélodie !}}== | ||
− | |||
=> changer les notes avec 2ème timer | => changer les notes avec 2ème timer |
Version du 3 avril 2015 à 14:55
Retour à la liste des Tps
Nous allons dans ce TP nous pencher sur l'utilisation des timers, notamment pour générer un signal, ceci en utilisant le moins possible les ressources du µcontrôleur. |
Comment générer un signal périodique
Nous allons dans cette partie comparer les 2 principales méthodes pour générer un signal périodique.
Dans toute cette partie, nous considérerons une sortie périodique telle que :
- le signal est carré (2 états '0' ou '1')
- de fréquence 1kHz
- de rapport cyclique 1/2
- sortie sur la patte PB3
avec une attente .... méthode basique !
Commençons tout simplement par utiliser une attente, comme nous l'avons fait dans le tp précédent.
Il suffit tout bonnement de configurer correctement la valeur d'attente.
Remarquons que pour plus de précision, il est possible d'utiliser la fonction _delay_us(xxx), dont le nom est suffisamment explicite pour se passer d'une description !
Écrire un programme générant le signal souhaité, et le visualiser avec un oscilloscope.
#include <util/delay.h>
int main()
{
// configuration e/s
while(1) // boucle
{
// modification de la sortie
// attente
_dela.....
}
}
Remarque : Comme expliqué en TD, cette méthode fonctionne mais entraîne une occupation à 100% du processeur, qui ne peut donc réaliser que cette action !
l'outil adapté : le timer !
Passons aux choses sérieuses en utilisant un timer (en l’occurrence le timer2 d'un atmega328p), et donnons tout d'abord les 2 figures qui nous permettront de le configurer facilement :
Remarquons tout de suite qu'il s'agit d'un compteur 8 bits, c'est à dire qu'il générera un débordement (overflow) lors du passage de 255 à 0.
Paramètres du timer
Comme pour toute utilisation du timer, il est nécessaire de choisir :
- la valeur du prédiviseur, et donc la fréquence de comptage : fcpt=fq/p
- le nombre de fois que le compteur s'incrémentera, n, avant d'être réinitialisé
On rappelle que sur les cartes arduino UNO, fq=16MHz
La figure ci après doit vous permettre de trouver facilement la relation entre les différentes valeurs et ainsi déterminer les valeurs n et p adaptées à notre problème.
Mode comparaison avec interruption
Nous pouvons désormais écrire le programme utilisant le timer2, qui devra utiliser le mode comparaison.
Autrement dit, on configure le timer pour que sa valeur soit réinitialisée à 0 lorsqu'il atteint la valeur OCR2A (ou si vous préférez la valeur n que vous avez calculé juste avant).
Comme indiqué sur la figure un peu plus haut, il s'agit de configurer les bits WGM2[2..0] dans le mode adéquat.
Il conviendra également de générer une interruption à chaque remise à zéro (ou plus exactement lors de chaque comparaison), et pour ce faire il suffit :
- autoriser l'interruption de comparaison : TIMSK2|=1<<OCIE2A
- autoriser globalement les interruptions : sei()
- la fonction d'interruption sera définie par : ISR(TIMER2_COMPA_vect)
La valeur de sortie sera alors modifiée à chaque interruption, et le µcontrôleur pourra vaquer à d'autres activités dans le programme principal.
Écrire le programme utilisant le Timer2, en vous servant du squelette suivant :
// Fonction de traitement Timer 2 Comparaison
ISR(TIMER2_COMPA_vect)
{
// il suffit de changer l'état de la sortie
}
int main()
{
// variables
// e/s
// configuration du timer2 (prédiviseur)
// autorisation d'interruption (cf ci dessus)
// boucle
while(1)
{
// tout se passe dans l'interruption, rien à faire ici
}
}
Sans interruption mais sans attente passive : le matériel fait tout
Par commodité, nous remettons (ci-contre) la documentation correspondante sous forme de schéma.
Ce mode de fonctionnement utilise le mode comparaison, et pilote également directement une sortie bien particulière.
L'idée générale est que lorsque le timer 2 (TCNT2) arrive à la même valeur que celle qui est contenue dans un registre (OCR2A ou OCR2B) une logique interne est capable de changer (ou pas) une sortie qui s'appelle OC2A ou OC2B.
Ce mode est essentiellement géré par les deux bits COM2A1 et COM2A0 comme indiqué dans le tableau ci-dessous :
- Mode non PWM pour la comparaison
COM2A1 | COM2A0 | Description |
---|---|---|
0 | 0 | Opération Normale PORT, OC2A déconnecté |
0 | 1 | Bascule OC2A sur la comparaison |
1 | 0 | Mise à 0 de OC2A sur la comparaison |
1 | 1 | Mise à 1 de OC2A sur la comparaison |
Nous avons ajouté les informations correspondant à la carte UNO entre parenthèse. Par exemple la sortie correspondante au bit b3 du PORTD est OC2A et correspond au numéro 11 de la carte UNO. L'autre broche gérée par le comparateur B est la broche 3 et se trouve entre parenthèse avec le registre de comparaison.
Modifier le programme précédent en supprimant la fonction d'interruption, et en pilotant directement la sortie depuis le timer, en vous servant du tableau précédent.
Indication : l'idée est de placer le timer en mode CTC (Clear Timer on Compare match) et la génération de signaux en mode basculement de OC2A.
int main()
{
// variables
// e/s
// configuration du timer2 (prédiviseur)
// mode de génération de signaux (COM2A1 COM2A0)
// boucle
while(1)
{
// Plus aucune utilisation processeur
}
}
Allons un peu plus loin ...
Il vous reste moins d'1h pour terminer le TP, passez à la partie suivante !
Complétons quelque peu notre programme en donnant la possibilité de modifier la fréquence du signal de sortie.
Nous ajoutons 2 boutons (A et B sur notre carte shieldInfo).
|
|
Écrire un programme répondant au cahier des charges
ISR(INT0_vect)
{
}
ISR(INT1_vect)
{
}
int main()
{
// variables
// e/s
// configuration du timer2 (prédiviseur)
// mode de génération de signaux (COM2A1 COM2A0)
// configuration des interruptions INT1 et INT2 (EIMSK et EICRA)
sei(); // autorisation des interruptions
// boucle
while(1)
{
// Plus aucune utilisation processeur
}
}
Du son !!!
Objectif
Nous allons écrire un programme permettant de jouer de la musique sur un buzzer, et ceci en chargeant le moins possible le µcontrôleur. Pour ce faire, nous utiliserons au maximum les timers à disposition.
Le buzzer sera connecté entre la masse et la sortie PB5 du µcontrôleur, et le tableau suivant donne les fréquences à générer pour obtenir les différentes notes.
Note | Do | Do# | Ré | Ré# | Mi | Fa | Fa# | Sol | Sol# | La | La# | Si |
---|---|---|---|---|---|---|---|---|---|---|---|---|
Fréquence (Hz) | 261,63 | 277,18 | 293,66 | 311,13 | 329,63 | 349,23 | 369,99 | 392 | 415,3 | 440 | 466,16 | 493,88 |
Période (µs) | 3822 | 3608 | 3405 | 3214 | 3037 | 2863 | 2703 | 2551 | 2408 | 2273 | 2145 | 2025 |
Configuration du Timer2
Le timer 2 sera bien entendu utilisé pour générer les notes sur le buzzer : en effet, c'est lui qui est connecté directement sur la patte PB5 !
Une seule valeur de prédiviseur devra être utilisée pour toutes les notes.
Choisissez une valeur de prédiviseur, et calculer alors la valeur de 'n', ceci pour chaque note
Remarque : On pourra astucieusement utiliser un tableur.
Saisir alors ces valeurs dans un tableau :
#define F_CPU 16000000UL
#define __DELAY_BACKWARD_COMPATIBLE__
#include <util/delay.h>
enum notes {Do, Dod, Re, Red, Mi, Fa, Fad, Sol, Sold, La, Lad, Si};
unsigned char freqNote[12]={ .... };
int main()
{
}
Tester vos notes
Commençons simplement par jouer toute la gamme. Il suffit pour cela de changer régulièrement de note.
Il convient bien évidemment de configurer correctement le timer; et de modifier régulièrement (à l'intérieur de la boucle "while(1)") la valeur de OCR2A.
Écrire ce programme, et tester
#define F_CPU 16000000UL
#define __DELAY_BACKWARD_COMPATIBLE__
#include <util/delay.h>
enum notes {Do, Dod, Re, Red, Mi, Fa, Fad, Sol, Sold, La, Lad, Si};
unsigned char freqNote[12]={ .... };
int main(){ // variables // e/s // configuration du timer2 (prédiviseur) // mode de génération de signaux (COM2A1 COM2A0) // boucle while(1) { // parcourir le tableau de notes et modifier la valeur de OCR2A
OCR2A= ...;
_delay_ms(xxx);
}}
Remarque : L'énumération est équivalente à dire que Do = 0, Dod = 1, Re = 2, ...
Préparons la suite
Afin de pouvoir écrire facilement une mélodie, nous allons écrire une fonction qui permet de changer la note, et également de modifier sa durée.
le prototype de la fonction sera :
void playnote(char n,unsigned int duree);
Modifier votre programme afin de pouvoir faire de la musique de la façon suivante :
#define F_CPU 16000000UL
#define __DELAY_BACKWARD_COMPATIBLE__
#include <util/delay.h>
enum notes {Do, Dod, Re, Red, Mi, Fa, Fad, Sol, Sold, La, Lad, Si};
unsigned char freqNote[12]={239,226,213,201,190,179,169,159,142,134,127};
#define blanche 400
#define noire 200
#define croche 100
void playnote(char n,unsigned int duree)
{
// modifier la valeur de OCR2A suivant la note n
OCR2A = ...;
// attendre la durée de la note
_delay_ms(duree);
}
int main()
{
// variables // e/s // configuration du timer2 (prédiviseur) // mode de génération de signaux (COM2A1 COM2A0) // boucle
while(1)
{
playnote(Do,noire);
playnote(Re,noire);
playnote(Mi,noire);
playnote(Re,noire);
}
}
Petite amélioration
On pourra légèrement modifier la fonction playnote pour introduire un "blanc" (pas de son) entre 2 notes.
Pour ce faire, il suffit d'arrêter pendant un temps très court (1ms) le timer en choisissant un prédiviseur nul, et ensuite remettre la "bonne" valeur de prédiviseur :
void playnote(char n,unsigned int duree)
{
// modifier la valeur de OCR2A suivant la note n
OCR2A = ...;
// attendre la durée de la note
_delay_ms(duree);
// arrêter le timer :
TCCR2A &= ~(...);
// pause dans le son
_delay_ms(1);
// remettre le timer en route
TCCR2A |= (.....);
}
Mélodie !
=> changer les notes avec 2ème timer