miércoles, 19 de agosto de 2015

Proyecto: Sonómetro libre gigante (2)

Bueno, nuevamente les traigo el proyecto del sonómetro libre gigante, tengo más fotos y los circuitos.

Bueno, acá van las fotos:











Como se puede obvservar, hay distintos módulos (display, control de potencia, tratamiento analógico de la señal y el arduino que se encarga de hacer el muestreo en el display y unos cálculos).

Acá les dejo un link de todos los circuitos necesarios y en algunos días les dejo el programa:

domingo, 16 de agosto de 2015

¿Sonar?

Bueno, luego de hacer las publicaciones del medidor de distancia por ultrasonido, del servo y de la pantalla, se me ocurrió realizar un radar. En el título lo puse entre signos de pregunta, porque no sé hasta que punto es factible realizarlo, pero si es necesario lo vamos a realizar en 2 o 3 publicaciones.

No es un proyecto propiamente dicho porque no va a tener utilidad. Quizá se pregunten ¿porqué?, bueno la razón o razones son sencillas:

1) Tiene un alcance ridículamente limitado (en las publicación del ultrasonido pude ver que tenía un máximo de 2 metros aproximadamente)

2) Los tiempos van a ser cruciales, por lo que parecería que no se va a poder mostrar en tiempo real el objeto.

Pero a pesar de esas cosas voy a desarrollarlo por "amor al arte" y ver que sale, quizá se pueda conseguir algo interesante.

Entonces, para empezar, voy a reciclar código (como no podía ser de otra forma) de la publicación del servo, de la pantalla y del ultrasonido.

La idea sería mostrar en la pantalla los puntos que se consideran que bloqueen la "vista" del ultrasonido.

Sin más, les dejo el código:


//pantalla
#include <Adafruit_GFX.h>                                    // Core graphics library
#include "SWTFT.h"                                           // Hardware-specific library
//algunos colores
#define negro     0x0000
#define azul      0x001F
#define rojo      0xF800
#define verde     0x07E0
#define cyan      0x07FF
#define magenta   0xF81F
#define amarillo  0xFFE0
#define blanco    0xFFFF
//de la pantalla
SWTFT tft;

//para el touch
#include <stdint.h>
#include "TouchScreen.h"
//declaraciones para el touch
#define YP A1                                                // must be an analog pin, use "An" notation!
#define XM A2                                                // must be an analog pin, use "An" notation!
#define YM 7                                                 // can be a digital pin
#define XP 6                                                 // can be a digital pin

#define MINPRESSURE 10
#define MAXPRESSURE 1000

TouchScreen ts = TouchScreen(XP, YP, XM, YM, 300);

//servo
#include <SoftwareServo.h>
SoftwareServo servo1;

//ultrasonido
#define trigPin 32
#define echoPin 34

int x = 0, y = 0, x1 = 0, y1 = 0, check = 1;
long distancia;

void setup() {
  servo1.attach(30);
  pinMode(trigPin, OUTPUT);
  pinMode(echoPin, INPUT);
  tft.reset();
  tft.begin();
  tft.fillScreen(negro);
  tft.setRotation(3);
  tft.fillCircle(50, 50, 10, blanco);
  Serial.begin(9800);
}

void loop() {
  tft.fillScreen(negro);
  for (int val = 0; val <= 180; val++)
  {
    servo1.write(val);
    ultrasonido();                                                                      //obtenemos la distancia hasta el objeto
    tft.fillCircle((160-(distancia*cos(val))), (240-((distancia*sin(val))/2)), 2, blanco);  //ubicamos el cursor en el medio abajo 
    SoftwareServo::refresh();                                                           //y situamos el punto donde esté el objeto
  }
  tft.fillScreen(negro);
  for (int val = 180; val >= 0; val--)
  {
    servo1.write(val);
    ultrasonido();                                                                      //obtenemos la distancia hasta el objeto
    tft.fillCircle((160-(distancia*cos(val))), (240-((distancia*sin(val))/2)), 2, blanco);  //ubicamos el cursor en el medio abajo
    SoftwareServo::refresh();                                                           //y situamos el punto donde esté el objeto
  }
}

