#include #include #include "FS.h" #include "SD.h" #include "SPI.h" #include "RTClib.h" #include #include "driver/pcnt.h" #include #include ///////////////////////////// /* DÉFINITIONS DU COMPTEUR */ ///////////////////////////// //#define PULSE_COMPTEUR #define PCNT_TEST_UNIT PCNT_UNIT_0 // Il existe 8 compteurs de 0 à 7 #define PCNT_TEST_CHANNEL PCNT_CHANNEL_0 #define PCNT_INPUT_SIG_IO GPIO_NUM_15 // Pulse Input GPIO #define PCNT_INPUT_CTRL_IO GPIO_NUM_2 // Control GPIO HIGH=count up, LOW=count down bool sem_compteur; // sémaphore pour le compteur int16_t count = 0; // variable de compteur float vRotReel; // vitesse réelle mesurée du moteur ///////////////////////////////// /*NUMÉRO DE SÉRIE DE L'APPAREIL*/ ///////////////////////////////// int numero_capteur = 0001; /////////////////////////// /* DÉFINITIONS DE LA RTC */ /////////////////////////// RTC_DS3231 rtc; //déclaration de la rtc //////////////////////////////// /* DÉFINITIONS DE LA CARTE SD */ //////////////////////////////// unsigned long freq_ecriture = 60000; // durée entre deux écritures sur la carte SD en micro secondes int fichier = 0; char horodatage[25]; //création du tableau pour contenir l'horodatage ////////////////////////////////// /* DÉFINITIONS DU DRIVER MOTEUR */ ////////////////////////////////// int vRotVis = 2400; //vitesse de rotation recherchée en tr/min Servo moteur; //création de l'objet moteur int minUs = 1000; // intervalle pwm mini int maxUs = 2000; // intervalle pwm maxi int pos = 135; // vitesse de départ, 135 donne environ 40Hz int moteurPin = 4; //pin de contrôle du BEC de l'ESC du moteur ESP32PWM pwm; // Activation du pwm //////////////////////////////// /* DÉFINITIONS DU BOUTON WIFI */ //////////////////////////////// int bouton_wifi = 2; // pin du bouton wifi bool sem_wifi; //sémaphore du minuteur du bouton wifi unsigned long wifi_minuteur = 0; // minuteur pour l'appui long sur le bouton wifi unsigned long compteur_minuteur = 0; // minuteur pour l'appui long sur le bouton wifi bool etat_bouton_wifi; /////////////////////////////////////// /* DÉFINITIONS DU POINT D'ACCES WIFI */ /////////////////////////////////////// const char *ssid = "IFV-Sporix"; // SSID const char *password = "12345678"; // PASSWORD, 8 caractères minimum WiFiServer server(80); // port du serveur /////////////// /* PAGE HTML */ /////////////// String html =" \ \ \

ESP32 IFV Soft access point

\

Web Server

