: Guide épique Robert Brun·11 juin 2021
Le microcontrôleur possède plusieurs temporisateurs qui peuvent exécuter différentes fonctions, telles que la génération d'un signal PWM . Pour que le temporisateur génère un signal PWM, il doit être préconfiguré en éditant le registre du temporisateur. Lorsque nous travaillons dans l'IDE Arduino, les minuteries sont configurées à notre insu dans la bibliothèque Arduino.h et obtiennent en fait les paramètres souhaités par les développeurs. Et ces paramètres ne sont pas très bons : la fréquence PWM par défaut est faible et les minuteries ne sont pas utilisées à leur plein potentiel. Regardons le PWM standard de l'ATmega328 (Arduino UNO/ Nano / Pro Mini ) :
Minuteur | Épingles | Fréquence | Résolution |
---|---|---|---|
Minuterie 0 | D5 et D6 | 976Hz | 8 bits (0-255) |
Minuterie 1 | D9 et D10 | 488Hz | 8 bits (0-255) |
Minuterie 2 | D3 et D11 | 488Hz | 8 bits (0-255) |
En fait, tous les temporisateurs peuvent facilement émettre un signal PWM de 64 kHz , et le temporisateur 1 – c'est même 16 bits, et à la fréquence qui lui a été donnée Arduino, pourrait fonctionner avec une résolution de 15 bits au lieu de 8, et cela, soit dit en passant, 32768 gradations de remplissage au lieu de 256 ! Alors pourquoi cette injustice ? La minuterie 0 est en charge de la synchronisation et est réglée de manière à ce que les millisecondes s'écoulent avec précision. Les autres minuteries sont ramenées à zéro pour éviter que l'amateur d'Arduino n'ait des problèmes inutiles. Cette approche est généralement compréhensible mais aurait fait au moins quelques fonctions standard pour une fréquence plus élevée, eh bien, sérieusement ! D'accord, s'ils ne l'ont pas fait, nous le ferons.
La génération PWM est réglée via les registres de temporisation. Ensuite, vous trouverez des “morceaux” de code prêts à l'emploi, que vous devez insérer dans setup(), et la fréquence PWM sera reconfigurée (le pré-délimiteur et le mode minuterie changeront). Vous pouvez toujours travailler avec le signal PWM avec la analogWrite()fonction, contrôlant le remplissage du PWM sur les broches standard.
// Pins D5 and D6 are 62.5kHz TCCR0B = 0b00000001; // x1 TCCR0A = 0b00000011; // fast pwm // Pins D5 and D6 - 31.4 kHz TCCR0B = 0b00000001; // x1 TCCR0A = 0b00000001; // phase correct // Pins D5 and D6 - 7.8 kHz TCCR0B = 0b00000010; // x8 TCCR0A = 0b00000011; // fast pwm // Pins D5 and D6 - 4 kHz TCCR0B = 0b00000010; // x8 TCCR0A = 0b00000001; // phase correct // Pins D5 and D6 - 976 Hz - default TCCR0B = 0b00000011; // x64 TCCR0A = 0b00000011; // fast pwm // Pins D5 and D6 - 490 Hz TCCR0B = 0b00000011; // x64 TCCR0A = 0b00000001; // phase correct // Pins D5 and D6 - 244 Hz TCCR0B = 0b00000100; // x256 TCCR0A = 0b00000011; // fast pwm // Pins D5 and D6 - 122 Hz TCCR0B = 0b00000100; // x256 TCCR0A = 0b00000001; // phase correct // Pins D5 and D6 - 61 Hz TCCR0B = 0b00000101; // x1024 TCCR0A = 0b00000011; // fast pwm // Pins D5 and D6 - 30 Hz TCCR0B = 0b00000101; // x1024 TCCR0A = 0b00000001; // phase correct
// Broches D9 et D10 - 62,5 kHz TCCR1A = 0b00000001 ; // 8 bits TCCR1B = 0b00001001 ; // x1 pwm rapide // Broches D9 et D10 - 31,4 kHz TCCR1A = 0b00000001 ; // 8 bits TCCR1B = 0b00000001 ; // phase x1 correcte // Broches D9 et D10 - 7,8 kHz TCCR1A = 0b00000001 ; // 8 bits TCCR1B = 0b00001010 ; // x8 rapide pwm // Broches D9 et D10 - 4 kHz TCCR1A = 0b00000001 ; // 8 bits TCCR1B = 0b00000010 ; // phase x8 correcte // Broches D9 et D10 - 976 Hz TCCR1A = 0b00000001 ; // 8 bits TCCR1B = 0b00001011 ; // pwm rapide x64 // Broches D9 et D10 - 490 Hz - par défaut TCCR1A = 0b00000001 ; // 8 bits TCCR1B = 0b00000011 ; // phase x64 correcte // Broches D9 et D10 - 244 Hz TCCR1A = 0b00000001 ; // 8 bits TCCR1B = 0b00001100 ; // x256 pwm rapide // Broches D9 et D10 - 122 Hz TCCR1A = 0b00000001 ; // 8 bits TCCR1B = 0b00000100 ; // phase x256 correcte // Broches D9 et D10 - 61 Hz TCCR1A = 0b00000001 ; // 8 bits TCCR1B = 0b00001101 ; // x1024 pwm rapide // Broches D9 et D10 - 30 Hz TCCR1A = 0b00000001 ; // 8 bits TCCR1B = 0b00000101 ; // phase x1024 correcte
// Broches D9 et D10 - 15,6 kHz 10 bits TCCR1A = 0b00000011 ; // 10 bits TCCR1B = 0b00001001 ; // x1 pwm rapide // Broches D9 et D10 - 7,8 kHz 10 bits TCCR1A = 0b00000011 ; // 10 bits TCCR1B = 0b00000001 ; // phase x1 correcte // Broches D9 et D10 - 2 kHz 10 bits TCCR1A = 0b00000011 ; // 10 bits TCCR1B = 0b00001010 ; // x8 rapide pwm // Broches D9 et D10 - 977 Hz 10 bits TCCR1A = 0b00000011 ; // 10 bits TCCR1B = 0b00000010 ; // phase x8 correcte // Broches D9 et D10 - 244 Hz 10 bits TCCR1A = 0b00000011 ; // 10 bits TCCR1B = 0b00001011 ; // pwm rapide x64 // Broches D9 et D10 - 122 Hz 10 bits TCCR1A = 0b00000011 ; // 10 bits TCCR1B = 0b00000011 ; // phase x64 correcte // Broches D9 et D10 - 61 Hz 10 bits TCCR1A = 0b00000011 ; // 10 bits TCCR1B = 0b00001100 ; // x256 pwm rapide // Broches D9 et D10 - 30 Hz 10 bits TCCR1A = 0b00000011 ; // 10 bits TCCR1B = 0b00000100 ; // phase x256 correcte // Broches D9 et D10 - 15 Hz 10 bits TCCR1A = 0b00000011 ; // 10 bits TCCR1B = 0b00001101 ; // x1024 pwm rapide // Broches D9 et D10 - 7,5 Hz 10 bits TCCR1A = 0b00000011 ; // 10 bits TCCR1B = 0b00000101 ; // phase x1024 correcte
// Broches D3 et D11 - 62,5 kHz TCCR2B = 0b00000001 ; // x1 TCCR2A = 0b00000011 ; // pwm rapide // Broches D3 et D11 - 31,4 kHz TCCR2B = 0b00000001 ; // x1 TCCR2A = 0b00000001 ; // phase correcte // Broches D3 et D11 - 8 kHz TCCR2B = 0b00000010 ; // x8 TCCR2A = 0b00000011 ; // pwm rapide // Broches D3 et D11 - 4 kHz TCCR2B = 0b00000010 ; // x8 TCCR2A = 0b00000001 ; // phase correcte // Broches D3 et D11 - 2 kHz TCCR2B = 0b00000011 ; // x32 TCCR2A = 0b00000011 ; // pwm rapide // Broches D3 et D11 - 980 Hz TCCR2B = 0b00000011 ; // x32 TCCR2A = 0b00000001 ; // phase correcte // Broches D3 et D11 - 980 Hz TCCR2B = 0b00000100 ; // x64 TCCR2A = 0b00000011 ; // pwm rapide // Broches D3 et D11 - 490 Hz - par défaut TCCR2B = 0b00000100 ; // x64 TCCR2A = 0b00000001 ; // phase correcte // Broches D3 et D11 - 490 Hz TCCR2B = 0b00000101 ; // x128 TCCR2A = 0b00000011 ; // pwm rapide // Broches D3 et D11 - 245 Hz TCCR2B = 0b00000101 ; // x128 TCCR2A = 0b00000001 ; // phase correcte // Broches D3 et D11 - 245 Hz TCCR2B = 0b00000110 ; // x256 TCCR2A = 0b00000011 ; // pwm rapide // Broches D3 et D11 - 122 Hz TCCR2B = 0b00000110 ; // x256 TCCR2A = 0b00000001 ; // phase correcte // Broches D3 et D11 - 60 Hz TCCR2B = 0b00000111 ; // x1024 TCCR2A = 0b00000011 ; // pwm rapide // Broches D3 et D11 - 30 Hz TCCR2B = 0b00000111 ; // x1024 TCCR2A = 0b00000001 ; // phase correcte
void setup() { // Pins D5 and D6 - 7.8 kHz TCCR0B = 0b00000010; // x8 TCCR0A = 0b00000011; // fast pwm // Pins D3 and D11 - 62.5 kHz TCCR2B = 0b00000001; // x1 TCCR2A = 0b00000011; // fast pwm // Pins D9 and D10 - 7.8 kHz 10bit TCCR1A = 0b00000011; // 10bit TCCR1B = 0b00000001; // x1 phase correct analogWrite(3, 15); analogWrite(5, 167); analogWrite(6, 241); analogWrite(9, 745); // yes, range 0-1023 analogWrite(10, 345); // yes, range 0-1023 analogWrite(11, 78); } void loop() { }
Si vous voulez ou avez vraiment besoin d'un PWM overclocké sur le temporisateur système (zéro) sans perte de fonctions temporelles, vous pouvez les corriger comme suit :
#define micros() (micros() >> CORRECT_CLOCK) #define millis() (millis() >> CORRECT_CLOCK) void fixDelay(uint32_t ms) { delay(ms << CORRECT_CLOCK); }
Les définitions doivent être placées avant de brancher les bibliothèques afin qu'elles entrent dans le code et remplacent les fonctions. La seule chose est que vous ne pouvez pas corriger le retard dans une autre bibliothèque de cette façon. Vous pouvez utiliser fixDelay() pour vous-même comme indiqué ci-dessus.
La chose la plus importante est CORRECT_CLOCK. Il s'agit d'un nombre entier égal au rapport entre le diviseur de minuterie par défaut et le nouveau défini (pour l'accélération PWM). Par exemple, nous réglons le PWM à 8 kHz. Dans la liste ci-dessus, nous voyons que le diviseur par défaut est 64, et 7,8 kHz sera 8, ce qui est huit fois plus petit. CORRECT_CLOCK est défini en conséquence.
#define CORRECT_CLOCK 8 void fixDelay(uint32_t ms) { delay(ms << CORRECT_CLOCK); } void setup() { pinMode(13, 1); // Pins D5 and D6 - 4 kHz TCCR0B = 0b00000010; // x8 TCCR0A = 0b00000001; // phase correct } void loop() { digitalWrite(13, !digitalRead(13)); fixDelay(1000); }
En plus de jouer manuellement avec les registres, il existe des bibliothèques prêtes à l'emploi qui vous permettent de modifier la fréquence PWM de l'Arduino. Jetons un coup d'œil à certains d'entre eux :
PWM library ( GitHub ) - une bibliothèque puissante qui vous permet de modifier la fréquence PWM sur les microcontrôleurs ATmega48 / 88 / 168 / 328 / 640 / 1280 / 1281 / 2560 / 2561, dont 328 sur UNO/Nano/Mini et 2560 sur un Arduino Méga.
GyverPWM library( GitHub ) - la bibliothèque que nous avons écrite avec mon ami. La bibliothèque permet un travail très flexible avec PWM sur le microcontrôleur ATmega328 (nous ajouterons Mega plus tard):