void ultrasonido() {

  long duracion;
a:;
  digitalWrite(trigPin, LOW);
  delayMicroseconds(5); 
  digitalWrite(trigPin, HIGH);
  delayMicroseconds(15);
  digitalWrite(trigPin, LOW);
  duracion = pulseIn(echoPin, HIGH);
  SoftwareServo::refresh();
  distancia = (duracion/2) / 29.1;
  if (distancia > 450)
  {
    goto a;                                                                              //si el objeto está muy lejos volvemos a
  }                                                                                      //buscar el valor
  Serial.println(distancia);
}


jueves, 13 de agosto de 2015

Servos

Hola, hoy quería mostrarles lo sencillo que resulta manejar un servo con arduino.

Antes de comenzar, voy a hacer una breve explicación de qué es un servo o servomotor de modelismo. Un servo es un dispositivo electromecánico que tiene la capacidad de ir y mantenerse en la posición que se le diga, recibe pulso periódico pwm, que varía de 1mS a 2mS, como muestra la imágen.


Bueno, ahora que saben más o menos qué es y como funciona, vamos al programa.

En la parte de ejemplos, en el ide de arduino, hay una opción que dice "servo", ahí se van a encontrar con 2 ejemplos "knob" y "sweep", si bien estos ejemplos sirven, creo que solo tienen una cantidad de pines limitados (9 y 10), por lo que no se pueden poner más que 2 servos (o al menos eso entendí). Por eso, buscando un poquito me topé con esta otra librería que parece que podría aceptar como salida cualquier pin, soporte o no salidas pwm.

El siguiente códgo lo saqué de este sitio y lo único que hice fue traducir los comentarios.

#include <SoftwareServo.h> 

SoftwareServo myservo;  // crea el objeto servo, para controlar el servo

int potpin = 0;  // pin analógico para conectar el potenciómetro
int val;    // variable que lee el valor analógico

