/********************************************************************************** * Surveillance ampli RF LDMOS par F6BQP V3.42a le 23/01/2023 * * Prevu pour fonctionner avec un Arduino NANO * * et afficheur LCD 20 X 4 4 bits parallele * **********************************************************************************/ #include // Librairie LCD #define T_REFRESH1 50 // msec bargraph refresh rate #define T_PEAKHOLD 600 // msec peak hold time before return #define T_pepHOLD 600 // msec pep hold time before return #define T_cliHOLD 200 // msec temps clignotement #define T_autotest 5000 // msec temps autotest //----------Constantes a modifier selon le materiel et ses preferences------------- const float VMAX = 64.9; // Tension max en Volts pour 5V sur A7 const unsigned int IMAX = 36200; // Courant max en mA pour 5V sur A6 const int calibrP = 1350; // 5V = 870W const int calibrT = 848; // correspond a 0° Celsius const byte Tempmin = 35; // seuil alarme temperature Min const byte Tempmax1 = 45; // seuil alarme temperature Max1 // const byte Tempmax1 = 37; // pour TEST const byte Tempmax2 = 55; // seuil alarme temperature Max2 // const byte Tempmax2 = 40; // pour TEST const float ROSmax1 = 2.0; // seuil alarme ROS Max1 const float ROSmax2 = 3.0; // seuil alarme ROS Max2 const unsigned int IntensMAX = 25000; // seuil alarme intensite max en mA // const unsigned int IntensMAX = 5000; // pour TEST const unsigned int TensionMAX = 51; // seuil alarme tension max en V // const unsigned int TensionMAX = 46; // pour TEST const unsigned int TensionMIN = 35; // seuil alarme tension min en V //----------------------- Affectation des Pins Arduino ---------------------------- #define LEDALIM 1 // LED ALIM #define LEDROS 4 // LED ROS #define LEDTEMP 5 // LED temperature #define FAN 17 // Ventilateur sur A3 #define TRANSMIT 13 // sortie emission int analogA0 = 0; // initialise A0 = FWD int analogA1 = 1; // initialise A1 = REF int analogA2 = 2; // initialise A2 = TEMP int analogA6 = 6; // initialise A6 = Courant int analogA7 = 7; // initialise A7 = Tension int PTT = 3; // entree PTT int overload = 2; // entree overload LiquidCrystal lcd(12, 11, 10, 9, 8, 7); // affectation pins afficheur //----------------------------- Declaration Variables ----------------------------- int sensor = 0; int Celsius; float direct = 1; float reflec = 0; unsigned long pwrdir = 0; // power forward (watts) unsigned long pwrref = 0; // power reflected (watts) unsigned int pwrdir_max = 0; // power forward max (for peak hold) unsigned int pwrref_max = 0;; // power reflected max (for peak hold) int anF = 0; // analog read forward power int anR = 0; // analog read reflected power unsigned int LectTension = 0; unsigned long Intensite = 0; // courant en mA unsigned long I_repos = 0; // courant en mA unsigned long Intensite_max = 0; unsigned long LectI; unsigned long LectI_repos; float ROS = 1.0; float Tension = 0.0; float afIntensite; float afI_repos; boolean AlarmTEMP = false; boolean AlarmROS = false; boolean AlarmINTENS = false; boolean AlarmIrepos = false; boolean AlarmTENSION = false; boolean AlarmPINPUT = false; byte fill[6]={ 0x20,0x00,0x01,0x02,0x03,0xFF }; // for fill 0=empty 5=full byte peak[7]={ 0x20,0x00,0x04,0x05,0x06,0x07,0x20 }; // for peak indicator int lmax[5]; // level max memory int dly[5]; // delay & speed // for peak return long lastTpep = 0; // update PEP display long lastTImax = 0; // update Imax display long lastTemps = 0; long lastTemps1 = 0; byte block[8][8]= { { 0x10,0x10,0x10,0x10,0x10,0x10,0x10,0x10 }, // character for fill the bar { 0x18,0x18,0x18,0x18,0x18,0x18,0x18,0x18 }, { 0x1C,0x1C,0x1C,0x1C,0x1C,0x1C,0x1C,0x1C }, { 0x1E,0x1E,0x1E,0x1E,0x1E,0x1E,0x1E,0x1E }, { 0x08,0x08,0x08,0x08,0x08,0x08,0x08,0x08 }, // character for peak level { 0x04,0x04,0x04,0x04,0x04,0x04,0x04,0x04 }, { 0x02,0x02,0x02,0x02,0x02,0x02,0x02,0x02 }, { 0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01 }, }; //-------------------------- Interruptions ---------------------------------------- void inter_overload() // si Pin > Pin MAX { digitalWrite(TRANSMIT, LOW); // Alarme ON --> +TX = 0V digitalWrite(LEDALIM, HIGH); // allume la LED ALIM (coupure +50V) AlarmPINPUT = true; // active alarme PINPUT } void inter_PTTchange() // si basculement PTT { if (digitalRead(PTT) == LOW) // si emission { digitalWrite(TRANSMIT, HIGH); // --> +TX = 13V digitalWrite(FAN, HIGH); // ventilateur ON delayMicroseconds(10000); // attente 10 mS digitalWrite(6, LOW); // ALC OFF } else // si reception { digitalWrite(TRANSMIT, LOW); // --> +TX = 0V analogWrite(6, 127); // ALC ON } } //------------------Routine d'initialisation au demarrage-------------------------- void setup() { pinMode (LEDALIM, OUTPUT); // sortie LED ALIM pinMode (LEDROS, OUTPUT); // sortie LED ROS pinMode (LEDTEMP, OUTPUT); // sortie LED temperature pinMode (FAN, OUTPUT); // sortie ventilateur pinMode (TRANSMIT, OUTPUT); // sortie emission digitalWrite(TRANSMIT, LOW); // interdit emission pinMode (PTT, INPUT_PULLUP); // entree PTT pinMode (overload, INPUT_PULLUP); // entree Overload analogWrite(6, 127); // ALC ON lcd.begin(20, 4); // 4 lignes de 20 caracteres for( int i=0 ; i<8 ; i++ ) // creation caracteres lcd.createChar( i,block[i] ); //----------------- Affichage lors de l'initialisation et Autotest ---------------- lcd.clear(); lcd.print ("ampli RF F6BQP V3.42a"); // 1er caractere ligne 1 lcd.setCursor(0, 2); // 1er caractere ligne 3 lcd.print ("Absence Tension +50V"); while (analogRead(analogA7) < 473) // attente +50V (473=30V) {digitalWrite(LEDALIM, HIGH);} // allume la LED ALIM (coupure +50V) digitalWrite(LEDALIM, LOW); // eteint la LED ALIM (retabli +50V) lcd.setCursor(1, 1); // 2ieme caractere ligne 2 lcd.print("Autotest en cours"); lcd.setCursor(0, 2); // 1er caractere ligne 3 lcd.print(" NE PAS TRANSMETTRE "); lastTemps = millis(); lastTemps1 = millis(); digitalWrite(TRANSMIT, HIGH); // passe en emission digitalWrite(FAN, HIGH); digitalWrite(LEDTEMP, HIGH); // allume LED temperature while (millis() < (lastTemps + T_autotest)) { LectI_repos = analogRead(analogA6); // mesure I repos I_repos = (LectI_repos * IMAX) / 1023; // calcul Irepos if (I_repos > 2500) { // si I repos > 2.5 A digitalWrite(TRANSMIT, LOW); // Alarme ON --> +TX = 0V digitalWrite(LEDALIM, HIGH); // allume la LED ALIM (coupure +50V) AlarmIrepos = true; // active alarme Irepos gestion_alarme (); } afI_repos = (float)I_repos; // conversion Irepos en float lcd.setCursor( 6,3 ); // affichage I repos lcd.print("Ir="); lcd.print (afI_repos/1000 ,1); lcd.print("A"); if (millis() > (lastTemps1 + T_cliHOLD)) {digitalWrite(LEDTEMP, digitalRead(LEDTEMP) ^ 1); // bascule LED TEMP digitalWrite(LEDROS, digitalRead(LEDROS) ^ 1); // bascule LED ROS lastTemps1 = millis(); } // reinitialise lastTemps1 } digitalWrite(TRANSMIT, LOW); // passe en reception digitalWrite(FAN, LOW); printtemplate(); // affiche le cadre attachInterrupt(digitalPinToInterrupt(2),inter_overload,FALLING); // sur D2 attachInterrupt(digitalPinToInterrupt(3),inter_PTTchange,CHANGE); // sur D3 } // FIN de l'initialisation //--------------------------- boucle principale (surveillance) -------------------- void loop() { //-------------------- Gestion ROS + mesure Irepos -------------------------------- direct = analogRead(analogA0); reflec = analogRead(analogA1); if (direct < 18) // si Pout < 1W {if (analogRead(analogA0) < 18) // confirmation Pout < 1W {LectI_repos = analogRead(analogA6);} // mesure I repos if (analogRead(analogA0) < 18) // confirmation Pout < 1W { I_repos = (LectI_repos * IMAX) / 1023; // calcul Irepos if (I_repos > 2500) { // si I repos > 2.5 A digitalWrite(TRANSMIT, LOW); // Alarme ON --> +TX = 0V digitalWrite(LEDALIM, HIGH); // allume la LED ALIM (coupure +50V) AlarmIrepos = true; // active alarme Irepos gestion_alarme (); } afI_repos = (float)I_repos; // conversion Irepos en float lcd.setCursor( 6,3 ); // affichage I repos lcd.print("Ir="); lcd.print (afI_repos/1000 ,1); } } lcd.setCursor(0, 3); // 1er caractere ligne 4 if (direct > (reflec + 1)) // si direct > reflec + 5mV {ROS = (direct + reflec) / (direct - reflec); lcd.print (ROS, 1);} // affiche le ROS else { lcd.print ("-.-"); // affiche -.- ROS = 1; } if (ROS > ROSmax1) { if (millis() > (lastTemps + T_cliHOLD)) {digitalWrite(LEDROS, digitalRead(LEDROS) ^ 1); // bascule LED ROS lastTemps = millis(); } // reinitialise lastTemps if (ROS > ROSmax2) {digitalWrite(TRANSMIT, LOW); // Alarme ON --> +TX = 0V digitalWrite(LEDALIM, HIGH); // allume la LED ALIM (coupure +50V) digitalWrite(LEDROS, HIGH); // allume la LED ROS AlarmROS = true; // active alarme ROS gestion_alarme (); } } else {digitalWrite(LEDROS, LOW);} // eteint la LED ROS direct = direct + 19; // ajoute 90mV (seuil diode X 0,45) reflec = reflec + 19; // ajoute 90mV (seuil diode X 0.45) pwrdir = pow(direct,2) / calibrP; // calcul puissance directe en W pwrref = pow(reflec,2) / calibrP; // calcul puissance réfléchie en W anF = map( pwrdir,0,800,0,65 ); // 65 = 13 x 5 colums, full scale 800W anR = map( pwrref,0,80,0,65 ); // 65 = 13 x 5 colums, full scale 80W bar( 0,anF ); bar( 1,anR ); if (pwrdir >= pwrdir_max) // we have a peak ! {lastTpep = millis(); pwrdir_max = pwrdir; // memorise pwrdir_max pwrref_max = pwrref;} // memorise pwrref_max if (millis() > (lastTpep + T_pepHOLD)) // clear the peak after hold time { pwrdir_max = pwrdir; pwrref_max = pwrref; } lcd.setCursor(16, 0); // 17ieme caractere ligne 1 if (pwrdir_max < 100) lcd.print(" "); if (pwrdir_max < 10) lcd.print(" "); lcd.print (pwrdir_max,DEC); // affiche puissance directe max lcd.setCursor(16, 1); // 17ieme caractere ligne 2 if (pwrref_max < 100) lcd.print(" "); if (pwrref_max < 10) lcd.print(" "); lcd.print (pwrref_max,DEC); // affiche puissance reflechie max //---------------------------- Affichage Tension ---------------------------------- LectTension = analogRead(analogA7); Tension = LectTension * VMAX / 1023; // calcul tension lcd.setCursor(8, 2); // 9ieme caractere ligne 3 lcd.print (Tension, 1); if (Tension > TensionMAX || Tension < TensionMIN) { digitalWrite(TRANSMIT, LOW); // Alarme ON --> +TX = 0V digitalWrite(LEDALIM, HIGH); // allume la LED ALIM (coupure +50V) AlarmTENSION = true; // active alarme TENSION gestion_alarme ();} //---------------------------- Affichage Courant ---------------------------------- LectI = analogRead(analogA6); Intensite = (LectI * IMAX) / 1023; // calcul intensite // if ( Intensite < 300 ) {Intensite = 0;} // ignore intensite < 300 mA if (Intensite > IntensMAX) { delay(1); // attente 1 mS LectI = analogRead(analogA6); // confirmation Intensite = (LectI * IMAX) / 1023; // calcul intensite if (Intensite > IntensMAX) // confirmation ? { digitalWrite(TRANSMIT, LOW); // Alarme ON --> +TX = 0 V digitalWrite(LEDALIM, HIGH); // allume la LED ALIM (coupure +50V) AlarmINTENS = true; // active alarme INTENS gestion_alarme (); } } if (direct > 10) // affichage intensite { if (Intensite >= Intensite_max) {lastTImax = millis(); Intensite_max = Intensite;} // memorise intensite max if (millis() > (lastTImax + T_pepHOLD)) // clear the peak after hold time { Intensite_max = Intensite; } lcd.setCursor(6, 3); // 7ieme caractere ligne 4 lcd.print( "I="); if (Intensite_max < 10000) lcd.print(" "); afIntensite = (float)Intensite_max; // conversion Intensite en float lcd.print (afIntensite/1000 ,1); } //---------------------------- Gestion temperature -------------------------------- // digitalWrite(LEDTEMP, HIGH); // pour mesure temps de boucle sensor = analogRead(analogA2); Celsius = (calibrT - sensor) / 3.14; // calcul temperature Celsius lcd.setCursor(16, 3); // 17ieme caractere ligne 3 lcd.print (Celsius); if (digitalRead(PTT) == HIGH) // si reception { if (Celsius < Tempmin) // si temperature < Tempmin {digitalWrite(FAN, LOW);} // FAN OFF } if (Celsius > Tempmax1) { if (millis() > (lastTemps + T_cliHOLD)) {digitalWrite(LEDTEMP, digitalRead(LEDTEMP) ^ 1); // bascule LED TEMP lastTemps = millis(); } // reinitialise lastTemps digitalWrite(FAN, HIGH); // ventilateur ON if (Celsius > Tempmax2) { delay(1); // attente 1 mS sensor = analogRead(analogA2); Celsius = (calibrT - sensor) / 3.14; // calcul temperature Celsius if (Celsius > Tempmax2) // confirmation ? {digitalWrite(TRANSMIT, LOW); // Alarme ON --> +TX = 0 V digitalWrite(LEDALIM, HIGH); // allume la LED ALIM (coupure +50V) digitalWrite(LEDTEMP, HIGH); // allume LED temperature AlarmTEMP = true; // active alarme TEMP gestion_alarme ();} } } else {digitalWrite(LEDTEMP, LOW);} // eteint LED temperature if (AlarmPINPUT == true) {gestion_alarme ();} } // FIN de la boucle principale void printtemplate () { lcd.setCursor( 0,0 ); lcd.print( "FWD 0W"); lcd.setCursor( 0,1 ); lcd.print( "REF 0W"); lcd.setCursor( 0,2 ); lcd.print( "SWR U= 0.0V Temp"); lcd.setCursor( 0,3 ); lcd.print( "-.- I=--.-A -- "); lcd.setCursor( 18,3 ); lcd.print (char(223)); // degres lcd.setCursor( 4,2 ); // barres verticales lcd.write( peak[3] ); lcd.setCursor( 4,3 ); lcd.write( peak[3] ); lcd.setCursor( 14,2 ); lcd.write( peak[3] ); lcd.setCursor( 14,3 ); lcd.write( peak[3] ); } void gestion_alarme () // messages d'alarme et attente passage en reception { detachInterrupt(1); // bloque le PTT digitalWrite(TRANSMIT, LOW); // Alarme ON --> +TX = 0V digitalWrite(LEDALIM, HIGH); // allume la LED ALIM (coupure +50V) lcd.clear(); lcd.setCursor(0, 1); // 1er caractere ligne 2 if (AlarmPINPUT == true) {lcd.print (" OVERLOAD Pin > 12W "); lcd.setCursor(0, 2); // 1er caractere ligne 3 lcd.print (" ou I > 28 A "); lcd.setCursor(6, 3); // 7ieme caractere ligne 4 lcd.print( "I="); if (Intensite_max < 10000) lcd.print(" "); afIntensite = (float)Intensite_max; // conversion Intensite en float lcd.print (afIntensite/1000 ,1); lcd.print(" A"); } if (AlarmTEMP == true) {lcd.print (" Temp trop elevee "); lcd.setCursor(16, 3); // 17ieme caractere ligne 4 lcd.print (Celsius); lcd.print (char(223)); } // degres if (AlarmROS == true) {lcd.print (" ROS > 3 ");} if (AlarmINTENS == true) {lcd.print (" I > 25A ");} if (AlarmTENSION == true) {lcd.print ("Alim > 50V ou < 35V ");} if (AlarmIrepos == true) {lcd.print (" I repos > 2.5A "); digitalWrite(LEDTEMP, LOW); digitalWrite(LEDROS, LOW);} while (true) { } // boucle infinie } //---------------------------------- BARGRAPH ------------------------------------- void bar ( int row,int lev ) // row = 0 ou 1 lev = 0 a 65 { lcd.setCursor( 3,row ); // 4ieme caractere ligne row // lcd.write( row ? ' ' : ' ' ); for( int i=1 ; i<14 ; i++ ) // 13 X 5 = 65 { int f=constrain( lev -i*5,0,5 ); // Level instant int p=constrain( lmax[row]-i*5,0,6 ); // Level maximum (=peak) if( f ) // si f > 0 lcd.write( fill[ f ] ); else // si f < 0 lcd.write( peak[ p ] ); } if( lev>lmax[row] ) { lmax[row] = lev; dly[row] = -(T_PEAKHOLD)/T_REFRESH1; // Starting delay value } // Negative=peak don't move else { if( dly[row]>0 ) lmax[row] -= dly[row]; // lmax = lmax - dly if( lmax[row]<0 ) lmax[row]=0; else dly[row]++; } } // FIN du programme