COUPE ROBOTIQUE DES IUT : Différence entre versions
Ligne 989 : | Ligne 989 : | ||
=={{Vert|Caméra }}== | =={{Vert|Caméra }}== | ||
==={{Bleu|Choix camera}}=== | ==={{Bleu|Choix camera}}=== | ||
+ | Nous avons testé 3 cameras différentes, la PiCam, la CMUCam3 | ||
+ | et la CMUCam5 Pixy | ||
+ | [[Fichier:Cmu Cam 3.jpeg|vignette|centré]] | ||
+ | [[Fichier:Cmu Cam 5.jpeg|vignette|vignette|centré]] | ||
+ | <br> |
Version du 24 janvier 2018 à 11:48
Sommaire:
- Présentation
- Présentation du projet
- Cahier des charges fonctionnel
- Solution techniques
- Étude et Réalisation des Différentes Parties
- Alimentation et Régulation
- Batterie
- Moteurs de roues
- Régulation de la tension d'alimentation
- Contrôle des moteurs CC par un dual H-Bridge (L298P)
- Principe de fonctionnement d'un H-Bridge (PONT-H)
- Le composant L298N
- Tests effectués avec un SHIELD Arduino (L298P)
- Mode PWM
- Positionnement du robot :Explication du principe
- Approximation par des segments de droites
- Estimation de la position du robot
- Étude et Réalisation Carte Encodeurs
- Roue codeuse
- Valeur reçu par les capteurs
- Interruption et timer
- suivons une ligne droite
- suivons une consigne
- Détection de balles de tennis
- Caméra
- Choix Caméra
- Caméra CMU cam 5
- Programme de gestion du cap
- Mise à l’arrêt du robot et le perçage du ballon
- Capteur de "Mise à l’arrêt" du robot
- Système "perçage du ballon"
- Réalisation Carte des "Entrées et Sorties" et du pont H
- Objectifs et Composants utilisés
- Schéma électrique de la carte (Eagle)
- Routage et Correspondance des pins (Eagle)
- La commande des Moteurs de Roues(L298N H-Bridge)
- Les signaux d'entrée et de sortie
- La carte final et les connecteurs
- Problèmes rencontrés :
- Alimentation et Régulation
- Code complet
- Vidéo de Démonstration
- Bibliographie/références
Sommaire
Présentation
Présentation du projet
Ce projet consiste à réaliser un robot pour participer à la coupe de robotique des GEII.Pour fabriquer ce robot nous devons respecter un cahier des charges défini par le : http:(...) Le principe de ce concours est d’envoyer un maximum de balles de tennis dans le camp adverse, sans entrer dans le camp adverse et sans jamais contrôler plus d’une balle à la fois
Cahier des charges fonctionnel
Schéma fonctionnel de degré II
Schéma fonctionnelle 1er degres
Solutions techniques
Pour fabriquer ce robot nous avions plusieurs choix pour procéder à la détection des balles de tennis ainsi que le contrôle et direction du robot vers la ligne noir ... , ainsi qu'au guidage vers la position (zone) du ballon . Nous avons choisi pour guider le robot un système hybride composé de roues codeuses et d'une caméra CMUCam Pixy, spécialisée dans la reconnaissance d'objets. La roue codeuse permettra d'effectuer le début du parcours, et sera remplacée par la caméra, plus précise, une fois la distance de détection atteinte. L'évitement des obstacles sera assuré par trois capteurs infrarouges SHARP GP2Y0A21YK0F, permettant la détection d'objets jusqu'à 80 centimètres de distance, qui, placés à l'avant du robot, permettront l'esquive si un obstacle est rencontré.
Etude et Réalisation des Differentes Parties
Alimentation et Régulation
Batterie
La batterie est imposée :
Tension | 12 V |
Capacité | 7 Ah |
Moteurs de roues
Les moteurs sont imposés.
- Caracteristique du moteur:
Marque | Dunkermotoren G 42*25 |
Tension | 15V |
In | 1.45 A |
Ifm | 10.9A |
Rpm | 3300 tr/mn |
- Tests en conditions réeles sur les moteurs montés sur le robot ( U=12V ) :
Résistance électrique | 4 ohm |
Courant à vide | 0.28 A |
Courant en charge nominale robot | 0.7A |
Courant en charge au démarage robot | 1.6A |
- Consommation maximum en courant pour les 2 moteurs du robot :
Imax = 2*1.6A = 3.2A
Régulation de la tension d'alimentation
- Les besoins :
Pour Arduino MEGA:
Caractéristiques techniques :
Operating Voltage | 5V |
Input Voltage (recommended) | 7-12V |
Input Voltage (limit) | 6-20V |
Total output current MAX | 800mA |
On constate qu'il est possible d'alimenter la carte Arduino MEGA directement avec la tension de la batterie ( 12.8V chargée).
Ce n'est toutefois pas recommandé, car le régulateur intégré dans l'Arduino chaufferait, ce qui pourrait endommager le microcontrôleur.
Solutions Alimentation Arduino MEGA:
- Tension d'alimentation inférieure à 12 V
- Tension régulé de 5V qu'on fait venir directement sur les pattes VCC d'Arduino:
Le courant maximum requis: 800mA
Pour les Moteurs de Roues :
- Tension maximum requise: 15V
- Courant maximum requis: 3.2A
- Utilisation d'un Convertisseur DC/DC
Nous avons utilisé une carte faite par des étudiants des années précédents.
Cette carte contient deux régulateurs à découpage:
- LM2596S fournit 5V ( 3A max )
- XL6009E1 qui fournit 12V ( 4A max )
Les régulateurs fournissent largement le nécessaire en tension et courant.
Pour le semestre suivant nous prévoyons à réaliser une autre carte d'alimentation, qui sera plus compacte.
Contrôle des moteurs CC par un dual H-Bridge (L298P)
Principe de fonctionnement d'un H-Bridge (PONT-H)
Le pont-H est une structure utilisée en électronique de puissance pour:
- controle moteurs
- convertisseurs et hacheurs
- onduleurs
- Principe: On active les commutateurs avec differents cominaisons pour obtenir le branchement voulu. Le courant va circuler dans un sens ou dans l'autre dans le moteur, ce qui va permettre d'inverser les sens de rotation du moteur. Avec le pont-H on peut également varier la vitesse en modulant la tension aux bornes du moteur.
Combinaisons de commutateurs possibles pour commander un moteur DC:
Sens + | Fermer A et D |
Sens - | Fermer B et C |
Freinage magnétique | A et C / B et D |
Arret libre | A,B,C,D ouverts |
Autres combinaison | INTERDITES |
Le composant L298N
Nous allons utiliser pour notre robot le composant L298N (traversant) qui a le meme principe de fonctionement que celui en CMS (L298P). Dans la figure suivante on peux voir le cablage du composant, les signaux de commande et les sorties d'alimentation MOTEUR. Dans le tableau nous avons les 4 modes possibles en actionant les entrées logique C et D ainsi que Venable (PWM) pour varier la tension d'alimentation des moteurs (0-12V).
Tests éffectués avec un SHIELD Arduino (L298P) (
Pour commander les moteurs nous allons utiliser le pont H L298P [ datasheet].
Le composant est ci-dessous:
Pour faire des tests nous avons utlisé le Motor Shield For Arduino. En connectant ce shield à l'arduino nous pouvons commander les deux moteurs (commande du sens et de la vitesse).
- PWM
Nous allons utiliser le shield en mode PWM, on placera donc les jumpers en conséquence.
- Borne du moteur
Nous avons deux bornes (bleues) pour connecter les moteurs CC. Les connecteurs mâles derrière sont identiques à celui des bornes bleues.
- PWRIN
Les moteurs peuvent être alimentés par une alimentation externe lorsque le courant du moteur dépasse les limites fournies par l'Arduino (Il est conseillé de séparer les alimentations d’Arduino et des moteurs). Le changement entre la puissance externe et l'Arduino est mis en œuvre par deux jumpers .
PWRIN: Alimentation externe.
VIN: Alimentation du Arduino.
On placera donc les jumpers d’alimentation sur PWRIN.
On doit avoir quelque chose comme cela:
- Signal de contrôle Tableau de vérité
E1 | M1 | E2 | M2 | Texte de l’en-tête | |
---|---|---|---|---|---|
L | X | Moteur 1 désactivé | L | X | Moteur 2 désactivé |
H | H | Moteur 1 en arrière | H | H | Moteur 2 en arrière |
H | L | Moteur 1 en avant | H | L | Moteur 2 en avant |
PWM | X | Contrôle vitesse PWM | PWM | X | Contrôle vitesse PWM |
NOTE:
H: Niveau haut
L: Niveau bas
X: N'importe quel niveau.
Mode PWM
Commande | Moteur | Pin Arduino | Pin Atmega328p | Signification |
---|---|---|---|---|
M1 | Gauche | 4 | PD4 | Contrôle du sens de rotation |
E1 (PWM) | Gauche | 5 | PD5 | Contrôle de la vitesse de rotation |
M2 | Droit | 7 | PD7 | Contrôle du sens de rotation |
E2 (PWM) | Droit | 6 | PD6 | Contrôle de la vitesse de rotation |
- Exemple de code
Nous allons gérer les moteurs par des signaux PWM (-255 a 255), le signe moins (-) indique que le moteur fonctionne en marche arrière, et le signe plus (+) qu'il fonctionne en marche avant. Ce code nous permet de gérer les deux moteurs par la fonction setVitesse(vG,vD). Dans la suite nous allons l'utiliser pour gérer le déplacement du robot.
Code exemple
#include <avr/io.h>//Librairie AVR
#define topPWM 255 //Valeur Max du PWM (8bits)
void initMoteur()//PWM sur PD5 et PD6
{
//fpwm = fq / (topPWM * p)
//fpwmi:fréquence du PWM
//fq:fréquence du quark
//topPWM: valeur maximum du PWM
//p:prédiviseur
//Déclaration de sorties
DDRD |= (1 << PD4);//Sens du moteur Gauche
DDRD |= (1 << PD5);//PWM du moteur Gauche
DDRD |= (1 << PD7);//Sens du moteur Droit
DDRD |= (1 << PD6);//PWM du moteur Droit
TCCR0B |= (1 << CS00) | (1 << CS01);//Prédiviseur P=64
TCCR0A |= (1 << WGM00) | (1 << WGM01);//Mode FAST PWM
TCCR0A |= 1 << COM0A1;//PWM sur OC0A
TCCR0A |= 1 << COM0B1;//PWM sur OC0B
OCR0A = 0;//Valeur de comparaison pour A --> PD6
OCR0B = 0;//Valeur de comparaison pour B --> PD5
}
void setMoteurG(int16_t vit)//fonction pour gérer le moteur gauche
{
if (vit < 0)
{
vit = -vit;
PORTD |= (1 << PD4);//Moteur Gauche en arrière
}
else PORTD &= ~ (1 << PD4);//Moteur Gauche en avant
if (vit > topPWM) vit = topPWM;//Si jamais on met une valeur supérieure à 255, la vitesse maximum sera 255
OCR0B = vit;//Action sur le PWM --> PD5
}
void setMoteurD(int16_t vit)//fonction pour gérer le moteur droit
{
if (vit < 0)
{
vit = -vit;
PORTD |= (1 << PD7);//Moteur Droit en arrière
}
else PORTD &= ~ (1 << PD7);//Moteur Droit en avant
if (vit > topPWM) vit = topPWM;//Si jamais on met une valeur supérieure a 255, la vitesse maximum sera 255
OCR0A = vit;//Action sur le PWM --> PD6
}
void setVitesse(int16_t vG, int16_t vD)//cette fonction nous permet gérer les deux moteurs avec "une seule" ligne
{
setMoteurD(vD);
setMoteurG(vG);
}
int main()
{
initMoteur();
while(1)
{
setVitesse(100,100);//Exemple d’utilisation
}
}
Positionnement du robot: Explication du principe
Approximation par des segments de droites
Le positionnement du robot est obtenu par odométrie, c'est à dire que la position est obtenue par intégration de petits déplacements. L'intérêt de l'odométrie est qu'elle est assez simple à mettre en oeuvre et qu'elle est fiable. Par contre, quand on intègre les déplacements, on intègre aussi l'erreur ce qui fait que l'erreur de position croît avec le temps.
Entre deux lectures, on peut savoir de combien s'est déplacée chaque roue codeuse et il faut à partir de cela en déduire la position du robot.
Appelons ∆d et ∆g les distances (en mm) parcourues respectivement par les roues droites et gauches entre deux lectures des LM soit un intervalle de temps Te. Connaissant la position du robot à l'instant n-1, on cherche la pose à l'instant n.
On a donc : ∆moy_n = (∆d_n + ∆g_n)/2
∆dif_n = ∆d_n - ∆g_n
∆x_n = ∆moy_n cos(theta_n-1)
∆y_n = ∆moy_n sin(theta_n-1)
∆theta_n = ∆dif_n/L
x_n = x_n-1 + ∆x_n
y_n = y_n-1 + ∆y_n
theta_n = theta_n-1 + ∆theta_n
Estimation de la position du robot
Etude et Réalisation Carte Encodeurs
L'odométrie nous permettra d'estimer la position du robot en mouvement, des le début et jusqu'à la détection de la balise par la caméra. C'est à partir de la mesure des déplacements des roues, qu'on pourra reconstituer le mouvement du robot. En partant d'une position initiale connue et en intégrant les déplacements mesurés, on peut ainsi calculer à chaque instant la position courante du véhicule.
Pour mesurer le déplacement des roues, nous allons utiliser un encodeur, monté sur l'axe de chaque roue.
- ENCODEUR DE BASE:
Ce montage basique permet de mesurer la vitesse de rotation à partir de la fréquence, mais il ne permet pas de connaître le sens de rotation.
- ENCODEUR EN QUADRATURE:
Celui-ci nous permettra de connaître à la fois le sens et la vitesse des roues. Il est composé d’un disque rotatif, 1 led infrarouge et 2 capteurs optique décalé un par rapport à l’autre de 90°. C’est justement ce décalage la qui va nous permettre de connaître le sens de rotation de la roue. Suivant le sens de rotation, nous auront deux signaux déphasés en avance/retard de 90°. La vitesse sera déterminé en fonction de la fréquence.
Les signaux de sortie à l’oscilloscope
- CAPTEUR TCUT 1300. Nous allons utiliser ce capteur il nous permettra d'avoir le sens et la vitesse de chaque roue.
The TCUT1300X01 is a compact transmissive sensor that includes an infrared emitter and two phototransistor detectors, located face-to-face in a surface mount package.
- Schéma et dimensionnement des composants
Emiter: If = 10mA (Vf = 1.2V), Re = (5V-Vf)/If = 380 ohm.
Collector: Ic sat = 0.4mA (If = 10mA), Rmin = Vce /Ic = 12.5 kohm. Vce sat = 0.4V max (pour Ic sat = 0.4mA, If = 10mA ). On prendra Rc = 15 kohm.
Nous devons obtenir les signaux correspondents:
Présence dent | Signal de sortie = 0 V |
Absence dent | Signal de sortie = 5 V |
- Schéma et routage en Eagle
- Réalisation carte (2 exemplaires) et installation
- Carte simple face CMS.
TEST RÉALISÉ SUR LA CARTE FINIE:
Présence dent | G1,G2,D1,D2 = 0 V |
Absence dent | G1,G2,D1,D2 = 4.8 V |
Conclusion: Les valeurs sont cohérentes et exploitables.
//
Roue codeuse
Pour determiner la position du robot nous allons utiliser deux Roues Codeuses et deux capteurs (TCUT 1300). Nous avons créé le roue sur le logiciel GEFAO et fabrication avec la machine charlyrobot. Vous trouverez ici le fichier de la roue (Fichier:RoueCodeuse.zip).
Valeurs recu par les capteurs
Cette roue sera placer sur l'arbre des motors et sur les deux capteurs, lorsque le moteur est en train de tourner les capteurs va nous envoyer une signal qui se resamblera à:
- Si la roue en avant
Il faut remarquer que la signal 1 est en avance. Donc sur chaque front montant de la signal 1, la signal 2 est à 0. Et sur chanque front descendate de la signal 1, la signal 2 est à 1. Ce raisonnement sera utiliser dans le code du proogramme.
- Si la roue en arriere
Il faut remarquer que la signal 1 est en retard. Donc sur chaque front montant de la signal 1, la signal 2 est à 1. Et sur chanque front descendate de la signal 1, la signal 2 est à 0. Ce raisonnement sera aussi utiliser dans le code du proogramme.
Il faut remarquer que si le robot avance, la roue codeuse est en arriere et si le robot recule la roue codeuse est en avant. Ce effet est du aux engranages qui sont sur les moteurs et la roue.
Interruptions et timer
- Interruptions INT0 et INT1: Code example
Dans cette partie nous allons utiliser les interruptions et les timers. On utilisera les inturruptions (INT0 et INT1) pour dechancher un code a chaque fois que les dents de la roue codeuse travers le flux lumineuse du capteurs (sur le front montant et front descendant). Nous allons aussi utliser un timer qui sera dechanché sur un temps assez petit ( <10ms) pour calculer un deltaX, un deltaY et un detaTeta qui nous permetront de determiner la position du robot a chaque instante.
Code example
#include <avr/io.h>//Libreries AVR
volatile int8_t NbDentsD = 0; //Nombre de dents captés par le capteur (TCUT 1300) de la roue codeuse Droit
volatile int8_t NbDentsG = 0; //Nombre de dents captés par le capteur (TCUT 1300) de la Roue Codeuse Gauge
ISR(INT0_vect)
{
//Sur chaque roue nous avons un emeteur et deux recepteurs, ces deux seront lus sur le pates PD2 et PB0
uint8_t Pin2 = PIND & (1 << PD2); //INTERRUPTION INT0 sur PD2
uint8_t Pin8 = PINB & (1 << PB0);
if (Pin2 != 0)
{
if (Pin8 == 0) NbDentsD++;//Si la roue droit avance dont on aditionne le dents
else NbDentsD--;//sinon on soustrait les dents, car cela signifie que le roue est en arrière
}
else if (Pin2 == 0)
{
if (Pin8 != 0) NbDentsD++;
else NbDentsD--;
}
}
ISR(INT1_vect)
{
uint8_t Pin3 = PIND & (1 << PD3); //INTERRUPTION INT1 sur PD3
uint8_t Pin9 = PINB & (1 << PB1);
if (Pin3 != 0)
{
if (Pin9 == 0) NbDentsG++;//Si la roue gauge avance dont on aditionne le dents
else {
NbDentsG--;//sinon on soustrait les dents, car cela signifie que le roue est en arrière
}
}
else if (Pin3 == 0)
{
if (Pin9 != 0) NbDentsG++;
else {
NbDentsG--;
}
}
}
void int0()
{
//CONFIGURATION INTERRUPTION INT0
//Pin2 --> PD2--> INT0 entrée
EIMSK |= 1 << INT0; //Autorisation interruption INT0
EICRA |= 1 << ISC00; //Activation sur changement d'etat sur PD2
}
void int1()
{
//CONFIGURATION INTERRUPTION INT1
//PIN3 -->PD3--> INT1 entrée
EIMSK |= 1 << INT1; //Autorisation interruption INT1
EICRA |= 1 << ISC10; //Activation sur changement d'etat sur PD3
}
void sei_interruption()
{
sei(); //Autorisation global d'interruptions
}
int main()
{
int0();//Interruption 0
int1();//Interruption 1
sei_interruption();//Autorisation d'interruptions
while(1)
{
}
}
- TIMER 1
En faite nous allons calculer la position du robot en ajoutant de petits deltas de postition lequels seront calculer sur un temps suffisamment petit ( <20ms) pour eviter que l'erreur soit grand. On utilisera un drapeau pour eviter de faire de calculs dans le ISR(TIMER1_COMPA_vect), nous allons les faire plutot sur while(1), cela nous permettra de reduire l'erreur.
Code example
#include <avr/io.h>//Libreries AVR
void timer1()//CONFIGURATION DU TIMER1
{
//Tt1=(n*p)/fq
//Tt1: periode du timer
//n=valuer de comparaison (OCR1A)
//p:prediviseur
//fq:frequence du quark (arduino uno 16Mhz)
TCCR1B |= (1 << CS10) | (1 << CS11); //p=64
TCCR1B |= 1 << WGM12; //RAZ mode CTC
TIMSK1 |= 1 << OCIE1A; //Autorisation d'interruption de comparaison A
OCR1A = 7500; //Comparaison n=7500
}
ISR(TIMER1_COMPA_vect)
{
newCalc = true;//drapeau
}
int main()
{
timer1();
while (1)
{
if (newCalc == true)
{
//Calculs de la position
newCalc = false;
}
}
}
Suivons une ligne droite
On sait que le moteurs n'ont pas la vitesse, donc il faut essayer de faire cela par code. On va donc faire setVitesse(Vmax * cos(Teta + PI / 4.0), Vmax * sin(Teta + PI / 4.0)), les cos et sin nous permettra jouer sur la vitesse et de cette facon faire que le robot suis une ligne droite. Il faut remarquer qu'il ne va pas s'arreter.
Code de Suivons une ligne droite
#include <avr/io.h>//Libreries AVR
#include <math.h>//Libreries Math pour les calculs de sinus, cosinus, etc;
#define topPWM 255 //Valeur Max du PWM (8bits)
float d = 51.2; //Rayon des roues
float Delta_Dent = PI * d / 60.0; //Espacement entre chaque dent de la roue.
float L = 230.0; //Distence entre les centre des roues.
volatile int8_t NbDentsD = 0; //Nombre de dents captés par le capteur TCUT 1300 de la roue codeuse Droit
volatile int8_t NbDentsG = 0; //Nombre de dents captés par le capteur TCUT 1300 de la Roue Codeuse Gauge
float Delta_D = 0;
float Delta_G = 0;
float Delta_moy = 0;
float Delta_dif = 0;
float Delta_teta = 0;
float Delta_X = 0;
float Delta_Y = 0;
float Position_X = 0;
float Position_X_precendent = 0;
float Position_Y = 0;
float Position_Y_precendent = 0;
float Teta = 0;
float Teta_precedent = 0;
float Teta_consigne = 0; //Angle qu'on veux attaindre
float X_consigne = 0; //Position en X qu'on veux attaindre
float Y_consigne = 0; //Position en Y qu'on veux attaindre
volatile boolean newCalc = true;//Drapeau qui nous permetra de
float Vmax = 0;
void int0()
{
//CONFIGURATION INTERRUPTION INT0
//Pin2 --> PD2--> INT0 entrée
EIMSK |= 1 << INT0; //Autorisation interruption INT0
EICRA |= 1 << ISC00; //Activation sur changement d'etat sur PD2
}
void int1()
{
//CONFIGURATION INTERRUPTION INT1
//PIN3 -->PD3--> INT1 entrée
EIMSK |= 1 << INT1; //Autorisation interruption INT1
EICRA |= 1 << ISC10; //Activation sur changement d'etat sur PD3
}
void timer1()//CONFIGURATION DU TIMER1
{
//Tt1=(n*p)/fq
//Tt1: periode du timer
//n=valuer de comparaison (OCR1A)
//p:prediviseur
//fq:frequence du quark (arduino uno 16Mhz)
TCCR1B |= (1 << CS10) | (1 << CS11); //p=64
TCCR1B |= 1 << WGM12; //RAZ mode CTC
TIMSK1 |= 1 << OCIE1A; //Autorisation d'interruption de comparaison A
OCR1A = 7500; //Comparaison n=7500
}
void initMoteur()//PWM sur PD5 et PD6
{
//fpwm = fq / (topPWM * p)
//fpwmi:frequence du PWM
//fq:frequance du quark
//topPWM: valeur maximun du PWM
//p:prediviseur
//Declaration de sorties
DDRD |= (1 << PD4);//Sens du motor Gauge
DDRD |= (1 << PD5);//PWM du motor Gauge
DDRD |= (1 << PD7);//Sens du motor Droit
DDRD |= (1 << PD6);//PWM du motor Droit
TCCR0B |= (1 << CS00) | (1 << CS01);//Prediviseur P=64
TCCR0A |= (1 << WGM00) | (1 << WGM01);//Mode FAST PWM
TCCR0A |= 1 << COM0A1;//PWM sur OC0A
TCCR0A |= 1 << COM0B1;//PWM sur OC0B
OCR0A = 0;//Valeur de comparaison pour A --> PD6
OCR0B = 0;//Valeur de comparaison pour B --> PD5
}
void sei_interruption()
{
sei(); //Autorisation global d'interruptions
}
ISR(INT0_vect)
{
//Sur chaque roue nous avons un emeteur et deux recepteurs, ces deux seront lu sur le pate PD2 et PB0
uint8_t Pin2 = PIND & (1 << PD2); //INTERRUPTION INT0 sur PD2
uint8_t Pin8 = PINB & (1 << PB0);
if (Pin2 != 0)
{
if (Pin8 == 0) NbDentsD++;//Si la roue droit avance dont on aditionne le dents
else NbDentsD--;//sinon on soustrait les dents, car cela signifie que le roue est en arrière
}
else if (Pin2 == 0)
{
if (Pin8 != 0) NbDentsD++;
else NbDentsD--;
}
}
ISR(INT1_vect)
{
uint8_t Pin3 = PIND & (1 << PD3); //INTERRUPTION INT1 sur PD3
uint8_t Pin9 = PINB & (1 << PB1);
if (Pin3 != 0)
{
if (Pin9 == 0) NbDentsG++;//Si la roue gauge avance dont on aditionne le dents
else {
NbDentsG--;//sinon on soustrait les dents, car cela signifie que le roue est en arrière
}
}
else if (Pin3 == 0)
{
if (Pin9 != 0) NbDentsG++;
else {
NbDentsG--;
}
}
}
ISR(TIMER1_COMPA_vect)
{
newCalc = true;
}
void setMoteurG(int16_t vit)//fontion pour gener le moteur gauge
{
if (vit < 0)
{
vit = -vit;
PORTD |= (1 << PD4);//Moteur Gauge en arrière
}
else PORTD &= ~ (1 << PD4);//Moteur Gauge en avant
if (vit > topPWM) vit = topPWM;//Si jamais on met une vauleur superior a 255, la vitesse maximun sera 255
OCR0B = vit;//Action sur le PWM --> PD5
}
void setMoteurD(int16_t vit)//fontion pour gener le moteur droit
{
if (vit < 0)
{
vit = -vit;
PORTD |= (1 << PD7);//Moteur Droite en arrière
}
else PORTD &= ~ (1 << PD7);//Moteur Droite en avant
if (vit > topPWM) vit = topPWM;//Si jamais on met une vauleur superior a 255, la vitesse maximun sera 255
OCR0A = vit;//Action sur le PWM --> PD6
}
void setVitesse(int16_t vG, int16_t vD)//cette fontion nous permet gerer les deux moteurs avec "une seule" ligne
{
setMoteurD(vD);
setMoteurG(vG);
}
int main()
{
int0();//Interruption 0
int1();//Interruption 1
timer1();//Timer 1
initMoteur();//PWM
sei_interruption();//Autorisation d'interruptions
while (1)//Boucle infinie
{
if (newCalc == true)
{
Delta_D = (NbDentsD * Delta_Dent);//pas de la roue droit à chaque declanchement du TIMER 1.
Delta_G = (NbDentsG * Delta_Dent); //pas de la roue gauge à chaque declanchement du TIMER 1.
NbDentsG = 0;//Remise à zéro à chaque declanchement du TIMER 1.
NbDentsD = 0;//Remise à zéro sà chaque declanchement du TIMER 1.
Delta_moy = (1 / 2.0) * (Delta_D + Delta_G);//pas du robot à chaque declanchement du TIMER 1.
Delta_dif = Delta_D - Delta_G;
Delta_teta = Delta_dif / L;//Calcul de Delta_teta en rad à chaque declanchement du TIMER 1.
Delta_X = Delta_moy * cos(Teta_precedent);//pas sur X du deplacement du robot à chaque declanchement du TIMER 1.
Delta_Y = Delta_moy * sin(Teta_precedent);//pas sur Y du deplacement du robot à chaque declanchement du TIMER 1.
Teta = Teta_precedent + Delta_teta;//Angle de desviation par rapport à l'axe X du robot.
Teta_precedent = Teta;//Angle de desviation precedent par rapport à l'axe X du robot.
Position_X = Position_X_precendent + Delta_X;//Potition actuelle du robot en X
Position_X_precendent = Position_X;//Position precedente du robot en X
Position_Y = Position_Y_precendent + Delta_Y;//Potition actuelle du robot en Y
Position_Y_precendent = Position_Y;//Position precedente du robot en Y
Vmax = 250;//valeur maximun qu'on impose aux moteurs
setVitesse(Vmax * cos(Teta + PI / 4.0), Vmax * sin(Teta + PI / 4.0));
newCalc = false;
}
}
}
Suivons une consigne
Cette fois nous pourrons donner une valeur en X ey Y et le robot sera capable d'arriver au point et s'arreté. La valeurs doit etre donnée en cm.
ws2812_config.h
//Code precedente (suivons un ligne)
int main()
{
//Configurations
while (1)//Boucle infinie
{
if (newCalc == true)
{
//Calculs
X_consigne = 100;//valeur donné en cm
Y_consigne = 50;//valeur donné en cm
Teta_consigne = atan((Y_consigne - Position_Y / 50.0) / (X_consigne - Position_X / 50.0)) - Teta;//Nous divisons par 50 pour avoir les valeur en cm
Vmax = 250;//valeur maximun qu'on impose aux moteurs
setVitesse(Vmax * cos(Teta_consigne + PI / 4.0), Vmax * sin(Teta_consigne + PI / 4.0));
newCalc = false;
}
}
}
Detection des balles de tennis
Caméra
Choix camera
Nous avons testé 3 cameras différentes, la PiCam, la CMUCam3 et la CMUCam5 Pixy