domingo, 27 de marzo de 2016

Mini osciloscopio con 2 arduino y QC12864. II (30Khz). Conexión paralelo.

En este post hacíamos un pequeño osciloscopio con un arduino nano y un lcd qc12864.
Tras comprobar las limitaciones de modificar el prescaler  en el arduino con casi todos sus pines analógicos trabajando, pensé en el otro pequeñín que tenía guardado en un cajón que podía ocuparse de hacer el muestreo de la señal, enviarselo al otro via serie que sería el encargado de los cálculos y la representación gráfica.
Al fin y al cabo los osciloscopios basados en arduino "más" decentes que he visto en internet funcionan así, aunque también hay otra opción con un solo arduino: conectarlos en serie.

De esta manera he logramos logrado una resolución de 6useg./pixel o lo que es lo mismo todo el ancho de pantalla menor de 1 milisegundo, con el potenciometro podemos variar la duración del muestreo hasta 200 mSeg. a partir de ahí el comportamiento del programa cambia y monitorea el valor de la señal.

El sketch viene preparado, descomentando una de las líneas finales del void(loop), para insertar un potenciometro al pin A7 y estabilizar la imagen, sobre todo en las frecuencias medias.



El funcionamiento del programa es similar al anterior, aunque ahora el arduino maestro envía una señal cuyo valor es la resolución requerida, tomada del potenciometro a través de A1, que a demás es la señal que indica al arduino esclavo que inicie el muestreo. ( Si hay un desajuste temporal, por ejemplo los encendemos en momentos distintos,  se va a quedar esperando esa señal indefinidamente, y habrá que reiniciar el arduino maestro para que coienze de nuevo - botón rset-).

Una vez hecho el muestreo se envían al maestro los datos a través del puerto serie, siendo el último dato el tiempo invertido en el muestreo, para que este los analice y represente en el lcd.

La lectura de la frecuencia es sólo una estimación y no es el propósito de este programa, aunque es bastante aproximada si vemos varias ondas en la pantalla,


Nuestro osciloscopio sólo admite valores de entrada entre 0 y +5V,  y otros valores lo pueden dañar, así que para visualizarlos  hay que usar divisores resistivos o  zener.



Para que funcionen  los sketch hay que tener instalada la librería U8glib.


Las conexiones son practicamente las mismas que en el osciloscopio anterior, con algunas pequeñas modificaciones para posibilitar la comunicación entre arduinos.




Los arduinos se conectan entre sí a través de los pines Tx y Rx con un cable cruzado, también deben compartir la masa, La señal se escanea a través del pin A0 del arduino esclavo.




Sketch 1. Arduino maestro (conectado al lcd) Descargar





#include "U8glib.h"


U8GLIB_ST7920_128X64_1X u8g(8, 9, 10, 11, 4, 5, 6, 7, 18, 17, 16);  //For ST7920_128X64 paralelo

unsigned long sensor;

int x;
int y[130];

boolean monitor;

int y2;
unsigned long ycal;

byte punto;
byte y3;

float mx;
byte mc;
byte fc;


void setup(void) {

  Serial.begin(115200);
}



