« SE3Binome2023-6 » : différence entre les versions
Ligne 417 : | Ligne 417 : | ||
} | } | ||
</syntaxhighlight><syntaxhighlight lang="c"> | </syntaxhighlight><syntaxhighlight lang="c"> | ||
void | void read_sound_HF() { | ||
ad_init_HF(); | |||
unsigned int db = MAX_INT - ad_capture(); | unsigned int db = MAX_INT - ad_capture(); | ||
PORTB &= ~(0b00100000); | |||
PORTF &= ~(0b11100000); | |||
if(db > 140){ | if(db > 140){ | ||
PORTF |= 0b00100000; | |||
} | } | ||
if (db > 160){ | if (db > 160){ | ||
PORTF |= 0b01100000; | |||
} | } | ||
if (db > | if (db > 190){ | ||
PORTF |= 0b11100000; | |||
} | } | ||
if (db > | if (db > 240){ | ||
PORTF |= 0b11100000; | |||
PORTB |= 0b00100000; | |||
} | } | ||
} | } |
Version du 12 juin 2024 à 20:55
Projet : Audio Spectrum Analyzer (Analyseur de Spectre Audio)
Fonctionnement du système:
- Capture du son : Le microphone convertit le son en signal électrique
- Pré-amplification : On amplifie le signal pour le traiter
- Filtres passe-bande : On utilise des filtres pour diviser le signal en plusieurs bande de fréquences
- Échantillonnage : Les signaux sont numérisés par l'AtMega pour être traités
- Traitement : L'AtMega analyse les signaux
- Contrôle des LEDs : Utilisation d'une matrice LED pour afficher l'amplitude aux différentes bandes de fréquence
Comment fonctionne un microphone ?
Un microphone est composé d'un diaphragme, d'un aimant en forme de E allongé et d'une bobine de cuivre attaché autour du diaphragme. Lorsqu'un son est émis, le diaphragme se "compresse" et fait bouger la bobine de haut en bas créer ainsi un courant induit dans le circuit. Notre microphone fonctionne différemment. Il s'agit d'un condensateur qui s'élargie ou se rétrécie dû à la membrane. Le principe de ces microphones est que ce condensateur a une armature fixe et une autre souple, permettant lorsque l'onde sonore arrive de modifier la distance entre ces deux armatures. Or la capacité d’un condensateur est définie par C = e S/ e où e est la permittivité, S la surface et e la distance entre les armatures, qui est variable dans notre cas. Or le courant i = C dU/dt où U est la tension aux bornes du condensateur. Ainsi l’onde sonore fait varier C, puis U, que l’on peut mesurer, après amplification car typiquement le signal en sortie d’un micro est de l’ordre de 0.1 mV à 10 mV. En pratique, la vibration est amplifiée électriquement par un transistor, dont le fonctionnement exige une alimentation électrique. La tension d’excitation nécessaire au fonctionnement d’un micro électret est de l’ordre de 3 à 5 volts continus.
Créer un amplificateur audio :
Il faut d'abord créer un ampli audio (AOP ?) pour utiliser le signal électrique, créé par le diaphragme et la bobine, dan notre circuit. Le micro étant très petit, le signal électrique est très faible et peu exploitable. Il faut donc l'amplifier pour que l'interface audio puisse reconnaître plus efficacement les fluctuations des tensions. La remarque sur la sortie est de l'ordre de 0.1 mV à 10mV, il faut donc une très grande amplification. Au moins de l'ordre de 103 . Malheureusement, nous nous sommes trompés dans nos composants et avons pris deux résistances de 810Ω dans la précipitation. Dû à cette erreur nous n'avons qu'une pré amplification de 2. Nous ne savions pas quel microphone nous allions utiliser, car la nature de notre projet nécessite des composants assez importants pour les autres années de SE et nous ne savions donc pas quel micro nous allions utiliser. Nous aurions dû faire plus attention à la datasheet du micro et refaire nos calculs en conséquence. Nous aution pû nous en douter, ces microphones sont utilisés dans les téléphones de nos jours. Ils doivent être petits mais aussi très sensibles et par conséquences, nécessite une grande amplification.
Choix des filtres :
Pour les filtres passe-bande on utilise des filtres actifs pour pouvoir les intégrés facilement dans un PCB. L'utilisation d'AOP sera obligatoire pour faciliter la création des filtres. Sans les AOP, nous serions obligés d'utiliser des self-inductances dans notre PCB, ce qui peut poser problèmes. Les AOP sont limités en hautes fréquences mais pas pour l'utilisation audio. Seulement les fréquences de 20Hz à 20kHz sont utilisées alors que la limite est de l'ordre du MHz.
Nous utilisons des AOP sous forme de cellules Sallen & key pour créer des filtres passe-bande d'ordre 2. Ils se comptent au nombre de 3 : un filtre Basse-Fréquence (20Hz-7kHz), un Mid-Fréquence(7kHz-14kHz) et un Haute-fréquence(14kHz-20kHz).
Évolution des brasures de la carte :
Dû à un manque d'AOP, nous n'avions pu faire que l'amplification et les filtres pour les Hautes et Basses Fréquences
KiCAD :
Pour analyser le signal audio, on le sépare en trois signaux filtré sur des bandes de fréquences spécifiques, pour cela on utilise trois filtres.
Pour chacun des filtres, on met à la suite simplement un filtre passe-bas et un filtre passe-haut.
Pour le premier filtre on conserve les fréquences allant de 20Hz à 7kHz environ.
Le second : 7kHz-14kHz
Le troisième: 14kHz-20kHz.
Il est à noté que que l'on peut changer facilement nos bandes de fréquences en choisissant de nouvelle résistance / condensateur
Code C et AtMega
EXPLICATIONS
Dans le bloc « prescaler » sert à configurer l’ADC ; plus précisément, le prescaler va diviser la fréquence de fonctionnement de l’AtMega (16 MHz sur un Arduino), par un facteur de division donné, afin de laisser le temps à analogRead
d’obtenir une mesure fiable, ainsi que pour avoir une bonne précision au niveau de la valeur lue. Ce prescaler est configuré au niveau du code d’Arduino afin d’obtenir un rapport de division de 128, ce qui est un rapport très important, la raison de ce choix est que les Arduino tournent aujourd’hui pour la plupart à 16MHz, mais le code d’Arduino doit être portable, et afin de tourner sur des fréquences aussi faibles de 1MHz en conservant la même fréquence pour l’ADC (afin d’en conserver la précision), il est impératif de mettre un rapport de division bien plus important pour les grandes fréquences ; voici un morceau du code de configuration d’analogRead
sur Arduino :
// Si la fréquence du processeur est de 16MHz
#if F_CPU >= 16000000 // 16 MHz / 128 = 125 KHz sbi(ADCSRA, ADPS2); sbi(ADCSRA, ADPS1); sbi(ADCSRA, ADPS0);
Dans notre cas F_CPU = 8000000 donc / 64
#if F_CPU >= 8000000 // 8 MHz / 64 = 125 KHz sbi(ADCSRA, ADPS2); sbi(ADCSRA, ADPS1); sbi(ADCSRA, ADPS0);
Comme vous pouvez le voir dans ce morceau de code, au niveau de l’AtMega, le rapport de division du prescaler est configuré par trois bits (ADPS[0-2]) du registre ADCSRA (nous l’avions mentionné sur notre schéma récapitulatif), ces trois bits fonctionnent de la façon suivante :
ADPS2 | ADPS1 | ADPS0 | Facteur de division |
---|---|---|---|
1 | 1 | 1 | 128 |
1 | 1 | 0 | 64 |
1 | 0 | 1 | 32 |
1 | 0 | 0 | 16 |
0 | 1 | 1 | 8 |
0 | 1 | 0 | 4 |
0 | 0 | 1 | 2 |
0 | 0 | 0 | 2 |
En pratique, le temps entre deux mesures est compris entre 16 et 20 µs, soit un peu plus que la théorie ; afin de ne pas vous encombrer avec la formule ou des mesures expérimentales, je vous propose de tableau résumant les temps et fréquences théoriques et pratiques obtenus sur un Arduino Uno : (Pour un F_CPU de 16 Mhz)
Rapport de division | Temps théorique | Temps min. | Temps max. | Fréquence moyenne |
---|---|---|---|---|
128 | 104 µs | 112 µs | 116 µs | 8.7 kHz |
64 | 52 µs | 54 µs | 64 µs | 17.2 kHz |
32 | 26 µs | 32 µs | 36 µs | 30.3 kHz |
16 | 13 µs | 16 µs | 20 µs | 55.5 kHz |
8 | 6.5 µs | 12 µs | 20 µs | 71.4 kHz |
4 | 3 µs | 8 µs | 12 µs | 100 kHz |
2 | 1.5 µs | 4 µs | 12 µs | 125 kHz |
Il faut toutefois mettre l’accent sur quelque chose d’important : plus la vitesse de l’ADC est élevée, plus la précision est faible, mais, pour une précision maximale, la vitesse maximale est de 15 kHz, un rapport de 64 fait donc déjà perdre en précision.
Ce tableau, en plus du registre ADCSRA, introduit un autre registre que nous n’avons pas encore vu, il s’agit de ADCSRB, qui lui est complémentaire, ce registre est grandement inutilisé :
7 | 6 | 5 | 4 | 3 | 2 | 1 | 0 |
---|---|---|---|---|---|---|---|
/ | ACME | / | / | MUX51 | ADTS2 | ADTS1 | ADTS0 |
Les trois bits ADTS[2-0] vont grandement nous intéresser, puisqu’ils permettent de sélectionner par quoi sera déclenché le convertisseur ; nous verrons ici simplement le cas où ADTS = 0b000, puisque les autres se réfèrent à des notions trop éloignées du sujet de l’article. Ce cas est celui où le convertisseur sera déclenché par ADIF, c’est-à-dire le registre activé en fin de conversion, permettant à notre ADC de boucler pour toujours, lisant continuellement la valeur de notre pin. Bien, il est l’heure de passer à la pratique, voici un morceau de code permettant de lire la valeur d’un pin en utilisant le free-running mode :
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 |
void setup() { // Désactivons l'ADC pour l'instant cbi(ADCSRA, ADEN); // Activation du free-running mode ADCSRB = 0x00; // On sélectionne notre pin (A0) ADMUX |= 0 & 0x07; // Important : préciser la référence de tension (ici, équivalent de DEFAULT) ADMUX |= (1 << REFS0); // Choix de la division du prescaler (ici, facteur 8) cbi(ADCSRA, ADPS2); sbi(ADCSRA, ADPS1); sbi(ADCSRA, ADPS0); // Ce bit doit être passé à 1 pour prendre en compte le free-running sbi(ADCSRA, ADATE); // Demande d'une interruption à la fin de la conversion sbi(ADCSRA, ADIE); // Réactivons l'ADC sbi(ADCSRA, ADEN); // On lance la première conversion sbi(ADCSRA, ADSC); // Ne pas oublier d'activer les interruptions matérielles sei(); } void loop() { // Code à exécuter lors d'une mesure analogique int value = (ADCH << 8) | ADCL; } |
INITIALISATION
Low Frequencies
Ceci est le code permettant d'initialiser l'AtMega pour lire la broche PF0(ADC0) et convertir la tension entrée en données numériques exploitables pour l'AtMega. Pour choisir la broche PF0(ADC0), il faut set ADMUX à 0x00.
void ad_init_LF(){ // Sélectionne un canal
ADCSRA &= ~(1 << ADEN); // Convertisseur désactivé
ADCSRB = 0x00;
ADCSRA |= (1 << ADPS2) | (1 << ADPS1) ;// | (1 << ADPS0)
ADCSRA &= ~(1 << ADPS0); // Division de fréquence 64 => 125KHz
// Comme vous pouvez le voir dans ce morceau de code, au niveau de l’AtMega, le rapport de division du prescaler est configuré par trois bits (ADPS[0-2]) du registre ADCSRA
ADCSRA &= ~(1 << ADATE); // Mode conversion unique
ADMUX &= 0x00;
ADMUX |= (1 << REFS0) | (1 << ADLAR); // Référence de mesure AVCC et de mettre les 1024 niveaux de tensions entre 0V et VCC+0.3V car REFS0 = 1. ADLAR = 1 donc on a 8 bits d'info
ADMUX &= ~(1 << REFS1);
// Selection du canal // Selection de ADC0 mais ensuite on séléctionnera ADC1 et ADC4 pour les autres passe-bandes
ADCSRA |= (1 << ADEN); // Convertisseur activé
}
Mid Frequencies
void ad_init_MF(){ // Sélectionne un canal
ADCSRA &= ~(1 << ADEN); // Convertisseur désactivé
ADCSRB = 0x00;
ADCSRA |= (1 << ADPS2) | (1 << ADPS1) ;// | (1 << ADPS0)
ADCSRA &= ~(1 << ADPS0); // Division de fréquence 64 => 125KHz
// Comme vous pouvez le voir dans ce morceau de code, au niveau de l’AtMega, le rapport de division du prescaler est configuré par trois bits (ADPS[0-2]) du registre ADCSRA
ADCSRA &= ~(1 << ADATE); // Mode conversion unique
ADMUX &= 0x00;
ADMUX |= (1 << REFS0) | (1 << ADLAR); // Référence de mesure AVCC et de mettre les 1024 niveaux de tensions entre 0V et VCC+0.3V car REFS0 = 1. ADLAR = 1 donc on a 8 bits d'info
ADMUX &= ~(1 << REFS1);
// Selection du canal // Selection de ADC0 mais ensuite on séléctionnera ADC1 et ADC4 pour les autres passe-bandes
ADMUX |= 0b00000010;
ADCSRA |= (1 << ADEN); // Convertisseur activé
}
High Frequencies
void ad_init_HF(){ // Sélectionne un canal
ADCSRA &= ~(1 << ADEN); // Convertisseur désactivé
ADCSRB = 0x00;
ADCSRA |= (1 << ADPS2) | (1 << ADPS1) ;// | (1 << ADPS0)
ADCSRA &= ~(1 << ADPS0); // Division de fréquence 64 => 125KHz
// Comme vous pouvez le voir dans ce morceau de code, au niveau de l’AtMega, le rapport de division du prescaler est configuré par trois bits (ADPS[0-2]) du registre ADCSRA
ADCSRA &= ~(1 << ADATE); // Mode conversion unique
ADMUX &= 0x00;
ADMUX |= (1 << REFS0) | (1 << ADLAR); // Référence de mesure AVCC et de mettre les 1024 niveaux de tensions entre 0V et VCC+0.3V car REFS0 = 1. ADLAR = 1 donc on a 8 bits d'info
ADMUX &= ~(1 << REFS1); // Selection du canal // Selection de ADC0 mais ensuite on séléctionnera ADC1 et ADC4 pour les autres passe-bandes
ADMUX |= 0b00000100;
ADCSRA |= (1 << ADEN); // Convertisseur activé
}
Les 3 broches d'entrée de l'AtMega de chaque Bande de fréquences
PF0 est ADC0 ==> MUX[0-3] = 0b0000 ; PF1 est ADC1 ==> MUX[0-3] = 0b0001 ; PF4 est ADC4 ==> MUX[0-3] = 0b0100 ;
Capture et conversion en numérique
unsigned int ad_capture(void) { // Acquisition de tension
ADCSRA |= (1 << ADSC); // Début de conversion
while (bit_is_set(ADCSRA, ADSC)); // Attente de la fin de conversion
return ADCH; // Résultat sur 8 bits car ADLAR=1
}
Lecture du son numérique
Ce code lit les basses fréquences. La fonction ad_init_LF vient initialiser les registres et chercher le PIN sur lequel convertir l'entrée analogique. La fonction ad_capture() va capturer la tension pendant un durée définie par les bits ADSP de l'AtMega et la transformer en une valeur numérique basée sur 1024 niveaux définis par les bits REFS.
void read_sound_LF() {
ad_init_LF();
unsigned int db = MAX_INT - ad_capture();
PORTB &= ~(0b00010000);
PORTD &= ~(0b11010000);
if(db > 20){
PORTD |= 0b00010000;
}
if (db > 40){
PORTD |= 0b01010000;
}
if (db > 60){
PORTD |= 0b11010000;
}
if (db > 80){
PORTD |= 0b11010000;
PORTB |= 0b00010000;
}
}
Si les db dépasse un certain certains seuils, la LED s'allume. Ces seuils sont arbitraires mais méritent d'être redesigner par rapport au circuit complet. Ce sont pas réellement des dB qui sortent de l'AtMega mais des "niveaux", il faut donc savoir quelles sont les valeurs maximales et minimales sortantes.
void read_sound_MF() {
ad_init_MF();
unsigned int db = MAX_INT - ad_capture();
PORTC &= ~(0b11000000);
PORTD &= ~(0b00000011);
if(db > 140){
PORTD |= 0b00000001;
}
if (db > 160){
PORTD |= 0b00000011;
}
if (db > 170){
PORTD |= 0b00000011;
PORTC |= 0b01000000;
}
if (db > 180){
PORTD |= 0b00000011;
PORTC |= 0b11000000;
}
}
void read_sound_HF() {
ad_init_HF();
unsigned int db = MAX_INT - ad_capture();
PORTB &= ~(0b00100000);
PORTF &= ~(0b11100000);
if(db > 140){
PORTF |= 0b00100000;
}
if (db > 160){
PORTF |= 0b01100000;
}
if (db > 190){
PORTF |= 0b11100000;
}
if (db > 240){
PORTF |= 0b11100000;
PORTB |= 0b00100000;
}
}
LED | Seuil d'allumage |
---|---|
1 | 20dB |
2 | 40dB |
3 | 60dB |
4 | 80dB |
GIT
https://archives.plil.fr/lgrevin/projet-se3-lilian-pierre
BIBLIOGRAPHIE et SOURCES
https://makezine.com/projects/audio-spectrum-analyzer/
https://zestedesavoir.com/articles/2106/arduino-les-secrets-de-lanalogique/
https://www.arnabkumardas.com/arduino-tutorial/adc-register-description/