void setup() 
{ 
  myservo.attach(2);  // vincual el servo al pin 2 void loop() 
{ 
  val = analogRead(potpin);            // lee el valor del pote
  val = map(val, 0, 1023, 0, 179);     // re-escala el valor del pote a 0 - 180
  myservo.write(val);                  // pone el servo en el ángulo escalado del pote
  delay(15);                           // le da tiempo al servo para llegar

  SoftwareServo::refresh();
} 



Bueno, el código parece ser simple, por lo que pude leer, la línea "SoftwareServo::refresh();" tiene que estar cada 50mS o menos, para que el servo funcione correctamente, por lo que eso podría llegar a ser un problema para programas largos, por lo que recomiendo (dependiendo del programa) agregar el siguiente código:


tiempo = micros();
colocar el código acá
Serial.print((micros() - tiempo));


Eso lo que va a hacer es calcular de forma casi precisa cuantos micro segundos pasaron mientras el código se estaba ejecutando, probablemente tenga un error, pero no creo que llegue al mS. En caso de que se superen los 50mS con el código, lo único que hay que hacer es colocar periódicamente "SoftwareServo::refresh();" y ya se soluciona (obviamente siempre teniendo en cuenta que el tiempo sin poner ese fragmento de código no supere los 50mS).

Descargas:

Librería del servo:

https://mega.co.nz/#!blhhhCpS!MpJYJ44tsdfn8ZwFCpdu7anjrVUms-0BXTSjYwX5rhY

lunes, 10 de agosto de 2015

Proyecto: Sonómetro libre gigante (1)

Hoy les traigo un proyecto que realicé con otros estudiantes de la facultad FCEIA, más específicamente con el grupo GEII (Grupo de Experimentación Innovativa e Instrumental, por sus siglas en español).

en su etapa final, haciendo las últimas pruebas antes de cerrar la caja.
En esta etapa solo queda encender el dBA y calibrarlo (que se hace una
 vez cerrada la caja)
Se trata de un sonómetro gigante que nos pidieron para medir el sonido de la calle. Consta de una plaqueta con varios circuitos (pre amplificador, amplificador, autorango y un verdadero valor eficaz), un arduino y un equipo de potencia. La plaqueta con los distintos circuitos, se encarga de preparar la señal y convertirla al verdadero valor eficaz para que el arduino (es un arduino UNO) lea el valor analógico y lo convierta a decibeles. Una vez convertido y procesado ese valor va al display (que consta de tiras de led simulando un 7 segmentos gigante). El consumo aproximado del display es de 3 a 4 amperes, por lo que ahí entra en juego la parte de potencia.

En otra entrada (cuando vuelva al GEII, porque ahora está en el receso de invierno) pondré algunos datos más, como los circuitos y el programa, así como algunas fotos y algún video.

viernes, 7 de agosto de 2015

Proyecto: integración de módulos (1)

Hoy les quiero mostrar un proyecto en el que llevo trabajando un poco más de 6 meses y sigo ampliándolo casi todos los días.


Se trata de un "dispositivo" a falta de un mejor nombre, que consta de un arduino MEGA 2560, la pantalla de mcufriend (hablé anteriormente en esta otra entrada), el transceptor NRF24L01 y el sensor de temperatura y humedad DTH11.








El proyecto lo realicé porque quería ver cuantas cosas podía integrar en un solo arduino, y como los arduino UNONANO y LEONARDO tenían pocos pines I/O decidí pasar al MEGA, con el cual puedo disponer de 53 pines I/O y 15 entradas analógicas. El sistema cuenta con: una comunicación con otro dispositivo (las direcciones todavía no se pueden cambiar) llamado "serv1", un reloj configurable, una linterna en pantalla y sensor de temperatura y humedad. A futuro estoy pensando en agregar un control remoto de domótica y un "generador de ondas" que serviría para generar señales cuadradas con ciertos patrones o frecuencias.






De momento el dispositivo solo puede establecer comunicación inalámbrica con la computadora (por puerto serie) con un arduino NANO y un tranceptor como interfaz (para recibir estoy usando el programa del chat que está en esta otra entrada), pero estoy seguro que si se obtiene el mismo equipo y realizando cambios mínimos (como la dirección de recepción y la dirección de emisión) se debería poder tener ambos equipos con una comunicación entre ellos.


En algunos días, subiré el programa y las librerías. Cuando yo considere que está terminado, voy a crear una página dedicada al proyecto completo para que se pueda ver sin tener que buscar viejas entradas.

martes, 4 de agosto de 2015

Medidor de distancia por ultrasonido

Esta vez les traigo un sensor que es capaz de medir las distancias mediante el sonido, manda un pulso y se cuenta cuanto tiempo demora en llegar, en función de eso se sabe la distancia.
El módulo en cuestión es el SFR05 y se alimenta con los 5V que puede entregar el arduino.



el programa es muy sencillo y no necesita librerías:


#define trigPin 13
#define echoPin 12

void setup() {
  Serial.begin (9600);
  pinMode(trigPin, OUTPUT);
  pinMode(echoPin, INPUT);
}

void loop() {
  long duracion, distancia;
  digitalWrite(trigPin, LOW);
  delayMicroseconds(2); 
  digitalWrite(trigPin, HIGH);
  delayMicroseconds(10);
  digitalWrite(trigPin, LOW);
  duracion = pulseIn(echoPin, HIGH);
  distancia = (duracion/2) / 29.1;
  }
  if (distancia >= 200 || distancia <= 0){
    Serial.println("Fuera de rango");
  }
  else {
    Serial.print(distancia);
    Serial.println(" cm");
  }
  delay(500);
}


sábado, 1 de agosto de 2015

Prueba led variante

Hoy les traigo un programa sencillo que puede llegar a ser útil para ver distintas intensidades de un led, sin usar el pwm (de esta forma se puede obtener mayor resolución) el problema es que consume todo el tiempo de procesamiento.

unsigned long tiempo;
int t = 12, g=0, h=6;

void setup() {

    pinMode(9, OUTPUT);          // Selecciono el pin 9 como salida
}

void loop() {

  tiempo = millis()+1000;        // guardo la cantidad de milisegundos
                                 // que va a haber dentro de 1 segundo
  while (tiempo > millis()) {    // 
    digitalWrite(9, HIGH);       // prendo el led 
    delay(t - g);                // durante un tiempo (t - g)
    digitalWrite(9, LOW);        // apago el led
    delay(g);                    // durante un tiempo g
  }

  tiempo = millis()+1000;        // 
                                 // 
  while (tiempo > millis()) {    // 
    digitalWrite(9, HIGH);       // hago lo mismo que arriba, pero con 
    delay(h);                    // h como variable en lugar de g
    digitalWrite(9, LOW);        // 
    delay(t-h);                  // 
  }
}