void loop(void) {


  u8g.firstPage();

  // Envio del tpo de muestreo

  sensor = analogRead(A1)/2;
 

if (sensor < 256) {    // Osciloscopio

  monitor=0;
 
   if ( sensor <64 arduino="" br="" con="" nbsp="" osciloscopio="" segundo="">

   Serial.write(byte(sensor));

 // Lectura de datos

  for ( punto = 0; punto <= 129; punto++ ) {
   
 while( Serial.available()==0){
 }

    y[punto] = Serial.read();
   
  
  }

sensor = (y[128]);   // tpo muestreo
 
   } // fin muestreo os. seg arduino
  

   else {   // muestreo propio arduino

   
y[129]=0;
mx=millis();


ycal=8*(sensor-64); // calibrado tiempo

 for ( punto = 0; punto <= 127; punto++ ) {  //muestreo pinA0 arduino maestro
    x=1;
    y[punto] = analogRead(A0);

  // delayMicroseconds(ycal);

   retardo0();

  }

 sensor= (millis()-mx)*4;
 
for ( punto = 0; punto <= 127; punto++ ) {
    y[punto] = y[punto]/4;
     y[129]=max(y[129],y[punto]);
  
  }
 
   
   }  // fin osciloscopio arduino  maestro

  
  /// Estimación frecuencia   f= picos/tpo muestreo

 
  fc = 0;
 
  mx = y[129]*.9  ;  // Pico > 10%


  for ( punto = 0; punto <= 127; punto++ ) {

    if ( (mx < y[punto]) && (mc == 0)) {
      fc = fc + 1;
      mc = 1;
    }

    if (y[punto] < mx) {
      mc = 0;
    }
  }
}                          // <-- br="" cierre="" nbsp="" osciloscopio="">

else {                         // monitoreo del pin A0

   monitor=1;

mx=millis();
   y[128]=0; //media total

x=100;


  y[127]=analogRead(A0)/4;
  retardo();
 
   for ( punto = 0; punto <= 126; punto++ ) {
   
 y[punto]=y[punto+1];

       y[128]=y[128]+y[punto];
  }


  mx=millis()-mx;

y[128]=y[128]/128;

}                    // cierre de monitoreo

  /// Dibujar

  do {

    draw();
  } while ( u8g.nextPage() );


//   delayMicroseconds(analogRead(A7)*50);   // Para congelar imagen entre muestreo y muestreo conviene controlar este parametro mediante potenciómetro.



}


void retardo0()    // retardo osciloscopio arduino maestro
{
   if (x   y[punto]=(analogRead(A0)+y[punto])/2;
  x=x+2;
     retardo0();
   }

}

void retardo()   // retardo monitor
{

if (x<((30*(analogRead(A1)-512)))) {
    y[127]=((analogRead(A0)/4)+y[127])/2;
  x=x+1;
retardo();
}

}

///////////////////    GRAFICA

void draw(void) {

  // graphic commands to redraw the complete screen should be placed here

  // u8g.drawFrame(0,0,128,56);  //marco exterior

  for ( punto = 21; punto <= 106; punto = punto + 42 ) {       // punto vertical cada 1V

    u8g.drawPixel(punto, 10);  u8g.drawPixel(punto, 21); u8g.drawPixel(punto, 32); u8g.drawPixel(punto, 43);

  }

 u8g.setFontPosTop();
 

  // Grafica de la onda

  y3 = 55 - 27 * float(y[0]) / 128;  ///  54/1024

  for ( punto = 1; punto <= 127; punto++ ) {


    ycal = 55 - 28 * float(y[punto]) / 128;    ///  54/1024


    u8g.drawLine(punto - 1, y3, punto, ycal);

    y3 = ycal;
  }


  // Leyenda

  u8g.setFont(u8g_font_04b_03);

  u8g.setPrintPos(0, 56);
 
  if (monitor==1){

    u8g.print(String(float(y[127])*5/255) +"V.  /"+ String(float(.17+mx/1000)) + " Seg.  Med: "+String(float(y[128])*5/255)+"V.");   ///   x5V
  }
  else {
    u8g.print( String(float((sensor+.5)/4)) + " mSeg. /Estm: " + String(long(float(fc / (sensor+.5) * 4000))) + " Hz ");
  }

}


<64 arduino="" br="" con="" nbsp="" osciloscopio="" segundo=""><-- br="" cierre="" nbsp="" osciloscopio="">
<64 arduino="" br="" con="" nbsp="" osciloscopio="" segundo=""><-- br="" cierre="" nbsp="" osciloscopio=""><64 arduino="" br="" con="" nbsp="" osciloscopio="" segundo=""><-- br="" cierre="" nbsp="" osciloscopio="">

<64 arduino="" br="" con="" nbsp="" osciloscopio="" segundo=""><-- br="" cierre="" nbsp="" osciloscopio="">
<64 arduino="" br="" con="" nbsp="" osciloscopio="" segundo=""><-- br="" cierre="" nbsp="" osciloscopio=""><64 arduino="" br="" con="" nbsp="" osciloscopio="" segundo=""><-- br="" cierre="" nbsp="" osciloscopio="">






Sketch 2. Arduino esclavo ( encargado de la obtención de datos.) Descargar

 


 void setup() {
  Serial.begin(115200);

    }

//////