\ \ "; /////////////////////////// /* FONCTIONS DU COMPTEUR */ /////////////////////////// void start_compteur() { esp_err_t error; Serial.println("Initialisation du compteur"); pcnt_config_t pcnt_config = { PCNT_INPUT_SIG_IO, // Pulse input gpio_num, if you want to use gpio16, pulse_gpio_num = 16, a negative value will be ignored PCNT_PIN_NOT_USED, // Control signal input gpio_num, a negative value will be ignored PCNT_MODE_KEEP, // PCNT low control mode PCNT_MODE_KEEP, // PCNT high control mode PCNT_COUNT_INC, // PCNT positive edge count mode PCNT_COUNT_DIS, // PCNT negative edge count mode //PCNT_H_LIM_VAL, // Maximum counter value //PCNT_L_LIM_VAL, // Minimum counter value PCNT_TEST_UNIT, // PCNT unit number PCNT_TEST_CHANNEL, // the PCNT channel }; if(pcnt_unit_config(&pcnt_config) == ESP_OK) //init unit Serial.println("Config Unit_0 = ESP_OK"); pcnt_set_filter_value(PCNT_TEST_UNIT, 100); /*Configure input filter value*/ pcnt_filter_enable(PCNT_TEST_UNIT); /*Enable input filter*/ pcnt_counter_pause(PCNT_TEST_UNIT); pcnt_counter_clear(PCNT_TEST_UNIT); /*Reset counter value*/ pcnt_counter_resume(PCNT_TEST_UNIT); /*Resume counting*/ } void stop_compteur() { /*Pause counter*/ pcnt_counter_pause(PCNT_TEST_UNIT); /*Reset counter value*/ pcnt_counter_clear(PCNT_TEST_UNIT); } void compte_tour() { if ( sem_compteur == false ) { compteur_minuteur = millis(); sem_compteur = true; } if ( sem_compteur == true && compteur_minuteur - millis() > freq_ecriture ){ /// vitesse = readPcntCounter_0() / 2; //vitesse en tours par min /// scribe_sd (); /// pcnt_counter_clear(); compteur_minuteur = millis(); sem_compteur = false; } } ///////////////////////// /* FONCTIONS DU MOTEUR */ ///////////////////////// void start_moteur() { ESP32PWM::allocateTimer(0); moteur.setPeriodHertz(50); moteur.attach(moteurPin, minUs, maxUs); delay(15); moteur.write(0); // envoi du 0 à l'ESC pour initialisation delay(1500); // délai permettant au variateur de détecter le zero moteur.write(pos); // vitesse initiale } void gaz_moteur() { if (vRotVis > vRotReel) { pos ++; } if (vRotReel > vRotVis) { pos --; } moteur.write(pos); } void stop_moteur() { moteur.write(0); delay(1000); moteur.detach(); } ///////////////////////// /// FONCTIONS SD CARD /// ///////////////////////// void listDir(fs::FS &fs, const char * dirname, uint8_t levels){ Serial.printf("Listing directory: %s\n", dirname); File root = fs.open(dirname); if(!root){ Serial.println("Failed to open directory"); return; } if(!root.isDirectory()){ Serial.println("Not a directory"); return; } File file = root.openNextFile(); while(file){ if(file.isDirectory()){ Serial.print(" DIR : "); Serial.println(file.name()); if(levels){ listDir(fs, file.name(), levels -1); } } else { Serial.print(" FILE: "); Serial.print(file.name()); Serial.print(" SIZE: "); Serial.println(file.size()); } file = root.openNextFile(); } } void createDir(fs::FS &fs, const char * path){ Serial.printf("Creating Dir: %s\n", path); if(fs.mkdir(path)){ Serial.println("Dir created"); } else { Serial.println("mkdir failed"); } } void removeDir(fs::FS &fs, const char * path){ Serial.printf("Removing Dir: %s\n", path); if(fs.rmdir(path)){ Serial.println("Dir removed"); } else { Serial.println("rmdir failed"); } } void readFile(fs::FS &fs, const char * path){ Serial.printf("Reading file: %s\n", path); File file = fs.open(path); if(!file){ Serial.println("Failed to open file for reading"); return; } Serial.print("Read from file: "); while(file.available()){ Serial.write(file.read()); } file.close(); } void writeFile(fs::FS &fs, const char * path, const char * message){ Serial.printf("Writing file: %s\n", path); File file = fs.open(path, FILE_WRITE); if(!file){ Serial.println("Failed to open file for writing"); return; } if(file.print(message)){ Serial.println("File written"); } else { Serial.println("Write failed"); } file.close(); } void appendFile(fs::FS &fs, const char * path, const char * message){ Serial.printf("Appending to file: %s\n", path); File file = fs.open(path, FILE_APPEND); if(!file){ Serial.println("Failed to open file for appending"); return; } if(file.print(message)){ Serial.println("Message appended"); } else { Serial.println("Append failed"); } file.close(); } void renameFile(fs::FS &fs, const char * path1, const char * path2){ Serial.printf("Renaming file %s to %s\n", path1, path2); if (fs.rename(path1, path2)) { Serial.println("File renamed"); } else { Serial.println("Rename failed"); } } void deleteFile(fs::FS &fs, const char * path){ Serial.printf("Deleting file: %s\n", path); if(fs.remove(path)){ Serial.println("File deleted"); } else { Serial.println("Delete failed"); } } void testFileIO(fs::FS &fs, const char * path){ File file = fs.open(path); static uint8_t buf[512]; size_t len = 0; uint32_t start = millis(); uint32_t end = start; if(file){ len = file.size(); size_t flen = len; start = millis(); while(len){ size_t toRead = len; if(toRead > 512){ toRead = 512; } file.read(buf, toRead); len -= toRead; } end = millis() - start; Serial.printf("%u bytes read for %u ms\n", flen, end); file.close(); } else { Serial.println("Failed to open file for reading"); } file = fs.open(path, FILE_WRITE); if(!file){ Serial.println("Failed to open file for writing"); return; } size_t i; start = millis(); for(i=0; i<2048; i++){ file.write(buf, 512); } end = millis() - start; Serial.printf("%u bytes written for %u ms\n", 2048 * 512, end); file.close(); } void sd_init(){ if(!SD.begin()){ Serial.println("Card Mount Failed"); return; } uint8_t cardType = SD.cardType(); if(cardType == CARD_NONE){ Serial.println("No SD card attached"); return; } Serial.print("SD Card Type: "); if(cardType == CARD_MMC){ Serial.println("MMC"); } else if(cardType == CARD_SD){ Serial.println("SDSC"); } else if(cardType == CARD_SDHC){ Serial.println("SDHC"); } else { Serial.println("UNKNOWN"); } uint64_t cardSize = SD.cardSize() / (1024 * 1024); Serial.printf("SD Card Size: %lluMB\n", cardSize); // créer un dossier avec un numéro incrémental } void scribe_sd (){ fichier ; writeFile(SD, "data", horodatage); } ///////////////////// /// FONCTIONS RTC /// ///////////////////// void rtc_init() { if (! rtc.begin()) { Serial.println("RTC introuvable !"); // ligne de debug à commenter en prod delay(2000); // ajouter un message d'erreur par clignotement de LED ESP.restart(); while (1); } if (rtc.lostPower()) { Serial.println("Veuillez régler l'heure et vérifier la pile du module RTC!"); // ligne de debug à commenter en prod // ajouter un message d'erreur par clignotement de LED } else { DateTime now = rtc.now(); // mise à l'heure de l'esp, à partir de la rtc Serial.println("mise à l'heure de l'esp, à partir de la rtc"); } } ////////////////////// /// FONCTIONS WIFI /// ////////////////////// void start_wifiAP() { pinMode(LED_BUILTIN, OUTPUT); Serial.println(); Serial.println("Configuring access point..."); //WiFi.softAP(ssid, password); // You can remove the password parameter if you want the AP to be open. IPAddress myIP = WiFi.softAPIP(); Serial.print("AP IP address: "); Serial.println(myIP); server.begin(); Serial.println("Server started"); } void wifi_AP() { WiFiClient client = server.available(); // listen for incoming clients if (client) { // if you get a client, Serial.println("New Client."); // print a message out the serial port String currentLine = ""; // make a String to hold incoming data from the client while (client.connected()) { // loop while the client's connected if (client.available()) { // if there's bytes to read from the client, char c = client.read(); // read a byte, then Serial.write(c); // print it out the serial monitor if (c == '\n') { // if the byte is a newline character // if the current line is blank, you got two newline characters in a row. // that's the end of the client HTTP request, so send a response: if (currentLine.length() == 0) { // HTTP headers always start with a response code (e.g. HTTP/1.1 200 OK) // and a content-type so the client knows what's coming, then a blank line: client.println("HTTP/1.1 200 OK"); client.println("Content-type:text/html"); client.println(); // the content of the HTTP response follows the header: client.print("Click here to turn ON the LED.
"); client.print("Click here to turn OFF the LED.
"); // The HTTP response ends with another blank line: client.println(); // break out of the while loop: break; } else { // if you got a newline, then clear currentLine: currentLine = ""; } } else if (c != '\r') { // if you got anything else but a carriage return character, currentLine += c; // add it to the end of the currentLine } // Check to see if the client request was "GET /H" or "GET /L": if (currentLine.endsWith("GET /H")) { digitalWrite(LED_BUILTIN, HIGH); // GET /H turns the LED on } if (currentLine.endsWith("GET /L")) { digitalWrite(LED_BUILTIN, LOW); // GET /L turns the LED off } } } // close the connection: client.stop(); Serial.println("Client Disconnected."); } ESP.restart(); } /////FONCTIONS COMMUNES////// void vigie_Wifi () { etat_bouton_wifi = digitalRead(bouton_wifi); if ( etat_bouton_wifi == HIGH && sem_wifi == false ) { sem_wifi = true; wifi_minuteur = millis(); } if ( etat_bouton_wifi == HIGH && sem_wifi == true && millis() - wifi_minuteur > 5000 ) { stop_moteur(); stop_compteur(); start_wifiAP(); wifi_AP(); } } void setup() { // serial just for feedback Serial.begin(9600); // RTC rtc_init(); // lecteur de carte SD sd_init (); // bouton wifi pinMode(bouton_wifi, INPUT); sem_wifi = false; // initialisation du sémaphore // moteur start_moteur(); // compteur de pulsations sem_compteur = false; // initialisation du sémaphore //pinMode(pulsePin,INPUT_PULLUP); start_compteur(); } void loop (){ vigie_Wifi(); compte_tour(); gaz_moteur(); scribe_sd(); delay(10000); }