RPiPico-Holonome2023/Strategie_prise_cerises.c

249 lines
8.8 KiB
C
Raw Normal View History

2023-03-26 14:34:06 +00:00
#include "stdio.h"
#include "Strategie_prise_cerises.h"
#include "Commande_vitesse.h"
#include "Geometrie.h"
#include "Geometrie_robot.h"
#include "Localisation.h"
#include "math.h"
2023-03-18 16:59:15 +00:00
#include "i2c_annexe.h"
// Rotation en rad/s pour accoster les cerises
#define ROTATION_CERISE 0.5f
// Distance la plus éloignée du bord où le robot est capable d'aspirer les cerises en longeant la bodure.
#define MAX_LONGE_MM 240
2023-03-26 14:34:06 +00:00
// Translation en mm/s pour aspirer les cerises
#define TRANSLATION_CERISE 45
2023-03-26 14:34:06 +00:00
2023-03-18 16:59:15 +00:00
void commande_rotation_contacteur_longer_A();
void commande_rotation_contacteur_longer_C();
2023-03-26 14:34:06 +00:00
void commande_translation_longer_vers_A();
void commande_translation_longer_vers_C();
enum longer_direction_t inverser_longe_direction(enum longer_direction_t direction);
2023-03-18 16:59:15 +00:00
double vitesse_accostage_mm_s=100;
/// @brief Fonction pour attraper les cerises sur les supports perpendiculaires à la bordure.
/// Le robot accoste, longe le support cerise vers la bordure, active la turbine, puis longe le support cerise jusqu'à son bout.
/// @param longer_direction : direction dans laquelle se trouve la bordure
/// @return ACTION_EN_COURS ou ACTION_TERMINEE
enum etat_action_t cerise_attraper_bordure(enum longer_direction_t longer_direction, uint32_t step_ms){
2023-03-26 14:34:06 +00:00
enum etat_action_t etat_action = ACTION_EN_COURS;
enum longer_direction_t longer_direction_aspire;
static uint32_t tempo_ms = 0;
static enum {
ATTRAPE_INIT,
ATTRAPE_VERS_BORDURE,
TURBINE_DEMARRAGE,
TURBINE_DEMARRAGE_TEMPO,
ASPIRE_LONGE,
ASPIRE_LIBRE
} etat_attrape=ATTRAPE_INIT;
switch(etat_attrape){
case ATTRAPE_INIT:
etat_attrape=ATTRAPE_VERS_BORDURE;
printf("ATTRAPE_VERS_BORDURE\n");
break;
case ATTRAPE_VERS_BORDURE:
cerise_longer_bord(longer_direction);
if( (longer_direction == LONGER_VERS_A) && (i2c_annexe_get_contacteur_butee_A() == CONTACTEUR_ACTIF) ||
(longer_direction == LONGER_VERS_C) && (i2c_annexe_get_contacteur_butee_C() == CONTACTEUR_ACTIF) ){
etat_attrape = TURBINE_DEMARRAGE;
printf("TURBINE_DEMARRAGE\n");
}
break;
case TURBINE_DEMARRAGE:
i2c_annexe_ferme_porte();
//i2c_annexe_active_turbine();
commande_vitesse_stop();
tempo_ms = 2000;
etat_attrape = TURBINE_DEMARRAGE_TEMPO;
printf("TURBINE_DEMARRAGE_TEMPO\n");
break;
case TURBINE_DEMARRAGE_TEMPO:
if(tempo_ms < step_ms){
etat_attrape = ASPIRE_LONGE;
printf("ASPIRE_LONGE\n");
}
tempo_ms -= step_ms;
break;
case ASPIRE_LONGE:
longer_direction_aspire = inverser_longe_direction(longer_direction);
cerise_longer_bord(longer_direction_aspire);
// La fonction cerise_longer_bord n'est efficace que tant que le robot a ses deux contacteur sur le support
// Le robot n'a les deux contacteurs sur le support que tant qu'il est à moins de 240mm (MAX_LONGE_MM) de la bordure
// En fonction du demi-terrain sur lequel se trouve le robot, on surveille la position en Z pour respecter cette condition
if( ((Localisation_get().y_mm > 1500) && (Localisation_get().y_mm < (3000 - MAX_LONGE_MM) )) ||
((Localisation_get().y_mm < 1500) && (Localisation_get().y_mm > (MAX_LONGE_MM))) ){
etat_attrape = ASPIRE_LIBRE;
printf("ASPIRE_LIBRE\n");
}
break;
case ASPIRE_LIBRE:
commande_vitesse_stop();
etat_action = ACTION_TERMINEE;
break;
}
return etat_action;
}
/// @brief Fonction pour accoster et longer une bordure
/// @param longer_direction : direction dans laquelle le robot va aller une fois le long de la bordure
/// @return ACTION_EN_COURS
enum etat_action_t cerise_longer_bord(enum longer_direction_t longer_direction){
2023-03-26 14:34:06 +00:00
static enum {
LONGE_INIT,
LONGE_VERS_A,
LONGE_VERS_C,
LONGE_COLLE
} etat_longer_bord=LONGE_INIT;
switch (etat_longer_bord){
case LONGE_INIT:
if(longer_direction==LONGER_VERS_A){
etat_longer_bord=LONGE_VERS_A;
}else{
// longer_direction==LONGER_VERS_C
etat_longer_bord=LONGE_VERS_C;
}
break;
case LONGE_VERS_A:
commande_translation_longer_vers_A();
if( (i2c_annexe_get_contacteur_longer_A() == CONTACTEUR_INACTIF) ||
(i2c_annexe_get_contacteur_longer_C() == CONTACTEUR_INACTIF) ){
etat_longer_bord = LONGE_COLLE;
printf("Longer colle\n");
}
break;
case LONGE_VERS_C:
commande_translation_longer_vers_C();
if( (i2c_annexe_get_contacteur_longer_A() == CONTACTEUR_INACTIF) ||
(i2c_annexe_get_contacteur_longer_C() == CONTACTEUR_INACTIF) ){
etat_longer_bord = LONGE_COLLE;
printf("Longer colle\n");
}
break;
case LONGE_COLLE:
if(cerise_accostage() == ACTION_TERMINEE){
etat_longer_bord = LONGE_INIT;
printf("Longer INIT\n");
}
}
return ACTION_EN_COURS;
2023-03-26 14:34:06 +00:00
}
enum etat_action_t cerise_accostage(void){
enum etat_action_t etat_action = ACTION_EN_COURS;
double rotation;
static enum {
CERISE_AVANCE_DROIT,
2023-03-18 16:59:15 +00:00
CERISE_TOURNE_CONTACTEUR_LONGER_A,
CERISE_TOURNE_CONTACTEUR_LONGER_C,
CERISE_ACCOSTE
} etat_accostage=CERISE_AVANCE_DROIT;
switch (etat_accostage)
{
case CERISE_AVANCE_DROIT:
commande_vitesse(vitesse_accostage_mm_s * cos(-M_PI/6), vitesse_accostage_mm_s * sin(-M_PI/6), 0);
2023-03-18 16:59:15 +00:00
if(i2c_annexe_get_contacteur_longer_A() == CONTACTEUR_ACTIF){
etat_accostage=CERISE_TOURNE_CONTACTEUR_LONGER_A;
2023-03-26 14:34:06 +00:00
printf("D ACCOSTE_TOURNE_A\n");
2023-03-18 16:59:15 +00:00
}
if(i2c_annexe_get_contacteur_longer_C() == CONTACTEUR_ACTIF){
etat_accostage=CERISE_TOURNE_CONTACTEUR_LONGER_C;
2023-03-26 14:34:06 +00:00
printf("D ACCOSTE_TOURNE_C\n");
2023-03-18 16:59:15 +00:00
}
if (i2c_annexe_get_contacteur_longer_A() == CONTACTEUR_ACTIF && i2c_annexe_get_contacteur_longer_C() == CONTACTEUR_ACTIF){
etat_accostage=CERISE_ACCOSTE;
2023-03-26 14:34:06 +00:00
printf("D ACCOSTE\n");
2023-03-18 16:59:15 +00:00
}
break;
2023-03-18 16:59:15 +00:00
case CERISE_TOURNE_CONTACTEUR_LONGER_A:
commande_rotation_contacteur_longer_A();
if(i2c_annexe_get_contacteur_longer_A() == CONTACTEUR_INACTIF){
if(i2c_annexe_get_contacteur_longer_C() == CONTACTEUR_INACTIF){
etat_accostage = CERISE_AVANCE_DROIT;
2023-03-26 14:34:06 +00:00
printf("A ACCOSTE_AVANCE\n");
2023-03-18 16:59:15 +00:00
}else{
etat_accostage = CERISE_TOURNE_CONTACTEUR_LONGER_A;
2023-03-26 14:34:06 +00:00
printf("A ACCOSTE_TOURNE_A\n");
2023-03-18 16:59:15 +00:00
}
}else if(i2c_annexe_get_contacteur_longer_C() == CONTACTEUR_ACTIF){
etat_accostage = CERISE_ACCOSTE;
2023-03-26 14:34:06 +00:00
printf("A ACCOSTE\n");
2023-03-18 16:59:15 +00:00
}
break;
2023-03-18 16:59:15 +00:00
case CERISE_TOURNE_CONTACTEUR_LONGER_C:
commande_rotation_contacteur_longer_C();
if(i2c_annexe_get_contacteur_longer_C() == CONTACTEUR_INACTIF){
if(i2c_annexe_get_contacteur_longer_A() == CONTACTEUR_INACTIF){
etat_accostage = CERISE_AVANCE_DROIT;
2023-03-26 14:34:06 +00:00
printf("C ACCOSTE_AVANCE\n");
2023-03-18 16:59:15 +00:00
}else{
etat_accostage = CERISE_TOURNE_CONTACTEUR_LONGER_C;
2023-03-26 14:34:06 +00:00
printf("C ACCOSTE_TOURNE_C\n");
2023-03-18 16:59:15 +00:00
}
}else if(i2c_annexe_get_contacteur_longer_A() == CONTACTEUR_ACTIF){
etat_accostage = CERISE_ACCOSTE;
2023-03-26 14:34:06 +00:00
printf("C ACCOSTE\n");
2023-03-18 16:59:15 +00:00
}
break;
2023-03-18 16:59:15 +00:00
case CERISE_ACCOSTE:
commande_vitesse_stop();
2023-03-26 14:34:06 +00:00
etat_accostage = CERISE_AVANCE_DROIT;
2023-03-18 16:59:15 +00:00
etat_action = ACTION_TERMINEE;
2023-03-26 14:34:06 +00:00
break;
default:
break;
}
return etat_action;
}
void commande_rotation_contacteur_longer_A(){
commande_rotation(ROTATION_CERISE, RAYON_ROBOT, 0);
}
void commande_rotation_contacteur_longer_C(){
commande_rotation(-ROTATION_CERISE, RAYON_ROBOT/2.0, -RAYON_ROBOT* RACINE_DE_3/2.0);
2023-03-26 14:34:06 +00:00
}
void commande_translation_longer_vers_A(){
// V_x : V * cos (60°) = V / 2
// V_y : V * sin (60°) = V * RACINE(3) / 2
commande_vitesse(TRANSLATION_CERISE/2., TRANSLATION_CERISE / 2. * RACINE_DE_3, 0);
}
void commande_translation_longer_vers_C(){
// V_x : -V * cos (60°) = -V / 2
// V_y : -V * sin (60°) = -V * RACINE(3) / 2
commande_vitesse(-TRANSLATION_CERISE/2., -TRANSLATION_CERISE / 2. * RACINE_DE_3, 0);
}
enum longer_direction_t inverser_longe_direction(enum longer_direction_t direction){
if(direction ==LONGER_VERS_A){
return LONGER_VERS_C;
}
return LONGER_VERS_A;
}