void loop(void) {

byte punto;

byte y[130];

int sensor;

unsigned long ycal;


  // Muestreo

  
 while( Serial.available()==0){
 }    

   sensor=Serial.read();


 //Prescaler
  //ADPS2 - ADPS1 - ADPS0 - Division Factor
  //0       - 0      - 0         ->2
  //0       - 0      - 1         ->2
  //0       - 1      - 0         ->4    >
  //0       - 1      - 1         ->8    ***
  //1       - 0      - 0         ->16
  //1       - 0      - 1         ->32
  //1       - 1      - 0         ->64
  //1       - 1      - 1         ->128
 
 // A partir de 4 funciona bien, el 2 distorsiona mucho. Cuanto más subimos el osciloscopio es más lento pero más preciso

switch(byte(1+6*sensor/64))
  {
    case 0: //  =2  distorsionado
      bitWrite(ADCSRA,ADPS2,0);  bitWrite(ADCSRA,ADPS1,0);  bitWrite(ADCSRA,ADPS0,0);
    break;
    case 1:
     bitWrite(ADCSRA,ADPS2,0);  bitWrite(ADCSRA,ADPS1,1);  bitWrite(ADCSRA,ADPS0,0);
    break;
    case 3:
      bitWrite(ADCSRA,ADPS2,0);  bitWrite(ADCSRA,ADPS1,1);  bitWrite(ADCSRA,ADPS0,1);
    break;
    case 4:
    bitWrite(ADCSRA,ADPS2,1);  bitWrite(ADCSRA,ADPS1,0);  bitWrite(ADCSRA,ADPS0,0);
    break;
    case 5:
      bitWrite(ADCSRA,ADPS2,1);  bitWrite(ADCSRA,ADPS1,0);  bitWrite(ADCSRA,ADPS0,1);
    break;
    case 6:
      bitWrite(ADCSRA,ADPS2,1);  bitWrite(ADCSRA,ADPS1,1);  bitWrite(ADCSRA,ADPS0,0);
    break;
    case 7:
      bitWrite(ADCSRA,ADPS2,1);  bitWrite(ADCSRA,ADPS1,1);  bitWrite(ADCSRA,ADPS0,1);;
    break;
 
  }

 
  //Analog Input A0
ADMUX=(1<<ADLAR)|(0<<REFS1)|(1<<REFS0)|(0<<MUX3)|(0<<MUX2)|(0<<MUX1)|(0<<MUX0);



 sensor=sensor*1.2;

  if (sensor ==0) {
  

///  MUESTREO turbo

 ycal = micros();

  //


  y[0] = analogReadFast();
  
    y[1] = analogReadFast();
  
    y[2] = analogReadFast();
  
    y[3] = analogReadFast();
  
    y[4] = analogReadFast();
  
    y[5] = analogReadFast();
  
    y[6] = analogReadFast();
  
    y[7] = analogReadFast();
  
    y[8] = analogReadFast();
  
    y[9] = analogReadFast();
  
    y[10] = analogReadFast();
  
    y[11] = analogReadFast();
  
    y[12] = analogReadFast();
  
    y[13] = analogReadFast();
  
    y[14] = analogReadFast();
  
    y[15] = analogReadFast();
  
    y[16] = analogReadFast();
  
    y[17] = analogReadFast();
  
    y[18] = analogReadFast();
  
    y[19] = analogReadFast();
  
        y[20] = analogReadFast();
  
    y[21] = analogReadFast();
  
    y[22] = analogReadFast();
  
    y[23] = analogReadFast();
  
    y[24] = analogReadFast();
  
    y[25] = analogReadFast();
  
    y[26] = analogReadFast();
  
    y[27] = analogReadFast();
  
    y[28] = analogReadFast();
  
    y[29] = analogReadFast();

        y[30] = analogReadFast();
  
    y[31] = analogReadFast();
  
    y[32] = analogReadFast();
  
    y[33] = analogReadFast();
  
    y[34] = analogReadFast();
  
    y[35] = analogReadFast();
  
    y[36] = analogReadFast();
  
    y[37] = analogReadFast();
  
    y[38] = analogReadFast();
  
    y[39] = analogReadFast();
        y[40] = analogReadFast();
  
    y[41] = analogReadFast();
  
    y[42] = analogReadFast();
  
    y[43] = analogReadFast();
  
    y[44] = analogReadFast();
  
    y[45] = analogReadFast();
  
    y[46] = analogReadFast();
  
    y[47] = analogReadFast();
  
    y[48] = analogReadFast();
  
    y[49] = analogReadFast();
    y[50] = analogReadFast();
  
    y[51] = analogReadFast();
  
    y[52] = analogReadFast();
  
    y[53] = analogReadFast();
  
    y[54] = analogReadFast();
  
    y[55] = analogReadFast();
  
    y[56] = analogReadFast();
  
    y[57] = analogReadFast();
  
    y[58] = analogReadFast();
  
    y[59] = analogReadFast();
    y[60] = analogReadFast();
  
    y[61] = analogReadFast();
  
    y[62] = analogReadFast();
  
    y[63] = analogReadFast();
  
    y[64] = analogReadFast();
  
    y[65] = analogReadFast();
  
    y[66] = analogReadFast();
  
    y[67] = analogReadFast();
  
    y[68] = analogReadFast();
  
    y[69] = analogReadFast();
        y[70] = analogReadFast();
  
    y[71] = analogReadFast();
  
    y[72] = analogReadFast();
  
    y[73] = analogReadFast();
  
    y[74] = analogReadFast();
  
    y[75] = analogReadFast();
  
    y[76] = analogReadFast();
  
    y[77] = analogReadFast();
  
    y[78] = analogReadFast();
  
    y[79] = analogReadFast();
        y[80] = analogReadFast();
  
    y[81] = analogReadFast();
  
    y[82] = analogReadFast();
  
    y[83] = analogReadFast();
  
    y[84] = analogReadFast();
  
    y[85] = analogReadFast();
  
    y[86] = analogReadFast();
  
    y[87] = analogReadFast();
  
    y[88] = analogReadFast();
  
    y[89] = analogReadFast();
        y[90] = analogReadFast();
  
    y[91] = analogReadFast();
  
    y[92] = analogReadFast();
  
    y[93] = analogReadFast();
  
    y[94] = analogReadFast();
  
    y[95] = analogReadFast();
  
    y[96] = analogReadFast();
  
    y[97] = analogReadFast();
  
    y[98] = analogReadFast();
  
    y[99] = analogReadFast();
 y[100] = analogReadFast();
  
    y[101] = analogReadFast();
  
    y[102] = analogReadFast();
  
    y[103] = analogReadFast();
  
    y[104] = analogReadFast();
  
    y[105] = analogReadFast();
  
    y[106] = analogReadFast();
  
    y[107] = analogReadFast();
  
    y[108] = analogReadFast();
  
    y[109] = analogReadFast();
  
    y[110] = analogReadFast();
  
    y[111] = analogReadFast();
  
    y[112] = analogReadFast();
  
    y[113] = analogReadFast();
  
    y[114] = analogReadFast();
  
    y[115] = analogReadFast();
  
    y[116] = analogReadFast();
    y[117] = analogReadFast();
    y[118] = analogReadFast();
    y[119] = analogReadFast();
    y[120] = analogReadFast();
    y[121] = analogReadFast();
    y[122] = analogReadFast();
    y[123] = analogReadFast();
    y[124] = analogReadFast();
    y[125] = analogReadFast();
    y[126] = analogReadFast();
    y[127] = analogReadFast();

  y[128]= (micros() - ycal)/250;

}

else {    // muestreo con retardo
 
 ycal = micros();

  for ( punto = 0; punto <= 127; punto++ ) {


    y[punto]=analogReadFast();
   

    delayMicroseconds(sensor);
  }
  y[128]= (micros() - ycal)/250;
}


  // ENVIO al  otro arduino

  for ( punto = 0; punto <= 127; punto++ ) {
   
  y[129]=max(y[129],y[punto]);  //Pico
   Serial.write(y[punto]);
    
  }
  Serial.write(y[128]); Serial.write(byte(y[129]));


    y[129]=0; // Pico

}

 //Read ADC
int analogReadFast()
{
ADCSRA|=(1<<ADSC);
 // ADSC is cleared when the conversion finishes
 while (bit_is_set(ADCSRA, ADSC));
        return ADCH;
}


No hay comentarios: